国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統(tǒng) > Android > 正文

Android 換膚技術(shù)資料整理

2019-12-12 03:21:07
字體:
供稿:網(wǎng)友

Android換膚技術(shù)總結(jié)

背景

縱觀現(xiàn)在各種Android app,其換膚需求可以歸為

- 白天/黑夜主題切換(或者別的名字,通常2套),如同花順/自選股/天天動聽等,UI表現(xiàn)為一個switcher。
- 多種主題切換,通常為會員特權(quán),如QQ/QQ空間。

對于第一種來說,目測應(yīng)該是直接通過本地theme來做的,即所有圖片/顏色的資源都在apk里面打包了。

而對于第二種,則相對復(fù)雜一些,由于作為一種線上服務(wù),可能上架新皮膚,且那么多皮膚包放在apk里面實(shí)在太占體積了,所以皮膚資源會在選擇后再進(jìn)行下載,也就不能直接使用android的那套theme。

技術(shù)方案

內(nèi)部資源加載方案和動態(tài)下載資源下載兩種。

動態(tài)下載可以稱為一種黑科技了,因?yàn)橥枰猦ack系統(tǒng)的一些方法,所以在部分機(jī)型和新的API上有時候可能有坑,但相對好處則很多

- 圖片/色值等資源由于是后臺下發(fā)的,可以隨時更新
- APK體積減小
- 對應(yīng)用開發(fā)者來說,換膚幾乎是透明的,不需要關(guān)心有幾套皮膚
- 可以作為增值服務(wù)賣錢!!

內(nèi)部資源加載方案

內(nèi)部資源加載都是通過android本身那套theme來做的,相對業(yè)務(wù)開發(fā)來說工作量更大(需要定義attr和theme),不同方案類似地都是在BaseActivity里面做setTheme,差別主要在解決以下2個問題的策略:

- setTheme后如何實(shí)時刷新,而不用重新創(chuàng)建頁面(尤其是listview里面的item)。
- 哪些view需要刷新,刷新什么(背景?字體顏色?ImageView的src?)。

自定義view

MultipleTheme

做自定義view是為了在setTheme后會去立即刷新,更新頁面UI對應(yīng)資源(如TextView替換背景圖和文字顏色),在上述項(xiàng)目中,則是通過對rootView進(jìn)行遍歷,對所有實(shí)現(xiàn)了ColorUiInterface的view/viewgroup進(jìn)行setTheme操作來實(shí)現(xiàn)即使刷新的。

顯然這樣太重了,需要把應(yīng)用內(nèi)的各種view/viewgroup進(jìn)行替換。

手動綁定view和要改變的資源類型

Colorful

這個…我們看看用法吧….

ViewGroupSetter listViewSetter = new ViewGroupSetter(mNewsListView);// 綁定ListView的Item View中的news_title視圖,在換膚時修改它的text_color屬性listViewSetter.childViewTextColor(R.id.news_title, R.attr.text_color);// 構(gòu)建Colorful對象來綁定View與屬性的對象關(guān)系mColorful = new Colorful.Builder(this)  .backgroundDrawable(R.id.root_view, R.attr.root_view_bg)  // 設(shè)置view的背景圖片  .backgroundColor(R.id.change_btn, R.attr.btn_bg)  // 設(shè)置背景色  .textColor(R.id.textview, R.attr.text_color)  .setter(listViewSetter) // 手動設(shè)置setter  .create(); // 設(shè)置文本顏色

我就是想換個皮膚,還得在activity里自己去設(shè)置要改變哪個view的什么屬性,對應(yīng)哪個attribute?是不是成本太高了?而且activity的邏輯也很容易被弄得亂七八糟。

動態(tài)資源加載方案

resource替換

覆蓋application的getResource方法,實(shí)現(xiàn)自己的resource,優(yōu)先加載本地皮膚包文件夾下的資源包,對于性能問題,可以通過attribute或者資源名稱規(guī)范(如需要換膚則用skin_開頭)來優(yōu)化,從而不對不換膚的資源進(jìn)行額外檢查開銷。

不過由于Android5.1源碼里,drawable初始化的時候調(diào)用的是loadDrawable,而不是resource.getDrawable,而loadDrawable是私有的方法,無法覆蓋,所以雖然很方便,卻無法繼續(xù)使用(不用關(guān)心任何皮膚相關(guān)的事情,android:color指定顏色就行了,神奇滴會自動換膚)。

