夜幕降臨憔涉,他走在馬路上,回想著今天發(fā)生的一切析苫,他不敢相信事情就這樣發(fā)生了兜叨。他最終還是決定撥打那個電話,掏出手機衩侥,解鎖屏幕国旷,突然一道強光從屏幕里毫無預兆的發(fā)射出來。他一個踉蹌倒在了馬路中央茫死。而他身后伴隨著的是一陣刺耳的剎車聲跪但。
手機被摔在一邊,屏幕停留在撥號頁面峦萎,白茫茫的頁面屡久。很顯然,所謂的強光就是亮白的頁面爱榔,這種頁面在大晚上被打開被环,眼睛會很不適應,所以
.
.
.
.
.
.
我們來探討一下夜間模式详幽。
問: android夜間模式的實現(xiàn)有幾種筛欢?
答: 有好多種。(:◎)≡
-
apk文件
首先我想到的是唇聘,通過下載額外的apk文件版姑,然后在獲取該apk文件中的資源文件,然后替換之迟郎。當然這種方式就不僅限于夜間模式了剥险,可以有各種模式,也就是更換主題了宪肖。采用這種方式的表制,我猜有健爬,微博和QQ。(:◎)≡
-
自定義View
把所有用到的view全部自定義一套浑劳,根據(jù)color和drawable命名規(guī)范,比如加_night后綴夭拌。根據(jù)當前設置的日夜間模式魔熏,在onDraw之前獲取對應的color或者drawable,然后再進行繪制鸽扁。采用這種方式的蒜绽,我猜有,網(wǎng)易新聞桶现。(:)≡
-
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文件慰于,你需要創(chuàng)建兩個drawable文件(一個日間文件钮科,一個帶_night的夜間文件)。如果你>的這個drawable文件是個<selector> </selector>
婆赠,不好意思绵脯,根據(jù)其中<item> </item>
的種類數(shù)量,你的每種<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目錄下呕缭,再創(chuàng)建一個帶-night后綴的values文件夾values-night
堵泽,再創(chuàng)建一個colors.xml
文件修己,然后創(chuàng)建顏色<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);
...
}
...
//需要切換的地方調(diào)用該方法
mNightModeHelper.toggle();
...
NightModeHelper在這里https://gist.github.com/slightfoot/c508cdc8828a478572e0
這樣就成功了尤辱,節(jié)省了很多繁冗的步驟。曾經(jīng)我一度以為這是我最后的救命稻草厢岂,直到我遇到了她光督。。塔粒。
-
AppCompatDelegate
-
AppCompatDelegate
-
AppCompatDelegate
重要的事情說三遍结借!恩,就是這個家伙卒茬,不知道什么時候(我是使用的是v23.2.0 Support包)船老,這個家伙里面有了一個這個方法setDefaultNightMode(xxx)
。這個方法可以設置四個值:
- MODE_NIGHT_NO 日間模式
- MODE_NIGHT_YES 夜間模式
- MODE_NIGHT_AUTO 根據(jù)時間自動切換日夜間模式
- MODE_NIGHT_FOLLOW_SYSTEM 默認模式圃酵,就是跟隨系統(tǒng)的設置柳畔,據(jù)說有可能以后會在android系統(tǒng)設置中添加日夜間模式的設置,此時如果你的app是默認模式辜昵,會根據(jù)系統(tǒng)設置變化日夜間模式
怎么用這個方法呢荸镊?
只要你的activity繼承AppCompatActivity
,app的sdk最低版本在14以上堪置,你在任何地方都可以調(diào)用AppCompatDelegate.setDefaultNightMode(xxx)
躬存,因為這是個靜態(tài)方法,設置完之后新開啟的頁面舀锨,都會采用新的模式岭洲。你需要在每次切換模式之后,把當前模式保存在本地坎匿,然后在下次打開app的時候盾剩,獲取當前模式并調(diào)用這個方法設置一下,就可以使app保持之前的模式替蔬。
接下來怎么去自定義自己的日夜間模式呢告私?
方法與之前的NightModeHelper
類似,創(chuàng)建帶-night后綴的文件夾(比如:values-night
)承桥,然后添加你的資源文件驻粟,資源文件需要相同的命名(比如:colors.xml
),這樣就把夜間資源和日間資源關聯(lián)起來了凶异。這里注意一下蜀撑,如果是drawable-xxhdpi
中的資源需要夜間模式挤巡,這些夜間的資源就應該放在drawable-night-xxhdpi
文件夾中。這里其實靈活性很大的酷麦,比如你可以在values-night
中創(chuàng)建一個strings.xml
矿卑,實現(xiàn)日夜間顯示不同的文本。甚至你可以在layout-night
中創(chuàng)建一個同樣名字的布局文件沃饶,實現(xiàn)日夜間顯示不同的布局母廷。
官方為了方便開發(fā)者,在最新的v23.2.0 Support包增加了一個Theme.AppCompat.DayNight
主題绍坝,如果繼承這個主題徘意,會有一些配置好的屬性,如果你喜歡官方的配色轩褐,可以考慮繼承這個主題椎咧,也會省下不少功夫。
還有什么把介?
還有個方法也提一下勤讽,就是setLocalNightMode(xxx)
,同樣可以設置上述提到的四個值拗踢。但是這個方法只能通過getDelegate().setLocalNightMode(xxx)
調(diào)用脚牍,這個方法只對當前activity有效果,并且設置完之后需要recreate()
才能使其生效巢墅。
完了沒诸狭?
還沒完,夜間模式還有個讓人頭疼的問題就是君纫,當前頁面和已經(jīng)開啟的頁面的實時更新驯遇。之前說的這么多夜間模式方案,都是讓之后新的頁面獲取新的資源蓄髓,而舊的頁面怎么辦叉庐?
如果你的頁面不是很復雜,那就把所有的View再重新設置一遍顏色或者圖片会喝《傅可以直接
setTextColor(R.color.xxx)
setBackground(R.drawable.xxx)
。也可以自定View肢执,重新獲取一下資源文件再設置一下枉阵。這里還有個前提是,需要更新的activity都得調(diào)用getDelegate().setLocalNightMode(xxx)
而不是AppCompatDelegate.setDefaultNightMode(xxx)
预茄,因為后者只是更改了一個全局變量岭妖,所以并不會使資源文件立即變化。如果你的頁面很復雜反璃,那就
.
.
.
.
.
.
recreate()
吧昵慌!(:◎)≡(如果你有好的方案,也特別希望你能告知我一下;打凇)
他沒辦法起身斋攀,強忍著疼痛,挪動這身體梧田,緩緩的拿起手機淳蔼。從車上走出來的司機,就在一旁目睹這一切裁眯,心想鹉梨,這年頭碰瓷越來越有水平了。顫巍巍的雙手觸摸著手機屏幕穿稳,他最終還是撥打了那個號碼存皂,110。司機越來越看不明白逢艘,只感覺這套路很深
.
.
.
.
.
.
“喂旦袋!是110嗎?我鑰匙丟家里了它改,你能找人幫忙開一下嗎疤孕?”
(最后很感謝這篇分享AppCompat v23.2?—?DayNight)