前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >简洁优雅地实现夜间模式

简洁优雅地实现夜间模式

作者头像
开发者技术前线
发布2020-11-23 15:27:50
1.6K0
发布2020-11-23 15:27:50
举报

夜间模式作为APP的一个重要的用户体验之一,很多应用有这个功能,目前,用户体验也是产品和设计较为看重的方面,越来越值得深入探究,本文将介绍在Android上一种夜间模式的实现方式。

前言

Android 6.0 Marshmallow 预览版中曾经短暂出现过相关的夜间模式的功能,只是在正式版中被移除了,在Android 7.0 Nougat上,用户们再次经历了「得而复失」的遗憾,在开发者预览版中,夜间模式和暗色模式先是开启,然后有再次被移除。而在正式版中,夜间模式也没有出现。但其实相关的代码一直存在于系统中,只是默认没有被开启。如何开启这项功能,可以参考少数派的这一篇文章,帮你找回 Android 7.0 夜间模式的 2 款应用(https://sspai.com/post/35273)

不过,今天要介绍的主要内容并不是关于系统的夜间模式,而是如何给我们开发的APP添加夜间模式的功能。毫不夸张的说,夜间模式现在已经是阅读类App的标配了。事实上,日间模式与夜间模式就是给APP定义并应用两套不同颜色的主题。用户可以自动或者手动的开启。我们先看两个我认为实现地很优雅的例子:知乎和Twitter。

这两个APP在切换的工程中,并没有出现闪现黑屏的情况,切换也比较顺滑。我们的目标就是利用Support Library实现同样的效果。

实现

添加依赖

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

由于Support Library在23.2.0的版本中才添加了Theme.AppCompat.DayNight主题,所以依赖的版本必须是高于23.2.0的,并且,这个特性支持的最低SDK版本为14,所以,需要兼容Android 4.0的设备,是不能使用这个特性的,在API Level 14以下的设备会默认使用亮色主题。不过现在4.0以下的设备应该比较少了吧,毕竟微信的minSdkVersion都设置为14了。

准备资源

1 让我们自己的主题继承并应用DayNight主题。

2 新建夜间模式资源文件夹:在res目录下新建values-night文件夹,然后在此目录下新建colors.xml文件在夜间模式下的应用的资源。当然也可以根据需要新建drawable-night,layout-night等后缀为-night的夜间资源文件夹。 我的valuesvalues-night目录下的colors.xml的内容如下:

3 使Activity继承自AppCompatActivity。

代码语言:javascript
复制
public class MainActivity extends AppCompatActivity {    
   // code here
    @Override
    protected void onCreate(Bundle savedInstanceState) {    }
}

应用

静态应用

在Application的继承类下设置初始主题。

代码语言:javascript
复制
public class App extends Application {    
   @Override
   public void onCreate() {        
      super.onCreate();        
       AppCompatDelegate.setDefaultNightMode(
            AppCompatDelegate.MODE_NIGHT_YES);        
       // other code here
   }
}

这里AppCompatDelegate.setDefaultNightMode()方法可以接受的参数值有4个:

  • MODE_NIGHT_NO. Always use the day (light) theme(一直应用日间(light)主题).
  • MODE_NIGHT_YES. Always use the night (dark) theme(一直使用夜间(dark)主题).
  • MODE_NIGHT_AUTO. Changes between day/night based on the time of day(根据当前时间在day/night主题间切换).
  • MODE_NIGHT_FOLLOW_SYSTEM(默认选项). This setting follows the system’s setting, which is essentially MODE_NIGHT_NO(跟随系统,通常为MODE_NIGHT_NO).

我们可以在任何时候调用这个方法,因为这个方法是静态的。但是这个值并不是一直存在的,每次在开启进程时需要重新设置。在上面的代码中,我是在onCreate()方法中设置的,网上也有大神建议在Activity或者Application的static代码块中设置。如下所示:

代码语言:javascript
复制
public class App extends Application {    
  static {        
        AppCompatDelegate
           .setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES);
   }    
   @Override
   public void onCreate() {        
      super.onCreate();        
   // ot
  }
}
动态应用

虽然上面的静态应用的设置非常简单,但是这种方式的应用场景还是太少了。我们更多的还是需要动态的根据需要动态的切换。

检测当前主题模式

代码语言:javascript
复制
int mode = getResources().getConfiguration().uiMode & Configuration.UI_MODE_NIGHT_MASK;

设置主题

代码语言:javascript
复制
 if(mode == Configuration.UI_MODE_NIGHT_YES) {
    AppCompatDelegate
       .setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_NO);
} else if(mode == Configuration.UI_MODE_NIGHT_NO) {
    AppCompatDelegate
          .setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES);
} else {    
  // blah blah
}recreate();

在调用recreate()方法之前,还可以创建一些动画进行过渡。而且,众所周知,调用recreate()需要对一些数据进行保存,例如fragment,CheckBox,RadioBox等。如下所示:

代码语言:javascript
复制
public class MainFragment extends Fragment {    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {        
           super.onCreate(savedInstanceState);          
           if (savedInstanceState != null) {              
            FragmentManager manager = getChildFragmentManager();
            doubanMomentFragment = (DoubanMomentFragment) manager.getFragment(savedInstanceState, "douban");
         } else {
             doubanMomentFragment = DoubanMomentFragment.newInstance();
         }
    }    
      @Override
     public void onSaveInstanceState(Bundle outState) {        
             super.onSaveInstanceState(outState);         
             FragmentManager manager = getChildFragmentManager();
           manager.putFragment        onCreate();        
        // other code here
   }
}

我们也可以把主题的值存储到SharedPreference中,已便于应用在下一次启动时自动应用主题。

Q&A

Q:系统默认的颜色不合我的口味怎么办?

A:使用主题属性,例如:textColor:?android:attr/textColorPrimary,color:?attr/colorControlNormal等。

Q:为什么我的WebView颜色没有变化?

A:因为WebView不能使用主题属性。WebView的颜色实际上取决于网页内容颜色。网页的默认背景色是白色,所以尽管设置了主题为AppCompatDelegate.MODE_NIGHT_YES,网页仍然是白色,所以看起来就很不搭了。所以,网页的内容和背景色等资源也需要适配了。

Q:为什么不直接设置为MODE_NIGHT_AUTO呢?

A:因为使用MODE_NIGHT_AUTO需要请求坐标权限,获取系统的位置。你肯定会说了,这尼玛不是坑爹吗?如果程序已经授予了坐标权限(location permission)(如果你的target SDK为23或者更高,需要考虑运行时权限),AppCompat会试着去获取上次保存的坐标,根据坐标来计算日出与日落的时间。如果程序没有位置权限或者LocationManager没有存储上次坐标的信息呢?系统或默认设置为早上6点钟为日出,下午10点为日落。用户调整系统时间,当前的主题也会随之改变。如果我们不希望用户在设定主题后,主题还会随着时间改变,MODE_NIGHT_AUTO就不适用了。

效果和代码

代码

https://github.com/TonnyL/PaperPlane

运行效果

在Android 6.0及以下的设备上,本项目运行时会有切换的过渡动画效果,但是不支持Android 7.0及以上的设备。

作者:TonnyL 链接:http://www.jianshu.com/p/dcfcfcbda7ac

精彩推荐

静默安装从入门到转行!

技术 - 资讯 - 感悟

END

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2017-07-31,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 开发者技术前线 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 添加依赖
  • compile 'com.android.support:appcompat-v7:25.1.0'
  • 准备资源
    • 静态应用
      • 动态应用
      • 代码
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档