android之ViewPager介绍

Posted 王俊凯夫人

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了android之ViewPager介绍相关的知识,希望对你有一定的参考价值。

一:ViewPager的含义:
ViewPager的功能就是可以使视图滑动,就像Lanucher左右滑动那样。
ViewPager用于实现多页面的切换效果,该类存在于Google的兼容包android-support-v4.jar里面.
ViewPager:
   1)ViewPager类直接继承了ViewGroup类,所有它是一个容器类,可以在其中添加其他的view类。
   2)ViewPager类需要一个PagerAdapter适配器类给它提供数据。
   3)ViewPager经常和Fragment一起使用,并且提供了专门的FragmentPagerAdapter和FragmentStatePagerAdapter类供Fragment中    的ViewPager使用。
  4)在编写ViewPager的应用的使用,还需要使用两个组件类分别是PagerTitleStrip类和PagerTabStrip类,PagerTitleStrip类直接继承  自ViewGroup类,而PagerTabStrip类继承PagerTitleStrip类,所以这两个类也是容器类。但是有一点需要注意,在定义XML的layout  的时候,这两个类必须是ViewPager标签的子标签,不然会出错。

如下:

<android.support.v4.view.ViewPager
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/viewPager"
        android:layout_gravity="center">
    	<android.support.v4.view.PagerTitleStrip
    	    android:id="@+id/pagertitle"
    	    android:layout_width="match_parent"
	    android:layout_height="wrap_content"
            android:layout_gravity="top"/>
</android.support.v4.view.ViewPager>
其中ViewPager为多页显示控件, pagerTitleStrip用于显示当前页面的标题,并且其android:layout_width必须设置成match_parent

二:完成一个简单的ViewPager例子
 a)在布局文件中配置ViewPager
 <android.support.v4.view.ViewPager
        android:id="@+id/viewpager"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"/> 
 b)在Activity中获取这这个ViewPager组件.
   并且可以设置全屏效果:
   requestWindowFeature(Window.FEATURE_NO_TITLE);
   getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
WindowManager.LayoutParams.FLAG_FULLSCREEN);
 c)加载要显示的页卡:初始化数据,即每一个页面显示的View.
   将其View添加到List集合中。
   LayoutInflater lf=getLayoutInflater();
   list=new ArrayList<View>();
   list.add(lf.inflate(R.layout.activity_layout1,null));
   list.add(lf.inflate(R.layout.activity_layout2,null));
   list.add(lf.inflate(R.layout.activity_layout3,null));
 d)给ViewPager设置数据适配器PagerAdapter
   viewPager.setAdapter(PagerAdatpser);
  实现PagerAdapter至少要实现四个方法:
    1)instantiateItem(ViewGroup, int)
        对显示的资源进行初始化。将显示的View
加入到ViewGroup中,然后作为返回值返回。
//container.addView(list.get(position));
//return list.get(position);
    2)destroyItem(ViewGroup, int, Object)
        PagerAdapter有一个缓存范围,如果在滑动过程
中超过了缓存范围,就会调用这个方法,销毁资源。
//container.removeView(list.get(position));
    3)getCount()获取要滑动的控件的数量
    4)isViewFromObject(View, Object)
       来判断显示的是否是同一个视图,一般将两个参数进行比较返回即可。

代码如下:

public class MainActivity extends Activity {

	private ViewPager pager;
	//每一个界面
	private List<View> views;
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		requestWindowFeature(Window.FEATURE_NO_TITLE);
		getWindow().setFlags(LayoutParams.FLAG_FORCE_NOT_FULLSCREEN, 
				LayoutParams.FLAG_FORCE_NOT_FULLSCREEN);
		setContentView(R.layout.activity_main);
		pager=(ViewPager) findViewById(R.id.pager);
		views=new ArrayList<View>();
		LayoutInflater li=getLayoutInflater();
		views.add(li.inflate(R.layout.f1, null));
		views.add(li.inflate(R.layout.f2, null));
		views.add(li.inflate(R.layout.f3, null));
		//需要给ViewPager设置适配器
		PagerAdapter adapter=new PagerAdapter() {
			
			@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 views.size();
			}

