双向数据绑定未更新 UI

Two-Way Databinding not updating UI

我想了解为什么设置一个值不会自动刷新 UI。如果我调用 binding.setItem,UI 会刷新。

我知道绑定对象包含更新的值,但是 UI 在设置 item.name 和 item.checked 后没有被刷新。

我做错了什么?我是否需要每次都调用 setItem 来刷新 UI?我认为这是不必要的,因为 UI 会在设置值后自动更新。

Item.java:

public class Item  {
    public String name;
    public Boolean checked;
}

MainActivity.java:

public class MainActivity extends AppCompatActivity {

    public Item item;
    ActivityMainBinding binding;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        item = new Item();
        item.checked = true;
        item.name = "a";

        binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
        binding.setItem(item);
    }

    public void button_onClick(View v) {
        item.checked = !item.checked;
        item.name = item.name  + "a";
        ///binding.setItem(item); --> ??? this updates UI, do I need to call this to refresh UI everytime???
    }
}

activity_main.xml:

<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">
    <data>
        <variable
            name="item"
            type="com.example.abc.twowaydatabinding.Item" />
    </data>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="vertical">

        <TextView
            android:id="@+id/tv_title"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="@={item.name}"
            android:textSize="20sp" />


        <Switch
            android:id="@+id/switch_test"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:checked="@={item.checked}" />

        <Button
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="change"
            android:onClick="button_onClick"/>

    </LinearLayout>
</layout>

build.gradle:

android {
...
    dataBinding{
        enabled=true
    }

}

根据开发者指南:

Any plain old Java object (POJO) may be used for data binding, but modifying a POJO will not cause the UI to update.

您可以按照开发人员指南的 "Observable Objects" 部分修改您的对象以更新您的 UI。

https://developer.android.com/topic/libraries/data-binding/index.html#data_objects

如前所述,使用 ObservableFields

public ObservableField<Boolean> checked = new ObservableField<>();

public ObservableField<String> name = new ObservableField<>();

并在 MainActivity 中进行更改

    item = new Item();
    item.checked.set(true);
    item.name.set("a");

    binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
    binding.setItem(item);
}

public void button_onClick(View v) {

    item.checked.set(!(item.checked.get()));
    item.name.set(item.name.get()  + "a");

}

顺便说一下,您可以查看有关数据绑定的博客文章系列 https://medium.com/google-developers/android-data-binding-2-way-your-way-ccac20f6313 and this https://medium.com/@fabioCollini/android-data-binding-f9f9d3afc761

为了更新 UI,字段需要是可观察的。只是更新了我的更改:

import android.databinding.BaseObservable;
import android.databinding.Bindable;

public class Item extends BaseObservable {
    private String name;
    private Boolean checked;
    @Bindable
    public String getName() {
        return this.name;
    }
    @Bindable
    public Boolean getChecked() {
        return this.checked;
    }
    public void setName(String name) {
        this.name = name;
        notifyPropertyChanged(BR.name);
    }
    public void setChecked(Boolean checked) {
        this.checked = checked;
        notifyPropertyChanged(BR.checked);
    }
}