利用ViewPager和CirclePageIndicator实现新闻循环滚动

Viewpager实现了左右循环滚动,但是无法实现类似网易新闻的循环滚动效果。所以,就自己简单写了一个。




原理:

1.在ViewPager的第一页,插入最后一页的内容。在最后一页,插入原本第一页的内容。

如:原本内容页为A、B、C,则添加后页签内容为 C、A、B、C、A。

2.当从当前的第1页,右滑进入第0页后,迅速将页面切为倒数第二页。

   当从当前的倒数第二页,左滑进入最后一页后,迅速将页面切为倒数第1页。

如:从第一个A,切到C,则页面迅速切换到最后一个C。


具体实现:

1.先看ViewPagerAdapter的3个方法:

  getCount() :指示页签的个数

    此处我们在原来的基础上+2


  destroyItem(View container, int position, Object object):当页签变化时,销毁指定位置的界面

   这里我们选择只销毁中间的界面


  instantiateItem(View container, int position): 当页签变化时,加载指定位置的界面

  此处做逻辑上的处理,当position为0时,加载最后一页。posiiton为最大时,加载第一页


public class PhotoNewsViewPagerAdapter extends PagerAdapter {
	private ArrayList<View> views = new ArrayList<View>();
	private List<Article> mData = new ArrayList<Article>();
	private Context mContext;
	
	public PhotoNewsViewPagerAdapter() {
		// TODO Auto-generated constructor stub
	}
	
	public void setData(Context context,List<Article> data) {
		mContext = context;
		
		mData.clear();
		mData.addAll(data);
		
		LayoutInflater inflater = LayoutInflater.from(context);
		views.clear();
		for(Article article:mData){
			View view = inflater.inflate(R.layout.activity_news_viewpager_item, null);
			views.add(view);
			
			ImageView image = (ImageView) view.findViewById(R.id.iv_photo);
			image.setOnClickListener(new OnImageClickListener(article));
			
			showPhoto(image, article);
			
			TextView title = (TextView) view.findViewById(R.id.tv_title);
			title.setText(article.getTitle());
		}
	}
	
	public List<Article> getData() {
		return mData;
	}

	@Override
	public int getCount() {
		return views.size()+2;
	}

	@Override
	public boolean isViewFromObject(View arg0, Object arg1) {
		return arg0 == arg1;
	}

	@Override
	public int getItemPosition(Object object) {
		return super.getItemPosition(object);
	}

	@Override
	public void destroyItem(View container, int position, Object object) {
		if(position>1&&position<views.size()){
			((ViewPager) container).removeView(views.get(getPosition(position)));
		}
	}

	@Override
	public Object instantiateItem(View container, int position) {
		try {
			((ViewPager) container).addView(views.get(getPosition(position)));
		} catch (Exception e) {
			// TODO: handle exception
		}
		return views.get(getPosition(position));
	}
	
	private int getPosition(int position){
		int count = views.size();
		if(position==0){
			return count-1;
		}else if(position==count+1){
			return 0;
		}else{
			return position-1;
		}
	}
	
	
	
	/**
	 * 显示图片
	 * @param view 显示的view
	 * @param newsBean 新闻对象
	 */
	private void showPhoto(ImageView view, Article newsBean) {
		String photoPath = SystemUtil.getPhotoUrl(mContext, newsBean.getPhotoPath());
		// 如果有图片,则显示
			//配置图片加载及显示选项(还有一些其他的配置,查阅doc文档吧)
			DisplayImageOptions options = new DisplayImageOptions.Builder()
						.showStubImage(R.drawable.ic_empty)    //在ImageView加载过程中显示图片
	    				.showImageForEmptyUri(R.drawable.ic_empty)  //image连接地址为空时
						.showImageOnFail(R.drawable.ic_empty)  //image加载失败
						.cacheInMemory(true)  //加载图片时会在内存中加载缓存
						.cacheOnDisc(true)   //加载图片时会在磁盘中加载缓存
						.displayer(new RoundedBitmapDisplayer(1))  //设置用户加载图片task(这里是圆角图片显示)
						.build();
			
			// 图片加载器
			ImageLoader imageLoader = ImageLoader.getInstance();
			
			// 图片加载监听事件
			ImageLoadingListener animateFirstListener = new SimpleImageLoadingListener();

			//先在加载前显示默认图片
			imageLoader.displayImage("", view, options, animateFirstListener);
			
			//加载缩略图
			imageLoader.displayImage(photoPath, view, options, animateFirstListener);
	}
	
