Android沉浸式狀態(tài)欄實(shí)現(xiàn)示例
應(yīng)用市場(chǎng)上App越來(lái)越多的出現(xiàn)沉浸式狀態(tài)欄的設(shè)計(jì)(如下圖所示)狀態(tài)欄和導(dǎo)航欄具有相同的顏色。Android在4.4開(kāi)始對(duì)于該種效果的支持,而在4.4之下,狀態(tài)欄只是黑框,無(wú)法控制。同時(shí)在4.4和5.0及其之上的版本對(duì)該種效果的支持又有所差異,因此要實(shí)現(xiàn)該種效果,可以將4.4歸為一類,5.0及其之上歸為一類。接下來(lái),我們將一步步來(lái)在4.4和5.0及其之上來(lái)實(shí)現(xiàn)如下所示效果。
導(dǎo)航欄問(wèn)題
在Android中,頂部導(dǎo)航欄目前常用的兩種實(shí)現(xiàn)方式,一個(gè)是通過(guò)Toolbar,一個(gè)是通過(guò)自定義View的方式來(lái)實(shí)現(xiàn)。兩種方式各有利弊。Toolbar為官方指定規(guī)范,開(kāi)發(fā)者使用更方便,但可拓展性差,對(duì)于一些特殊的展示效果無(wú)法實(shí)現(xiàn),而通過(guò)自定義方式的方式,可以支持更多展示效果,但卻需要我們寫(xiě)更多的代碼。兩種方式在實(shí)現(xiàn)狀態(tài)欄沉浸上也有所差別。
去掉Title
Toolbar默認(rèn)主題會(huì)具有一個(gè)title,當(dāng)我們使用Toolbar的時(shí)候,而沒(méi)有去掉title,應(yīng)用則會(huì)crash,報(bào)出如下所示錯(cuò)誤。
因此在使用Toolbar 的時(shí)候,我們需要style中添加如下屬性配置
<item name="windowNoTitle">true</item>
當(dāng)然我們也可以通過(guò)代碼動(dòng)態(tài)去掉title,但當(dāng)我們的主題從Theme.AppCompat作為父類繼承的時(shí)候,通過(guò)代碼并不可以去掉title。
自定義導(dǎo)航欄
當(dāng)我們未設(shè)置windowNoTitle
屬性的時(shí)候,在導(dǎo)航欄之上有title。顯然和我們要實(shí)現(xiàn)導(dǎo)航欄的沉浸式有所違背,因此實(shí)現(xiàn)對(duì)于導(dǎo)航欄的沉浸,
<item name="windowNoTitle">true</item>
該配置是必不可少的。
設(shè)置狀態(tài)欄透明
去掉title之后,是否我們就可以實(shí)現(xiàn)上述的效果了呢?
這個(gè)時(shí)候,我們發(fā)現(xiàn)狀態(tài)欄還是黑色,并沒(méi)有沉浸,需要我們將狀態(tài)欄設(shè)置為透明。
<item name="android:windowTranslucentStatus">true</item>
該屬性只有在在4.4和高于4.4版本上可以進(jìn)行該屬性的配置,但是在更低版本上則無(wú)法使用。配置該屬性之后,執(zhí)行效果如下圖所示。
解決導(dǎo)航欄上移問(wèn)題
這個(gè)時(shí)候,Toolbar被整體上移了,導(dǎo)致其部分功能也進(jìn)入了狀態(tài)欄之下,包括導(dǎo)航欄的內(nèi)容也到了狀態(tài)欄位置之中,顯然這是不符合我們最初的要求的。如何解決這個(gè)問(wèn)題?我們?cè)赥oolbar中添加fitSystemWindows屬性,即可使得toolbar的上部空出一個(gè)高度,使得Toolbar內(nèi)容部分脫離狀態(tài)欄。
<android.support.v7.widget.Toolbar android:id="@+id/toolbar" android:fitsSystemWindows="true" android:layout_width="match_parent" android:layout_height="wrap_content"> </android.support.v7.widget.Toolbar>
得到我們最終想要得到的效果
自定義導(dǎo)航欄與之實(shí)現(xiàn)類似。
fitsSystemWindows屬性
前面對(duì)Toolbar的設(shè)置是在Toolbar中添加的fitSystemWindows
屬性,那么當(dāng)我們將其屬性添加到Toolbar所在的最外層的布局會(huì)怎么樣呢?
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_toolbar" android:layout_width="match_parent" android:layout_height="match_parent" android:fitsSystemWindows="true" tools:context="com.example.netease.toolbardemo.activity.ToolbarActivity"> <android.support.v7.widget.Toolbar android:id="@+id/toolbar" android:layout_width="match_parent" android:layout_height="wrap_content"> </android.support.v7.widget.Toolbar> </RelativeLayout>
執(zhí)行之后,可以看到和之前未設(shè)置狀態(tài)欄透明時(shí)的效果相同。
那么這個(gè)fitSystemWindows工作的原理是什么呢?通過(guò)上述實(shí)驗(yàn),不難發(fā)現(xiàn),對(duì)于沉浸狀態(tài)欄的控制,該屬性起到了一個(gè)很關(guān)鍵的作用。
接下來(lái)通過(guò)一個(gè)實(shí)驗(yàn)來(lái)驗(yàn)證下,該屬性所起的作用,在Toolbar所在的布局中,在布局的底部添加一個(gè)Button.
<android.support.v7.widget.Toolbar android:id="@+id/toolbar" android:fitsSystemWindows="true" android:layout_width="match_parent" android:layout_height="wrap_content"> </android.support.v7.widget.Toolbar> <Button android:text="Test" android:layout_width="match_parent" android:layout_height="wrap_content" android:background="@color/colorAccent" android:layout_alignParentBottom="true"/>
當(dāng)我們將該屬性設(shè)置到按鈕上,又會(huì)發(fā)生什么呢?
<Button android:text="Test" android:fitsSystemWindows="true" android:layout_width="match_parent" android:layout_height="wrap_content" android:background="@color/colorAccent" android:layout_alignParentBottom="true"/>
通過(guò)比較可以很明顯的看出,設(shè)置了 fitsSystemWindows
屬性的View在其上部被設(shè)置了一個(gè)padding。根據(jù)之前做的實(shí)驗(yàn),我們可以知道當(dāng)我們?cè)O(shè)置了窗口狀態(tài)欄透明之后,整個(gè)內(nèi)容視圖會(huì)向上移動(dòng)了一個(gè)狀態(tài)欄的高度,而當(dāng)前為該View增加的padding的大小是不是和其高度相同呢?
Button btn = (Button) findViewById(R.id.test_btn); Log.i("padding", btn.getPaddingTop()+""); Rect frame = new Rect(); getWindow().getDecorView().getWindowVisibleDisplayFrame(frame); Log.i("height", frame.top+"");
獲取按鈕的padding高度和狀態(tài)欄的高度,我們可以得到如下日志。
通過(guò)實(shí)驗(yàn)我們可以得出結(jié)論, fitSystemWindows
屬性會(huì)對(duì)所設(shè)置的View增加一個(gè)top padding,因此當(dāng)我們?cè)趯?shí)現(xiàn)讓導(dǎo)航欄沉浸的時(shí)候,設(shè)置窗口狀態(tài)欄的透明會(huì)使得視圖整體上移,而借助 fitSystemWindows
屬性的功能,為視圖中最頂部的View設(shè)置一個(gè)和狀態(tài)欄高度相同的padding,使得導(dǎo)航欄不會(huì)被頂?shù)綘顟B(tài)欄內(nèi)。
當(dāng)我們?cè)谝粋€(gè)視圖中,多個(gè)View設(shè)置該屬性時(shí),發(fā)現(xiàn)只有第一個(gè)設(shè)置該屬性的View會(huì)起作用,在視圖布局上,自上而下的第一個(gè)View其作用。層級(jí)上則為最頂級(jí)的View上首先其作用。因此其功能歸納為:
- 為設(shè)置該屬性的View增加一個(gè)和狀態(tài)欄高度相同的toppadding
- 當(dāng)視圖中有多個(gè)View被設(shè)置了該屬性,那么只布局上最頂部的View起作用
5.0及其以上
至此,我們可以完美的實(shí)現(xiàn)一個(gè)狀態(tài)欄的沉浸,上述的實(shí)現(xiàn)是在Android 4.4版本上,在視圖的最上部,會(huì)有一個(gè)黑色漸變的陰影,而在5.0設(shè)備上的展示效果如下所示,在狀態(tài)欄上整個(gè)都會(huì)有一個(gè)陰影。當(dāng)然不同廠家對(duì)此也有自己的一些優(yōu)化,比如魅族在4.4上是不具有陰影的。
對(duì)于5.0及其之上,官方提供了對(duì)狀態(tài)欄顏色控制的相應(yīng)API,我們可以通過(guò)代碼來(lái)控制狀態(tài)欄的顏色,實(shí)現(xiàn)如下效果。
實(shí)現(xiàn)代碼
if(Build.VERSION.SDK_INT >= 21) { Window window = getWindow(); //取消設(shè)置透明狀態(tài)欄,使 ContentView 內(nèi)容不再沉浸到狀態(tài)欄下 window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS); //需要設(shè)置這個(gè) flag 才能調(diào)用 setStatusBarColor 來(lái)設(shè)置狀態(tài)欄顏色 window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS); //設(shè)置狀態(tài)欄顏色 window.setStatusBarColor(getResources().getColor(R.color.yx_red)); }
據(jù)此可以看出,當(dāng)我們?cè)?.0及其之上的實(shí)現(xiàn)中,可以不用進(jìn)行狀態(tài)欄透明的設(shè)置和 fitSystemWindows
屬性的設(shè)置,直接通過(guò)代碼來(lái)控制,但為了適應(yīng)4.4版本,建議在代碼中仍然按之前的方式實(shí)現(xiàn),如果想在5.0及其高版本中實(shí)現(xiàn)去陰影,再手動(dòng)在代碼中控制。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android手動(dòng)檢查并申請(qǐng)權(quán)限方法
今天小編就為大家分享一篇Android手動(dòng)檢查并申請(qǐng)權(quán)限方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-07-07Android開(kāi)發(fā)之圖形圖像與動(dòng)畫(huà)(二)Animation實(shí)現(xiàn)圖像的漸變/縮放/位移/旋轉(zhuǎn)
Android 平臺(tái)提供了兩類動(dòng)畫(huà),一類是Tween動(dòng)畫(huà),就是對(duì)場(chǎng)景里的對(duì)象不斷的進(jìn)行圖像變化來(lái)產(chǎn)生動(dòng)畫(huà)效果;旋轉(zhuǎn)、平移、放縮和漸變等等,感興趣的朋友可以了解下啊,希望本文對(duì)你有所幫助2013-01-01使用Fragment來(lái)處理Andoird app的UI布局的實(shí)例分享
這篇文章主要介紹了使用Fragment來(lái)處理Andoird appUI布局的實(shí)例分享,Fragment的出現(xiàn)緩解了代碼依賴于Activity而造成的臃腫狀況,需要的朋友可以參考下2016-02-02Android 滑動(dòng)監(jiān)聽(tīng)RecyclerView線性流+左右劃刪除+上下移動(dòng)
這篇文章主要介紹了Android 滑動(dòng)監(jiān)聽(tīng)RecyclerView線性流+左右劃刪除+上下移動(dòng)的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-09-09Android 優(yōu)雅的實(shí)現(xiàn)通用格式化編輯
這篇文章主要介紹了Android 優(yōu)雅的實(shí)現(xiàn)通用格式化編輯,幫助大家更好的理解和學(xué)習(xí)使用Android,感興趣的朋友可以了解下2021-03-03Android中設(shè)置WebView禁止縮放網(wǎng)頁(yè)的步驟
在Android中如果你想要禁止WebView縮放網(wǎng)頁(yè),可以通過(guò)設(shè)置WebView的一些屬性來(lái)實(shí)現(xiàn),這篇文章主要給大家介紹了關(guān)于Android中設(shè)置WebView禁止縮放網(wǎng)頁(yè)的步驟,需要的朋友可以參考下2024-05-05Android實(shí)現(xiàn)監(jiān)聽(tīng)電話呼叫狀態(tài)的方法
這篇文章主要介紹了Android實(shí)現(xiàn)監(jiān)聽(tīng)電話呼叫狀態(tài)的方法,涉及Android權(quán)限控制及電話狀態(tài)監(jiān)聽(tīng)的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10完美解決android M上鎖屏情況下,禁止pc通過(guò)MTP訪問(wèn)手機(jī)存儲(chǔ)單元
下面小編就為大家?guī)?lái)一篇完美解決android M上鎖屏情況下,禁止pc通過(guò)MTP訪問(wèn)手機(jī)存儲(chǔ)單元。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-04-04