分別用ToolBar和自定義導(dǎo)航欄實(shí)現(xiàn)沉浸式狀態(tài)欄
一、ToolBar
1、在build.gradle中添加依賴,例如:
compile 'com.android.support:appcompat-v7:23.4.0'
2、去掉應(yīng)用的ActionBar??梢允切薷闹黝}theme為“NoActionBar”,例如:
<style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar">
或者不修改主題為"NoActionBar",而在主題的style下,添加:
<item name="windowNoTitle">true</item> <item name="windowActionBar">false</item>
第二個(gè)屬性代表是否用ActionBar代替TitleBar。
其實(shí),剛學(xué)的時(shí)候,感覺很納悶,怎么又多了個(gè)TitleBar?后來查了很久才發(fā)現(xiàn),3.0以前,狀態(tài)欄下面的是標(biāo)題欄(只能顯示標(biāo)題等少量信息),3.0以后就變成了應(yīng)用欄,也就是ActionBar。
另外,我測(cè)試的時(shí)候,activity是繼承于AppCompatActivity,主題是AppCompat類型的。這種情況下,必須要像上面那樣寫才有效果,少寫或值不同的話,要么沒效果,要么報(bào)錯(cuò)。
最后,上面兩個(gè)屬性的說明可在android.R.attr這個(gè)類中查看。
3、在xml中為ToolBar添加屬性
android:fitsSystemWindows="true" android:minHeight="?attr/actionBarSize"
fitsSystemWindows是ToolBar實(shí)現(xiàn)沉浸式狀態(tài)欄的關(guān)鍵,其大概情況是,如果設(shè)為true,就會(huì)調(diào)整這個(gè)view去留一些空間給系統(tǒng)窗口,如果不設(shè)置或設(shè)為false,ToolBar就會(huì)和狀態(tài)欄重疊在一起。
而第二個(gè)屬性中,它的值全寫是"?android:attr/actionBarSize",其意思是引用當(dāng)前主題中的actionBarSize這個(gè)屬性。更多相關(guān)說明可查看官方文檔中Accessing Resources的部分。
上面兩個(gè)屬性可在android.view.View這個(gè)類中查看。
4、在java中添加判斷sdk版本的代碼并在用戶的系統(tǒng)是4.4及以上時(shí)設(shè)置狀態(tài)欄為透明
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS); }
無論是ToolBar,還是自定義導(dǎo)航欄,這個(gè)操作都是實(shí)現(xiàn)沉浸式狀態(tài)欄的關(guān)鍵。
因?yàn)樵O(shè)置狀態(tài)欄為透明的這個(gè)屬性,要4.4以上才能使用,所以4.4以下的系統(tǒng)是不能夠?qū)崿F(xiàn)沉浸式狀態(tài)欄的。而在4.4到5.0的系統(tǒng)中,狀態(tài)欄是全透明的,也就是它的顏色會(huì)跟你的ToolBar和自定義導(dǎo)航欄的顏色一樣。而在5.0以上的系統(tǒng)中,則是半透明的,也就看起來會(huì)比較深暗。
而我在6.0的系統(tǒng)上測(cè)試時(shí),發(fā)現(xiàn)這一步?jīng)]設(shè)置和設(shè)置了的,從效果上看,區(qū)別就是沒設(shè)置時(shí)狀態(tài)欄顏色淺一點(diǎn),而且ToolBar的padding top為0,而設(shè)置了的顏色就深一點(diǎn),padding top為狀態(tài)欄的高度。具體有什么影響,還不清楚。但這會(huì)讓自定義導(dǎo)航的外觀變形,它會(huì)增加狀態(tài)欄的高度,但又沒有讓這部分與狀態(tài)欄重疊,就導(dǎo)致效果變形。
5、最后在java中添加
setSupportActionBar(mToolbar);
ToolBar的布局代碼:
<?xml version="1.0" encoding="utf-8"?> <android.support.v7.widget.Toolbar xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" android:id="@+id/tool_bar" android:layout_width="match_parent" android:layout_height="wrap_content" android:background="@color/colorPrimary" android:elevation="4dp" android:fitsSystemWindows="true" android:minHeight="?android:attr/actionBarSize" app:title="ToolBar" app:subtitle="toolbar"/>
效果圖(Android 6.0):
二、自定義導(dǎo)航欄TopBar
1、設(shè)置窗口為無標(biāo)題,上面第2步中的兩個(gè)方法都可以實(shí)現(xiàn),或者是在java中添加如下代碼:
requestWindowFeature(Window.FEATURE_NO_TITLE);
注意在添加這句代碼時(shí),確保是在加載布局內(nèi)容之前,也就是onCreate的setContentView之前。在《Android群英傳》“Android控件架構(gòu)”,這一節(jié)中解釋了為什么requestWindowFeature()需要在setContentView()之前。
另外,我發(fā)現(xiàn)如果該activity是繼承AppCompatActivity的話,只寫上面的這句代碼是沒有變化的,顯示的還是ActionBar。但如果是繼承FragmentActivity的話,就有效果,也就說上面第2步中的第二個(gè)方法,只添加其中任意一個(gè)屬性都是可以的。至于是什么原因,我還沒弄清楚。
2、同上面第4步,判斷系統(tǒng)版本并按需設(shè)置狀態(tài)欄為透明
3、獲取狀態(tài)欄的高度
protected int getStatusHeight() { try { Class<?> c = Class.forName("com.android.internal.R$dimen"); // 獲得與字符串對(duì)應(yīng)的Class對(duì)象 Object object = c.newInstance(); // 創(chuàng)建這個(gè)Class的實(shí)例對(duì)象 Field field = c.getField("status_bar_height"); // 拿到字符串對(duì)應(yīng)的變量 int x = Integer.parseInt(field.get(object).toString()); // 通過這個(gè)實(shí)例對(duì)象拿到這個(gè)變量的值,再轉(zhuǎn)換類型,最后轉(zhuǎn)為整型,變?yōu)橐粋€(gè)資源id return getResources().getDimensionPixelSize(x); } catch (Exception e) { e.printStackTrace(); } return 0; }
這部分代碼是利用Java的反射機(jī)制來實(shí)現(xiàn)的,因?yàn)檫@個(gè)internal包默認(rèn)會(huì)被sdk/platforms/android-version中的android.jar給移除掉,所以無法直接調(diào)用或查看這個(gè)包中的類。如果要使用的話,可以借助這個(gè)開源項(xiàng)目https://github.com/anggrayudi/android-hidden-api。
4、獲取自定義TopBar的高度并修改布局參數(shù)
protected void setStatusBar() { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { final ViewGroup viewGroup = (ViewGroup) findViewById(R.id.top_bar); final int statusHeight = getStatusHeight(); viewGroup.post(new Runnable() { @Override public void run() { int topBarHeight = viewGroup.getHeight(); LinearLayout.LayoutParams layoutParams = (LinearLayout.LayoutParams) viewGroup.getLayoutParams(); layoutParams.height = statusHeight + topBarHeight; viewGroup.setLayoutParams(layoutParams); } }); } }
因?yàn)樵趇nclude這個(gè)TopBar的布局文件中,其父布局是LinearLayout,而TopBar的父布局是RelativeLayout,所以這里先要轉(zhuǎn)成ViewGroup,等getLayoutParams時(shí),再轉(zhuǎn)成LinearLayout.LayoutParams。
TopBar的布局:
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/top_bar" android:layout_width="match_parent" android:layout_height="49dp" android:background="@color/colorPrimary" android:gravity="bottom"> <RelativeLayout android:layout_width="match_parent" android:layout_height="49dp"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerInParent="true" android:text="@string/app_name" android:textSize="24sp" android:textColor="#ffffff"/> </RelativeLayout> </RelativeLayout>
因?yàn)檫@個(gè)布局的高度會(huì)在代碼中動(dòng)態(tài)地修改,即49dp加上狀態(tài)欄的高度,所以只有一個(gè)層級(jí)的結(jié)構(gòu)的話,那導(dǎo)航欄的內(nèi)容就會(huì)往上偏。所以要嵌套多一層來維持導(dǎo)航欄的高度,同時(shí)在最外層的布局中,添加android:gravity="bottom"這個(gè)屬性來保證導(dǎo)航欄不往上偏。
效果圖(Android 6.0):
以上就是本文的全部?jī)?nèi)容,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來一定的幫助,同時(shí)也希望多多支持腳本之家!
- 微軟IE Developer Toolbar安裝使用簡(jiǎn)要圖文說明
- jquery toolbar與網(wǎng)頁浮動(dòng)工具條具體實(shí)現(xiàn)代碼
- Java Swing中的工具欄(JToolBar)和分割面版(JSplitPane)組件使用案例
- Toolbar制作菜單條過程詳解
- iOS應(yīng)用中使用Toolbar工具欄方式切換視圖的方法詳解
- iOS中的導(dǎo)航欄UINavigationBar與工具欄UIToolBar要點(diǎn)解析
- Android5.0+ CollapsingToolbarLayout使用詳解
- Android自定義ActionProvider ToolBar實(shí)現(xiàn)Menu小紅點(diǎn)
- Android自定義Toolbar使用方法詳解
- 深入理解Android 5.0中的Toolbar
相關(guān)文章
android開發(fā)教程之清除android數(shù)據(jù)緩存示例(清除本地?cái)?shù)據(jù)緩存)
android清除本地?cái)?shù)據(jù)緩存示例,主要功能清除內(nèi)/外緩存,清除數(shù)據(jù)庫,清除sharedPreference,清除files和清除自定義目錄,需要的朋友可以參考下2014-02-02Android Service服務(wù)詳細(xì)介紹及使用總結(jié)
這篇文章主要介紹了Android Service 服務(wù)的詳細(xì)資料,網(wǎng)上關(guān)于Android Service 服務(wù)的文章比較多,但是不是很全面,不夠細(xì)致,畢竟是Android 四大組件之一,重要性不用說,這里總結(jié)下,需要的朋友可以參考下2016-12-12Android中實(shí)現(xiàn)EditText密碼顯示隱藏的方法
這篇文章主要介紹了Android中實(shí)現(xiàn)EditText密碼顯示隱藏的方法,需要的朋友可以參考下2017-01-01Android中ViewPager你所不知道的優(yōu)化技巧分享
提到ViewPager想必各位同學(xué)一點(diǎn)都不陌生,它是Android中最常用的組件之一,這篇文章小編就帶大家一起來看看ViewPager一些新的優(yōu)化方式吧2024-04-04ViewPager的setOnPageChangeListener方法詳解
這篇文章主要介紹了ViewPager的setOnPageChangeListener方法詳解,非常不錯(cuò),具有參考解決借鑒價(jià)值,需要的朋友可以參考下2016-12-12Android studio報(bào): java.lang.ExceptionInInitializerError 錯(cuò)誤
本篇文章主要介紹了Android studio報(bào): java.lang.ExceptionInInitializerError錯(cuò)誤的解決方法,具有很好的參考價(jià)值。下面跟著小編一起來看下吧2017-03-03