web-dev-qa-db-ja.com

MutableLiveDataがUIで更新されない

PDATE:別のフラグメントに移動してこれに戻ると、TextViewが更新されます...

UIでMutableLiveDataを取得して、setValue()またはpostValue()を使用して新しい値に更新できません。 MutableLiveDataをObservableDataに変更することでこれを機能させることができますが、ポイントはObservableDataではなくLiveDataを使用することです。

私は問題なく他のビューで動作している同様のものがいます。ここで何が起こっているのかわからない...考えられる唯一のことは、カメラのアクティビティ(意図による)とこのフラグメントの間を行き来していることです。アクティビティの結果では、フラグメントからViewModelにデータを設定しています。

XMLとViewModelの間に双方向のデータバインディングがあるため、フラグメントの値にオブザーバーを設定する必要があるとは思いません。

my_fragment.XML

<?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"
    xmlns:tools="http://schemas.Android.com/tools">

    <data>
        <variable
            name="myViewModel"
            type="com.example.myapplication.MyViewModel" />
    </data>

    <LinearLayout
        Android:layout_width="fill_parent"
        Android:layout_height="wrap_content"
        Android:orientation="vertical">

            <TextView
            Android:id="@+id/textView"
            Android:layout_width="wrap_content"
            Android:layout_height="wrap_content"
            Android:text="@={myViewModel.photosCount}"
            tools:text="1" />

           <Button
               Android:id="@+id/btnTakePhoto"
               Android:layout_width="wrap_content"
               Android:layout_height="wrap_content"
               Android:text="Take Picture"
               Android:onClick="@{myViewModel::navClicked}"/>

    </LinearLayout>

</layout>

MyViewModel.Java

private MutableLiveData<String> photosCount = new MutableLiveData<>();
private MutableLiveData<Boolean> takePhoto = new MutableLiveData<>();

public MyViewModel() {
    photosCount.setValue("0"); // This correctly sets the value to "0" in UI
    takePhoto.setValue(true);
}

public MutableLiveData<String> getPhotosCount() {
    return photosCount;
}

public MutableLiveData<Boolean> getTakePhoto() {
    return takePhoto;
}

public void storeImage() {
    ....
    Log.d(TAG, "Updating UI count to: " + String.valueOf(count)); // this shows the correct value that should be updated in the UI
    photosCount.setValue(String.valueOf(count));
    Log.d(TAG, "Updated value: " + photosCount.getValue()); // this shows the correct updated value
    ...
}

public void navClicked(@NonNull View view) {
    if (view.getId() == R.id.btnTakePhoto) {
        takePhoto.setValue(true);
    }
}

さて、LiveDataは値の変更時にUIを更新することを保証していないので、これはバインディングディスパッチの問題かもしれないと思いました。これで問題は解決しました...

MyFragment.Java

private MyViewModel myViewModel;
MyFragmentBinding binding;

@Nullable
@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
    super.onCreateView(inflater, container, savedInstanceState);

    myViewModel = new ViewModelProvider(this).get(MyViewModel.class);

    binding = DataBindingUtil.inflate(inflater, R.layout.my_fragment, container, false);
    binding.setMyViewModel(myViewModel);

    return binding.getRoot();
}

@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
    viewModel.getTakePhoto().observe(getViewLifecycleOwner(), takePhoto -> {
        if (takePhoto) {
            Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

            if (intent.resolveActivity(getActivity().getPackageManager()) != null) {
                File photo = viewModel.createImageFile();
                if (photo != null) {
                    Uri photoURI = FileProvider.getUriForFile(getActivity(),"com.example.fileprovider", photo);
                    intent.putExtra(MediaStore.EXTRA_OUTPUT, photoURI);
                    this.startActivityForResult(intent, Constants.RequestCodes.MY_REQUEST_IMAGE_CAPTURE);
                }
            }
        }
    });
}

@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    if (resultCode == Activity.RESULT_OK) {
        if (requestCode == Constants.RequestCodes.MY_REQUEST_IMAGE_CAPTURE) {
            viewModel.storeImage();
        }
    }
}

この場合も、photosCountをMutableLiveDataからObservableDataに切り替えると、問題は修正されますが、これはLiveDataではありません。

4
DevinM

公式ドキュメントから:

/**
 * Sets the {@link LifecycleOwner} that should be used for observing changes of
 * LiveData in this binding. If a {@link LiveData} is in one of the binding expressions
 * and no LifecycleOwner is set, the LiveData will not be observed and updates to it
 * will not be propagated to the UI.
 *
 * @param lifecycleOwner The LifecycleOwner that should be used for observing changes of
 *                       LiveData in this binding.
 */z

必要な手順:

  1. ビューのバインディングを適用した後は、必ずactivityOrFragmentBinding.lifecycleOwner = thisを設定してください。 activityOrFragmentBindingバインディング変数名に置き換えます。
  2. 常にViewModelをデータバインディングに設定します。
0