1、系统提供的组件使用DataBinding
比较简单,只需要在xml布局文件里面直接引用就好.
在build.gradle
文件中添加配置
android {
dataBinding{
enabled = true
}
}
系统会根据布局文件名称自动生成相应的DataBinding
类,例如
布局文件activity_main.xml
会生成ActivityMainBinding
类 ,类名生成规则为驼峰命名:文件名(首字母大写) + Binding
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
<variable
name="viewModel"
type="com.package.ViewModel" />
</data>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<TextView
android:id="@+id/textView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@{viewmodel.text}"
/>
</LinearLayout>
</layout>
activity
中使用:
var viewBinding:ActivityMainBinding = DataBindingUtil.setContentView(this,R.layout.activity_main)
viewBinding.viewModel = MainViewModel()
fragment
中使用:
override fun onCreateView(
inflater: LayoutInflater,
container: ViewGroup?,
savedInstanceState: Bundle?): View? {
var viewBinding:FragmentMainBinding = FragmentMainBinding.inflate(inflater,container,false).apply {
viewmodel = MainViewModel()
}
return viewDataBinding.root
}
这里 android:text="@{viewmodel.text}"
对text进行设置
在Edittext
中可以使用android:text="@={viewmodel.text}"
进行双向绑定,关键是这个=
号;
若需要在xml布局文件中使用系统类的属性,则需要引入.例:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
<variable
name="viewModel"
type="com.package.ViewModel" />
<import type="android.view.View"/>
</data>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<TextView
android:id="@+id/textView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:visibility="@{viewmodel.hidden ? View.GONE:View.VISIBLE}"
/>
</LinearLayout>
</layout>
通过引入 <import type="android.view.View"/>
可以使用View.VISIBLE
、View.GONE
属性
2、自定义View 属性使用绑定
有时候自定义view要用DataBinding时需要通过@BindingAdapter
设置
class MineView constructor(context:Context):View(context){
var paramA:String = ""
var paramB:String = ""
companion object{
//单向绑定
@BindingAdapter("app:paramA")
@JvmStatic
fun setParamA(view:MineView,param:String){
if(view.paramA != param){
view.paramA = param
}
}
//双向绑定
@BindingAdapter("app:paramB")
@JvmStatic
fun setParamB(view:MineView,param:String){
if(view.paramB!= param){
view.paramB = param
}
}
@InverseBindingAdapter(attribute = "app:paramB")
@JvmStatic
fun getParamB(view:MineView):String{
return view.paramB
}
//这个地方是双向绑定的关键,attrChange 是通知dataBinding 内容变更,然后取回当前View变更内容
@BindingAdapter("app:paramBAttrChanged")
@JvmStatic
fun setListeners(
view:MineView,
attrChange: InverseBindingListener
) {
//这个地方去判断调用时机,可以保存attrChange,在适当的时候调用
when(paramB changed){
attrChange.onChange()
}
}
}
}
单向绑定设置比较简单@BindingAdapter("app:paramA")
方法名为setParamA
,单向绑定嘛就是传入参数:
<MineView
app:paramA="@{viewmodel.paramA}"/>
双向绑定 set
、get
方法,最主要的是setListeners
方法,名字可以随意,只是@BindingAdapter
中是"app:paramBAttrChanged"
,在属性后面跟上AttrChanged
,attrChange
为通知系统内容变更的回调,当在MineView
内部paramB
发生变更时,通过attrChange
去通知对paramB
进行绑定的对象更新自己缓存中的内容 如:
<MineView
app:paramB="@={viewmodel.paramB}"/>
希望对您有帮助.