使用 BindingAdapter 时数据绑定器无法识别 lambda 中的类型化参数

Typed parameter in a lambda not recognized by the data binder when using BindingAdapter

我正在尝试绑定以下元素:

xml:

<android.support.v7.widget.SwitchCompat
    ...
    bind:onCheckedChanged="@{(isChecked) -> viewModel.onCheckedChanged(isChecked)}"
    .../>

视图模型:

class MyViewModel() {
    fun onCheckedChanged(isChecked: Boolean) {
        ...
    }
}

使用 BindingAdapter:

@BindingAdapter("onCheckedChanged")
fun bindOnCheckedChanged(view: SwitchCompat, onCheckedChanged: (Boolean) -> Unit) {
    view.setOnCheckedChangeListener(
        { _, isChecked ->
            if (view.isPressed) onCheckedChanged(isChecked)
        }
    )
}

我得到的错误是这个:

data binding error ****msg:cannot find method onCheckedChanged(java.lang.Object) in class MyViewModel

数据绑定器似乎无法将 isChecked 识别为 Boolean。我试图像 isChecked:Boolean 那样强制输入 xml,但出现了一堆不同的错误。

现在我通过使用 Any 而不是 Boolean 使其工作,但我觉得这是错误的:

@BindingAdapter("onCheckedChanged")
fun bindOnCheckedChanged(view: SwitchCompat, onCheckedChanged: (Any) -> Unit) {
    ...

fun onCheckedChanged(isChecked: Any) {
    val isSwitchChecked = isChecked as? Boolean ?: return
    ...

有谁知道如何让它在正确的函数签名下工作?

根据 Binding Adapters documentation:“事件处理程序只能与接口一起使用或抽象 classes 与一个抽象方法一起使用,如下例所示:

@BindingAdapter("android:onLayoutChange")
fun setOnLayoutChangeListener(
    view: View,
    oldValue: View.OnLayoutChangeListener?,
    newValue: View.OnLayoutChangeListener?
) {
  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
      if (oldValue != null) {
          view.removeOnLayoutChangeListener(oldValue)
      }
      if (newValue != null) {
          view.addOnLayoutChangeListener(newValue)
      }
  }
}

DB docs never specify you can use Kotlin lambdas in Binding Adapters 在这种情况下,您只需要使用一种方法创建接口,如下所示:

interface CustomOnCheckedListener {
    fun onChecked(isChecked: Boolean)
} 

那么你的绑定适配器:

@BindingAdapter("onCheckedChanged")
fun bindOnCheckedChanged(view: SwitchCompat, onCheckedChanged: CustomCheckListener) {
    view.setOnCheckedChangeListener(
        { _, isChecked ->
            if (view.isPressed) onCheckedChanged.onChecked(isChecked)
        }
    )
}

在您的 ViewModel class 中,您需要创建一个与侦听器方法具有相同签名的函数:

fun onChecked(isChecked: Boolean){
    //some code here
}

并在您的 xml 中简单地传递一个对您的函数的引用:

"@{viewModel::onChecked}"

解决方案:(KOTLIN)

使用lambda参数


| val lambda: (arg1, arg2, ...) -> 输入 |


1️⃣ ➖ Lambda 实现:

MainActivity.kt

val lambda: (TextView?) -> Unit = {
    Log.i(TAG, "lambda: ${it?.text}")
}

2️⃣ ➖ BindingAdapter 实现:

Bindings.kt

@BindingAdapter("lambda")
fun View.lambda(block: (TextView?) -> Unit) {
    // Your logic
}

3️⃣ ➖ 使用数据绑定和绑定适配器:

activity_main.xml

<layout xmlns:android="http://schemas.android.com/apk/res/android">

   <data>

       <variable
           name="activityMain"
           type="com.veldan.mvi.ui.activities.MainActivity" />
   </data>

   <View
        lambda="@{activityMain.lambda}"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

</layout>