本篇包含:实现一屏多页、自定义PageTransformer、OnPageChangeListener方法参数的解释和应用
使用
ViewPager在开发过程中的使用就不用多说了,那肯定是很多场景都会用到,其实使用它很简单,就两部就能使用了:
- 拿到一个ViewPager实例
- 设置一个适配器
其中适配器的实现方式有三种:
方式 | 场景 |
---|---|
PagerAdapter | 主要是用在比较单一的View时比较方便 |
FragmentPagerAdapter | 主要是用在内容为Fragment时,当这些Fragment比较少时使用 |
FragmentStatePagerAdapter | 主要是用在内容为Fragment时,当Fragment比较多时使用,可保存Fragment的状态 |
具体的使用方式这里就不介绍了,都比较简单。
注:对于Fragment的创建,尽量做到需要创建时再创建
一屏多页
这里可以先看看实现的效果
难看是难看了点,但是效果是有了,就好比麻雀虽丑,五脏还是俱全的。其实这个知识点,挺简单,并没有想象中那么难,要做到的其实也就几步:
- 给ViewPager的父容器添加 android:clipChildren="false" 和
android:layerType="software"属性 - 为ViewPager添加layout_marginLeft和layout_marginRight,并设一个值
- 再拿到实例,设置Adapter即可
这样基本就能出效果,就能看到下一页的内容能显示在屏幕中了,这两点的含义也比较简单,clipChildren就是是否裁剪子View,默认为true的,layerType这个是用来关闭硬件加速的;第二点的margin是让一页的宽度不要占满整个屏幕,这样才能让下一页或上一页的显示出来。
这时候会发现,每一页都是挨在一起的,我们怎么把他们分开呢,其实在不知道的情况下,我们可以给把自己内容加上一个padding,那么就分开了,但是Android给我们提供了方法pager.setPageMargin,这个就是设置页与页之间的间距的。
这时候细心的同学又会发现一个问题,下一页为什么总是在滚动完之后才被显示出来,会闪一下,这个效果可不友好,这时候其实是由于ViewPager的缓存机制在作怪,我们可以通过pager.setOffscreenPageLimit来解决这个问题,这个值默认是1,所以它每次到下一页都会闪一下,但是不代表它只缓存了一页,其实是三页,这里先不解释为什么,那么我们现在只需要把这个值设置为2,就能让下一页的出来,而且保证不闪一下,可以先这么理解,值为1时,只管当前页一定先加载出来的,值为2时,保证当前页和下一页一定被加载出来了。
好这一部分的实现就到这里。
自定义PageTransformer
这个东西,如果你看过他是什么的话,应该就不怕了
public interface PageTransformer {
void transformPage(View page, float position);
}
你看,是不是纸老虎,就一个方法,两个参数而已,注释我删了=_=|。
其中page和position是配对的,page是position所对应的那个view,其实最关键就是这个position了,它其实也很简单,你可以先这么理解,当所有页都被加载出来之后,这些页面都会对应着一个position。
当静止不滑动的时候,当前的position就是0,它右边的就是1,2,3...它左边的就是-1,-2,-3...所以我们想操作某一页都可以以当前页为基准,那样就所有页面都能操作了。
当滑动时,position就是变化的了,其中从右往左滑的时候,是所有position都变小,即从position到position-1,反之亦然;Android给我们也提供过一些简单的切换效果,我就假设你知道了,这里呢,其实我还可以来个跟简单点的,就只是透明度的变化,左右两边的我都默认为0.3,中间的是1,所以透明度切换的过程就是从1到0.3或者0.3到1。而postion是从postion到postion-1,这里我们可以看到,这是一个分段函数,都是一元一次方程,分界线是postion为0的点,只管[-1,1]的区间的画,大概是这个样子:
求不吐槽照片,电脑画不太熟,太慢=_=|
这个我们知道position就是x,alpha值就是y,在小于0的部分函数关系就是:
y=0.7(1+x)+0.3
大于0的部分函数关系是:
y=0.7(1-postion)+0.3
0的部分归到哪边都一样,所以我们就能写这一部分的代码了
public class AlphaTransformer implements ViewPager.PageTransformer {
@Override
public void transformPage(View page, float position) {
if (position <= 0) {
float alpha = 0.7f * (1 + position) + 0.3f;
page.setAlpha(alpha);
} else {
float alpha = 0.7f * (1 - position) + 0.3f;
page.setAlpha(alpha);
}
}
}
然后再调用pager.setPageTransformer(true,new AlphaTransformer());
这样就实现了。
有没有看到左右两边不那么红了,其实就是因为透明度被改变了,这里就没有弄动态图了,效果已经出来了。有的朋友说等等,间距变大了,对,我这里把间距调大了。。感觉挨在一起不好看,虽然现在也不大好看=_=|。
再来看个复杂点的效果
模拟器录的,有点快,但是多看两遍还是可以看出其规律的。
这里我把ViewPager的布局和加载的Fragment换了下,之前的那个就不大好看了,这里也把页面循环滑动去掉了。
好我们来分析一下,它默认是要显示出后边几页,并像卡片一样叠在下边,这里可以看到是总共三张,这里边用到了透明度变化,x方向偏移,y方向偏移,以及x和y方向的缩放,这个其实是项目中的一个卡片式布局的备选方案,结果没用上,用了透明度那个,当然还有背景颜色的变化下文介绍,还有一个页面跳转的动画比较炫之后会介绍。
我们来通过position分析一下,再不滑动的情况下
(-INF,-1)的时候,即左边的第二页及其左边的,其实是看不到的,所以我们只需要将其透明度设置为0,甚至不管都可以。
[-1,0],即左边第一页,若是滑动的话是从当前页往左滑的时候,透明度从1到0,宽高同比缩小。
(0,3],即右边的1,2,3页,首先x都偏移到了正中间,且宽高都有缩放,y向上偏移还与缩放比有关,这里第三页,滑动时进来的,从0到1的透明度变化,小于3的透明度都是1,都显示出来了的。
(3,INF),即右边4及其以后的页,看不到,直接不管或者透明度为0即可。
其实这个到最后就是使用的和动画有关的东西,变化的情况都告诉了,就是一个函数关系而已,这里就不去仔细分析怎么得出来的了,代码是最好的老师,所以这里就直接上代码。还有一点就是画图太不熟悉。
public class CardTransformer implements ViewPager.PageTransformer {
private static final float MIN_SCALE = 0.9f;
private static final int PAGE_OFFSET = 40;
@SuppressLint("NewApi")
@Override
public void transformPage(View view, float position) {
//position从左往右滑时,根据缓存了多少页,每一页的变化都是currPos到currPos+1;反之则为currPos到currPos-1;所以根据pos的大小可以控制当前页的位置,
//例如下边改变translationX,让后几页一样显示到当前页,当然这里让它能被看到,
//再调整ViewPager的宽高来控制协作控制其位置
int pageWidth = view.getWidth();
int pageHeight = view.getHeight();
float scaleFactor;
if (position < -1) { // [-Infinity,-1)
view.setAlpha(0);
} else if (position <= 0) {
view.setAlpha(1 + position);
scaleFactor = MIN_SCALE + (1 - MIN_SCALE)
* (1 - Math.abs(position));
view.setScaleX(scaleFactor);
view.setScaleY(scaleFactor);
} else {
view.setAlpha(1 + position);
if (position <= 3) {
if (position <= 2) {
view.setAlpha(1);
} else {
view.setAlpha(1 - (position - 2));
}
view.setTranslationX(pageWidth * -position);
} else {
view.setAlpha(0);
}
scaleFactor = MIN_SCALE + (1 - MIN_SCALE)
* (1 - Math.abs(position));
view.setScaleX(scaleFactor);
view.setScaleY(scaleFactor);
view.setTranslationY(-position * PAGE_OFFSET - (pageHeight - pageHeight * scaleFactor) / 2);
}
}
}
这样就可以了。
注:其实举了这两个例子,为的就是想要说明一点postion可以代表ViewPager的每一页,可以自己去操控他,唯一需要配合使用的就是一屏多页和mOffscreenPageLimit参数,来控制在显示的时候是否能被加载并且看见。
OnPageChangeListener方法的参数
使用过ViewPager的朋友应该都知道,想要监听ViewPager的滚动情况,我们都可以使用设置这个监听去实现,而实现它需要实现他的三个方法:
- onPageSelected(int position)
- onPageScrollStateChanged(int state)
- onPageScrolled(int position, float positionOffset, int positionOffsetPixels)
第一个用的应该是最多的,就是某一页被确定选中的时候,被回调,注意我的措辞,被确定选中,而不是直接说选中,即意味如果滑了一大半,且方向是滑了一大半的方向,那么虽然还在滑动,但是这个方法就会被回调。大家可以自己验证一下,我自己通过打log发现的,去看源码之后发现确实是这样这样,在手指拿起的时候就会去分发这个被选中的方法。
第二个是,滑动状态改变的时候回调,有三种状态:
- SCROLL_STATE_IDLE
- SCROLL_STATE_DRAGGING
- SCROLL_STATE_SETTLING
依次表示,滑动结束,手指拖着滑动中,手指离开后滑动,这就包含了滑动的所有阶段。
第三个,也是在监听滚动变化情况的时候用的最多的一个方法,这里有三个参数,
- position:表示变化趋势,滑动完成后如果变大了,那么就是往左滑,反之亦然;在滑动过程中,如果往左滑,position是不变的,直到滑动结束变为1;往右滑,postion是比滑动之前的当前页小1直到结束。
- positionOffset:这个就厉害了,它从小到大就是往左滑(0,1),反之往右滑(1,0),注意到没有0和1的区间都是开区间,这是因为不管是0到1还是1到0都是无限趋近于1结束或开始,而0是在滑动结束之后都会变成0。
- positionOffsetPixels:这个就和positionOffset变化情况一致,只不过它不再是0到1的变化,而是0到Pager宽度的变化情况了。
到这里这一部分的概念和含义都介绍完了,下边我再介绍一下onPageScrolled在实际运用中的技巧,其中主要是postion和positionOffset的应用。
在这个方法中,position和position+1分别代表左边的View和右边的View,这一点很神奇,捋一捋之后发现果然是这样,可参照前边position的介绍;而positionOffset是(0,1)的变化,这样就方便我们做变化效果了。这里举个例子,因为要实现一屏多页,那么如果想要很好的实现背景颜色的渐变,那么目前最好的方法我认为就是通过在这个方法中处理。
这里先上个效果,因为代码实在太简单:
由于是模拟器录制的,然后mp4转gif总感觉不大理想,大家将就看看吧,就是有个渐变的效果,直接上关键代码:
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
int color = mItems.get(position % mItems.size());
if (positionOffset > 0) {
color = Utils.getColor(positionOffset, mItems.get(position % mItems.size()),
mItems.get((position+1) % mItems.size()));
}
mRoot.setBackgroundColor(color);
}
/**
* 根据fraction获取颜色,实现渐变
*/
public static int getColor(float fraction, Object startColor, Object endColor) {
int startInt = (Integer) startColor;
int startA = (startInt >> 24) & 0xff;
int startR = (startInt >> 16) & 0xff;
int startG = (startInt >> 8) & 0xff;
int startB = startInt & 0xff;
int endInt = (Integer) endColor;
int endA = (endInt >> 24) & 0xff;
int endR = (endInt >> 16) & 0xff;
int endG = (endInt >> 8) & 0xff;
int endB = endInt & 0xff;
return (int) ((startA + (int) (fraction * (endA - startA))) << 24) |
(int) ((startR + (int) (fraction * (endR - startR))) << 16) |
(int) ((startG + (int) (fraction * (endG - startG))) << 8) |
(int) ((startB + (int) (fraction * (endB - startB))));
}
第一个方法没啥好解释的,就是保证在切换的时候背景颜色从当前页面的背景颜色变化到下一页的颜色,有可能是上一页也有可能是下一页,所以这里的postion和position+1就体现出高级的用处来了,第二个方法,不是我自己写的,我是从Android的颜色变换的估值器ArgbEvaluator中拿过来的。就是一个根据变化情况,去一个开始颜色到结束颜色的其中一个比例值(ARGB),其实去掉位运算之后就是一个直线的一元一次方程而已。
到这里,这篇想说的内容都写完了,若有不对的地方,请指正,最后奉上源码,其中有的地方注释掉了,保留的这版代码是最后效果的代码,但是我相信看完这一篇之后你一定知道怎么还原每个部分的代码。