Android状态栏设置

  • 首先需要知道状态栏能设置哪些内容。
    (1)状态栏是否显示
    (2)状态栏颜色设置
    (3)状态栏图标和字体颜色修改
  • 其次,了解“沉浸模式”。
  • 最后,如何通过Theme实现状态栏颜色修改。

1. 状态栏隐藏/显示

默认情况下,状态栏是显示的,那如何隐藏/显示呢?

  • 方法1:设置Theme主题
    在AndroidManifest.xml文件中修改theme为android:theme=”@android:style/Theme.NoTitleBar.Fullscreen”
  • 方法2:在setContentView方法前执行如下代码
requestWindowFeature(Window.FEATURE_NO_TITLE) 
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,WindowManager.LayoutParams.FLAG_FULLSCREEN);
  • 方法3:通过View的setSystemUiVisibility方法

View.SYSTEM_UI_FLAG_VISIBLE:显示状态栏,Activity不全屏显示(恢复到有状态的正常情况)。
View.INVISIBLE:隐藏状态栏,同时Activity会伸展全屏显示。
View.SYSTEM_UI_FLAG_FULLSCREEN:Activity全屏显示,且状态栏被隐藏覆盖掉。
View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN:Activity全屏显示,但状态栏不会被隐藏覆盖,状态栏依然可见,Activity顶端布局部分会被状态遮住。
View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION:效果同View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
View.SYSTEM_UI_LAYOUT_FLAGS:效果同View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
View.SYSTEM_UI_FLAG_HIDE_NAVIGATION:隐藏虚拟按键(导航栏)。有些手机会用虚拟按键来代替物理按键。
View.SYSTEM_UI_FLAG_LOW_PROFILE:状态栏显示处于低能显示状态(low profile模式),状态栏上一些图标显示会被隐藏。

  • 方法4:getWindow().addFlags getWindow().clearFlags
getWindow().addFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN) //隐藏状态栏 
getWindow().clearFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN) //显示状态栏

2. 状态栏颜色

默认是Dark样式,字体颜色为白色。
android4.4之前的版本,状态栏的背景色是不能修改的。在4.4版本,google增加了设置状态栏背景透明的方法。在5.x以上版本,google新增了设置状态栏背景色的方法。

2.1 android4.4 版本设置状态栏颜色

  • 首先把状态栏设置为透明
getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);

设置为透明后,页面布局从状态栏开始,状态栏遮盖在页面布局上。

  • 其次,修改状态栏颜色
    本质上并不是修改状态栏颜色,而是修改了页面布局某组件的背景色,而透过透明状态栏看到的就是这个组件。实现的方式有两种:

方法1:适配系统窗口
android:fitsSystemWindows="true"
在页面布局的最外层设置。设置后的效果:状态栏的颜色显示最外层View的背景色

方法2:占位法

ViewGroup rootView = (ViewGroup) getWindow().getDecorView().findViewById(android.R.id.content);
rootView.setPadding(0, getStateBarHeight(), 0, 0);
// 根布局添加占位状态栏
ViewGroup decorView = (ViewGroup) getWindow().getDecorView();
View statusBarView = new View(this);
ViewGroup.LayoutParams lp = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
                        getStateBarHeight());
statusBarView.setBackgroundColor(color);
decorView.addView(statusBarView, lp);

2.2 android 5.+及以上系统设置状态栏颜色

android5.+新增状态栏颜色设置的方法

window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);  
window.setStatusBarColor(colorId);  

android5.+以上系统,我自定义了两个方法:

  • setStatusColor(Activity activity,int color)
    将状态栏背景色设置为任意颜色(除白色外)
  • setStatusColorWhite(Activity activity)
    android6.0+以上系统将状态栏背景色设置为白色,字体设置为黑色样式。而android5.+不能设置状态栏字体颜色,如果状态栏设置为白色,则状态栏字体颜色看不清(默认白色),因此设置为透明色。
    调用哪个方法视具体情况而定,方法的实现见StatusBarUtils工具类。

3. 状态栏字体和图标样式设置

在6.0以后Google增加了修改字体颜色的方法。
系统默认的状态栏字体样式是白色。可通过代码设置为黑字。

//设置为黑色
window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR);

//设置为白色
window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN);

4. 沉浸模式

沉浸模式必须在android4.4及其以上版本才能实现。
沉浸模式可理解为Activity全屏,但是状态栏覆盖显示在界面顶部。沉浸模式效果:

状态栏背景透明

状态栏背景透明,覆盖在图片上
具体方法见:StatusBarUtils.setImmersionModel(Activity activity)

5. 通过设置主题style实现

values文件夹中的styles文件:

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

    <style name="AppTheme.Main" parent="AppTheme"/>

values-v19文件夹中的styles文件:

    <style name="AppTheme.Main" parent="AppTheme">
        <item name="android:windowTranslucentStatus">true</item>
    </style>

values-v21文件夹中的styles文件:

    <style name="AppTheme.Main" parent="AppTheme">
        <item name="android:windowTranslucentStatus">true</item>
        <!--Android 5.x开始需要把颜色设置透明,否则导航栏会呈现系统默认的浅灰色-->
        <item name="android:windowTranslucentNavigation">true</item>
    </style>

android:windowTranslucentNavigation对应的是底部的虚拟键的背景。

