在ListView中实现多选并不困难,因为它是自带多选模式的,主要包括CHOICE_MODE_MULTIPLE
和CHOICE_MODE_MULTIPLE_MODAL
。如果你想要更深入的了解这两者是如何工作的。那么这篇文章是非常适合你的。
他们之间的区别大概就是CHOICE_MODE_MULTIPLE模式的特点在于他本身没有排斥性,在能选择item的情况下,也可以响应普通点击事件。
CHOICE_MODE_MULTIPLE_MODAL和CHOICE_MODE_MULTIPLE恰恰相反,他是对普通点击操作和多选操作是排斥的,一旦有一个item被选中,即进入到多选状态,item的onclick事件被屏蔽。这种排斥性也是他比CHOICE_MODE_MULTIPLE多了个MODAL的原因。
我们也可以利用这个多选模式完成RecyclerView的多选。
首先,导入库。将以下行添加到您的build.gradle:
compile 'com.bignerdranch.android:recyclerview-multiselect:+'
接下来,创建一个MultiSelector实例。在我的示例应用程序中,我在我的Fragment添加如下代码:
public class SDTxtListActivity extends AppCompatActivity {
RecyclerView recycler;
ArrayList txtFiles;
private MultiSelector multiSelector = new MultiSelector();
...
}
MultiSelector是管理多选的核心对象。但是,您必须告诉MultiSelector何时进入/离开选择模式。
MultiSelector multiSelector = new MultiSelector();
multiSelector.setSelectable(true); // enter selection mode
MultiSelector multiSelector = new MultiSelector();
multiSelector.setSelectable(false); // leave selection mode
当选择某一项时,您还必须将ViewHolder告诉MultiSelector。
MultiSelector multiSelector = new MultiSelector();
multiSelector.setSelectable(true); // enter selection mode
multiSelector.setSelected(myViewHolder, true); // set myViewHolder to selected
您的ViewHolder必须实现SelectableHolder界面。该库提供了一个名为SwappingHolder的默认实现,用于处理被选择/未选择的视觉效果。
public class TxtFileHodler extends SwappingHolder implements View.OnClickListener,View.OnLongClickListener{ //(1)
private FileMsgModel dm;
private TextView textView;
public TxtFileHodler(View itemView) {
super(itemView,multiSelector); //(2)
textView = (TextView) itemView.findViewById(R.id.txtName);
itemView.setLongClickable(true);
itemView.setOnClickListener(this);
itemView.setOnLongClickListener(this);
}
public void bindView(FileMsgModel fm){
this.dm = fm;
String txtName = dm.getFileName();
textView.setText(txtName);
}
@Override
public void onClick(View view) {
....
}
@Override
public boolean onLongClick(View view) { //(6)
if(!multiSelector.isSelectable()){ //(3)
multiSelector.setSelectable(true); //(4)
multiSelector.setSelected(TxtFileHodler.this,true); //(5)
return true;
}
return false;
}
}
创建一个子类SwappingHolder(1)的ViewHolder。当选择/取消选择时,MultiSelector可以通过SelectableHolder接口与ViewHolder进行通信。因此,您必须在ViewHolder的构造函数(2)中将MultiSelector传递给SwappingHolder。确保MultiSelector尚未处于选择模式(3),然后进入选择模式(4)。最后选择项目(5)。在这个例子中,我们选择长按(6)进入选择模式。您可以决定什么触发选择模式。
在选择模式下单击项目时,还可以通知MultiSelector。但是,您只需在选择模式下通知MultiSelector。当不在选择模式时,您可以正常处理项目点击。
该库提供了一种名为tapSelection()的方便方法。如果MultiSelector处于选择模式,则此方法将切换项目的选定状态并返回true。如果MultiSelector未处于选择模式,则此方法返回false。
public class TxtFileHodler extends SwappingHolder implements View.OnClickListener,View.OnLongClickListener{
private FileMsgModel dm;
private TextView textView;
public TxtFileHodler(View itemView) {
super(itemView,multiSelector);
textView = (TextView) itemView.findViewById(R.id.txtName);
itemView.setLongClickable(true);
itemView.setOnClickListener(this);
itemView.setOnLongClickListener(this);
}
public void bindView(FileMsgModel fm){
this.dm = fm;
String txtName = dm.getFileName();
textView.setText(txtName);
}
@Override
public void onClick(View view) {
if(!multiSelector.tapSelection(TxtFileHodler.this)){
selectTxt(dm);
}
}
....
}
要获得CHOICE_MODE_MULTIPLE_MODAL同样的效果,您可以实现ActionMode.Callback。
但是,该库提供了一个名为ModalMultiSelectorCallback的抽象类。
private ModalMultiSelectorCallback mActionModeCallback
= new ModalMultiSelectorCallback(mMultiSelector) { // (1)
...
};
private class TxtFileHodler extends SwappingHolder
implements View.OnClickListener, View.OnLongClickListener {
...
@Override
public boolean onLongClick(View view) {
if (!mMultiSelector.isSelectable()) {
((AppCompatActivity) getActivity()).startSupportActionMode(mActionModeCallback); // (2)
mMultiSelector.setSelectable(true);
mMultiSelector.setSelected(MyViewHolder.this, true);
return true;
}
return false;
}
}
首先创建一个ModalMultiSelectorCallback的实例,传入你的MultiSelector(1)。然后在启动ActionMode(2)时将此回调传递到系统中。
ModalMultiSelectorCallback公开ActionMode.Callback界面中通常用于创建/准备/响应/销毁ActionMode菜单项的所有方法:
public boolean onCreateActionMode(ActionMode mode, Menu menu);
public boolean onPrepareActionMode(ActionMode mode, Menu menu);
public boolean onActionItemClicked(ActionMode mode, MenuItem item);
public void onDestroyActionMode(ActionMode mode);
对于这个例子,我们使用想要在选择模式中这样做:
private ModalMultiSelectorCallback mActionModeCallback = new ModalMultiSelectorCallback(multiSelector) {
/**
* actionmode的菜单处理
* 进入多选模式后,ActionBar/ToolBar上会弹出一个菜单,这个菜单需要我们自定义
*
*/
@Override
public boolean onCreateActionMode(ActionMode actionMode, Menu menu) {
MenuInflater inflater = actionMode.getMenuInflater();
inflater.inflate(R.menu.sd_card_txt_menu,menu);
return true;
}
@Override
public boolean onPrepareActionMode(ActionMode actionMode, Menu menu) {
return false;
}
//检查是否选中了某一项
@Override
public boolean onActionItemClicked(ActionMode mode, MenuItem item) {
switch (item.getItemId()){
case R.id.action_delete:
mode.finish();
List files = new ArrayList<>();
for(int i = txtFiles.size()-1;i>=0;i--){
if(multiSelector.isSelected(i,0)){ // (1)
FileMsgModel fileModel = txtFiles.get(i);
files.add(fileModel);
}
}
Intent intent=new Intent();
intent.putExtra("MESSAGE", (Serializable) files);
setResult(2,intent);
finish();
multiSelector.clearSelections(); // (2)
return true;
default:
break;
}
return false;
}
/**
* 关闭多选模式
*/
@Override
public void onDestroyActionMode(ActionMode actionMode) {
new TxtFileAdapter(txtFiles).clearSelections();
}
};
private class TxtFileAdapter extends RecyclerView.Adapter {
private ArrayList files;
public TxtFileAdapter(ArrayList txtFiles) {
this.files = txtFiles;
}
@Override
public TxtFileHodler onCreateViewHolder(ViewGroup parent, int viewType) {
LayoutInflater inflater = LayoutInflater.from(SDTxtListActivity.this);
View view = inflater.inflate(R.layout.item,parent,false);
return new TxtFileHodler(view);
}
@Override
public void onBindViewHolder(TxtFileHodler holder, int position) {
FileMsgModel txt = files.get(position);
//我们可以用这行代码替换选中后的背景
holder.setSelectionModeBackgroundDrawable(getResources().getDrawable(R.drawable.sd_txt_back_drawable));
holder.bindView(txt);
}
@Override
public int getItemCount() {
return files.size();
}
//关闭Model
public void clearSelections() {
multiSelector.setSelectable(false);
notifyDataSetChanged();
}
}
我们可以访问每个选定的项目(1)并对其做处理。完成对所有项目的操作后,还需要清除选择(2)。