分组布局的话,在平时使用的频率超级高.所以非常重要.
此篇文章为BaseRecyclerViewAdapterHelper源码解读第六篇,开源库地址,如果没有看过之前5篇文章的同学可以先去看看,大神可直接跳过.
BaseRecyclerViewAdapterHelper源码解读(一) 封装简单的adapter和万能的BaseViewHolder
BaseRecyclerViewAdapterHelper源码解读(二) 添加header和footer
BaseRecyclerViewAdapterHelper源码解读(三) 添加动画
BaseRecyclerViewAdapterHelper源码解读(四) 上拉加载更多
BaseRecyclerViewAdapterHelper源码解读(五) header和footer完善
实体类必须继承SectionEntity
public class MySection extends SectionEntity<Video> {
private boolean isMore;
public MySection(boolean isHeader, String header) {
super(isHeader, header);
}
public MySection(Video t) {
super(t);
}
}
我的理解:上面的Video是实体类,MySection是包装Video的实体类,之所以这样做,是为了让实体类Video可以去继承其他类,这样就不影响实体类了.
adapter构造需要传入两个布局id,第一个是item的,第二个是head的,在convert方法里面加载item数据,在convertHead方法里面加载head数据
public SectionAdapter(int layoutResId, int sectionHeadResId, List data) {
super(layoutResId, sectionHeadResId, data);
}
@Override
protected void convert(BaseViewHolder helper, MySection item) {
helper.setImageUrl(R.id.iv, (String) item.t);
}
@Override
protected void convertHead(BaseViewHolder helper,final MySection item) {
helper.setText(R.id.header, item.header);
helper.setOnClickListener(R.id.more, new View.OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(context,item.header+"more..",Toast.LENGTH_LONG).show();
}
});
}
public abstract class SectionEntity<T> implements Serializable {
/**
* 是否是分组header
*/
public boolean isHeader;
/**
* 包装的实体类,可以没有
*/
public T t;
/**
* 分组header标题
*/
public String header;
public SectionEntity(boolean isHeader, String header) {
this.isHeader = isHeader;
this.header = header;
this.t = null;
}
public SectionEntity(T t) {
this.isHeader = false;
this.header = null;
this.t = t;
}
}
首先,我们有了这个抽象的父类之后,需要实现分组的列表的实体类就必须继承自该抽象父类,然后我们在BaseSectionQuickAdapter中才好根据实体类里面的属性isHeader来分辨是否是header.
public abstract class BaseSectionQuickAdapter<T extends SectionEntity, K extends BaseViewHolder>
extends BaseQuickAdapter<T, K> {
/**
* 分组header布局id
*/
protected int mSectionHeadResId;
/**
* 分组header类型
*/
protected static final int SECTION_HEADER_VIEW = 0x00000444;
/**
* Same as QuickAdapter#QuickAdapter(Context,int) but with
* some initialization data.
*
* @param layoutResId The layout resource id of each item.
* @param sectionHeadResId The section head layout id for each item
* @param data A new list is created out of this one to avoid mutable list
*/
public BaseSectionQuickAdapter(int layoutResId, int sectionHeadResId, List data) {
super(layoutResId, data);
this.mSectionHeadResId = sectionHeadResId;
}
@Override
protected int getDefItemViewType(int position) {
//item的类型 根据实体类里的一个属性 分组RecyclerView里,要么是分组header,要么是普通item
return mData.get(position).isHeader ? SECTION_HEADER_VIEW : 0;
}
@Override
protected K onCreateDefViewHolder(ViewGroup parent, int viewType) {
//创建ViewHolder 如果是分组header,那么view是mSectionHeadResId加载出来的
if (viewType == SECTION_HEADER_VIEW)
return createBaseViewHolder(getItemView(mSectionHeadResId, parent));
//分组内的item,则用默认的方法创建ViewHolder
return super.onCreateDefViewHolder(parent, viewType);
}
@Override
protected boolean isFixedViewType(int type) {
//分组header也是特殊布局,也需要进行跨格子(在GridLayoutManager中,比如SpanCount为2,那么分组header的跨度就是2)
return super.isFixedViewType(type) || type == SECTION_HEADER_VIEW;
}
@Override
public void onBindViewHolder(K holder, int position) {
switch (holder.getItemViewType()) {
case SECTION_HEADER_VIEW:
//如果是分组header,那么需要设置为满Span 即占满
setFullSpan(holder);
//绑定数据 这是分组header
convertHead(holder, getItem(position - getHeaderLayoutCount()));
break;
default:
super.onBindViewHolder(holder, position);
break;
}
}
/**
* 分组header绑定数据
*
* @param helper ViewHolder
* @param item 实体类
*/
protected abstract void convertHead(K helper, T item);
}
这里代码不多,我就直接全部拿出来了.
我们的这个BaseSectionQuickAdapter是继承的BaseQuickAdapter这样的话就不必重写一个重复的方法,可以专注于分组的逻辑.
1. 首先,外界需要通过构造方法传入分组header的布局id,这点无需解释,肯定是需要的.
2. 因为分组header也只是普通的item,特殊的item是headerView,footerView,emptyView,loadMoreView;所以在getDefItemViewType()里面根据实体类里的isHeader属性来区分是分组header类型还是普通item类型.
3. 上面获取了该索引处的type之后,在onCreateDefViewHolder()的时候,就可以直接通过创建分组header所对应的ViewHolder,创建ViewHolder时传入的View是通过mSectionHeadResId创建出来的.
4. 当使用GridLayoutManager时需要将分组header的跨度设置为SpanCount,即占满.所以需要在isFixedViewType()将分组header加上.
5. 绑定数据:当我们在onBindViewHolder()中绑定数据时,需要增加一种类型,就是分组header,用convertHead()暴露给外界调用者去绑定数据.