Android 状态栏工具——一行代码实现状态栏字体变黑

状态栏(一)
状态栏(二)

经过前两篇的介绍我们对如何修改状态栏的效果有了大致的了解,本篇介绍一种使用更加简单的方式,可以先拉到最后查看效果图。

设置主题效果
注意: 要设置fitsSystemWindows为true

<style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar">

<item name="android:fitsSystemWindows">true</item>
<item name="colorPrimary">@color/colorPrimary</item>
<item name="colorPrimaryDark">@color/colorPrimaryDark</item>
<item name="colorAccent">@color/colorAccent</item>
</style>
1
2
3
4
5
6
7
参考SystemBarTint对状态栏的改变方式,添加下面这个类方法,用来设置 API 19 以上的状态栏效果

我们的界面绘制时在DecorView中的,这个类的核心思想就是:
1.获取DecorView
2.绘制状态栏高度的View
3.将view添加到DecorView中

所以我们对状态栏的操作,其实就是对view的操作。在这个类中默认view不显示,改变状态栏的效果时,要调用

setStatusBarTintEnabled(true)
1
注意:使用这个方法的前提是 状态栏是透明的,因为进行了相关的判断(可以自行修改)

public class SystemBarTintManager {

public static final int DEFAULT_TINT_COLOR = 0x99000000;
private boolean mStatusBarAvailable;
private boolean mStatusBarTintEnabled;
private View mStatusBarTintView;

@TargetApi(19)
public SystemBarTintManager(Activity activity) {

    Window win = activity.getWindow();
    //获取DecorView
    ViewGroup decorViewGroup = (ViewGroup) win.getDecorView();

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        // 检查主题中是否有透明的状态栏
        int[] attrs = {android.R.attr.windowTranslucentStatus};
        TypedArray a = activity.obtainStyledAttributes(attrs);
        try {
            mStatusBarAvailable = a.getBoolean(0, false);
        } finally {
            a.recycle();
        }
        WindowManager.LayoutParams winParams = win.getAttributes();
        int bits = WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS;//状态栏透明
        if ((winParams.flags & bits) != 0) {
            mStatusBarAvailable = true;
        }
    }
    if (mStatusBarAvailable) {
        setupStatusBarView(activity, decorViewGroup);
    }
}

/**
 * 初始化状态栏
 *
 * @param context
 * @param decorViewGroup
 */
private void setupStatusBarView(Activity context, ViewGroup decorViewGroup) {
    mStatusBarTintView = new View(context);
    //设置高度为Statusbar的高度
    FrameLayout.LayoutParams params = new FrameLayout.LayoutParams(FrameLayout.LayoutParams.MATCH_PARENT, getStatusBarHeight(context));
    params.gravity = Gravity.TOP;
    mStatusBarTintView.setLayoutParams(params);
    mStatusBarTintView.setBackgroundColor(DEFAULT_TINT_COLOR);
    //默认不显示
    mStatusBarTintView.setVisibility(View.GONE);
    //decorView添加状态栏高度的View
    decorViewGroup.addView(mStatusBarTintView);
}

/**
 * 获取状态栏高度
 *
 * @param activity
 * @return
 */
private int getStatusBarHeight(Activity activity) {
    int statusBarHeight = 0;
    int resourceId = activity.getResources().getIdentifier("status_bar_height", "dimen", "android");
    if (resourceId > 0) {
        statusBarHeight = activity.getResources().getDimensionPixelSize(resourceId);
    }
    return statusBarHeight;
}

/**
 * 显示状态栏
 */
public void setStatusBarTintEnabled(boolean enabled) {
    mStatusBarTintEnabled = enabled;
    if (mStatusBarAvailable) {
        mStatusBarTintView.setVisibility(enabled ? View.VISIBLE : View.GONE);
    }
}

/**
 * 设置状态栏颜色
 *
 * @param color
 */
public void setStatusBarTintColor(int color) {
    if (mStatusBarAvailable) {
        mStatusBarTintView.setBackgroundColor(color);
    }
}

}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
工具类代码
在这段代码中当我们的API 为4.4.+ 但是小于 5.0时使用上述的类

/**
* 修改状态栏颜色,支持4.4以上版本
* @param colorId 颜色
*/
public static void setStatusBarColor(Activity activity, int colorId) {

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        Window window = activity.getWindow();
        window.setStatusBarColor(colorId);
    } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        //使用SystemBarTintManager,需要先将状态栏设置为透明
        setTranslucentStatus(activity);
        SystemBarTintManager systemBarTintManager = new SystemBarTintManager(activity);
        systemBarTintManager.setStatusBarTintEnabled(true);//显示状态栏
        systemBarTintManager.setStatusBarTintColor(colorId);//设置状态栏颜色
    }
}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
完整代码

import android.annotation.TargetApi;
import android.app.Activity;
import android.graphics.Color;
import android.os.Build;
import android.support.annotation.IntDef;
import android.view.View;
import android.view.Window;
import android.view.WindowManager;

import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

/**

  • Created by rhm on 2018/1/13.
    */

