RecyclerView的超强辅助Graywater——基础实操篇

关于Graywater的系列文章

  1. RecyclerView的超强辅助Graywater——理论篇
  2. RecyclerView的超强辅助Graywater——基础实操篇
  3. RecyclerView的超强辅助Graywater——点击事件
  4. RecyclerView的超强辅助Graywater——综合实操篇

原本上一篇写完之后就想接着就把实操篇文章给完成了的,但是没想到新需求一来,就加了2周的班,到今天才空下来,才有时间继续完成这篇文章。

上一篇讲解了Graywater的一些基础理论知识,这一篇文章将讲解Graywater的基本使用方法,并搭配了一个基础Demo。下一篇则会讲解Graywater的进阶用法。

注意

以下所使用的Graywater类库,非Graywater最新类库,比最新的旧一点,但是使用起来没有任何问题,个人认为这版比最新的要好理解很多。Graywater官方的类库已经更新到最新,我使用的这个版本Github地址:GraywaterPrimaryDemo Github地址,大家在项目里面可以找到graywater类库。

使用Graywater的步骤

还是要先祭出这张原理图,里面依然是熟悉的5个类,但是多了一些与步骤相关的文字。在GraywaterAdapter实例化之前,我们需要做一些准备工作,也就是将以下步骤中相关类全部创建完成。这是我在实际使用后,觉得最佳的一个文件创建顺序。


带步骤的原理图.png

基础Demo预览效果

根据上面所说的步骤,我们首先来看看如何使用Graywater做出一个最简单的列表。

Graywater简单列表Demo.gif

先展示一下GraywaterPrimaryDemo项目的目录结构

目录结构.png

接下来开始使用Graywater实现上面GIF图的效果,要想使用Graywater,肯定要先导入类库啦。

准备工作:导入Graywater类库

从GraywaterPrimaryDemo中将graywater文件夹拷贝到根目录下,在settings.gradle文件中添加graywater

include ':app', 'graywater'

然后选择菜单栏 File --> Project Structure,选择app--->选择Denpendencies--->选择Module dependency,选择graywater进行依赖。如下图。

屏幕快照 2018-11-06 上午12.23.24 2.png

成功导入graywater后,我们就可以开始愉快的写代码了。

第一步:新建布局文件adapter_item.xml

布局文件很简单,只有一个ImageView和TextView,一个用来显示图片,一个用来显示文本,就不贴出代码了,大家可以在最后Github GraywaterPrimaryDemo地址里下载完整代码查看。

第二步:创建需要的model类

首先将官方Demo中的Primitive拷贝到model目录下,为什么一定要拷贝这个接口,原因后面再讲解。

因为Demo里面的数据类型只有一种,每个Item就是一串文本,和一个图片url(这里大家最好是用对象来管理)。所以只需要创建一个EntertainPrimitive类来管理所有数据就可以了。同时记得EntertainPrimitive类需要实现Primitive接口。model目录下所有的类都需要实现Primitive接口。

第三步:创建对应的ViewHolder文件

从GraywaterPrimaryDemo的GIF图上可以看到,每一个Item就只有一个ImageView和一个TextView。所以我们需要创建一个ViewHolder来管理这2个控件。首先也是将官方Demo中的PrimitiveViewHolder拷贝到viewholder目录下,然后新建EntertainViewHolder来继承PrimitiveViewHolder,为什么也要拷贝这个的原因也在后面再讲,我们先关注核心步骤。

在EntertainViewHolder中创建一个ImageView和Textview,使用itemView来查找,这跟在RecyclerView.Adapter使用时是一样的。

    private ImageView img;
    private TextView title;

    public EntertainViewHolder(View itemView) {
        super(itemView);
        img = itemView.findViewById(R.id.img);
        title = itemView.findViewById(R.id.name);
    }

    public ImageView getImg() {
        return img;
    }

    public TextView getTitle() {
        return title;
    }

第四步:创建对应的ViewHolderCreator文件

