android 获取 item的位置,RecyclerView 滚动和获取指定位置Item的完整方案

 

假设现在我们要获取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;

}

}

 
posted @ 2022-05-07 17:17  新感觉  阅读(4430)  评论(0编辑  收藏  举报