RecyclerView Adapter

  1. 为RecyclerView提供更简单的适配器实现方式,不断更新完好中。

  2. Demo视频演示

  3. GitHub地址

  4. 博客

使用

  • 类库还在开发中。临时没有公布到Jcenter,所以须要在yourProject.gradle文件里加入例如以下代码进行依赖
    allprojects {
repositories {
maven { url 'https://dl.bintray.com/chendongmarch/maven' }
}
}
  • yourApp.gradle文件里加入依赖
    compile 'com.march.lib-adapter:lib-adapter:1.0.0'

BaseViewHolder 的使用

  • 通用ViewHolder,内部使用SparseArray实现View的缓存。
//BaseViewHolder

//获取控件
public <T extends View> T getView(int resId)
public <T extends View> T getView(String resId)
//设置可见
public RvViewHolder setVisibility(int resId, int v)
//文字
public RvViewHolder setText(int resId, String txt)
//图片
public RvViewHolder setImg(int resId, int imgResId)
//监听
public RvViewHolder setClickLis(int resId, View.OnClickListener listener)

通用适配器

单类型数据适配

  • SimpleRvAdapter
//一个简单的实现,实体类不须要再去实现RvQuickInterface接口
SimpleRvAdapter simpleAdapter =
new SimpleRvAdapter<Demo>(self, demos, R.layout.rvquick_item_a) {
@Override
public void onBindView(RvViewHolder holder, Demo data, int pos) {
holder.setText(R.id.item_a_tv, data.title);
}
};

多类型数据适配

  • TypeRvAdapter

    1. 多类型数据适配时须要接受ITypeAdapterModel(接口)类型的的数据,因此进行多类型数据适配时,你的Model须要实现ITypeAdapterModel告知Adapter数据的type

    2. addType(int type,int res)方法用来给每种类型的数据加入不同的布局文件,达到多类型适配的目的。

// Demo类必须实现 ITypeAdapterModel 接口
TypeRvAdapter<Demo> typeAdapter =
new TypeRvAdapter<Demo>(context, data) {
@Override
public void onBindView(RvViewHolder holder, Demo data, int pos, int type) {
//依据类型绑定数据
switch (type) {
case Demo.CODE_DETAIL:
holder.setText(R.id.item_quickadapter_type_title, data.getmDemoTitle()).setText(R.id.item_quickadapter_desc, data.getmDescStr());
break;
case Demo.JUST_TEST:
holder.setText(R.id.item_quickadapter_title, data.getmDemoTitle());
break;
}
} };
typeAdapter.addType(Demo.CODE_DETAIL, R.layout.item_layout_a)
.addType(Demo.JUST_TEST, R.layout.item_layout_b);

九宫格模式适配

  • SectionRvAdapter

    1. 每一个Header以下由多个item。相似微信九宫格照片展示,SectionRvAdapter实际上是多类型适配(TypeRvAdapter)的一种特殊形式,内置了SectionHeader类型的数据,用来更简单的实现九宫格模式的数据适配,由于多类型数据适配的特点他都能够使用,也就是说你能够定义多种类型的数据展示。

    2. 作为ItemHeader的数据类型须要继承AbsSectionHeader(抽象类)

class ItemHeader extends AbsSectionHeader {
String itemHeaderTitle;
}
  1. 作为每一个Section内容的数据类型,假设他是单类型的不须要做其它操作。假设有多种类型的内容须要实现ITypeAdapterModel,具体參照多类型数据适配多类型数据适配
class Content {
int contentIndex
String contentTitle;
}

使用 ISectionRule 配置数据

  1. ISectionRule接口,进行九宫格模式适配使用 ISectionRule 配置数据时,须要加入ISectionRule,这是一种规则。adapter会依据你提供的规则自己主动生成Header

  2. 提供了两种构造方法

