Android沉浸狀態(tài)欄(StatusBar)兼容方案

所謂"沉浸狀態(tài)欄"的實現(xiàn)需要兩點:

  1. 設置狀態(tài)欄為透明或者半透明狀態(tài);
  2. 整體布局可以置于狀態(tài)欄下方.

1. 狀態(tài)欄的配置

對于狀態(tài)欄的配置有兩種方式:

  1. 在manifest中配置Activity的theme屬性
    在theme文件中添加android:windowTranslucentStatus屬性(當然僅支持api19, 即Android4.4及以上, 且根據(jù)這個屬性名稱可知只是半透明);
    <style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar">
        ...
        <item name="android:windowTranslucentStatus">true</item>
    </style>

效果:

鍵盤隱藏 鍵盤顯示
Android5.0鍵盤未彈出
Android5.0鍵盤彈出
  1. 通過代碼動態(tài)設置
    相對于通過theme的屬性配置, 這種方案可以實現(xiàn)狀態(tài)欄全透明.
    /**
     * 設置狀態(tài)欄為全透明
     * 通過設置theme的方式無法達到全透明效果
     *
     * @param activity
     */
    @TargetApi(19)
    private static void transparencyStatusBar(Activity activity) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
            //5.0及其以上
            Window window = activity.getWindow();
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
                    | View.SYSTEM_UI_FLAG_LAYOUT_STABLE);
            window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
            window.setStatusBarColor(Color.TRANSPARENT);
        } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            //4.4及其以上
            activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        }
    }

以上兩種方案都沒能達到想要的效果. 如上圖所示,現(xiàn)在有兩個問題:

問題1. 整體布局上移, statusBar遮擋contentView顯示的內容;
問題2. 鍵盤遮擋了輸入框, 我們在manifest里給activity聲明的android:windowSoftInputMode="adjustResize"失效了;

針對以上兩個問題, 現(xiàn)在有兩種著手方案:

  1. 設置Activity的根布局 android:fitsSystemWindows="true"
  2. 頂部view添加paddingTop屬性, 高度取當前手機的statusbar高度

方案一, 可以使得整體contentView下移至statusBar下方, 同時問題2解決, 但是此時的statusBar顏色是整體background的顏色, 未能解決問題. 此方案放棄

下面著重介紹方案二:

2. 以上兩個問題的解決

2.1解決問題1: contentView上移, statusBar遮擋內容顯示

解決方案很簡單, 就是給頂部的view加個paddingTop, 而這個paddingTop的值取當前手機的statusbar高度.
比如這里我給自定義的toolbar添加了paddingTop

    //動態(tài)獲取statusBar的高度
    private static int sHeight = -1;
    public static int getStatusBarHeight(Context context) {
        if (sHeight == -1) {
            //獲取status_bar_height資源的ID
            int resourceId = context.getResources().getIdentifier("status_bar_height", "dimen", "android");
            if (resourceId > 0) {
                //根據(jù)資源ID獲取響應的尺寸值
                sHeight = context.getResources().getDimensionPixelSize(resourceId);
            }
        }
        return sHeight;
    }
    //給頂部的view添加paddingTop
    int height = getStatusBarHeight(context);
    customeToolbar.setPadding(root.getPaddingRight(), height + root.getPaddingTop(),
                              root.getPaddingLeft(), root.getPaddingBottom());

看效果


Android5.0
2.2 解決問題2: 鍵盤遮擋輸入框

經過測試發(fā)現(xiàn)影響adjustResize生效以及contentView上移 的關鍵是設置contentView根布局 android:fitsSystemWindows="true"

那么我們只需要
1.重寫View的fitSystemWindows(Rect insets)的方法, 使得其控制contentView位移的功能失效,

public class CustomInsetsLinearLayout extends LinearLayout {

    private int[] mInsets = new int[4];

    public CustomInsetsLinearLayout(Context context) {
        super(context);
    }

    public CustomInsetsLinearLayout(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public CustomInsetsLinearLayout(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }

    public final int[] getInsets() {
        return mInsets;
    }

    @Override
    protected final boolean fitSystemWindows(Rect insets) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            // Intentionally do not modify the bottom inset. For some reason,
            // if the bottom inset is modified, window resizing stops working.

            mInsets[0] = insets.left;
            mInsets[1] = insets.top;
            mInsets[2] = insets.right;

            insets.left = 0;
            insets.top = 0;
            insets.right = 0;
        }

        return super.fitSystemWindows(insets);
    }
}
  1. 同時配置android:fitsSystemWindows="true"保留對于adjustResize屬性的影響;
<?xml version="1.0" encoding="utf-8"?>
<com.example.j.verticaltextdemo.view.CustomInsetsLinearLayout
    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">

    <include layout="@layout/center_title_toolbar" />

    <Space
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1" />

    <EditText
        android:id="@+id/et1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="bottom"
        android:hint="請輸入" />

</com.example.j.verticaltextdemo.view.CustomInsetsLinearLayout>

最終效果展示

Android 5.0及其以上的顯示效果 Android4.4的顯示效果
解決問題2
Android4.4

3. 完全沉浸效果

  • 設置狀態(tài)欄全透明后的效果展示
Android6.0及其以上 Android5.x Android4.4
Android6.0
Android5.0
Android4.4

4. 進階篇: 淺色狀態(tài)欄及兼容性配置

目前市面上的淺色狀態(tài)欄基本都是 白底黑字, 比如 [微博]
支持這種設置的有Android6.0及其以上; MIUI v6及以上, Flyme 4.0及以上, 所以我的兼容方案如下:

    1. 對于支持[狀態(tài)欄透明 + 狀態(tài)欄文字圖標黑白切換]的機型(Android 6.0及以上,MIUI v6及以上, Flyme 4.0及以上), 全透明狀態(tài)欄 + 黑色文字圖標
    1. 對于僅支持[狀態(tài)欄透明]的機型(Android 4.4及其以上), 采用: 半透明狀態(tài)欄 + 白色文字圖標
    1. 對于不支持的機型(Android 4.4以下), 采用默認狀態(tài): 黑色狀態(tài)欄 + 白色文字圖標

具體實現(xiàn)

    /**
     * Android 6.0 原生
     *
     * @param activity
     * @param dark
     */
    private static void setAndroidNativeLightStatusBar(Activity activity, boolean dark) {
        //狀態(tài)欄字體圖標顏色
        View decor = activity.getWindow().getDecorView();
        if (dark) {
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
                decor.setSystemUiVisibility(View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR //淺色狀態(tài)欄(字體圖標白色)
                        | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN //contentView 全屏(置于statusbar之下)
                        | View.SYSTEM_UI_FLAG_LAYOUT_STABLE);
            }
        } else {
            decor.setSystemUiVisibility(View.SYSTEM_UI_FLAG_VISIBLE);
        }
    }

附:

具體代碼參見 https://github.com/JesseWo/ImmersedStatusBar

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

推薦閱讀更多精彩內容