有时候需要列出大量的选择数据,供用户选择,如果直接用布局写一个个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 方法,里面完全可以自己操作,可操作性很强大。