当前位置:   article > 正文

android v7兼容包RecyclerView的使用(四)——点击事件的不同方式处理

v7添加两种方式

前三篇文章
android v7兼容包RecyclerView的使用(三)——布局管理器的使用
android v7兼容包RecyclerView的使用(二)
android v7兼容包RecyclerView的使用(一)
介绍了RecyclerView的使用以及常见的相关类和布局管理器的灵活之处。写了这么多篇,还没涉及到用户交互,那么怎么处理点击事件呢。

RecyclerView中你会惊奇的发现,该类中并没有OnItemClickListener监听器监听我们的单击事件,也没有OnItemLongClickListener监听器监听我们的长按事件。取而代之的是OnItemTouchListener监听器,那么该怎样实现我们的点击事件和长按事件呢。

我们的代码是基于上篇文章布局管理器的代码,在其基础上加入事件监听。

假设让我们自己来事件点击事件,我们比方会使用ViewHolder来间接处理事件。首先在适配器中添加监听器接口。

  1. interface OnItemClickListener {
  2. void onClick(View v);
  3. }
  4. interface OnItemLongClickListener {
  5. void onLongClick(View v);
  6. }
  7. private OnItemClickListener onClickListener;
  8. private OnItemLongClickListener onLongClickListener;
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

然后重载构造函数,使其能够接收监听器实例

  1. public CardViewAdapter(String[] data) {
  2. this(data, null, null);
  3. }
  4. public CardViewAdapter(String[] data, OnItemClickListener onClickListener) {
  5. this(data, onClickListener, null);
  6. }
  7. public CardViewAdapter(String[] data, OnItemClickListener onClickListener,
  8. OnItemLongClickListener onLongClickListener) {
  9. this.data = data;
  10. this.onClickListener = onClickListener;
  11. this.onLongClickListener = onLongClickListener;
  12. }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14

终于会调用三參数的构造器,在该构造器内完毕赋值。

将原来的ViewHolder构造函数进行改造,使其处理点击事件,当然也能够直接在onCreateViewHolder函数里处理点击事件。

  1. public ViewHolder(View itemLayoutView,
  2. final OnItemClickListener onClickListener,
  3. final OnItemLongClickListener onLongClickListener) {
  4. super(itemLayoutView);
  5. info = (TextView) itemLayoutView.findViewById(R.id.info_text);
  6. itemLayoutView.setOnClickListener(new OnClickListener() {
  7. @Override
  8. public void onClick(View v) {
  9. //在监听器不为空的时候,进行回调
  10. if (onClickListener != null) {
  11. onClickListener.onClick(v);
  12. }
  13. }
  14. });
  15. itemLayoutView.setOnLongClickListener(new OnLongClickListener() {
  16. @Override
  17. public boolean onLongClick(View v) {
  18. //在监听器不为空的时候,进行回调
  19. if (onLongClickListener != null) {
  20. onLongClickListener.onLongClick(v);
  21. }
  22. //返回true,消费掉该事件,阻止其继续传递
  23. return true;
  24. }
  25. });
  26. }
  • 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

细致一看,事实上代码还是挺多的,那么让我们调用一下。

  1. mAdapter = new CardViewAdapter(data,new OnItemClickListener() {
  2. @Override
  3. public void onClick(View v) {
  4. TextView info = (TextView) v.findViewById(R.id.info_text);
  5. Toast.makeText(getApplicationContext(), "单击"+info.getText(), Toast.LENGTH_LONG).show();
  6. }
  7. },new OnItemLongClickListener() {
  8. @Override
  9. public void onLongClick(View v) {
  10. TextView info = (TextView) v.findViewById(R.id.info_text);
  11. Toast.makeText(getApplicationContext(), "长按"+info.getText(), Toast.LENGTH_LONG).show();
  12. }
  13. });
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

