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

标签 android kotlin data-binding android-databinding

我有一个 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/

    相关文章:

    android - OpenGL ES 顶点/索引

    java - 将数据传递到查询中,而不是更新

    Android:有没有办法在应用程序中实现条码扫描器?

    java - kotlin 注释处理器中的可空类型

    wpf - 如何转义 WPF 绑定(bind)路径中的斜杠字符,或者如何解决?

    c# - WPF MVVM - 绑定(bind)到来自祖先 View 模型的属性

    java - Android 上的 Jsoup - OutOfMemory 错误

    kotlin - 我应该如何在 Kotlin 中将一个集合过滤到另一个集合

    c# - 如何在 XAML 中设置 DataGrid 的 ItemsSource?

    android - 如何避免在我的 Android Room 存储库中出现许多 AsyncTask 类?