RecycleView滑动锚点定位

RecycleView滑动,顶部tab跟着变化位置,类似于淘宝详情页,效果是这样的


Tab跟随RecycleView滑动

1.实现思路

简单讲就是监听RecycleView的滑动,根据滑动调整tab位置。相应的监听tab点击事件,滑动recycleView到相应位置。

2.滑动RecycleView调整tab位置

2.1获得滑动位置

LayoutManager提供了获得首个可见item的方法
int position = layoutManager.findFirstVisibleItemPosition();

2.2将滑动位置转换为类型位置

这里我item类型有普通类型和大类的标题类型,通过instanceof判断类型是否是我所属的大类的标题类型,然后根据大类标题类型到顶部的距离判断,再根据类型名称进行定位。
然后通过getTop获得到顶部的距离,根据距离进行分类间的跳转。
int top = mRecyclerView.getChildAt(0).getTop();

2.3完整代码

mRecyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
            @Override
            public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
                super.onScrolled(recyclerView, dx, dy);
                //不捕获自动滚动,非用户触摸产生的滑动
                if (isFormTabScroll) {
                    return;
                }

                //监听滑动距离,改变tab选中位置
                int position = layoutManager.findFirstVisibleItemPosition();
                int realPosition = position - mAdapter.getHeaderSize();

                //因为有1个header所以>=-1
                if (realPosition >= -1 && realPosition < mAdapter.getList().size()) {
                    //通过判断分类卡片位置定位tab
                    if (mealCardIndxList.size() == 0) {//记录大类的流量卡片位置
                        for (int i = 0; i < mAdapter.getList().size(); i++) {
                            if (mAdapter.getList().get(i) instanceof CardFlowMeal) {
                                mealCardIndxList.add(i);
                            }
                        }
                    }

                    if (mealCardIndxList.size() + 1 == indicator.getTitles().size()) {//防止下标越界异常
                        //判断当前item所处类定位tab
                        for (int i = 0; i < mealCardIndxList.size(); i++) {
                            if (i == 0) {
                                if (realPosition >= -1 && realPosition < mealCardIndxList.get(i)) {
                                    if (indicator.getCurrentPosition() != 0) {
                                        indicator.setCurrentItem(0);
                                        break;
                                    }
                                }
                            } else if (realPosition >= mealCardIndxList.get(i - 1) && realPosition < mealCardIndxList.get(i)) {
                                if (indicator.getCurrentPosition() != i) {
                                    indicator.setCurrentItem(i);
                                    break;
                                }
                            }
                            //可能会有i==0的情况下属于如下情况 所以不加else if
                            if (i == mealCardIndxList.size() - 1) {
                                if (realPosition >= mealCardIndxList.get(i)) {
                                    if (indicator.getCurrentPosition() != i + 1) {
                                        indicator.setCurrentItem(i + 1);
                                    }
                                }
                            }
                        }
                    }
                }
            }

            @Override
            public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
                super.onScrollStateChanged(recyclerView, newState);

                switch (newState) {
                    case RecyclerView.SCROLL_STATE_DRAGGING:
                        isFormTabScroll = false;
                        break;
                    case RecyclerView.SCROLL_STATE_IDLE:
                        if (flowDetailActivity != null && !flowDetailActivity.isCloseNps()) {

                            UIUtils.postDelayRunnable(new Runnable() {
                                @Override
                                public void run() {
                                    if (!flowDetailActivity.isShowNps()) {
                                        flowDetailActivity.showNPS();
                                    }
                                }
                            }, 100);
                        }
                        break;
                    default:
                        break;
                }
            }
        });

3.点击tab位置RecycleView滑动到相应位置

通过RecycleView的scrollBy()scrollToPosition()两个方法实现。为scrollToPosition()只会保证滑动的position出现在视野中,不会保证该position在顶端,所以需要通过scrollBy()完成置顶的滑动。

    public void moveRecycleViewToPosition(int n) {
        isFormTabScroll = true;

        //先从RecyclerView的LayoutManager中获取第一项和最后一项的Position
        int firstItem = layoutManager.findFirstVisibleItemPosition();
        int lastItem = layoutManager.findLastVisibleItemPosition();
        //然后区分情况
        if (n < firstItem) {
            //当要置顶的项在当前显示的第一个项的前面时
            mRecyclerView.scrollToPosition(n);
            if (n != 0) {
                mIndex = n;
                needMoveToTop = true;
            }
        } else if (n <= lastItem) {
            //当要置顶的项已经在屏幕上显示时
            int top = mRecyclerView.getChildAt(n - firstItem).getTop();
            mRecyclerView.scrollBy(0, top);
        } else {
            //当要置顶的项在当前显示的最后一项的后面时
            mRecyclerView.scrollToPosition(n);
            //这里这个变量是用在RecyclerView滚动监听里面的
            mIndex = n;
            needMoveToTop = true;
        }
    }

可以看到在该position不在屏幕中时,指定了一个字段needMoveToTop = true,因为需要在scrollToPosition()方法后再执行scrollBy()完成置顶操作。这个标记是监听RecycleView滑动完成之后用来标识完成scrollBy()最后置顶滑动的,具体代码如下:

        mRecyclerView.getViewTreeObserver().addOnScrollChangedListener(new ViewTreeObserver.OnScrollChangedListener() {
            @Override
            public void onScrollChanged() {

                //将改item滑动到顶部
                if (needMoveToTop) {
                    needMoveToTop = false;
                    //获取要置顶的项在当前屏幕的位置,mIndex是记录的要置顶项在RecyclerView中的位置
                    int n = mIndex - layoutManager.findFirstVisibleItemPosition();
                    if (0 <= n && n < mRecyclerView.getChildCount()) {
                        //获取要置顶的项顶部离RecyclerView顶部的距离
                        int top = mRecyclerView.getChildAt(n).getTop();
                        //最后的移动
                        mRecyclerView.scrollBy(0, top);
                    }
                }
            }
        });

总体实现难度并不大,至此就完成了滑动锚点定位的逻辑。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,980评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,178评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,868评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,498评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,492评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,521评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,910评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,569评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,793评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,559评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,639评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,342评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,931评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,904评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,144评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,833评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,350评论 2 342

推荐阅读更多精彩内容