Android双向数据绑定与Double(Kotlin)



我有一个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_quantityEditText中,我会得到以下错误:

...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类。有现成的ObservableBooleanObservableCharObservableFloat和其他一些如果打开上一句中的链接,您应该会在左侧窗格中看到所有类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

还有另一种进行双向数据绑定的方法,我在这里不讨论,因为它已经得到了回答。给你。

最新更新