// 直接配置 item header 和 content 的 layout 资源
public SectionRvAdapter(Context context,List<ID> originDatas,
int headerLayoutId, int contentLayoutId)
// 仅仅加入 header 的 layout 资源。content的资源能够使用addType方法加入
public SectionRvAdapter(Context context, List<ID> originDatas,
int headerLayoutId)
// ItemHeader表示header的数据类型,Content表示内部数据的数据类型
// 初始化,加入header 和 content的布局文件
adapter = new SectionRvAdapter<ItemHeader, Content>(
this,
contents,
R.layout.item_header_header,
R.layout.item_header_content) {
@Override
protected void onBindItemHeader(BaseViewHolder holder, ItemHeader data, int pos, int type) {
holder.setText(R.id.info1, data.getItemHeaderTitle());
} @Override
protected void onBindContent(BaseViewHolder holder, Content data, int pos, int type) {
holder.setText(R.id.tv, String.valueOf(data.contentIndex));
}
}; // 加入ISectionRule
adapter.addItemHeaderRule(new ISectionRule<ItemHeader, Content>() {
@Override
public ItemHeader buildItemHeader(int currentPos, Content preData, Content currentData, Content nextData) {
// 生成header数据
return new ItemHeader("create new header " + currentData.contentIndex);
} @Override
public boolean isNeedItemHeader(int currentPos, Content preData, Content currentData, Content nextData) {
// 什么时候创建header(当是第一个数据或者index是7的倍数时,插入一个header)
return currentPos == 0 || currentData.contentIndex % 7 == 1;
}
});
mRv.setAdapter(adapter);

使用 HashMap 配置数据

  1. 一个 ItemHeader 下有多个 Content。相似Map<ItemHeader,Content>的数据结构,能够选择在外面构造好数据来进行数据适配

  2. HashMap是无序的,为了保证数据的有序性,使用LinkedHashMap

  3. 相同的也提供了两种构造方法

// 直接配置 item header 和 content 的 layout 资源
public SectionRvAdapter(Context context,
LinkedHashMap<IH, List<ID>> originDatas,
int headerLayoutId, int contentLayoutId) // 仅仅加入 header 的 layout 资源,content的资源能够使用addType方法加入
public SectionRvAdapter(Context context,
LinkedHashMap<IH, List<ID>> originDatas,
int headerLayoutId)
final LinkedHashMap<ItemHeader, List<Content>> map = new LinkedHashMap<>();
adapter = new SectionRvAdapter<ItemHeader, Content>(this, map,
R.layout.item_header_header,
R.layout.item_header_content) {
@Override
protected void onBindItemHeader(BaseViewHolder holder, ItemHeader data, int pos, int type) {
holder.setText(R.id.info1, data.getItemHeaderTitle());
} @Override
protected void onBindContent(BaseViewHolder holder, Content data, int pos, int type) {
TextView tv = (TextView) holder.getView(R.id.tv);
}
};

监听事件

支持单击、双击和长按事件

三种事件

public interface OnItemListener<D> {
// 单击事件
void onClick(int pos, BaseViewHolder holder, D data);
// 长按事件
void onLongPress(int pos, BaseViewHolder holder, D data);
// 双击事件
void onDoubleClick(int pos, BaseViewHolder holder, D data);
}

实现须要的事件

抽象类的实现,能够选择性的实现须要的方法

public abstract class SimpleItemListener<D> implements OnItemListener<D> {

    @Override
public void onClick(int pos, BaseViewHolder holder, D data) { } @Override
public void onLongPress(int pos, BaseViewHolder holder, D data) { } @Override
public void onDoubleClick(int pos, BaseViewHolder holder, D data) { }
} adapter.setItemListener(new SimpleItemListener <GuideData>() {
@Override
public void onClick(int pos, BaseViewHolder holder, GuideData data) {
Toast.makeText(mContext, "单击事件", Toast.LENGTH_SHORT).show();
} @Override
public void onLongPress(int pos, BaseViewHolder holder, GuideData data) {
Toast.makeText(mContext, "长按事件", Toast.LENGTH_SHORT).show();
} @Override
public void onDoubleClick(int pos, BaseViewHolder holder, GuideData data) {
Toast.makeText(mContext,"双击事件", Toast.LENGTH_SHORT).show();
}
});

SectionRvAdapter 事件

adapter.setItemListener(new SimpleItemListener<ItemModel>() {
@Override
public void onClick(int pos, BaseViewHolder holder, ItemModel data) {
// 当是Content数据类型
if (data.getRvType() == AbsAdapter.TYPE_ITEM_DEFAULT) {
Content content = (Content) data.get();
Toast.makeText(SectionAdapterTest.this, content.contentTitle, Toast.LENGTH_SHORT).show();
}else{
// 当是ItemHeader数据类型
}
}
});

数据更新

为了简化数据更新的方法。内置了数据更新的部分方法

内置更新方法

// 插入一条数据
public void insert(int pos, D data)
// 更新数据,isUpdate为false时仅仅会加入数据不会更新显示
public void notifyDataSetChanged(List<D> data, boolean isUpdate)

分页更新方法

