1.绪论

现在基本上所有的应用都会去实现沉浸式状态栏,这个是应用的标配,如果你开发的应用没有,那这个吐槽点就多了,“这美工有审美观么”“程序猿这么菜,沉浸式都不会?”….. 咳咳….. 开个玩笑啊,各有各的设计思想,不能怪程序猿。

2.问题

那么说到沉浸式状态栏的问题是什么呢?不知道大家有没有遇到过,应用在android7.0系统以下的手机上运行,沉浸式状态栏是正常的,但是在7.0以上的手机上运行就感觉没有沉浸式了,是分层的。无论怎么修改状态栏背景色都没用,看下图:图1是7.0以下运行效果,图2是7.0以上运行效果。

android 首页变灰色 我的android手机 灰色_android 首页变灰色

android 首页变灰色 我的android手机 灰色_沉浸式状态栏_02

看到上面两幅图,明显感觉到7.0系统做了处理,在状态栏上蒙了一层灰色背景。

3.解决方法

首先,先普及下知识:
DecorView是整个Window界面的最顶层View,它只有一个子元素为LinearLayout。代表整个Window界面,包含通知栏,标题栏,内容显示栏三块区域。
DecorView这个大家应该很熟悉,通过getWindow().getDecorView()就可以得到此对象,在6.0以上,我们可以通过以下方式设置状态栏字体变黑:

getWindow().getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN|View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR);

那么可以猜想状态栏蒙灰肯定跟这个view有关。
接下来我们就去看DecorView的源码,对比Android6.0和7.0有什么不同。你会发现Android7.0以下,DecorView是PhoneWindow的内部类,而在7.0以上,是一个单独的类,并且有新的属性和方法。新的属性如:mSemiTransparentStatusBarColor,看字面意思应该就是我们要找的,我们对它进行跟踪,与它相关的代码如下:

DecorView(Context context, int featureId, PhoneWindow window,WindowManager.LayoutParams params) {
        super(context);
        ......//省略无关代码
        mForceWindowDrawsStatusBarBackground = context.getResources().getBoolean(
                R.bool.config_forceWindowDrawsStatusBarBackground)
                && context.getApplicationInfo().targetSdkVersion >= N;
                //设置默认的值,灰色
        mSemiTransparentStatusBarColor = context.getResources().getColor(
                R.color.system_bar_background_semi_transparent, null /* theme */);
        ......//省略无关代码
    }
private int calculateStatusBarColor() {
    int flags = mWindow.getAttributes().flags;
    return (flags & FLAG_TRANSLUCENT_STATUS) != 0 ? mSemiTransparentStatusBarColor
                : (flags & FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS) != 0 ? mWindow.mStatusBarColor : Color.BLACK;
}

calculateStatusBarColor这个方法就是计算得到状态栏的颜色值,其中FLAG_TRANSLUCENT_STATUS是透明标识,如果flags与FLAG_TRANSLUCENT_STATUS相与不等于0的话就选择默认灰色值mSemiTransparentStatusBarColor。
这个calculateStatusBarColor方法在updateColorViews方法中调用,而updateColorViews方法又在onWindowDragResizeStart、onWindowDragResizeEnd等方法调用(相关代码就不再帖了,请自行查看源码),看到这里就知道为啥怎么修改状态栏颜色也没用的原因了,这个背景色是动态算出来的。由此见得,只有将这个mSemiTransparentStatusBarColor变量值改为透明的就ok了。

那么怎么改呢?

解决思路:首先,我们通过getWindow().getDecorView()可以获取到这个DecorView类的对象,然后通过反射修改这个对象的成员mSemiTransparentStatusBarColor变量值。代码如下:

if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.N){
    try {
        Class decorViewClazz = Class.forName("com.android.internal.policy.DecorView");
        Field field = decorViewClazz.getDeclaredField("mSemiTransparentStatusBarColor");
        field.setAccessible(true);  
        field.setInt(getWindow().getDecorView(), Color.TRANSPARENT);  //改为透明
    } catch (Exception e) {}
}

注意:这段代码需要在setContentView方法前调用。

核心代码都写完了,赶紧试试吧!如果你觉得这篇文章对你有用,那么赞一个或者留个言吧~

最后推荐给大家一个能够在线查看源码的网站:http://androidxref.com/