RecyclerView的滚动事件分析
来源:互联网 发布:数据质量维度有效性 编辑:程序博客网 时间:2024/05/18 10:08
列表的滚动一般分为两种:
手指按下 -> 手指拖拽列表移动 -> 手指停止拖拽 -> 抬起手指
手指按下 -> 手指快速拖拽后抬起手指 -> 列表继续滚动 -> 停止滚动
从上面可以看出,滚动状态分为:
|--静止|--滚动 |--被迫拖拽移动 |--自己滚动
上面的过程的状态变化如下:
静止 -> 被迫拖拽移动 -> 静止
静止 -> 被迫拖拽移动 -> 自己滚动 -> 静止
<!--more-->
监听RecyclerView的滚动
好了,我们分析完滚动的过程,再看看如何监听RecyclerView的滚动.查看源码是最好的方法.
看源码
查看RecyclerView的源码,我们可以看到以下代码:
/** * Set a listener that will be notified of any changes in scroll state or position. * @param listener Listener to set or null to clear * @deprecated Use {@link #addOnScrollListener(OnScrollListener)} and * {@link #removeOnScrollListener(OnScrollListener)} */@Deprecatedpublic void setOnScrollListener(OnScrollListener listener) { mScrollListener = listener;}/** * Add a listener that will be notified of any changes in scroll state or position. * <p>Components that add a listener should take care to remove it when finished. * Other components that take ownership of a view may call {@link #clearOnScrollListeners()} * to remove all attached listeners.</p> * @param listener listener to set or null to clear */public void addOnScrollListener(OnScrollListener listener) { if (mScrollListeners == null) { mScrollListeners = new ArrayList<>(); } mScrollListeners.add(listener);}
也就是说有两种方式可以监听滚动事件:
其中 setOnScrollListener 已经过时(
设置的监听器源码如下:
public abstract static class OnScrollListener { /** * Callback method to be invoked when RecyclerView's scroll state changes. * @param recyclerView The RecyclerView whose scroll state has changed. * @param newState The updated scroll state. One of {@link #SCROLL_STATE_IDLE}, * {@link #SCROLL_STATE_DRAGGING} or {@link #SCROLL_STATE_SETTLING}. */ public void onScrollStateChanged(RecyclerView recyclerView, int newState){} /** * Callback method to be invoked when the RecyclerView has been scrolled. This will be * called after the scroll has completed. * <p> * This callback will also be called if visible item range changes after a layout * calculation. In that case, dx and dy will be 0. * * @param recyclerView The RecyclerView which scrolled. * @param dx The amount of horizontal scroll. * @param dy The amount of vertical scroll. */ public void onScrolled(RecyclerView recyclerView, int dx, int dy){}}
在滚动过程中,此监听器会回调两个方法.
onScrollStateChanged : 滚动状态变化时回调
onScrolled : 滚动时回调这两者的区别在于: 状态与过程
举例子
注 : 以下源码可在最后的地址中找到.
demoRv = (RecyclerView) findViewById(R.id.demo_rv);layoutManager = new LinearLayoutManager(this);demoRv.setLayoutManager(layoutManager);demoRv.addItemDecoration(new DividerItemDecoration(this, DividerItemDecoration.VERTICAL_LIST));bookAdapter = new BookAdapter();bookAdapter.fillList(MockService.getBookList());demoRv.setAdapter(bookAdapter);demoRv.addOnScrollListener(new RecyclerView.OnScrollListener() { @Override public void onScrollStateChanged(RecyclerView recyclerView, int newState) { super.onScrollStateChanged(recyclerView, newState); Log.i(TAG, "-----------onScrollStateChanged-----------"); Log.i(TAG, "newState: " + newState); } @Override public void onScrolled(RecyclerView recyclerView, int dx, int dy) { super.onScrolled(recyclerView, dx, dy); Log.i(TAG, "-----------onScrolled-----------"); Log.i(TAG, "dx: " + dx); Log.i(TAG, "dy: " + dy); Log.i(TAG, "CHECK_SCROLL_UP: " + recyclerView.canScrollVertically(TAG_CHECK_SCROLL_UP)); Log.i(TAG, "CHECK_SCROLL_DOWN: " + recyclerView.canScrollVertically(TAG_CHECK_SCROLL_DOWN)); }});
以上代码中输出了主要个几个信息:
newState : 目前的状态
dx : 水平滚动距离
dy : 垂直滚动距离
onScrollStateChanged 方法
在
recyclerView : 当前在滚动的RecyclerView
newState : 当前滚动状态.
其中newState有三种值:
//停止滚动public static final int SCROLL_STATE_IDLE = 0;//正在被外部拖拽,一般为用户正在用手指滚动public static final int SCROLL_STATE_DRAGGING = 1;//自动滚动开始public static final int SCROLL_STATE_SETTLING = 2;
onScrolled 方法
在
recyclerView : 当前滚动的view
dx : 水平滚动距离
dy : 垂直滚动距离
真机实践
运行代码
运行以上代码,然后按照上面的滚动过程分别进行两种滚动.
第一种方式缓慢滚动结果如下:
I/MainActivity: -----------onScrollStateChanged-----------I/MainActivity: newState: 1I/MainActivity: -----------onScrolled-----------I/MainActivity: dx: 0I/MainActivity: dy: -6I/MainActivity: CHECK_SCROLL_UP: trueI/MainActivity: CHECK_SCROLL_DOWN: true------------------------n个onScrolled--------------------I/MainActivity: -----------onScrolled-----------I/MainActivity: dx: 0I/MainActivity: dy: -2I/MainActivity: CHECK_SCROLL_UP: trueI/MainActivity: CHECK_SCROLL_DOWN: falseI/MainActivity: -----------onScrollStateChanged-----------I/MainActivity: newState: 0
第二种快速滚动结果如下:
I/MainActivity: -----------onScrollStateChanged-----------I/MainActivity: newState: 1I/MainActivity: -----------onScrolled-----------I/MainActivity: dx: 0I/MainActivity: dy: 59I/MainActivity: CHECK_SCROLL_UP: trueI/MainActivity: CHECK_SCROLL_DOWN: true--------------------------n个onScrolled-------------------I/MainActivity: -----------onScrolled-----------I/MainActivity: dx: 0I/MainActivity: dy: 54I/MainActivity: CHECK_SCROLL_UP: trueI/MainActivity: CHECK_SCROLL_DOWN: trueI/MainActivity: -----------onScrollStateChanged-----------I/MainActivity: newState: 2I/MainActivity: -----------onScrolled-----------I/MainActivity: dx: 0I/MainActivity: dy: 56I/MainActivity: CHECK_SCROLL_UP: trueI/MainActivity: CHECK_SCROLL_DOWN: true--------------------------n个onScrolled-------------------I/MainActivity: -----------onScrolled-----------I/MainActivity: dx: 0I/MainActivity: dy: 14I/MainActivity: CHECK_SCROLL_UP: trueI/MainActivity: CHECK_SCROLL_DOWN: trueI/MainActivity: -----------onScrolled-----------I/MainActivity: dx: 0I/MainActivity: dy: 1I/MainActivity: CHECK_SCROLL_UP: trueI/MainActivity: CHECK_SCROLL_DOWN: trueI/MainActivity: -----------onScrollStateChanged-----------I/MainActivity: newState: 0
分析结果
且在滚动过程中发现:
1.滚动方向
dy > 0 时为向上滚动
dy < 0 时为向下滚动
2.回调过程
缓慢拖拽回调过程:
1. newState = RecyclerView.SCROLL_STATE_DRAGGING;2. dy 多次改变3. newState = RecyclerView.SCROLL_STATE_IDLE
快速滚动回调过程:
1. newState = RecyclerView.SCROLL_STATE_DRAGGING;2. dy 多次改变3. newState = RecyclerView.SCROLL_STATE_SETTLING;4. dy 多次改变5. newState = RecyclerView.SCROLL_STATE_IDLE;
3.顶端与底部
以上信息中还打印了
RecyclerView.canScrollVertically(-1)
的值表示是否滚动到顶部
封装
基于以上,我们可以封装一个可以回调滚动状态和方向的RecyclerView.
先建立事件监听的接口publicinterfaceOnScrollCallback { void onStateChanged(ScrollRecycler recycler, int state); voidonScrollUp(ScrollRecycler recycler,int dy); void onScrollDown(ScrollRecycler recycler, int dy); }
再写一个类
RecyclerView
,在类中添加以下方法:
public void setOnScrollCallback(final OnScrollCallback callback) { if (callback == null) { return; } addOnScrollListener(new OnScrollListener() { @Override public void onScrollStateChanged(RecyclerView recyclerView, int newState) { super.onScrollStateChanged(recyclerView, newState); callback.onStateChanged(ScrollRecycler.this, newState); } @Override public void onScrolled(RecyclerView recyclerView, int dx, int dy) { super.onScrolled(recyclerView, dx, dy); if (dy > 0) { callback.onScrollDown(ScrollRecycler.this, dy); } else { callback.onScrollUp(ScrollRecycler.this, dy); } } });}
作者: 一点点征服
出处:http://www.cnblogs.com/ldq2016/
本文版权归作者所有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接,否则保留追究法律责任的权利
- RecyclerView的滚动事件分析
- RecyclerView的滚动事件分析
- RecyclerView的滚动事件分析
- RecyclerView滚动事件分析
- RecyclerView的滚动事件研究
- RecyclerView的滚动事件OnScrollListener研究
- RecyclerView滚动事件
- 居中滚动的Recyclerview
- RecyclerView smoothScrollToPosition的滚动时间
- 强大的滚动控件RecyclerView
- RecyclerView的点击事件
- RecyclerView的事件监听
- RecyclerView的点击事件
- RecyclerView的点击事件
- RecyclerView的点击事件
- RecyclerView的点击事件
- RecyclerView 的点击事件!
- RecyclerView的点击事件
- <Notepad++>FingerText插件创建自己的代码块
- 用Swing开发GUI程序(五)JScrollpane
- 二叉树递归,非递归遍历二叉树
- 简单排序算法汇总(算法四版)
- 《剑指offer》刷题笔记(递归和循环):跳台阶
- RecyclerView的滚动事件分析
- 转载: Java ConcurrentModificationException异常原因和解决方法
- MYSQL创建用户与授权
- 为什么java的构造方法中this()或者super()要放在第一行
- Gabage Collection (GC)
- 统计汉字,字母,数字等字符
- JavaScript从入门到放弃(三)
- 深度学习中的欠拟合和过拟合简介
- 获得程序当前路径System.getProperty("user.dir")