给RecyclerView封装个Adapter吧(更优雅的添加点击事件)

给RecyclerView封装个Adapter吧(更优雅的添加点击事件)


RecyclerView强大,好用,但是使用率很高的ItemClickListener却没有添加。

- 你想要控制其显示的方式,请通过布局管理器LayoutManager

- 你想要控制Item间的间隔(可绘制),请通过ItemDecoration

- 你想要控制Item增删的动画,请通过ItemAnimator

- 你想要控制点击、长按事件,请自己写(擦,这点尼玛。)

无疑让我们在欣喜中多了一丝疑虑。有的哥们干脆从入门到放弃了。其实这也是这个框架的强大之处——定制。

其中一个原因是RecyclerView的Item支持动画。举个栗子。如果你开启了RecyclerView的Item动画,当你删除了position位于1的Item之后, 此时位于23456...的position会向前一格变成12345,而被删除的那个view也会留在RecycleView上继续执行动画直到结束,于是这时候RecycleView中会留有两个position为1的View....此时点击那个被删除的view的话就蛋疼了。。因为Adapter中已经没有这个item了,如果用position 1去adapter中取的话,得到的是原来positon为2的那个item.....

还有就是RecyclerView负责控制/框架,LayoutManager负责计算布局,假设将ItemClickListener放到RecyclerView上,如果要实现点击事件,首先需要确定每一个item的点击区域。但是RecyclerView无法知道每一个item的点击区域,因为LayoutManager是可以由开发者来实现的,也就是说两个View的区域是允许重叠的。如果点了A和B重叠区域到底是触发A还是B,又必须要由LayoutManager来决定。所以还不如直接放到LayoutManager中,但如果放到LayoutManager中的话需要给RecyclerView添加OnTouchListener,看上去又很别扭(要是外部给RecyclerView设置OnTouchListener会覆盖掉这个导致ItemClickListener失效)。所以干脆交给开发者,你长按或者点击,自己实现。。。

点击事件的实现,有常见的三种方法:

通过 RecyclerView已有的方法 addOnItemTouchListener()实现

在创建 ItemView时添加点击监听

当 ItemViewattachRecyclerView时实现

从以上三种方式的实现过程可知:

三种均可实现 ItemView的点击事件和长按事件的监听.

第一种方式可以很方便获取用户点击的坐标.并且节省资源

第二种和第三种方式可以很方便对 ItemView中的子 View进行监听

第一种方式和第三种方式可以写在单独的类中,相对于第二种写在 Adapter的方式可使代码更独立整洁,

既然RecyclerViewrecycle这么优雅的解耦,我们肯定要选个解耦的方案。而RecycleViewd的API也提供了RecyclerView.addOnItemTouchListener方法。so,我们选择第一种方案:

OnItemTouchListener 源码:

···

public static interface OnItemTouchListener {

     public boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e); 

     public void onTouchEvent(RecyclerView rv, MotionEvent e); 

     public void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept); }

···

此接口还提供了一个实现类,且官方推荐使用该实现类 SimpleOnItemTouchListener:

/**

* An implementation of {@linkRecyclerView.OnItemTouchListener} that has empty method bodies and

* default return values.

*

* You may prefer to extend this class if you don't need to override all methods. Another

* benefit of using this class is future compatibility. As the interface may change, we'll

* always provide a default implementation on this class so that your code won't break when

* you update to a new version of the support library.

*/

public static class SimpleOnItemTouchListener implements RecyclerView.OnItemTouchListener { 

    @Override 

    public

boolean onInterceptTouchEvent(RecyclerViewrv,MotionEvente){returnfalse;}

@Override

public void onTouchEvent(RecyclerViewrv,MotionEvente){}

@Override

public void onRequestDisallowInterceptTouchEvent(booleandisallowIntercept){}}

在触摸接口中,当触摸时会回调一个 MotionEvent对象,看到MotionEvent然后想到了什么?通过使用 GestureDetectorCompat来解析用户的操作!GestureDetectorCompat就是处理手势的类:手势探测器,它比GestureDetector能更好兼容低版本的api,但使用方法是一致的。而GestureDetectorCompat还提供了一个外部类SimpleOnGestureListener,这个类实现了上面GestureDetectorCompat接口的所有方法,但全都是空实现,这样继承SimpleOnGestureListener的时候就不用实现每一个方法了,我们需要点击和长按,只需要实现:

boolean onSingleTapUp(MotionEvent e)voidonLongPress(MotionEvent e)

然后根据点击的坐标得到viewhoder:

View childView=recyclerView.findChildViewUnder(e.getX(),e.getY());if(childView!=null){RecyclerView.ViewHoldervh=recyclerView.getChildViewHolder(childView);onItemClick(vh);//点击回掉}

全部代码:

public abstract class OnRecyclerItemClickListener extends RecyclerView.SimpleOnItemTouchListener{

private GestureDetectorCompatmGestureDetector;

private RecyclerViewrecyclerView;

public OnRecyclerItemClickListener(finalRecyclerViewrecyclerView){

this.recyclerView=recyclerView;

mGestureDetector=new GestureDetectorCompat(recyclerView.getContext(),newGestureDetector.SimpleOnGestureListener(){

@Override

public boolean onSingleTapUp(MotionEvente){

View childView=recyclerView.findChildViewUnder(e.getX(),e.getY());if(childView!=null){

RecyclerView.ViewHoldervh=recyclerView.getChildViewHolder(childView);

onItemClick(vh);}

return true;}

@Override

public void onLongPress(MotionEvente){

View childView=recyclerView.findChildViewUnder(e.getX(),e.getY());

if(childView!=null){

RecyclerView.ViewHoldervh=recyclerView.getChildViewHolder(childView);onItemLongClick(vh);}}});}//点击事件交给mGestureDetector处理@Override public boolean onInterceptTouchEvent(RecyclerViewrv,MotionEvente){mGestureDetector.onTouchEvent(e);returnfalse;}//点击回掉public abstract void onItemClick(RecyclerView.ViewHoldervh);//长按监听public abstract void onItemLongClick(RecyclerView.ViewHoldervh);}

使用:

recyclerView.addOnItemTouchListener(newOnRecyclerItemClickListener(recyclerView){

@Override 

public void onItemClick(RecyclerView.ViewHoldervh){

intadapterPosition=vh.getAdapterPosition();//当前item的位置Log.i(TAG,"onItemClick: "+adapterPosition);}

@Override

public void onItemLongClick(RecyclerView.ViewHoldervh){Log.i(TAG,"onItemLongClick: ");}});

你可能感兴趣的:(给RecyclerView封装个Adapter吧(更优雅的添加点击事件))