安卓沉浸式状态栏初体验

初次接触沉浸式,本文先从纯色入手。

安卓状态栏沉浸式从大于等于19即安卓4.4开始有了第一种实现方案即使用SystemBarTintManager:

 

            Window window =activity.getWindow();
            window.setFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS,
                    WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            //使用SystemBarTint库使4.4版本状态栏变色,需要先将状态栏设置为透明
            SystemBarTintManager tintManager = new SystemBarTintManager(activity);
            tintManager.setStatusBarTintEnabled(true);
            tintManager.setStatusBarTintColor(color);

其中,后面的调用需要先把状态栏设置为透明。

而在android21即安卓5.0之后,安卓给出了比较简单的设置状态栏颜色的方法:

            Window window = activity.getWindow();
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
            window.setStatusBarColor(color);
这种方法则需要标志位先清除状态栏半透明标志,使能回执系统条背景标志。

经过测试,这种方法确实会设置状态栏的颜色,不过状态栏上面会根据系统不同,有不同样式的蒙层,比如MIUI8里面会有一个爆丑的蒙层。

如果不需要,可以退回到第一种方案。


沉浸式不仅需要改变背景的颜色,还需要改变文字的颜色。

文字默认是白色的,当我们使用较浅的颜色作为状态栏背景时,文字是看不清楚的。

改变文字的颜色,目前为止方法有限。

到了android6.0之后,谷歌才放出了修改文字为深色的方法:

            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
            window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN|View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR);
其中,SYSTEM_UI_FLAG_LIGHT_STATUS_BAR为6.0才有的标记,调用需要标志位有以上限制。

除了6.0以外,各个手机厂商有自己的修改文字深浅风格的方法,比如MIUI:

    private static boolean setMiuiStatusBarMode(Window window, boolean dark) {
        boolean result = false;
        if (window != null) {
            Class clazz = window.getClass();
            try {
                int darkModeFlag = 0;
                Class layoutParams = Class.forName("android.view.MiuiWindowManager$LayoutParams");
                Field  field = layoutParams.getField("EXTRA_FLAG_STATUS_BAR_DARK_MODE");
                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);//清除黑色字体
                }
                result=true;
            }catch (Exception e){

            }
        }
        return result;
    }
再比如魅族的Flyme:

    private static boolean setFlymeStatusBarMode(Window window, boolean dark) {
        boolean result = false;
        if (window != null) {
            try {
                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);
                result = true;
            } catch (Exception e) {

            }
        }
        return result;
    }


最后给出完整的代码:
/**
 * 顶部状态栏帮助类
 * Created by liangy on 2017/1/11.
 */
public class StatusBarHelper {

    // 背景色
    public static final int COLOR_DARK = Color.parseColor("#009EE6");
    public static final int COLOR_LIGHT = Color.parseColor("#FFFFFF");

    // 明暗模式:决定状态栏文字渲染的深浅
    public static final int LIGHT_MODE_NONE = 0;
    public static final int LIGHT_MODE_ANDROID_M = 1;
    public static final int LIGHT_MODE_MI = 2;
    public static final int LIGHT_MODE_FLYME = 3;

    private static int getLightMode(){
        int result = LIGHT_MODE_NONE;
        if (RomHelper.isMiui()){
            result = LIGHT_MODE_MI;
        }else if (RomHelper.isFlyme()){
            result = LIGHT_MODE_FLYME;
        }else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M){
            result = LIGHT_MODE_ANDROID_M;
        }
        return result;
    }

    /**
     *
     * @param activity
     * @param dark 把文字设置成深色
     * @return
     */
    public static boolean setStatusBarMode(Activity activity, boolean dark){

        // step 1. 根据模式调整状态栏状态
        int mode = getLightMode();
        boolean result = false;
        switch (mode){
            case LIGHT_MODE_NONE:
                break;
            case LIGHT_MODE_ANDROID_M:
                result = setAndroidMStatusBarMode(activity.getWindow(), dark);
                break;
            case LIGHT_MODE_MI:
                result = setMiuiStatusBarMode(activity.getWindow(), dark);
                break;
            case LIGHT_MODE_FLYME:
                result = setFlymeStatusBarMode(activity.getWindow(), dark);
                break;
        }

        // step 2. 如果能够成功修改图标和文字的模式,则继续修改背景色与之匹配
        if (true){
            if (dark){
                StatusBarHelper.setStatusBarColor(activity, COLOR_LIGHT);
            }else{
                StatusBarHelper.setStatusBarColor(activity, COLOR_DARK);
            }
        }
        return result;
    }

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

        /**
         * 使用这段会带上安卓默认的蒙层,特别在MIUI上面丑的一比
         */
        /*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);
            return true;
        } else */
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT){
            Window window =activity.getWindow();
            window.setFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS,
                    WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            //使用SystemBarTint库使4.4版本状态栏变色,需要先将状态栏设置为透明
            SystemBarTintManager tintManager = new SystemBarTintManager(activity);
            tintManager.setStatusBarTintEnabled(true);
            tintManager.setStatusBarTintColor(color);
            return true;
        }
        return false;
    }

    /**
     * 设置状态栏图标为深色和魅族特定的文字风格
     * 可以用来判断是否为Flyme用户
     * @param window 需要设置的窗口
     * @param dark 是否把状态栏字体及图标颜色设置为深色
     * @return  boolean 成功执行返回true
     *
     */
    private static boolean setFlymeStatusBarMode(Window window, boolean dark) {
        boolean result = false;
        if (window != null) {
            try {
                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);
                result = true;
            } catch (Exception e) {

            }
        }
        return result;
    }

    /**
     * 设置状态栏字体图标为深色,需要MIUIV6以上
     * @param window 需要设置的窗口
     * @param dark 是否把状态栏字体及图标颜色设置为深色
     * @return  boolean 成功执行返回true
     *
     */
    private static boolean setMiuiStatusBarMode(Window window, boolean dark) {
        boolean result = false;
        if (window != null) {
            Class clazz = window.getClass();
            try {
                int darkModeFlag = 0;
                Class layoutParams = Class.forName("android.view.MiuiWindowManager$LayoutParams");
                Field  field = layoutParams.getField("EXTRA_FLAG_STATUS_BAR_DARK_MODE");
                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);//清除黑色字体
                }
                result=true;
            }catch (Exception e){

            }
        }
        return result;
    }

    private static boolean setAndroidMStatusBarMode(Window window, boolean dark){
        if (!dark){
            window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN);
        }else{
            window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
            window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
            window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN|View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR);
        }
        return true;
    }
}

参考链接:https://www.zhihu.com/question/31994153

吐槽一下:回答赞成最高的那个哥们给的那个链接的代码也是可以

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值