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-02-27 09:01:32

    npm yarn 命令对比

    Yarn 是 Facebook, Google, Exponent 和 Tilde 开发的一款新的 JavaScript 包管理工具。就像我们可以从官方文档了解那样,它的目的是解决这些团队使用 npm 面临的少数问题.

  • 2020-02-29 20:47:34

    Nuxt 特有函数和变量

    asyncData方法使得你能够在渲染组件之前异步获取数据。该方法在服务端中执行的,所以,请求数据时,不存在跨域问题。返回的数据将与 data() 返回的数据进行合并。由于asyncData方法是在组件 初始化 前被调用的,所以在方法内是没有办法通过 this 来引用组件的实例对象。

  • 2020-03-01 19:00:46

    触发onclick事件元素的获取

    自动生成元素的onclick事件 event.target返回触发事件的元素 event.currentTarget返回绑定事件的元素

  • 2020-03-03 09:46:42

    JS实现HTML标签转义及反转义

    简单说一下业务场景,前台用户通过input输入内容,在离开焦点时,将内容在div中显示。 这时遇到一个问题,如果用户输入了html标签,则在div显示中,标签被解析。 由于是纯前端操作,不涉及后端,因此需要通过js对输入内容进行转义。

  • 2020-03-03 09:51:38

    写一个可插入自定义标签的 Textarea 组件

    为了实现这个功能,我最先想的是改造一个 <textarea> 然后我想到了 contenteditable (链接指向 mozilla.org) 这一属性 这是一个 html5 的属性,可以让元素内容可编辑