去年的Google IO 大会上,Android 团队发布了一个数据绑定框架(Data Binding Library)。依赖它可以在 Android 项目中实现 MVVM 模式,其主要功能(解决的问题)有:
1、可以直接在 layout 布局 xml 文件中绑定数据了,无需再 findViewById 然后手工设置数据了。
2、也可以用来绑定事件的操作。
3、数据可以被观察和设置,以便在需要的时候使其自动更新。
本 Sample 并没有严格遵循MVVM或者MVP,因为它既使用了ViewModel,又使用了Presenter。其总体架构图为:
1、Presenter 仍然负责业务逻辑;
2、View 的显示和刷新交给 ViewModel 来处理,ViewModel 和 View 之间的交互通过 Data Binding 来完成(其实,本例中并不是完全通过 ViewModel 来跟 View 交互,仍保留有 Presenter 对 View 的操作);
3、View 的 XML 文件需要实现 Data Binding。
本文将通过待办事项列表页(对象项目中的tasks包)来进行分析。
准备事项
在 build.gradle 文件中添加:
dataBinding {
enabled = true
}
数据绑定
1、XML 文件定义了<data>标签:
<data>
<import type="android.view.View" />
<variable
name="tasks"
type="com.example.android.architecture.blueprints.todoapp.tasks.TasksViewModel" />
<variable
name="actionHandler"
type="com.example.android.architecture.blueprints.todoapp.tasks.TasksContract.Presenter" />
</data>
2、XML 文件使用数据绑定:
<TextView
android:id="@+id/filteringLabel"
……
android:text="@{tasks.currentFilteringLabel}" />
<TextView
android:id="@+id/noTasksMain"
……
android:text="@{tasks.noTasksLabel}" />
<LinearLayout
android:id="@+id/noTasks"
……
android:visibility="@{tasks.notEmpty ? View.GONE : View.VISIBLE}" />
……
3、向 ViewModel 中设置数据:
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
TasksFragBinding tasksFragBinding = TasksFragBinding.inflate(inflater, container, false);
// 在<data>标签中定义的<variable>都会有其 set 方法
tasksFragBinding.setTasks(mTasksViewModel);
tasksFragBinding.setActionHandler(mPresenter);
4、通知 ViewModel 来进行刷新:
public void showTasks(List<Task> tasks) {
mListAdapter.replaceData(tasks);
mTasksViewModel.setTaskListSize(tasks.size());
}
// TasksViewModel 继承 BaseObservable,调用其notifyPropertyChanged()方法
public void setTaskListSize(int taskListSize) {
mTaskListSize = taskListSize;
notifyPropertyChanged(BR.noTaskIconRes);
notifyPropertyChanged(BR.noTasksLabel);
notifyPropertyChanged(BR.currentFilteringLabel);
notifyPropertyChanged(BR.notEmpty);
notifyPropertyChanged(BR.tasksAddViewVisible);
}
5、当然,要实现 XML 中定义的方法:
// TasksViewModel 继承 BaseObservable,使用 @Bindable 来注解方法
@Bindable
public String getCurrentFilteringLabel() {
switch (mPresenter.getFiltering()) {
case ALL_TASKS:
return mContext.getResources().getString(R.string.label_all);
case ACTIVE_TASKS:
return mContext.getResources().getString(R.string.label_active);
case COMPLETED_TASKS:
return mContext.getResources().getString(R.string.label_completed);
}
return null;
}
@Bindable
public String getNoTasksLabel() {
switch (mPresenter.getFiltering()) {
case ALL_TASKS:
return mContext.getResources().getString(R.string.no_tasks_all);
case ACTIVE_TASKS:
return mContext.getResources().getString(R.string.no_tasks_active);
case COMPLETED_TASKS:
return mContext.getResources().getString(R.string.no_tasks_completed);
}
return null;
}
@Bindable
public boolean isNotEmpty() {
return mTaskListSize > 0;
}
……
事件绑定
数据绑定就不需要调用findViewById(),而事件绑定则可以帮助减少setOnClickListener()等这些事件注册方法。比如,定义 TextView 标签的点击事件,调用了 actionHandler 对象的 addNewTask() 方法。
<TextView
android:id="@+id/noTasksAdd"
……
android:onClick="@{() -> actionHandler.addNewTask()}"
android:visibility="@{tasks.tasksAddViewVisible ? View.VISIBLE : View.GONE}" />
而该方法则在 TasksPresenter 中进行了定义:
@Override
public void addNewTask() {
mTasksView.showAddTask();
}
通过以上分析,可以看出跟mvp最大的区别是:通过引用 Data Binding 库,简化了 View 中UI的控制逻辑,ViewModel 担任了该项职责。