Android从SDK29推出深色模式主题,使用xxxxx.xxxxxx.DayNight深色模式主题就可以完成深色模式颜色反转。(其实深色模式很久之前就支持,只是google不提供最短捷径方法,适配请看正文 二)
但作为一个优秀的猿类,不应该使用这种系统级支持。
前言:
一、双重属性(前提设置好支持深色的主题)
1.<item name="android:forceDarkAllowed">?</item>
:是否使用系统自带的深色模式主题(不设置默认为true)
我可以在我们的主题或者控件属性中设置它,例如:
//View中
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:forceDarkAllowed="true">
</LinearLayout>
//主题中
<style name="AppTheme.NoActionBar">
<item name="android:forceDarkAllowed">true</item>
</style>
2.系统浅色和深色模式切换时,App中的Activity会调用recreate方法重新绘制UI,不管你是否使用深色模式主题或者android:forceDarkAllowed属性为false。解决Ui重新绘制的方法就是由被动改变设置成主动通知。
Activity中的android:configChanges=“uiMode”。解决了Ui重新绘制问题。
但手机设置发生变化时,Android提供了一个可以接受到变化的方法:
我们可以在Activity或者Fragment中重写此方法。
override fun onConfigurationChanged(newConfig: Configuration) {
super.onConfigurationChanged(newConfig)
newConfig.uiMode
}
3.交叉使用
forcedarkallowed | true | false |
---|---|---|
设置了uimode | 不调用recreate,但Ui会适配深色模式(可能内部做了主题切换,不会闪屏效果不错) | 不调用recreate,Ui也不会发生改变 |
未设置uimode | 调用recreate,Ui会适配深色模式 | 调用recreate,Ui也不会发生改变 |
正文
二、自适配深色模式
但我们使用两套资源去适配深色模式的时候,不管你是否设置android:configChanges="uiMode"或者true都不会影响深色模式的切换。(如果在App内切换还是要调用recreate的:主动调用或者别设置uimode)
具体详见:https://blog.csdn.net/qq_20521573/article/details/76222085?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522163350114316780264086163%2522%252C%2522scm%2522%253A%252220140713.130102334…%2522%257D&request_id=163350114316780264086163&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2allsobaiduend~default-3-76222085.first_rank_v2_pc_rank_v29&utm_term=values-night&spm=1018.2226.3001.4187 (引用大佬的文章,可以多点点关注)
在两套资源完备的情况下,App的深色模式就会跟随系统的深色模式变化。
三、控制App内深色模式的*切换。
在Android的AppCompatActivity中提供了一个方法
AppCompatDelegate.setDefaultNightMode()
这个方法可以使所有继承了AppCompatActivity的子类Activity中的内容在深色和浅色模式下转换。
(非AppCompatActivity的继承者是不会发生变化的,例如使用:PreferenceActivity,FragmentActivity)
四、应对recreate闪屏问题(App内使用了AppCompatDelegate.setDefaultNightMode())
1.在有切换深色和浅色模式按钮的activity中使用上述(交叉使用)的第一种方式(不刷新Ui,系统级反射)。
2.在有切换深色和浅色模式按钮的activity中准备两套完整的主题,并设置Uimode,但需要做切换时,单独切换主题。