有所了解的都知道databinding出现的意义就在于系统自动帮助我们生成Data和View之间的connection.而在Adapter中出现的ViewHolder就是我们手动生成的connection,下面我们就来看看databinding是如何完全取代ViewHolder的角色吧。(在此之前请看如何配置环境,在上述链接里有)
- public class DynamicActivity extends Activity {
private View.OnClickListener itemClickListener;
private GridView gridView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_dynamic);
itemClickListener = new View.OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(getBaseContext(), v.getClass().getSimpleName(), Toast.LENGTH_SHORT).show();
}
};
gridView = (GridView) findViewById(R.id.gridView);
MMyAdapter adapter = new MMyAdapter(getBaseContext());
adapter.setOnclickListener(itemClickListener);
gridView.setAdapter(adapter);
}
}
listener用于处理item的一些click事件,当然也可以直接写到adapter里面去,但是考虑到adapter还是比较纯粹的处理view和data的关系,所以把这个还是放到了activity层。
接着是activity_main.xml文件比较简单就一个gridview
- <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<GridView
android:id="@+id/gridView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:numColumns="3" />
</RelativeLayout>
接着是比较关键的adapter
- /**
* Created by zhangxiaang on 15/7/6.
*/
public class MMyAdapter extends BaseAdapter {
private Context mContext;
private LayoutInflater inflater;
private List<String> mlist;
public View.OnClickListener itemClickListener;
private ItemmBinding binding;
public MMyAdapter(Context context) {
this.mContext = context;
inflater = LayoutInflater.from(context);
mlist = new ArrayList<String>();
for (int i = 0; i < 100; i++) {
mlist.add(i, "item" + i);
}
}
@Override
public int getCount() {
return mlist == null ? 0 : mlist.size();
}
@Override
public Object getItem(int position) {
return mlist.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
if (convertView == null) {
binding = DataBindingUtil.inflate(inflater, R.layout.itemm, parent, false);
convertView = binding.getRoot();
convertView.setTag(binding);
} else {
binding = (ItemmBinding) convertView.getTag();
}
binding.setVariable(BR.item, mlist.get(position));
binding.setAdapter(this);
return convertView;
}
public void setOnclickListener(View.OnClickListener itemClickListener) {
this.itemClickListener = itemClickListener;
}
}
其中的item布局如下:
- <layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<data>
<variable
name="item"
type="String" />
<variable
name="adapter"
type="com.liangfeizc.databindingsamples.dynamic.MMyAdapter" />
</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:clickable="true"
app:text="@{item}"
app:onClickListener="@{adapter.itemClickListener}"/>
</LinearLayout>
</layout>
可以看到的是在adapter中完全没有 viewHolder的身影,在getView中,开始直接判断当前的convertview是否为空,进而来初始化这个binding对象,而convertView就是这个binding对象所对应的container的root view类似一个view-tree的rootView角色,然后同样的给当前convertView打一个标签方便后续取出,如果不为空的话就直接取出当前convertView的binding对象。
处理完convertView的视图部分接着就是处理convertView的数据部分,而数据部分应该在包含在convertview的binding中进行赋值。(感兴趣的可以去看看生成的binding代码,自动的替我们生成了<variable>标签里面数据的setter函数)
最后把设置完了view和data的converView直接返回就ok 啦,大家跑一边看看databinding带来的便利吧。
Ps:由于在rufi的案例中已经给了一个recyclerView的案例,但是我个人觉得recyclerView中已经封装了ViewHolder,所以再使用databinding的话就显得有点多余了。毕竟databinding的存在意义就是取代掉viewHolder的。
欢迎各位提出自己的一些想法或者实现过程中遇到的问题。