Android UI之系统栏(System Bars)

Android系统栏包括状态栏(status bar)和导航栏(navigation bar)。

开发者可以对系统进行操作,具体包括:

  1. 弱化系统栏(Dimming the System Bars)
  2. 隐藏状态栏(Hiding the Status Bar)
  3. 隐藏导航栏(Hiding the Navigation Bar)
  4. 使用沉浸式全屏模式(Using Immersive Full-Screen Mode)
  5. 响应系统栏的改变(Responding to UI Visibility Changes)
  6. 状态栏上色
  7. 导航栏上色

弱化系统栏

原文参见

什么是弱化系统栏?简而言之就是弱化系统栏对用户的视觉冲击,可以使用户将更多注意力转移到App提供的内容。

要求:弱化系统栏操作,只有Android 4.0或更高版本的设备支持。

效果

  1. 状态栏消失(仍占位);导航栏图标变为小圆点
  2. 不会改变serContentView()方法中参数布局的大小
  3. 用户操作会使系统栏重新可见

实现

// This example uses decor view, but you can use any visible view.
View decorView = getActivity().getWindow().getDecorView();
int uiOptions = View.SYSTEM_UI_FLAG_LOW_PROFILE;
decorView.setSystemUiVisibility(uiOptions);

取消弱化系统栏:

View decorView = getActivity().getWindow().getDecorView();
// Calling setSystemUiVisibility() with a value of 0 clears
// all flags.
decorView.setSystemUiVisibility(0);

隐藏状态栏

原文参见

什么是隐藏状态栏?简而言之就是使状态栏区域的内容消失(占位隐藏);或让状态栏的整个区域都不可见(完全隐藏)。

分类

  1. 占位隐藏
  2. 完全隐藏

占位隐藏实现

这里介绍的实现方法基于Android 4.1及更高的版本。低版本的隐藏参见原文...

int uiOptions = View.SYSTEM_UI_FLAG_FULLSCREEN;//隐藏状态栏
visibleView.setSystemUiVisibility(uiOptions);

完全隐藏实现

这里介绍的实现方法基于Android 4.1及更高的版本。低版本的隐藏参见原文...

int uiOptions = View.SYSTEM_UI_FLAG_FULLSCREEN//隐藏状态栏
                |View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN;//无论状态栏隐藏与否,让布局在状态栏区域显示
visibleView.setSystemUiVisibility(uiOptions);

隐藏导航栏

原文参见

要求:隐藏导航栏操作在Android4.0版本引进。

分类

  1. 占位隐藏
  2. 完全隐藏

占位隐藏实现

int uiOptions = View. SYSTEM_UI_FLAG_HIDE_NAVIGATION;//隐藏导航栏
visibleView.setSystemUiVisibility(uiOptions);

完全隐藏实现

int uiOptions = View. SYSTEM_UI_FLAG_HIDE_NAVIGATION//隐藏导航栏
                |View. SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION;//无论导航栏隐藏与否,让布局在导航栏区域显示
visibleView.setSystemUiVisibility(uiOptions);

使用沉浸式全屏模式

原文参见

响应系统栏的改变

原文参见

状态栏上色

参考1


参考2

对于状态栏:

  1. 隐藏
  2. 显示

对于状态栏的隐藏,上文已经介绍。这里介绍状态栏按照需求进行显示的问题。那么,状态栏的显示有哪些需求呢?我分析如下:

  1. 指定状态栏背景颜色
  2. 指定状态栏元素颜色

对于第二个问题,待以后研究,这里只分析如何指定状态栏背景(色)。

API21及以上

设置状态栏背景色的实现可以调用如下方法:

public abstract class Window {
  /**
  For this to take effect,the window must be drawing the system bar backgrounds with 
  android.view.WindowManager.LayoutParams#FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS and 
  android.view.WindowManager.LayoutParams#FLAG_TRANSLUCENT_STATUS must not be set
  **/
  public abstract void setStatusColor(int color);
}

注意:

  1. 该方法在Android 5.0(API 21)
  2. 引入通过注释可知,要使方法生效,需要其他的操作配合:
    • window不能设置android.view.WindowManager.LayoutParams#FLAG_TRANSLUCENT_STATUS标签
    • window必须设置WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS标签

API20及以下

