Material Design系列之沉浸式狀態欄詳解以及適配

Material Design系列之沉浸式狀態欄詳解以及適配

簡介

沉浸式Translucent官方的解釋是:整個APP充滿整個屏幕,沒有顯示狀態欄和底部導航欄,像看電影或看小說一樣。

現在所說的沉浸式狀態欄就是系統的狀態欄的顏色變得和頂部導航欄顏色一致或者相近,類似QQ的效果。

正式開發

Google在Android 5.0及以上自動實現了沉浸式效果,狀態欄的顏色跟隨style里主題的colorPrimaryDark屬性的顏色一致

<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
        <!-- Customize your theme here. -->
        <item name="colorPrimary">@color/colorPrimary</item>
        <item name="colorPrimaryDark">@color/colorPrimaryDark</item>
        <item name="colorAccent">@color/colorAccent</item>
    </style>

也可以通過設置values-v21中樣式屬性來解決:

<item name="android:statusBarColor">@color/colorPrimary</item>

或者通過代碼來設置:

getWindow().setStatusBarColor()來設置

Android 4.4以上5.0以下解決

API低于KitKat(19)版本無法實現,在4.4以上可以設置狀態欄為透明的。

方法1、新建values-19資源文件夾,在對應的主題里面設置:

<item name="android:windowBackground">@color/colorAccent</item>
<item name="android:windowTranslucentStatus" tools:targetApi="kitkat">true</item>

或者用代碼在setContentView之前設置:

getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);

然后給布局最外層布局設置你想要在狀態欄填充的背景顏色或設置<item name="android:windowBackground">@color/colorPrimary</item>,并設置android:fitsSystemWindows="true"。然后給剩下的布局加上一層正常的背景色。

注意不要給toolbar設置android:fitsSystemWindows="true",會造成一些問題。比如scrollview里嵌套edittext,彈出軟件盤會造成toolbar出現錯位。

這樣做的話就要多一層背景。

方法2:修改toolbar或者其他導航欄的高度

給這些導航欄設置在原本的高度上增加一個系統狀態欄的高度,并設置一個paddingTop的值,這個值也是系統狀態欄的高度。

還是在values-v19設置<item name="android:windowTranslucentStatus" tools:targetApi="kitkat">true</item>
或者代碼在setContentView之前設置:getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);

獲取狀態欄的高度status_bar_height:

/**
     * 利用反射獲取狀態欄高度
     * @return
     */
    public static int getStatusBarHeight(Activity activity) {
        int result = 0;
        //獲取狀態欄高度的資源id
        int resourceId = activity.getResources().getIdentifier("status_bar_height", "dimen", "android");
        if (resourceId > 0) {
            result = activity.getResources().getDimensionPixelSize(resourceId);
        }
        return result;
    }

    /**
        這個方法詳細一點,知道怎么獲取的
      * 獲取狀態欄的高度
      * @param context
      * @return
     */
    private int getStatusBarHeight(Context context) {
        // 反射手機運行的類:android.R.dimen.status_bar_height.
        int statusHeight = -1;
        try {
            Class<?> clazz = Class.forName("com.android.internal.R$dimen");
            Object object = clazz.newInstance();
            String heightStr = clazz.getField("status_bar_height").get(object).toString();
            int height = Integer.parseInt(heightStr);
            statusHeight = context.getResources().getDimensionPixelSize(height);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return statusHeight;
    }

然和把獲取到的值設置到頂部導航欄里去,增加toolbar的高度,并設置paddingTop的值,保證toolbar高度改變不會使里面的內容位置發生改變

ViewGroup.LayoutParams layoutParams = toolbar.getLayoutParams();
            layoutParams.height = layoutParams.height+getStatusBarHeight(this);
            layoutParams.width = WindowManager.LayoutParams.MATCH_PARENT;
            toolbar.setLayoutParams(layoutParams);
toolbar.setPadding(
    toolbar.getPaddingLeft(),
    toolbar.getPaddingTop()+getStatusBarHeight(this),
    toolbar.getPaddingRight(),
    toolbar.getPaddingBottom());

注意導航欄高度不能設置為wrap_content

方法3:給狀態欄增加一個占位視圖

在setContentView之前設置全屏,getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);然后獲取根布局的視圖,然后設置一個paddingTop的值,然后把你想要的顏色的一個布局填充進去

