Android 兩個(gè)ViewPager的聯(lián)動(dòng)效果的實(shí)現(xiàn)
前言
以前做的項(xiàng)目,導(dǎo)航欄基本上是在頂部或者是在底部,但是最近開(kāi)發(fā)的一款app,剛開(kāi)始拿到設(shè)計(jì)圖也是很懵逼的,導(dǎo)航欄居然是在中間,what fuck!設(shè)計(jì)圖如下:

導(dǎo)航欄在中間就會(huì)涉及到兩個(gè)viewpager之間的聯(lián)動(dòng),viewpager的高度適應(yīng)等問(wèn)題,現(xiàn)在來(lái)紀(jì)錄一下是怎么解決問(wèn)題的?希望給有同樣需求的提供一定的幫助。
(一)Viewpager 高度自適應(yīng)
系統(tǒng)自動(dòng)viewpager 不能設(shè)置wrap_content;
自定義viewpager,注意高度的設(shè)置否則底部空白的問(wèn)題
網(wǎng)上也會(huì)有很多相關(guān)的教程,我選擇了其中一個(gè)。具體代碼如下:
public class WrapContentHeightViewPager extends ViewPager {
private int current;
private int height = 0;
private boolean scrollble = true;
public WrapContentHeightViewPager(Context context) {
super(context);
}
public WrapContentHeightViewPager(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
if (getChildCount() > current) {
View child = getChildAt(current);
child.measure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED));
int h = child.getMeasuredHeight();
height = h;
}
heightMeasureSpec = MeasureSpec.makeMeasureSpec(height, MeasureSpec.EXACTLY);
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}
public void resetHeight(int current) {
this.current = current;
if (getChildCount() > current) {
LinearLayout.LayoutParams layoutParams = (LinearLayout.LayoutParams) getLayoutParams();
if (layoutParams == null) {
layoutParams = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, height);
} else {
layoutParams.height = height;
}
setLayoutParams(layoutParams);
}
}
@Override
public boolean onTouchEvent(MotionEvent ev) {
if (!scrollble) {
return true;
}
return super.onTouchEvent(ev);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
return scrollble && super.onInterceptTouchEvent(ev);
}
public boolean isScrollble() {
return scrollble;
}
public void setScrollble(boolean scrollble) {
this.scrollble = scrollble;
}
}
(二)Viewpager 的聯(lián)動(dòng)
聯(lián)動(dòng)ViewPager的意思就是當(dāng)一個(gè)viewpager在滑動(dòng)的時(shí)候,另外一個(gè)ViewPager也跟著滑動(dòng),而且兩者是同步的。
如果ViewPager有關(guān)于移動(dòng)距離的回調(diào)接口,這事兒就好辦了,遺憾的是沒(méi)有,只有一個(gè)OnPageChangeListener,我試過(guò)在OnPageChangeListener中根據(jù)onPageScrolled(int position, float positionOffset, int positionOffsetPixels)的參數(shù)來(lái)做,但是失敗了。
沒(méi)辦法只有改造一下OnPageChangeListener,讓它可以實(shí)現(xiàn)兩個(gè)viewpager的聯(lián)動(dòng),難點(diǎn)在于對(duì)滑動(dòng)的距離一個(gè)計(jì)算。
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);
}
}
}
(三)使用方法
xml布局
<?xml version="1.0" encoding="utf-8"?>
<android.support.v4.widget.NestedScrollView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<cn.yznu.gdmapoperate.ui.widget.WrapContentHeightViewPager
android:id="@+id/body_vp"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
<com.flyco.tablayout.SlidingTabLayout
android:id="@+id/tabLayout"
android:layout_width="match_parent"
android:layout_height="40dp"
android:layout_centerHorizontal="true"
android:layout_gravity="center_horizontal"
android:background="@color/colorPrimaryDark"
android:paddingBottom="10dp"
app:tl_indicator_color="#000"
app:tl_indicator_margin_top="10dp"
app:tl_indicator_width_equal_title="true"
app:tl_tab_space_equal="true"
app:tl_textSelectColor="#f00"
app:tl_textUnselectColor="#fff"
app:tl_textsize="17sp" />
<cn.yznu.gdmapoperate.ui.widget.WrapContentHeightViewPager
android:id="@+id/header_vp"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
</LinearLayout>
</android.support.v4.widget.NestedScrollView>
activity中的配置
bodyVp.addOnPageChangeListener(new BaseLinkPageChangeListener(bodyVp, headerVp) {
@Override
public void onPageSelected(int position) {
super.onPageSelected(position);
pageScrollToTop();
bodyVp.resetHeight(position);//設(shè)置viewpager高度
headerVp.resetHeight(position);
}
});
headerVp.addOnPageChangeListener(new BaseLinkPageChangeListener(headerVp, bodyVp) {
@Override
public void onPageSelected(int position) {
super.onPageSelected(position);
tabLayout.onPageSelected(position);
}
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
super.onPageScrolled(position, positionOffset, positionOffsetPixels);
tabLayout.onPageScrolled(position, positionOffset, positionOffsetPixels);
bodyVp.resetHeight(position);
headerVp.resetHeight(position);
}
});
大功搞成,看一下效果圖

