RecyclerView将指定项滑动到顶部显示

前言:

一般我们用 Recycleview.smoothScrollToPosition(0)滑动到顶部,具有滚动效果,但是如果我们想滚动到任意指定位置,那么smoothScrollToPosition()就不能保证所指定item位于屏幕顶部,比如无法将处于屏幕中部的item滑动到屏幕顶部。那么一下提供下我解决的方法。

解决方法按照效果是否平滑分为两种:(至于平滑和非平滑的效果的不同,类比下面效果图便知道)

第一种:非平滑效果的实现方式
    private void moveToPosition(int position) {
        if (position != -1) {
            mRecyclerView.scrollToPosition(position);
            LinearLayoutManager mLayoutManager =
                    (LinearLayoutManager) mRecyclerView.getLayoutManager();
            mLayoutManager.scrollToPositionWithOffset(position, 0);
        }
    }

实现效果:

这里写图片描述

 

第二种:平滑效果的实现方式

首先获取第一个可见位置和最后一个可见位置,分三种情况:
1.如果如果跳转位置在第一个可见位置之前,就smoothScrollToPosition()可以直接跳转;
2.如果跳转位置在第一个可见项之后,最后一个可见项之前smoothScrollToPosition()不会滚动,此时调用smoothScrollBy来滑动到指定位置;
3.如果要跳转的位置在最后可见项之后,则先调用smoothScrollToPosition()将要跳转的位置滚动到可见位置,在addOnScrollListener()里通过onScrollStateChanged控制,调用smoothMoveToPosition,再次执行判断;

    
    //目标项是否在最后一个可见项之后
    private boolean mShouldScroll;
    //记录目标项位置
    private int mToPosition;
    
    /**
     * 使指定的项平滑到顶部
     *
     * @param mRecyclerView
     * @param position      待指定的项
     */
    private void smoothMoveToPosition(RecyclerView mRecyclerView, final int position) {
        int firstItemPosition = -1;
        int lastItemPosition = -1;

        //todo 获取第一个和最后一个可见位置方式1
        // 第一个可见位置
        firstItemPosition = mRecyclerView.getChildLayoutPosition(mRecyclerView.getChildAt(0));
        // 最后一个可见位置
        lastItemPosition = mRecyclerView.getChildLayoutPosition(mRecyclerView.getChildAt(mRecyclerView.getChildCount() - 1));

        //todo 获取第一个和最后一个可见位置方式2
        // 判断是当前layoutManager是否为LinearLayoutManager
        // 只有LinearLayoutManager才有查找第一个和最后一个可见view位置的方法
//        RecyclerView.LayoutManager layoutManager = mRecyclerView.getLayoutManager();
//        if (layoutManager instanceof LinearLayoutManager) {
//            LinearLayoutManager linearManager = (LinearLayoutManager) layoutManager;
//            //获取第一个可见view的位置
//            firstItemPosition = linearManager.findFirstVisibleItemPosition();
//            //获取最后一个可见view的位置
//            lastItemPosition = linearManager.findLastVisibleItemPosition();
//
//        }

        Log.i(TAG, "smoothMoveToPosition: firstItemPosition::" + firstItemPosition + " lastItemPosition::" + lastItemPosition + "\n");

        if (position < firstItemPosition) {
            // 第一种可能:跳转位置在第一个可见位置之前
            mRecyclerView.smoothScrollToPosition(position);
        } else if (position <= lastItemPosition) {
            // 第二种可能:跳转位置在第一个可见位置之后,在最后一个可见项之前
            int movePosition = position - firstItemPosition;
            if (movePosition >= 0 && movePosition < mRecyclerView.getChildCount()) {
                int top = mRecyclerView.getChildAt(movePosition).getTop();
                mRecyclerView.smoothScrollBy(0, top);//dx>0===>向左  dy>0====>向上
            }
        } else {
            // 第三种可能:跳转位置在最后可见项之后
            mRecyclerView.smoothScrollToPosition(position);
            mToPosition = position;
            mShouldScroll = true;
        }
    }