后面两个參数能够传空值,代表不设置监听器。
执行效果图例如以下。
RecyclerView点击事件
我们会发现,上面的代码耦合性还是有点高,事件直接与适配器发生了耦合,除此之外,我们还应该有更好的方法来处理这个点击事件。是的,不是有OnItemTouchListener监听器吗,再配合手势不就能够吗。
好了,看代码吧,详解在凝视中。

  1. package cn.edu.zafu.layoutmanager;
  2. import android.content.Context;
  3. import android.support.v7.widget.RecyclerView;
  4. import android.view.GestureDetector;
  5. import android.view.MotionEvent;
  6. import android.view.View;
  7. /**
  8. * 监听器,实现OnItemTouchListener接口
  9. *
  10. * @author lizhangqu
  11. *
  12. * 2015-3-12
  13. */
  14. public class RecyclerItemClickListener implements
  15. RecyclerView.OnItemTouchListener {
  16. private OnItemClickListener mListener;
  17. private GestureDetector mGestureDetector;
  18. // 点击回调
  19. public interface OnItemClickListener {
  20. public void onItemClick(View view, int position);
  21. public void onItemLongClick(View view, int position);
  22. }
  23. public RecyclerItemClickListener(Context context,
  24. final RecyclerView recyclerView, OnItemClickListener listener) {
  25. mListener = listener;
  26. // 识别并处理手势
  27. mGestureDetector = new GestureDetector(context,
  28. new GestureDetector.SimpleOnGestureListener() {
  29. @Override
  30. public boolean onSingleTapUp(MotionEvent e) {
  31. // 轻击触摸屏后,弹起,必须返回true,否则无法触发单击
  32. return true;
  33. }
  34. @Override
  35. public void onLongPress(MotionEvent e) {
  36. // 长按
  37. // 依据findChildViewUnder(float x, float y)来算出哪个item被选择了
  38. View childView = recyclerView.findChildViewUnder(
  39. e.getX(), e.getY());
  40. // 有item被选则且监听器不为空触发长按事件
  41. if (childView != null && mListener != null) {
  42. mListener.onItemLongClick(childView,
  43. recyclerView.getChildPosition(childView));
  44. }
  45. }
  46. });
  47. }
  48. @Override
  49. public boolean onInterceptTouchEvent(RecyclerView view, MotionEvent e) {
  50. View childView = view.findChildViewUnder(e.getX(), e.getY());
  51. if (childView != null && mListener != null
  52. && mGestureDetector.onTouchEvent(e)) {
  53. // 触发单击事件
  54. mListener.onItemClick(childView, view.getChildPosition(childView));
  55. return true;
  56. }
  57. return false;
  58. }
  59. @Override
  60. public void onTouchEvent(RecyclerView view, MotionEvent motionEvent) {
  61. }
  62. }
  • 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
  • 66
  • 67
  • 68
  • 69
  • 70

在activity中调用

  1. mRecyclerView.addOnItemTouchListener(new RecyclerItemClickListener(getApplicationContext(), mRecyclerView, new OnItemClickListener() {
  2. @Override
  3. public void onItemLongClick(View view, int position) {
  4. Toast.makeText(getApplicationContext(), "长按"+data[position], Toast.LENGTH_SHORT).show();
  5. }
  6. @Override
  7. public void onItemClick(View view, int position) {
  8. Toast.makeText(getApplicationContext(), "短按"+data[position], Toast.LENGTH_SHORT).show();
  9. }
  10. }));
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

非常明显,另外一种方式与适配器进行了解耦。应该说优于第一种方法。
那么还有没有方法处理点击事件呢。让我们从万能的github上搜索一下。
https://github.com/lucasr/twoway-view
在该项目的sample文件夹下有个类里面有这样一段代码

  1. final ItemClickSupport itemClick = ItemClickSupport.addTo(mRecyclerView);
  2. itemClick.setOnItemClickListener(new OnItemClickListener() {
  3. @Override
  4. public void onItemClick(RecyclerView parent, View child, int position, long id) {
  5. mToast.setText("Item clicked: " + position);
  6. mToast.show();
  7. }
  8. });
  9. itemClick.setOnItemLongClickListener(new OnItemLongClickListener() {
  10. @Override
  11. public boolean onItemLongClick(RecyclerView parent, View child, int position, long id) {
  12. mToast.setText("Item long pressed: " + position);
  13. mToast.show();
  14. return true;
  15. }
  16. });
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18

上述代码有个ItemClickSupport 类,对的,该类就是提供事件的支持。那么该类在哪里呢,事实上它在该项目的core文件夹下。我直接将其该文件夹下的代码拷至我的项目的包中,删除一个不相关的类,就直接使用了,当然还须要拷一个资源文件就是ids.xml

实际执行效果呢,是跟上面两种方式是一样的,那么它的实现由什么差别呢,事实上与另外一种方式没什么大的差别,基本上就是OnItemTouchListener 加手势实现的,仅仅只是其逻辑可能更加严谨,设计更加优秀罢了。除此之外,该文件夹下另一个ItemSelectionSupport类,该类提供了item选择的功能,提供了单选多选方式,然而呢,在我測试时发现存在bug,所以呢,这个类的使用就跳过了。

至此,RecyclerView的点击事件就处理完了。

源代码下载
http://download.csdn.net/detail/sbsujjbcy/8495337

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Cpp五条/article/detail/286669
推荐阅读
相关标签
  

闽ICP备14008679号