EntertainViewHolder创建好了,接下来就需要新建一个viewholdercreator类来生成viewholder。在viewholdercreator目录下创建EntertainViewHolderCreator类,实现GraywaterAdapter.ViewHolderCreator接口,同时要实现接口中的create方法和getViewType方法。

public class EntertainViewHolderCreator implements GraywaterAdapter.ViewHolderCreator {
    @Override
    public RecyclerView.ViewHolder create(ViewGroup parent) {
        return new EntertainViewHolder(GraywaterAdapter.inflate(parent, R.layout.adapter_item));
    }

    @Override
    public int getViewType() {
        return R.layout.adapter_item;
    }
}

create方法返回创建的viewholder对象。
getViewType方法返回的就是item的layoutId。

EntertainViewHolderCreator就是EntertainViewHolder的制造器,专门负责创建EntertainViewHolder对象。有一个需要注意的地方,在new一个EntertainViewHolder对象时,参数是需要传入itemView的,这里使用的GraywaterAdapter.inflate(),查看源码可知,其实就是对LayoutInflater.from(parent.getContext()).inflate();再封装了一层。

GraywaterAdapter.inflate(parent, R.layout.adapter_item)  //实际上就是调用的LayoutInflater.from(parent.getContext()).inflate();

第五步:创建对应的Binder文件

数据model和视图viewholder都创建好了,接下来就可以使用Binder来绑定数据了。创建EntertainBinder类实现GraywaterAdapter.Binder接口,接口里需要传入model的泛型和viewholder的泛型,因为这里是EntertainPrimitive数据所对应的部分,所以直接传EntertainPrimitive和EntertainViewHolder就好。

实现接口里的方法,主要需要重写下面3个方法:

getViewHolderType() : 返回ViewHolder的类型。
bind() : 将model的数据绑定到view上,这里就类似于RecyclerView.Adapter的onBindViewHolder方法。
unbind() :做一些清理操作,一般在里面解除监听事件。

GraywaterPrimaryDemo里的EntertainBinder类也写的很简单

public class EntertainBinder implements GraywaterAdapter.Binder<EntertainPrimitive, EntertainViewHolder> {
    @NonNull
    @Override
    public Class<EntertainViewHolder> getViewHolderType() {
        return EntertainViewHolder.class;
    }

    @Override
    public void prepare(@NonNull EntertainPrimitive model, List<GraywaterAdapter.Binder<? super EntertainPrimitive, ? extends EntertainViewHolder>> binders, int binderIndex) {

    }

    @Override
    public void bind(@NonNull EntertainPrimitive model, @NonNull EntertainViewHolder holder, @NonNull List<GraywaterAdapter.Binder<? super EntertainPrimitive, ? extends EntertainViewHolder>>
            binders, int binderIndex, @NonNull GraywaterAdapter.ActionListener<EntertainPrimitive, EntertainViewHolder> actionListener) {
        Picasso.get().load(model.getUrls().get(binderIndex)).placeholder(R.mipmap.ic_launcher).into(holder.getImg());
        holder.getTitle().setText(model.getTitles().get(binderIndex));
    }

    @Override
    public void unbind(@NonNull EntertainViewHolder holder) {}

}

第六步:创建对应的ItemBinder文件

基本数据model类,视图类viewholder,视图创建器类viewholdercreator类,数据视图绑定类binder类创建好后,就可以创建最后一个ItemBinder类了,这里我们创建对应的EntertainItemBinder类来对EntertainBinder类进行管理。

EntertainItemBinder类的一个作用就是使用getBinderList()方法返回当前所管理所有Binder对象。

    public List<GraywaterAdapter.Binder<? super EntertainPrimitive, ? extends PrimitiveViewHolder>> getBinderList(@NonNull final EntertainPrimitive model, int position) {
        return new ArrayList<GraywaterAdapter.Binder<? super EntertainPrimitive, ? extends PrimitiveViewHolder>>() {{
            for (String s : model.getTitles()) {
                add(entertainBinder);
            }
        }};
    }

