目录
当白夜模式切换时,activity会销毁重新加载,谷歌是希望重新加载Activity可以刷新页面UI,但我的App并没有适配深色模式,这样用户体验就很不好,解决办法就是:
在AndroidManifest.xml中给Activiyty添加或追加 android:configChanges="uiMode" 属性即可:
configChanges 参数详解:
- api 'com.android.support:appcompat-v7:24.1.1' 或者更高版本
- 或使用androidx的依赖都可以
- /**
- * 暗黑模式适配工具类
- */
- public class DarkModeUtils {
-
- public static final String KEY_MODE = "white_night_mode_sp";
-
- /**
- * 在 Application 的 onCreate() 方法中调用
- */
- public static void init(Application application) {
- int nightMode = getNightModel(application);
- AppCompatDelegate.setDefaultNightMode(nightMode);
- }
-
- /**
- * 应用夜间模式
- */
- public static void applyNightMode(Context context) {
- AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES);
- setNightModel(context, AppCompatDelegate.MODE_NIGHT_YES);
- }
-
- /**
- * 应用日间模式
- */
- public static void applyDayMode(Context context) {
- AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_NO);
- setNightModel(context, AppCompatDelegate.MODE_NIGHT_NO);
- }
-
- /**
- * 跟随系统主题时需要动态切换
- */
- public static void applySystemMode(Context context) {
- AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_FOLLOW_SYSTEM);
- setNightModel(context, AppCompatDelegate.MODE_NIGHT_FOLLOW_SYSTEM);
- }
-
- /**
- * 判断App当前是否处于暗黑模式状态
- */
- public static boolean isDarkMode(Context context) {
- int nightMode = getNightModel(context);
- if (nightMode == AppCompatDelegate.MODE_NIGHT_FOLLOW_SYSTEM) {
- int applicationUiMode = context.getResources().getConfiguration().uiMode;
- int systemMode = applicationUiMode & Configuration.UI_MODE_NIGHT_MASK;
- return systemMode == Configuration.UI_MODE_NIGHT_YES;
- } else {
- return nightMode == AppCompatDelegate.MODE_NIGHT_YES;
- }
- }
-
- private static int getNightModel(Context context) {
- SharedPreferences sp = context.getSharedPreferences(KEY_MODE, Context.MODE_PRIVATE);
- return sp.getInt(KEY_MODE, AppCompatDelegate.MODE_NIGHT_FOLLOW_SYSTEM);
- }
-
- public static void setNightModel(Context context, int nightMode) {
- SharedPreferences sp = context.getSharedPreferences(KEY_MODE, Context.MODE_PRIVATE);
- sp.edit().putInt(KEY_MODE, nightMode).apply();
- }
-
- }
对图片和颜色创建暗黑模式的文件夹,切换暗黑模式时系统会自动使用该文件夹下的资源:
如果添加或追加了 android:configChanges="uiMode"属性, 在模式切换后会回调Activity的onConfigurationChanged(),建议在BaseActivity中重写该方法,并在里面做相应的适配.
可以调用上面工具类中的 DarkModeUtils.isDarkMode(this) 方法判断是否是暗黑模式.也可以调用工具类中的方法切换App的白夜模式,注意事项看下面:
(1) 在 Application 的 onCreate() 方法中调用 DarkModeUtils.init(this) 初始化工具类;
(2) 调用 DarkModeUtils.applySystemMode(this) 方法自适应系统白夜模式切换;
(3) 如果使用的 androidx的依赖直接调用即可.
(4) 如果使用support依赖需要手动调用Activity的 recreate()方法重建页面或针对页面所有View做不同适配.
直接调用recreate()方法重建页面示例:
针对页面每个View去设置白夜模式的示例:
(5) 具体原因看下面源码:
androidx版本:
- /**
- * Sets the default night mode. This is the default value used for all components, but can
- * be overridden locally via {@link #setLocalNightMode(int)}.
- *
- *
This is the primary method to control the DayNight functionality, since it allows
- * the delegates to avoid unnecessary recreations when possible.
- *
- *
If this method is called after any host components with attached
- * {@link AppCompatDelegate}s have been 'started', a {@code uiMode} configuration change
- * will occur in each. This may result in those components being recreated, depending
- * on their manifest configuration.
- *
- *
Defaults to {@link #MODE_NIGHT_FOLLOW_SYSTEM}.
- *
- * @see #setLocalNightMode(int)
- * @see #getDefaultNightMode()
- */
- public static void setDefaultNightMode(@NightMode int mode) {
- switch (mode) {
- case MODE_NIGHT_NO:
- case MODE_NIGHT_YES:
- case MODE_NIGHT_FOLLOW_SYSTEM:
- case MODE_NIGHT_AUTO_TIME:
- case MODE_NIGHT_AUTO_BATTERY:
- if (sDefaultNightMode != mode) {
- sDefaultNightMode = mode;
- applyDayNightToActiveDelegates();
- }
- break;
- default:
- Log.d(TAG, "setDefaultNightMode() called with an unknown mode");
- break;
- }
- }
support版本:
- public static void setDefaultNightMode(int mode) {
- switch(mode) {
- case -1:
- case 0:
- case 1:
- case 2:
- sDefaultNightMode = mode;
- break;
- default:
- Log.d("AppCompatDelegate", "setDefaultNightMode() called with an unknown mode");
- }
- }
对比后可以发现androidx切换暗黑模式后,自己主动调用了applyDayNightToActiveDelegates()方法,使Activity重建。而support上没有,只是赋值。所以support版本上使用需要自己调用Activity的recreate()方法重建。