夜晚的故事(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()吧!(:◎)≡(如果你有好的方案,也特別希望你能告知我一下?。?/p>

他沒辦法起身,強忍著疼痛,挪動這身體,緩緩的拿起手機。從車上走出來的司機,就在一旁目睹這一切,心想,這年頭碰瓷越來越有水平了。顫巍巍的雙手觸摸著手機屏幕,他最終還是撥打了那個號碼,110。司機越來越看不明白,只感覺這套路很深
.
.
.
.
.
.
“喂!是110嗎?我鑰匙丟家里了,你能找人幫忙開一下嗎?”

(最后很感謝這篇分享AppCompat v23.2?—?DayNight

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 173,466評論 25 708
  • 前言 由于項目需要,近段時間開發的夜間模式功能。主流的方案如下:1、通過切換theme實現2、通過resource...
    三十二蟬閱讀 14,236評論 8 87
  • 前言 隨著一款APP應用功能的不斷完善,用戶群體的不斷增多,APP的更新也就不僅僅局限于功能需求,如何做好良好的用...
    采蘑菇的里奧馬閱讀 28,068評論 43 146
  • 博文出處:對于Android日夜間模式實現的探討,歡迎大家關注我的博客,謝謝!0x0001====== 關于 An...
    俞其榮閱讀 10,706評論 16 74
  • 男生在咖啡店的沙發上臥著,這正是小島的午睡時刻。透過玻璃窗發現,少年時動心過的車馬慢和書信慢,其實是跟著正午的艷陽...
    子回閱讀 529評論 0 1