BaseRecyclerViewAdapterHelper使用

原文:https://www.jianshu.com/p/9cf594d76975

对于RecyclerView, 我们重复编写着那一个又一个的列表界面,有的要分组,有的要添加广告头部、有的要不同类型item排列、等等需求,主要代码有大部分是重复的,如果有一个框架能够帮快速的解决各种需求,那该多好。

BRVAH是GitHub上一个强大的RecyclerAdapter框架,它能节约开发者大量的开发时间,它集成了大部分列表常用需求解决方案。 该框架于2016年4月10号发布,至今,经历了700多次迭代。

presentation by吴思博

github:https://github.com/CymChad/BaseRecyclerViewAdapterHelper

official website:www.recyclerview.org

Author(Brvah) blog:http://www.jianshu.com/u/f958e66439f0

它能干

BaseRecyclerViewAdapterHelper使用_第1张图片

BaseRecyclerViewAdapterHelper使用_第2张图片

解放你的双手,让你少码代码!

BaseRecyclerViewAdapterHelper使用_第3张图片

列表、子布局

BaseRecyclerViewAdapterHelper使用_第4张图片

一行代码轻松切换5种动画

BaseRecyclerViewAdapterHelper使用_第5张图片

让你感觉又回到ListView时代!

BaseRecyclerViewAdapterHelper使用_第6张图片

哼、列表需求来呀!

来了什么好

 

BaseRecyclerViewAdapterHelper使用_第7张图片

提高效率

BRVAH是Github上的一个很棒的开源项目,主要作用是帮助我们更加高效的使用Recyclerview控件,处理项目中常见需求的Adapter,使用起来非常方便。

BRVAH主要是针对Adapter来设计的,为我们提供了一般情况下的BaseQuickAdapter,和几个特定需求下的Adapter,BaseMultiItemQuickAdapter用于复杂类布局列表;BaseItemDraggableAdapter用于拖拽移动和滑动删除类列表;BaseSectionQuickAdapter用于带Section头部View的列表。

一、框架引入:

BaseRecyclerViewAdapterHelper使用_第8张图片

二、基本使用

Activity代码:(和普通的一样)

BaseRecyclerViewAdapterHelper使用_第9张图片

正常的adapter写法:

onCreateViewHolder(这个方法主要生成为每个Item inflater出一个View,方法返回的是一个ViewHolder。 方法是把View直接封装在ViewHolder中,然后我们面向的是ViewHolder这个实例,这个ViewHolder需要我们自己去编写。直接省去了当初Listview的convertView.setTag(holder)和convertView.getTag()这些繁琐的步骤。)和onBindViewHolder(这个方法主要用于适配渲染数据到View中。方法提供给你了一个viewHolder,而不是原来的convertView)等方法。

BaseRecyclerViewAdapterHelper使用_第10张图片

使用BRAVH的Adapter:

首先需要继承BaseQuickAdapter,然后BaseQuickAdapter中第一个泛型T是数据实体类型,第二个BaseViewHolder是ViewHolder,其目的是为了支持扩展ViewHolder。

BaseRecyclerViewAdapterHelper使用_第11张图片

可以通过viewHolder.getView(viewId)获取该控件。 通过viewHolder.getLayoutPosition()获取当前item的position。(为什么有数据不显示?检查一下的RecyclerView是否设置了LayoutManager。)

代码量是3:1的比例!

和原始的adapter相对,减少了70%的代码量。

实现思路:

找到重复部分代码,抽取到基类,非重复部分用抽象方法代替,具体让子类实现。具体代码BaseQuickAdapter如下:(onCreateViewHolder方法和普通adapter一样,对onBindViewHolder方法进行了提取,使用抽象方法convert(抽象方法)代替。)

BaseRecyclerViewAdapterHelper使用_第12张图片

接下来再看看BaseViewHolder怎么写的:( 把ViewHolder进行了封装,利用了SparseArray,加快了查找已创建好控件的速度,没有创建再使用findviewbyId再存入SparseArray)

