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),代码中还获取不到当前的模式


  • 2017-01-16 15:09:40

    Javascript模块化编程(一):模块的写法

    随着网站逐渐变成"互联网应用程序",嵌入网页的Javascript代码越来越庞大,越来越复杂。网页越来越像桌面程序,需要一个团队分工协作、进度管理、单元测试等等......开发者不得不使用软件工程的方法,管理网页的业务逻辑。

  • 2017-01-16 15:16:24

    Javascript模块化编程(二):AMD规范

    AMD是"Asynchronous Module Definition"的缩写,意思就是"异步模块定义"。它采用异步方式加载模块,模块的加载不影响它后面语句的运行。所有依赖这个模块的语句,都定义在一个回调函数中,等到加载完成之后,这个回调函数才会运行。

  • 2017-01-16 15:19:24

    RequireJS 入门指南

    如今最常用的JavaScript库之一是RequireJS。最近我参与的每个项目,都用到了RequireJS,或者是我向它们推荐了增加RequireJS。在这篇文章中,我将描述RequireJS是什么,以及它的一些基础场景。

  • 2017-01-16 15:22:30

    CommonJS规范

    CommonJS模块规范。 根据这个规范,每个文件就是一个模块,有自己的作用域。在一个文件里面定义的变量、函数、类,都是私有的,对其他文件不可见。

  • 2017-01-17 15:42:03

    vue-cli 发布(译)

    当我们真正开发一个应用的时候,我们不可避免的会用到一大堆的工具,模块化、预处理器、热模块加载、代码校验和测试。这些工具对于一个需要长期维护的大型应用是必须的,但是项目初始化将会是让人痛苦的事情。这就是为什么我们做了 vue-cli 。