上一篇文章介紹了用 Kotlin 代碼寫布局副女,但是有個問題管行,如果我在原來的 xml 里設(shè)置了 style 怎么辦番挺?怎么在代碼里設(shè)置 style 呢庶近?
據(jù)我所知击碗,Android 官方是沒有提供設(shè)置 style 的 API 的忌栅,那么就沒法用 style 了嗎瓶竭?當(dāng)然不是视乐,我們可以換個思路男摧。
換個思路
我們用 style 是為了減少相同樣式 View 的重復(fù)代碼蔬墩,將重復(fù)代碼抽成 style 達(dá)到復(fù)用的效果译打,其實我們上篇文章中那種寫法就已經(jīng)達(dá)到了這個目的,就是 NumButton
那個類拇颅,我們把重復(fù)的屬性設(shè)置都放到了這個類里奏司,看,這樣是不是已經(jīng)滿足設(shè)置 style 的需求了樟插。
什么韵洋?你就要用 style。好吧黄锤,那就再教大家一個小技巧來用代碼設(shè)置 style搪缨。
就要用 style
大家記不記得 View 的構(gòu)造方法中有一個是四個參數(shù)的:
/**
* ...
* @param defStyleAttr An attribute in the current theme that contains a
* reference to a style resource that supplies default values for
* the view. Can be 0 to not look for defaults.
* 當(dāng)前主題中的一個屬性,包含對為視圖提供默認(rèn)值的樣式資源的引用鸵熟「北啵可以為 0 以不查找默認(rèn)值
* @param defStyleRes A resource identifier of a style resource that
* supplies default values for the view, used only if
* defStyleAttr is 0 or can not be found in the theme. Can be 0
* to not look for defaults.
* 為視圖提供默認(rèn)值的樣式資源的資源標(biāo)識符,僅在 defStyleAttr 為 0 或在主題中找不到時使用流强”越欤可以為 0 以不查找默認(rèn)值。
* ...
*/
public View(context: Context, attrs: AttributeSet?, defStyleAttr: Int, defStyleRes: Int)
著重看一下后兩個參數(shù)的意思打月,其中第四個參數(shù) defStyleRes 不就是我們要找的設(shè)置 style 的地方嘛队腐。那么我們想要給 View 設(shè)置 style 可以這樣寫:
// 注意,這里的第三個參數(shù)傳 0 僵控,原因上面源碼注釋里已經(jīng)寫了
val btn0 = Button(context, null, 0, R.style.StyleCalBtn)
這樣我們寫的 style 就能用了香到。但是... 還有一個問題,有的 View 就沒給提供四個參數(shù)的構(gòu)造方法咋辦报破?就比如 AppCompatButton
,像這種情況我們怎么設(shè)置 style 呢千绪?這時候就需要看一下 View 的第三個參數(shù) defStyleAttr 的含義了充易。
defStyleAttr 的解釋當(dāng)前主題中的一個屬性 attr,這個屬性里有默認(rèn) style 的引用荸型,那么盹靴,我們把這個默認(rèn) style 的引用換成我們自己的不就達(dá)到了設(shè)置 style 的目的了嘛。
先看一下 AppCompatButton
的 attr 名字叫什么:
public AppCompatButton(@NonNull Context context, @Nullable AttributeSet attrs) {
this(context, attrs, R.attr.buttonStyle);
}
根據(jù) AppCompatButton
的源碼可以知道瑞妇,它的 attr 叫 buttonStyle稿静,然后我們就可以自定義一個 theme,把 buttonStyle 的值給覆蓋掉:
<style name="Theme.CalBtn" parent="Theme.CustomViewGroup">
<!-- 注意這里前面加了 android: -->
<item name="android:buttonStyle">@style/StyleCalBtn</item>
</style>
那么我們怎么用這個主題呢辕狰?直接傳到 View 的第三個參數(shù)上改备?這樣不行,第三個參數(shù)要的是一個 attr蔓倍,不能傳 theme悬钳。那怎么辦呢盐捷?
我們可以從 Context 上做手腳,通過 ContextThemeWrapper
給 Context 再包裝一層:
fun Context.toTheme(@StyleRes style: Int) = ContextThemeWrapper(this, style)
這樣 View 在通過 context.obtainStyledAttributes()
方法獲取屬性值的時候默勾,就可以加載我們設(shè)置的主題了碉渡。
新的寫法就可以這樣:
// 注意,這里不能只傳 context 母剥,如果只傳 context 它內(nèi)部會走 (context, null, 0) 這個構(gòu)造滞诺,導(dǎo)致設(shè)置的 style 無效
val btn0 = AppCompatButton(context.toTheme(R.style.Theme_CalBtn), null, android.R.attr.buttonStyle)
這下可以用代碼設(shè)置 style 了吧。