完美解决RecyclerView点击事件、长按事件、子项点击事件
自从Google推出了RecyclerView之后,便可以完全取代ListView,个人感觉唯一的美中不足是对于itemView的各种点击事件不够完美。观点只代表个人看法。应最近项目需求实现itemView的子项点击事件,便写篇博客记录一下,若是能够帮到你,我深感荣幸。接下来,便对RecyclerView进行简单的封装,使得它更方便实现各种点击事件。
我们都知道,对与RecyclerView的使用,是创建一个adapter类,然后在adapter类中再创建一个ViewHolder的内部类。我们要做的,正是对这两个类进行封装,让其实现itemView点击事件、长按事件、子项点击事件。
首先,我的处理方式是,对于开发者来说,只需要对adapter进行setxxx()方法的调用,例如设置itemView的点击事件:adapter.setOnRecyclerViewItemClickListener(...);对该方法传入自定义的接口即可。也就是说,我们需要自定义一个adapter类。那我们就先创建一个类,命名为BaseRecylerAdapter,此后,我们也应当创建一个BaseViewHolder类,接下来开始搞事情。
BaseRecylerAdapter类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
|
public abstract class BaseRecyclerAdapter extends RecyclerView.Adapter<BaseViewHolder> implements View.OnClickListener ,View.OnLongClickListener { private OnRecyclerViewItemClickListener onRecyclerViewItemClickListener; private OnRecyclerViewItemLongClickListener onRecyclerViewItemLongClickListener; private OnSubViewClickListener onSubViewClickListener; @Override public void onBindViewHolder(BaseViewHolder holder, int position) { holder.itemView.setTag(position); holder.onBind(position); if (onRecyclerViewItemClickListener != null ) { holder.itemView.setOnClickListener( this ); } if (onRecyclerViewItemLongClickListener != null ) { holder.itemView.setOnClickListener( this ); } if (onSubViewClickListener != null ) { holder.setSubViewClickListener(onSubViewClickListener,position); } } public void setOnRecyclerViewItemClickListener(OnRecyclerViewItemClickListener onRecyclerViewItemClickListener) { this .onRecyclerViewItemClickListener = onRecyclerViewItemClickListener; } public void setOnRecyclerViewItemLongClickListener(OnRecyclerViewItemLongClickListener onRecyclerViewItemLongClickListener) { this .onRecyclerViewItemLongClickListener = onRecyclerViewItemLongClickListener; } public void setOnSubViewClickListener(OnSubViewClickListener listener){ this .onSubViewClickListener = listener; } @Override public void onClick(View v) { if (v.getTag() != null ) { int position = ( int ) v.getTag(); onRecyclerViewItemClickListener.onItemClick(position); } } @Override public boolean onLongClick(View v) { if (v.getTag() != null ){ int position = ( int )v.getTag(); onRecyclerViewItemLongClickListener.onItemLongClick(position); } return true ; } public interface OnRecyclerViewItemClickListener { void onItemClick( int position); } public interface OnSubViewClickListener{ void onSubViewClick(View v, int position); } public interface OnRecyclerViewItemLongClickListener { void onItemLongClick( int position); } } |
可以看到我们在类中创建了三个接口类
1
2
3
4
5
6
7
8
9
10
11
|
public interface OnRecyclerViewItemClickListener { void onItemClick( int position); } public interface OnSubViewClickListener{ void onSubViewClick(View v, int position); } public interface OnRecyclerViewItemLongClickListener { void onItemLongClick( int position); } |
这三个接口便是用于点击事件的回调,看名字就能分别出各自的功能。itemView的点击回调public interface OnRecyclerViewItemClickListener
,itemView的长按public interface OnRecyclerViewItemLongClickListener
,子项View的点击回调public interface OnSubViewClickListener
。都是点击事件的处理,没有点击发送怎么行呢,对吧!所以,这个类还实现了View.OnClickListener
和View.OnLongClickListener
这两个接口,本别实现itemView的点击事件和长按事件。
可以看到,BaseRecyclerAdapter继承自RecyclerView.Adapter<BaseViewHolder>
,此时我们只需要实现onBindViewHolder
这个方法即可。来分析这个方法。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
@Override public void onBindViewHolder(BaseViewHolder holder, int position) { holder.itemView.setTag(position); holder.onBind(position); if (onRecyclerViewItemClickListener != null ) { holder.itemView.setOnClickListener( this ); } if (onRecyclerViewItemLongClickListener != null ) { holder.itemView.setOnClickListener( this ); } if (onSubViewClickListener != null ) { holder.setSubViewClickListener(onSubViewClickListener,position); } } |
可以看出 这个方法里都是操作我们自定义的BaseViewHolder类。接下来就是三个空判断,也就是说,我们若是没有设置相应的点击事件,就不会初始化对应的点击事件,这样的处理方式还是很常见的。处理这个点击事件最麻烦的就是position的问题,因此我们使用的技巧是,对View对象设置tag的方式。查看源码便知道,View有个方法 setTag(Object obj)
; 我们就可以将对应的position赋值给这个tag,我们使用View的getTag()
方法就可以得到对应点击View的position了。在BaseRecylerAdapter类实现的点击接口和长按接口就可以知道这样的操作,类容如下。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
@Override public void onClick(View v) { if (v.getTag() != null ) { int position = ( int ) v.getTag(); onRecyclerViewItemClickListener.onItemClick(position); } } @Override public boolean onLongClick(View v) { if (v.getTag() != null ){ int position = ( int )v.getTag(); onRecyclerViewItemLongClickListener.onItemLongClick(position); } return true ; } |
BaseViewHolder类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
|
public abstract class BaseViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener{ private BaseRecyclerAdapter.OnSubViewClickListener onSubViewClickListener; public BaseViewHolder(View itemView) { super (itemView); findViewById(itemView); } /** * 传入子项点击事件所需参数 * @param listener 自定义的接口 * @param tagPosition tag */ public void setSubViewClickListener(BaseRecyclerAdapter.OnSubViewClickListener listener, int tagPosition){ this .onSubViewClickListener = listener; initSubViewClick(tagPosition); } /** * 通过id匹配控件(开发者自行实现) * @param itemView 父布局 */ abstract protected void findViewById(View itemView); /** * 用于装载数据(开发者自行实现) * @param position 当前位置 */ abstract protected void onBind( int position); /** * 初始化子项的点击事件(为子项设置tag) * @param tagPosition tag */ protected void initSubViewClick( int tagPosition){ } /** * 实现子项点击事件的转发 * @param v */ @Override public void onClick(View v) { if (v.getTag() != null ) { int position = ( int ) v.getTag(); onSubViewClickListener.onSubViewClick(v,position); } } } |
这是个抽象类,也就是说,在使用的时候需要实现其中的抽象方法。为了逻辑清晰,我在这里写了两个抽象方法
1
2
3
4
5
6
7
8
9
10
11
|
/** * 通过id匹配控件(开发者自行实现) * @param itemView 父布局 */ abstract protected void findViewById(View itemView); /** * 用于装载数据(开发者自行实现) * @param position 当前位置 */ abstract protected void onBind( int position); |
看注释也就很清楚这两个方法的作用是什么,这里就不多说了。
到此,我们已经实现了itemView的点击和长按事件,接下来我们来实现对itemView子项的点击事件。
在BaseViewHolder类中,也实现了一个View的点击事件接口。子项的点击方式和itemView的点击事件是一样的套路,使用tag。接下来我们来看个例子,就明白了。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
|
public class RecyclerAdapterMyActivity extends BaseRecyclerAdapter{ private List<MyActivityBean> list; public RecyclerAdapterMyActivity(List<MyActivityBean>list){ this .list = list; } @Override public BaseViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_recycler_myactivity_activity,parent, false ); ViewHolder holder = new ViewHolder(view); return holder; } @Override public int getItemCount() { return list.size(); } public class ViewHolder extends BaseViewHolder { private TextView tv_name,tv_title,tv_content; private Button activityBtnChat; private Button activityBtnCancel; ViewHolder(View itemView) { super (itemView); } @Override protected void findViewById(View itemView) { tv_name = itemView.findViewById(R.id.tv_my_activity_name); tv_title = itemView.findViewById(R.id.tv_my_activity_title); tv_content = itemView.findViewById(R.id.tv_my_activity_content); activityBtnChat = itemView.findViewById(R.id.activity_btn_chat); activityBtnCancel = itemView.findViewById(R.id.activity_btn_cancel); } @Override protected void onBind( int position) { MyActivityBean bean = list.get(position); tv_name.setText(bean.getName()); tv_title.setText(bean.getTitle()); tv_content.setText(bean.getContent()); } @Override protected void initSubViewClick( int tagPosition) { activityBtnChat.setTag(tagPosition); activityBtnCancel.setTag(tagPosition); activityBtnChat.setOnClickListener( this ); activityBtnCancel.setOnClickListener( this ); } } } |
这段代码是最近项目中的一小段代码。其中,adapter类继承BaseRecyclerAdapter,viewHolder类继承BaseViewHolder。尤其要注意的是ViewHolder的构造方法中一定要有super(itemView);
其余的方法都会按照正确的逻辑执行。若要实现itemView的子项点击事件,需要重写父类的initSubViewClick(int tagPosition)
; 方法。其中参数tagPosition便是对应的itemVIew处于RecyclerView中的位置。在这里是为两个button添加点击事件,先为其设置tag,再设置点击事件,我们这里的setOnClickListener(this) ;
参数传的是this,是因为,我们再父类中实现了View的onClick(View v);
方法。
这样,我们便完成了各类点击事件。
使用方法也很简单,就是直接操作你的adapter就可以了,调用adapter.setXxxx(...)
即可方便地实现各种点击事件。当然,要是你地需求是Touchu事件,或子项地长按事件等,都可以通过这样类似地方式来实现。
最后
方法不止一种,这样地操作方式,这只是我的一种思考。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:https://www.jianshu.com/p/8991b9226bdf