美文网首页androidAndroid知识Android开发经验谈
一行代码实现RecyclerView的多选批量操作

一行代码实现RecyclerView的多选批量操作

作者: 大飛機 | 来源:发表于2017-06-13 17:59 被阅读2287次

    MultiSelectAdapter项目地址

    装B部分

    在Android上古时代,如果人类想要实现列表展示,需要用到ListView,虽然用起来很简单,但是很不灵活,随后,Google推出了新的替代品RecyclerView,那RecyclerView确实很灵活,只有你想不到的没有它做不到的,但是呢,天下没有免费的午餐,灵活带来的代价就是更多的Coding,我觉得灵活的本质就是只抽象更底层的逻辑,至于具体的场景就由你们这些码农去做吧,哈哈哈...

    依稀记得,在ListView的时代,多选是ListView自带的功能,Item的点击回调也是自带的功能,用起来还算方便,但是到了RecyclerView的时代呢,虽然很灵活,但是一切都得自己去实现,多选就是其中需要自己实现的功能之一。

    据我了解,如果你想在RecyclerView上实现多选功能,正规的做法是

    1. 在定义Item的时候,要把选中状态View埋进Item的布局中
    2. 在Adapter中定义一个全局变量暂且叫isSelectMode来表示当前状态
    3. 在Adapter中定义一个记录选中状态的集合(HashMap或SparseBooleanArray)
    4. 监听Item的点击事件,被点击时往上面的集合中插入选中或者未选中记录
    5. 刷新列表,在onBindViewHolder中显示选中或未选中的标记
    6. ...

    可以看到,我只想实现一个简单的多选却要写这么多的代码,着实很码农

    为了解决这个问题,我设计了一个Library,可以通过一行代码实现从一个普通的RecyclerView到一个支持多选的RecyclerView的华丽转换

    ✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨

    ✨✨✨✨✨✨✨MultiSelectAdapter✨✨✨✨✨✨✨✨

    ✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨✨

    虽然名字很土,但是用起来很舒服🙂 谁用谁知道

    “那怎么使用呢?真的只需要一行代码就能实现吗?”

    “额,O__O "… 其实,理论上讲,不管什么程序,都可以一行搞定,不是吗?”

    “💩”

    先说一下具体的怎么使用吧


    使用介绍

    recycler.adapter = MultipleSelect.with(Activity).adapter(YourAdapter).build();
    

    对应的显示效果如下:

    普通模式 多选模式
    <img src="https://img.haomeiwen.com/i599109/dc99a22926753055.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240" /> <img src="https://img.haomeiwen.com/i599109/f98c3b1f0fda7e42.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240"/>

    我擦嘞,还真的是一行耶,不过,如果你想要更多不同的显示效果,就得多敲几行了,具体怎么使用我就不多bb了,本文主要想讲一讲这个Library的源码部分,

    至于使用方法请看这里,写的很详细,喜欢的话请记得点个Star:
    MultiSelectAdapter项目地址

    源码部分

    MultiSelectAdapter源码

    时序图

    其实这个项目本身没有用到任何复杂的技术,都是些很常用的东西,只不过是对普普通通的一些代码的封装,我觉得作为程序员,对语言使用的熟练度,高级的技巧固然重要,但更重要的是封装的思想,这种思想是通用的,这就像我们学习外语,会很多单词,很多句式,很多语法结构很重要,但是如果不能用这些学到的东西来准确的表达你的意思,那学这些也没有意义。

    整体结构:

    <img src="https://img.haomeiwen.com/i599109/f5c28b4fef6ad473.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240" width="400" title="结构"/>

    从上面的图可以看出MultipleAdapter使用装饰模式,将用户的Adapter进行加工,然后将装饰后的ViewHolder返回RecyclerView,而装饰的具体过程交给DecorateFactory执行。

    了解了大体的结构以后,我们去看看代码吧,老规矩,从使用的角度入手,一步一步的来吧。

     MultipleAdapter adapter = MultipleSelect
                    .with(Activity)
                    .adapter(YourAdapter)
                    .ignoreViewType(ItemViewType)
                    .stateChangeListener(StateChangeListener)
                    .decorateFactory(? extends DecorateFactory)
                    .customMenu(? extends MenuBar)
                    .build();
    

    这个项目代码是Kotlin写的,跟Java大同小异,应该没什么阅读障碍吧

    首先看看

    MultipleSelect

    MultipleSelect是一个单例,主要是为了构造MultipleAdapter,通过Builder模式,方便用户传入初始化的配置,其中主要的参数有:

    参数 类型 说明
    adapter RecyclerView.Adapter 用户定义的Adapter
    ignoreViewType Integer[] 忽略多选的ItemViewType
    decorateFactory DecorateFactory 多选样式生成器
    customMenu MenuBar 自定义的MenuBar,一般继承CustomMenuBar,传入menu id
    stateChangeListener StateChangeListener 一些回调

    最后通过build()方法构造一个MultipleAdapter并将用户的配置通过构造方法传入参数

    MultipleAdapter

    /**
     * val开头代表这个是常量
     * 该类继承自RecyclerView.Adapter,
     * 同时持有用户的Adapter,利用装饰者
     * 模式,对用户的Adapter进行加工
     */
    class MultipleAdapter(val adapter: RecyclerView.Adapter<RecyclerView.ViewHolder>,
                          val stateChangeListener: StateChangeListener?,
                          val popupToolbar: MenuBar?,
                          val ignoreType: Array<Int>?,
                          val decorateFactory: DecorateFactory,
                          val duration: Long) : RecyclerView.Adapter<RecyclerView.ViewHolder>(), MenuController {
        // showState一共有四种:默认,默认到选择,选择,选择到默认
        var showState = ViewState.DEFAULT
        // 记录选中状态的数组
        val selectIndex = SparseBooleanArray()
        var selectNum = 0
    }
    
    MultipleAdapter.onCreateViewHolder
    override fun onCreateViewHolder(viewGroup: ViewGroup, position: Int): RecyclerView.ViewHolder {
        // 通过调用用户的Adapter,生成原始的ViewHolder
        val outerHolder = adapter.onCreateViewHolder(viewGroup, position)
        // 判断这个ViewHolder是否允许多选
        // 如果是否,则直接返回用户的ViewHolder即可
        if (isIgnore(position))
            return outerHolder
        // 否则,通过decorateFactory对outerHolder加工
        return decorateFactory.decorate(outerHolder, this)
    }
    

    暂时先跳过decorateFactory.decorate这个方法,后面会这种介绍

    MultipleAdapter.onBindViewHolder
    override fun onBindViewHolder(viewHolder: RecyclerView.ViewHolder, position: Int) {
        // 如果类型不是我们定义的ViewHolder
        // 直接返回
        if (viewHolder !is BaseViewHolder) {
            adapter.onBindViewHolder(viewHolder, position)
            return
        }
        
        /**
         * 先调用外界的绑定ViewHolder
         */
        adapter.onBindViewHolder(viewHolder.viewHolder, position)
        
        /**
         * 如果被忽略,则不往下走
         */
        if (isIgnore(position))
            return
        // 根据selectIndex的数据调用BaseViewHolder.selectStateChanged,
        // 选择/未选中
        if (selectIndex.get(position)) {
            viewHolder.selectStateChanged(SelectState.SELECT)
        } else {
            viewHolder.selectStateChanged(SelectState.UN_SELECT)
        }
        // 切换选择模式或普通模式
        viewHolder.showStateChanged(showState)
    }
    

    MultipleAdapter主要的方法就是onCreateViewHolder和onBindViewHolder,其中:

    • onCreateViewHolder:生成支持多选的ViewHolder
    • onBindViewHolder:根据showState和selectIndex刷新状态

    MultipleAdapter暂时停到这里,我们跳转到decorateFactory.decorate(...)中瞧一瞧

    DecorateFactory.decorate
    interface DecorateFactory {
    
        /**
         * 通过这个方法呢,可以将一个普通的Recycler item 转换为一个支持多选的item
         * 是不是很神奇呀
         */
        fun decorate(viewHolder: RecyclerView.ViewHolder, adapter: MultipleAdapter): BaseViewHolder;
    
    }
    

    这是个接口,形参是通过用户的Adapter创建的ViewHolder,返回值是我们自己的BaseViewHolder,我们找一个子类看看具体的实现:

    abstract class CustomViewFactory : DecorateFactory,AnimationInterface {
    
        override fun decorate(viewHolder:ViewHolder, adapter:MultipleAdapter):BaseViewHolder {
            val context = viewHolder.itemView.context
            // 调用onCreateRootView创建整个Item的容器
            val root = onCreateRootView(context)
            val rootParams = ViewGroup.LayoutParams(viewHolder.itemView.layoutParams)
            root.layoutParams = rootParams
            return createViewHolder(context, root, viewHolder, adapter)
        }
    
    
        /**
         * 这个
         */
        fun createViewHolder(context:Context,
                             root: android.view.ViewGroup,
                             viewHolder:ViewHolder,
                             adapter:MultipleAdapter): BaseViewHolder {
            // 调用onCreateSelectView和onCreateNormalView
            // 生成选中和未选中状态View
            val selectView = onCreateSelectView(context)
            val defaultView = onCreateNormalView(context)
            val selectRoot = FrameLayout(context)
            selectRoot.id = R.id.id_select_view
            val layoutParams = FrameLayout.LayoutParams(FrameLayout.LayoutParams.MATCH_PARENT, FrameLayout.LayoutParams.MATCH_PARENT)
            layoutParams.gravity = Gravity.CENTER
            selectRoot.addView(defaultView, layoutParams)
            selectRoot.addView(selectView, layoutParams)
            selectRoot.visibility = android.view.View.GONE
             // 调用onBindSelectView将selectRoot和viewHolder.itemView加入到root中
            onBindSelectView(root, viewHolder.itemView, selectRoot)
    
            selectRoot.measure(root.width, root.height)
            selectRoot.visibility = android.view.View.GONE
            // 生成CustomViewHolder,CustomViewHolder
            // 继承自BaseViewHolder
            return CustomViewHolder(root, viewHolder, adapter, this, selectRoot, selectView, defaultView)
        }
    
        /**
         * 生成默认的SelectView
         */
        abstract fun onCreateSelectView(context:Context): android.view.View
    
        /**
         * 生成默认的UnSelectView
         */
        abstract fun onCreateNormalView(context:Context): android.view.View
    
    
        /**
         * 创建最外层的View
         */
        abstract fun onCreateRootView(context:Context): android.view.ViewGroup
    
    
        /**
         * 绑定SelectView
         */
        abstract fun onBindSelectView(root:ViewGroup, itemView:View, selectView:View)
    

    我们看到,CustomViewFactory也是一个抽象类,只实现了一部分逻辑,再找一个它的子类继续往下看

    class CheckBoxFactory(val color: Int = Color.RED,
                          val duration: Int = 300,
                          val gravity: Int = Gravity.RIGHT,
                          val marginDp:Int = 8) : CustomViewFactory() {
        // 显示动画,如果懒得加动画,直接VISIBLE即可
        override fun onShowAnimation(itemView: View, selectView: View) {
            selectView.visibility = View.VISIBLE
        }
        // 隐藏动画,同上
        override fun onHideAnimation(itemView: View, selectView: View) {
            selectView.visibility = View.GONE
        }
        // 创建选中状态时候显示的View,这里显示一个CheckBox样式的图片
        override fun onCreateSelectView(context: android.content.Context): View {
            val imageView = android.widget.ImageView(context)
            imageView.setImageResource(R.drawable.ic_check_box_black_24dp)
            imageView.setColorFilter(color)
            return imageView
        }
        // 创建未选中状态时候显示的View,这里显示一个CheckBox样式的图片
        override fun onCreateNormalView(context: android.content.Context): View {
            val imageView = android.widget.ImageView(context)
            imageView.setColorFilter(color)
            imageView.setImageResource(R.drawable.ic_check_box_outline_blank_black_24dp)
            return imageView
        }
        // 创建selectView和ItemView的容器
        // 这决定了你的Item最终的布局方式
        override fun onCreateRootView(context: android.content.Context): ViewGroup {
            return FrameLayout(context)
        }
        
        // 设置两个View在RootLayout中的布局
        override fun onBindSelectView(root: ViewGroup, itemView: View, selectView: View) {
            root.removeAllViews()
            root.addView(itemView)
            val params = FrameLayout.LayoutParams(FrameLayout.LayoutParams.WRAP_CONTENT, FrameLayout.LayoutParams.WRAP_CONTENT)
            params.gravity = gravity
            params.leftMargin = marginDp.toPx(root.context)
            params.topMargin = marginDp.toPx(root.context)
            params.rightMargin = marginDp.toPx(root.context)
            params.bottomMargin = marginDp.toPx(root.context)
            root.addView(selectView, params)
        }
    

    上面说到DecorateFactory.decorate的返回值是BaseViewHolder

    BaseViewHolder
    // 抽象类,主要监听了Item的Touch事件,实现监听长按和点击事件
    abstract class BaseViewHolder(val view: View,
                                  val viewHolder: RecyclerView.ViewHolder,
                                  val adapter: MultipleAdapter) : RecyclerView.ViewHolder(view) {
    
        val onTouchListener = OnTouchListener(adapter,this)
        init {
            viewHolder.itemView.setOnTouchListener(onTouchListener)
        }
    
        abstract fun selectStateChanged(state: Int)
    
        open fun showStateChanged(toState: Int) {}
        
    }
    

    BaseViewHolder同样是抽象类,找一个它的子类看一下

    class CustomViewHolder(view: View,
                           viewHolder: RecyclerView.ViewHolder,
                           adapter: MultipleAdapter,
                           val animationInterface: AnimationInterface,
                           val selectViewContainer: View,
                           val selectView: View,
                           val unSelectView: View) : BaseViewHolder(view, viewHolder, adapter) {
        init {
            selectView.setOnTouchListener(onTouchListener)
            unSelectView.setOnTouchListener(onTouchListener)
        }
    
        // 选中或者未选中时候,隐藏和显示对应的View
        override fun selectStateChanged(state: Int) {
            if(state == SelectState.UN_SELECT){
                selectView.visibility = INVISIBLE
                unSelectView.visibility = VISIBLE
            }else if(state == SelectState.SELECT){
                selectView.visibility = VISIBLE
                unSelectView.visibility = INVISIBLE
            }
        }
    
        // 选择模式和普通模式的转换
        override fun showStateChanged(toState: Int) {
            when(toState){
                ViewState.DEFAULT -> {
                    selectViewContainer.visibility = GONE
                }
                ViewState.DEFAULT_TO_SELECT -> {
                    animationInterface.onShowAnimation(itemView,selectViewContainer)
                }
                ViewState.SELECT -> {
                    selectViewContainer.visibility = VISIBLE
                }
                ViewState.SELECT_TO_DEFAULT -> {
                    animationInterface.onHideAnimation(itemView,selectViewContainer)
                }
            }
        }
    }
    

    走到这一步,onCreateViewHolder和onBindViewHolder的实际流程就走完了,剩下的逻辑就简单了

    上面我们介绍BaseViewHolder的时候讲到,BaseViewHolder会监听ItemView的onTouch事件,然后区分长按还是点击,如果是长按,就会调用 MultipleAdapter.onItemLongClick()方法,如果是点击,调用MultipleAdapter.onItemClick()方法

    MultipleAdapter.onItemLongClick
    fun onItemLongClick(position: Int): Boolean {
        if (isIgnore(position))
            return false
        selectIndex.clear()
        // 如果当前是普通模式,在onItemLongClick之后
        // 应该切换到选择模式
        if (showState == ViewState.DEFAULT) {
            selectMode(false)
            // 长按那一条默认选中
            selectIndex.put(position, true)
            stateChangeListener?.onSelect(position, selectNum)
        } else if (showState == ViewState.SELECT) {
             // 否则恢复到普通模式
            selectNum = 0
            cancel()
        }
        // 刷新RecyclerView
        notifyDataSetChanged()
        handler.postDelayed(run, duration)
        return true
    }
    
    MultipleAdapter.onItemClick
    /**
     * 在选择模式中的点击才在这里处理
     * 正常模式的话,会传递给调用者的
     * adapter
     */
    fun onItemClick(position: Int) {
        if (isIgnore(position))
            return
        if (showState != ViewState.SELECT)
            return
        // 置为相反的选择状态
        selectIndex.put(position, !selectIndex[position])
        selectNum += if (selectIndex[position]) 1 else -1
        popupToolbar?.onUpdateTitle(selectNum,getTotal())
        if (selectIndex[position]) {
            stateChangeListener?.onSelect(position, selectNum)
        } else {
            stateChangeListener?.onUnSelect(position, selectNum)
        }
        if (selectNum <= 0) {
            cancel()
        } else {
            notifyItemChanged(position)
        }
    }
    

    时序图

    <img src="https://img.haomeiwen.com/i599109/349044bc73d65152.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240" width="400"/>

    到这里,MultiSelectAdapter主要逻辑大概讲清楚了,剩下的都是些皮毛,如果有疑惑可以去看看源代码,就像我前面说的,这个Library没有使用什么高深的技术和高级的接口调用,有的只是一些对于封装的想法。

    如果发现有什么错误,或有什么更先进的想法,请一定要告诉我。

    相关文章

      网友评论

      本文标题:一行代码实现RecyclerView的多选批量操作

      本文链接:https://www.haomeiwen.com/subject/qvfjqxtx.html