// 简化分页载入的更新。调用该方法实现增量更新,不会所有刷新。isAllData为true时表示data是所有数据。为false时表示是追加的数据
public void appendTailRangeData(List<D> data, boolean isAllData)

SectionRvAdapter 追加更新

// SectionRvAdapter比較特别,须要使用单独的更新方法

// 使用SectionRule配置数据时,使用此方法更新
public void updateDataAndItemHeader(List<ID> data)
// 使用Map配置数据时,使用此方法更新
public void updateDataAndItemHeader(Map<IH, List<ID>> map)
// 分页载入更新数据时调用,仅支持使用SectionRule配置数据
public void appendSectionTailRangeData(List<ID> data)

Module

使用Module配置附加功能,眼下有HFModule(加入Header和Footer)、LoadMoreModule(预载入很多其它)

加入 Header 和 Footer

  • HFModule
// 生成和加入module
HFModule hfModule = new HFModule(mContext,
R.layout.header_footer_headerly,
R.layout.header_footer_footerly, mRv);
adapter.addHFModule(hfModule);
// 更改Header 和 Footer的数据,相似数据的配置,你能够在实现的方法里绑定数据和监听事件
adapter = new SimpleRvAdapter<HFModel>(mContext, hfModels, R.layout.header_footer_item) {
@Override
public void onBindHeader(BaseViewHolder header) {
super.onBindHeader(header);
} @Override
public void onBindFooter(BaseViewHolder footer) {
super.onBindFooter(footer);
}
};
// 当仅仅想加入Header或Footer时,使用常亮HFModule.NO_RES表示没有资源
HFModule hfModule = new HFModule(mContext,HFModule.NO_RES,
HFModule.NO_RES, mRv); // 隐藏Header 和 Footer
public void setFooterEnable(boolean footerEnable)
public void setHeaderEnable(boolean headerEnable)

预载入很多其它

  • LoadMoreModule

new LoadMoreModule(int preLoadNum, OnLoadMoreListener lis),preLoadNum表示提前几个Item进行预载入,preLoadNum越大预载入的越提前

载入数据完毕之后须要调用mLoadMoreModule.finishLoad();结束本次载入,保证下次载入能够生效

        // 触发之后1500秒后载入数据
LoadMoreModule loadMoreM = new LoadMoreModule(4, new OnLoadMoreListener() {
@Override
public void onLoadMore(final LoadMoreModule mLoadMoreModule) {
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
List<LoadMoreModel> tempData = new ArrayList<LoadMoreModel>();
for (int i = 0; i < 9; i++) {
tempData.add(new LoadMoreModel("new is " + i));
}
adapter.appendTailRangeData(tempData, false);
mLoadMoreModule.finishLoad();
}
}, 1500);
}
});
adapter.addLoadMoreModule(loadMoreM);

其它

adapterId 区分

为了区分不同的适配器,生成了adapterId,用来检測当前RecyclerView使用的是不是这个adapter

public boolean isThisAdapter(RecyclerView rv)

Sample


//内部类实现
quickAdapter = new TypeRvAdapter<Demo>(self, demos) {
@Override
public void onBindView(RvViewHolder holder, Demo data, int pos, int type) {
// 给控件绑定数据,必须实现
} @Override
public void onBindHeader(RvViewHolder header) {
//给Header绑定数据和事件,不须要能够不实现
} @Override
public void onBindFooter(RvViewHolder footer) {
//给footer绑定数据和事件,不须要能够不实现
}
}; //继承实现
public class MyAdapter extends TypeRvAdapter<Demo> { public MyAdapter(Context context, List<Demo> data) {
super(context, data);
} @Override
public void onBindView(RvViewHolder holder, Demo data, int pos, int type) {
// 给控件绑定数据,必须实现
} @Override
public void onBindHeader(RvViewHolder header) {
//给Header绑定数据和事件,不须要能够不实现
} @Override
public void onBindFooter(RvViewHolder footer) {
//给footer绑定数据和事件,不须要能够不实现
}
}

