主 View 模型如下所示:
class MainVM{
public ObservableField<String> title;
public ObservableField<Boolean> isFlexible;
}
主要布局如下:
<layout>
<date><variable name="item" type="MainVM"></data>
<LinearLayout>
<TextView text="@{item.title}"/>
<CustomCtrl1 vm="@{item.isFlexible}">
</LinearLayout>
</layout>
CustomCtrl
的布局看起来有点像 <layout>
<date><variable name="item" type="boolean"></data>
<LinearLayout>
...
<Switch checked="@{item}"/>
...
</LinearLayout>
</layout>
问题是
ObservableField
中的 MainVM
在传递给 CustomCtrl
时转换为 boolean 值,然后在 CustomCtrl
中更改 boolean 值不会影响 MainVM
。第一个想法是将 CustomCtrl's
View 模型从 Boolean
更改为 ObservableField<Boolean>
但由于某种原因这是不允许的。所以问题是 - 在内部控制中传递
ObservableField
的正确方法是什么。 最佳答案
最好的方法是直接使用 2 路绑定(bind)。这需要 Android Studio 2.1 及更高版本。 Android Studio 2.2 修复了自定义控件可能会遇到的通货膨胀错误。如果您使用了包含而不是自定义控件,那将是相当简单的:
<layout>
<date><variable name="item" type="MainVM"></data>
<LinearLayout>
<TextView android:text="@{item.title}"/>
<include layout="@layout/other" app:vm="@={item.isFlexible}">
</LinearLayout>
</layout>
和其他人的布局:
<layout>
<date><variable name="item" type="boolean"></data>
<LinearLayout>
...
<switch android:checked="@={item}"/>
...
</LinearLayout>
</layout>
对于自定义控件,您必须自己实现监听器。这意味着您必须拥有该属性的监听器。我认为这应该有效(我现在在平板电脑上,所以我无法验证):
@InverseBindingMethods({
InverseBindingMethod(type = CustomControl.class, attribute="vm")})
public class CustomCtrl extends View {
private CustomCtrlBinding binding;
private InverseBindingAdapter listener;
public CustomCtrl(...) {
binding = ...
binding.addOnPropertyChangedCallback(new OnPropertyChangedCallback() {
@Overriide
public void OnPropertyChanged(Observable sender, int propertyId) {
if (listener != null) {
listener.onChange();
}
}
});
}
@Bindable
public boolean getVm() { return binding.getItem(); }
public void setVm(boolean vm) {
binding.setItem(vm);
}
@BindingAdapter("vmAttrChanged")
public static void setListener(CustomCtrl view,
InverseBindingListener listener) {
view.listener = listener;
}
}
然后你有 2 路绑定(bind):
<layout>
<date><variable name="item" type="MainVM"></data>
<LinearLayout>
<TextView android:text="@{item.title}"/>
<CustomCtrl app:vm="@={item.isFlexible}">
</LinearLayout>
</layout>
和自定义控件布局:
<layout>
<date><variable name="item" type="boolean"></data>
<LinearLayout>
...
<switch android:checked="@={item}"/>
...
</LinearLayout>
</layout>
我通常会编写一个自定义控件,该控件具有自己的属性监听器,但 InverseBindingListener 在紧要关头工作。