Android開發中難免遇到需要自定義控件的需求,有些是產品的要求在Android標准控件庫中沒有滿足要求的,有些是開發過程中沒有代碼的可復用,自己定義的。
一個好的自定義控件應當和Android本身提供的控件一樣,封裝了一系列的功能以供開發者使用,不僅具有完備的功能,也需要高效的使用內存和CPU。Android本身提供了
一些指標:
1. 應當遵守Android標准的規范(命名,可配置,事件處理等)。
2. 在XML布局中科配置控件的屬性。
3. 對交互應當有合適的反饋,比如按下,點擊等。
4. 具有兼容性, Android版本很多,應該具有廣泛的適用性。
Android已經提供了一系列基礎控件和xml屬性來幫助你創建自定義控件。
1. View的子類
View在Android是最基礎的幾個控件之一, 所有的控件均繼承自View,你也可以直接繼承View也可以繼承其他的控件比如ImageView等。
當然,你至少需要提供一個構造函數,其中Context和AttributeSet作為參數。 舉例如下:
class PieChart extends View {
public PieChart(Context context, AttributeSet attrs) {
super(context, attrs);
}
}
2. 自定義屬性
一個完美的自定義控件也可以添加xml來配置屬性和風格。 要實現這一點,可按照下列步驟來做:
1) 添加自定義屬性
到xml文件中
2) 在xml的中,指定屬性的值
3) 在view中獲取xml中的值
4) 將獲取的值應用到view中
下面繼續舉例說明:
添加 到你的程序中,習慣上一般是放在res/values/attrs.xml文件中,例如:
這段代碼聲明了兩個自定義的屬性 showText和labelPosition,他們屬於一個自定義的實體PieChat。
一旦定義好了屬性,就可以在xml中使用這些屬性了,下面是一個簡單的例子:
可以看到和標准的Android的組件一樣,唯一的差別在他們屬於不同的命名空間,標准的組件的命名空間一般是http://schemas.android.com/apk/res/android,
而我們自定義的命名空間是http://schemas.android.com/apk/res/[your package name]。注意到xmlns:custom中的custom了嗎?你可以使用任意的字符,但是
要和下面的控件的定義中的字符要保持一致。
另外一個需要注意的是, xml中的tag:com.example.customviews.charting.PieChart,需要的完整的包名,如果你的自定義控件是個內部類(好吧,這麼奇葩),
也必須給全路徑,假設PieChat有個內部類PieView,如果在XML中引用它,需要這樣使用:com.example.customviews.charting.PieChart$PieView
3) 應用自定義的屬性值
當View被創建的時候,可以通過AttributeSet讀取所有的定義在xml中的屬性,在構造函數中通過obtainStyledAttributes讀取attrs,
該方法會返回一個TypeArray數組。通過TypeArray可以讀取到已經定義在XML中的方法。下面的例子展示了讀取上文中的xml屬性值。
public PieChart(Context context, AttributeSet attrs) {
super(context, attrs);
TypedArray a = context.getTheme().obtainStyledAttributes(
attrs,
R.styleable.PieChart,
0, 0);
try {
mShowText = a.getBoolean(R.styleable.PieChart_showText, false);
mTextPos = a.getInteger(R.styleable.PieChart_labelPosition, 0);
} finally {
a.recycle();
}
}
需要強調的是, TypeArray使用完畢後需要銷毀,不然會發生內存洩露。
4) 添加自定義的方法和事件
自定義屬性很強大,但缺點也很明顯,它只能在view初始化的時候被應用到控件中。 為了添加更加靈活的行為, 可以為每一個屬性添加getter和setter
對。下面的代碼段展示了PieChat的屬性showText
public boolean isShowText() {
return mShowText;
}
public void setShowText(boolean showText) {
mShowText = showText;
invalidate();
requestLayout();
}
在setShowText中調用了invalidate()和requestLayout(), 保證了view能及時的更新。在你的自定義View中,如果有屬性被改變並且需要立即生效時,
你也必須調用這個方法。 這樣系統會立即重新繪制view。 同樣的,如果view的尺寸或者形狀發生了變化,你也必須調用requestLayout(). 不然會引起
很多問題。
一般你也需要添加事件回調來和調用者溝通。 例如PieChat暴露了OnCurrentItemChanged來通知調用者pie chat發生了旋轉。
在開發過程中,很容易忘記添加一些屬性和事件,特別是作者是這個自定義View的唯一使用者的時候。為使View有更普遍的適用性,應當花些時間考慮的更加周全。
你最好是暴露所有的可能改變外觀和行為的屬性。當然這也對你提出了更高的要求,不然怎麼進不呢。