Android文档在描述如何使用布局XML文件创建绑定类方面做得很好。但是我有几个问题。
是否有一种方法可以为通过编程实例化的自定义视图创建数据绑定类?例如,可以说我有两个自定义视图类,我想在不使用任何XML的情况下以编程方式绑定相同的视图模型对象。课程如下:
class MyViewModel {
}
class MyCustomView extends View {
}
class MyAnotherCustomView extends MyCustomView {
}
现在说我实例化mycustomview/hyanothercustomview使用:
MyCustomView customView = new MyCustomView(context);
在这种情况下,如何使用数据绑定?这是否可以使用官方的Android数据绑定框架?如果没有,还有哪些其他框架/库可用或建议实现此目标?
我的第二个问题是第一个问题的跟进。可以说,在我的第一个问题中不可能实现我想要的东西。然后,我将不得不定义一个my_custom_view.xml
文件。这看起来像这样:
<?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>
<com.example.name.MyCustomView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:text="@{user.firstName}"/>
</layout>
现在,如果我想使用mycustomView的一个子类,请保持绑定逻辑相同,我是否必须创建一个新的XML文件my_another_custom_view.xml
,仅将mycustomview替换为Myanothothercustomview来定义相同的绑定?
第一个问题的答案是"否"Android数据绑定需要XML生成绑定类。
在您的第二个问题中,您提供了一种可以正常工作的解决方案。如果您走这条路线,那么一种方法是使用ViewDataBinding Base类设置器来设置您的变量。我可以想象一种这样的方法:
public void addCustomView(LayoutInflater inflater, ViewGroup container, User user) {
ViewDataBinding binding = DataBindingUtil.inflate(inflater,
this.layoutId, container, true);
binding.setVariable(BR.user, user);
}
在这里,我假设选择了哪种自定义视图由字段layoutId
确定。每个可能的布局都必须定义一个user
类型User
的变量。
我不知道您使用的细节,但是如果您想动态选择要加载的自定义视图,则可以使用ViewStub。如果您在加载自定义视图时没有任何巨大的开销,那么您也可以使用可见性做同样的事情。
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
<import type="android.view.View"/>
<variable name="user" type="com.example.User"/>
<variable name="viewChoice" type="int"/>
</data>
<FrameLayout ...>
<!-- All of your outer layout, which may include binding
to the user variable -->
<ViewStub android:layout="@layout/myCustomView1"
app:user="@{user}"
android:visiblity="@{viewChoice == 1} ? View.VISIBLE : View.GONE"/>
<ViewStub android:layout="@layout/myCustomView2"
app:user="@{user}"
android:visiblity="@{viewChoice == 2} ? View.VISIBLE : View.GONE"/>
</FrameLayout>
</layout>