文/程序员男神
前言
最近项目升级上线,科室提出一个需求,希望可以全选一次把项目确费调,不需要一个个去确费。自己开发过程中发现,每次选中之后,上下滑动就会出现选中错乱的问题,通过了解,这是Recyclerview的复用性质导致的。
项目效果
原理分析
原理:RecycleView具有复用性,条目中的checkBox的选中状态在上下滑动的时候可能会被复用而导致混乱;如果RecycleView条目中的checkBox都有相应的数据源,刷新的时候每个条目中的checkBox会赋予相应的状态,也就不会发生混乱,相当于适配器将数据源赋给指定的控件一样。
解决以上问题可以采取以下思路:
1、首先声明一个map,用来存储checkbox的状态。
//用来记录所有checkbox的状态
private Map<Integer, Boolean> checkStatus = new HashMap<>();
2、循环数据源的集合,初始化checkbox的选中状态。
private void initData() {
list = new ArrayList<String>();
for (int i = 0; i < 50; i++) {
list.add("CheckBox" + i);
}
initCheck(false);
}
//更改集合内部存储的状态
public void initCheck(boolean flag) {
for (int i = 0; i < list.size(); i++) {
//更改指定位置的数据
checkStatus.put(i, flag);
}
}
3、重写Adapter的getView方法时,为每个checkbox添加事件响应并记录选择状态,通过获取获取状态记录值获取所有选择的checkbox值。
@Override
public void onBindViewHolder(@NonNull MyViewHolder holder, final int position) {
holder.checkBox.setText(list.get(position));
//清除监听器
holder.checkBox.setOnCheckedChangeListener(null);
//设置选中状态
holder.checkBox.setChecked(checkStatus.get(position));
//再设置一次CheckBox的选中监听器,当CheckBox的选中状态发生改变时,把改变后的状态储存在Map中
holder.checkBox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
checkStatus.put(position, isChecked);
//check状态一旦改变,保存的check值也要发生相应的变化
}
});
}
完整的Adapter的代码
/**
* 描述: adapter
* 作者|时间: djj on 2019/4/26 17:16
* 博客地址: http://www.jianshu.com/u/dfbde65a03fc
*/
public class MyAdapter extends RecyclerView.Adapter<MyAdapter.MyViewHolder> {
private List<String> list;
private Context mContext;
//用来记录所有checkbox的状态
private Map<Integer, Boolean> checkStatus = new HashMap<>();
public MyAdapter(Context mContext) {
this.mContext = mContext;
initData();
}
private void initData() {
list = new ArrayList<String>();
for (int i = 0; i < 50; i++) {
list.add("CheckBox" + i);
}
initCheck(false);
}
//全选
public void selectAll() {
initCheck(true);
notifyDataSetChanged();
}
//全不选
public void unSelectAll() {
initCheck(false);
notifyDataSetChanged();
}
//更改集合内部存储的状态
public void initCheck(boolean flag) {
for (int i = 0; i < list.size(); i++) {
//更改指定位置的数据
checkStatus.put(i, flag);
}
}
@NonNull
@Override
public MyViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
View view = LayoutInflater.from(mContext).inflate(R.layout.item_recyclerview, parent, false);
return new MyViewHolder(view);
}
@Override
public void onBindViewHolder(@NonNull MyViewHolder holder, final int position) {
holder.checkBox.setText(list.get(position));
//清除监听器
holder.checkBox.setOnCheckedChangeListener(null);
//设置选中状态
holder.checkBox.setChecked(checkStatus.get(position));
//再设置一次CheckBox的选中监听器,当CheckBox的选中状态发生改变时,把改变后的状态储存在Map中
holder.checkBox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
checkStatus.put(position, isChecked);
//check状态一旦改变,保存的check值也要发生相应的变化
}
});
}
@Override
public int getItemCount() {
if (list != null) {
return list.size();
}
return 0;
}
class MyViewHolder extends RecyclerView.ViewHolder {
private CheckBox checkBox;
public MyViewHolder(View itemView) {
super(itemView);
checkBox = itemView.findViewById(R.id.check_box);
}
}
}
效果GIF图
下面就是我们效果图:
我们demo效果
完整的代码路径,需要的请下载:
https://github.com/hellodonj/CheckBoxRecyclerView.git