1. 图片轮播
学习资料:
十分感谢两位同学
两个同学的思路大同小异,都没有采用将getCount()
设置Integer.MAX_VALUE
的常规思路,采用原理是基本是一致的,看懂一个,另外一个自然也就懂了
在ViewPager
的item
队列中,首先添加原队尾,也就是5
;然后再添加原队首1
,最终比实际item
多出两个
原理图也直接拿来用了,我实际用坚果手机测试,这种方式带来的优化效果比设置 Integer.MAX_VALUE
方式几乎可以忽略。占用内存相差不大,使用这种方式,循环多次时,占用内存增长速度感觉稍微慢一点点。个人感觉,性能方面两种方式基本是一样的。但这种方式,更符合真正的循环轮播吧
1.1 BannerAdapter
主要是startUpdate()
方法的重写
public class BannerAdapter extends PagerAdapter {
private List<String> imageViews = new ArrayList<>();
private int bannerSize;
@Override
public int getCount() {
bannerSize = imageViews.size();
return bannerSize;
}
public void setImageViews(List<String> imageViews) {
this.imageViews.addAll(imageViews);
}
@Override
public boolean isViewFromObject(View view, Object object) {
return view == object;
}
@Override
public Object instantiateItem(ViewGroup container, int position) {
ImageView imageView = new ImageView(container.getContext());
imageView.setScaleType(ImageView.ScaleType.CENTER_CROP);
String url = imageViews.get(position);
Glide.with(container.getContext()).load(url).into(imageView);
container.addView(imageView);
return imageView;
}
@Override
public void destroyItem(ViewGroup container, int position, Object object) {
container.removeView((View) object);
}
@Override
public void startUpdate(ViewGroup container) {
ViewPager vp = (ViewPager) container;
int position = vp.getCurrentItem();
if (position == 0) {
position = bannerSize - 2;
vp.setCurrentItem(position, false);
} else if (position == (bannerSize - 1)) {
position = 1;
vp.setCurrentItem(position, false);
}
}
}
在startUpdate()
方法中,根据滑动到的当前的position
来确定最终的CurrentItem
。当position
为0
时,就设置CurrentItem
为原队尾;当positon
为新队列的最后时,就设置CurrentItem
为1
vp.setCurrentItem(position, false)
中第2个参数,是去除动画效果
1.2 Activity
布局文件:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:clipChildren="false">
<android.support.v4.view.ViewPager
android:id="@+id/banner_activity_vp"
android:layout_width="match_parent"
android:layout_height="180dp"
android:layout_centerInParent="true"
android:layout_marginLeft="30dp"
android:layout_marginRight="30dp"
android:clipChildren="false" />
</RelativeLayout>
RelativeLayout
和ViewPager
的布局中都使用,android:clipChildren="false"
,主要是为了让ViewPager
一屏幕可以显示多个item
,显示多少则由android:layout_marginLeft
这个系列的属性控制
Activity代码:
public class BannerActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_banner);
initView();
}
private void initView() {
List<String> imageViews = new ArrayList<>();
imageViews.add(Urls.IMG_URL3);//添加队尾
imageViews.add(Urls.IMG_URL1);
imageViews.add(Urls.IMG_URL2);
imageViews.add(Urls.IMG_URL3);
imageViews.add(Urls.IMG_URL1);//添加队首
ViewPager vp = (ViewPager) findViewById(R.id.banner_activity_vp);
BannerAdapter adapter = new BannerAdapter();
vp.setPageMargin(10);
adapter.setImageViews(imageViews);
vp.setOffscreenPageLimit(3);
vp.setAdapter(adapter);
vp.setCurrentItem(1);//显示img_url_1
}
}
在imageViews
中,多添加了两个img_url
,由于添加了一个新的队首,设置vp.setCurrentItem(1)
2. 切换动画
学习资料
2.1 BannerTransformer
public class BannerTransformer implements ViewPager.PageTransformer {
private static final float MAX_SCALE = 1.0f;
private static final float MIN_SCALE = 0.8f;
@Override
public void transformPage(View page, float position) {
if (position < -1) {//(-Infinity,-1)
page.setScaleX(MIN_SCALE);
page.setScaleY(MIN_SCALE);
} else if (position <= 0) {//[-1,0]
float scaleFactor = MIN_SCALE+(1-Math.abs(position))*(MAX_SCALE-MIN_SCALE);
page.setScaleX(scaleFactor);
page.setScaleY(scaleFactor);
} else if (position <= 1) {//(0,1]
float scaleFactor = MIN_SCALE+(1-Math.abs(position))*(MAX_SCALE-MIN_SCALE);
page.setScaleX(scaleFactor);
page.setScaleY(scaleFactor);
} else {//(1,Infinity)
page.setScaleX(MIN_SCALE);
page.setScaleY(MIN_SCALE);
}
}
}
切换动画使用vp.setPageTransformer(true, new BannerTransformer())
动画就涉及到了一个缩放效果
一个page的position为0代表它处于中间,为1代表它完全处于右边,为-1代表它完全处于左边
由page_a
切换到page_b
,page_a
从0.0f --> -1.0f
,而page_b
从1.0f --> 0.0f
2.2 问题
遇到一个坑,查了一个多小时,暂时也没有考虑出到底啥原因
一开始在BannerAdapter
中,重写的是finshUpdate()
,右滑没有问题,左滑遇到一个问题
手指从右侧向左滑时,3个page
每循环一次后,就会出现问题,左面的page
会闪一下,然后大小变成原始的正常大小,设置的缩放效果无效
将BannerAdapter
中的finshUpdate()
改为startUpdata()
方法后,就好了
若使用设置Integer.MAX_VALUE
的方式,使用同一个切换动画,没有任何问题。
3. 最后
腊月二十九,2016,马上就真的过去了,新年新气象
有错误,请指出
共勉 :)