低版本的系统是不支持给状态栏着色的,但却可以通过透明状态栏+透明背景颜色来实现相同的效果。

  1. 将系统状态栏设置为透明
    • getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
  2. 设置Activity根布局的背景颜色为预期状态栏背景色

注意:当设置了状态栏为透明后,Activity会相当于一个FullScreen的全屏设置,窗口会占满整个屏幕,整体的内容会往上移动一段状态栏高度的距离,这样就会导致状态栏覆盖到我们的内容。这时,我们需要在根布局上设置android:fitsSystemWindows="true",这样系统会帮我们重新调整窗口的位置避免出现覆盖的情况(无非就是给我们的窗口加上一个padding值)

沉浸式状态栏

public static void immersingStatusBar(Activity activity) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
            //5.0以上
            Window window = activity.getWindow();
            //清除导航栏和状态栏的半透明属性
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            //设置系统的ui
            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) {
            //将窗口的状态栏设置成透明的
            activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        }
}

沉浸式导航栏

  • 调用如下方法

关键代码

public void immersingNvigationBar(Activity activity) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {//5.0以上
            Window window = activity.getWindow();
            
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION);

            window.getDecorView().setSystemUiVisibility(
                    View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION//布局在导航栏区域有值
                            | View.SYSTEM_UI_FLAG_LAYOUT_STABLE);//布局稳定

            window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
            window.setNavigationBarColor(Color.TRANSPARENT);
        } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {//4.4版本到5.0以下
            Window window = activity.getWindow();
            //将系统栏设置成透明的
            window.addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION);
        }
    }
  • 设置关键元素的margin(包含顶部元素和底部元素)

是否有导航栏

public boolean checkDeviceHasNavigationBar(Context context) {
        boolean hasNavigationBar = false;
        Resources rs = context.getResources();
        int id = rs.getIdentifier("config_showNavigationBar", "bool", "android");
        if (id > 0) {
            hasNavigationBar = rs.getBoolean(id);
        }
        try {
            Class systemPropertiesClass = Class.forName("android.os.SystemProperties");
            Method m = systemPropertiesClass.getMethod("get", String.class);
            String navBarOverride = (String) m.invoke(systemPropertiesClass, "qemu.hw.mainkeys");
            if ("1".equals(navBarOverride)) {
                hasNavigationBar = false;
            } else if ("0".equals(navBarOverride)) {
                hasNavigationBar = true;
            }
        } catch (Exception e) {
        }
        return hasNavigationBar;
    }

获取导航栏高度

    private int getNavigationBarHeight() {
        Resources resources = getResources();
        int resourceId = resources.getIdentifier("navigation_bar_height", "dimen", "android");
        int height = resources.getDimensionPixelSize(resourceId);
        Log.v("dbw", "Navi height:" + height);
        return height;
    }

设置底部元素margin

    //设置底部,也可以通过fitsSystemWindows属性实现,但对该属性暂不熟悉,因此使用下面方法保证底部元素不与导航栏重叠
    if(checkDeviceHasNavigationBar(this)){
        FrameLayout.LayoutParams param = (FrameLayout.LayoutParams) llBottom.getLayoutParams();
        param.bottomMargin = getNavigationBarHeight();
        llBottom.setLayoutParams(param);
    }

设置顶部元素margin

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            FrameLayout.LayoutParams param = (FrameLayout.LayoutParams) llBack.getLayoutParams();
            param.topMargin = getStateBarHeight();
            llBack.setLayoutParams(param);
}

沉浸式系统栏

public void immersingSystemBar(Activity activity) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {//5.0以上
            Window window = activity.getWindow();
            //清除导航栏和状态栏的半透明属性
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS
                    | WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION);

            //设置系统的ui
            window.getDecorView().setSystemUiVisibility(
                    View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION//布局在导航栏区域有值
                            | 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);
            window.setNavigationBarColor(Color.TRANSPARENT);
        } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {//4.4版本到5.0以下
                Window window = activity.getWindow();
                window.addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
                window.addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION);
        }
}

备注

  1. 青葱西点模块首页采用沉浸式系统栏
  2. 青葱西点商品详情界面采用沉浸式状态栏
  3. 罗曼蒂克首页采用沉浸式状态栏
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容