仿京東金融兩個(gè)ViewPager聯(lián)動(dòng)效果

前言

產(chǎn)品讓做一個(gè)仿京東金融的效果,在網(wǎng)上搜了搜,就找到一個(gè)通過重寫ViewPage實(shí)現(xiàn)聯(lián)動(dòng)的案例,但效果不理想.最后實(shí)在沒找到適合參考的案例,所以記錄下來,希望能給有同樣需求的童鞋一些幫助.

首先感謝我的同事ChenWei,沒有他的幫助番枚,絕對(duì)達(dá)不到現(xiàn)在的效果审轮。里面好多難點(diǎn)都是他搞定的。

12081141-a5c57479bc7e5049.gif
all.gif

技術(shù)難點(diǎn)

1欧宜、兩個(gè)ViewPager的聯(lián)動(dòng)
2坐榆、滑動(dòng)任一列表時(shí),headerVp和其他列表都跟著滑動(dòng)
3、切換頁(yè)面時(shí)所有列表回到初始位置
4冗茸、有個(gè)頁(yè)面有懸停吸頂欄,我們是用CoordinatorLayout+AppBarLayout+RecyclerView利用behavior實(shí)現(xiàn)的,怎么獲取此頁(yè)面滑動(dòng)的距離席镀、怎么讓其跟隨別的列表滑動(dòng)及怎么歸位

一、兩個(gè)ViewPager的聯(lián)動(dòng)

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>

既然要聯(lián)動(dòng),兩個(gè)ViewPager自然就要相互監(jiān)聽.

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滑動(dòng)過程中,如上圖所示,滑動(dòng)完整一頁(yè)時(shí)bodyVp滑動(dòng)距離為屏幕寬度screenWidth,而headerVp滑動(dòng)距離為headerVp自身寬度再加上右邊那個(gè)白色的marging值(為headerWidth + margin),那么當(dāng)bodyVp滑動(dòng)距離為bodyX時(shí),headVp滑動(dòng)距離headerX就為bodyX / screenWidth * (headerWidth + margin);

然后調(diào)用headerVp.scrollTo(headerX, 0),headerVp就能跟隨滑動(dòng)了.

下面是封裝的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

其實(shí)headerVp如果主要用來展示數(shù)據(jù),沒有復(fù)雜的觸摸操作的話,可以吧bodyVp放在上層,只是將headerVp占用的那塊空間用透明布局填充.這時(shí)無論是滑動(dòng)headerVp,還是bodyVp其實(shí)都是在滑動(dòng)bodyVp,這樣只需讓headerVp跟著bodyVp滑動(dòng)就行了.

二夏漱、滑動(dòng)任一列表時(shí),headerVp和其他列表都跟著滑動(dòng)

首先說一點(diǎn),headerVp和另外的列表隨著當(dāng)前頁(yè)面向上滑動(dòng)時(shí),如果headerVp不可見了,他們將不會(huì)再向上滑動(dòng).

followingSlidding.gif

1.計(jì)算當(dāng)前l(fā)ist滑動(dòng)距離

  1. scrollView:
    scrollView有個(gè)onScrollChanged()方法,我是寫了一個(gè)類繼承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坐標(biāo)值.
在scrollView界面就可以監(jiān)聽到滑動(dòng)的距離了

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移動(dòng)的距離了.
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

這時(shí)通過監(jiān)聽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. 跟隨移動(dòng)

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

三豪诲、切換頁(yè)面時(shí)所有列表回到初始位置

scrollView和recycleView比較簡(jiǎn)單,重點(diǎn)說下coordinateLayout的那種情況.

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

coordinateLayout和recycleView和appBarLayout必須都要調(diào)用對(duì)應(yīng)的方法才能完成復(fù)位.尤其是appBarLayout很容易忽略.

總結(jié)

由于篇幅原因,還有很多細(xì)節(jié)沒有展開講,感興趣的童鞋就麻煩看下源碼吧...

https://github.com/Alphabet111/ViewPagerDemo

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市挂绰,隨后出現(xiàn)的幾起案子屎篱,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,277評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件交播,死亡現(xiàn)場(chǎng)離奇詭異专肪,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)堪侯,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門嚎尤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人伍宦,你說我怎么就攤上這事芽死。” “怎么了次洼?”我有些...
    開封第一講書人閱讀 163,624評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵关贵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我卖毁,道長(zhǎng)揖曾,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,356評(píng)論 1 293
  • 正文 為了忘掉前任亥啦,我火速辦了婚禮炭剪,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘翔脱。我一直安慰自己奴拦,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評(píng)論 6 392
  • 文/花漫 我一把揭開白布届吁。 她就那樣靜靜地躺著错妖,像睡著了一般。 火紅的嫁衣襯著肌膚如雪疚沐。 梳的紋絲不亂的頭發(fā)上暂氯,一...
    開封第一講書人閱讀 51,292評(píng)論 1 301
  • 那天,我揣著相機(jī)與錄音亮蛔,去河邊找鬼痴施。 笑死,一個(gè)胖子當(dāng)著我的面吹牛尔邓,可吹牛的內(nèi)容都是我干的晾剖。 我是一名探鬼主播锉矢,決...
    沈念sama閱讀 40,135評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼梯嗽,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了沽损?” 一聲冷哼從身側(cè)響起灯节,我...
    開封第一講書人閱讀 38,992評(píng)論 0 275
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后炎疆,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體卡骂,經(jīng)...
    沈念sama閱讀 45,429評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評(píng)論 3 334
  • 正文 我和宋清朗相戀三年形入,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了全跨。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,785評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡亿遂,死狀恐怖浓若,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情蛇数,我是刑警寧澤挪钓,帶...
    沈念sama閱讀 35,492評(píng)論 5 345
  • 正文 年R本政府宣布,位于F島的核電站耳舅,受9級(jí)特大地震影響碌上,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜浦徊,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評(píng)論 3 328
  • 文/蒙蒙 一馏予、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧盔性,春花似錦吗蚌、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至暂筝,卻和暖如春箩言,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背焕襟。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工陨收, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人鸵赖。 一個(gè)月前我還...
    沈念sama閱讀 47,891評(píng)論 2 370
  • 正文 我出身青樓务漩,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親它褪。 傳聞我的和親對(duì)象是個(gè)殘疾皇子饵骨,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容