给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时添加点击监听
  • 当 ItemViewattach RecyclerView时实现

从以上三种方式的实现过程可知:
三种均可实现 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 {@link RecyclerView.OnItemTouchListener} that has empty method bodies and
 * default return values.
 * <p>
 * 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(RecyclerView rv, MotionEvent e) {
        return false;
    }

    @Override
    public void onTouchEvent(RecyclerView rv, MotionEvent e) {
    }

    @Override
    public void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept) {
    }
}

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

boolean onSingleTapUp(MotionEvent e)
void onLongPress(MotionEvent e)

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

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

全部代码:

public abstract class OnRecyclerItemClickListener extends RecyclerView.SimpleOnItemTouchListener
{
private GestureDetectorCompat mGestureDetector;
private RecyclerView recyclerView;

public OnRecyclerItemClickListener(final RecyclerView recyclerView)
{
    this.recyclerView = recyclerView;
    mGestureDetector = new GestureDetectorCompat(recyclerView.getContext(), new GestureDetector.SimpleOnGestureListener()
    {
        @Override
        public boolean onSingleTapUp(MotionEvent e)
        {
            View childView = recyclerView.findChildViewUnder(e.getX(), e.getY());
            if (childView != null)
            {
                RecyclerView.ViewHolder vh = recyclerView.getChildViewHolder(childView);
                onItemClick(vh);
            }
            return true;
        }

        @Override
        public void onLongPress(MotionEvent e)
        {
            View childView = recyclerView.findChildViewUnder(e.getX(), e.getY());
            if (childView != null)
            {
                RecyclerView.ViewHolder vh = recyclerView.getChildViewHolder(childView);
                onItemLongClick(vh);
            }
        }
    });
}
 //点击事件交给mGestureDetector处理
@Override
public boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e)
{
    mGestureDetector.onTouchEvent(e);
    return false;
}
    //点击回掉
public abstract void onItemClick(RecyclerView.ViewHolder vh);
    //长按监听
public abstract void onItemLongClick(RecyclerView.ViewHolder vh);
}

使用:

    recyclerView.addOnItemTouchListener(new OnRecyclerItemClickListener(recyclerView)
    {
        @Override
        public void onItemClick(RecyclerView.ViewHolder vh)
        {
            int adapterPosition = vh.getAdapterPosition();//当前item的位置
            Log.i(TAG, "onItemClick: "+adapterPosition);
        }

        @Override
        public void onItemLongClick(RecyclerView.ViewHolder vh)
        {
            Log.i(TAG, "onItemLongClick: ");
        }
    });

传送门

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,716评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,558评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,431评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,127评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,511评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,692评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,915评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,664评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,412评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,616评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,105评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,424评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,098评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,096评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,869评论 0 197
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,748评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,641评论 2 271

推荐阅读更多精彩内容