Selector用法<致健忘的自己>

时间:2022-12-01 10:55:56

      1.Selector中文的意思选择器,在Android中常常用来作组件的背景,这样做的好处是省去了用代码控制实现组件在不同状态下不同的背景颜色或图片的变换:

        selector标签,可以添加一个或多个item子标签,而相应的状态是在item标签中定义的,定义的xml文件可以分为两种类型:Color-Selector  、  Color-Drawble

      2.Selector的两种xml具体不同:

        关于的Selector一些属性整理如下:

  • android:state_enabled: 设置触摸或点击事件是否可用状态,一般只在false时设置该属性,表示不可用状态
  • android:state_pressed: 设置是否按压状态,一般在true时设置该属性,表示已按压状态,默认为false
  • android:state_selected: 设置是否选中状态,true表示已选中,false表示未选中
  • android:state_checked: 设置是否勾选状态,主要用于CheckBox和RadioButton,true表示已被勾选,false表示未被勾选
  • android:state_checkable: 设置勾选是否可用状态,类似state_enabled,只是state_enabled会影响触摸或点击事件,而state_checkable影响勾选事件
  • android:state_focused: 设置是否获得焦点状态,true表示获得焦点,默认为false,表示未获得焦点
  • android:state_window_focused: 设置当前窗口是否获得焦点状态,true表示获得焦点,false表示未获得焦点,例如拉下通知栏或弹出对话框时,当前界面就会失去焦点;另外,ListView的ListItem获得焦点时也会触发true状态,可以理解为当前窗口就是ListItem本身
  • android:state_activated: 设置是否被激活状态,true表示被激活,false表示未激活,API Level 11及以上才支持,可通过代码调用控件的setActivated(boolean)方法设置是否激活该控件
  • android:state_hovered: 设置是否鼠标在上面滑动的状态,true表示鼠标在上面滑动,默认为false,API Level 14及以上才支持

        添加了下面两个属性之后,则会在状态改变时出现淡入淡出效果,但必须在API Level 11及以上才支持

  • android:enterFadeDuration 状态改变时,新状态展示时的淡入时间,以毫秒为单位
  • android:exitFadeDuration  状态改变时,旧状态消失时的淡出时间,以毫秒为单位

       Color-Selector:颜色状态选则,可以和color一样使用,颜色会随着组件的状态改变而改变,位置位于/res/color(没有则自己new)/自己起的名字.xml

        代码中引用:R.Color.自己起的名字.xml

        Xml中的引用:@color/自己起的名字.xml

在/res/color/btn_color_selector.xml

<?xml version="1.0" encoding="utf-8"?>

<selector xmlns:android="http://schemas.android.com/apk/res/android">

<item android:state_pressed="true" android:color="#ffff0000"/> <!-- pressed -->
<item android:state_focused="true" android:color="#ff0000ff"/> <!-- focused -->
<item android:color="#ff000000"/> <!-- default -->

</selector>

在Layout中调用

<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!"
android:textSize="16sp"
android:gravity="center"
android:textColor="@color/btn_color_seletor"
/>

      Color-Drawble:背景状态选则,可以和background一样使用,背景会随着组件的状态改变而改变,位置位于/res/drawable/自己起的名字.xml

        代码中引用:R.Drawable.自己起的名字.xml

        Xml中的引用:@drawable/自己起的名字.xml


在/res/drawable/btn_background_selector.xml
<?xml version="1.0" encoding="utf-8"?>

<selector xmlns:android="http://schemas.android.com/apk/res/android">

<item android:state_selected="true" android:drawable="@drawable/timg001" />
<item android:state_focused="true" android:drawable="@drawable/timg001" />
<item android:state_pressed="true" android:drawable="@drawable/timg001" />
<item android:drawable="@drawable/timg" />

</selector>


在Lauout的中的调用

<Button
android:layout_width="wrap_content"
android:layout_height="100dp"
android:text="Hello World!"
android:textSize="16sp"
android:gravity="center"
android:textColor="@color/btn_color_seletor"
android:background="@drawable/btn_background_selector"
/>

     3.知识拓展:关于ListView的ListItem样式,有两种设置方式,一种是在ListView标签里设置android:listSelector属性,另一种是在ListItem的布局layout里设置android:background。但是,这两种设置的结果却有着不同。同时,使用ListView时也有些其他需要注意的地方,总结如下:

     a.android:listSelector设置的ListItem默认背景是透明的,不管你在selector里怎么设置都无法改变它的背景。所以,如果想改ListItem的默认背景,只能通过第二种方式,在ListItem的布局layout里设置android:background

     b.当触摸点击ListItem时,第一种设置方式下,state_pressedstate_focusedstate_window_focused设为true时都会触发,而第二种设置方式下,只有state_pressed会触发。

     c.当ListItem里有Button或CheckBox之类的控件时,会抢占ListItem本身的焦点,导致ListItem本身的触摸点击事件会无效。那么,要解决此问题,有三种解决方案:

  • 将Button或CheckBox换成TextView或ImageView之类的控件
  • 设置Button或CheckBox之类的控件设置focusable属性为false
  • 设置ListItem的根布局属性android:descendantFocusability="blocksDescendants"

第三种是最方便,也是推荐的方式,它会将ListItem根布局下的所有子控件都设置为不能获取焦点。android:descendantFocusability属性的值有三种,其中,ViewGroup是指设置该属性的View,本例中就是ListItem的根布局:

  • beforeDescendants:ViewGroup会优先其子类控件而获取到焦点
  • afterDescendants:ViewGroup只有当其子类控件不需要获取焦点时才获取焦点
  • blocksDescendants:ViewGroup会覆盖子类控件而直接获得焦点