夜晚的故事(android夜间模式实现)

夜幕降临,他走在马路上,回想着今天发生的一切,他不敢相信事情就这样发生了。他最终还是决定拨打那个电话,掏出手机,解锁屏幕,突然一道强光从屏幕里毫无预兆的发射出来。他一个踉跄倒在了马路中央。而他身后伴随着的是一阵刺耳的刹车声。

手机被摔在一边,屏幕停留在拨号页面,白茫茫的页面。很显然,所谓的强光就是亮白的页面,这种页面在大晚上被打开,眼睛会很不适应,所以
.
.
.
.
.
.
我们来探讨一下夜间模式。

问: android夜间模式的实现有几种?

答: 有好多种。(:◎)≡

  • apk文件

首先我想到的是,通过下载额外的apk文件,然后在获取该apk文件中的资源文件,然后替换之。当然这种方式就不仅限于夜间模式了,可以有各种模式,也就是更换主题了。采用这种方式的,我猜有,微博和QQ。(:◎)≡

想要了解,可以参考http://www.eoeandroid.com/thread-102060-1-1.html

  • 自定义View

把所有用到的view全部自定义一套,根据color和drawable命名规范,比如加_night后缀。根据当前设置的日夜间模式,在onDraw之前获取对应的color或者drawable,然后再进行绘制。采用这种方式的,我猜有,网易新闻。(:)≡

  • setTheme设置style

这种方式,也是我在之前项目中采取的方式。可以感受一下。
采用这种方式,你需要这样
(colors.xml)

<color name="color_a">#ffffff</color>
<color name="color_a_night">#000000</color>

然后这样
(attrs.xml)

<attr name="colorA" format="color"/>

再这样
(styles.xml)

<style name="Theme.A">
     <item name="colorA">@color/color_a</item>
</style>
<style name="Theme.A.Night">
     <item name="colorA">@color/color_a_night</item>
</style>

这样
(activity_xxx.xml)

 <View 
        android:id="@+id/xxx"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:background="?attr/colorA"/>

如果此处你设置的background是一个drawable文件,你需要创建两个drawable文件(一个日间文件,一个带_night的夜间文件)。如果你>的这个drawable文件是个<selector> </selector>,不好意思,根据其中<item> </item>的种类数量,你的每种<item> </item>需要再double一份。

最后再这样
(xxxActivity.java)

@Override 
protected void onCreate(Bundle savedInstanceState) { 
    ...
    if(!isNightMode()) { 
        setTheme(R.style.Theme_A); 
    } else {
        setTheme(R.style.Theme_A_Night); 
    }
    setContentView(R.layout.activity_xxx);
    ...
 }

如果需要切换日夜间,先保存切换的模式,然后通过recreate()重启activity

如果需要在代码中去进行颜色设置,你可以这样

     int[] attrs = { R.attr.colorA };
     TypedArray ta = context.getTheme().obtainStyledAttributes(attrs);
     int color = ta.getColor(ArrayUtils.indexOf(attrs, R.attr.colorA), 0);
     ta.recycle();
     textView.setTextColor(color);

基本的流程就是这样。不管你受没受够,反正我受够了。我就感觉自己挖了一个坑,当color和drawable越来越多,我就越陷越深,这是什么样的一种感受,没错,就是绝望。

  • NightModeHelper

这种方式离我即将要说的方式越来越近了,你仅需要在res目录下,再创建一个带-night后缀的values文件夹values-night,再创建一个colors.xml文件,然后创建颜色<colorname="color_a">#000000</color>,注意,这里颜色命名不用添加_night后缀,与values文件夹中的颜色命名相同即可,只是这里的色值需要更换成夜间的。
然后在xxxActivity.java

private NightModeHelper mNightModeHelper;

 @Override 
protected void onCreate(Bundle savedInstanceState) { 
    ...
    mNightModeHelper = new NightModeHelper(this, R.style.Theme_A);
    setContentView(R.layout.activity_xxx);
    ...
 }

 ...
//需要切换的地方调用该方法
mNightModeHelper.toggle();
...

NightModeHelper在这里https://gist.github.com/slightfoot/c508cdc8828a478572e0

这样就成功了,节省了很多繁冗的步骤。曾经我一度以为这是我最后的救命稻草,直到我遇到了她。。。

  • AppCompatDelegate

  • AppCompatDelegate

  • AppCompatDelegate

重要的事情说三遍!恩,就是这个家伙,不知道什么时候(我是使用的是v23.2.0 Support包),这个家伙里面有了一个这个方法setDefaultNightMode(xxx)。这个方法可以设置四个值:

  • MODE_NIGHT_NO 日间模式
  • MODE_NIGHT_YES 夜间模式
  • MODE_NIGHT_AUTO 根据时间自动切换日夜间模式
  • MODE_NIGHT_FOLLOW_SYSTEM 默认模式,就是跟随系统的设置,据说有可能以后会在android系统设置中添加日夜间模式的设置,此时如果你的app是默认模式,会根据系统设置变化日夜间模式
怎么用这个方法呢?

只要你的activity继承AppCompatActivity,app的sdk最低版本在14以上,你在任何地方都可以调用AppCompatDelegate.setDefaultNightMode(xxx),因为这是个静态方法,设置完之后新开启的页面,都会采用新的模式。你需要在每次切换模式之后,把当前模式保存在本地,然后在下次打开app的时候,获取当前模式并调用这个方法设置一下,就可以使app保持之前的模式。

接下来怎么去自定义自己的日夜间模式呢?

方法与之前的NightModeHelper类似,创建带-night后缀的文件夹(比如:values-night),然后添加你的资源文件,资源文件需要相同的命名(比如:colors.xml),这样就把夜间资源和日间资源关联起来了。这里注意一下,如果是drawable-xxhdpi中的资源需要夜间模式,这些夜间的资源就应该放在drawable-night-xxhdpi文件夹中。这里其实灵活性很大的,比如你可以在values-night中创建一个strings.xml,实现日夜间显示不同的文本。甚至你可以在layout-night中创建一个同样名字的布局文件,实现日夜间显示不同的布局。

官方为了方便开发者,在最新的v23.2.0 Support包增加了一个Theme.AppCompat.DayNight主题,如果继承这个主题,会有一些配置好的属性,如果你喜欢官方的配色,可以考虑继承这个主题,也会省下不少功夫。

还有什么?

还有个方法也提一下,就是setLocalNightMode(xxx),同样可以设置上述提到的四个值。但是这个方法只能通过getDelegate().setLocalNightMode(xxx)调用,这个方法只对当前activity有效果,并且设置完之后需要recreate()才能使其生效。

完了没?

还没完,夜间模式还有个让人头疼的问题就是,当前页面和已经开启的页面的实时更新。之前说的这么多夜间模式方案,都是让之后新的页面获取新的资源,而旧的页面怎么办?

  • 如果你的页面不是很复杂,那就把所有的View再重新设置一遍颜色或者图片。可以直接setTextColor(R.color.xxx) setBackground(R.drawable.xxx)。也可以自定View,重新获取一下资源文件再设置一下。这里还有个前提是,需要更新的activity都得调用getDelegate().setLocalNightMode(xxx)而不是AppCompatDelegate.setDefaultNightMode(xxx),因为后者只是更改了一个全局变量,所以并不会使资源文件立即变化。

  • 如果你的页面很复杂,那就
    .
    .
    .
    .
    .
    .
    recreate()吧!(:◎)≡(如果你有好的方案,也特别希望你能告知我一下!)

他没办法起身,强忍着疼痛,挪动这身体,缓缓的拿起手机。从车上走出来的司机,就在一旁目睹这一切,心想,这年头碰瓷越来越有水平了。颤巍巍的双手触摸着手机屏幕,他最终还是拨打了那个号码,110。司机越来越看不明白,只感觉这套路很深
.
.
.
.
.
.
“喂!是110吗?我钥匙丢家里了,你能找人帮忙开一下吗?”

(最后很感谢这篇分享AppCompat v23.2 — DayNight

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,454评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,553评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,921评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,648评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,770评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,950评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,090评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,817评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,275评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,592评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,724评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,409评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,052评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,815评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,043评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,503评论 2 361
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,627评论 2 350

推荐阅读更多精彩内容