有时候需要列出大量的选择数据,供用户选择,如果直接用布局写一个个CheckBox 的话不是很合理,所以需要在ListView中加载大量的CheckBox ,但是加载了大量的CheckBox 的时候 ListView的OnItemClickListener 并没有响应,查了一些资料,发现在加载ListView 的Item布局的根目录中添加一个属性 :
`android:descendantFocusability="blocksDescendants"`
此属性有3个可选值 :
- beforeDescendants:viewgroup会优先其子类控件而获取到焦点
- afterDescendants:viewgroup只有当其子类控件不需要获取焦点时才获取焦点
- blocksDescendants:viewgroup会覆盖子类控件而直接获得焦点
当设置了该属性之后,布局的子View就无法获取焦点,这是ListView就能响应OnItemClickListener。
这个时候我们就可以响应点击事件,然后找到CheckBox,设置他们的checked 状态,这种方式可以实现单选效果,但是这并不是我想要的,我试图找出一种简单一点的实现方式。
经过网上查阅资料,发现ListView 中有个属性叫做:
android:choiceMode="singleChoice"
该属性也有3个可选属性:
- singleChoice 单选
- none 默认,并没有选择属性
- multipleChoice 多选
但是设置了这些之后还是无法点击,因为ListView 的子View 需要实现了Checkable 接口才能选中,笔者这里是自定义了一个FrameLayout 让他实现Checkable 接口,然后用这个FrameLayout 作为ListView 子View 的跟布局 , 下面说一下Checkable 中有的方法。
package android.widget;
/**
* Defines an extension for views that make them checkable.
*
*/
public interface Checkable {
/**
* Change the checked state of the view
*
* @param checked The new checked state
*/
void setChecked(boolean checked);
/**
* @return The current checked state of the view
*/
boolean isChecked();
/**
* Change the checked state of the view to the inverse of its current state
*
*/
void toggle();
}
接口里面有3个方法:
-
setCheck(boolean checked)
设置选中状态,我大胆猜测,ListView Item 被点击的时候会调用该方法 -
isCheckd()
判断是否已经选中 -
toggle()
这个方法顾名思义,就是开关的意思。
所以我就在这个方法里面做文章,我遍历FrameLayout所有的子View ,判断每一个子View是否是Checkable 的实例(intanceof 关键字),如果是,我就强转成为Checkable ,然后调用 setCheck(checked) 方法。
这样处理的话,就会变成FrameLayout里面所有可选择的控件都会被选中。当然了,说了这么多可能还是没表达的太清晰,我下面就把我的代码贴出来:
首先是FrameLayout
public class CheckableFrameLayout extends FrameLayout implements Checkable{
public boolean mChecked = false;
public CheckableFrameLayout(Context context, AttributeSet attrs,
int defStyle) {
super(context, attrs, defStyle);
}
public CheckableFrameLayout(Context context, AttributeSet attrs) {
super(context, attrs);
}
public CheckableFrameLayout(Context context) {
super(context);
}
@Override
public void setChecked(boolean checked) {
if (mChecked != checked) {
mChecked = checked;
refreshDrawableState();
for (int i = 0, len = getChildCount(); i < len; i++) {
View child = getChildAt(i);
if(child instanceof Checkable){
((Checkable) child).setChecked(checked);
}
}
}
}
@Override
public boolean isChecked() {
return mChecked;
}
@Override
public void toggle() {
setChecked(!mChecked);
}
}
然后我是ListView 的定义, 我这里设置的是单选模式
<ListView
android:id="@+id/event_lv_name"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1"
android:choiceMode="singleChoice"
android:cacheColorHint="@null"
android:listSelector="@drawable/common_bg"
android:divider="@drawable/line"
android:dividerHeight="1dp"
android:scrollbars="@null" >
</ListView>
最后贴出我的ListView 的 item 布局,需要注意的是CheckBox,需要设置2个属性 android:focusable="false"
android:clickable="false"
让他不可点击,因为我们已经在跟布局中做了处理
<com.example.CheckableFrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:padding="10dp" >
<CheckBox
android:focusable="false"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:button="@drawable/common_checkbox"
android:clickable="false" />
</com.example.CheckableFrameLayout>
这样处理之后,点击ListView的item的时候,被点的Item中的CheckBox就会被选中,其他的就会取消选中
最后,如果我们需要获取被点击了的数据的话,ListView有个方法叫做getCheckedItemPosition()
可以获取选中的Item的位置,通过这个位置我们完全可以获取到数据。
最后拓展一下,Checkable 是所有的View 都可以实现的,同时如果大家想要实现不同样式的选中效果,可以在setCheck()方法中做处理,比如要设置选中背景色的话, 就可以调用setBackground 方法,里面完全可以自己操作,可操作性很强大。