美文网首页Android开发程序员Android进阶之旅
RecyclerView 滚动和获取指定位置Item的完整方案

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

作者: 凌峰 | 来源:发表于2017-03-09 09:56 被阅读21305次

    先看这篇文章:
    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时候会产生偏移量错误。

    要获取正确的位置,我采用的是这样的逻辑:

    1. 首先为每一个ItemView setTag,内容是这个Item在RecyclerView当中的position,在RecyclerViewAdapter中:
    @Override
    public void onBindViewHolder(..., int position){
    ViewHolder vHolder;
    ...
    vHolder.getView().setTag(position);
    }
    
    1. 判断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的实际位置
    }
    
    1. 当执行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;
                                }
                            }
    

    相关文章

      网友评论

      • 對9當歌:outputPortLayoutMgr 是什么参数
      • 萌萌的白天:就是不知道如何获取position 已经设置tag 跪求
        超饿困因子:github上有源码吗?
        萌萌的白天:@凌峰 谢谢 之前 一已经搞定了
        凌峰:layoutMgr.getChildAt(0) 这是recyclerview窗口中第一个view,.getTag是获取这个view在队列中的实际位置
      • 風的記憶:大佬 。。。position 是怎么来的啊 和楼上的 哥们一样处在懵逼中啊!
        凌峰:position是recycleview中数据的位置,不是显示的位置。如果比如我们想让队列跳到第37项item上,那么position就是37
      • GYLEE:请问一下步骤2中的 position 是怎样获得的,谢谢。
      • 07b0c4d53d10:请问onBindViewHolder 中这么会有ViewHolder呢?不明白甚么意思?
        lsys:方法参数中就有一个holder

      本文标题:RecyclerView 滚动和获取指定位置Item的完整方案

      本文链接:https://www.haomeiwen.com/subject/spqwgttx.html