我有一个ViewModel
类定义如下:
class StockLoadTaskModel : ViewModel() {
....
....
var d: Double = 10.0
}
这与以下布局有关:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools">
<data>
<import type="android.view.View" />
<import type="it.kfi.lorikeetmobile.extras.Converter" alias="Converter"/
<variable
name="viewModel"
type="it.kfi.lorikeetmobile.stock.models.StockLoadTaskModel" />
<variable
name="view"
type="it.kfi.lorikeetmobile.stock.ui.movements.StockLoadTaskFragment
</data>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
...
<com.google.android.material.textfield.TextInputLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginStart="8dp"
android:layout_marginTop="4dp"
android:layout_marginEnd="8dp">
<com.google.android.material.textfield.TextInputEditText
android:id="@+id/et_code"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="@string/hint_et_item_code"
android:text="@={viewModel.itemCode}" />
</com.google.android.material.textfield.TextInputLayout>
<com.google.android.material.textfield.TextInputLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginStart="8dp"
android:layout_marginTop="4dp"
android:layout_marginEnd="8dp">
<com.google.android.material.textfield.TextInputEditText
android:id="@+id/et_quantity"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:inputType="numberDecimal"
android:text="@={Converter.doubleToString(d)}"
android:hint="@string/quantity" />
</com.google.android.material.textfield.TextInputLayout>
<com.google.android.material.textfield.TextInputLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginStart="8dp"
android:layout_marginTop="4dp"
android:layout_marginEnd="8dp">
<com.google.android.material.textfield.TextInputEditText
android:id="@+id/et_note"
android:lines="3"
android:scrollbars="vertical"
android:overScrollMode="ifContentScrolls"
android:gravity="top"
android:inputType="textMultiLine"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="@string/hint_et_note"
android:text="@={viewModel.selectedItem.detail.note}"/>
</com.google.android.material.textfield.TextInputLayout>
...
</LinearLayout>
我还有以下Converter
对象:
object Converter {
@JvmStatic
@InverseMethod("stringToDouble")
fun doubleToString(value: Double?): String? {
if (value == null) {
return null
}
return DecimalFormat(ClientConfiguration.currentConfig.decimalFormat).format(value)
}
@JvmStatic
fun stringToDouble(value: String?): Double? {
if (value == null) {
return null
}
val v = DecimalFormat(ClientConfiguration.currentConfig.decimalFormat).parse(value)
return v.toDouble()
}
}
如果我设置:android:text="@={Converter.doubleToString(d)}"
(双向数据绑定(,在id为et_quantity
的EditText
中,我会得到以下错误:
...error: cannot find symbol
如果我把它改成单向数据绑定,比如:android:text="@{Converter.doubleToString(d)}"
,它就可以工作了。绑定管理器似乎无法识别反向方法。
有人能帮我吗?非常感谢。
为什么会发生错误
当您像在示例android:text="@={Converter.doubleToString(d)}"
中那样定义双向数据绑定时,问题是:当用户在其中键入数据时,哪个函数/对象将接收从EditText
返回的数据?数据应该传递给Converter.doubleToString
还是Converter
的其他静态函数?可能是Converter.doubleToString(d)
的结果,还是d
的变量?
你必须精确
您希望它是d
,编译器希望它是Converter.doubleToString(d)
的结果。事实上,两者都不起作用。
另一个问题是EditText
确实使用字符。它对double、int、float、byte、short、boolean或任何其他非字符串的东西一无所知。
这意味着为了实现双向数据绑定,您的来源:
- 必须返回字符串类型的值
- 必须是可赋值的
如何解决此问题
Android架构组件向我们介绍了ObservableField
类。有现成的ObservableBoolean
、ObservableChar
、ObservableFloat
和其他一些如果打开上一句中的链接,您应该会在左侧窗格中看到所有类Observable...
没有ObservableString
,但ObservableField
接受泛型类型。因此,您可以将作为数据绑定一部分的变量定义为ObservableField<String>("defaultValueHere")
。
所以你应该拥有的是:
class StockLoadTaskModel : ViewModel() {
....
....
var d: Double = 10.0
var dataBindingVariable = ObservableField<String>(d.toString())
}
dataBindingVariable
将始终返回您绑定到的EditText
的内容。您可以获得该值并安全地转换为double。
class StockLoadTaskModel : ViewModel() {
....
....
var d: Double = 10.0
var dataBindingVariable =
object: ObservableField<String>(d.toString()) {
override fun set(value: String?) {
super.set(value)
// a value has been set
d = value.toDoubleOrNull() ?: d
}
}
}
布局声明将类似于输入字段的声明:
<com.google.android.material.textfield.TextInputLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginStart="8dp"
android:layout_marginTop="4dp"
android:layout_marginEnd="8dp">
<com.google.android.material.textfield.TextInputEditText
android:id="@+id/et_quantity"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:inputType="numberDecimal"
android:text="@={viewModel.dataBindingVariable}"
android:hint="@string/quantity" />
</com.google.android.material.textfield.TextInputLayout>
并且将不需要object Converter
。
还有另一种进行双向数据绑定的方法,我在这里不讨论,因为它已经得到了回答。给你。