实现单选及多选的选择对话框

先看效果图:


实现单选及多选的选择对话框_第1张图片
pick.png

思路:

使用DialogFragment、RecyclerView、CheckBox

准备:

圆角Drawable,checkbox Drawable,checkButtonDrawable,字体颜色 Drawable

开发的时候应先把所需要的所有UI准备好之后 再进行开发,而不是边开发边找ui图或者编写xml文件

开始Code:

1. 整体布局文件




    

        

        
    

    

    

    

    

Tips: 用Space可以用来占位

2.PickerDialog

  /**
   * 新建一个dialog
   *
   * @param maxSelected 最大可选数
   * @param title       标题
   * @param list        数据源
   * @return
   */
  public static PickerDialog newInstance(int maxSelected, String title, ArrayList list) {
      Bundle args = new Bundle();
      args.putInt(MAX_NUM, maxSelected);
      args.putString(TITLE, title);
      args.putParcelableArrayList(SOURCE, list);
      PickerDialog fragment = new PickerDialog();
      fragment.setArguments(args);
      return fragment;
  }

IContent是一个接口,有一个getDesc()方法 用于显示单位的名称,由于需要将其序列化,所以IContent 需要继承自Parcelable接口。

public interface IContent extends Parcelable{

    String getDesc();

}

然后给RecyclerView设置一下adapter,布局方式以及间隔就好了

·····
·····
·····
·····
·····
·····
·····
·····
·····
·····
·····
吗?当然不是囖!重点才刚开始,精髓都在adapter里

3.PickerAdapter

public class PickerAdapter extends 
RecyclerView.Adapter {

···
public PickerAdapter(int maxSelected, List list, Context context) {
        this.maxSelected = maxSelected;
        mList = list;
        this.context = context;
    }

    @Override
    public ItemHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View v = LayoutInflater.from(context).inflate(R.layout.item_picker, null, false);
        return new ItemHolder(v);
    }

    @Override
    public void onBindViewHolder(ItemHolder holder, int position) {
        ···
  }

  static class ItemHolder extends RecyclerView.ViewHolder {
        CheckBox cbx;

        public ItemHolder(View itemView) {
            super(itemView);
            cbx = (CheckBox) itemView.findViewById(R.id.checkbox);
        }
    }

}

item_picker.xml



经过上面的步骤一个简单的adapter就写好了,但是有经验的开发一眼就知道上面的代码有复用所带来的显示问题。

由于ListView/RecyclerView的复用机制,如果我们对第一个Item中的CheckBox进行了选中操作,那么当你向上滑动的时候会发现下面的Item中的CheckBox会自动选中了。相信不少人都曾经遇到过这样的问题,通过goole或者stackoverflow,知道不能用view去保存item视图的状态,于是选择去使用数据来控制,这样确实可以基本解决这个问题。

但是如果我们每个数据都再加上一个布尔值用于记录的话,这代价就有点略大了。为什么呢?一是费时二是费力三是完全没必要。其实Android为我们提供了一种完美的数据结构来解决这个问题:SparseBooleanArray ←_←

相信不少看过android内存优化、性能优化的同学都知道这个东西,然后这些文章都只告诉你使用SparseArray替代HashMap,然后会写一堆关于存储结构的东西,告诉你这个更适合android。当时我就比较疑惑问什么SparseArray默认的key都是Integer类型的。那么,现在的代码就是这样了:

    ···
    private SparseBooleanArray mCheckStates = new SparseBooleanArray();
    @Override
    public void onBindViewHolder(ItemHolder holder, int position) {

        holder.cbx.setTag(position);
        holder.cbx.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
            @Override
            public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
                int pos = (int) buttonView.getTag();
                if (isChecked) {
                    mCheckStates.put(pos, true);
                    //do something
                } else {
                    mCheckStates.delete(pos);
                    //do something
                }
            }
        });
        holder.cbx.setText(mList.get(position).getDesc());
        holder.cbx.setChecked(mCheckStates.get(position, false));
    }
    ···

就这些代码就解决了复用的问题,而且完全不必去给数据项新增一个布尔字段,到这里是不是就恍然大悟了

到了这一步后,我们就开始实现单选模式

实现单选

单选比较简单,点击之后关闭dialog然后通过一个回调将选中的值传回去就可以了,当然我们需要先判断一下是否已经有选中了的值,如果有选中了的值了,那么久先将其置为未选中状态,那么怎么知道是否有选中的值呢?SparseArray又立功了

    ···
    @Override
    public void onBindViewHolder(ItemHolder holder, int position) {

        holder.cbx.setTag(position);
        holder.cbx.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
            @Override
            public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {

                int pos = (int) buttonView.getTag();
                if (isChecked) {
                    if (maxSelected == 1 && maxSelected == mCheckStates.size()) {
                        int pre = mCheckStates.keyAt(0);
                        mCheckStates.clear();
                        notifyItemChanged(pre);
                    } 
                    mCheckStates.put(pos, true);
                    //do something
                    if (mOnSelectChangeListener != null) {
                        mOnSelectChangeListener.onSelect(pos, mCheckStates.size());
                    }

                } else {
                    mCheckStates.delete(pos);
                    //do something else
                    if (mOnSelectChangeListener != null) {
                        mOnSelectChangeListener.unSelect(pos);
                    }
                }
            }
        });
        holder.cbx.setText(mList.get(position).getDesc());
        holder.cbx.setChecked(mCheckStates.get(position, false));
    }

    public interface OnSelectChangeListener {

        void onSelect(int pos, int selectedSize);

        void unSelect(int pos);
    }
    ···

实际效果:
[图片上传失败...(image-e4ebf6-1521086193167)]

实现多选

多选分为2种

1.有限制选择个数

由于当选择到最大可选数时,即使把checkbox设为disable也无法控制选中状态,所以需要在代码里置为未选中状态setChecked(!isChecked),但是由于setChecked也会调用onCheckedChanged方法,导致引起死循环,所以需要加锁进行控制

    if (mCheckStates.size() == maxSelected) {
       //不然cbx改变状态.
       lockState = true;
       buttonView.setChecked(!isChecked);
       lockState = false;
       Toast.makeText(context, "最多可选" + maxSelected + "个", Toast.LENGTH_SHORT).show();
       return;
   }

实际效果:
[图片上传失败...(image-588799-1521086193168)]
2.无限制选择个数

无限制就不需要做什么额外的操作,默认就是无限制的,只需要返回选中的数据集就了.
实际效果:
[图片上传失败...(image-8c6eba-1521086193168)]

接着获取选中的集合

public ArrayList getSelectedItems() {
        selectItems.clear();
        for (int i = 0; i < mCheckStates.size(); i++) {
            if (mCheckStates.valueAt(i)) {
                selectItems.add(mList.get(mCheckStates.keyAt(i)));
            }
        }
        return selectItems;
}

最后

在PickerDialog中写一个回调接口,将选中的数据集传递回去,就大功告成了

    /**
     * 选择后的回调,返回选中的list集合
     * * @param 
     */
    public interface OnSelectedListener  {
        void onSelected(List contents);
    }

扩展:

  • 在屏幕旋转时保存选中的数据
    复写onSaveInstanceState和onViewStateRestored函数,当改变屏幕方向时会调用这两个方法
    @Override
    public void onSaveInstanceState(Bundle outState) {
        super.onSaveInstanceState(outState);
        //保存数据
        //···
        outState.putInt(SELECTED_NUM, hasSelectedNum);
        outState.putString(SELECTED_POS_SET, adapter.getSelectedPos());
    }

    @Override
    public void onViewStateRestored(@Nullable Bundle savedInstanceState) {
        super.onViewStateRestored(savedInstanceState);
        //可以在这里设置格数,横屏有3格,竖屏两格
        Configuration newConfig = getActivity().getResources().getConfiguration();
        if (newConfig.orientation == Configuration.ORIENTATION_PORTRAIT) {
            layoutManager.setSpanCount(2);
        } else {
            layoutManager.setSpanCount(3);
        }
        recyclerView.setLayoutManager(layoutManager);

        if (savedInstanceState == null) {
            return;
        }
        //恢复数据
        //···
        hasSelectedNum = savedInstanceState.getInt(SELECTED_NUM, 0);
        selectedPos = savedInstanceState.getString(SELECTED_POS_SET);
        tipsTv.setText(String.format(getString(R.string.has_selected), String.valueOf(hasSelectedNum)));
        if (adapter != null) {
            adapter.setSelectedPosSet(getSelectedPos());
            adapter.setList(mList);
        }

最后的最后

项目地址:https://github.com/vienan/PickerDialog

更新:

2018-3-15
为了避免列表中checkbox的复用问题,除了以上的方法还可以使用DataBinding技术去改变对应对象的值,不过最好的方法还是避免在列表中使用checkbox或RadioButton等等的东西,使用StateListDrawable来代替他们,效果一样,但省去了复用的麻烦,也不必处理前面两者的事件

你可能感兴趣的:(实现单选及多选的选择对话框)