快速解决Android7.0下沉浸式状态栏变灰的问题
1.绪论
现在基本上所有的应用都会去实现沉浸式状态栏,这个是应用的标配,如果你开发的应用没有,那这个吐槽点就多了,“这美工有审美观么”“程序猿这么菜,沉浸式都不会?”….. 咳咳….. 开个玩笑啊,各有各的设计思想,不能怪程序猿。
2.问题
那么说到沉浸式状态栏的问题是什么呢?不知道大家有没有遇到过,应用在android7.0系统以下的手机上运行,沉浸式状态栏是正常的,但是在7.0以上的手机上运行就感觉没有沉浸式了,是分层的。无论怎么修改状态栏背景色都没用,看下图:图1是7.0以下运行效果,图2是7.0以上运行效果。
看到上面两幅图,明显感觉到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/