我有一个 ViewModel
类定义如下:
class StockLoadTaskModel : ViewModel() {
....
....
var d: Double = 10.0
}
这绑定(bind)到以下布局:<?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)}"
(双向数据绑定(bind)),在 EditText
编号为 et_quantity
我收到以下错误:...error: cannot find symbol
如果我将其更改为单向数据绑定(bind),例如:android:text="@{Converter.doubleToString(d)}"
, 有用。看起来绑定(bind)管理器无法识别逆向方法。有谁能够帮助我?谢谢你。
最佳答案
为什么会发生错误?
当您像在示例中那样定义双向数据绑定(bind)时 android:text="@={Converter.doubleToString(d)}"
问题是:什么函数/对象将接收您返回的数据 来自 EditText
当用户输入数据时?是否应将数据传递给 Converter.doubleToString
或者 Converter
的其他一些静态函数?也许是 Converter.doubleToString(d)
的结果或至d
多变的?
你必须准确。
你期望它是 d
,编译器期望它是 Converter.doubleToString(d)
的结果.实际上,两者都行不通。
另一个问题是 EditText
确实对字符进行操作。它对 double、int、float、byte、short、boolean 或任何其他不是字符串的东西一无所知。
这意味着为了实现双向数据绑定(bind)您的源:
如何解决问题?
Android架构组件向我们介绍
ObservableField
类(class)。有现成可用ObservableBoolean
, ObservableChar
, ObservableFloat
和其他一些人。如果您打开上一句中的链接,您应该会看到所有类 Observable...
在左侧 Pane 中。没有
ObservableString
但是 ObservableField
接受泛型类型。因此,您可以将作为数据绑定(bind)一部分的变量定义为 ObservableField<String>("defaultValueHere")
.所以你应该拥有的是:
class StockLoadTaskModel : ViewModel() {
....
....
var d: Double = 10.0
var dataBindingVariable = ObservableField<String>(d.toString())
}
dataBindingVariable
将始终返回 EditText
的内容你把它绑定(bind)到。您可以获得该值并安全地转换为 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
.还有另一种进行双向数据绑定(bind)的方法,我在这里不讨论,因为它已经得到了回答。 Here it is .
关于Android 双向数据绑定(bind)与 Double (Kotlin),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63280275/