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 и свойством user модели 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
}

Что я делаю не так? Нужно ли мне добавлять конкретного наблюдателя?

Спасибо


person Faustino Gagneten    schedule 19.10.2018    source источник


Ответы (3)


Благодаря @communistWatermelon я смог решить часть проблемы, но этого было недостаточно. Вот полное решение:

С одной стороны, мы должны инициализировать свойство value MutableLiveData, как сказал @communistWatermelon:

var loginModel: MutableLiveData<LoginModel> = MutableLiveData()

init {
    loginModel.value = LoginModel()
}

С другой стороны, нам нужно правильно настроить привязку в макете.

В случае @ {variable.field} связыватель сгенерирует фрагмент кода для получения значения путем вызова variable.getField (). Обратите внимание, что связыватель неявно префикс слова «получить», если он не указан. Итак, @ {variable.getField ()} - явный допустимый параметр.

В случае @ = {variable.field} связыватель создаст тот же код, что и раньше, но с дополнительным кодом для получения значения из представления и установки его обратно для вашего объекта.

Прочитав это, нам нужно изменить способ привязки

android:text="@{viewModel.loginModel.password}"
android:text="@{viewModel.loginModel.user}"

to

android:text="@={viewModel.loginModel.password}"
android:text="@={viewModel.loginModel.user}"

ИЗМЕНИТЬ

Для тех, кто не решил, используя приведенное выше объяснение, попробуйте зарегистрировать lifecycleOwner в своем фрагменте с помощью следующей строки:

binding.setLifecycleOwner(this)

Хорошее кодирование!

person Faustino Gagneten    schedule 20.10.2018
comment
спасибо за ответ, мне просто не хватало loginModel.value = LoginModel() этой строки :( - person Priyanka; 09.07.2021

Это случилось и со мной, за исключением того, что произошло: я использовал MutableLiveData в своей ViewModel без регистрации моего фрагмента / активности как LifeCycleOwner в ViewDataBinding.

Вопрос не столкнется с этой проблемой, потому что

    binding.setLifecycleOwner(this)

Это может не иметь отношения к вопросу, но если кто-то еще приземлится здесь, как я, дважды проверьте, что вы выполняете setLifecycleOwner в своей привязке.

person Jacques.S    schedule 13.05.2019

MutableLiveData<LoginModel>() не инициализирует свойство value объекта MutableLiveData. Скорее всего, вам потребуется установить значение в вызове init, например:

init {
    loginModel = MutableLiveData<LoginModel>()
    loginModel.value = LoginModel()
}

Поскольку вы не устанавливаете значение, это вызовет исключение NullPointerException в

return repository.login(loginModel.value!!)
person communistWatermelon    schedule 19.10.2018
comment
Не работает. Теперь значение не пусто, но в нем нет данных из написанного мной Edittext. РЕДАКТИРОВАТЬ: я смог найти ошибку благодаря вам - person Faustino Gagneten; 22.10.2018