android夜间模式Support Library

介绍

Android官方在Support Library 23.2.0中已经加入了夜间主题。也就是只需要通过更换主题便可实现日间模式和夜间模式的切换。

坑很多,看完再想想符合不符合自己需求。

一、实现夜间模式需要的配置
1、引入support包

 implementation 'com.android.support:appcompat-v7:28.0.0'

2、让我们项目的主题继承夜间模式主题,在style中设置如下主题:

 <style name="AppTheme" parent="Theme.AppCompat.DayNight">
        <!-- Customize your theme here. -->
        <item name="colorPrimary">@color/colorPrimary</item>
        <item name="colorPrimaryDark">@color/colorPrimaryDark</item>
        <item name="colorAccent">@color/colorAccent</item>
    </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

<?xml version="1.0" encoding="utf-8"?>
<alpha
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="500"
    android:fromAlpha="0.0"
    android:toAlpha="1.0" />

fade_out

<?xml version="1.0" encoding="utf-8"?>
<alpha
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:duration="500"
    android:fromAlpha="1.0"
    android:toAlpha="0.0" />

但是如果当前界面数据很多,界面需要加载时长,那就不适合此方案了。

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的界面就会闪白屏一次,第二次打开不会有问题。
然后你退出这个界面,你的上个界面会被刷新。

这个不知道为什么。。。。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值