
自主实现滑动指示条
先上一个基本效果图:
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数组 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装入数组中 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);// 设置动画初始位置 } //页面改变监听器 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); } else if (currIndex == 2) { animation = new TranslateAnimation(two,0); } break; case 1: if (currIndex == 0) { animation = new TranslateAnimation(offset,one,0); } break; case 2: if (currIndex == 0) { animation = new TranslateAnimation(offset,two,0); } else if (currIndex == 1) { animation = new TranslateAnimation(one,0); } break; } currIndex = arg0; animation.setFillAfter(true);// True:图片停在动画结束位置 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.重点解析
从易到难一步步来讲。
(1)initCursorPos()---初始化指示器位置
游标在初始化显示时,我们要根据屏幕宽度来显示游标位置。先看看这部分代码:
//初始化指示器位置 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);// 设置动画初始位置 } 可能有些同学不明白的位置在于,初始化位置的偏移量为什么这么算,下面,我画了张图,看下就应该明白了。
最后对于偏移的方法,可用的很多,这里仿网上的代码用了matrix;当然大家可以用其它的偏移方法,一样。
(2)MyPagechangelistener()---页面改变监听器
代码如下 :
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); } else if (currIndex == 2) { animation = new TranslateAnimation(two,0); } break; case 1: if (currIndex == 0) { animation = new TranslateAnimation(offset,0); } break; case 2: if (currIndex == 0) { animation = new TranslateAnimation(offset,0); } else if (currIndex == 1) { animation = new TranslateAnimation(one,0); } break; } currIndex = arg0; animation.setFillAfter(true);// True:图片停在动画结束位置 animation.setDuration(300); cursor.startAnimation(animation); } 原理是这样,根据滑动到的页面,把游标滑动找指定位置。
这里可能有难度的地方在于,数学……
我画了一张图,解释从第一个页面到第二个页面时的距离为什么是“游标宽度+offset*2”,其它距离类似。
使用Fragment实现VIEwPager滑动
效果图:
在第一个页面加一个Btn
第一页面向第二页面滑动
第二页面向第三个页面滑动
一些说明:
FragmentPagerAdapter派生自PagerAdapter,它是用来呈现Fragment页面的,这些Fragment页面会一直保存在fragment manager中,以便用户可以随时取用。
这个适配器最好用于有限个静态fragment页面的管理。尽管不可见的视图有时会被销毁,但用户所有访问过的fragment都会被保存在内存中。因此fragment实例会保存大量的各种状态,这就造成了很大的内存开销。所以如果要处理大量的页面切换,建议使用FragmentStatePagerAdapter.
每一个使用FragmentPagerAdapter的VIEwPager都要有一个有效的ID集合,有效ID的集合就是Fragment的集合(感谢夫诸同学的提示)
对于FragmentPagerAdapter的派生类,只需要重写getItem(int)和getCount()就可以了。
具体实现:
1、适配器实现――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(); } } 这里有三个函数,根据第一部分的官方文档,可知,对于FragmentPagerAdapter的派生类,只重写getItem(int)和getCount()就可以了。
对于构造函数,这里申请了一个Fragment的List对象,用于保存用于滑动的Fragment对象,并在创造函数中初始化:
public FragAdapter(FragmentManager fm,List<Fragment> fragments) { super(fm); // Todo auto-generated constructor stub mFragments=fragments; } 然后在getItem(int arg0)中,根据传来的参数arg0,来返回当前要显示的fragment。
最后,getCount()返回用于滑动的fragment总数;
从构造函数所以看出,我们要构造Fragment的集合才行,所以下面我们就先产生我们所需要的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
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不太熟悉的同学,先看看这篇文章:《AndroID Fragment完全解析,关于碎片你所需知道的一切》
第二个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>
public class Fragment2 extends Fragment { @OverrIDe public VIEw onCreateVIEw(LayoutInflater inflater,Bundle savedInstanceState) { // Todo auto-generated method stub VIEw vIEw=inflater.inflate(R.layout.layout2,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,Bundle savedInstanceState) { // Todo auto-generated method stub VIEw vIEw=inflater.inflate(R.layout.layout3,false); return vIEw; } } 3、主activity实现
核心代码:
public class MainActivity extends FragmentActivity { @OverrIDe protected voID onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentVIEw(R.layout.activity_main); //构造适配器 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); } } 首先有一个最值得注意的地方:Activity派生自FragmentActivity,其实这是有关Fragment的基础知识,只有FragmentActivity才能内嵌fragment页面,普通Activity是不行的。
这段代码主要分为两步,第一步:构造适配器;第二步:设定适配器。
先看构造适配器的过程:
//构造适配器 List<Fragment> fragments=new ArrayList<Fragment>(); fragments.add(new Fragment1()); fragments.add(new Fragment2()); fragments.add(new Fragment3()); FragAdapter adapter = new FragAdapter(getSupportFragmentManager(),fragments);
构造一个fragment列表,然后将上面的三个Fragment类对应的实例添加进去,最后生成FragAdapter实例。
至于第二步,设定适配器,没什么好讲的。
以上是内存溢出为你收集整理的Android中ViewPager实现滑动条及与Fragment结合的实例教程全部内容,希望文章能够帮你解决Android中ViewPager实现滑动条及与Fragment结合的实例教程所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)