下面的代码应该很多人都熟悉:
1
2
3
4
5
6
|
<?xml version=
"1.0"
encoding=
"utf-8"
?>
<item android:drawable=
"@drawable/numpad_button_bg_selected"
android:state_selected=
"true"
></item>
<item android:drawable=
"@drawable/numpad_button_bg_pressed"
android:state_pressed=
"true"
></item>
<item android:drawable=
"@drawable/numpad_button_bg_normal"
></item>
</selector>
|
这是一个给button使用的背景选择,这种不同状态显示不同背景的xml文件我们称为selector。其实selector的本质是一个drawable对象。
如果要用java代码实现上述的selector该如何实现呢?答案是StateListDrawable。
我们先看一个实现StateListDrawable的最简单例子:
1
2
3
4
|
StateListDrawable drawable =
new
StateListDrawable();
drawable.addState(
new
int[]{android.R.attr.state_focused}, mFocusedDrawable);
drawable.addState(
new
int[]{android.R.attr.state_enabled, android.R.attr.state_pressed}, mEnabledPressedDrawable);
drawable.addState(
new
int[0], mDefaultDrawable);
|
这段代码中我们添加了三种状态。
下面我们用更规范的代码来演示如何在java中使用StateListDrawable,以及她对应的xml写法:
首先定义一个获得StateListDrawable对象的方法:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
private StateListDrawable addStateDrawable(Context context, int idNormal, int idPressed, int idFocused) {
StateListDrawable sd =
new
StateListDrawable();
Drawable normal = idNormal == -1 ?
null
: context.getResources().getDrawable(idNormal);
Drawable pressed = idPressed == -1 ?
null
: context.getResources().getDrawable(idPressed);
Drawable focus = idFocused == -1 ?
null
: context.getResources().getDrawable(idFocused);
//注意该处的顺序,只要有一个状态与之相配,背景就会被换掉
//所以不要把大范围放在前面了,如果sd.addState(new[]{},normal)放在第一个的话,就没有什么效果了
sd.addState(
new
int[]{android.R.attr.state_enabled, android.R.attr.state_focused}, focus);
sd.addState(
new
int[]{android.R.attr.state_pressed, android.R.attr.state_enabled}, pressed);
sd.addState(
new
int[]{android.R.attr.state_focused}, focus);
sd.addState(
new
int[]{android.R.attr.state_pressed}, pressed);
sd.addState(
new
int[]{android.R.attr.state_enabled}, normal);
sd.addState(
new
int[]{}, normal);
return
sd;
}
|
其中,就如注释中所讲的addState
的顺序相当重要。
使用ddStateDrawable
1
2
3
|
//……前面对Button的声明略去
okBtn.setBackgroundDrawable(addStateDrawable(
this
, R.drawable.btn_normal, R.drawable.btn_selected, R.drawable.btn_selected));
cancelBtn.setBackgroundDrawable(addStateDrawable(
this
, R.drawable.btn_normal, R.drawable.btn_selected, R.drawable.btn_selected));
|
xml方式实现:
下面对应的具体实例,由于是做背景用,该xml将放于/res/drawable下面(StateList中第一个匹配当前状态的item会被使用。因此,如果第一个item没有任何状态特性的话,那么它将每次都被使用,这也是为什么默认的值必须总是在最后)
selector的定义:
1
2
3
4
5
6
7
|
<?xml version=
"1.0"
encoding=
"utf-8"
?>
<item android:state_pressed=
"true"
android:drawable=
"@drawable/btn_selected"
/>
<item android:state_focused=
"true"
android:drawable=
"@drawable/btn_selected"
/>
<item android:state_enabled=
"true"
android:drawable=
"@drawable/btn_normal"
/>
<item android:drawable=
"@drawable/btn_normal"
/>
</selector>
|
使用selector
1
2
3
4
5
6
7
8
9
10
11
|
<Button
android:id=
"@+id/canel"
android:layout_width=
"wrap_content"
android:layout_height=
"wrap_content"
android:text=
"@string/btn_cancel"
android:layout_margin=
"10dip"
android:layout_weight=
"1"
android:textColor=
"#ffffffff"
android:textSize=
"15sp"
android:background=
"@drawable/button_drawable"
/>
|