仿京东金融两个ViewPager联动效果

前言

产品让做一个仿京东金融的效果,在网上搜了搜,就找到一个通过重写ViewPage实现联动的案例,但效果不理想.最后实在没找到适合参考的案例,所以记录下来,希望能给有同样需求的童鞋一些帮助.

首先感谢我的同事ChenWei,没有他的帮助,绝对达不到现在的效果。里面好多难点都是他搞定的。

12081141-a5c57479bc7e5049.gif
all.gif

技术难点

1、两个ViewPager的联动
2、滑动任一列表时,headerVp和其他列表都跟着滑动
3、切换页面时所有列表回到初始位置
4、有个页面有悬停吸顶栏,我们是用CoordinatorLayout+AppBarLayout+RecyclerView利用behavior实现的,怎么获取此页面滑动的距离、怎么让其跟随别的列表滑动及怎么归位

一、两个ViewPager的联动

viewpagerLink.gif

布局如下:

<RelativeLayout
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      android:layout_below="@+id/tabLayout"
      android:clipChildren="false"
      >
    <android.support.v4.view.ViewPager

        android:id="@+id/body_vp"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        />
    <android.support.v4.view.ViewPager
        android:id="@+id/header_vp"
        android:layout_width="match_parent"
        android:layout_height="160dp"
        android:layout_marginLeft="30dp"
        android:layout_marginRight="30dp"
        android:clipChildren="false"
        />
  </RelativeLayout>

既然要联动,两个ViewPager自然就要相互监听.

bodyVp.addOnPageChangeListener(new BaseLinkPageChangeListener(bodyVp, headerVp) {
      @Override public void onPageSelected(int position) {
        super.onPageSelected(position);
        pageScrollToTop();
      }
    });
    headerVp.addOnPageChangeListener(new BaseLinkPageChangeListener(headerVp, bodyVp) {
      @Override public void onPageSelected(int position) {
        super.onPageSelected(position);
        tabLayout.onPageSelected(position);
      }
    });
17_33_52__07_16_2018.jpg

在ViewPager滑动过程中,如上图所示,滑动完整一页时bodyVp滑动距离为屏幕宽度screenWidth,而headerVp滑动距离为headerVp自身宽度再加上右边那个白色的marging值(为headerWidth + margin),那么当bodyVp滑动距离为bodyX时,headVp滑动距离headerX就为bodyX / screenWidth * (headerWidth + margin);

然后调用headerVp.scrollTo(headerX, 0),headerVp就能跟随滑动了.

下面是封装的OnPagerChangeLIstener:

public class BaseLinkPageChangeListener implements ViewPager.OnPageChangeListener {

  private ViewPager linkViewPager;
  private ViewPager selfViewPager;

  private int pos;

  public BaseLinkPageChangeListener(ViewPager selfViewPager, ViewPager linkViewPager) {
    this.linkViewPager = linkViewPager;
    this.selfViewPager = selfViewPager;
  }

  @Override
  public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {

    int marginX = ((selfViewPager.getWidth() + selfViewPager.getPageMargin()) * position
        + positionOffsetPixels) * (linkViewPager.getWidth() + linkViewPager.getPageMargin()) / (
        selfViewPager.getWidth()
            + selfViewPager.getPageMargin());

    if (linkViewPager.getScrollX() != marginX) {
      linkViewPager.scrollTo(marginX, 0);
    }
  }

  @Override public void onPageSelected(int position) {
    this.pos=position;
  }

  @Override public void onPageScrollStateChanged(int state) {
    if (state == ViewPager.SCROLL_STATE_IDLE) {
      linkViewPager.setCurrentItem(pos);
    }
  }
}

ps

其实headerVp如果主要用来展示数据,没有复杂的触摸操作的话,可以吧bodyVp放在上层,只是将headerVp占用的那块空间用透明布局填充.这时无论是滑动headerVp,还是bodyVp其实都是在滑动bodyVp,这样只需让headerVp跟着bodyVp滑动就行了.

二、滑动任一列表时,headerVp和其他列表都跟着滑动

首先说一点,headerVp和另外的列表随着当前页面向上滑动时,如果headerVp不可见了,他们将不会再向上滑动.

followingSlidding.gif

1.计算当前list滑动距离

  1. scrollView:
    scrollView有个onScrollChanged()方法,我是写了一个类继承NestedScrollView,
    然后重写onScrollChanged()方法,再把top暴露出去.
@Override protected void onScrollChanged(int l, int t, int oldl, int oldt) {
    super.onScrollChanged(l, t, oldl, oldt);
    if (this.onScrollChangedListener != null) {
      onScrollChangedListener.onScrollChanged(t, oldt);
    }
  }

top就是scrollView顶部的y坐标值.
在scrollView界面就可以监听到滑动的距离了

scrollView.setOnScrollChangedListener(new OnScrollChangedListener() {
      @Override public void onScrollChanged(int top, int oldTop) {
        if (isPageVisible()) {
          ((MainActivity) getActivity()).pageScrollTo(Math.min(top, maxScrollDisY()));
        }
      }
    });
  1. recycleVIew
    直接用recyclerView.computeVerticalScrollOffset()就可以获取recycleView移动的距离了.
recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
      @Override public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
        super.onScrolled(recyclerView, dx, dy);
        if (isPageVisible()) {
          if (layoutManager.findFirstVisibleItemPosition() == 0) {
            ((MainActivity) getActivity()).pageScrollTo(recyclerView.computeVerticalScrollOffset());
          } else {
            ((MainActivity) getActivity()).pageScrollTo(maxScrollDisY());
          }
        }
      }
    });
  1. CoordinatorLayout+AppBarLayout+RecyclerView

这时通过监听appBarLayout来获取距离

appBarLayout.addOnOffsetChangedListener(new AppBarLayout.OnOffsetChangedListener() {
      @Override public void onOffsetChanged(AppBarLayout appBarLayout, int verticalOffset) {
        if (isPageVisible()) {
          ((MainActivity) getActivity()).pageScrollTo(
              Math.min(Math.abs(verticalOffset), maxScrollDisY()));
        }
      }
    });

2. 跟随移动

  1. headerVp直接调用headerVp.setTranslationY(-distance);
  2. scrollView调用scrollView.scrollTo(0, disY);
  3. recycleView用layoutManager.scrollToPositionWithOffset(0, -disY);
  4. layoutManager.scrollToPositionWithOffset(0, -disY);
    coordinatorLayout.scrollTo(0, disY);

三、切换页面时所有列表回到初始位置

scrollView和recycleView比较简单,重点说下coordinateLayout的那种情况.

@Override public void pageScrollToTop() {
    layoutManager.scrollToPositionWithOffset(0, 0);
    appBarLayout.setExpanded(true);
    coordinatorLayout.scrollTo(0, 0);
  }

coordinateLayout和recycleView和appBarLayout必须都要调用对应的方法才能完成复位.尤其是appBarLayout很容易忽略.

总结

由于篇幅原因,还有很多细节没有展开讲,感兴趣的童鞋就麻烦看下源码吧...

https://github.com/Alphabet111/ViewPagerDemo

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

推荐阅读更多精彩内容