Android官方在Support Library 23.2.0中已经加入了夜间主题。也就是只需要通过更换主题便可实现日间模式和夜间模式的切换。
坑很多,看完再想想符合不符合自己需求。
一、实现夜间模式需要的配置
1、引入support包
implementation 'com.android.support:appcompat-v7:28.0.0'
2、让我们项目的主题继承夜间模式主题,在style中设置如下主题:
3.新建drawable-night和values-night的资源目录。
values-night目录下新建和存放与夜间模式相关的value文件。
drawable文件夹放日间模式图片,drawable-night文件夹放夜间模式图片。
value的color文件夹设置日间模式颜色,value-night的color文件夹设置夜间模式颜色
然后在项目所有的xml配置中涉及有图片和颜色的地方,都在对应地方设置日夜间模式两种。
日夜间状态有两种切换方法
AppCompatDelegate.setDefaultNightMode(mode) 可以设置全局的夜间模式
AppCompatDelegate.setLocalNightMode(mode)可以设置局部的夜间模式
mode值有四种
MODE_NIGHT_NO: 亮色(light)主题,不使用夜间模式
MODE_NIGHT_YES:暗色(dark)主题,使用夜间模式
MODE_NIGHT_AUTO:根据当前时间自动切换 亮色(light)/暗色(dark)主题(22:00-07:00时间段内自动切换为夜间模式)
MODE_NIGHT_FOLLOW_SYSTEM(默认选项):设置为跟随系统,通常为MODE_NIGHT_NO
设置完后,一般都需要调用recreate()才能生效。
我们一般会选择保存下日夜间模式的状态
在Application中进行初始化。
根据app上次退出的状态来判断是否需要设置夜间模式,提前在SharedPreference中存了一个是否是夜间模式的boolean值
if (SettingDBHelper.getIsNightTheme()) {
AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES);
}else {
AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_NO);
}
以上就是一个比较常规的根据系统实现夜间模式的实现方法了。
然而,没有坑那就对不起系统自带几个字。
没有坑那就对不起系统自带几个字。
网上教程很大,但是大多数人只是写个demo完事儿,正式用到项目的很少很少,先看完符合不符合自身需求吧,有更好的方案也感谢能提出建议。
1、recreate()带来的闪屏问题以及数据问题
如果是在设置中,可以采取重新调起当前界面的方式,而不recreate,加上一个动画,相对来说也比较温和。
if (SettingDBHelper.getIsNightTheme()) {
//变成夜间模式
AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES)
} else {
//变成日间模式
AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_NO)
}
startActivity(Intent(this, ActivitySetting::class.java))
overridePendingTransition(R.anim.fade_in, R.anim.fade_out)
finish()
fade_in
fade_out
但是如果当前界面数据很多,界面需要加载时长,那就不适合此方案了。
2、已经实例化的界面,不能切换状态
在设置界面设置完了,返回之前的界面,还都是原来的状态,这个真的好气。传统的日夜间模式方案(每个界面建两个theme那种)
要在以前的界面写一大堆set和回调。如果这种也这么写,那这种方案就毫无意义了。
根据大家的方案也是通知之前的界面recreate(),实测这种老是会崩溃之类的。。。fu,ck,太坑了。
所以,最后我采取了一种相对折中的方案。因为我的设置是在mainactivity的一个fragment点进去的activity。(如果不是更麻烦,太麻烦了,放弃吧孩子。)
在Mainactivity中的,如果切换了日夜间模式。
override fun onResume() {
super.onResume()
if (SPUtils.get(this, "isChangeTheme", false) as Boolean) {
SPUtils.put(this, "isChangeTheme", false)
startActivity(Intent(this, ActivityMain::class.java))
overridePendingTransition(R.anim.fade_in, R.anim.fade_out)
finish()
}
}
也要切换到应该在的fragment
navigation.selectTab(initPostion)
这也太勉强,也是勉强实现了吧,,,,
3、对,还有坑。
如果你的程序中大量应用了webview,那么你基本可以放弃这种方案了。
首先这种方案不提供webview的实现方案,如果是线上的链接,你需要让前端做两套。
如果是本地模板实现,需要写两套html css
在需要引用的地方判断引用哪种。
你以为这就是webview的坑了么,不是的!!!
有webview的应用,在android 9.0(目前占有率35%)的大多数机型都会出现一个问题,在你设置了夜间模式后,第二次打开app,此时默认是夜间模式,然后你打开有webview的界面就会闪白屏一次,第二次打开不会有问题。
然后你退出这个界面,你的上个界面会被刷新。
这个不知道为什么。。。。