一提到沉浸式状态栏,第一个浮现在脑海里的词就是“碎片化”。碎片化是让 Android 开发者很头疼的问题,相信没有哪位开发者会不喜欢“write once, run anywhere”的感觉,碎片化让我们不得不耗费精力去校验代码在各个系统版本、各个机型上是否有效。因此以前我一直把沉浸式状态栏看作一块难啃的骨头,但是该面对的问题迟早还是要面对,所以,不如就此开始吧。
沉浸式状态栏的实现
方法一:通过设置 Theme 主题设置状态栏透明
因为 API21 之后(也就是 android 5.0 之后)的状态栏,会默认覆盖一层半透明遮罩。且为了保持4.4以前系统正常使用,故需要三份 style 文件,即默认的values(不设置状态栏透明)、values-v19、values-v21(解决半透明遮罩问题)。
1 | //valuse |
由图可见,设置之后布局的内容延伸到了状态栏。但有些场景下,我们还是需要状态栏那块位置存在的(然而不存在的)。有三种解决方法:
法一:设置 fitsSystemWindows 属性
引用一下官方对该属性的解释吧:
android:fitsSystemWindows
Boolean internal attribute to adjust view layout based on system windows such as the status bar. If true, adjusts the padding of this view to leave space for the system windows. Will only take effect if this view is in a non-embedded activity.
当该属性设置 true 时,会在屏幕最上方预留出状态栏高度的 padding。
在布局的最外层设置 android:fitsSystemWindows="true"
属性。当然,也可以通过代码设置:
1 | /** |
通过该设置保留状态栏高度的 paddingTop
后,再设置状态栏的颜色。就可以达到设想的效果。但这种方式实现有些问题,例如我们想设置状态栏为蓝色,只能通过设置最外层布局的背景为蓝色来实现,然而一旦设置后,整个布局就都变成了蓝色,只能在下方的布局内容里另外再设置白色背景,而这样就存在过度绘制了。而且设置了 fitsSystemWindows=true
属性的页面,在点击 EditText 调出 软键盘时,整个视图都会被顶上去。
法二:布局里添加占位状态栏
法一:在根布局加入一个占位状态栏,这样虽然整个内容页面时顶到头的,但是因为在内容布局里添加了一个占位状态栏,所以效果与设想的一致。
1 | <View |
通过反射获取状态栏高度:
1 | /** |
设置占位视图高度
1 | View statusBar = findViewById(R.id.statusBarView); |
当然,除了从布局文件中添加这一方式之外,一样可以在代码中添加。比较推荐使用代码添加的方式,方便封装使用。
1 | /** |
法三:代码中设置 paddingTop 并添加占位状态栏
手动给根视图设置一个 paddingTop
,高度为状态栏高度,相当于手动实现了 fitsSystemWindows=true
的效果,然后再在根视图加入一个占位视图,其高度也设置为状态栏高度。
1 | //设置 paddingTop |
个人认为最优解应该是第三种方法,通过这种方法达到沉浸式的效果后面也可以很方便地拓展出渐变色的状态栏。
方法二:代码中设置
通过在代码中设置,实现方法一中在 Theme 主题样式里设置的属性,便于封装。
1 | if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { |
但是从图片中也看到了,该方案会导致一个问题就是导航栏颜色变灰。
经测试,在 5.x 以下导航栏透明是可以生效的,但 5.x 以上导航栏会变灰色(正常情况下我们期望导航栏保持默认颜色黑色不变),但因为设置了FLAG_TRANSLUCENT_NAVIGATION
,所以即使代码中设置 getWindow().setNavigationBarColor(Color.BLACK);
也是不起作用的。但如果不设置该 FLAG ,状态栏又无法被置为隐藏和设置透明。
方案二:全屏模式的延伸
通过设置 FLAG ,让应用内容占用系统状态栏的空间,经测试该方式不会影响对导航栏的设置。
1 | /** |
验证其他使用场景
侧滑菜单
使用 AS 自动创建 Navigation Drawer Activity
,布局结构为:
- DrawerLayout
- include :内容布局,默认使用 ToolBar
- NavigationView :侧滑布局
这里只调用了 fullScreen()
, 测试一下运行结果如何:
可以看到都有不尽如人意的地方,4.4 系统中内容视图是可以正常延伸到状态栏中,但侧滑菜单中却在上方出现了白条,而在 6.0 中侧滑菜单上会有半透明遮罩。针对 6.0 侧滑菜单半透明遮罩问题,通过设置为 NavigationView
设置属性 app:insetForeground="#00000000"
即可解决。针对 4.4 侧滑菜单白条问题,经过测试,通过对最外层布局设置 setFitsSystemWindows(true)
和 setClipToPadding(false)
可以解决,所以这里对之前的 fitsSystemWindows
方法稍加修改:
1 | /** |
这样是解决了上述的问题,既然延伸内容没问题了,那就开开心心地像上面一样调用 addStatusViewWithColor()
方法增加个占位状态栏,解决一下内容顶到头的问题吧:
可以看到,效果依然不是我们想要的,虽然占位状态栏是有了,但是却也覆盖到了侧滑菜单上,并且即使设置了 android:fitsSystemWindows="true"
也并没有什么卵用,内容布局依然顶到了头部。这里有两种解决方法:1. 第一种方案是网上提到比较多的,改变 ToolBar
的高度,并增加状态栏高度的 paddingTop
,这也是
ImmersionBar 库采用的方案。2. 第二种方案其实思路与第一种差不多,就是将原有的内容布局从 DrawerLayout
中移除,并添加到线性布局(布局中已有占位状态栏),之后再将这个线性布局添加到 DrawerLayout
中成为新的内容布局,此谓狸猫换太子。
1 | /** |
一番操作后,效果如下:
对于内容视图未使用到 ToolBar
的情况方案二依然可以适用。
ActionBar
上述代码在使用 ActionBar 时可以完美适配吗?测试后效果如下图所示
可以看到,通过添加指定颜色的占位状态来达到沉浸效果的方案,在 4.4 系统上效果是正常的,但是在 6.0 上,在状态栏和 Actionbar 之间会有阴影,这个阴影是主题的效果。不知道大家还记不记得 Theme 主题里的几个设计颜色的属性:
colorPrimary
指定 ActionBar 的颜色,colorPrimaryDark
指定状态栏颜色,经过测试,在主题里将二者设为统一颜色,状态栏和 ActionBar 之间不会有黑边。自然,我们除了在 Theme 主题里设置,还可以直接在代码里通过上文提到过的代码修改 5.x 以上系统的状态栏颜色:
1 | Window window = activity.getWindow(); |
但是因为 setStatusBarColor()
方法的参数无法传入 Drawble ,所以这种方式是无法实现渐变色状态栏的效果的。所以还是应该聚焦在怎么解决 ActionBar 阴影的问题,上面说了,既然这个阴影是 Theme 的效果,那就肯定有移除这种效果的方法,一种解决方法是更改主题为 ActionBar 不带阴影的主题样式:
1 | <style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar"> |
还有第二种更简单的方式,那就是直接在代码里设置去除阴影:
1 | /** |
并且因为内容是位于 ActionBar 之下的,我们还必须给内容视图是指一个 paddingTop,高度为状态栏高度+ActionBar 高度,才可以使内容正常显示。我们给 ActionBar 设置一个渐变色试试看:
1 | //drawble 文件夹内新建 shape 渐变色 |
至此,尝试适配了几种比较常见的使用场景的沉浸式状态栏,效果也都还比较符合预期。真正去处理这个问题时会发现其实问题也没有想象中的那么复杂。最后附上 Github 源码。
Stay hungry. Stay foolish.
下篇博客再见。