1. 前言
Databinding提供了很多有用的注解,介绍两个介绍几个常用的注解
@BindingAdapter
使用BindingAdapter注解来创建一个自定义的xml属性。当这个属性以正确的类型设置到布局文件中时,data binding 框架会触发被注解的方法。注解方法必须是静态的。
对于一些View本身就没有android:xxxx或者app:xxx属性,我们是使用不了,但是在databinding里,使用BindingAdapter即是没有相应的属性xxx,也可以使用。
@BindingAdapter(value = {"android:imageUrl", "android:placeHolder", "android:error"}, requireAll = false)
public static void loadImage(ImageView imageView, String url, Drawable placeHolder, Drawable error) {
ImageLoader.bind(imageView, url, placeHolder, error, true, DiskCacheStrategy.ALL, null);
}
再例如一些属性,没有对应的set方法,或者方法签名不同,BindingAdapter注释可以帮我们来解决这个问题。 像View的android:paddingLeft属性,是没有对应的直接进行设置的方法的,只有setPadding(left, top, right, bottom),而我们又不可能为了使用Data Binding去继承修改这种基础的View(即便修改了,还有一堆继承它的View呢,类似这种情况,一般不需要我们手动处理,绝大多数框架已经处理了)。
伪代码如下:
@BindingAdapter("android:paddingLeft")
public static void setPaddingLeft(View view, int padding) {
view.setPadding(padding,
view.getPaddingTop(),
view.getPaddingRight(),
view.getPaddingBottom());
}
2. 代码实现
package com.example.databinding.databindingdemo.model;
import android.databinding.BaseObservable;
import android.databinding.Bindable;
import android.databinding.BindingAdapter;
import android.widget.ImageView;
import com.example.databinding.databindingdemo.BR;
import com.example.databinding.databindingdemo.R;
/**
* Created by user on 2018/3/8.
*/
public class UserInfo extends BaseObservable{
private String name;
@Bindable
private String nickname;
@Bindable
private int resId = R.mipmap.ic_launcher;
private int paLeft = 100;
@Bindable
private boolean vip = true;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getNickname() {
return nickname;
}
public void setNickname(String nickname) {
this.nickname = nickname;
notifyPropertyChanged(BR.nickname);
}
public int getResId() {
return resId;
}
public void setResId(int resId) {
this.resId = resId;
notifyPropertyChanged(BR.resId);
}
public int getPaLeft() {
return paLeft;
}
public void setPaLeft(int paLeft) {
this.paLeft = paLeft;
}
public boolean isVip() {
return vip;
}
public void setVip(boolean vip) {
this.vip = vip;
}
@BindingAdapter("imageUrl")
public static void loadImage(ImageView imageView, int resId){
imageView.setImageResource(resId);
}
}
package com.example.databinding.databindingdemo;
import android.databinding.DataBindingUtil;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import com.example.databinding.databindingdemo.adapter.AppAdapter;
import com.example.databinding.databindingdemo.databinding.ActivityMainBinding;
import com.example.databinding.databindingdemo.handler.MyClick;
import com.example.databinding.databindingdemo.listener.UserClickListener;
import com.example.databinding.databindingdemo.model.UserInfo;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity {
private ActivityMainBinding mainBinding;
private UserInfo userInfo;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mainBinding = DataBindingUtil.setContentView(this, R.layout.activity_main);
userInfo = new UserInfo();
userInfo.setName("name1");
mainBinding.setUser(userInfo);
}
}
@BindingMethods
如果View本身支持这种属性的set,只是xml中的属性名字和java代码中的方法名不相同,这时候我们可以使用BindingMethods注释,处理改问题。
例如:android:tint是给ImageView加上着色的属性,可以在不换图的前提下改变图标的颜色。如果我们直接对android:tint使用data binding,由于会去查找setTint方法,而该方法不存在,则会编译出错。而实际对应的方法,应该是setImageTintList。
这时候我们就可以使用BindingMethod指定属性的绑定方法:
@BindingMethods({
@BindingMethod(type = BindTextView.class,
attribute = “android:bold”,
method = “setTextBold”),
})
注:即是我们没有做任何处理,直接使用android:tint=“@{@color/colorAccent}”也没有报错,是因为dataBinding已经帮我们定义好了转换方法了,在你使用android:tint=“@{@color/colorAccent}"时,已经自动使用setImageTintList()方法生成代码了。所以一般我们也不需要做这种错误的处理,大多数类似的情况框架都已经处理了。
案例如下:
package com.example.databinding.databindingdemo.widget;
import android.content.Context;
import android.databinding.BindingMethod;
import android.databinding.BindingMethods;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import android.widget.TextView;
/**
* Created by user on 2018/3/8.
*/
@BindingMethods({
@BindingMethod(
type = BindTextView.class,
attribute = "android:bold",
method = "setTextBold"),
})
public class BindTextView extends TextView {
public BindTextView(Context context) {
super(context);
}
public BindTextView(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}
public BindTextView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
public void setTextBold(boolean isVIP){
getPaint().setFakeBoldText(isVIP);
}
}
package com.example.databinding.databindingdemo;
import android.databinding.DataBindingUtil;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import com.example.databinding.databindingdemo.adapter.AppAdapter;
import com.example.databinding.databindingdemo.databinding.ActivityMainBinding;
import com.example.databinding.databindingdemo.handler.MyClick;
import com.example.databinding.databindingdemo.listener.UserClickListener;
import com.example.databinding.databindingdemo.model.UserInfo;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity {
private ActivityMainBinding mainBinding;
private UserInfo userInfo;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mainBinding = DataBindingUtil.setContentView(this, R.layout.activity_main);
userInfo = new UserInfo();
userInfo.setName("name1");
mainBinding.setUser(userInfo);
}
}
AndroidDatabinding的使用(一)
AndroidDatabinding的使用之双向绑定(二)
AndroidDatabinding的使用之事件绑定(三)
AndroidDatabinding的使用进阶之注解(四)
AndroidDatabinding原理分析(五)