Android中的Data Binding初探 (三)
本文接 《Android中的Data Binding初探 (二)》
MVVM中的Model
我们可以用任何POJO作为data binding的Model,但是直接修改POJO对象,不能直接更新UI。
Android的Data Binding模块给提供了通知机制,有3种类型,分别对应于类(Observable),字段(ObservableField),集合类型(Observable Collections)。
把这些observable对象绑定到View后,当observable对象更新后,UI会自动更新。
Observable Objects用法
我们需要把POJO继承自BaseObservable,才能获得通知UI的能力
private static class User extends BaseObservable { private String firstName; private String lastName; @Bindable public String getFirstName() { return this.firstName; } @Bindable public String getFirstName() { return this.lastName; } public void setFirstName(String firstName) { this.firstName = firstName; notifyPropertyChanged(BR.firstName); } public void setLastName(String lastName) { this.lastName = lastName; notifyPropertyChanged(BR.lastName); } }
Bindable标签在编译时会自动生成BR类,但Model中的数据发生改变时,我们在Set方法中调用notifyPropertyChanged通知UI更新。
ObservableFields用法
创建支持Observable的POJO类还是有点麻烦,
ObservableFields可以简化我们的POJO对象:
private static class User extends BaseObservable { public final ObservableField<String> firstName = new ObservableField<>(); public final ObservableField<String> lastName = new ObservableField<>(); public final ObservableInt age = new ObservableInt(); }
通过以下方式访问和修改字段值
user.firstName.set("Google"); int age = user.age.get();
对应基础数据类型有ObservableInt、ObservableFloat、ObservableBoolean等可以使用。
Observable Collections用法
DataBinding中提供了一些支持通知机制的集合类型,比如ObservableArrayList,ObservableArrayMap。
ObservableArrayMap的使用跟Map一样
ObservableArrayMap<String, Object> user = new ObservableArrayMap<>(); user.put("firstName", "Google"); user.put("lastName", "Inc."); user.put("age", 17);
在Layout中使用ObservableArrayMap中的数据
<data> <import type="android.databinding.ObservableMap"/> <variable name="user" type="ObservableMap<String, Object>"/> </data> … <TextView android:text='@{user["lastName"]}' android:layout_width="wrap_content" android:layout_height="wrap_content"/> <TextView android:text='@{String.valueOf(1 + (Integer)user["age"])}' android:layout_width="wrap_content" android:layout_height="wrap_content"/>
MVVM中的ViewModel
Android中的ViewModel是自动生成的Binding类(继承自android.databinding.ViewDataBinding)
创建Binding对象
我们一般使用Binding对象的静态方法创建Binding对象:
MyLayoutBinding binding = MyLayoutBinding.inflate(layoutInflater); MyLayoutBinding binding = MyLayoutBinding.inflate(LayoutInflater, viewGroup, false);
有时候我们需要使用DataBindingUtil创建Binding对象
ViewDataBinding binding = DataBindingUtil.inflate(LayoutInflater, layoutId, parent, attachToParent); ViewDataBinding binding = DataBindingUtil.bindTo(viewRoot, layoutId);
设置View的id
使用DataBinding以后,我们一般不需要设置View的id,但是我们有时候也会需要,
设置id后,ViewDataBinding类会自动生成对应的字段,比如:
<layout xmlns:android="http://schemas.android.com/apk/res/android"> <data> <variable name="user" type="com.example.User"/> </data> <LinearLayout android:orientation="vertical" android:layout_width="match_parent" android:layout_height="match_parent"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@{user.firstName}" android:id="@+id/firstName"/> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@{user.lastName}" android:id="@+id/lastName"/> </LinearLayout> </layout>
对应的id会自动生成:
public final TextView firstName; public final TextView lastName;
Variables
在Layout中data区域定义的变量,或自动在Binding类中生成get/set方法
<data> <import type="android.graphics.drawable.Drawable"/> <variable name="user" type="com.example.User"/> <variable name="image" type="Drawable"/> <variable name="note" type="String"/> </data>
生成的方法如下:
public abstract com.example.User getUser(); public abstract void setUser(com.example.User user); public abstract Drawable getImage(); public abstract void setImage(Drawable image); public abstract String getNote(); public abstract void setNote(String note);
Dynamic Variables
有时候,我们无法知道确切的binding类,比如RecyclerView Adapter可以使用任意的layout,
所以我们的binding类需要动态生成。
我们需要在onBindViewHolder方法中给变量赋值,比如我们的layout中声明了一个item变量,
我们通过BindingHolder的getBinding返回一个binding对象,调用setVariable方法给item变量赋值
public void onBindViewHolder(BindingHolder holder, int position) { final T item = mItems.get(position); holder.getBinding().setVariable(BR.item, item); holder.getBinding().executePendingBindings(); }
binding对象需要在onCreateViewHolder中创建
ViewDataBinding binding = DataBindingUtil.inflate(LayoutInflater.from(viewGroup.getContext()), R.layout.list_item,viewGroup,false); BindingHolder holder = new BindingHolder(binding.getRoot()); holder.setBinding(binding);
本文参考谷歌官方的 Data Binding Guide </div>