BaseRecyclerViewAdapterHelper使用_第13张图片

BaseRecyclerViewAdapterHelper使用_第14张图片

三、强大的功能

1.复杂布局实现(多布局)

在实际应用中经常会遇到各种样式的列表、宫格和列表同时存在、分类列表等情况。

1.1多布局

在云阅读中的使用:“评论页面”(优点: 代码量少,节省时间,简洁,清晰,易维护)

BaseRecyclerViewAdapterHelper使用_第15张图片

普通多布局写法:

getItemViewType方法,根据位置获取当前item的类型,在onCreateViewHolder和onBindViewHolder中匹配类型进行操作。

BaseRecyclerViewAdapterHelper使用_第16张图片

BRAVH是怎么实现?

1、实体类(UserComment)实现MultiItemEntity接口,在设置数据的时候,给每一个数据设置itemType。

BaseRecyclerViewAdapterHelper使用_第17张图片

评论的实体类:

BaseRecyclerViewAdapterHelper使用_第18张图片

2、adapter继承BaseMultiItemQuickAdapter类,在构造方法中调用addItemType()方法加入定义的itemType和对应布局, 在Activity中实例化即可。

BaseRecyclerViewAdapterHelper使用_第19张图片

和原始的adapter相对,减少70%的代码量,更加清晰易于维护。

原理分析:

BaseRecyclerViewAdapterHelper使用_第20张图片

BaseRecyclerViewAdapterHelper使用_第21张图片

1.2宫格和列表的混排样式

关于Grid和List的混排样式,Grid样式是一行有多个,而List样式是一行只有一个。 我们可以把List样式看成是Grid样式,它就相当于把一个Grid的item拉长了的样子。

BaseRecyclerViewAdapterHelper使用_第22张图片

列表与网格混排的布局效果,我们可以创建adapter继承BaseMultiItemQuickAdapter添加对应item类型的布局文件,也可以在Activity中创建GridLayoutManager对象,设置spanSize属性,通过Adapter的setSpanSizeLookup()方法设置每种item类型对应的spanSize。设置Recyclerview的addItemDecoration()方法设置添加分割线或设置item间距。

在创建GridLayoutManager对象时,spanSize的参数需要设置,它的作用就是使原来一个item占满一行变为可以最多三个item占满一行。

而设置setSpanSizeLookup()方法返回的是对应每种item类型返回具体的横跨大小。比如代码中TYPE_GRID类型的item在设置的spanSize是1,而GridLayoutManager设置的spanSize是3,那么该类型的item就会以3个item占满一行,相当于每个item占一行的1/3。

代码片段:

BaseRecyclerViewAdapterHelper使用_第23张图片

在我们使用addItemDecoration()添加分割线的方法中对这种混排的列表设置item间距的时候,在getItemOffsets()方法里,通过GridLayoutManager.LayoutParams获取spanSize来确定item类型设置对应间距, 其中spanIndex表示当前行item对应的下标位置,从左到右依次从0开始。

1.3 Recyclerview嵌套Recyclerview的复杂布局

我们“文学漫画”产品中,那些地方用到这种布局?例如蜗牛的领读ReadTrendFragment、Lofter发现音乐Fragment、 云阅读书店BookStoreFragment。

BaseRecyclerViewAdapterHelper使用_第24张图片

BaseRecyclerViewAdapterHelper使用_第25张图片

蜗牛的领读ReadTrendFragmentLofter发现的音乐Fragment

这种界面布局样式,最外面一个RecyclerView,它里面嵌套一个RecyclerView。继承BaseMultiItemQuickAdapter在重写的convert()方法中实例化子Recyclerview。

BaseRecyclerViewAdapterHelper使用_第26张图片

需要注意的是Recyclerview嵌套Recyclerview会出现子Recyclerview抢焦点的问题, 导致界面顶部部分控件被挤出, 只需在最外面的Recyclerview的包裹容器中设置属性

Android:descendantFocusability="blocksDescendants"即可。

2.实现了点击监听事件

BaseRecyclerViewAdapterHelper使用_第27张图片

BRVAH为我们提供好了全面的item和item子View的监听事件,我们只需在继承它提供的Adapter的基础上,通过adapter来调用对应的监听。 设置监听子View前,我们需要在adapter中对应的item的子View注册对应的监听事件。

2.1对于item的监听事件处理

直接在Activity中添加item点击事件,和listview一样。

BaseRecyclerViewAdapterHelper使用_第28张图片

item长按监听事件

BaseRecyclerViewAdapterHelper使用_第29张图片

2.2对item中子View的监听事件处理

首先需要在adapter中,注册子View的监听事件。

BaseRecyclerViewAdapterHelper使用_第30张图片

BaseRecyclerViewAdapterHelper使用_第31张图片

2.3也可以通过viewHolder.getView(viewId)获取该控件,再设置监听。

BaseRecyclerViewAdapterHelper使用_第32张图片

3实现列表加载动画效果

BaseRecyclerViewAdapterHelper使用_第33张图片

3.1默认动画

我们只需将自建的adapter继承它对应满足需求的Adapter,然后在Activity中实例化,通过openLoadAnimation()方法完成特定的动画效果。

BaseRecyclerViewAdapterHelper使用_第34张图片

BRVAH支持5种动画:

渐显、缩放、从下到上,从左到右、从右到左

3.2自定义动画

我们也可以自定义动画,通过实现BaseAnimation这个类,重写

getAnimators(View view)方法来完成自定义动画。

BaseRecyclerViewAdapterHelper使用_第35张图片

3.3动画其他设置

动画默认只执行一次,如果想重复执行可设置: mQuickAdapter.isFirstOnly(false);

设置不显示动画数量:adapter.setNotDoAnimationCount(count);

由于进入界面的item都是很多的速度进来的所以不会出现滑动显示的依次执行动画效果,这个时候会一起执行动画,如果觉得这样的效果不好可以使用setNotDoAnimationCount设置第一屏item不执行动画, 但是如果需要依次执行动画可以重写startAnim让第一个屏幕的item动画延迟执行即可。

BaseRecyclerViewAdapterHelper使用_第36张图片

如何做到的?

首先判断是否开启动画,然后判断是否是自定义动画还是用户选择的自带动画,然后对动画的操作元素进行遍历执行,执行时间为300毫秒,由于上面说了每次填充数据都会调用,所以如何不判断的话,就会导致上下滑动每次都会重复调用动画,动画本身是会耗费性能的。添加一个mLastPosition来存储滑动过的位置,然后判断滑动的位置是否被滑动过,这样就可以避免每次都添加动画了。不过为了满足喜欢动画多过于性能的开发者,如果你想要每次滑动都带动画可以设置isFirstOnly属性即可,默认是不开启的。

BaseRecyclerViewAdapterHelper使用_第37张图片

4添加头部、尾部只需要一行代码。

BaseRecyclerViewAdapterHelper使用_第38张图片

添加:

删除指定view:

删除所有:

BaseRecyclerViewAdapterHelper使用_第39张图片

默认出现了头部就不会显示Empty,和尾部,配置以下方法也支持同时显示:

默认头部尾部都是占满一行,如果需要不占满可以配置:

其中HeaderView和FooterView可以包含多个View,LoadingMoreView在BRVAH中固定成了一个。无论HeaderView里面包含了多少个Header,HeaderView会被整体当成一个Item,所以当Adapter需要刷新HeaderView的时候始终都是刷新position = 0的位置(同理,FooterView也是一个道理)

这里需要重点理解的是HeaderView中Header的数量和RecycleView整个Item数量之间的关系,BRVAH中,HeaderView和FooterView都是LinerLayout,向这个LinerLayout中增加View是不影响RecycleView的ItemCount的。千万不要想当然的认为Add了几个Header然后ItemCount就会加几了。这点理解了源代码理解起来就很容易了

5.实现Recyclerview刷新

