Android中ViewPager實(shí)現(xiàn)滑動(dòng)條及與Fragment結(jié)合的實(shí)例教程
自主實(shí)現(xiàn)滑動(dòng)指示條
先上一個(gè)基本效果圖:
1.XML布局
布局代碼如下:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" tools:context="com.example.testviewpage_2.MainActivity" > <ImageView android:id="@+id/cursor" android:layout_width="fill_parent" android:layout_height="wrap_content" android:scaleType="matrix" android:src="@drawable/a" /> <android.support.v4.view.ViewPager android:id="@+id/viewpager" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_gravity="center"/> </LinearLayout>
采用線性垂直布局,在滑動(dòng)頁(yè)面的上方添加一個(gè)小水平條。
2.JAVA代碼
public class MainActivity extends Activity { private View view1, view2, view3; private List<View> viewList;// view數(shù)組 private ViewPager viewPager; // 對(duì)應(yīng)的viewPager private ImageView cursor; private int bmpw = 0; // 游標(biāo)寬度 private int offset = 0;// // 動(dòng)畫圖片偏移量 private int currIndex = 0;// 當(dāng)前頁(yè)卡編號(hào) @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); viewPager = (ViewPager) findViewById(R.id.viewpager); LayoutInflater inflater = getLayoutInflater(); view1 = inflater.inflate(R.layout.layout1, null); view2 = inflater.inflate(R.layout.layout2, null); view3 = inflater.inflate(R.layout.layout3, null); viewList = new ArrayList<View>();// 將要分頁(yè)顯示的View裝入數(shù)組中 viewList.add(view1); viewList.add(view2); viewList.add(view3); //初始化指示器位置 initCursorPos(); viewPager.setAdapter(new MyPagerAdapter(viewList)); viewPager.setOnPageChangeListener(new MyPageChangeListener()); } //初始化指示器位置 public void initCursorPos() { // 初始化動(dòng)畫 cursor = (ImageView) findViewById(R.id.cursor); bmpw = BitmapFactory.decodeResource(getResources(), R.drawable.a) .getWidth();// 獲取圖片寬度 DisplayMetrics dm = new DisplayMetrics(); getWindowManager().getDefaultDisplay().getMetrics(dm); int screenW = dm.widthPixels;// 獲取分辨率寬度 offset = (screenW / viewList.size() - bmpw) / 2;// 計(jì)算偏移量 Matrix matrix = new Matrix(); matrix.postTranslate(offset, 0); cursor.setImageMatrix(matrix);// 設(shè)置動(dòng)畫初始位置 } //頁(yè)面改變監(jiān)聽(tīng)器 public class MyPageChangeListener implements OnPageChangeListener { int one = offset * 2 + bmpw;// 頁(yè)卡1 -> 頁(yè)卡2 偏移量 int two = one * 2;// 頁(yè)卡1 -> 頁(yè)卡3 偏移量 @Override public void onPageSelected(int arg0) { Animation animation = null; switch (arg0) { case 0: if (currIndex == 1) { animation = new TranslateAnimation(one, 0, 0, 0); } else if (currIndex == 2) { animation = new TranslateAnimation(two, 0, 0, 0); } break; case 1: if (currIndex == 0) { animation = new TranslateAnimation(offset, one, 0, 0); } else if (currIndex == 2) { animation = new TranslateAnimation(two, one, 0, 0); } break; case 2: if (currIndex == 0) { animation = new TranslateAnimation(offset, two, 0, 0); } else if (currIndex == 1) { animation = new TranslateAnimation(one, two, 0, 0); } break; } currIndex = arg0; animation.setFillAfter(true);// True:圖片停在動(dòng)畫結(jié)束位置 animation.setDuration(300); cursor.startAnimation(animation); } @Override public void onPageScrolled(int arg0, float arg1, int arg2) { } @Override public void onPageScrollStateChanged(int arg0) { } } /** * ViewPager適配器 */ public class MyPagerAdapter extends PagerAdapter { public List<View> mListViews; public MyPagerAdapter(List<View> mListViews) { this.mListViews = mListViews; } @Override public boolean isViewFromObject(View arg0, Object arg1) { // TODO Auto-generated method stub return arg0 == arg1; } @Override public int getCount() { // TODO Auto-generated method stub return mListViews.size(); } @Override public void destroyItem(ViewGroup container, int position, Object object) { // TODO Auto-generated method stub container.removeView(mListViews.get(position)); } @Override public Object instantiateItem(ViewGroup container, int position) { // TODO Auto-generated method stub container.addView(mListViews.get(position)); return mListViews.get(position); } } }
3.重點(diǎn)解析
從易到難一步步來(lái)講。
(1)initCursorPos()---初始化指示器位置
游標(biāo)在初始化顯示時(shí),我們要根據(jù)屏幕寬度來(lái)顯示游標(biāo)位置。先看看這部分代碼:
//初始化指示器位置 public void initCursorPos() { // 初始化動(dòng)畫 cursor = (ImageView) findViewById(R.id.cursor); bmpw = BitmapFactory.decodeResource(getResources(), R.drawable.a) .getWidth();// 獲取圖片寬度 DisplayMetrics dm = new DisplayMetrics(); getWindowManager().getDefaultDisplay().getMetrics(dm); int screenW = dm.widthPixels;// 獲取分辨率寬度 offset = (screenW / viewList.size() - bmpw) / 2;// 計(jì)算偏移量 Matrix matrix = new Matrix(); matrix.postTranslate(offset, 0); cursor.setImageMatrix(matrix);// 設(shè)置動(dòng)畫初始位置 }
可能有些同學(xué)不明白的位置在于,初始化位置的偏移量為什么這么算,下面,我畫了張圖,看下就應(yīng)該明白了。
最后對(duì)于偏移的方法,可用的很多,這里仿網(wǎng)上的代碼用了matrix;當(dāng)然大家可以用其它的偏移方法,一樣。
(2)MyPageChangeListener()---頁(yè)面改變監(jiān)聽(tīng)器
代碼如下 :
public class MyPageChangeListener implements OnPageChangeListener { int one = offset * 2 + bmpw;// 頁(yè)卡1 -> 頁(yè)卡2 偏移量 int two = one * 2;// 頁(yè)卡1 -> 頁(yè)卡3 偏移量 @Override public void onPageSelected(int arg0) { Animation animation = null; switch (arg0) { case 0: if (currIndex == 1) { animation = new TranslateAnimation(one, 0, 0, 0); } else if (currIndex == 2) { animation = new TranslateAnimation(two, 0, 0, 0); } break; case 1: if (currIndex == 0) { animation = new TranslateAnimation(offset, one, 0, 0); } else if (currIndex == 2) { animation = new TranslateAnimation(two, one, 0, 0); } break; case 2: if (currIndex == 0) { animation = new TranslateAnimation(offset, two, 0, 0); } else if (currIndex == 1) { animation = new TranslateAnimation(one, two, 0, 0); } break; } currIndex = arg0; animation.setFillAfter(true);// True:圖片停在動(dòng)畫結(jié)束位置 animation.setDuration(300); cursor.startAnimation(animation); }
原理是這樣,根據(jù)滑動(dòng)到的頁(yè)面,把游標(biāo)滑動(dòng)找指定位置。
這里可能有難度的地方在于,數(shù)學(xué)……
我畫了一張圖,解釋從第一個(gè)頁(yè)面到第二個(gè)頁(yè)面時(shí)的距離為什么是“游標(biāo)寬度+offset*2”,其它距離類似。
使用Fragment實(shí)現(xiàn)ViewPager滑動(dòng)
效果圖:
在第一個(gè)頁(yè)面加一個(gè)Btn
第一頁(yè)面向第二頁(yè)面滑動(dòng)
第二頁(yè)面向第三個(gè)頁(yè)面滑動(dòng)
一些說(shuō)明:
FragmentPagerAdapter派生自PagerAdapter,它是用來(lái)呈現(xiàn)Fragment頁(yè)面的,這些Fragment頁(yè)面會(huì)一直保存在fragment manager中,以便用戶可以隨時(shí)取用。
這個(gè)適配器最好用于有限個(gè)靜態(tài)fragment頁(yè)面的管理。盡管不可見(jiàn)的視圖有時(shí)會(huì)被銷毀,但用戶所有訪問(wèn)過(guò)的fragment都會(huì)被保存在內(nèi)存中。因此fragment實(shí)例會(huì)保存大量的各種狀態(tài),這就造成了很大的內(nèi)存開(kāi)銷。所以如果要處理大量的頁(yè)面切換,建議使用FragmentStatePagerAdapter.
每一個(gè)使用FragmentPagerAdapter的ViewPager都要有一個(gè)有效的ID集合,有效ID的集合就是Fragment的集合(感謝夫諸同學(xué)的提示)
對(duì)于FragmentPagerAdapter的派生類,只需要重寫getItem(int)和getCount()就可以了。
具體實(shí)現(xiàn):
1、適配器實(shí)現(xiàn)——FragmentPagerAdapter
先看完整代碼,再細(xì)講:
public class FragAdapter extends FragmentPagerAdapter { private List<Fragment> mFragments; public FragAdapter(FragmentManager fm,List<Fragment> fragments) { super(fm); // TODO Auto-generated constructor stub mFragments=fragments; } @Override public Fragment getItem(int arg0) { // TODO Auto-generated method stub return mFragments.get(arg0); } @Override public int getCount() { // TODO Auto-generated method stub return mFragments.size(); } }
這里有三個(gè)函數(shù),根據(jù)第一部分的官方文檔,可知,對(duì)于FragmentPagerAdapter的派生類,只重寫getItem(int)和getCount()就可以了。
對(duì)于構(gòu)造函數(shù),這里申請(qǐng)了一個(gè)Fragment的List對(duì)象,用于保存用于滑動(dòng)的Fragment對(duì)象,并在創(chuàng)造函數(shù)中初始化:
public FragAdapter(FragmentManager fm,List<Fragment> fragments) { super(fm); // TODO Auto-generated constructor stub mFragments=fragments; }
然后在getItem(int arg0)中,根據(jù)傳來(lái)的參數(shù)arg0,來(lái)返回當(dāng)前要顯示的fragment。
最后,getCount()返回用于滑動(dòng)的fragment總數(shù);
從構(gòu)造函數(shù)所以看出,我們要構(gòu)造Fragment的集合才行,所以下面我們就先產(chǎn)生我們所需要的Fragment類;
2、三個(gè)Fragment類
第一個(gè)Fragment類:
XML:(layout1.xml)
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:background="#ffffff" android:orientation="vertical" > <Button android:id="@+id/fragment1_btn" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="show toast" /> </LinearLayout>
在其中加入了一個(gè)Btn
public class Fragment1 extends Fragment { @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { // TODO Auto-generated method stub View view= inflater.inflate(R.layout.layout1, container, false); //對(duì)View中控件的操作方法 Button btn = (Button)view.findViewById(R.id.fragment1_btn); btn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { // TODO Auto-generated method stub Toast.makeText(getActivity(), "點(diǎn)擊了第一個(gè)fragment的BTN", Toast.LENGTH_SHORT).show(); } }); return view; } }
在onCreateView()中返回要顯示的View,上面這段代碼簡(jiǎn)單演示了如何對(duì)視圖里的控件進(jìn)行操作,難度不大,不再細(xì)講,如果對(duì)Fragment不太熟悉的同學(xué),先看看這篇文章:《Android Fragment完全解析,關(guān)于碎片你所需知道的一切》
第二個(gè)Fragment類:
XML代碼:(layout2.xml)原生代碼,沒(méi)有做任何更改
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:background="#ffff00" android:orientation="vertical" > </LinearLayout>
public class Fragment2 extends Fragment { @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { // TODO Auto-generated method stub View view=inflater.inflate(R.layout.layout2, container, false); return view; } }
第三個(gè)Fragment類:
XML代碼:(layout3.xml)同樣,原生代碼,沒(méi)做任何更改
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:background="#ff00ff" android:orientation="vertical" > </LinearLayout>
java代碼:
public class Fragment3 extends Fragment { @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { // TODO Auto-generated method stub View view=inflater.inflate(R.layout.layout3, container, false); return view; } }
3、主activity實(shí)現(xiàn)
核心代碼:
public class MainActivity extends FragmentActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); //構(gòu)造適配器 List<Fragment> fragments=new ArrayList<Fragment>(); fragments.add(new Fragment1()); fragments.add(new Fragment2()); fragments.add(new Fragment3()); FragAdapter adapter = new FragAdapter(getSupportFragmentManager(), fragments); //設(shè)定適配器 ViewPager vp = (ViewPager)findViewById(R.id.viewpager); vp.setAdapter(adapter); } }
首先有一個(gè)最值得注意的地方:Activity派生自FragmentActivity,其實(shí)這是有關(guān)Fragment的基礎(chǔ)知識(shí),只有FragmentActivity才能內(nèi)嵌fragment頁(yè)面,普通Activity是不行的。
這段代碼主要分為兩步,第一步:構(gòu)造適配器;第二步:設(shè)定適配器。
先看構(gòu)造適配器的過(guò)程:
//構(gòu)造適配器 List<Fragment> fragments=new ArrayList<Fragment>(); fragments.add(new Fragment1()); fragments.add(new Fragment2()); fragments.add(new Fragment3()); FragAdapter adapter = new FragAdapter(getSupportFragmentManager(), fragments);
構(gòu)造一個(gè)fragment列表,然后將上面的三個(gè)Fragment類對(duì)應(yīng)的實(shí)例添加進(jìn)去,最后生成FragAdapter實(shí)例。
至于第二步,設(shè)定適配器,沒(méi)什么好講的。
- Android 開(kāi)發(fā)之BottomBar+ViewPager+Fragment實(shí)現(xiàn)炫酷的底部導(dǎo)航效果
- Android App中ViewPager與Fragment結(jié)合的一些問(wèn)題解決
- Android App中使用ViewPager+Fragment實(shí)現(xiàn)滑動(dòng)切換效果
- Android App在ViewPager中使用Fragment的實(shí)例講解
- Android基于ViewPager Fragment實(shí)現(xiàn)選項(xiàng)卡
- Android中ViewPager實(shí)現(xiàn)滑動(dòng)指示條及與Fragment的配合
- Android中ViewPager和Fragment的使用
- Android實(shí)現(xiàn)Tab布局的4種方式(Fragment+TabPageIndicator+ViewPager)
- 詳解Android中fragment和viewpager的那點(diǎn)事兒
- Android中ViewPager獲取當(dāng)前顯示的Fragment
相關(guān)文章
Android實(shí)現(xiàn)用戶圓形頭像和模糊背景
這篇文章主要介紹了Android實(shí)現(xiàn)用戶圓形頭像和模糊背景 ,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-04-04Android 加載大圖、多圖和LruCache緩存詳細(xì)介紹
這篇文章主要介紹了Android 加載大圖、多圖和LruCache緩存詳細(xì)介紹的相關(guān)資料,需要的朋友可以參考下2016-10-10Android HTTP網(wǎng)絡(luò)請(qǐng)求的異步實(shí)現(xiàn)
這篇文章主要介紹了Android HTTP網(wǎng)絡(luò)請(qǐng)求的異步實(shí)現(xiàn),感興趣的小伙伴們可以參考一下2016-07-07Android Webview的postUrl與loadUrl加載頁(yè)面實(shí)例
這篇文章主要介紹了Android Webview的postUrl與loadUrl加載頁(yè)面實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03Android百度地圖應(yīng)用之MapFragment的使用
這篇文章主要為大家詳細(xì)介紹了Android百度地圖應(yīng)用之MapFragment的使用的相關(guān)資料,需要的朋友可以參考下2016-06-06Android實(shí)現(xiàn)底部導(dǎo)航欄功能
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)底部導(dǎo)航欄功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06