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

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();

对应的显示效果如下:

普通模式 多选模式

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

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

源码部分

MultiSelectAdapter源码

时序图

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

整体结构:

从上面的图可以看出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="http://upload-images.jianshu.io/upload_images/599109-349044bc73d65152.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240" width="400"/>

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

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

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

推荐阅读更多精彩内容