android 快速实现夜间模式

2019-05-18 12:38:41

参考文章  android 快速实现夜间模式

最近项目中遇到了一个问题,夜间模式在8.0以上的手机中不起作用,查看了一下原因,是夜间模式实现方法的问题。分两种情况介绍一下


先看一下运行效果:




第一种 是目前项目中用到的,是以前的开发人员写的,存在一些问题(不兼容8.0以上的版本),已解决


第二种是Support Library 中自带的一种方法。


 


第一种:

第一步 需要添加夜间模式资源,以-night为后缀

日间模式                    对应                  夜间模式


drawable                                              drawable-night


values                                                     values-night


drawable-hdpi                                     drawable-night-hdpi


。。。


目录结构如下:




例如:color 的色值:在不同的value 下,命名相同,但是色值可以随意设置,适应不同的模式


图片,边框样式等也是一样的道理。




第二步 在application 中添加 切换方法:

public class MyApplication extends Application {

    private static Resources sRes;

    @Override

    public void onCreate() {

        super.onCreate();

        init(this);

    }

    public static void init(Context context) {

        sRes = context.getResources();

    }

    /**

     * 切换 夜间模式

     * @param on true 夜间, false  日间

     */

    public static void updateNightMode(boolean on) {

        DisplayMetrics dm = sRes.getDisplayMetrics();

        Configuration config = sRes.getConfiguration();

        config.uiMode &= ~Configuration.UI_MODE_NIGHT_MASK;

        config.uiMode |= on ? Configuration.UI_MODE_NIGHT_YES : Configuration.UI_MODE_NIGHT_NO;

        sRes.updateConfiguration(config, dm);

    }

}

第三步 调用方法修改主题:

 MyApplication.updateNightMode( false/true );  // 需要保存当前主题的状态。根据状态,进行切换


切换完后 刷新 


recreate();// 会有闪屏,可以使用下面的方法

或者


finish();

startActivity(new Intent( this, this.getClass()));

overridePendingTransition(0, 0);

注意:以上方法只适合 8.0以下的手机,需要兼容的话,就需要用到第二种方法了

 


第二种:

使用Support Library 23.2中新添加的夜间模式主题,并且可以向下兼容最低api 14的,基本满足绝大多数的手机。


介绍:

设置 当前的模式


        AppCompatDelegate.setDefaultNightMode(int mode);


它有四个可选值,分别是:

                    MODE_NIGHT_NO: 使用亮色(light)主题,不使用夜间模式

                    MODE_NIGHT_YES:使用暗色(dark)主题,使用夜间模式

                    MODE_NIGHT_AUTO:根据当前时间自动切换 亮色(light)/暗色(dark)主题

                    MODE_NIGHT_FOLLOW_SYSTEM(默认选项):设置为跟随系统,通常为 MODE_NIGHT_NO


使用方法:

这种夜间模式只需要按照以下几部就可以使用了:


第一步:

添加夜间模式目录,和第一种方法的第一步一致


第二步:

需要   com.android.support:appcompat 23.2及以上的版本


及 在app  的bulid.gradle 中添加(一般创建项目的时候会自动添加):


dependencies {

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

 

}

第三步:

Activity须继承AppCompatActivity   (一定要继承,否则无效)


第四步:

在 values—>styles.xml 中 修改你的主题样式,将你的主题 parent  改为  Theme.AppCompat.DayNight 或者 它的子类


如: Theme.AppCompat.DayNight.NoActionBar


    <style name="AppTheme" parent="Theme.AppCompat.DayNight.NoActionBar">

 

        <!-- Customize your theme here. -->

        <item name="colorAccent">@color/colorAccent</item>

 

    </style>

第四步:

切换主题


建议在application  onCreate 中切换,


AppCompatDelegate.setDefaultNightMode(int mode);


注意:主题的当前模式 ,需要保存到本地( 可以通过SharePreference),代码中还获取不到当前的模式


  • 2020-01-16 08:52:22

    Vue函数式调用组件创建公共组件

    所有组件都需要这么去调用,就会有些许麻烦而且不太美观。像Loading、Toast等这些组件,一页面可以经常用到而且每次显示的内容都可能不一样,这样的话用js的方式【this.$xxx.show(option)】去调用就方便很多,而且代码也更整洁。

  • 2020-01-17 08:37:26

    css transition分别指定多个属性

    transition有四个属性,很多人都会遗忘,分别是transition-property,transition-duration,transition-timing-function,transition-delay,尤其是transition-delay,这个可以实现延迟动画

  • 2020-01-17 08:44:57

    vue keepalive 前进刷新后退不刷新终极解决方案

    另外,我们做路由的时候要有意的根据页面等级做出路由的长度 比如 /a是一级的页面/a/b是二级的页面,下面的文章大家也可以通过判断path的长度来计算rank值,不用有意自定了 这样做的好处有两点,一个就是前进刷新,后退不刷新,还有就是,如果我们做页面进出效果的时候也能排上用场。

  • 2020-01-17 15:28:24

    深入理解vue中的slot与slot-scope

    vue中关于插槽的文档说明很短,语言又写的很凝练,再加上其和methods,data,computed等常用选项使用频率、使用先后上的差别,这就有可能造成初次接触插槽的开发者容易产生“算了吧,回头再学,反正已经可以写基础组件了”,于是就关闭了vue说明文档。