实际应用中,很多地方比如 Button、Tab、ListItem 等都是不同状态有不同的展示形状。举个例子,一个按钮的背景,默认时是一个形状,按下时是一个形状,不可操作时又是另一个形状。有时候,不同状态下改变的不只是背景、图片等,文字颜色也会相应改变。而要处理这些不同状态下展示什么的问题,就要用 selector 来实现了。
selector 标签,可以添加一个或多个 item 子标签,而相应的状态是在 item 标签中定义的。定义的 xml 文件可以作为两种资源使用:drawable 和 color。作为drawable 资源使用时,一般和 shape 一样放于 drawable 目录下,item必须指定 android:drawable 属性;作为 color 资源使用时,则放于 color 目录下,item必须指定 android:color 属性。
下面,看看都有哪些状态可以设置呢?
以下是 bg_btn_selector.xml 的代码,用于按钮的背景:
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<!-- 当前窗口失去焦点时 -->
<item android:drawable="@drawable/bg_btn_lost_window_focused" android:state_window_focused="false" />
<!-- 不可用时 -->
<item android:drawable="@drawable/bg_btn_disable" android:state_enabled="false" />
<!-- 按压时 -->
<item android:drawable="@drawable/bg_btn_pressed" android:state_pressed="true" />
<!-- 被选中时 -->
<item android:drawable="@drawable/bg_btn_selected" android:state_selected="true" />
<!-- 被激活时 -->
<item android:drawable="@drawable/bg_btn_activated" android:state_activated="true" />
<!-- 默认时 -->
<item android:drawable="@drawable/bg_btn_normal" />
</selector>
下面则是 text_btn_selector.xml 的代码,用于按钮的文本颜色:
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<!-- 当前窗口失去焦点时 -->
<item android:color="@android:color/black" android:state_window_focused="false" />
<!-- 不可用时 -->
<item android:color="@android:color/background_light" android:state_enabled="false" />
<!-- 按压时 -->
<item android:color="@android:color/holo_blue_light" android:state_pressed="true" />
<!-- 被选中时 -->
<item android:color="@android:color/holo_green_dark" android:state_selected="true" />
<!-- 被激活时 -->
<item android:color="@android:color/holo_green_light" android:state_activated="true" />
<!-- 默认时 -->
<item android:color="@android:color/white" />
</selector>
最后,则是在控件中的引用:
<Button
android:id="@+id/btn_default"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_margin="8dp"
android:background="@drawable/bg_btn_selector"
android:text="默认按钮"
android:textColor="@color/text_btn_selector" />
那么,在使用过程中,有几点还是需要注意和了解的:
另外,selector标签下有两个比较有用的属性要说一下,添加了下面两个属性之后,则会在状态改变时出现淡入淡出效果,但必须在API Level 11及以上才支持:
最后,关于 ListView 的 ListItem 样式,有两种设置方式,一种是在 ListView 标签里设置 android:listSelector 属性,另一种是在 ListItem 的布局 layout 里设置android:background。但是,这两种设置的结果却有着不同。同时,使用ListView时也有些其他需要注意的地方,总结如下:
当 ListItem 里有 Button 或 CheckBox 之类的控件时,会抢占 ListItem 本身的焦点,导致 ListItem 本身的触摸点击事件会无效。那么,要解决此问题,有三种解决方案:
将 Button 或 CheckBox 换成 TextView 或 ImageView 之类的控件
设置 Button 或 CheckBox 之类的控件设置 focusable 属性为 false
设置 ListItem 的根布局属性
android:descendantFocusability=“blocksDescendants”
第三种是最方便,也是推荐的方式,它会将 ListItem 根布局下的所有子控件都设置为不能获取焦点。android:descendantFocusability 属性的值有三种,其中,ViewGroup是指设置该属性的 View,本例中就是ListItem的根布局: