版权声明:本文源自简书tianma,转载请务必注明出处: http://www.jianshu.com/p/e893e4c569fb
最近想动手实现 RecyclerView
的“自动加载更多”功能,即当 RecyclerView
滑动到底部时,执行加载更多操作。这里的关键在于,需要监听RecyclerView是否滑动到底部。
分析
RecyclerView
有 addOnScrollListener(OnScrollListener)
方法,可以为 RecyclerView
添加滚动监听,其中 OnScrollListener
有两个回调函数:
- onScrollStateChanges(RecyclerView recyclerView, int newState): 在
RecyclerView
的滚动状态发生改变时回调 - onScrolled(RecyclerView recyclerView, int dx, int dy): 在
RecyclerView
滚动时回调
在 OnScrollListener
回调中,可以获取RecyclerView的滚动状态,我们只需要通过继承 OnScrollListener
并复写上面的两个回调函数,便可以实现对 RecyclerView
滑动到底部的监听。
实现
我们先定义一个回调接口 BottomListener
用以监听控件是否滚动到底部:
public interface BottomListener {
/**
* 滑动到底部时回调
*/
void onScrollToBottom();
}
然后,我们继承 RecyclerView.OnScrollListener
和实现 BottomListener
:
import android.support.v7.widget.GridLayoutManager;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.support.v7.widget.StaggeredGridLayoutManager;
/**
* 实现了RecyclerView滚动到底部监听的OnScrollListener
*/
public class RecyclerViewScrollListener extends RecyclerView.OnScrollListener implements BottomListener {
// 最后几个完全可见项的位置(瀑布式布局会出现这种情况)
private int[] lastCompletelyVisiblePositions;
// 最后一个完全可见项的位置
private int lastCompletelyVisibleItemPosition;
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
RecyclerView.LayoutManager layoutManager = recyclerView.getLayoutManager();
// 找到最后一个完全可见项的位置
if (layoutManager instanceof StaggeredGridLayoutManager) {
StaggeredGridLayoutManager manager = (StaggeredGridLayoutManager) layoutManager;
if (lastCompletelyVisiblePositions == null) {
lastCompletelyVisiblePositions = new int[manager.getSpanCount()];
}
manager.findLastCompletelyVisibleItemPositions(lastCompletelyVisiblePositions);
lastCompletelyVisibleItemPosition = getMaxPosition(lastCompletelyVisiblePositions);
} else if (layoutManager instanceof GridLayoutManager) {
lastCompletelyVisibleItemPosition = ((GridLayoutManager) layoutManager).findLastCompletelyVisibleItemPosition();
} else if (layoutManager instanceof LinearLayoutManager) {
lastCompletelyVisibleItemPosition = ((LinearLayoutManager) layoutManager).findLastCompletelyVisibleItemPosition();
} else {
throw new RuntimeException("Unsupported LayoutManager.");
}
}
private int getMaxPosition(int[] positions) {
int max = positions[0];
for (int i = 1; i < positions.length; i++) {
if (positions[i] > max) {
max = positions[i];
}
}
return max;
}
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
RecyclerView.LayoutManager layoutManager = recyclerView.getLayoutManager();
// 通过比对 最后完全可见项位置 和 总条目数,来判断是否滑动到底部
int visibleItemCount = layoutManager.getChildCount();
int totalItemCount = layoutManager.getItemCount();
if (newState == RecyclerView.SCROLL_STATE_IDLE) {
if (visibleItemCount > 0 && lastCompletelyVisibleItemPosition >= totalItemCount - 1) {
onScrollToBottom();
}
}
}
@Override
public void onScrollToBottom() {
}
}
上述大体思路就是: 找到 "最后完全可见项的位置(lastCompletelyVisibleItemPosition)",通过比较 lastCompletelyVisibleItemPosition 是否是 RecyclerView
最后一项,来判断是否滑动到底部。
在 RecyclerView
上应用加载更多:
recyclerView.addOnScrollListener(new RecyclerViewScrollListener() {
@Override
public void onScrollToBottom() {
// 加载更多
doLoadMore();
}
});
// ...
private void doLoadMore() {
// TODO load more
}
参考链接:
通过重写OnScrollListener来监听RecyclerView是否滑动到底部
特别注意
个人认为上面的参考链接中给出的方案是有待商榷的,原博主计算的是 "最后一个可见项的位置(lastVisibleItemPosition)",而本文中计算的是 "最后一个完全可见项的位置(lastCompletelyVisibleItemPosition)" , 很明显 RecyclerView
列表中最后一个元素可见的时候,RecyclerView
并不一定滑动到最底部了。当然了,具体采取哪种方案,需要根据实际需求来确定。