Android 使用带参数的自定义绑定适配器侦听器方法的自定义视图

Android custom view using custom binding adapter listener method with parameter

我正在开发一个自定义搜索视图,我需要添加一个侦听器,以便视图模型可以使用数据绑定对查询执行搜索,我目前在使用查询参数设置绑定适配器时遇到问题,在这里相关部分是:

这是带有侦听器事件的自定义视图:

class MySearchView @JvmOverloads constructor(
    context: Context,
    attrs: AttributeSet? = null,
    defStyleAttr: Int = android.R.attr.editTextStyle
) : AppCompatEditText(context, attrs, defStyleAttr) {

    var onSearchListener: ((query: String) -> Unit)? = null
}

这是绑定适配器:

@BindingAdapter("onSearchListener")
fun MySearchView.setOnSearchListener(event: (query: String) -> Unit) {
    onSearchListener = event
}

这是 XMl 布局部分:

<com.xxx.MySearchView
    android:id="@+id/txt_search"
    android:layout_width="0dp"
    android:layout_height="wrap_content"
    android:hint="@string/label_search"
    app:layout_constraintEnd_toEndOf="parent"
    app:layout_constraintStart_toStartOf="parent"
    app:layout_constraintTop_toTopOf="parent"
    app:onSearchListener="@{(query) -> viewModel.onSearchTriggered(query)}"/>

最后我在视图模型中的方法:

fun onSearchTriggered(query: String) {
    Log.d("search", "search: $query")
}

当我编译 XML 失败时告诉我它找不到上面的方法,这是错误:

cannot find method onSearchTriggered(java.lang.Object) in class com.xxx.MyViewModel

我之前尝试过使用不带参数的自定义侦听器并且一切正常所以我认为我在添加参数时一定是做错了什么,有什么想法吗?

我认为这行不通,因为无法在 XML 中指定参数类型,但我希望传递 viewModel::onSearchTriggered 行得通,但看起来行不通。您可以使用以下方法之一来执行此操作:

  1. 您可以在 ViewModel 中声明变量而不是函数:
val onSearchTriggered = { query: String ->
    Log.d("search", "search: $query")
}

然后您可以像这样直接传递变量:

<com.xxx.MySearchView
    android:id="@+id/txt_search"
    android:layout_width="0dp"
    android:layout_height="wrap_content"
    android:hint="@string/label_search"
    app:layout_constraintEnd_toEndOf="parent"
    app:layout_constraintStart_toStartOf="parent"
    app:layout_constraintTop_toTopOf="parent"
    app:onSearchListener="@{viewModel.onSearchTriggered}"/>
  1. 您可以为您的侦听器创建一个接口:
interface OnSearchListener {
    fun onSearch(String)
}

@BindingAdapter("onSearchListener")
fun MySearchView.setOnSearchListener(listener: OnSearchListener) {
    onSearchListener = listener::onSearch
}

然后您可以像这样使用它:

<com.xxx.MySearchView
    android:id="@+id/txt_search"
    android:layout_width="0dp"
    android:layout_height="wrap_content"
    android:hint="@string/label_search"
    app:layout_constraintEnd_toEndOf="parent"
    app:layout_constraintStart_toStartOf="parent"
    app:layout_constraintTop_toTopOf="parent"
    app:onSearchListener="@{viewModel::onSearchTriggered}"/>