總結(jié)
一頓亂寫(xiě),個(gè)人觀點(diǎn)僅供參考,如有不對(duì)的地方,請(qǐng)直接直出
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android使用ViewPager完成app引導(dǎo)頁(yè)
- Android ViewPager實(shí)現(xiàn)滑動(dòng)指示條功能
- Android使用viewpager實(shí)現(xiàn)畫(huà)廊式效果
- android使用ViewPager實(shí)現(xiàn)圖片自動(dòng)切換
- Android自定義引導(dǎo)玩轉(zhuǎn)ViewPager的方法詳解
- android使用viewpager計(jì)算偏移量實(shí)現(xiàn)選項(xiàng)卡功能
- Android使用ViewPager快速切換Fragment時(shí)卡頓的優(yōu)化方案
- Android利用ViewPager實(shí)現(xiàn)帶小圓球的圖片滑動(dòng)
- Android Studio使用ViewPager+Fragment實(shí)現(xiàn)滑動(dòng)菜單Tab效果
- Android自定義ViewPager實(shí)現(xiàn)縱向滑動(dòng)翻頁(yè)效果
- 深入了解ViewPager2的使用
相關(guān)文章
Android創(chuàng)建文件實(shí)現(xiàn)對(duì)文件監(jiān)聽(tīng)示例
Android創(chuàng)建文件實(shí)現(xiàn)對(duì)文件監(jiān)聽(tīng),可以用android.os.FileObserver;類(lèi)來(lái)實(shí)現(xiàn),下面是實(shí)現(xiàn)代碼,內(nèi)有注釋2014-01-01
Android 使用 RxJava2 實(shí)現(xiàn)倒計(jì)時(shí)功能的示例代碼
本篇文章主要介紹了Android 使用 RxJava2 實(shí)現(xiàn)倒計(jì)時(shí)功能的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-03-03
android實(shí)現(xiàn)系統(tǒng)信息推送
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)系統(tǒng)信息推送,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04
六款值得推薦的android(安卓)開(kāi)源框架簡(jiǎn)介
同事整理的android(安卓)開(kāi)源框架,個(gè)個(gè)都堪稱(chēng)經(jīng)典。32 個(gè)贊!2014-06-06
Android客戶端與服務(wù)端數(shù)據(jù)加密傳輸方案詳解
這篇文章主要為大家介紹了Android客戶端與服務(wù)端數(shù)據(jù)加密傳輸方案詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01
Android開(kāi)發(fā)筆記之探秘WebView
瀏覽器控件是每個(gè)開(kāi)發(fā)環(huán)境都具備的,這為馬甲神功提供了用武之地,windows的有webbrowser,android和ios都有webview。只是其引擎不同,相對(duì)于微軟的webbrowser,android及ios的webview的引擎都是webkit,對(duì)Html5提供支持。本篇主要介紹android的webview。2014-08-08
android獲取照片的快照 思路及實(shí)現(xiàn)方法
android獲取照片的快照 思路及實(shí)現(xiàn)方法,需要的朋友可以參考一下2013-06-06
Android App開(kāi)發(fā)的自動(dòng)化測(cè)試框架UI Automator使用教程
UI Automator為Android程序的UI開(kāi)發(fā)提供了測(cè)試環(huán)境,這里我們就來(lái)看一下Android App開(kāi)發(fā)的自動(dòng)化測(cè)試框架UI Automator使用教程,需要的朋友可以參考下2016-07-07
AndroidHttpClient詳解及調(diào)用示例
本文給大家介紹AndroidHttpClient結(jié)構(gòu)、使用方式及調(diào)用示例詳解,需要的朋友可以參考下2015-10-10
Android View的事件分發(fā)機(jī)制深入分析講解
事件分發(fā)從手指觸摸屏幕開(kāi)始,即產(chǎn)生了觸摸信息,被底層系統(tǒng)捕獲后會(huì)傳遞給Android的輸入系統(tǒng)服務(wù)IMS,通過(guò)Binder把消息發(fā)送到activity,activity會(huì)通過(guò)phoneWindow、DecorView最終發(fā)送給ViewGroup。這里就直接分析ViewGroup的事件分發(fā)2023-01-01

