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