No ViewHolder!!! 一个非官方的纯java版 databinding(拒绝xml配置).

写在前面

我们的目标是 No ViewHolder and No Adapter.

官方的databinding的确十分厉害,各种xml绑定,然后自动生成一波文件,各种吊的飞起,不过容易让人抓不住重点。为了加深理解,我写了这个纯java版databindng, 不需要xml各种配置android:text="@{...}",同时进一步加了绑定Adapter

时间仓促,只粗略的实现了小部分功能。基于注解的性能也有待优化,但它已经极大地提升了我的开发效率。觉得它不错的话,可以一起维护这个项目,向No ViewHolder的目标迈进~

预览

常规的电商首页

所需代码量

实现这样一个带Header, 带上拉加载的列表需要多少代码呢?

  • 两个无聊的 javabean
  • 一个轮播控件
  • 一个Activity(真的不含 Adapter 啊)
  • 然后没有然后了。。。

特性

  • data -> view的单向绑定
  • 支持常用控件的绑定,同时增加了官方没有的Adapter绑定。支持Header上拉加载
  • 代码极其简洁, 无需实例化View, 也没有Adapter, 连ViewHoler 也没有。。。
  • 支持绑定行为的自定义
  • 配合 Rxjava + Lambda 简直上天

源码

https://github.com/fashare2015/NoViewHolder

gralde 依赖

// 1. Add it in your root build.gradle at the end of repositories:
allprojects {
    repositories {
        ...
        maven { url 'https://jitpack.io' }
    }
}
// 2. Add the dependency in your app/build.gradle
dependencies {
        compile 'com.github.fashare2015:NoViewHolder:1.0.1'
}

绑定 Data 和 View

这一块和官方差不多,只是xml配置换成了java注解配置。

绑定单个 View

首先,你手头有一个javabean,就是你在图中看到的妹子列表Item如:

public class MeiZhi {
    @BindImageView(id=R.id.iv_image, placeHolder = R.mipmap.ic_launcher)
    public String url;  // 把 url 绑定在 ImageView 上
    @BindTextView(id=R.id.tv_title)
    public String desc; // 把 desc 绑定在 TextView 上
}

基本等同于官方的android:text="@{meizhi.desc}",用过databinding的话应该秒懂的。。。

绑定列表

当然,服务端返回的肯定是个妹子的列表,你手头还会有一个HomeInfo的东东。

public class HomeInfo {
    // 妹子列表区
    @BindRecyclerView(id = R.id.rv_meizhi, layout = R.layout.item_meizhi)
    private List<MeiZhi> results = new ArrayList<>();   // 把 List 绑定在 RecyclerView 上
    // banner
    @BindViewPager(id = R.id.vp_banner, layout = R.layout.item_banner)
    private List<MeiZhi> bannerInfo;    // 把 List 绑定在 ViewPager 上
}

这部分是官方没有的,相应的还提供了 @BindListView

绑定 header

像上面的配置,banner和妹子列表是分开的,不会一起滑动的。因此,提供了向RecyclerView中添加Header的注解——@BindRvHeader. 让我们把banner加进RecyclerView

public class HomeInfo {
    // 妹子列表区
    @BindRecyclerView(id = R.id.rv_meizhi, layout = R.layout.item_meizhi)
    private List<MeiZhi> results = new ArrayList<>();   // 把 List 绑定在 RecyclerView 上
    // banner
    @BindRvHeader(id = R.id.rv_meizhi, layout = R.layout.layout_banner, itemType = 0) // 增加这一行 !!!
    @BindViewPager(id = R.id.vp_banner, layout = R.layout.item_banner)
    private List<MeiZhi> bannerInfo;    // 把 List 绑定在 ViewPager 上
}

绑定点击事件

提供了@BindItemClick@BindClick

public class MainActivity extends AppCompatActivity {
    ...
    @BindItemClick(id = R.id.vp_banner)
    NoOnItemClickListener<MeiZhi> clickBanner = (view, data, pos) -> toast("click Banner: " + pos + ", "+ data.toString());
    @BindItemClick(id = R.id.rv_meizhi)
    NoOnItemClickListener<MeiZhi> clickMeiZhi = (view, data, pos) -> toast("click MeiZhi: " + pos + ", "+ data.toString());
}

更新 UI

前面只是一系列绑定关系的配置,还需要一个接口触发他们:

  • 初始化:根据 R.id.XXX 初始化相应的 View 和 Adapter,为后续更新UI做准备
mNoViewHolder = new NoViewHolder.Builder(this)
                .initView(new HomeInfo()) // 一定要提供`注解信息`的类,否则无法初始化。
                .build();
  • 更新UI: mNoViewHolder.notifyDataSetChanged(homeInfo); 自动根据 homeInfo 里提供的注解信息,找到相应的控件,并把数据刷新上去。
// 在请求的 onSuccess() 中刷新界面,本例使用了 Rxjava 和 lambda
homeInfoObservable.subscribe(homeInfo -> {
        mHomeInfo.getResults().addAll(homeInfo.getResults());           // 更新 妹子列表 info
        if(homeInfo.getResults().size() >= 6)
            mHomeInfo.setBannerInfo(homeInfo.getResults().subList(0, 6));   // 更新 bannerInfo
        mNoViewHolder.notifyDataSetChanged(mHomeInfo);  // mHomeInfo 发生变化, 通知 UI 及时刷新
}

全局配置——自定义行为

当你需要自定义的时候 (比如替换图片加载库,默认Glide)。可以这样: 如下,即把@BindTextView的行为override掉了。

    static NoViewHolder.Options mDataOptions = new NoViewHolder.DataOptions()
            .setBehaviors(new BindTextView.Behavior() {
                @Override
                public void onBind(TextView targetView, BindTextView annotation, Object value) {
                    targetView.setText("fashare 到此一游" + value);
                }
            });
    static {
        NoViewHolder.setDataOptions(mDataOptions);
    }

总结

水平有限,实现的比较粗糙。但我觉得这个思路还行,用起来简洁性也丝毫不比官方的差。觉得它不错的话,可以一起维护这个项目,向No ViewHolder的目标迈进~

一些类似实现

https://github.com/Kelin-Hong/MVVMLight

https://github.com/evant/binding-collection-adapter

感谢

https://github.com/hongyangAndroid/baseAdapter (基于它封装的)

https://github.com/mcxtzhang/all-base-adapter

完全掌握Android Data Binding

[译]关于 Android Adapter,你的实现方式可能一直都有问题