			//对超出范围的资源进行销毁
			@Override
			public void destroyItem(ViewGroup container, int position,
					Object object) {
				// TODO Auto-generated method stub
				//super.destroyItem(container, position, object);
				container.removeView(views.get(position));
			}
			//对显示的资源进行初始化
			@Override
			public Object instantiateItem(ViewGroup container, int position) {
				// TODO Auto-generated method stub
				//return super.instantiateItem(container, position);
				container.addView(views.get(position));
				return views.get(position);
			}
			
		};
		pager.setAdapter(adapter);
		
		//给ViewPager添加事件监听
		pager.setOnPageChangeListener(new OnPageChangeListener() {
			
			@Override
			public void onPageSelected(int arg0) {
				// TODO Auto-generated method stub
				Toast.makeText(MainActivity.this, "您选择了:"+arg0+"页面", 0).show();
			}
			
			@Override
			public void onPageScrolled(int arg0, float arg1, int arg2) {
				// TODO Auto-generated method stub
				
			}
			
			@Override
			public void onPageScrollStateChanged(int arg0) {
				// TODO Auto-generated method stub
				
			}
		});
	}
}
布局文件就一个ViewPager,就不贴代码了,

效果如图:


三:实现了滑动页面的的简单实现方法,但有时,仅仅实现页面滑动是不够的,还要有标题栏才会显得更友好。
所以需要使用android.support.v4包中的两个控件PagerTabStrip与PagerTitleStrip,他们都是用来实现标题栏的,但各自有些不同。
  a)PagerTitleStrip是ViewPager的一个关于当前页面、上一个页面和下一个页面的一个非交互的指示器。
它经常作为ViewPager控件的一个子控件被被添加
在XML布局文件中。在布局文件中,它必须作为子
控件添加在ViewPager中。而且要使用它的
android:layout_gravity性设置为TOP或BOTTOM来
将标题显示在ViewPager的顶部或底部。每个页面的标
题是通过适配器的getPageTitle(int)函数提供给ViewPager
的标题内容。
  b)PagerTabStrip是ViewPager的一个关于当前页面、上一个页面和下一个页面的一个可交互的指示器。
它经常作为ViewPager控件的一个子控件被被添加
在XML布局文件中。在你的布局文件中,将它作为
子控件添加在ViewPager中。而且要将它的 
android:layout_gravity 属性设置为TOP或BOTTOM
来将它显示在ViewPager的顶部或底部。每个页面
的标题是通过适配器的getPageTitle(int)函数提
供给ViewPager的标题内容。
  注意:其实这两个实现的效果基本差不多,但有两点不同:
          1)PagerTabStrip在当前页面下,会有一个下划线条来提示当前页面的Tab是哪个。
          2)PagerTabStrip的Tab是可以点击的,当用户点击某一个Tab时,当前页面就
 会跳转到这个页面,而PagerTitleStrip
 则没这个功能。
         3)做标题栏需要重写适配器中的
getPageTitle(int position)返回标题内容。
所有标题可以放在List集合中,每次在
该方法中返回指定位置的标题。
        4)可以自定义PagerTabStrip的样式
tabStrip = (PagerTabStrip)findViewById(R.id.pagertab);
//取消tab下面的长横线
tabStrip.setDrawFullUnderline(true);
//设置tab的背景色
tabStrip.setBackgroundColor(Color.BLUE);
//设置当前tab页签的下划线颜色
tabStrip.setTabIndicatorColor(Color.RED);
tabStrip.setTextColor(Color.CYAN);
实例代码:

布局:

<RelativeLayout 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.support.v4.view.ViewPager
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:id="@+id/pager" >
        <android.support.v4.view.PagerTabStrip
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_gravity="top"
            android:id="@+id/title"
            />
          
     </android.support.v4.view.ViewPager>

</RelativeLayout>

activity:

public class TestActivity extends Activity {

	private ViewPager pager;
	//每一个界面
	private List<View> views;
	//标题
	private String[] titles={"新闻","娱乐","军事"};
	
	private PagerTabStrip t;
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		requestWindowFeature(Window.FEATURE_NO_TITLE);
		getWindow().setFlags(LayoutParams.FLAG_FORCE_NOT_FULLSCREEN, 
				LayoutParams.FLAG_FORCE_NOT_FULLSCREEN);
		setContentView(R.layout.activity_test);
		
		t=(PagerTabStrip) findViewById(R.id.title);
		//自定义table的样式
		t.setBackgroundColor(Color.MAGENTA);
		t.setTextColor(Color.BLUE);
		t.setTextSize(0, 25f);
		t.setTabIndicatorColor(Color.GREEN);
		
