欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android中ViewPager實現(xiàn)滑動指示條及與Fragment的配合

 更新時間:2016年03月23日 12:03:55   作者:harvic880925  
這篇文章主要介紹了Android中ViewPager實現(xiàn)滑動指示條及與Fragment的配合,使用Fragment實現(xiàn)ViewPager的滑動是一種比較推薦的做法,需要的朋友可以參考下

自主實現(xiàn)滑動指示條
先上效果圖:

2016323115400211.png (300×500)2016323115431597.png (300×500)

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> 

采用線性垂直布局,在滑動頁面的上方添加一個小水平條。

2、JAVA代碼
先給出全部代碼,然后再逐步講解。

public class MainActivity extends Activity { 
 
 private View view1, view2, view3; 
 private List<View> viewList;// view數(shù)組 
 private ViewPager viewPager; // 對應的viewPager 
 
 private ImageView cursor; 
 private int bmpw = 0; // 游標寬度 
 private int offset = 0;// // 動畫圖片偏移量 
 private int currIndex = 0;// 當前頁卡編號 
 
 @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>();// 將要分頁顯示的View裝入數(shù)組中 
  viewList.add(view1); 
  viewList.add(view2); 
  viewList.add(view3); 
 
  //初始化指示器位置 
  initCursorPos(); 
   
  viewPager.setAdapter(new MyPagerAdapter(viewList)); 
  viewPager.setOnPageChangeListener(new MyPageChangeListener()); 
 
 } 
  
 //初始化指示器位置 
 public void initCursorPos() { 
  // 初始化動畫 
  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;// 計算偏移量 
 
  Matrix matrix = new Matrix(); 
  matrix.postTranslate(offset, 0); 
  cursor.setImageMatrix(matrix);// 設(shè)置動畫初始位置 
 } 
  
 
 //頁面改變監(jiān)聽器 
 public class MyPageChangeListener implements OnPageChangeListener { 
 
  int one = offset * 2 + bmpw;// 頁卡1 -> 頁卡2 偏移量 
  int two = one * 2;// 頁卡1 -> 頁卡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:圖片停在動畫結(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); 
  } 
 } 
 
} 

從易到難一步步來講。
1、MyPagerAdapter類

一般我們對于適配器的實現(xiàn)總是new一個PageAdapter的實例。我們這里做了一點稍微的更改,將其集合成一個類,內(nèi)容都沒變,只是多了一個構(gòu)造函數(shù)而已。所以針對這個類的具體代碼,我就不再細講,如果對其中的復寫的函數(shù)為什么要這么寫不理解的同學,請看《ViewPager 詳解(二)---詳解四大函數(shù)》

2、initCursorPos()---初始化指示器位置
游標在初始化顯示時,我們要根據(jù)屏幕寬度來顯示游標位置。先看看這部分代碼:

//初始化指示器位置 
public void initCursorPos() { 
 // 初始化動畫 
 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;// 計算偏移量 
 
 Matrix matrix = new Matrix(); 
 matrix.postTranslate(offset, 0); 
 cursor.setImageMatrix(matrix);// 設(shè)置動畫初始位置 
} 

可能有些同學不明白的位置在于,初始化位置的偏移量為什么這么算,下面,我畫了張圖,看下就應該明白了。

2016323115509630.png (960×540)

最后對于偏移的方法,可用的很多,這里仿網(wǎng)上的代碼用了matrix;當然大家可以用其它的偏移方法,一樣。

3、MyPageChangeListener()---頁面改變監(jiān)聽器
代碼如下 :

public class MyPageChangeListener implements OnPageChangeListener { 
 
 int one = offset * 2 + bmpw;// 頁卡1 -> 頁卡2 偏移量 
 int two = one * 2;// 頁卡1 -> 頁卡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:圖片停在動畫結(jié)束位置 
  animation.setDuration(300); 
  cursor.startAnimation(animation); 
 } 

原理是這樣,根據(jù)滑動到的頁面,把游標滑動找指定位置。
這里可能有難度的地方在于,數(shù)學……
我畫了一張圖,解釋從第一個頁面到第二個頁面時的距離為什么是“游標寬度+offset*2”,其它距離類似。

2016323115543616.png (960×540)

這篇就到這了,而且這個難度不太大,講的可能不太細。
源碼中,給大家列出了一個有Tab交互的Demo,圖片如下:

2016323115611176.png (300×533)2016323115625900.png (300×533)

使用Fragment實現(xiàn)ViewPager滑動
使用Fragment實現(xiàn)ViewPager滑動是Android官方比較推薦的一種做法,我們先再來看一下效果圖:
在第一個頁面加一個Btn

2016323115641975.png (300×533)

第一頁面向第二頁面滑動

2016323115656400.png (300×533)

第二頁面向第三個頁面滑動

2016323115713962.png (300×533)

1、適配器實現(xiàn)——FragmentPagerAdapter
先看完整代碼,再細講:

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(); 
 } 
 
} 

這里有三個函數(shù),根據(jù)第一部分的官方文檔,可知,對于FragmentPagerAdapter的派生類,只重寫getItem(int)和getCount()就可以了。
對于構(gòu)造函數(shù),這里申請了一個Fragment的List對象,用于保存用于滑動的Fragment對象,并在創(chuàng)造函數(shù)中初始化:

public FragAdapter(FragmentManager fm,List<Fragment> fragments) { 
 super(fm); 
 // TODO Auto-generated constructor stub 
 mFragments=fragments; 
} 