其中用于获取RecyclerView的第一个和最后一个可见项的方式有两种,上述代码中已经贴出来了。顺便在贴一遍

//方式1:
        // 第一个可见位置
        firstItemPosition = mRecyclerView.getChildLayoutPosition(mRecyclerView.getChildAt(0));
        // 最后一个可见位置
        lastItemPosition = mRecyclerView.getChildLayoutPosition(mRecyclerView.getChildAt(mRecyclerView.getChildCount() - 1));


//方式2
  // 判断是当前layoutManager是否为LinearLayoutManager
        // 只有LinearLayoutManager才有查找第一个和最后一个可见view位置的方法
        RecyclerView.LayoutManager layoutManager = mRecyclerView.getLayoutManager();
        if (layoutManager instanceof LinearLayoutManager) {
            LinearLayoutManager linearManager = (LinearLayoutManager) layoutManager;
            //获取第一个可见view的位置
            firstItemPosition = linearManager.findFirstVisibleItemPosition();
            //获取最后一个可见view的位置
            lastItemPosition = linearManager.findLastVisibleItemPosition();

        }

其中,针对第三种情况——跳转的位置在最后可见项之后的情况,还需要添加如下代码:

  //监听事件的设置,仅仅是为了第三种情况,即:要跳转的位置在可见项之后
        mRecyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
            @Override
            public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
                super.onScrollStateChanged(recyclerView, newState);
                if (mShouldScroll && RecyclerView.SCROLL_STATE_IDLE == newState) {//
                    mShouldScroll = false;
                    smoothMoveToPosition(mRecyclerView, mToPosition);
                }
            }
        });

实现效果:
这里写图片描述

至此完结,小伙伴,如有问题请留言。

 

 

参考文章:
RecyclerView滑动到指定位置,并指定位置在顶部

在Android开发中,我们可以使用RecyclerView来实现网格布局,并且可以通过指定的方式滑动指定的position。 首先,我们需确保已经在目的build.gradle文件中添加了RecyclerView的依赖。 在布局文件中,我们将RecyclerView添加到指定的位置。例如,如果我们希望将RecyclerView添加到activity_main.xml文件的某个LinearLayout中,可以使用以下代码: ```xml <LinearLayout ... android:id="@+id/linear_layout" ...> <androidx.recyclerview.widget.RecyclerView android:id="@+id/recycler_view" ... /> </LinearLayout> ``` 在Activity或Fragment中,我们需要将布局文件中的RecyclerView与代码中的RecyclerView关联起来,并为RecyclerView设置LayoutManager和Adapter。LayoutManager决定了RecyclerView的布局方式,可以选择GridLayoutManager来实现网格布局。Adapter负责为RecyclerView提供数据,并控制每个item的显示。 ```java LinearLayout linearLayout = findViewById(R.id.linear_layout); RecyclerView recyclerView = findViewById(R.id.recycler_view); GridLayoutManager layoutManager = new GridLayoutManager(this, 2); // 2代表每行显示2个item recyclerView.setLayoutManager(layoutManager); CustomAdapter adapter = new CustomAdapter(dataList); // 自定义Adapter,提供数据 recyclerView.setAdapter(adapter); ``` 接下来,我们可以通过RecyclerView的smoothScrollToPosition()方法来平滑地滑动指定的position。例如,如果我们想要滑动到第10个item,可以使用以下代码: ```java recyclerView.smoothScrollToPosition(9); // RecyclerView中position从0开始计数 ``` 这个方法会使RecyclerView平稳地滑动指定位置,并且会自动滑动到该的前台,以便用户能够看到该。 总结来说,我们可以通过在布局文件中添加RecyclerView并设置LayoutManager和Adapter来实现网格布局。然后,通过RecyclerView的smoothScrollToPosition()方法,我们可以滑动指定的position。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值