RecyclerView高速通用适配Adapter的更多相关文章

  1. 为RecyclerView打造通用Adapter

    ##RecycleView简单介绍 RecyclerView控件和ListView的原理有非常多相似的地方,都是维护少量的View来进行显示大量的数据.只是RecyclerView控件比ListVie ...

  2. 为RecyclerView打造通用Adapter 让RecyclerView更加好用

    原文出处: 张鸿洋 (Granker,@鸿洋_ ) 一.概述 记得好久以前针对ListView类控件写过一篇打造万能的ListView GridView 适配器,如今RecyclerView异军突起, ...

  3. RecyclerView的通用适配器

    本来这一个主题应该早就写了,只是项目多,属于自己的时间不多,所以现在才开动!! 前一段时间写了一篇文章,是关于ListView,GriView万能适配器,没有看过的同学,可以先看看那篇文章,然后在来学 ...

  4. RecyclerView打造通用的万能Adapter

    既然想做到通用那么现在摆在面前的就三个问题:数据怎么办?布局怎么办? 绑定怎么办?.数据决定采用泛型,布局打算直接构造传递,绑定显示效果肯定就只能回传. 1 基本改造 数据决定采用泛型,布局打算直接构 ...

  5. RecyclerView的通用适配器,和滚动时不加载图片的封装

    对于RecyclerView我们需要使用RecyclerAdapter,使用方式与ListViewAdapter类似,具体代码大家可以在网上搜索,这里就只教大家使用封装后的简洁RecyclerAdap ...

  6. RecyclerView更通用——listView的onItemClick,onLongItemClick,addHeaderView,addFooterView

    一.点击事件 setOnItemClickListener,setOnItemLongClickListener RecyclerView中虽然没有提供上面这两个接口,但是给我们提供了另外一个接口:O ...

  7. 设计通用的Adapter

    2019年3月18日 10:53:27 参照博文:Android 快速开发系列 打造万能的ListView GridView 适配器 一 BaseViewHolder public class Bas ...

  8. 通用的Adapter

    activity_main.xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android&qu ...

  9. wesome-android

    awesome-android Introduction android libs from github System requirements Android Notice If the lib ...

随机推荐

  1. 关于json-lib中日期类型转换的分析与问题解决

    说明:本文中的json-lib版本为 <dependency> <groupId>net.sf.json-lib</groupId> <artifactId& ...

  2. 使用python3的typing模块提高代码健壮性

    前言:很多人在写完代码一段时间后回过头看代码,很可能忘记了自己写的函数需要传什么参数,返回什么类型的结果,就不得不去阅读代码的具体内容,降低了阅读的速度,加上Python本身就是一门弱类型的语言,这种 ...

  3. SVN服务迁移备份操作步骤

    SVN服务备份操作步骤 1.准备源服务器和目标服务器 源服务器:192.168.1.250 目标服务器:192.168.1.251 root/rootroot 2.对目标服务器(251)装SVN服务器 ...

  4. IE6中 PNG 背景透明的最佳解决方案

    为什么要使用 PNG 图片? 简 单来说,使用 PNG 格式比起 GIF 来表现色彩更丰富,特别是表现渐变以及背景透明的渐变要比GIF格式出色很多.目前,最新的浏览器基本上都支持PNG格式.唯独有万恶 ...

  5. github创建远程仓库

    创建远程仓库 当你已经在本地创建了一个Git仓库后,又想在GitHub创建一个Git仓库,并且让这两个仓库进行远程同步,这样,GitHub上的仓库既可以作为备份,又可以让其他人通过该仓库来协作,真是一 ...

  6. tensorflow 从入门到上天教程一

    tensorflow 是一个google开源的深度学习的框架,执行性能良好,值得使用. caffe,caffe2 通过配置就可以拼凑一个深度学习框架,大大简化流程但也依赖大量的开源库,性能也不错.20 ...

  7. 微信JS-SDK 选取手机照片并进行上传

    项目中遇到需要选取照片上传的需求,因为网页运行在微信的浏览器里面,所以用微信的 js-sdk 提供的选取照片功能,来进行项目开发.实际开发中需要用到微信web开发者工具,详细参考链接:https:// ...

  8. 2017-11-22 Intall Ubuntu Log

    重启之后进入不了系统,安装工具check defect也不好用(问题尚不清楚),决定重做系统 直接用u盘(Universal_USB_Installer制作的安装工具,之前用都没有问题)安装,前面一起 ...

  9. 小程序web-view组件

    不久前微信小程序发布了web-view组件,这个消息在各个圈里引起不小的涟漪.近期正好在做小程序的项目,便研究了一下这个让大家充满期待的组件.   1,web-view这个组件是什么鬼? 官网的介绍: ...

  10. Pyhton编程(六)之基本数据类型-集合(补充)

    集合(set) 集合其实就是一个无序的,自动去重的数据集合,它主要的作用是用来去重和进行关系测试,集合的定义方法如下: name=set("czp") /name=set({1,2 ...