自定義view的時(shí)候号醉,有時(shí)需要用到自定義屬性苍苞,方便我們定制View矾飞。一般來說挥萌,自定義屬性過程如下:
- 定義屬性:在values下的attrs.xml內(nèi)編寫declare-styleable標(biāo)簽來定義屬性;
- 使用屬性:在布局文件中通過獲取
- 獲取屬性:在自定義view中使用TypedArray獲取自定義的屬性亡问。
下面按照自定義View的流程講講自定義屬性:
首先說明一下自定義View的四個(gè)構(gòu)造函數(shù)的區(qū)別:
public class CustomView extends View {
public CustomView(Context context) {
super(context);
}
public CustomView(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}
public CustomView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
public CustomView(Context context, @Nullable AttributeSet attrs, int defStyleAttr, int defStyleRes) {
super(context, attrs, defStyleAttr, defStyleRes);
}
}
在使用過程中官紫,一般都是聯(lián)級(jí)調(diào)用。第一個(gè)構(gòu)造函數(shù)用處并不大州藕,主要在Java代碼中聲明View時(shí)才使用束世;在布局文件中使用自定義的View,則調(diào)用的是第二個(gè)構(gòu)造函數(shù)床玻;第三毁涉,第四個(gè)構(gòu)造函數(shù)是與系統(tǒng)主題有關(guān)的,從參數(shù)defStyleAttr和defStyleRes也可以看得出來锈死;也就是說贫堰,在自定義View時(shí),如果不需要view跟隨主題改變待牵,則前兩個(gè)構(gòu)造函數(shù)就足夠了其屏。
關(guān)于AttributeSet
在第二個(gè)構(gòu)造函數(shù)中,有個(gè)AttributeSet參數(shù)缨该,那這個(gè)參數(shù)有啥用的呢偎行?
查看源碼,我們可以發(fā)現(xiàn)AttributeSet是個(gè)接口,該接口用于解析xml布局中的屬性蛤袒。舉個(gè)例子熄云,假設(shè)定義了一個(gè)CustomerView,在布局中使用:
<com.example.developmc.customview.CustomView
android:layout_width="100dp"
android:layout_height="100dp" />
然后我們?cè)跇?gòu)造函數(shù)中用如下代碼打印AttributeSet中的屬性:
public CustomView(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
int attributeCount = attrs.getAttributeCount();
for (int i = 0; i < attributeCount; i++) {
String attrName = attrs.getAttributeName(i);
String attrVal = attrs.getAttributeValue(i);
Log.e("AttributeSet:", "attrName = " + attrName + " , attrVal = " + attrVal);
}
}
打印結(jié)果如下:
attrName = layout_width , attrVal = 100.0dip
attrName = layout_height , attrVal = 100.0dip
可以看到AttributeSet包含了所有在布局中定義的屬性,并且能夠按順序地取得各個(gè)屬性的name和value妙真。換言之缴允,AttributeSet用于解析View在xml布局中的所有屬性的name和value,這也是自定義View要使用第二個(gè)構(gòu)造函數(shù)的原因珍德。
細(xì)心觀察一下练般,我們?cè)诓季种?strong>layout_width的值是100dp,但打印出來的值卻是100dip,這單位不對(duì)呀菱阵!抱著疑問踢俄,我們修改一下布局:
<com.example.developmc.customview.CustomView
android:layout_width="@dimen/dimen_100"
android:layout_height="100dp" />
其中dimen_100的定義是:
<dimen name="dimen_100">100dp</dimen>
再次打印,結(jié)果如下:
attrName = layout_width , attrVal = @2131165262
attrName = layout_height , attrVal = 100.0dip
可以看到layout_width變成了一個(gè)奇怪的值@2131165262晴及,這個(gè)值是怎么來的呢都办?
我們知道,Android會(huì)在R.java中為定義的屬性生成資源標(biāo)識(shí)符(一個(gè)十六進(jìn)制的數(shù)值)虑稼。在app/build/generated/r/debug下找到并打開R.java琳钉,找到dimen_100,對(duì)應(yīng)的值是0x7f07004e蛛倦,將這個(gè)數(shù)值轉(zhuǎn)為十進(jìn)制歌懒,正好就是“2131165262”!
到這里溯壶,我們可以得出結(jié)論及皂,當(dāng)在布局中直接賦值(如:android:layout_width="100dp"),Attribute拿到的值且改,數(shù)值是正確的验烧,但單位可能會(huì)不對(duì); 當(dāng)在布局中為屬性賦引用值(如:android:layout_width="@dimen/dimen_100")又跛,Attribute拿到的是該值對(duì)應(yīng)的資源標(biāo)識(shí)符碍拆!總的來說,Attribute解析出來的屬性值并不能直接使用慨蓝!不要怕感混,TypedArray就是用于簡(jiǎn)化這方面的工作的。
關(guān)于TypedArray
先來回憶一下礼烈,我們是如何同時(shí)使用Arrtibute和TypedArray的:
首先弧满,新建文件attrs.xml,為自定義View添加一個(gè)自定義屬性:
<declare-styleable name="CustomView">
<attr name="customWidth" format="dimension"/>
</declare-styleable>
然后在布局文件中使用自定義的屬性customWidth:
<com.example.developmc.customview.CustomView
android:layout_width="@dimen/dimen_100"
android:layout_height="100dp"
app:customWidth="@dimen/dimen_100"/>
最后在構(gòu)造函數(shù)中用TypedArray解析customWidth的value此熬,代碼如下:
public CustomView(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
int attributeCount = attrs.getAttributeCount();
for (int i = 0; i < attributeCount; i++) {
String attrName = attrs.getAttributeName(i);
String attrVal = attrs.getAttributeValue(i);
Log.e("AttributeSet:", "attrName = " + attrName + " , attrVal = " + attrVal);
}
TypedArray array = context.obtainStyledAttributes(attrs,R.styleable.CustomView);
float width = array.getDimension(R.styleable.CustomView_customWidth,200f);
Log.e("width:", "widthVal = "+String.valueOf(width));
array.recycle();
}
打印結(jié)果是:
attrName = layout_width , attrVal = @2131165262
attrName = layout_height , attrVal = 100.0dip
attrName = customWidth , attrVal = @2131165262
widthVal = 350.0
測(cè)試虛擬機(jī)的像素是:560dpi谱秽,那么通過計(jì)算洽蛀,100dp對(duì)應(yīng)的像素就是350dip
通過打印結(jié)果,可以看到使用Attribute和TypedArray的區(qū)別:如果使用Attribute疟赊,拿到的結(jié)果并不能直接使用,需要進(jìn)一步處理峡碉;而TypedArray則直接取得了正確的數(shù)值近哟,簡(jiǎn)化了這個(gè)步驟。所以在使用自定義屬性時(shí)鲫寄,我們總是應(yīng)該使用TypedArray的方式獲取屬性值吉执。
這里需要注意的一點(diǎn)是:每次使用完TypedArray之后,要記得調(diào)用recycle()回收地来。這是為什么呢戳玫?
從上述代碼我們是通過context.obtainStyledAttributes獲取TypeadArray實(shí)例的,并不是通過new實(shí)例的方式獲取的未斑。事實(shí)上咕宿,TypedArray類,沒有公有的構(gòu)造函數(shù)蜡秽,是一個(gè)典型的單例模式府阀,程序在運(yùn)行時(shí)維護(hù)一個(gè)TypedArray池,使用時(shí)芽突,向該池中請(qǐng)求一個(gè)實(shí)例试浙,用完之后,調(diào)用 recycle() 方法來釋放該實(shí)例寞蚌,從而使其可被其他模塊復(fù)用田巴。
關(guān)于declare-styleable
我們一般在attrs.xml中通過<declare-styleable>標(biāo)簽聲明自定義的屬性;先看看下面的代碼:
<resources>
<declare-styleable name="CustomView">
<attr name="customWidth" format="dimension"/>
</declare-styleable>
<attr name="customHeight" format="dimension"/>
</resources>
在上述attrs.xml中挟秤,我們自定義了兩個(gè)屬性:customWidth和customHeight壹哺,其中customHeight沒有聲明在declare-styleable。運(yùn)行代碼后煞聪,在生成的R.java文件中斗躏,可以同時(shí)看到這兩個(gè)屬性:
public static final class attr {
public static final int customWidth=0x7f0100ce;
public static final int customHeight=0x7f010001;
}
可以看到定義在declare-styleable中與直接用attr定義沒有實(shí)質(zhì)的不同,系統(tǒng)都會(huì)為我們?cè)赗.attr中生成響應(yīng)的屬性昔脯。但不同的是啄糙,如果聲明在declare-styleable中,系統(tǒng)除了在R.java的attr類下生成資源標(biāo)識(shí)符云稚,還會(huì)為我們?cè)赗.java內(nèi)的styleable類中生成相關(guān)的屬性:
public static final class styleable {
public static final int[] CustomView = {
0x7f0100ce
};
}
可以看到customWidth屬性對(duì)應(yīng)的標(biāo)識(shí)符0x7f0100ce被保存在styleable內(nèi)的數(shù)組中隧饼。如上所示,R.styleable.CustomView是一個(gè)int[]静陈,而里面的元素正是declare-styleable內(nèi)聲明的元素燕雁,這個(gè)數(shù)組在自定義View的構(gòu)造函數(shù)中獲得屬性值時(shí)會(huì)用到诞丽。
將自定義屬性分組聲明在declare-styleabe中的作用就是系統(tǒng)會(huì)自動(dòng)為我們生成這些東西,如果不聲明在declare-styleable中拐格,我們也可以在需要的時(shí)候自己構(gòu)建這個(gè)數(shù)組僧免,只是比較麻煩。當(dāng)我們有多個(gè)自定義View需要用到同一個(gè)自定義屬性時(shí)捏浊,就不能同時(shí)在兩個(gè)declare-styleabe下聲明同一個(gè)屬性了(編譯不通過)懂衩,這時(shí)就可以把這個(gè)屬性直接定義在attr下了,然后在需要使用時(shí)金踪,自己構(gòu)建數(shù)組引用即可浊洞。
本文就到這里,謝謝各位胡岔。