6. StatusBarUtils工具类

public class StatusBarUtils {


    /**
     * 设置状态栏颜色
     * 状态栏没有覆盖在页面布局上
     * 设置为白色,使用{@link StatusBarUtils#setStatusColorWhite(Activity)} 方法
     * @param activity  activity
     * @param color 状态栏颜色
     */
    public static void setStatusColor(Activity activity,int color){

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
            //5.0 以上直接设置状态栏颜色
            Window window = activity.getWindow();
            //After LOLLIPOP not translucent status bar
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            //Then call setStatusBarColor.
            window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
            activity.getWindow().setStatusBarColor(color);
//
        } else if(Build.VERSION.SDK_INT == Build.VERSION_CODES.KITKAT){
            activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            ViewGroup rootView = (ViewGroup) activity.getWindow().getDecorView().findViewById(android.R.id.content);
            rootView.setPadding(0, getStateBarHeight(activity), 0, 0);
            //根布局添加占位状态栏
            ViewGroup decorView = (ViewGroup) activity.getWindow().getDecorView();
            View statusBarView = new View(activity);
            ViewGroup.LayoutParams lp = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
                    getStateBarHeight(activity));
            statusBarView.setBackgroundColor(color);
            decorView.addView(statusBarView, lp);
        }

    }

    public static int getStateBarHeight(Activity activity){
        int result = 0;
        int resourceId = activity.getResources().getIdentifier("status_bar_height", "dimen", "android");
        if (resourceId > 0) {
            result = activity.getResources().getDimensionPixelSize(resourceId);
        }
        return result;
    }

    /**
     * 状态栏颜色设置为白色
     * 状态栏没有覆盖在页面布局上
     * android6.0以上系统,状态栏字体设置为黑色
     * android5.+系统,状态栏设置为透明
     * android4.4 系统,状态栏设置为透明
     * android 4.4以下系统,不能设置状态栏颜色
     * @param activity
     */
    public static void setStatusColorWhite(Activity activity){

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            //5.0 以上直接设置状态栏颜色
            Window window = activity.getWindow();
            //After LOLLIPOP not translucent status bar
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            //Then call setStatusBarColor.
            window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
            activity.getWindow().setStatusBarColor(Color.WHITE);
            //设置状态栏字体颜色
            window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR);
        } else if(Build.VERSION.SDK_INT == Build.VERSION_CODES.LOLLIPOP){
            //5.+ 不能设置状态栏字体颜色,状态栏直接设置为白色,则状态栏字体颜色看不清,因此设置为透明色
            setTranslucentStatus(activity);
        }else if(Build.VERSION.SDK_INT == Build.VERSION_CODES.KITKAT){
            activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            ViewGroup rootView = (ViewGroup) activity.getWindow().getDecorView().findViewById(android.R.id.content);
            rootView.setPadding(0, getStateBarHeight(activity), 0, 0);
            //根布局添加占位状态栏
            ViewGroup decorView = (ViewGroup) activity.getWindow().getDecorView();
            View statusBarView = new View(activity);
            ViewGroup.LayoutParams lp = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
                    getStateBarHeight(activity));
            statusBarView.setBackgroundColor(Color.WHITE);
            decorView.addView(statusBarView, lp);
        }

    }


    /**
     * 设置状态栏透明
     * @param activity
     */
    public static void setTranslucentStatus(Activity activity) {
        //设置5.0及其以上系统状态栏透明
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
            Window window = activity.getWindow();
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            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);
        }
    }


    /**
     * 设置沉浸模式,并且状态栏颜色为透明
     * 其中状态栏为透明且覆盖在页面布局上
     * android系统必须4.4及其以上
     * @param activity activity
     */
    public static void setImmersionModel(Activity activity) {
        //设置5.0及其以上系统状态栏透明
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
            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) {
            activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        }
    }

}

第三方库:compile 'com.jaeger.statusbarutil:library:1.4.0'

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 230,622评论 6 544
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 99,716评论 3 429
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 178,746评论 0 383
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 63,991评论 1 318
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 72,706评论 6 413
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 56,036评论 1 329
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 44,029评论 3 450
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 43,203评论 0 290
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 49,725评论 1 336
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 41,451评论 3 361
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 43,677评论 1 374
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 39,161评论 5 365
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 44,857评论 3 351
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 35,266评论 0 28
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 36,606评论 1 295
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 52,407评论 3 400
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 48,643评论 2 380

推荐阅读更多精彩内容

  • 前言:关于这个状态栏变色到底叫透明状态栏、沉浸状态栏、变色状态栏,可能大家在看网上文章时也是各抒己见,概念乱七八糟...
    芒果味的你呀阅读 13,899评论 11 50
  • 前言 首先请大家看几张图: 以上的效果,一般我们统称为沉浸式状态栏。其实,这种叫法不是很准确,而且也没有沉浸式状态...
    宇是我阅读 3,921评论 2 28
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,775评论 25 708
  • 唇启玉齿润香舌 明眸善睐言无多 料是无意念某人 一泓镜水影孤月
    狂奔的路痴阅读 245评论 0 0
  • 这么久的相处,这么久的分别,感觉和父母成了最熟悉的陌生人,陌生纯粹是精神上的,所以我们之间的焦点都集中在了生活上,...
    LAZYCAT窝阅读 433评论 0 0