Android DataBinding ——事件处理

DataBinding允许您编写表达式去处理从View分发的事件。例如onClick()方法。事件属性名称由监听器方法的名称确定,但有少数例外。例如,View.OnClickListener有一个方法onClick(),因此此事件绑定的属性为android:onClick。

有一些专门的点击事件处理者(注:它们也接受View.OnClickListener参数)需要使用android:onClick以外的属性来避免冲突。您可以使用以下属性来避免这些类型的冲突:

Class Listener setter Attribute
SearchView setOnSearchClickListener(View.OnClickListener) android:onSearchClick
ZoomControls setOnZoomInClickListener(View.OnClickListener) android:onZoomIn
ZoomControls setOnZoomOutClickListener(View.OnClickListener) android:onZoomOut

您可以使用以下机制来处理事件:

  • 方法引用:在表达式中,可以引用符合监听器方法签名的方法。当表达式计算为方法引用时,DataBinding将方法引用和所有者对象封装到一个监听听器中,并设置到该View上。如果表达式的计算结果为null,则DataBinding不会创建监听器,该View这个属性的监听器为null。

  • 监听器绑定:这些是在事件发生时计算的lambda表达式。DataBinding总是创建一个监听器,并设置到这个View。当事件被分发时,监听器将计算lambda表达式。

1. 方法引用

事件可以直接绑定到处理方法上,类似于android:onClick可以分发到Activity中的方法。与View的onClick属性相比,一个主要优点是表达式在编译时处理,因此,如果该方法不存在或其签名不正确,则会收到编译时错误。

方法引用和监听器绑定之间的主要区别在于,实际的监听器实现是在绑定数据时创建的,而不是在触发事件时创建的。如果希望在事件发生时对表达式求值,则应使用监听器绑定。

若要将事件分配给其处理程序,请使用普通绑定表达式,该值为要调用的方法名。例如,考虑以下布局数据对象示例:

java 复制代码
class MyHandlers {
    fun onClickFriend(view: View) { ... }
}

绑定表达式可以将视图的单击监听器分发给onClickFriend()方法,如下所示:

xml 复制代码
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">
   <data>
       <variable name="handlers" type="com.example.MyHandlers"/>
       <variable name="user" type="com.example.User"/>
   </data>
   <LinearLayout
       android:orientation="vertical"
       android:layout_width="match_parent"
       android:layout_height="match_parent">
       <TextView android:layout_width="wrap_content"
           android:layout_height="wrap_content"
           android:text="@{user.firstName}"
           android:onClick="@{handlers::onClickFriend}"/>
   </LinearLayout>
</layout>

注释:这里表达式里是直接引用的函数。这个函数的签名要和onClickListener的onClick的签名一样

2. 监听者绑定

监听者绑定是在事件发生时运行的绑定表达式。它们类似于方法引用,但允许您运行任意的数据绑定表达式。此功能可与Gradle 2.0及更高版本的Android Gradle插件一起使用。

在方法引用中,方法的参数必须与事件监听者的参数匹配。在监听者绑定中,只要返回值必须与监听者的预期返回值匹配(除非它预期为void)即可。例如,考虑以下具有onSaveClick() 方法示例:

java 复制代码
class Presenter {
    fun onSaveClick(task: Task){}
}

然后,您可以将click事件绑定到onSaveClick() 方法,如下所示:

xml 复制代码
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">
    <data>
        <variable name="task" type="com.android.example.Task" />
        <variable name="presenter" type="com.android.example.Presenter" />
    </data>
    <LinearLayout android:layout_width="match_parent" android:layout_height="match_parent">
        <Button android:layout_width="wrap_content" android:layout_height="wrap_content"
        android:onClick="@{() -> presenter.onSaveClick(task)}" />
    </LinearLayout>
</layout>

在表达式中使用回调时,DataBinding会自动创建必要的监听者并为事件注册它。当视图激发事件时,DataBinding将计算给定的表达式。与正则绑定表达式一样,在评估这些监听者表达式时,您仍然可以获得DataBinding的空安全和线程安全性。

在上面的例子中,我们还没有定义传递给onClick(view)的视图参数。监听者绑定为监听者参数提供了两种选择:可以忽略方法的所有参数,也可以命名所有参数。如果您喜欢命名参数,可以在表达式中使用它们。例如,上面的表达式可以写如下:

java 复制代码
android:onClick="@{(view) -> presenter.onSaveClick(task)}"

或者,如果您想在表达式中使用参数,它可以按如下方式工作:

java 复制代码
class Presenter {
    fun onSaveClick(view: View, task: Task){}
}
java 复制代码
android:onClick="@{(theView) -> presenter.onSaveClick(theView, task)}"

可以将lambda表达式与多个参数一起使用:

java 复制代码
class Presenter {
    fun onCompletedChanged(task: Task, completed: Boolean){}
}
java 复制代码
<CheckBox android:layout_width="wrap_content" android:layout_height="wrap_content"
      android:onCheckedChanged="@{(cb, isChecked) -> presenter.completeChanged(task, isChecked)}" />

如果正在监听的事件返回的值的类型不是void,则表达式也必须返回相同类型的值。例如,如果要监听长按事件,则表达式应返回布尔值。

java 复制代码
class Presenter {
    fun onLongClick(view: View, task: Task): Boolean { }
}
java 复制代码
android:onLongClick="@{(theView) -> presenter.onLongClick(theView, task)}"

如果由于null对象而无法计算表达式,则数据绑定将返回该类型的默认值。例如,引用类型返回null,int返回0,boolean返回false等。

如果需要将表达式与谓词一起使用(例如,三元),可以使用void作为一个符号值。

java 复制代码
android:onClick="@{(v) -> v.isVisible() ? doSomething() : void}"

3. 避免复杂的监听者

监听者表达式非常强大,可以使代码非常易于阅读。另一方面,包含复杂表达式的监听者会使布局难以阅读和维护。这些表达式应该像将可用数据从UI传递到回调方法一样简单。您应该在监听者表达式调用的回调方法中实现任何业务逻辑。(注:业务逻辑都写到ViewModel的一个入口函数,databing绑定那个函数即可)

👀关注公众号:Android老皮!!!欢迎大家来找我探讨交流👀

相关推荐
下位子1 分钟前
『OpenGL学习滤镜相机』- Day3: 着色器基础 - GLSL 语言
android·opengl
bqliang3 分钟前
Jetpack Navigation 3:领航未来
android·android studio·android jetpack
云存储小天使12 分钟前
安卓蛙、苹果蛙为什么难互通?
android
陈大头铃儿响叮当2 小时前
Android Studio升级后,Flutter运行android设备报错
android·flutter·android studio
勤劳打代码3 小时前
isar_flutter_libs 引发 Namespace not specified
android·flutter·groovy
奔跑吧 android4 小时前
【android bluetooth 协议分析 18】【PBAP详解 2】【车机为何不显示电话号码为空的联系人信息】
android·蓝牙电话·hfp·pbap·电话簿
深盾科技4 小时前
安卓二次打包技术深度拆解:从逆向篡改到防护逻辑
android
4Forsee4 小时前
【Android】消息机制
android·java·前端
2501_915921436 小时前
iOS 虚拟位置设置实战,多工具协同打造精准调试与场景模拟环境
android·ios·小程序·https·uni-app·iphone·webview
龚礼鹏6 小时前
Android 图像显示框架三——演示demo以及解析
android·交互