然后在getItem(int arg0)中,根據(jù)傳來的參數(shù)arg0,來返回當前要顯示的fragment,下面是getItem的官方解釋,難度不大,不再細講。
public abstract Fragment getItem (int position)

Return the Fragment associated with a specified position.

最后,getCount()返回用于滑動的fragment總數(shù);

從構(gòu)造函數(shù)所以看出,我們要構(gòu)造Fragment的集合才行,所以下面我們就先產(chǎn)生我們所需要的Fragment類;
2、三個Fragment類
第一個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> 

在其中加入了一個Btn

Java代碼:

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); 
   
  //對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(), "點擊了第一個fragment的BTN", Toast.LENGTH_SHORT).show(); 
   } 
  }); 
  return view; 
 } 
} 

在onCreateView()中返回要顯示的View,上面這段代碼簡單演示了如何對視圖里的控件進行操作,難度不大,不再細講。第二個Fragment類:

XML代碼:(layout2.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="#ffff00" 
 android:orientation="vertical" > 
  
 
</LinearLayout> 

java代碼:

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; 
 } 
 
} 

第三個Fragment類:
XML代碼:(layout3.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="#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實現(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); 
 } 
 
} 

首先有一個最值得注意的地方:Activity派生自FragmentActivity,其實這是有關(guān)Fragment的基礎(chǔ)知識,只有FragmentActivity才能內(nèi)嵌fragment頁面,普通Activity是不行的。

這段代碼主要分為兩步,第一步:構(gòu)造適配器;第二步:設(shè)定適配器。
先看構(gòu)造適配器的過程:

//構(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)造一個fragment列表,然后將上面的三個Fragment類對應的實例添加進去,最后生成FragAdapter實例。
至于第二步,設(shè)定適配器,沒什么好講的。
4、可能出現(xiàn)的問題
問題:在MainActivity中,當寫到這句:fragments.add(new Fragment1()); 向Fragment列表中添加Fragement對象實例時,會提示“無法將Fragment1()轉(zhuǎn)換為fragment”

解決辦法 :這是因為導入包不一致,一般的問題在于:在Fragment1中導入的是android.app.Fragment, 而在這里導入類確是:android.support.v4.app.Fragment,包不同當然無法轉(zhuǎn)換,統(tǒng)一導入為android.support.v4.app.Fragment之后就正常了
 

相關(guān)文章

  • Android SDK Manager國內(nèi)無法更新的解決方案

    Android SDK Manager國內(nèi)無法更新的解決方案

    本文主要介紹Android SDK Manager國內(nèi)無法更新的解決方案,這里提供了解決方法,及簡單說明實現(xiàn)流程,有興趣的小伙伴可以參考下
    2016-09-09
  • Android編程雙重單選對話框布局實現(xiàn)與事件監(jiān)聽方法示例

    Android編程雙重單選對話框布局實現(xiàn)與事件監(jiān)聽方法示例

    這篇文章主要介紹了Android編程雙重單選對話框布局實現(xiàn)與事件監(jiān)聽方法,涉及Android雙重單選對話框的界面布局與事件監(jiān)聽、響應等相關(guān)操作技巧,需要的朋友可以參考下
    2017-10-10
  • AndroidStudio3 支持 Java8 了請問你敢用嗎

    AndroidStudio3 支持 Java8 了請問你敢用嗎

    Google 發(fā)布了 AS 3.0,以及一系列的 Support 包,有意思的新東西挺多,AS3里面有一個亮眼的特性就是支持J8。接下來通過本文給大家分享AndroidStudio3 支持 Java8 的相關(guān)內(nèi)容,感興趣的朋友一起看看吧
    2017-11-11
  • Android實現(xiàn)全屏截圖或長截屏功能

    Android實現(xiàn)全屏截圖或長截屏功能

    這篇文章主要為大家詳細介紹了Android實現(xiàn)全屏截圖或長截屏功能,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-05-05
  • Android 標準Intent的使用詳解

    Android 標準Intent的使用詳解

    這篇文章主要介紹了Android 標準Intent的使用詳解的相關(guān)資料,需要的朋友可以參考下
    2017-03-03
  • Android中圖片圓角三種實現(xiàn)方法

    Android中圖片圓角三種實現(xiàn)方法

    大家好,本篇文章主要講的是Android中圖片圓角三種實現(xiàn)方法,感興趣的同學趕快來看一看吧,對你有幫助的話記得收藏一下
    2022-02-02
  • Android SeekBar在刷新使用中需要注意的問題

    Android SeekBar在刷新使用中需要注意的問題

    SeekBar在刷新使用中需要注意的問題:在使用SeekBar的過程中需要注意刷新頻率,避免頻繁刷新造成的性能問題;同時,需要對SeekBar的監(jiān)聽事件進行適當?shù)膬?yōu)化,減少回調(diào)次數(shù),提高響應速度
    2023-05-05
  • Android studio 如何刪除項目 module

    Android studio 如何刪除項目 module

    本篇文章主要介紹了Android studio 如何刪除項目module的相關(guān)知識,具有很好的參考價值。下面跟著小編一起來看下吧
    2017-05-05
  • android指定DatePickerDialog樣式并不顯示年的實現(xiàn)代碼

    android指定DatePickerDialog樣式并不顯示年的實現(xiàn)代碼

    下面小編就為大家?guī)硪黄猘ndroid指定DatePickerDialog樣式并不顯示年的實現(xiàn)代碼。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧,祝大家游戲愉快哦
    2016-08-08
  • Android多線程之同步鎖的使用

    Android多線程之同步鎖的使用

    本篇文章主要介紹了Android多線程之同步鎖的使用,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-08-08

最新評論