//        增加一個占位視圖
        ViewGroup rootView = (ViewGroup) this.getWindow().getDecorView().findViewById(android.R.id.content);
        rootView.setPadding(0, getStatusBarHeight(this), 0, 0);
        ViewGroup decorView = (ViewGroup) this.getWindow().getDecorView();
        View statusBarView = new View(this);
        ViewGroup.LayoutParams lp = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
                getStatusBarHeight(this));
        statusBarView.setBackgroundColor(Color.parseColor("#ffee66"));
        decorView.addView(statusBarView, lp);

注意頂部導航欄不要加android:fitsSystemWindows="true"

使用以上方案在當含有DrawLayout這種側滑菜單時,側滑菜單會出現下面這種情況:

側滑問題圖
方案一:

照樣設置getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);

然后給DrawLayout布局設置如下屬性:

android:fitsSystemWindows="true"
android:clipToPadding="false"

在使用上面方法2,修改頂部導航欄的高度,即可解決問題。

方案二

照樣設置getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);

然后給DrawLayout布局設置如下屬性:

android:fitsSystemWindows="true"
android:clipToPadding="false"

使用上面方法三直接添加占位視圖會出現占位視圖顯示在側滑界面上。這種情況的
解決方案就是將DrawLayout原有的內容視圖移除,然后新建一個線性布局,把狀態欄的占位布局添加進去,然后將這個線性布局添加到drawlayout中成為新的內容視圖。

 //要在內容布局增加狀態欄,否則會蓋在側滑菜單上
        ViewGroup rootView = (ViewGroup) this.getWindow().getDecorView().findViewById(android.R.id.content);
        //DrawerLayout 則需要在第一個子視圖即內容試圖中添加padding
        View parentView = rootView.getChildAt(0);
        LinearLayout linearLayout = new LinearLayout(this);
        linearLayout.setOrientation(LinearLayout.VERTICAL);
        View statusBarView = new View(this);
        ViewGroup.LayoutParams lp = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
                getStatusBarHeight(this));
        //設置外層狀態欄的顏色
        statusBarView.setBackgroundColor(Color.parseColor("#3F51B5"));
        //添加占位狀態欄到線性布局中
        linearLayout.addView(statusBarView, lp);
        //側滑菜單
        DrawerLayout drawerlayout = (DrawerLayout) parentView;
        //內容視圖
        View content = findViewById(R.id.coor_layout);
        //將內容視圖從 DrawerLayout 中移除
        drawerlayout.removeView(content);
        //將內容視圖添加到線性布局里
        linearLayout.addView(content, content.getLayoutParams());
        //將帶有占位狀態欄的新的線性布局內容視圖設置給 DrawerLayout
        drawerlayout.addView(linearLayout, 0);

如何給狀態欄填充漸變色

面對這種情況只能用填充占位視圖這一招了。給占位視圖設置drawable背景即可。只要是4.4以上的系統都要這樣操作。

狀態欄特殊情況:白底黑字

在原生系統中,Android 6.0以下的系統的狀態欄字體都是白色的,除了國內少數幾家廠商可以修改:小米和魅族。
這種情況下只能是適配6.0以上系統以及6.0以下的小米和魅族手機

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            this.getWindow()
                    .getDecorView()
                    .setSystemUiVisibility(View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR);
 }else{

 }

小米狀態欄變色官方文檔

魅族狀態欄變色官方文檔

經過測試,使用添加占位視圖的方式在小米狀態欄變色后無效,可以使用改變頂部導航欄高度來處理。魅族尚未測試。

遇到的問題

狀態欄透明導致android:windowSoftInputMode屬性失效,軟鍵盤遮住輸入框。

解決方案一:(尚未經過全面測試,可能別的手機上會出問題,小米可以)

頁面最外層布局改成ScrollView,并設置android:fitsSystemWindows="true"

解決方案二:

自行Google“Android中adjustResize失效的解決辦法”

參考文章:Android 沉浸式狀態欄的實現

Github示例代碼,歡迎star

?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,835評論 6 534
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,676評論 3 419
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,730評論 0 380
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,118評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,873評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,266評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,330評論 3 443
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,482評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,036評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,846評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,025評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,575評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,279評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,684評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,953評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,751評論 3 394
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,016評論 2 375

推薦閱讀更多精彩內容