这个框架也帮我们实现了,下拉刷新,上拉加载等,傻瓜式实现。

BaseRecyclerViewAdapterHelper使用_第40张图片

5.1上拉加载

在按照BRVAH框架设置完adapter后,在Activity中让类实现BaseQuickAdapter.RequestLoadMoreListener接口,重写onLoadMoreRequested( )方法,在方法中调用mAdapter.addData()来添加新的数据,接着设置mAdapter.loadMoreComplete(),在数据都加载完后设置mAdapter.loadMoreEnd(false)显示数据加载完毕。

BaseRecyclerViewAdapterHelper使用_第41张图片

默认第一次加载会进入回调,如果不需要可以配置:

mQuickAdapter.disableLoadMoreIfNotFullPage();

5.2设置自定义加载布局

mQuickAdapter.setLoadMoreView(newCustomLoadMoreView());

BaseRecyclerViewAdapterHelper使用_第42张图片

BaseRecyclerViewAdapterHelper使用_第43张图片

(注意:如果上拉结束后,下拉刷新需要再次开启上拉监听,需要使用setNewData方法填充数据。)

5.3下拉刷新

BaseRecyclerViewAdapterHelper使用_第44张图片

BaseRecyclerViewAdapterHelper使用_第45张图片

6.实现分组布局

BaseRecyclerViewAdapterHelper使用_第46张图片

设置MySection类继承SectionEntity,创建不同的构造方法来设置item是否有header。 在adapter中,增加了convertHead()方法来加载head数据。在Activity中根据数据创建不同的MySection对象加入集合,设置给adapter。

BaseRecyclerViewAdapterHelper使用_第47张图片

BaseRecyclerViewAdapterHelper使用_第48张图片

根据数据确定不同的样式,用不同的构造方法设置item布局。

在adapter中,构造方法需要传入两个不同的布局id,第一个是item的layout id,第二个是head,item的数据加载在convert()方法中,head的数据加载在convertHead()方法中。

BaseRecyclerViewAdapterHelper使用_第49张图片

 

7.设置空布局

一行代码搞定。

//没有数据的时候默认显示该布局

mQuickAdapter.setEmptyView(getView());

8.实现Recyclerview拖拽滑动删除

BaseRecyclerViewAdapterHelper使用_第50张图片

添加RecyclerView的拖拽和滑动移除很简单,只需adapter继承BaseItemDraggableAdapter类,在Activity中,添加OnItemDragListener和OnItemSwipeListener两个接口,通过adapter配置基本属性即可。

Activity使用代码:

BaseRecyclerViewAdapterHelper使用_第51张图片

拖拽和滑动删除的回调方法

 

BaseRecyclerViewAdapterHelper使用_第52张图片

adapter需要继承BaseItemDraggableAdapter

BaseRecyclerViewAdapterHelper使用_第53张图片

默认不支持多个不同的ViewType之间进行拖拽,如果开发者有所需求:

9自定义ViewHolder

需要继承BaseViewHolder

publicclassMovieViewHolderextendsBaseViewHolder

然后修改adapter的第二个泛型为自定义的ViewHolder

publicclassDataBindingUseAdapterextendsBaseQuickAdapter

注意:需要单独建一个外部类继承BaseViewHolder,否则部分机型会出现ClassCastException,如果是内部类的构造方法要是public,定义的那个类也最好是public。

10添加data方法

BaseRecyclerViewAdapterHelper使用_第54张图片

11其它

树形列表,查看官方网站

四、总结

找到重复部分代码,抽取到基类,非重复部分用抽象方法代替,具体让子类实现。提高了代码复用率, 减少了代码量, 轻松添加各种点击事件,多布局, 一行代码添加加载动画效果,添加头部、尾部、下拉刷新、上拉加载、让你感觉又回到ListView时代! 最重要的是在我们“文学漫画”产品中也可以很好的接入。

 

BaseRecyclerViewAdapterHelper使用_第55张图片

Thank you!!


 

你可能感兴趣的:(android)