简体   繁体   中英

How to make EditText observe a ViewModel's LiveData and forward user input to the ViewModel without using data binding

I'm trying to come up with a way to have an EditText update the data of a ViewModel and simultaneously observe that data for any changes (eg changes brought about by manipulating the DB). Is there a way to do this without using the data binding library?

The main problem I'm facing while simply using MutableLiveData is the following:

when the user enters text in the EditText, a TextWatcher pokes the ViewModel to update its data, which in turn will set the new text to the MutableLiveData object. Because the EditText is observing the LiveData, the onChange is triggered and sets the text of the EditText accordingly, which in turn will trigger the TextWatcher again creating an infinite loop.

I also ran into this problem since I don't like the databinding library. I did as @kAliert said, but in my ViewModel to keep the logic there. I just added a simple catch on the function that receives my text changes events in the ViewModel . It works well.

fun editTextChanged(newText: String) {
    if (newText == textLiveData.value) {
        return
    }
}

May be a little late, but hope useful for somebody. I had this problem and what I did was consider a string member for the viewmodel class to maintain last value of editText. Everytime it would be to return livedata, It checks it first is there difference between new edittext input(getData() parameter) and its past value (saved in myVariable):

  public class MyViewModel extends ViewModel {
  private MutableLiveData<DataType> data;
  private String myVariable; //maintainer variable

  public LiveData<DataType> getData(String input){
    if(data == null || !myVariable.equals(input)){
      myVariable = input;
      data = new MutableLiveData<>();
      loadData(data, input);
    }
    return data;
  }

and in your owner(eg. Activity) observe data changes:

myViewModel = ViewModelProviders.of(this).get(MyViewModel.class);    
editText.addTextChangedListener(new TextWatcher() {
  @Override
  public void beforeTextChanged(CharSequence s, int start, int count, int after) {    
  }

  @Override
  public void onTextChanged(CharSequence s, int start, int before, int count) {
     myViewModel.getData(s).observe(this, new Observer<DataType>() {
      @Override
      public void onChanged(<DataType> data) {
        //do everything you want like update UI by data
      }
    });
  }

  @Override
  public void afterTextChanged(Editable s) {         
  }
});

我认为您也可以使用 LiveData.distinctUntilChanged(),它只会在设置新值时发出数据。

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM