标签:asc var 模式 文件名 ica bindings amp padding 世界
了解一门新技术,一般从2W1H入手:
DataBinding翻译过来就是数据绑定,把数据绑定在控件上。本篇讲述的都是单向绑定,即数据绑定到控件上。现在已经支持双向绑定,也就是说,还可以把控件绑定在数据上,后续介绍。
DataBinding可以代替findViewById,让代码更简洁,而且比注解框架(如ButterKnife)效率高。
开始学习吧……
环境要求:
在module(如:app)的build.gradle中添加dataBinding的使能开关
android {
compileSdkVersion 24
buildToolsVersion "23.0.2"
defaultConfig {
...
}
// add
dataBinding{
enabled true
}
...
然后使用Sync now同步。
创建一个User类:
public class User{
String name;
String nickname;
boolean isMale;
int age;
public User(String name, String nickname, boolean isMale, int age) {
this.name = name;
this.nickname = nickname;
this.isMale = isMale;
this.age = age;
}
/**
* Getter and Setter,省略
*/
...
}
步骤:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools">
<data>
<variable
name="user"
type="com.zjun.databinding.demo.bean.User" />
</data>
<LinearLayout
android:id="@+id/activity_fast_use"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:padding="@dimen/activity_vertical_margin"
tools:context="com.zjun.databinding.demo.FastUseActivity">
<LinearLayout style="@style/StyleItemParent">
<Button
android:id="@+id/btn_load"
style="@style/StyleBeforeText"
android:text="数据1" />
<Button
android:id="@+id/btn_change"
style="@style/StyleBeforeText"
android:text="数据2" />
</LinearLayout>
<LinearLayout style="@style/StyleItemParent">
<TextView
style="@style/StyleBeforeText"
android:text="姓名:" />
<TextView
style="@style/StyleAfterText"
android:text="@{user.name}"
android:onClick="@{user.onNameClick}"/>
</LinearLayout>
<LinearLayout style="@style/StyleItemParent">
<TextView
style="@style/StyleBeforeText"
android:text="昵称:" />
<TextView
style="@style/StyleAfterText"
android:text="@{user.nickname ?? user.name}"
android:onLongClick="@{user.onNicknameLongClick}"/>
</LinearLayout>
<LinearLayout style="@style/StyleItemParent">
<TextView
style="@style/StyleBeforeText"
android:text="性别:" />
<TextView
style="@style/StyleAfterText"
android:textColor="@{user.male ? 0xFF0000FF : 0xFFFF0000}"
android:text=‘@{user.male ? @string/male : @string/female}‘ />
</LinearLayout>
<LinearLayout style="@style/StyleItemParent">
<TextView
style="@style/StyleBeforeText"
android:text="年龄:" />
<TextView
style="@style/StyleAfterText"
android:textColor="@{user.age < 14 || user.age > 65 ? 0xFFFF0000 : 0xFF000000}"
android:text=‘@{String.valueOf(user.age) + " years old"}‘ />
</LinearLayout>
</LinearLayout>
</layout>
上面的数据绑定中,主要是把数据显示android:text属性中(点击事件暂不考虑),必须注意语法:
"@{user.age + ``}"
,或 "@{String.valueOf(user.age)}"
"@{user.nickname ?? user.name}"
,代表user.nickname为null时显示user.name,否则显示自己。等同于"@{user.nickname == null ? user.name : user.nickname}"
>
<
android:text=‘@{user.male ? "男" : "女"}‘
android:text=‘@{user.male ? @string/male : @string/female}‘
public class FastUseActivity extends AppCompatActivity {
private ActivityFastUseBinding mBinding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// 换掉setContentView()
// setContentView(R.layout.activity_fast_use);
mBinding= DataBindingUtil.setContentView(this, R.layout.activity_fast_use);
}
public void onClick(View view) {
User user = new User("张君宝", "张三丰", true, 30);
mBinding.setUser(user);
}
...
}
其中:
运行结果:
可以设置控件的点击与长按事件。
A. 先定义监听事件处理方法
在Bean类中添加点击事件:
public class User{
String name;
String nickname;
boolean isMale;
int age;
...
public void onNameClick(View view) {
Toast.makeText(view.getContext(), name + " is Clicked", Toast.LENGTH_SHORT).show();
}
public boolean onNicknameLongClick(View view) {
Toast.makeText(view.getContext(), nickname + " is long Clicked", Toast.LENGTH_SHORT).show();
return true;
}
B. 在xml中使用
...
<TextView
...
android:text="@{user.name}"
android:onClick="@{user.onNameClick}"/>
...
<TextView
...
android:text="@{user.nickname ?? user.name}"
android:onLongClick="@{user.onNicknameLongClick}"/>
...
注意:
布局文件中,经常会用到include子布局,那如何把父布局中的数据传递给子布局呢?方法就是通过自定义属性。
【1】子布局layout_include_params.xml——与之前一样,定义数据模块,然后使用就OK:
<?xml version="1.0" encoding="utf-8"?>
<layout ...
xmlns:tools="http://schemas.android.com/tools">
<data>
<variable
name="user"
type="com.zjun.databinding.demo.bean.User" />
</data>
<LinearLayout
...>
<LinearLayout style="@style/StyleItemParent">
<TextView
...
android:text="姓名:" />
<TextView
...
android:text="@{user.name}" />
</LinearLayout>
<LinearLayout ...>
<TextView
...
android:text="年龄:" />
<TextView
...
android:text="@{String.valueOf(user.age) + ` years old`}" />
</LinearLayout>
</LinearLayout>
</layout>
【2】父布局activity_include_params_post.xml(添加命名空间app,用于自定义属性,然后在include标签中传递数据app:user="@{user}"
):
<?xml version="1.0" encoding="utf-8"?>
<!--为了传递数据给子布局,需要添加自定义属性,先添加命名空间app-->
<layout ...
xmlns:app="http://schemas.android.com/apk/res-auto" >
<data>
<variable
name="user"
type="com.zjun.databinding.demo.bean.User" />
</data>
<RelativeLayout
...>
<!--通过app:user来传递数据-->
<include
layout="@layout/layout_include_params"
app:user="@{user}" />
</RelativeLayout>
</layout>
这里的List的显示,表示把List集合中的对象单独拎出来显示,非适配器Adapter。如把两个User放入List中,然后在父布局中把对象取出来,再传递给子布局。
【1】父布局activity_list_show.xml:
<?xml version="1.0" encoding="utf-8"?>
<layout ...
xmlns:app="http://schemas.android.com/apk/res-auto" >
<data>
<!--import导入User类,与java代码的作用一样。这样下面可以直接引用,否则要写完整的包名+类名。如果导入了两个类名一样的类,可以使用别名alias来区分,变量中type写别名-->
<import type="com.zjun.databinding.demo.bean.User" />
<variable
name="userList"
type="java.util.List<User>" />
</data>
<LinearLayout
... >
<!--与操作数组一样,用中括号获取List中的对象-->
<include
layout="@layout/layout_include_list"
...
app:user="@{userList[0]}" />
<include
layout="@layout/layout_include_list"
...
app:user="@{userList[1]}" />
</LinearLayout>
</layout>
【2】子布局layout_include_list.xml:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android">
<data>
<variable
name="user"
type="com.zjun.databinding.demo.bean.User" />
</data>
<TextView
...
android:text="@{user.name + `[` + user.nickname + `, ` + (user.male ? `男` : `女`) + `, ` + user.age +`]`}" />
</layout>
【3】代码中设置数据:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityListShowBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_list_show);
List<User> userList = new ArrayList<>();
userList.add(new User("黄药师", "东邪", true, 30));
userList.add(new User("欧阳锋", "西毒", true, 33));
binding.setUserList(userList);
}
结果:
注意:
"<"
和">"
android:text="@{userList[0]}"
,即使你想打印对象的地址,也不行。这样使用的后果是:每个databing都报“程序包不存在”的错误,连之前正常的也错了。会懵逼的(user.male ? `男` : `女`)
,必须用小括号括起来这里专门使用了一个类DBUtils,来封装Databinding的注解
public class DBUtils {
/**
* 使用DataBinding来加载图片
* 使用@BindingAdapter注解,注解值(这里的imageUrl)可任取,注解值将成为自定义属性
* 此自定义属性可在xml布局文件中使用,自定义属性的值就是这里定义String类型url
* 《说明》:
* 1. 方法名可与注解名一样,也可不一样
* 2. 第一个参数必须是View,就是自定义属性所在的View
* 3. 第二个参数就是自定义属性的值,与注解值对应。这是数组,可多个
* 这里需要INTERNET权限,别忘了
*
* @param imageView ImageView控件
* @param url 图片网络地址
*/
@BindingAdapter({"imageUrl"})
public static void loadImage(ImageView imageView, String url) {
if (url == null) {
imageView.setImageResource(R.mipmap.ic_launcher);
} else {
Glide.with(imageView.getContext()).load(url).into(imageView);
}
}
}
先把命名空间写上:xmlns:app="http://schemas.android.com/apk/res-auto"
再使用自定义属性imageUrl
<?xml version="1.0" encoding="utf-8"?>
<layout ...
xmlns:app="http://schemas.android.com/apk/res-auto" >
<data>
<variable
name="imgUrl1"
type="String" />
<variable
name="imgUrl2"
type="String" />
</data>
<RelativeLayout
...>
<!--使用自定义属性imageUrl-->
<ImageView
...
app:imageUrl="@{imgUrl1}" />
<ImageView
...
app:imageUrl="@{imgUrl2}" />
</RelativeLayout>
</layout>
// 获取Activity***Binding就省略了
binding.setImgUrl1("http://avatar.csdn.net/4/9/8/1_a10615.jpg");
binding.setImgUrl2(null);
运行结果:
ListView的展示全靠BaseAdapter,在DataBinding中也不例外。这个相对来复杂点,先把步骤写下:
再一步一步来完成:
public class CommonAdapter extends BaseAdapter {
private Context mContext;
private List<User> mDataList;
private int layoutId; // 条目布局ID
private int variableId; // DataBinding的变量ID,可通过类似R文件的BR文件来获取
public CommonAdapter(Context context, List<User> dataList, int layoutId, int variableId) {
this.mContext = context;
this.mDataList = dataList;
this.layoutId = layoutId;
this.variableId = variableId;
}
@Override
public int getCount() { return mDataList.size(); }
@Override
public Object getItem(int position) { return mDataList.get(position); }
@Override
public long getItemId(int position) { return position; }
@Override
public View getView(int position, View convertView, ViewGroup parent) {
// 没有ViewHolder的复用,但Databinding内部已经实现了复用
ViewDataBinding binding;
if (convertView == null) {
binding = DataBindingUtil.inflate(LayoutInflater.from(mContext), layoutId, parent, false);
} else {
binding = DataBindingUtil.getBinding(convertView);
}
binding.setVariable(variableId, mDataList.get(position));
return binding.getRoot();
}
}
<?xml version="1.0" encoding="utf-8"?>
<layout ...
xmlns:app="http://schemas.android.com/apk/res-auto"
>
<data >
<variable
name="lvAdapter"
type="com.zjun.databinding.demo.adapter.CommonAdapter" />
<!--或:type="android.widget.BaseAdapter" />-->
</data>
<RelativeLayout
...>
<!--使用自定义属性adapter,因为有setAdapter()方法,所以无需定义-->
<ListView
...
app:adapter="@{lvAdapter}"
/>
</RelativeLayout>
</layout>
这个简单,上面都已经讲过了的
<?xml version="1.0" encoding="utf-8"?>
<layout ...
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
<variable
name="user"
type="com.zjun.databinding.demo.bean.User" />
</data>
<LinearLayout
...>
<ImageView
...
app:imageUrl="@{user.icon}" />
<TextView
...
android:text="@{user.name}" />
</LinearLayout>
</layout>
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityListViewBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_list_view);
List<User> list = new ArrayList<>();
for (int i = 0; i < 100; i++) {
// 3个网络图片1个本地图片,依次循环
User user = new User("user" + i, (i & 0x03) < 3 ? "http://avatar.csdn.net/4/9/8/1_a10615.jpg" : null);
list.add(user);
}
// 创建Adapter。BR类似与R文件,用于存储变量名称。位置也与R一样,在app包名下
CommonAdapter adapter = new CommonAdapter(this, list, R.layout.item_list_view, BR.user);
binding.setLvAdapter(adapter);
}
结果:
在普通的ListView中,更新数据后要立马展示到界面上,需要notifyDataSetChanged()。在但这里,Databinding里使用的是观察者模式。
实现步骤:
notifyPropertyChanged(int variableId);
。一般把它放在Setter方法中,因为属性值都是通过这里改变的。代码
Bean类:
// 1、继承BaseObservable
public class Member extends BaseObservable{
private String name;
private String icon;
public Member(String name, String icon) {
this.name = name;
this.icon = icon;
}
// 2、在属性值的Getter上添加@Bindable注解
@Bindable
public String getName() {
return name;
}
// 3、在更新的地方添加属性更新通知方法:notifyPropertyChanged(filedId);
public void setName(String name) {
this.name = name;
notifyPropertyChanged(com.zjun.databinding.demo.BR.name);
}
public String getIcon() {
return icon;
}
public void setIcon(String icon) {
this.icon = icon;
}
public void onItemClick(View view) {
setName(name + "【已更新】");
}
}
布局文件(在整个条目上设置点击事件,也可以设置在某个控件上):
<?xml version="1.0" encoding="utf-8"?>
<layout ...
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
<variable
name="member"
type="com.zjun.databinding.demo.bean.Member" />
</data>
<!--设置点击事件-->
<LinearLayout
...
android:onClick="@{member.onItemClick}">
<ImageView
...
app:imageUrl="@{member.icon}" />
<TextView
...
android:text="@{member.name}" />
</LinearLayout>
</layout>
Activity中的代码同上,不过我把CommonAdapter的User改成了泛型,以便通用。
结果:
还有两种方法:ObserableField和ObservableMap,见最后的参考1或2
前面我们加载图片的时候,使用了注解@BindingAdapter({"imageUrl"})
,和static静态方法来添加自定义属性。但static没有非static功能多,有时就要用到对象等,怎么办?
DataBinding里也可以通过component组件的方式,来实现非static来添加自定义属性:
代码实现:
public class NotStaticUtils {
@BindingAdapter({"imgSrc"})
public void showImg(ImageView imageView, String src) {
if (src == null) {
imageView.setImageResource(R.mipmap.ic_launcher);
} else {
Glide.with(imageView.getContext()).load(src).into(imageView);
}
}
}
public class MyComponent implements DataBindingComponent{
private NotStaticUtils utils;
@Override
public NotStaticUtils getNotStaticUtils() {
if (utils == null) {
utils = new NotStaticUtils();
}
return utils;
}
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// 方法一、设置默认的组件
DataBindingUtil.setDefaultComponent(new MyComponent());
ActivityNotStaticBinding dataBinding = DataBindingUtil.setContentView(this, R.layout.activity_not_static);
// 方法二、直接在构造中传入,这里是单独地给本DataBinding指定自定义组件
// ActivityMainBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_main, new Component());
dataBinding.setSrc1("http://avatar.csdn.net/4/9/8/1_a10615.jpg");
dataBinding.setSrc2(null);
}
这个跟上面一样,就是名称改了一下而已。核心如下:
<variable
name="src1"
type="String" />
...
<!--使用自定义属性imgSrc-->
<ImageView
...
app:imgSrc="@{src1}" />
Fragment都有独立的布局,好像与Activity没有什么关系,但DataBinding中有一个方法可以绑定View:DataBindingUtil.bind(View root)
。
<layout ...>
<data>
<variable name="province" type="String" />
<variable name="city" type="String" />
</data>
<LinearLayout ...>
<TextView ...
android:text="@{province}" />
<TextView ...
android:text="@{city}" />
</LinearLayout>
</layout>
获取DataBinding的方式有三种:
private FragmentShowBinding mBinding;
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
return inflater.inflate(R.layout.fragment_show, container, false);
// 或,在这里获取DataBinding
// mBinding = DataBindingUtil.inflate(inflater, R.layout.fragment_show, container, false);
// return mBinding.getRoot();
}
@Override
public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
// 绑定View
mBinding = DataBindingUtil.bind(view);
mBinding.setProvince("浙江");
mBinding.setCity("宁波");
// 或,获取父类ViewDataBinding,然后通过设置变量的方法
// ViewDataBinding binding = DataBindingUtil.bind(view);
// binding.setVariable(BR.province, "江西");
// binding.setVariable(BR.city, "赣州");
}
RecyclerView跟ListView一样,有一个适配器Adapter。但要正常显示,RecyclerView还需要一个LayoutManager(第一次就不小心就掉此坑了)。
网上有的方法是只在item条目里使用Databinding,RecyclerView还是通过findViewById来获取。但这里是Databinding,所以我这里就不要findViewById了。
在Databinding的世界里,如果这个View有setter方法,直接使用即可,没有我们可以自定义属性(见2.6)。比如要添加分割线,自己去玩吧。
直接看代码来看步骤逻辑:
activity_recycler_view.xml
<?xml version="1.0" encoding="utf-8"?>
<layout ...>
<data >
<variable
name="rvLayoutManager"
type="android.support.v7.widget.LinearLayoutManager" />
<variable
name="rvAdapter"
type="com.zjun.databinding.demo.adapter.RVAdapter" />
<!--或:type="android.support.v7.widget.RecyclerView.Adapter" />-->
</data>
<RelativeLayout ...>
<android.support.v7.widget.RecyclerView
...
app:layoutManager="@{rvLayoutManager}"
app:adapter="@{rvAdapter}"
/>
</RelativeLayout>
</layout>
跟上面ListView的条目布局一样
item_recycler_view.xml
<?xml version="1.0" encoding="utf-8"?>
<layout ...>
<data>
<variable name="member" type="com.zjun.databinding.demo.bean.Member" />
</data>
<LinearLayout ...>
<ImageView ...
app:imageUrl="@{member.icon}" />
<TextView ...
android:text="@{member.name}" />
</LinearLayout>
</layout>
有两种写法,第一种是官方的,第二种通用性好。官方的里面,条目布局都不用(哈哈,是不是可以在新手面前装一下?但查看代码时,不能直接跳到布局文件,要显示也可以,根据上一节Fragment自己去修改吧)
一、官方办法
public class RVAdapter extends RecyclerView.Adapter<RVAdapter.RVHolder> {
private Context mContext;
private List<Member> mDataList;
public RVAdapter(Context context, List<Member> list) {
this.mContext = context;
this.mDataList = list;
}
@Override
public RVHolder onCreateViewHolder(ViewGroup parent, int viewType) {
return RVHolder.create(LayoutInflater.from(mContext), parent);
}
@Override
public void onBindViewHolder(RVHolder holder, int position) {
holder.bindTo(mDataList.get(position));
}
@Override
public int getItemCount() {
return mDataList == null ? 0 :mDataList.size();
}
static class RVHolder extends RecyclerView.ViewHolder {
// 创建一个静态获取方法
static RVHolder create(LayoutInflater inflater, ViewGroup parent) {
ItemRecyclerViewBinding binding = ItemRecyclerViewBinding.inflate(inflater, parent, false);
return new RVHolder(binding);
}
ItemRecyclerViewBinding mBinding;
private RVHolder(ItemRecyclerViewBinding binding) {
super(binding.getRoot());
this.mBinding = binding;
}
public void bindTo(Member member) {
mBinding.setMember(member);
// 它使数据绑定刷新所有挂起的更改。这官方的解释好难懂,其实功能就是让数据立即展示在布局上
mBinding.executePendingBindings();
}
}
}
activity中使用:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityRecyclerViewBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_recycler_view);
// 模拟数据
List<Member> list = new ArrayList<>();
for (int i = 0; i < 100; i++) {
// 1个网络图片1个本地图片,依次循环
Member member = new Member("user" + i, (i & 0x01) == 0 ? "http://avatar.csdn.net/4/9/8/1_a10615.jpg" : null);
list.add(member);
}
// 设置布局管理器,及适配器
binding.setRvLayoutManager(new LinearLayoutManager(this, LinearLayoutManager.VERTICAL, false));
binding.setRvAdapter(new RVAdapter(this, list));
}
二、通用办法
public class RVAdapter extends RecyclerView.Adapter<RVAdapter.RVHolder> {
private Context mContext;
private List<Member> mDataList;
private int mLayoutId; // 条目布局文件ID
private int mVariableId; // DataBinding变量ID
public RVAdapter(Context context, List<Member> list, int layoutId, int variableId) {
this.mContext = context;
this.mDataList = list;
this.mLayoutId = layoutId;
mVariableId = variableId;
}
@Override
public RVHolder onCreateViewHolder(ViewGroup parent, int viewType) {
ViewDataBinding binding = DataBindingUtil.inflate(LayoutInflater.from(mContext), mLayoutId, parent, false);
RVHolder holder = new RVHolder(binding.getRoot());
holder.binding = binding;
return holder;
}
@Override
public void onBindViewHolder(RVHolder holder, int position) {
holder.binding.setVariable(mVariableId, mDataList.get(position));
// 别忘记这句代码
holder.binding.executePendingBindings();
}
@Override
public int getItemCount() {
return mDataList == null ? 0 :mDataList.size();
}
class RVHolder extends RecyclerView.ViewHolder {
ViewDataBinding binding;
RVHolder(View view) {
super(view);
}
}
}
activity中把setAdapter()代码,改成这句这就好了:
binding.setRvAdapter(new RVAdapter(this, list, R.layout.item_recycler_view, BR.member));
DataBinding的错误很难找,但基本都是在xml文件中出错的。要仔细
很难找到具体原因,一个一个排查吧:
DataBinding快速入门(还在用findViewById?)
标签:asc var 模式 文件名 ica bindings amp padding 世界
原文地址:http://blog.csdn.net/a10615/article/details/52781956