MVVM - MutableLiveData пользовательской модели не обновляется в ViewModel с привязкой к данным и всегда имеет значение null
Резюме:
Когда я пытаюсь отправить в хранилище пользовательскую модель, MutableLiveData имеет значение null. Я думаю, что это из-за наблюдателя MutableLiveData.
Пожалуйста, прочитайте до конца.
ViewModel
class LoginViewModel @Inject constructor(
private val repository: MyRepository) : ViewModel() {
var loginModel: MutableLiveData<LoginModel>
init {
loginModel = MutableLiveData<LoginModel>()
}
fun loadUser(): LiveData<Response<Custom<Token>>> {
return repository.login(loginModel.value!!)
}
}
Как вы можете видеть здесь, у меня есть MutableLiveData
Моя LoginModel что-то вроде этого
data class LoginModel(var user : String, var password : String)
Фрагмент был определен с использованием привязки данных и привязки с ViewModel выше.
login_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"
tools:context="com.app.example.view.BaseActivity">
<data>
<variable
name="viewModel"
type="com.app.example.view.login.LoginViewModel" />
</data>
<android.support.constraint.ConstraintLayout
android:id="@+id/activityMain"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@drawable/bg_login">
<android.support.v7.widget.CardView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_marginBottom="80dp"
android:layout_marginLeft="16dp"
android:layout_marginRight="16dp"
android:layout_marginTop="80dp"
app:cardCornerRadius="7dp"
app:cardElevation="22dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center|top"
android:layout_marginTop="60dp"
android:text="@string/bienvenido_text"
android:textAllCaps="true"
android:textSize="20sp" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_margin="20dp"
android:orientation="vertical">
<android.support.design.widget.TextInputLayout
android:layout_width="match_parent"
android:layout_height="wrap_content">
<EditText
android:id="@+id/etEmail"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginLeft="20dp"
android:layout_marginRight="20dp"
android:cursorVisible="true"
android:text="@{viewModel.loginModel.user}"
android:gravity="center|start|bottom"
android:hint="@string/email_text"
android:inputType="textEmailAddress"
android:maxLength="50"
android:paddingBottom="10dp"
android:textSize="18sp" />
</android.support.design.widget.TextInputLayout>
<android.support.design.widget.TextInputLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
app:passwordToggleEnabled="true">
<android.support.design.widget.TextInputEditText
android:id="@+id/etPassword"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginLeft="20dp"
android:layout_marginRight="20dp"
android:layout_marginTop="30dp"
android:cursorVisible="true"
android:gravity="center|start|bottom"
android:hint="@string/contrasenia_text"
android:text="@{viewModel.loginModel.password}"
android:inputType="textPassword"
android:maxLength="50"
android:paddingBottom="10dp"
android:textSize="18sp" />
</android.support.design.widget.TextInputLayout>
<Button
android:id="@+id/btnServerLogin"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_margin="15dp"
android:padding="10dp"
android:text="@string/ingresar_text"
android:textSize="18sp" />
</LinearLayout>
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="bottom|center"
android:layout_marginBottom="40dp"
android:orientation="horizontal">
</LinearLayout>
</android.support.v7.widget.CardView>
</android.support.constraint.ConstraintLayout>
</layout>
Используя привязку данных, я реализовал связь между etEmail и пользователем свойства Model UserModel, а также реализовал связь между etPassword и паролем свойства.
когда пользователь нажимает на btnServerLogin, LoginFragment выполнит следующий код
binding.btnServerLogin.setOnClickListener {
loginViewModel!!.loadUser().observe(this, Observer<Response<Custom<Token>>> { this.handleResponse(it) })
}
И здесь проблема.
kotlin.KotlinNullPointerException
Там причина из-за loginModel.value! нулевой
fun loadUser(): LiveData<Response<Custom<Token>>> {
return repository.login(loginModel.value!!)
}
Значение MutableLiveData всегда равно нулю. Я думал, что это изменится в то же время, когда вы набираете свой EditText сообщения электронной почты или пароля.
Вот код LoginFragment OnActivityCreated, в котором я инициализировал ViewModel
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
setHasOptionsMenu(true)
DeviceUtils.setTranslucentStatusBar(activity!!.window, R.color.colorPrimaryDark)
loginViewModel = ViewModelProviders.of(this, viewModelFactory)
.get(LoginViewModel::class.java)
binding.setLifecycleOwner(this)
binding.viewModel = loginViewModel
binding.btnServerLogin.setOnClickListener {
mPostsViewModel!!.loadUser().observe(this, Observer<Response<RespuestaModel<JwtTokenModel>>> { this.handleResponse(it) })
}
return
}
Что я делаю неправильно? Должен ли я добавить конкретного Обозревателя?
Спасибо
2 ответа
Благодаря @communistWatermelon я смог решить часть проблемы, но этого было недостаточно. Вот полное решение:
С одной стороны, мы должны инициализировать свойство value MutableLiveData как @communistWatermelon:
var loginModel: MutableLiveData<LoginModel> = MutableLiveData()
init {
loginModel.value = LoginModel()
}
С другой стороны, нам нужно правильно установить привязку в макете
В случае @{variable.field} связыватель сгенерирует кусок кода, чтобы получить значение, вызвав variable.getField(). Обратите внимание, что связыватель неявно префиксует слово "get", если оно не указано. Так что @{variable.getField()} является явным допустимым параметром.
В случае @={variable.field} связыватель создаст тот же код, что и раньше, но с дополнительным кодом для получения значения из View и установки его обратно в ваш объект
Прочитав это, нам нужно изменить способ привязки
android:text="@{viewModel.loginModel.password}"
android:text="@{viewModel.loginModel.user}"
в
android:text="@={viewModel.loginModel.password}"
android:text="@={viewModel.loginModel.user}"
Хорошее кодирование!
Это случилось и со мной, за исключением того, что произошло: я использовал MutableLiveData в своей ViewModel без регистрации моего фрагмента / активности как LifeCycleOwner в ViewDataBinding.
Вопрос не столкнется с этой проблемой, потому что
binding.setLifecycleOwner(this)
Это может не иметь отношения к вопросу, но если кто-то еще приземлится здесь, как я, дважды проверьте, что вы выполняете setLifecycleOwner в своей привязке.
MutableLiveData<LoginModel>()
не инициализирует value
собственность MutableLiveData
объект. Скорее всего, вам нужно будет установить значение в init
звоните, вот так:
init {
loginModel = MutableLiveData<LoginModel>()
loginModel.value = LoginModel()
}
Поскольку вы не устанавливаете значение, это приведет к возникновению исключения NullPointerException в
return repository.login(loginModel.value!!)