		pager=(ViewPager) findViewById(R.id.pager);
		views=new ArrayList<View>();
		LayoutInflater li=getLayoutInflater();
		views.add(li.inflate(R.layout.f1, null));
		views.add(li.inflate(R.layout.f2, null));
		views.add(li.inflate(R.layout.f3, null));
		//需要给ViewPager设置适配器
		PagerAdapter adapter=new PagerAdapter() {
			
			//提供标题的内容
			@Override
			public CharSequence getPageTitle(int position) {
				// TODO Auto-generated method stub
				return titles[position];
			}
			
			@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 views.size();
			}

			//对超出范围的资源进行销毁
			@Override
			public void destroyItem(ViewGroup container, int position,
					Object object) {
				// TODO Auto-generated method stub
				//super.destroyItem(container, position, object);
				container.removeView(views.get(position));
			}
			//对显示的资源进行初始化
			@Override
			public Object instantiateItem(ViewGroup container, int position) {
				// TODO Auto-generated method stub
				//return super.instantiateItem(container, position);
				container.addView(views.get(position));
				return views.get(position);
			}
			
		};
		pager.setAdapter(adapter);
	}
}

效果图:


四:PagerTabStrip和PagerTitleStrip都不适合用在实际用途中,

当要在实际运用中,我们就要自己去实现相关的功能。一般使用交互Tab的实现。
   步骤:a)布局文件使用TextView完成tab的切换卡。
           使用ImageView进行分割
           使用ViewPager进行翻页
         b)在Activity中获取TextView对象,
  并分别设置点击事件。点击TextView
  可以切换当前选中的卡片:
   mPager.setCurrentItem(index);
c)在Activity中获取ViewPager对象。
  添加卡片内容,设置PagerAdapter适配器。
d)给ViewPager对象设置setOnPageChangeListener();
  可以监听某个卡片被选中的事件监听器。
  修改TextView的字体大小和字体颜色。
实例:

布局:

<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">
    <LinearLayout 
        android:layout_width="match_parent"
        android:layout_height="40dp"
        android:orientation="horizontal"
        android:background="#D5D4C3"
        android:gravity="center">
        <TextView 
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:textSize="18sp"
            android:layout_weight="1"
            android:text="聊天"
            android:id="@+id/tv1"
            android:gravity="center"/>
        <TextView 
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:textSize="18sp"
            android:layout_weight="1"
            android:text="发现"
            android:id="@+id/tv2"
            android:gravity="center"/>
        <TextView 
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:textSize="18sp"
            android:layout_weight="1"
            android:text="通讯录"
            android:id="@+id/tv3"
            android:gravity="center"/>
    </LinearLayout>
    <View 
        android:layout_width="match_parent"
        android:layout_height="2dp"
        android:background="#68AE94"/>
    <android.support.v4.view.ViewPager
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:id="@+id/pager" />
</LinearLayout>
activity:

public class MyTabActivity extends Activity {

	private ViewPager pager;
	private List<View> views;
	//放标签页
	private List<TextView>tvs=new ArrayList<TextView>();
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_my_tab);
		initTextView();
		//初始化ViewPager组件
		initView();
		initViewPager();
	}
	
	public void initTextView() {
		// TODO Auto-generated method stub
		TextView tv1=(TextView) findViewById(R.id.tv1);
		tv1.setTextColor(Color.BLUE);
		TextView tv2=(TextView) findViewById(R.id.tv2);
		TextView tv3=(TextView) findViewById(R.id.tv3);
		//添加点击事件
		//OnClickListener click=new MyClickListener();
		tv1.setOnClickListener(new MyClickListener(0));
		tv2.setOnClickListener(new MyClickListener(1));
		tv3.setOnClickListener(new MyClickListener(2));
		tvs.add(tv1);
		tvs.add(tv2);
		tvs.add(tv3);
	}	
	private class MyClickListener implements OnClickListener{

		private int index;
		public MyClickListener(int index) {
			// TODO Auto-generated constructor stub
			this.index=index;
		}
		@Override
		public void onClick(View v) {
			// TODO Auto-generated method stub
			//改变ViewPager当前显示页面
			pager.setCurrentItem(index);
		}
	}

	//初始化ViewPager中显示的数据
	public void initView() {
		// TODO Auto-generated method stub
		views=new ArrayList<View>();
		LayoutInflater li=getLayoutInflater();
		views.add(li.inflate(R.layout.f1, null));
		views.add(li.inflate(R.layout.f2, null));
		views.add(li.inflate(R.layout.f3, null));
	}
	
	
	public void initViewPager() {
		// TODO Auto-generated method stub
		pager=(ViewPager) findViewById(R.id.pager);
		PagerAdapter adapter=new MyPagerAdapter();
		pager.setAdapter(adapter);
		pager.setOnPageChangeListener(new OnPageChangeListener() {
			
			@Override
			public void onPageSelected(int index) {
				// TODO Auto-generated method stub
				for(int i=0;i<tvs.size();i++){
					if(i==index){
						tvs.get(i).setTextColor(Color.BLUE);
					}else{
						tvs.get(i).setTextColor(Color.rgb(55,55,55));
					}
				}
			}
			
			@Override
			public void onPageScrolled(int arg0, float arg1, int arg2) {
				// TODO Auto-generated method stub
				
			}
			
			@Override
			public void onPageScrollStateChanged(int arg0) {
				// TODO Auto-generated method stub
				
			}
		});
	}	
	private class MyPagerAdapter extends PagerAdapter{

		@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 views.size();
		}
		//对超出范围的资源进行销毁
		@Override
		public void destroyItem(ViewGroup container, int position,
				Object object) {
			// TODO Auto-generated method stub
			//super.destroyItem(container, position, object);
			container.removeView(views.get(position));
		}
		//对显示的资源进行初始化
		@Override
		public Object instantiateItem(ViewGroup container, int position) {
			// TODO Auto-generated method stub
			//return super.instantiateItem(container, position);
			container.addView(views.get(position));
			return views.get(position);
		}
	}
}

效果图:


五:前面介绍ViewPager的普通实现方法,但android官方最推荐的一种实现方法却是使用fragment。
使用Fragment的时候需要使用的适配器为FragmentPagerAdapter。
FragmentPagerAdapter是PagerAdapter的子类,专门用于处理Fragment翻页的适配器。
这个适配器最好用于有限个静态fragment页面的管理。尽管不可见的视图有时会被销毁,但用户所有访问过的fragment都会被保存在内存中。因此fragment实例会保存大量的各种状态,这就造成了很大的内存开销。
所以如果要处理大量的页面切换,建议使用FragmentStatePagerAdapter.对于FragmentPagerAdapter的派生类,只需要重写getItem(int)和getCount()就可以了。

步骤:a)写一个Activity继承FragmentActivity,
        专门处理Frament的Activity,重写onCreate方法
List<Fragment> fragments=new ArrayList<Fragment>();
fragments.add(new Fragment1());
fragments.add(new Fragment2());
fragments.add(new Fragment3()); 
FragAdapter adapter = new FragAdapter(
getSupportFragmentManager(), fragments);      
        //设定适配器
        ViewPager vp = (ViewPager)findViewById(R.id.viewpager);
        vp.setAdapter(adapter);


      b)FragAdapter是extends FragmentPagerAdapter。
        //提供构造器
private List<Fragment> mFragments;
public FragAdapter(FragmentManager fm,List<Fragment> fragments) {
super(fm);
mFragments=fragments;
}
        //重写方法
@Override
public Fragment getItem(int arg0) {
return mFragments.get(arg0);
}
@Override
public int getCount() {
return mFragments.size();
}

实例:

布局:和上一个一样

activity:

public class MyFragmentActivity extends FragmentActivity {

	private ViewPager pager;
	private List<Fragment>list;
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_my_tab);
		list=new ArrayList<Fragment>();
		list.add(new ChatFragment());
		list.add(new FindFragment());
		list.add(new ContactFragment());
		pager=(ViewPager) findViewById(R.id.pager);
		MyAdapter adapter=new MyAdapter(getSupportFragmentManager());
		pager.setAdapter(adapter);
	}
	
	//处理Fragment和ViewPager的适配器
	private class MyAdapter extends FragmentPagerAdapter{

		public MyAdapter(FragmentManager fm) {
			super(fm);
			// TODO Auto-generated constructor stub
		}

		@Override
		public Fragment getItem(int arg0) {
			// TODO Auto-generated method stub
			return list.get(arg0);
		}

		@Override
		public int getCount() {
			// TODO Auto-generated method stub
			return list.size();
		}
		
	}
}
效果图:


有想要代码的可以:下载链接




以上是关于android之ViewPager介绍的主要内容,如果未能解决你的问题,请参考以下文章

Android:使用Tab检测单个片段viewpager

Android TabLayout ViewPager 不会在 backstack 上膨胀标签片段

android之ViewPager介绍

Android使用片段在viewpager中的页面滚动上放置动画

Android:导航抽屉片段内的Viewpager

ViewPager 未从代码、android、eclipse 更新