近期打开爱奇艺的时候,突然发现它首页的轮播图跟以前不一样了,左右两个图都能显示一部分,且有滑动时缩放的效果,一直不知咋样实现,然后在鸿洋的微信公众号中看到了一篇仿魅族Banner效果的文章,最后在参考中放出了链接,然后就参考了这篇文章,就在此做个记录。废话不多说了,先上个效果图
布局文件
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:clipChildren="false"
android:orientation="vertical">
<com.ly.viewpagerdemo.LoopViewPager
android:id="@+id/looviewpager"
android:layout_width="match_parent"
android:layout_height="200dp"
android:layout_marginLeft="20dp"
android:layout_marginRight="20dp" />
LinearLayout>
这个属性android:clipChildren 很关键,是否裁剪子view
这里的LoopViewPager,是参考开源框架 LoopingViewPager来的,使用Integer.MAX_VALUE 来实现左右循环滚动的,使用方式是一致的
package com.ly.viewpagerdemo;
import android.content.Context;
import android.os.Handler;
import android.os.Message;
import android.support.v4.view.PagerAdapter;
import android.support.v4.view.ViewPager;
import android.util.AttributeSet;
import android.util.DisplayMetrics;
import android.view.WindowManager;
import android.view.animation.Interpolator;
import android.widget.Scroller;
import java.lang.ref.WeakReference;
import java.lang.reflect.Field;
import java.util.concurrent.atomic.AtomicBoolean;
public class LoopViewPager extends ViewPager {
OnPageChangeListener mOuterPageChangeListener;
private LoopPagerAdapterWrapper mAdapter;
private MyHandler mHandler;
private final static int HANDLE_LOOP_MSG = 101;
private AtomicBoolean isAutoLoop = new AtomicBoolean();
@Override
public void setAdapter(PagerAdapter adapter) {
mAdapter = new LoopPagerAdapterWrapper(adapter);
super.setAdapter(mAdapter);
isAutoLoop.set(false);
setCurrentItem(0, false);
}
@Override
public PagerAdapter getAdapter() {
return mAdapter != null ? mAdapter.getRealAdapter() : mAdapter;
}
/**
* 获取当前真实的item
*
* @return
*/
public int getRealItem() {
return mAdapter != null ? mAdapter.toRealPosition(super.getCurrentItem()) : 0;
}
public void setCurrentItem(int item, boolean smoothScroll) {
int realItem = mAdapter.toInnerPosition(item);
super.setCurrentItem(realItem, smoothScroll);
}
@Override
public void setOnPageChangeListener(OnPageChangeListener listener) {
mOuterPageChangeListener = listener;
}
public LoopViewPager(Context context) {
super(context);
init();
}
public LoopViewPager(Context context, AttributeSet attrs) {
super(context, attrs);
init();
}
private void init() {
super.setOnPageChangeListener(onPageChangeListener);
try {
Field scrollerField = ViewPager.class.getDeclaredField("mScroller");
scrollerField.setAccessible(true);
Field interpolator = ViewPager.class.getDeclaredField("sInterpolator");
interpolator.setAccessible(true);
Scroller scroller = new Scroller(getContext(), (Interpolator) interpolator.get(null)) {
@Override
public void startScroll(int startX, int startY, int dx, int dy, int duration) {
//控制滑动速度
super.startScroll(startX, startY, dx, dy, (int) (1300 * (double) Math.abs(dx) / getWidth(getContext())));
}
};
scrollerField.set(this, scroller);
} catch (Exception e) {
e.printStackTrace();
}
}
public void autoLoop(boolean isAuto) {
if (mHandler == null) {
mHandler = new MyHandler(getContext());
}
if (isAuto) {
mHandler.sendEmptyMessageDelayed(HANDLE_LOOP_MSG, 3000);
} else {
mHandler.removeCallbacksAndMessages(null);
}
isAutoLoop.set(isAuto);
}
private OnPageChangeListener onPageChangeListener = new OnPageChangeListener() {
private float mPreviousOffset = -1;
private float mPreviousPosition = -1;
@Override
public void onPageSelected(int position) {
int realPosition = mAdapter.toRealPosition(position);
if (mPreviousPosition != realPosition) {
mPreviousPosition = realPosition;
if (mOuterPageChangeListener != null) {
mOuterPageChangeListener.onPageSelected(realPosition);
}
}
}
@Override
public void onPageScrolled(int position, float positionOffset,
int positionOffsetPixels) {
int realPosition = position;
if (mAdapter != null) {
realPosition = mAdapter.toRealPosition(position);
if (positionOffset == 0
&& mPreviousOffset == 0
&& (position == 0 || position == mAdapter.getCount() - 1)) {
setCurrentItem(realPosition, false);
}
}
mPreviousOffset = positionOffset;
if (mOuterPageChangeListener != null) {
if (realPosition != mAdapter.getRealCount() - 1) {
mOuterPageChangeListener.onPageScrolled(realPosition,
positionOffset, positionOffsetPixels);
} else {
if (positionOffset > .5) {
mOuterPageChangeListener.onPageScrolled(0, 0, 0);
} else {
mOuterPageChangeListener.onPageScrolled(realPosition,
0, 0);
}
}
}
}
@Override
public void onPageScrollStateChanged(int state) {
switch (state) {
case SCROLL_STATE_DRAGGING:
if (isAutoLoop.get()) {
mHandler.removeCallbacksAndMessages(null);
}
break;
case SCROLL_STATE_IDLE:
if (isAutoLoop.get()) {
mHandler.sendEmptyMessageDelayed(HANDLE_LOOP_MSG, 3000);
}
break;
case SCROLL_STATE_SETTLING:
break;
}
if (mOuterPageChangeListener != null) {
mOuterPageChangeListener.onPageScrollStateChanged(state);
}
}
};
private class MyHandler extends Handler {
WeakReference mWeakReference;
public MyHandler(Context context) {
mWeakReference = new WeakReference<>(context);
}
@Override
public void handleMessage(Message msg) {
final Context context = mWeakReference.get();
if (context == null) {
return;
}
switch (msg.what) {
case HANDLE_LOOP_MSG:
int curItem = getCurrentItem();
setCurrentItem(++curItem);
break;
}
}
}
public int getWidth(Context context) {
WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
DisplayMetrics outMetrics = new DisplayMetrics();
wm.getDefaultDisplay().getMetrics(outMetrics);
return outMetrics.widthPixels;
}
}
package com.ly.viewpagerdemo;
import android.os.Parcelable;
import android.support.v4.view.PagerAdapter;
import android.view.View;
import android.view.ViewGroup;
public class LoopPagerAdapterWrapper extends PagerAdapter {
private PagerAdapter mAdapter;
LoopPagerAdapterWrapper(PagerAdapter adapter) {
this.mAdapter = adapter;
}
@Override
public void notifyDataSetChanged() {
super.notifyDataSetChanged();
}
public int toInnerPosition(int realPosition) {
return realPosition + getCount() / 2 - getCount() % getRealCount();
}
int toRealPosition(int position) {
return position % getRealCount();
}
@Override
public int getCount() {
return Integer.MAX_VALUE;
}
public int getRealCount() {
return mAdapter.getCount();
}
public PagerAdapter getRealAdapter() {
return mAdapter;
}
@Override
public Object instantiateItem(ViewGroup container, int position) {
int realPosition = position % getRealCount();
return mAdapter.instantiateItem(container, realPosition);
}
@Override
public void destroyItem(ViewGroup container, int position, Object object) {
int realPosition = position % getRealCount();
mAdapter.destroyItem(container, realPosition, object);
}
@Override
public void finishUpdate(ViewGroup container) {
mAdapter.finishUpdate(container);
}
@Override
public boolean isViewFromObject(View view, Object object) {
return mAdapter.isViewFromObject(view, object);
}
@Override
public void restoreState(Parcelable bundle, ClassLoader classLoader) {
mAdapter.restoreState(bundle, classLoader);
}
@Override
public Parcelable saveState() {
return mAdapter.saveState();
}
@Override
public void startUpdate(ViewGroup container) {
mAdapter.startUpdate(container);
}
@Override
public void setPrimaryItem(ViewGroup container, int position, Object object) {
mAdapter.setPrimaryItem(container, position, object);
}
}
package com.ly.viewpagerdemo;
import android.content.Context;
import android.support.v4.view.PagerAdapter;
import android.support.v4.view.ViewPager;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
public class MainActivity extends AppCompatActivity {
private LoopViewPager looviewpager;
Context context;
int[] resIds = new int[]{R.mipmap.img1, R.mipmap.img2, R.mipmap.img3, R.mipmap.img4, R.mipmap.img5, R.mipmap.img6};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
context = this;
looviewpager = (LoopViewPager) findViewById(R.id.looviewpager);
looviewpager.setAdapter(new MyAdapter());
looviewpager.setOffscreenPageLimit(3);
looviewpager.setPageTransformer(true, new ViewPager.PageTransformer() {
float scale = 0.9f;
@Override
public void transformPage(View page, float position) {
if (position >= 0 && position <= 1) {
page.setScaleY(scale + (1 - scale) * (1 - position));
} else if (position > -1 && position < 0) {
page.setScaleY(1 + (1 - scale) * position);
} else {
page.setScaleY(scale);
}
}
});
looviewpager.autoLoop(true);
}
class MyAdapter extends PagerAdapter {
@Override
public int getCount() {
return resIds.length;
}
@Override
public boolean isViewFromObject(View view, Object object) {
return view == object;
}
@Override
public Object instantiateItem(ViewGroup container, int position) {
View view = View.inflate(context, R.layout.item_viewpager, null);
RoundImageView itemImage = (RoundImageView) view.findViewById(R.id.item_image);
itemImage.setImageResource(resIds[position]);
container.addView(view);
return view;
}
@Override
public void destroyItem(ViewGroup container, int position, Object object) {
container.removeView((View) object);
}
}
}
ViewPager的setPageTransformer是用来控制view的切换动画的
looviewpager.setPageTransformer(true, new ViewPager.PageTransformer() {
float scale = 0.9f;
@Override
public void transformPage(View page, float position) {
if (position >= 0 && position <= 1) {
page.setScaleY(scale + (1 - scale) * (1 - position));
} else if (position > -1 && position < 0) {
page.setScaleY(1 + (1 - scale) * position);
} else {
page.setScaleY(scale);
}
}
});
Android 三种方式实现圆角ImageView
http://blog.csdn.net/zz7zz7zz/article/details/50602748
仿魅族BannerView,图片轮播控件,支持多种模式切换…
https://github.com/pinguo-zhouwei/MZBannerView
viewpager 左右无限循环
https://github.com/imbryk/LoopingViewPager