先看这篇文章:
RecyclerView的滚动事件分析
假设现在我们要获取RecyclerView中指定位置(position)的ItemView,大部分的文章是这么建议的:
int position;
LinearLayoutManager layoutMgr;
int firstPosition = layoutMgr.findFirstVisibleItemPosition();
View v = layoutMgr.getChildAt(position - firstPosition);
基本的思路是,先获取当前Window中第一个可见Item的position,然后计算指定位置距离这个可见位置的偏移量,最后通过getChildAt获取指定位置的ItemView。
在API 25.1.0中,我们发现LayoutManager计算逻辑并不是这样的。layoutMgr.findFirstVisibleItemPosition()
获取到的位置,实际就是字面的意思,即是LayoutManager在Recycler中找到的第一个可见的Item。特别是如果position不在可视区域,需要RecyclerView先执行一个滚动时,第一个可见的Item不一定是RecyclerView的第一个Child,在计算position-firstPosition时候会产生偏移量错误。
要获取正确的位置,我采用的是这样的逻辑:
- 首先为每一个ItemView setTag,内容是这个Item在RecyclerView当中的position,在RecyclerViewAdapter中:
@Override
public void onBindViewHolder(..., int position){
ViewHolder vHolder;
...
vHolder.getView().setTag(position);
}
- 判断LayoutManager当前Window所显示的Item范围是否包含了指定位置,如果不在范围内,则需要先scroll一下:
RecyclerView recyclerView;
View itemView;
...
int firstPosition = layoutMgr.findFirstVisibleItemPosition();
int lastPosition = layoutMgr.findLastVisibleItemPosition();
if (position < firstPosition || position > lastPosition){
recyclerView.smoothScrollToPosition(position); //滚动到指定的位置
//这里需要处理滚动监听,请看步骤3
}
else {
//直接获取ItemView
itemView = layoutMgr.getChildAt(position - (int) layoutMgr.getChildAt(0).getTag());//通过获取Child(0)的tag得到第一个Child的实际位置
}
- 当执行scroll时,需要判断什么时候scroll执行完毕,RecyclerView处于新的位置。我们需要实现一个滚动监听器和两个状态判断量:
boolean isDragging;//判断scroll是否是用户主动拖拽
boolean isScrolling;//判断scroll是否处于滑动中
recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
int firstPosition;
switch(newState){
case RecyclerView.SCROLL_STATE_SETTLING:
if (!outputPortDragging && !outputPortScrolling){
outputPortScrolling = true; //a scrolling occurs
}
break;
case RecyclerView.SCROLL_STATE_DRAGGING:
outputPortDragging = true; //如果是用户主动滑动recyclerview,则不触发位置计算。
break;
case RecyclerView.SCROLL_STATE_IDLE:
if (!outputPortDragging && outputPortScrolling){
outputPortDragging = false;
outputPortScrolling = false;
firstPosition = outputPortLayoutMgr.findFirstVisibleItemPosition();
int lastPos = outputPortLayoutMgr.findLastVisibleItemPosition();
//N.B.: firstVisibleItemPosition is not the first child of layoutmanager
itemView = layoutMgr.getChildAt(position-(int) outputPortLayoutMgr.getChildAt(0).getTag())); //由于滚动事件会多次触发IDLE状态,我们只需要在第一次IDLE被触发时获取ItemView。
}
break;
}
}