主 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 在紧要关头工作。

10-08 14:46