MVVM - 自定义模型的 MutableLiveData 未更新为具有数据绑定的 ViewModel,并且始终为 null
MVVM - MutableLiveData of Custom Model not been updating into ViewModel with databinding and always is null
总结:
当我尝试向存储库发送自定义模型时,MutableLiveData 为空。我认为是因为 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 和模型 UserModel 的 属性 user 之间的关系,并且我实现了 etPassword 和 属性 password
之间的关系
当用户点击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
}
我做错了什么?我需要添加一个特定的观察者吗?
谢谢
MutableLiveData<LoginModel>()
不会初始化 MutableLiveData
对象的 value
属性。您可能需要在 init
调用中设置一个值,如下所示:
init {
loginModel = MutableLiveData<LoginModel>()
loginModel.value = LoginModel()
}
因为您没有设置值,所以这样做会导致
处的 `NullPointerException
return repository.login(loginModel.value!!)
感谢@communistWatermelon,我解决了一部分问题,但这还不够。这是完整的解决方案:
一方面,我们必须初始化 MutableLiveData 的值 属性,正如@communistWatermelon 所说:
var loginModel: MutableLiveData<LoginModel> = MutableLiveData()
init {
loginModel.value = LoginModel()
}
另一方面,我们需要在布局中正确设置绑定
In the case of @{variable.field} the binder will generate a piece of code to get the value by calling variable.getField(). Note that the binder implicitly prefix the “get” word if it is not provided. So @{variable.getField()} is an explicit valid option.
In the case of @={variable.field} the binder will create the same code as before, but with additional code for getting the value from the View and setting it back to your object
看完之后,我们需要改变绑定的方式
android:text="@{viewModel.loginModel.password}"
android:text="@{viewModel.loginModel.user}"
至
android:text="@={viewModel.loginModel.password}"
android:text="@={viewModel.loginModel.user}"
编辑
对于那些没有使用上述解释解决问题的人,请尝试使用以下行在您的片段中注册 lifecycleOwner:
binding.setLifecycleOwner(this)
写得不错!
这也发生在我身上,除了发生的事情是我在我的 ViewModel 上使用了 MutableLiveData 而没有将我的 fragment/activity 注册为 ViewDataBinding 的 LifeCycleOwner。
问题不会遇到这个问题,因为
binding.setLifecycleOwner(this)
这可能与问题无关,但如果其他人像我一样登陆这里,请仔细检查您是否对绑定执行了 setLifecycleOwner。
总结:
当我尝试向存储库发送自定义模型时,MutableLiveData 为空。我认为是因为 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 和模型 UserModel 的 属性 user 之间的关系,并且我实现了 etPassword 和 属性 password
之间的关系当用户点击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
}
我做错了什么?我需要添加一个特定的观察者吗?
谢谢
MutableLiveData<LoginModel>()
不会初始化 MutableLiveData
对象的 value
属性。您可能需要在 init
调用中设置一个值,如下所示:
init {
loginModel = MutableLiveData<LoginModel>()
loginModel.value = LoginModel()
}
因为您没有设置值,所以这样做会导致
处的 `NullPointerExceptionreturn repository.login(loginModel.value!!)
感谢@communistWatermelon,我解决了一部分问题,但这还不够。这是完整的解决方案:
一方面,我们必须初始化 MutableLiveData 的值 属性,正如@communistWatermelon 所说:
var loginModel: MutableLiveData<LoginModel> = MutableLiveData()
init {
loginModel.value = LoginModel()
}
另一方面,我们需要在布局中正确设置绑定
In the case of @{variable.field} the binder will generate a piece of code to get the value by calling variable.getField(). Note that the binder implicitly prefix the “get” word if it is not provided. So @{variable.getField()} is an explicit valid option.
In the case of @={variable.field} the binder will create the same code as before, but with additional code for getting the value from the View and setting it back to your object
看完之后,我们需要改变绑定的方式
android:text="@{viewModel.loginModel.password}"
android:text="@{viewModel.loginModel.user}"
至
android:text="@={viewModel.loginModel.password}"
android:text="@={viewModel.loginModel.user}"
编辑
对于那些没有使用上述解释解决问题的人,请尝试使用以下行在您的片段中注册 lifecycleOwner:
binding.setLifecycleOwner(this)
写得不错!
这也发生在我身上,除了发生的事情是我在我的 ViewModel 上使用了 MutableLiveData 而没有将我的 fragment/activity 注册为 ViewDataBinding 的 LifeCycleOwner。
问题不会遇到这个问题,因为
binding.setLifecycleOwner(this)
这可能与问题无关,但如果其他人像我一样登陆这里,请仔细检查您是否对绑定执行了 setLifecycleOwner。