	private class OnImageClickListener implements OnClickListener{
		
		private Article article;
		
		public OnImageClickListener(Article article) {
			this.article = article;
		}
		
		@Override
		public void onClick(View v) {

		}
	};
}



2.给ViewPager加页面切换监听

		indicator.setOnPageChangeListener(new OnPageChangeListener() {
			//当前的索引
			private int currentIndex = 1;
			
			@Override
			public void onPageSelected(int arg0) {
				currentIndex = arg0;
			}
			
			@Override
			public void onPageScrolled(int arg0, float arg1, int arg2) {
			}
			
			@Override
			public void onPageScrollStateChanged(int arg0) {
				//滑动停止后,再切换,效果会好一点
				if(arg0 == 0){
					if(currentIndex==0){
						viewPager.setCurrentItem(photoAdapter.getCount()-2, false);
					}else if(currentIndex==photoAdapter.getCount()-1){
						viewPager.setCurrentItem(1, false);
					}
				}
			}
		});
至此,ViewPager循环切换就做好了,接下来就是底部页签指示标志的修改。


3.由于我们多加了两个页面,导致CirclePageIndicator显示的时候会多两个点,所以要修改其源码。


增加mIsLoopViewPager 属性,兼容循环ViewPager。

修改下面的代码,把第一个和最后一个点,屏蔽掉。

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);

        ...

        //Draw stroked circles
        int startIndex = 0;
        int endIndex = count;
        if(mIsLoopViewPager){
        	startIndex = 1;
        	endIndex = count-1;
        }
        for (int iLoop = startIndex; iLoop < endIndex; iLoop++) {
            float drawLong = longOffset + (iLoop * threeRadius);
            if (mOrientation == HORIZONTAL) {
                dX = drawLong;
                dY = shortOffset;
            } else {
                dX = shortOffset;
                dY = drawLong;
            }
            // Only paint fill if not completely transparent
            if (mPaintPageFill.getAlpha() > 0) {
                canvas.drawCircle(dX, dY, pageFillRadius, mPaintPageFill);
            }

            // Only paint stroke if a stroke width was non-zero
            if (pageFillRadius != mRadius) {
                canvas.drawCircle(dX, dY, mRadius, mPaintStroke);
            }
        }

        ...
    }

    对CirclePageIndicator不清楚的,参考 http://my.oschina.net/u/1403288/blog/208402


    收工。



一个轻量级的viewpager指示器 ,类似于nexus5 启动器的效果。它可以自定义指示器上小圆点的样式和动画效果。可以用于引导页。项目地址:https://github.com/ongakuer/CircleIndicator 效果图:如何使用1. xml布局中创建CircleIndicator是配合ViewPager使用的,一般如下布局:<RelativeLayout         android:layout_width="match_parent"         android:layout_height="match_parent">         <android.support.v4.view.ViewPager             android:id="@ id/viewpager_default"             android:layout_width="match_parent"             android:layout_height="match_parent"/>         <me.relex.circleindicator.CircleIndicator             android:id="@ id/indicator_default"             android:layout_centerInParent="true"             android:layout_width="fill_parent"             android:layout_height="40dp"/>     </RelativeLayout>2. java代码中// DEFAULT ViewPager defaultViewpager = (ViewPager) findViewById(R.id.viewpager_default); CircleIndicator defaultIndicator = (CircleIndicator) findViewById(R.id.indicator_default); DemoPagerAdapter defaultPagerAdapter = new DemoPagerAdapter(getSupportFragmentManager()); defaultViewpager.setAdapter(defaultPagerAdapter);//为ViewPager设置适配器 defaultIndicator.setViewPager(defaultViewpager);//将ViewPager添加到CircleIndicator中,以便监听ViewPager的滚动等事件DemoPagerAdapter是ViewPager的适配器,这个需要你去实现。属性说明属性名称类型说明ci_widthdimension小圆点的宽度ci_heightdimension小圆点的高度ci_margindimension小圆点间的间距ci_animatorreferenceViewPager页面切换时,给小圆点设置动画。设置当前的小圆点恢复到未选中时的状态的动画。例如:当前正从a切换到b,那么该属性设置的是a的动画。ci_animator_reversereference和"ci_animator"属性一样,只不过该属性设置的是b的动画。ci_drawablereference设置当前或选中的小圆点的样式,如颜色、圆角等。ci_drawable_unselectedreference和"ci_drawable"一样,只是该属性设置的是未选中的小圆点的样式。点击上面的"下载源码"下载完整的demo工程。demo简要说明xml布局:<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"     xmlns:app="http://schemas.android.com/apk/res-auto"     android:layout_width="match_parent"     android:layout_height="match_parent"     android:orientation="vertical">     <RelativeLayout         android:layout_width="match_parent"         android:layout_height="0dp"         android:layout_weight="1">         <android.support.v4.view.ViewPager             android:id="@ id/viewpager_default"             android:layout_width="match_parent"             android:layout_height="match_parent"/>         <me.relex.circleindicator.CircleIndicator             android:id="@ id/indicator_default"             android:layout_centerInParent="true"             android:layout_width="fill_parent"             android:layout_height="40dp"/>     </RelativeLayout>     <RelativeLayout         android:layout_width="match_parent"         android:layout_height="0dp"         android:layout_weight="1">         <android.support.v4.view.ViewPager             android:id="@ id/viewpager_custom"             android:layout_width="match_parent"             android:layout_height="match_parent"/>         <me.relex.circleindicator.CircleIndicator             android:id="@ id/indicator_custom"             app:ci_width="10dp"             app:ci_height="4dp"             app:ci_margin="6dp"             app:ci_animator="@animator/indicator_animator"             app:ci_animator_reverse="@animator/indicator_animator_reverse"             app:ci_drawable="@drawable/black_radius_square"             android:layout_centerInParent="true"             android:layout_width="fill_parent"             android:layout_height="40dp"/>     </RelativeLayout>     <RelativeLayout         android:layout_width="match_parent"         android:layout_height="0dp"         android:layout_weight="1">         <android.support.v4.view.ViewPager             android:id="@ id/viewpager_unselected_background"             android:layout_width="match_parent"             android:layout_height="match_parent"/>         <me.relex.circleindicator.CircleIndicator             android:id="@ id/indicator_unselected_background"             android:layout_centerInParent="true"             android:layout_width="fill_parent"             app:ci_width="6dp"             app:ci_height="6dp"             app:ci_animator="@animator/indicator_no_animator"             app:ci_drawable="@drawable/white_radius"             app:ci_drawable_unselected="@drawable/black_radius"             android:layout_height="40dp"             />     </RelativeLayout> </LinearLayout>定义了3组ViewPager和指示器。第一组是采用默认样式的,被选中的小圆点带有放大的动画。第2组比较全,既定义了动画又修改了样式。第3组是取消动画的同时使用了ci_drawable_unselected属性。适配器Adapter:public class DemoPagerAdapter extends FragmentPagerAdapter {         private int pagerCount = 5;         private Random random = new Random();         public DemoPagerAdapter(FragmentManager fm) {             super(fm);         }         @Override public Fragment getItem(int i) {             return ColorFragment.newInstance(0xff000000 | random.nextInt(0x00ffffff));         }         @Override public int getCount() {             return pagerCount;         }     }定义了5个页面或Fragment,通过ColorFragment生成Fragment。Fragment:public class ColorFragment extends Fragment {     private static final String ARG_COLOR = "color";     private int mColor;     public static ColorFragment newInstance(int param1) {         ColorFragment fragment = new ColorFragment();         Bundle args = new Bundle();         args.putInt(ARG_COLOR, param1);         fragment.setArguments(args);         return fragment;     }     public ColorFragment() {     }     @Override     public void onCreate(Bundle savedInstanceState) {         super.onCreate(savedInstanceState);         if (getArguments() != null) {             mColor = getArguments().getInt(ARG_COLOR);         }     }     @Override     public View onCreateView(LayoutInflater inflater, ViewGroup container,             Bundle savedInstanceState) {         View v = inflater.inflate(R.layout.color_fragment, container, false);         v.setBackgroundColor(mColor);         return v;     } }生成空的页面,没有任何控件,只是设置了页面的背景色。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值