序言
之前我有讲过Android换肤功能的实现,具体看juejin.cn/post/725848... 。本篇就来讲讲另外一个类似功能,深色模式适配。从Android10以来,官方直接在系统层面支持了深色模式的功能。
适配深色主题的好处
- 为弱视以及对强光敏感的用户提高可视性
- 省电,OLED屏幕跟电阻屏LCD屏幕不一样,如果是黑色,则可以不发光,减少电池的损耗
- 更换UI风格,降低审美疲劳
- 专业,逼格更高,这可能是某些开发者的主要原因,哈哈
开始适配
主要适配的流程有以下两点:
- 添加-night的资源,如res/values-night/colors.xml,适配文字和背景底色
- 应用内提供切换为深色模式、浅色模式以及跟随系统的界面
自动适配
Android10提供Force Dark功能,通过给themes.xml或styles.xml的style节点下添加item节点。大项目可以先用自动适配,然后慢慢进行手动适配。
xml
<style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar">
<item name="android:forceDarkAllowed" tools:targetApi="Q">true</item>
</style>
手动适配
对自动适配的颜色不满意的地方,可以采用手动适配。那就要自行提供-night的资源了。主题和样式覆盖的优先级可以参考我的这篇文章juejin.cn/post/719151... ,所以它能覆盖掉系统定义的。
切换深/浅色模式
最终调用AppCompatDelegate.setDefaultNightMode切换深/浅色模式,切换过程建议在启动页完成,因为这个系统的切换API有一定的延迟。
kotlin
// 始终以软件保存的暗色模式为准
if (SPUtils.readBoolean(this, Environment.PREF_DAY_NIGHT_MODE)) {
AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES)
} else {
AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_NO)
}
startActivity(Intent(this@SplashActivity, MainActivity::class.java))
// 淡入淡出效果
overridePendingTransition(R.anim.fade_in, R.anim.fade_out)
finish()
其他注意事项
当我们调用AppCompatDelegate.setDefaultNightMode()方法的时候,系统会自动重新重建Activity,如果你不想要重新创建,则可在AndroidManifest.xml的activity节点中指定 android:configChanges="uiMode",然后重写onConfigurationChanged()方法,自己处理切换逻辑。
kotlin
override fun onConfigurationChanged(newConfig: Configuration) {
super.onConfigurationChanged(newConfig)
val nightMode = newConfig.uiMode and Configuration.UI_MODE_NIGHT_MASK
if (nightMode == Configuration.UI_MODE_NIGHT_YES) {
// 切换为深色模式,在这里写你的逻辑...
}
if (nightMode == Configuration.UI_MODE_NIGHT_NO) {
// 切换为浅色模式,在这里写你的逻辑...
}
}
总结
深色主题的适配其实并不复杂,主要注意在启动页调用AppCompatDelegate.setDefaultNightMode()方法来加载预设主题配置。适配流程不清晰的回头看前面的内容,其实主要也就是添加暗色资源,然后调用系统的切换方法。另外多语言的适配也是一个提升App质量的一个方面,juejin.cn/post/733879... 。