ListView使用多选模式好处
交互与数据分离,在多选状态下不需要修改数据源,在最后确定的时候获取选择索引来确定选择的数据。
ListView模式
CHOICE_MODE_NONE:普通模式;
CHOICE_MODE_SINGLE:单选模式;
CHOICE_MODE_MULTIPLE:多选模式;
CHOICE_MODE_MULTIPLE_MODAL:多选模式(与ActionMode配合使用)。
CHOICE_MODE_MULTIPLE 与 CHOICE_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());
设置方式
XML布局文件设置(多选模式设置):
android:layout_width="match_parent"
android:layout_height="match_parent"
android:choiceMode="multipleChoice"/>
代码设置:
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.Callback和MultiChoiceModeListener,后者继承自前者,并新增一个item选择与取消选择的回调方法,但是此方法只在CHOICE_MODE_MULTIPLE_MODAL模式下有效,所以MultiChoiceModeListener监听只适用于CHOICE_MODE_MULTIPLE_MODAL模式。
ListView如果使用CHOICE_MODE_MULTIPLE_MODAL模式,可以自动进入ActionMode模式,使用setMultiChoiceModeListener(MultiChoiceModeListener listener)设置监听回调。