说一下为什么要写for循环,Demo中可以看到,每一个福原爱Item就是一个子数据,对应着一个ViewHolder和一个Binder,所以一个EntertainPrimitive的model里有多少个titles就对应了有多少个子数据,就添加到binderlist中,再一起返回,数据就能显示出来了。

最后一步,创建PrimitiveAdapter

所有准备工作做好了,就可以创建适配器PrimitiveAdapter来继承GraywaterAdapter了,注意这里因为要传入3个泛型参数,分别是model、viewholder和model的class类型

public class PrimitiveAdapter extends GraywaterAdapter<Primitive, PrimitiveViewHolder, Class<? extends Primitive>> {

    public PrimitiveAdapter() {
        register(new EntertainViewHolderCreator(), EntertainViewHolder.class);  //将creator和对应的viewholder绑定

        EntertainBinder entertainBinder = new EntertainBinder();
        EntertainItemBinder entertainItemBinder = new EntertainItemBinder(entertainBinder);
        register(EntertainPrimitive.class, entertainItemBinder, null);  //将itemBinder和指定的数据类型绑定
    }

    @Override
    protected Class<? extends Primitive> getModelType(Primitive model) {
        return model.getClass();
    }
}

可以看到有2个register()方法,一个是绑定viewholder和viewholdercreator的,一个是绑定itemBinder和数据类型的。实际上register()方法就是把对应的数据放入到map集合中,然后迅速查找对应的对象。每个数据类型的register只需要写一次就够了。可以看到第二个register()方法的第三个参数传入了null值,这是因为第三个参数是拿来传入点击事件的listener的。没有点击事件就可以直接传入null,需要点击事件就让ItemBinder类实现GraywaterAdapter.ActionListener接口,再传入register方法中。

疑问解答

前面的部分留下了2个疑问,为什么一定要将Primitive和PrimitiveViewHolder类拷贝到对应目录下,并需要继承。我们可以从上面ItemBinder传入的泛型看到是Primitive和PrimitiveViewHolder。这就是泛型的魅力了,在不同的数据类型的情况下,比如还有SportsPrimitive、NewsPrimitive或SportsViewHolder和NewsViewHolder类,因为都实现了Primitive接口,继承了PrimitiveViewHolder类的原因,就能在PrimitiveAdapter中统一管理。

Primitive接口还有一个作用,就是用来区别不同的数据类型。这个可以在Graywater的官方Demo中看到,接下来的一篇Graywater的进阶使用也会讲到。

显示RecyclerView

到这里,Graywater的部分就创建完了。我们只需要在MainActivity中,将EntertainPrimitive赋值,再传给PrimitiveAdapter就可以使用了。

    private void initData() {
        List<String> urls = new ArrayList<>();
        List<String> titles = new ArrayList<>();

        for (int i = 0; i < 40; i++) {
            urls.add("http://n.sinaimg.cn/ent/4_img/upload/0b3147ad/107/w1024h683/20181023/M34_-hmuuiyw5724330.jpg");
            titles.add("item " + (i + 1) + " : " + "福原爱现身退役发布会挥泪告别 甜美比心畅聊感慨");
        }

        mEntertainPrimitive = new EntertainPrimitive(urls, titles);
    }

    private void initContentView() {
        mRecyclerView = findViewById(R.id.recyclerview);
        mRecyclerView.setLayoutManager(new LinearLayoutManager(this, LinearLayoutManager.VERTICAL, false));
        mPrimitiveAdapter = new PrimitiveAdapter();
        mRecyclerView.setAdapter(mPrimitiveAdapter);
    }

    private void updateRecyclerView() {
        mPrimitiveAdapter.add(mEntertainPrimitive);
        mPrimitiveAdapter.notifyDataSetChanged();
    }

总结

按照步骤一步步下来,创建Graywater的过程还是很清晰的。大家有什么疑问可以在评论区提问。

GraywaterPrimaryDemo Github地址
如果对大家有帮助的话,希望大家点下star,谢谢,

如果对你有帮助的话,点赞、评论、赞赏都是对我的鼓励,也是支持我写下去的动力,谢谢!

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

推荐阅读更多精彩内容