ListView使用多选模式好处

交互与数据分离,在多选状态下不需要修改数据源,在最后确定的时候获取选择索引来确定选择的数据。

ListView模式

  • CHOICE_MODE_NONE:普通模式;
  • CHOICE_MODE_SINGLE:单选模式;
  • CHOICE_MODE_MULTIPLE:多选模式;
  • CHOICE_MODE_MULTIPLE_MODAL:多选模式(与ActionMode配合使用)。

CHOICE_MODE_MULTIPLECHOICE_MODE_MULTIPLE_MODAL 区别: 前者能够同时响应ListView Item点击事件,与Item选择事件; 后者在未进入ActionMode模式下响应Item点击事件,不响应Item选择事件。在进入ActionMode模式后不响应Item点击事件,响应Item选择事件。 同时后者如果Item长按能够自动进入ActionMode模式(也可以使用 ListView.setItemChecked(0,true)来选择一项Item进入ActionMode模式),当所有Item全部取消选择后,自动退出ActionMode模式CHOICE_MODE_MULTIPLE_MODAL使用方式:

listView = (ListView) findViewById(R.id.list_view);
listView.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE_MODAL);
listView.setMultiChoiceModeListener(new ListView.MultiChoiceModeListener());

设置方式

  1. XML布局文件设置(多选模式设置):
 <ListView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:choiceMode="multipleChoice"/>
  1. 代码设置: listView.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE);

选择

ListView多选状态是记录到ListView控件中,由其父类AbsListView实现。 列表Item最外层需要实现Checkable接口,比如CheckBox、CheckedTextView等控件。 如果需要使用容器类控件比如LinearLayout,可以重写控件,实现Checkable接口。 比如:

public class CheckableLinearLayout extends LinearLayout implements Checkable {

    private boolean mChecked = false;

    public CheckableLinearLayout(Context context) {
        super(context);
    }

    public CheckableLinearLayout(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
    }

    public CheckableLinearLayout(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    @Override
    public void toggle() {
        setChecked(!mChecked);
    }

    @Override
    public boolean isChecked() {
        return mChecked;
    }

    @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);
                }
            }
        }
    }

}

获取选择数据

获取到最终选择结果为选中项索引集合,是一个SparseBooleanArray,记录了操作过的item选择状态(如果选中item再取消同样会记录,状态为false)。 SparseBooleanArray checkedItemPositions = listView.getCheckedItemPositions();

API

// 清除选中 public void clearChoices(); // 根据索引判断是否选中 public boolean isItemChecked(int position); // 获取选中数量 public int getCheckedItemCount(); // 根据索引设置选中状态 public void setItemChecked(int position, boolean value);

注意

  • 如果ListView Item内有按钮会抢Item点击事件,解决方法为在Item根布局增加 android:descendantFocusability="blocksDescendants"
  • ActionMode监听回调有两种ActionMode.CallbackMultiChoiceModeListener,后者继承自前者,并新增一个item选择与取消选择的回调方法,但是此方法只在CHOICE_MODE_MULTIPLE_MODAL模式下有效,所以MultiChoiceModeListener监听只适用于CHOICE_MODE_MULTIPLE_MODAL模式。
  • ListView如果使用CHOICE_MODE_MULTIPLE_MODAL模式,可以自动进入ActionMode模式,使用setMultiChoiceModeListener(MultiChoiceModeListener listener)设置监听回调。