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

来源:互联网 发布:访客网络还是 编辑:程序博客网 时间:2024/06/05 14:33

一行代码实现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中显示选中或未选中的标记

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

为了解决这个问题,我设计了一个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)    }    // 设置两个ViewRootLayout中的布局    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)    }}

时序图

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

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

阅读全文
0 0
原创粉丝点击