Fork me on GitHub

重构——引入Databinding

添加DataBinding支持

在Android Studio上使用,需要在module级别的build.gradle上添加对DataBinding的支持

1
2
3
4
5
6
android {
....
dataBinding {
enabled = true
}
}

布局文件

DataBinding的layout files和普通的非DataBinding布局文件是有一些区别的,下面是一个基础的使用了DataBinding的布局文件

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">

<data>
<variable
name="user"
type="com.example.User" />
</data>

<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">

<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{user.firstName}" />

<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{user.lastName}" />
</LinearLayout>
</layout>

事件处理
点击事件onClick

android:onClick="@{()->viewModel.logout()}

1
2
3
4
5
6
7
8
9
10
11
<Button
android:id="@+id/log_out"
android:layout_width="268dp"
android:layout_height="32dp"
android:layout_centerHorizontal="true"
android:layout_marginTop="40dp"
android:onClick="@{()->viewModel.logout()}"
android:background="@drawable/round_corner_logout"
android:text="@string/log_out"
android:textColor="#ffffff"
android:textSize="14sp" />

ViewPager
setCurrentItem(int item)
  • 在ViewModel创建public ObservableInt mCurrentIndex = new ObservableInt();
  • 在xml关联app:currentItem="@{viewModel.mCurrentIndex}"
  • 通过设置mCurrentIndex的值,即可改变ViewPager的显示位置
    1
    2
    3
    public void setCurrentItem(int currentIndex) {
    mCurrentIndex.set(currentIndex);
    }
setAdapter(PagerAdapter adapter)
  • 在ViewModel创建public FragmentPagerAdapter mAdapter;
  • setViewModel前,需要初始化mAdapter
  • 在xml关联app:adapter="@{viewModel.mAdapter}"
setOnPageChangeListener(OnPageChangeListener listener)
  • 在ViewModel实现接口implements OnPageChangeListener
  • 重写OnPageChangeListener的方法

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    public void onPageScrollStateChanged(int state) {

    }

    public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {

    }

    public void onPageSelected(int position) {
    mCurrentIndex.set(position);
    }
  • 在xml关联app:onPageChangeListener="@{viewModel}"

View的可见状态控制
  • 引入View <import type="android.view.View" />
  • 通过?:来控制显示
    1
    android:visibility="@{viewModel.isVisible?View.VISIBLE:View.GONE}"
颜色控制
  • 通过?:来控制显示
    1
    android:textColor="@{viewModel.isBlue?@color/blue:@color/gray}"
Include布局

Include这个布局标签在DataBinding布局里面使用有点特殊, 因为它需要我们传递绑定的方法和数据对象。比如我们有以下的include布局:

1
2
3
4
5
6
7
8
9
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">
<data>
<import type="android.view.View" />
<variable
name="viewModel"
type="com.zhhx.fragment.IbmsMainFragment" />
</data>
</layout>

那么我们在另一个xml引用时,就需要传递这个include需要绑定的方法和数据:

1
2
3
<include
layout="@layout/include_sub_system_airconditioner"
app:viewModel="@{viewModel}" />

ObservableField

使用ObservableField可以引用复杂的数据类型,如

1
public ObservableField<SubsysBean> subsysBean = new ObservableField<>();

在数据更新时,直接set即可,不需要对SubsysBean实现Bindable注解。

1
subsysBean.set(info);

Databinding自动判空

使用时不需要判空,如:

1
2
info.Video.getData().setObjList(null);
subsysBean.set(info);

1
2
3
<TextView
...
android:text="@{String.valueOf(viewModel.subsysBean.Video.data.ObjList.get(2).ValidCount)}" />

查看源码可知,虽然进行了判空,但是对数组越界没有进行处理,所以要 谨慎防止数组越界

1
2
3
4
5
6
if (viewModelSubsysBeanVideoDataGetObjList != null) {
// read viewModel.subsysBean.get().Video.data.getObjList().get(1)
viewModelSubsysBeanVideoDataGetObjListGetInt1 = viewModelSubsysBeanVideoDataGetObjList.get(1);
// read viewModel.subsysBean.get().Video.data.getObjList().get(2)
viewModelSubsysBeanVideoDataGetObjListGetInt2 = viewModelSubsysBeanVideoDataGetObjList.get(2);
}

TextView的int类型数值的显示

直接使用int类型会报错,可以调用String.valueOf()方法

<TextView
   ...
   android:text="@{String.valueOf(viewModel.subsysBean.Video.data.ObjList.get(2).ValidCount)}" />