public class StatusBarUtil {
public final static int TYPE_MIUI = 0;
public final static int TYPE_FLYME = 1;
public final static int TYPE_M = 3;//6.0

@IntDef({TYPE_MIUI,
        TYPE_FLYME,
        TYPE_M})
@Retention(RetentionPolicy.SOURCE)
@interface ViewType {
}

/**
 * 修改状态栏颜色,支持4.4以上版本
 * @param colorId 颜色
 */
public static void setStatusBarColor(Activity activity, int colorId) {

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        Window window = activity.getWindow();
        window.setStatusBarColor(colorId);
    } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        //使用SystemBarTintManager,需要先将状态栏设置为透明
        setTranslucentStatus(activity);
        SystemBarTintManager systemBarTintManager = new SystemBarTintManager(activity);
        systemBarTintManager.setStatusBarTintEnabled(true);//显示状态栏
        systemBarTintManager.setStatusBarTintColor(colorId);//设置状态栏颜色
    }
}

/**
 * 设置状态栏透明
 */
@TargetApi(19)
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.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) { //19
        activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
    }
}

/**
 * 设置沉浸式状态栏
 *
 * @param fontIconDark 状态栏字体和图标颜色是否为深色
 */
public static void setImmersiveStatusBar(Activity activity,boolean fontIconDark) {
    setTranslucentStatus(activity);
    if (fontIconDark) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            setStatusBarFontIconDark(activity,TYPE_M);
        } else if (OSUtils.isMiui()) {
            setStatusBarFontIconDark(activity,TYPE_MIUI);
        } else if (OSUtils.isFlyme()) {
            setStatusBarFontIconDark(activity,TYPE_FLYME);
        } else {//其他情况下我们将状态栏设置为灰色,就不会看不见字体
            setStatusBarColor(activity,Color.LTGRAY);//灰色
        }
    }
}

/**
 * 设置文字颜色
 */
public static void setStatusBarFontIconDark(Activity activity,@ViewType int type) {
    switch (type) {
        case TYPE_MIUI:
            setMiuiUI(activity,true);
            break;
        case TYPE_M:
            setCommonUI(activity);
            break;
        case TYPE_FLYME:
            setFlymeUI(activity,true);
            break;
    }
}

//设置6.0的字体
public static void setCommonUI(Activity activity) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        activity.getWindow().getDecorView().setSystemUiVisibility(
                View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
                        | View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR);
    }
}

//设置Flyme的字体
public static void setFlymeUI(Activity activity,boolean dark) {
    try {
        Window window = activity.getWindow();
        WindowManager.LayoutParams lp = window.getAttributes();
        Field darkFlag = WindowManager.LayoutParams.class.getDeclaredField("MEIZU_FLAG_DARK_STATUS_BAR_ICON");
        Field meizuFlags = WindowManager.LayoutParams.class.getDeclaredField("meizuFlags");
        darkFlag.setAccessible(true);
        meizuFlags.setAccessible(true);
        int bit = darkFlag.getInt(null);
        int value = meizuFlags.getInt(lp);
        if (dark) {
            value |= bit;
        } else {
            value &= ~bit;
        }
        meizuFlags.setInt(lp, value);
        window.setAttributes(lp);
    } catch (Exception e) {
        e.printStackTrace();
    }
}

//设置MIUI字体
public static void setMiuiUI(Activity activity,boolean dark) {
    try {
        Window window = activity.getWindow();
        Class clazz = activity.getWindow().getClass();
        Class layoutParams = Class.forName("android.view.MiuiWindowManager$LayoutParams");
        Field field = layoutParams.getField("EXTRA_FLAG_STATUS_BAR_DARK_MODE");
        int darkModeFlag = field.getInt(layoutParams);
        Method extraFlagField = clazz.getMethod("setExtraFlags", int.class, int.class);
        if (dark) {    //状态栏亮色且黑色字体
            extraFlagField.invoke(window, darkModeFlag, darkModeFlag);
        } else {
            extraFlagField.invoke(window, 0, darkModeFlag);
        }
    } catch (Exception e) {
        e.printStackTrace();
    }
}

}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
通用方法为
设置字体颜色为黑色
setImmersiveStatusBar(Activity activity,boolean fontIconDark)
设置状态栏透明
setTranslucentStatus(Activity activity)
设置状态栏的颜色
setStatusBarColor(Activity activity, int colorId)
1
2
3
4
5
6
使用方式
public class MainActivity extends AppCompatActivity {

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    StatusBarUtil.setImmersiveStatusBar(this,true);
}

}
1
2
3
4
5
6
7
8
9
第二个参数是设置是否为黑色字体。
我们还可以调用下面代码设置状态栏颜色。

StatusBarUtil.setStatusBarColor(Activity activity, int colorId)
1
效果

使用时只需要一行代码就可以改变状态栏的字体颜色,是不是很简单呢。

请在github上下载完整的内容,工具类在statusbarutil中。
博客地址:
https://blog.csdn.net/Fessible_Max/article/details/79052063

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

推荐阅读更多精彩内容