Translucent System Bar 的最佳實踐
近幾天準備抽空總結Android一些系統(tǒng)UI的實踐使用崔梗,于是開始動手建了一個庫 AndroidSystemUiTraining 夜只,邊擼代碼邊寫總結
今天開寫第一篇,對 Translucent System Bar 的實踐做一些總結蒜魄。說起 Translucent System Bar 的特性扔亥,可能有些朋友還比較陌生,這里做一下簡單的介紹谈为。
看上圖旅挤,Android 4.4之前,即使我們打開手機app伞鲫,我們還總是能看到系統(tǒng)頂部那條黑乎乎的通知欄粘茄,這樣會使得app稍顯突兀。于是Android 4.4開始秕脓,便引入了Translucent System Bar的系特性柒瓣,用于彌補系統(tǒng)通知欄突兀之處。(估計也是向ios學習吠架,因為ios一大早就有這個特性)芙贫。我們先來看看 Translucent System Bar 新特性引入后,發(fā)生了什么樣的變化傍药。下面截取了 中華萬年歷的天氣預報界面 和 QQ音樂主界面 的效果(兩個界面的效果實現(xiàn) Translucent System Bar 的方式有些區(qū)別磺平,下文會細講)
可以看到,系統(tǒng)的通知欄和app界面融為一體拐辽,媽媽再也不用面對黑乎乎的通知欄了拣挪。有關 Translucent System Bar 的特性就暫且介紹到此。
工程簡介
先簡單介紹一下工程的結構俱诸,核心部分已經(jīng)圈出菠劝,待我逐一講解
主要的操作都在style.xml 和 AndroidManifest.xml 中,Activity里面沒有任何涉及到Translucent System Bar設置的代碼乙埃,所以可以忽略不看闸英。
ColorTranslucentBarActivity 和 ImageTranslucentBarActivity 分別用于展示兩種不同實現(xiàn)方式的效果
要在Activity中使用 Translucent System Bar 特性,首先需要到AndroidManifest中為指定的Activity設置Theme介袜。但是需要注意的是甫何,我們不能直接在values/style.xml直接去自定義 Translucet System Bar 的Theme,因為改特性僅兼容 Android 4.4 開始的平臺遇伞,所以直接在values/style.xml聲明引入辙喂,工程會報錯。有些開發(fā)者朋友會在代碼中去判斷SDK的版本,然后再用代碼設置Theme巍耗。雖然同樣可以實現(xiàn)效果秋麸,但個人并不推崇這種做法。我所采取的方法則是建立多個SDK版本的values文件夾炬太,系統(tǒng)會根據(jù)SDK的版本選擇合適的Theme進行設置灸蟆。大家可以看到上面我的工程里面有values、values-v19亲族、values-v21炒考。
第一種方式
第一種方式,需要做下面三步設置
1霎迫、在values斋枢、values-v19、values-v21的style.xml都設置一個 Translucent System Bar 風格的Theme
values/style.xml
<style name="ImageTranslucentTheme" parent="AppTheme">
<!--在Android 4.4之前的版本上運行知给,直接跟隨系統(tǒng)主題-->
</style>
values-v19/style.xml
<style name="ImageTranslucentTheme" parent="Theme.AppCompat.Light.DarkActionBar">
<item name="android:windowTranslucentStatus">true</item>
<item name="android:windowTranslucentNavigation">true</item>
</style>
values-v21/style.xml
<style name="ImageTranslucentTheme" parent="Theme.AppCompat.Light.DarkActionBar">
<item name="android:windowTranslucentStatus">false</item>
<item name="android:windowTranslucentNavigation">true</item>
<!--Android 5.x開始需要把顏色設置透明瓤帚,否則導航欄會呈現(xiàn)系統(tǒng)默認的淺灰色-->
<item name="android:statusBarColor">@android:color/transparent</item>
</style>
上面需要注意的地方是,無論你在哪個SDK版本的values目錄下涩赢,設置了主題戈次,都應該在最基本的values下設置一個同名的主題。這樣才能確保你的app能夠正常運行在 Android 4.4 以下的設備谒主。否則朝扼,肯定會報找不到Theme的錯誤。
2霎肯、在AndroidManifest.xml中對指定Activity的theme進行設置
<activity
android:name=".ui.ImageTranslucentBarActivity"
android:label="@string/image_translucent_bar"
android:theme="@style/ImageTranslucentTheme" />
3擎颖、在Activity的布局文件中設置背景圖片,同時观游,需要把android:fitsSystemWindows設置為true
activity_image_translucent_bar.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@mipmap/env_bg"
android:fitsSystemWindows="true">
</RelativeLayout>
到此搂捧,第一種實現(xiàn)方式完成,大家可以看看下面的效果
就跟中華萬年歷的天氣預報效果界面一樣懂缕,系統(tǒng)的整個導航欄都融入了app的界面中允跑,背景圖片填滿了整個屏幕,看起來舒服很多搪柑。這里還有一個android:fitsSystemWindows設置需要注意的地方聋丝,后面會在細講。接下來看第二種實現(xiàn)工碾。
方式二
相比中華萬年歷弱睦,QQ音樂采用的是另外一種實現(xiàn)的方式,它將app的Tab欄和系統(tǒng)導航欄分開來設置渊额。
由于它的Tab欄是純色的况木,所以只要把系統(tǒng)通知欄的顏色設置和Tab欄的顏色一致即可垒拢,實現(xiàn)上相比方法一要簡單很多。同樣要到不同SDK版本的values下火惊,創(chuàng)建一個同名的theme求类,在values-v21下,需要設置系統(tǒng)導航欄的顏色:
values-v21/style.xml
<style name="ColorTranslucentTheme" parent="Theme.AppCompat.Light.DarkActionBar">
<item name="android:windowTranslucentStatus">false</item>
<item name="android:windowTranslucentNavigation">true</item>
<item name="android:statusBarColor">@color/color_31c27c</item>
</style>
再到ColorTranslucentBarActivity的布局文件activity_color_translucent_bar.xml中設置Tab欄的顏色
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fitsSystemWindows="true"
android:orientation="vertical">
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="55dp"
android:background="@color/color_31c27c">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:text="QQ Music"
android:textColor="@android:color/white"
android:textSize="20sp" />
</RelativeLayout>
</LinearLayout>
到此屹耐,我們就可以得到和QQ音樂主界面一樣的效果了尸疆。
到此,就大體介紹完了 Translucent System Bar 的兩種實現(xiàn)方式了张症。
android:fitsSystemWindows的“踩坑”
通過前面的兩種方式仓技,大家估計會留意到一個地方鸵贬,就是所有實現(xiàn) Translucent System Bar 效果的Activity俗他,都需要在根布局里設置 android:fitsSystemWindows="true" 。設置了該屬性的作用在于阔逼,不會讓系統(tǒng)導航欄和我們app的UI重疊兆衅,導致交互問題。這樣說可能比較抽象嗜浮,看看下面兩個效果圖的對比就知道了羡亩。
注:上面的演示效果,是借助了我的另一個開源項目危融,詳情請戳:AndroidAlbum
這樣的話畏铆,如果我有10個Activity要實現(xiàn)這種效果,就要在10個布局文件中做設置吉殃,非常麻煩辞居。所以,想到一種方法蛋勺,在theme中加上如下的android:fitsSystemWindows設置:
<item name="android:fitsSystemWindows">true</item>
發(fā)現(xiàn)果真可以了瓦灶。所有要實現(xiàn) Translucent System Bar 的Activity,只需要設置了這個theme即可,改起來也很方便抱完≡籼眨可惜,后來出現(xiàn)了一個BUG巧娱,讓我還是得老老實實的回去布局文件中設置碉怔。
Toast打印出來的文字都往上偏移了。這里也是我疏忽的地方禁添,因為在布局文件中設置是對View生效撮胧,而到了theme進行設置則是對Window生效了,兩者在實現(xiàn)上就不一樣了上荡。所以趴樱,最終只能改回原來的方式去實現(xiàn)馒闷。
實踐總結
最后做一下小小的總結:
- 方式一適用于app中沒有導航欄,且整體的背景是一張圖片的界面叁征;
- 方式二適用于app中導航欄顏色為純色的界面纳账;
- android:fitsSystemWindows設置要在布局文件中,不要到theme中設置捺疼;
怎樣疏虫,介紹到這里,你會使用 Translucent System Bar 了嗎啤呼?趕快到你的app中引入吧卧秘!
補充更新(2016-02-19)
一些熱心的網(wǎng)友反饋,在Android 4.4平臺上使用第二種方法失效官扣。我立馬跑到Android4.4的真機運行一遍翅敌,果真出現(xiàn)下面的bug,頂部變成黑白漸變了惕蹄。
在此蚯涮,先為自己的疏忽向廣大讀者說聲抱歉。以后會最大程度的避免這種低級錯誤的產(chǎn)生卖陵。下面給出此Bug的修復方案:
第一步:去到 ColorTranslucentBarActivity 的布局文件中遭顶,將布局劃分成為標題布局和內容布局兩部分;
第二步:將 ColorTranslucentBarActivity 的根布局顏色設置與標題布局的顏色一致泪蔫,并將內容布局設置為白色棒旗;
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/color_31c27c"
android:fitsSystemWindows="true"
android:orientation="vertical">
<!--標題布局-->
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="55dp"
android:background="@color/color_31c27c">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:text="QQ Music"
android:textColor="@android:color/white"
android:textSize="20sp" />
</RelativeLayout>
<!--內容布局-->
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@android:color/white"
android:orientation="vertical">
<Button
android:id="@+id/btn_show_toast"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Show a toast" />
</LinearLayout>
</LinearLayout>
經(jīng)過以上兩步,即可在 4.4 平臺上實現(xiàn) Translucent System Bar 的效果 撩荣。最后附上修復bug后的效果圖一張铣揉。
補充更新(2016-02-22)
很多童鞋反應,在每個布局文件中都要寫上 android:fitsSystemWindows="true" ,有沒有更佳方便的方法婿滓,本人當時沒有思路老速。今天收到coder_sharp 童鞋反饋的一種更為簡便的思路
個人把他的思路,整理成代碼凸主,如下:
public abstract class TranslucentBarBaseActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
supportRequestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(getLayoutResId());//把設置布局文件的操作交給繼承的子類
ViewGroup contentFrameLayout = (ViewGroup) findViewById(Window.ID_ANDROID_CONTENT);
View parentView = contentFrameLayout.getChildAt(0);
if (parentView != null && Build.VERSION.SDK_INT >= 14) {
parentView.setFitsSystemWindows(true);
}
}
/**
* 返回當前Activity布局文件的id
*
* @return
*/
abstract protected int getLayoutResId();
}
所有需要實現(xiàn)效果的界面繼承以上的父類橘券,并實現(xiàn) getLayoutResId 抽象方法即可,就可以不用在布局文件中不斷做重復操作了卿吐,具體代碼詳見工程中的 TranslucentBarBaseActivity 和 BestTranslucentBarActivity旁舰。
補充更新(2016-02-25)
近幾天在琢磨 Material Design 的一些新控件效果,意外的發(fā)現(xiàn)上面提到的第二種方式嗡官,在將原 values-v21/style.xml
<style name="ColorTranslucentTheme" parent="Theme.AppCompat.Light.DarkActionBar">
....
....
<item name="android:statusBarColor">@color/color_31c27c</item>
</style>
換成
<style name="ColorTranslucentTheme" parent="Theme.AppCompat.Light.DarkActionBar">
....
....
<item name="android:statusBarColor">@android:color/transparent</item>
</style>
之后箭窜,依舊可以實現(xiàn)同樣的效果。那么衍腥,到了這里你就可以發(fā)現(xiàn)磺樱,上面提到的兩種方式從本質上其實是殊途同歸(最終總結得到的就是一種方式)纳猫!
我是熱愛技術,喜歡開源和分享的Clock竹捉,很享受寫文和其他開發(fā)者們探討問題的樂趣芜辕,其中很多bug都是其他細心的開發(fā)者發(fā)現(xiàn)的。如果你對我寫的內容感興趣块差,歡迎關注我的簡書侵续,我很樂意把我開發(fā)中一些有趣的東西分享到簡書中來(雖然目前僅僅是Android,但相信以后肯定會有其他內容的)憨闰,希望與其他開發(fā)者共同探討状蜗,共同進步!
分享即美德鹉动,最后附上源代碼地址:
https://github.com/D-clock/AndroidSystemUiTraining
歡迎關注我的簡書轧坎,以及:
- 知乎:https://www.zhihu.com/people/d_clock
- 知乎專欄(技術視界):https://zhuanlan.zhihu.com/coderclock
- 知乎專欄(閱讀視界):https://zhuanlan.zhihu.com/readerclock
- 個人博客:http://blog.coderclock.com/
-
微信公眾號: