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

先看效果图:


pick.png

思路:

使用DialogFragment、RecyclerView、CheckBox

准备:

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

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

开始Code:

1. 整体布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:orientation="vertical"
              android:padding="10dp">

    <android.support.v7.widget.Toolbar
        android:id="@+id/toolbar"
        android:layout_width="match_parent"
        android:layout_height="?android:actionBarSize">

        <TextView
            android:id="@+id/tip_tv"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text=""/>

        <TextView
            android:id="@+id/title_tv"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:text="Title"/>
    </android.support.v7.widget.Toolbar>

    <android.support.v7.widget.RecyclerView
        android:id="@+id/recycler_view"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_weight="1">

    </android.support.v7.widget.RecyclerView>

    <android.support.v4.widget.Space
        android:layout_width="match_parent"
        android:layout_height="10dp"/>

    <TextView
        android:id="@+id/submit"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="right"
        android:background="@drawable/blue_button_background"
        android:padding="10dp"
        android:text="确定"/>
</LinearLayout>

Tips: 用Space可以用来占位

2.PickerDialog

  /**
   * 新建一个dialog
   *
   * @param maxSelected 最大可选数
   * @param title       标题
   * @param list        数据源
   * @return
   */
  public static PickerDialog newInstance(int maxSelected, String title, ArrayList<? extends IContent> 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<T extends IContent> extends 
RecyclerView.Adapter<PickerAdapter.ItemHolder> {

···
public PickerAdapter(int maxSelected, List<T> 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

<?xml version="1.0" encoding="utf-8"?>
<CheckBox xmlns:android="http://schemas.android.com/apk/res/android"
          android:id="@+id/checkbox"
          android:layout_width="match_parent"
          android:layout_height="100dp"
          android:layout_gravity="center"
          android:background="@drawable/selector_item_bg"
          android:button="@drawable/selector_check_button"
          android:gravity="right|center_vertical"
          android:paddingRight="10dp"
          android:textColor="@drawable/selector_text"/>

经过上面的步骤一个简单的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<T> 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 <T>
     */
    public interface OnSelectedListener <T extends IContent> {
        void onSelected(List<T> 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来代替他们,效果一样,但省去了复用的麻烦,也不必处理前面👆两者的事件

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容