[老实李] Android夜间模式轻松实现

效果图:

[老实李] Android夜间模式轻松实现_第1张图片
yejianmoshigif.gif

Android 官方在 Support Library 23.2.0 中已经加入了夜间主题。也就是只需要通过更换主题便可实现日间模式和夜间模式的切换。
1、所以第一步先在gradle下面添加依赖:

 compile 'com.android.support:appcompat-v7:25.3.1'

2、修改Apptheme,让其继承夜间模式主题

 

3、res目录下新建night 资源目录
新建 drawable-night 和 values-night 的资源目录。如果要适配不同分辨率的屏幕则可新建 drawable-night-hdip、drawable-night-xhdpi 等目录来存放不同分辨率的图片资源。values-night 目录下存放与夜间模式相关的 value 文件。本篇文章讲解仅以夜间模式和日间模式的颜色为例,在 values-night 目录下新建 color.xml 文件。

[老实李] Android夜间模式轻松实现_第2张图片
image.png

4、启动 App 时检测是否处于夜间模式

如果是则切换至夜间主题。这个需要在自己项目的 Application 中实现。可在自己项目的 Application 中添加以下代码:

这里需要介绍一下有关夜间模式的几个常量值
AppCompatDelegate.setDefaultNightMode(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

override fun onCreate() {
super.onCreate()
setNightMode()
}
/**
*初始化夜间模式
*/
fun setNightMode(){
val nightMode = defaultSharedPreferences.getBoolean("night", false)
if (nightMode) {
AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES)
}else{
AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_NO)
}
}

5、设置页面点击 switchPreference时切换白天/夜间模式

/**
* ClassName:SettingFragment
* Description:
*/
class SettingFragment : PreferenceFragment(), Preference.OnPreferenceChangeListener {
    override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View {
        addPreferencesFromResource(R.xml.setting)
        val switchPreference = findPreference("night") as SwitchPreference
        switchPreference.onPreferenceChangeListener = this
        return super.onCreateView(inflater, container, savedInstanceState)
    }
   override fun onPreferenceChange(preference: Preference, objValue: Any): Boolean {
        val key = preference.key
        if ("night" == key) {
            var nightMode = defaultSharedPreferences.getBoolean("night", false)
            if (nightMode) {
                AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_NO)
            } else {
                AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES)
            }
            startActivity(Intent(activity, SettingActivity::class.java))
            activity.overridePendingTransition(R.anim.activity_enter_alpha,R.anim.activity_enter_alpha)
            activity.finish()
        }
        return true
    }   
}

在设置界面中对SwitchPreference设置onPreferenceChangeListener,这样当夜间模式变化的时候就先启动SettingActivity,然后再将activity finish掉,中间加上渐变的过渡动画解决闪屏问题。
动画如下:







6、解决返回主界面,主界面的UI没有更新的问题

解决这个问题可以在切换模式后从设置页面发送一个广播,然后在 MainActivity 中接收到这个广播后重启 MainActivity 即可。根据官方的推荐更换夜间模式后需要调用 recreate() 方法刷新页面。但是 recreate() 方法巨坑无比,调用 recreate() 方法引起了诸多问题。因此解决这个问题并没有在 MainActivity 调用中调用 recreate() 方法。而是在 SettingActivity 中重写了 onKeyDown() 方法。如果切换了夜间模式则在返回时发出一个广播结束掉 MainActivity ,然后调用 startActivity() 重启了 MainActivity 并添加了启动动画,让用户感觉是只是返回了主页面。

SettingActivity:

override fun onKeyDown(keyCode: Int, event: KeyEvent?): Boolean {
if (keyCode == KeyEvent.KEYCODE_BACK && event?.action == KeyEvent.ACTION_DOWN) {
goBack()
return true
}
return super.onKeyDown(keyCode, event)
}

fun goBack() {
EventBus.getDefault().post(NightEvent())
val intent = Intent(this, MainActivity::class.java)
startActivity(intent)
overridePendingTransition(R.anim.activity_enter_alpha, R.anim.activity_exit)
finish()
}

MainActivity(别忘了EventBus的注册和反注册):

/**
* 接收eventbus方法
*/
fun onEventMainThread(itemBean: NightEvent) {
print("mainactivityonEventMainThread")
finish()
}

大功告成~

你可能感兴趣的:([老实李] Android夜间模式轻松实现)