自定義LayoutInflator.Factory

開源項(xiàng)目可參照Android-Skin-Loader。

即setFactory使用自定義的LayoutInflator.Factory,可以重點(diǎn)關(guān)注該項(xiàng)目中的SkinInflaterFactory和SkinManager(實(shí)現(xiàn)了自己的getColor、getDrawable、getBitmap、getColorStateList等等方法)。

需要自定義一個tag比如app:customStyle,重寫所有的style,轉(zhuǎn)成set方法,這樣帶來的犧牲就是增加了換膚的成本,要寫很多style,自己去set,并不完全透明了。

Hack Resources internally

黑科技方法,直接對Resources進(jìn)行hack,Resources.Java:

// Information about preloaded resources. Note that they are not// protected by a lock, because while preloading in zygote we are all// single-threaded, and after that these are immutable.private static final LongSparseArray<Drawable.ConstantState>[] sPreloadedDrawables;private static final LongSparseArray<Drawable.ConstantState> sPreloadedColorDrawables  = new LongSparseArray<Drawable.ConstantState>();private static final LongSparseArray<ColorStateList> sPreloadedColorStateLists  = new LongSparseArray<ColorStateList>();

直接對Resources里面的這三個LongSparseArray進(jìn)行替換,由于apk運(yùn)行時的資源都是從這三個數(shù)組里面加載的,所以只要采用interceptor模式:

public class DrawablePreloadInterceptor extends LongSparseArray<Drawable.ConstantState>

自己實(shí)現(xiàn)一個LongSparseArray,并通過反射set回去,就能實(shí)現(xiàn)換膚,具體getDrawable等方法里是怎么取preload數(shù)組的,可以自己看Resources的源碼。

等等,就這么簡單?,NONO,少年你太天真了,怎么去加載xml,9patch的padding怎么更新,怎么打包/加載自定義的皮膚包,drawable的狀態(tài)怎么刷新,等等。這些都是你需要考慮的,在存在插件的app中,還需要考慮是否會互相覆蓋resource id的問題,進(jìn)而需要修改apt,把resource id按位放在2個range。

手Q和獨(dú)立版QQ空間使用的是這種方案,效果挺好。

總結(jié)

盡管動態(tài)加載方案比較黑科技,可能因?yàn)橄到y(tǒng)API的更改而出問題,但相對來說

好處有

- 靈活性高,后臺可以隨時更新皮膚包
- 相對透明,開發(fā)者幾乎不用關(guān)心有幾套皮膚,不用去定義各種theme和attr,甚至連皮膚包的打包都可以交給設(shè)計(jì)或者專門的同學(xué)
- apk體積節(jié)省

存在的問題

沒有完善的開源項(xiàng)目,如果我們采用動態(tài)加載的第二種方案,需要的項(xiàng)目功能包括:

- 自定義皮膚包結(jié)構(gòu)
- 換膚引擎,加載皮膚包資源并load,實(shí)時刷新。
- 皮膚包打包工具
- 對各種rom的兼容

如果有這么一個項(xiàng)目的話,就一勞永逸了,有興趣的同學(xué)可以聯(lián)系一下,大家一起搞一搞。

內(nèi)部加載方案大同小異,主要解決的都是即時刷新的問題,然而從目前的一些開源項(xiàng)目來看,仍然沒有特別簡便的方案。讓我選的話,我寧愿讓界面重新創(chuàng)建,比如重啟activity,或者remove所有view再添加回來(或者你可能想遍歷rootview,然后一個個檢查是否需要換膚然后set…)。

感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!

發(fā)表評論 共有條評論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 手机| 晋州市| 莱阳市| 石城县| 航空| 天峨县| 延川县| 遵义县| 南雄市| 福鼎市| 道真| 当阳市| 石柱| 枣庄市| 邯郸县| 德阳市| 浑源县| 商都县| 婺源县| 齐河县| 博白县| 成安县| 尼勒克县| 大宁县| 霍州市| 凤冈县| 林周县| 防城港市| 钟山县| 新河县| 名山县| 高碑店市| 红河县| 闻喜县| 宜良县| 古蔺县| 眉山市| 洪雅县| 洪泽县| 礼泉县| 洛川县|