首页 > 解决方案 > 使用带参数的自定义绑定适配器侦听器方法的 Android 自定义视图

问题描述

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

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

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

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

标签: androidkotlindata-bindingparameter-passingandroid-binding-adapter

解决方案


我不认为这会起作用,因为不能在 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}"/>

推荐阅读