现状
夜间模式是android换肤的一种,关于换肤的相关知识总结,大家可以参考这篇文章Android换肤技术总结-markzhai
夜间模式现状
夜间模式与换肤的不同:
- 夜间模式可以说是大部分应用发展到一定阶段都要实现的功能,而换肤功的需求广度不如夜间模式。
- 实现难度上来讲,换肤的实现会比夜间模式更复杂些,但是实现方式思路也已经比较成熟。
- 换肤的实现一般都是需要从apk外部加载皮肤资源,毕竟皮肤的种类比较多,都放到apk中不现实。而夜间模式则直接放到apk中就可以。
为什么要重新实现一个夜间模式库:
由于工程需要在原有基础上修改,就希望能有一个侵入性尽量小,以不影响现有工程的库,来实现夜间模式。考察了一些现有的夜间模式或换肤库,感觉工程需要修改的内容太多,所以决定做一个这样的库。
夜间模式要解决两个问题:
- 切换夜间模式后,新打开页面要以新的样式展示。
- 切换夜间模式时,已打开页面的刷新。
新页面的样式展示的解决方式:
新页面以新的样式的展示是一个比较容易解决的问题,替换resource这种类似的黑科技排出在外,我这里有两种实现思路,解决方式不同,其解决上面提到的问题2(刷新页面)的实现方式也相应的需要改变。
- 使用attr引用的方式实现两套主题,在Activity的onCreate方法中,在super.onCreate之前setTheme, Activity就可以以设置的Theme的样式展示。
- 使用官方在support appcompat 23.2.1中加入的夜间模式,使用
AppCompatDelegate.setDefaultNightMode(int mode)
来全局启用需要的夜间模式,之后再打开新Activity将以新的方式展示。
刷新已打开页面的解决方式:
针对attr方式实现两套主题的方式,刷新已打开界面的的解决方式,我这里由于没有实践过,就不敢枉自猜测解决方式了。下面我们着重介绍以官方夜间模式为基础的刷新已打开页面样式的实现思路。
官方提供了夜间模式的支持,但是对于已经打开的页面还是需要重启来刷新样式。然而这种方式用户体验并不好。这样我们还是要解决不重启页面刷新页面的问题。那么有没有办法不重启就重新加载样式呢,答案是:没有,没有这样一个全局性的东西(至少我还没发现,如果有麻烦告诉我下)。我们只能针对性的对我们的View
主动修改background
,textColor
等来使我们的页面达到刷新的效果。
这里呢我们又有两个问题需要解决:
- 哪些
view
需要刷新。 - 刷新哪些属性。
- 新样式从哪儿来。
对于问题1、2的解决我这里参考来张鸿洋老师的博客和开源库。思路就是,代理我们Activity中View
的创建过程,以通过他们在XML中设置的属性来判断是否我们需要刷新的View
。如果是就将View
和属性(比如android:background="@color/gray"
)cache起来,备用。
对于问题3(新样式从哪儿来)的解决,我们这里利用来官方提供的夜间模式的来解决。在使用官方提供的AppCompatDelegate.setDefaultNightMode(int mode)
方法之后虽然不能刷新已打开页面,但是我们同时再调用activity.getDelegate().applyDayNight()
方法就之后,其实Activity中的Resource资源已久发生了改变。这个我们可以通过源码了解到,applyDayNight()方法最终调用了这样一段函数:
这个时候Resource会刷新自己已经缓存的资源内容。这个时候我们为Activity中的空间setBackgroundDrawable(mResource.getDrawable(int resId))
已经是新的了(这个时候虽然资源id没变,但是它们指向的内容以及发生了变化)。
我们就这样解决了刷新问题。思路就是这样的。
遇到的问题:
在实践上面的方法中,也遇到了一些问题:
-
ListView
中为item
设置了selector
资源,刷新没有效果。这个通过反射设置Activity
的resource
置空,让Activity
重新生产resource
来解决。 -
Activity
中new TextView(this)
创建的view
不能刷新,需要通过LayoutInflater.from(MainActivity.this).inflate(R.layout.item_layout, linearLayout, false);
这种加载布局的方式来创建才能被刷新,因为我们需要让控件的创建走代理来已cache到新创建的控件。
我实现的轮子:
我通过上面的思路实现了一个库,尽量的降低了侵入性。要参考代码的可以走这里。