Android 使用viewpager实现无限循环(定时+手动)
作者:mrr 发布时间:2023-12-09 07:43:56
标签:android,viewpager,无限循环
循环轮播的方法有两种,一种是使用定时器另外一种是使用手指拨动,相比较而言,定时器实现循环播放比较容易,只要在定时器的消息里加上简单几段代码即可,下面分别就这两种方法给大家详解,具体详情请看下文吧。
int count = adapter.getCount();
if (count > 1) { // 多于1个,才循环
int index = viewPager.getCurrentItem();
index = (index + 1) % count;
viewPager.setCurrentItem(index, true);
}
但如何在手指拨动时,也支持左右无限循环的功能呢?这个问题,我折腾了好久,网上资料多不可用。
比较可笑的是在适配器里,设置数量为最大值;然后,在当前位置为中间的某个值,比如:
public class MyAdapter extends PagerAdapter{
@Override
public int getCount() {
return Integer.MAX_VALUE;
}
@Override
public boolean isViewFromObject(View arg0, Object arg1) {
return arg0 == arg1;
}
@Override
public void destroyItem(View container, int position, Object object) {
((ViewPager)container).removeView(mImageViews[position % count]);
}
public Object instantiateItem(View container, int position) {
View view = mImageViews[position % count];
((ViewPager)container).addView(view, 0);
return view;
}
}
protected void onCreate(Bundle savedInstanceState) {
viewPager.setAdapter(new MyAdapter());
//设置ViewPager的默认项, 设置为总数的100倍,一开始才能往左滑动
viewPager.setCurrentItem(count * 100);
}
这种做法,实际上,并没有真正达到左右无限循环;只是一般情况下,很难达到边界(不可滑动)的情况。
通过查阅大量的资料,我发现ViewPager自身并不支持左右无限循环的功能,不得不说,这点真是太失败了。
由于ViewPager自身不支持,我们只能自己找方法了!下面是我实现左右无限循环的方法:
用于显示的mViews,比数据源mList,多了两个节点元素(头节点0:b和尾节点5:e用于跳转)
下图的不带箭头的红线,是mViews根据mList初始化的情况;带箭头的红线是跳转的情况。
具体代码如下:
public class AdapterCycle extends PagerAdapter
implements ViewPager.OnPageChangeListener{
private Context mContext; // 上下文
private LayoutInflater mInflater; // 用于解XML
private LinkedList<View> mViews; // <span style="font-family: Arial, Helvetica, sans-serif;">用于</span><span style="font-family: Arial, Helvetica, sans-serif;">显示的View</span>
private List<Drawable> mList; // 数据源<span style="font-family: Arial, Helvetica, sans-serif;">Drawable</span>
private ViewPager mViewPager; //页面
public AdapterCycle(Context context, ViewPager viewPager,
List<Drawable> list)
{
mContext = context;
mInflater = LayoutInflater.from(context);
mViewPager = viewPager;
if (list != null) {
//无论是否多于1个,都要初始化第一个(index:0)
mViews = new LinkedList<View>();
ImageView view = (ImageView) mInflater.inflate(
R.layout.activity_main_item_cycle, null);
Drawable drawable = list.get(list.size() - 1);
view.setImageDrawable(drawable);
mViews.add(view);
//注意,如果不只1个,mViews比mList多两个(头尾各多一个)
//假设:mList为mList[0~N-1], mViews为mViews[0~N+1]
// mViews[0]放mList[N-1], mViews[i]放mList[i-1], mViews[N+1]放mList[0]
// mViews[1~N]用于循环;首位之前的mViews[0]和末尾之后的mViews[N+1]用于跳转
// 首位之前的mViews[0],跳转到末尾(N);末位之后的mViews[N+1],跳转到首位(1)
if( list.size() > 1) { //多于1个要循环
for (Drawable d : list) { //中间的N个(index:1~N)
ImageView v = (ImageView) mInflater.inflate(
R.layout.activity_main_item_cycle, null);
v.setImageDrawable(d);
mViews.add(v);
}
//最后一个(index:N+1)
view = (ImageView) mInflater.inflate(
R.layout.activity_main_item_cycle, null);
drawable = dList.get(0);
view.setImageDrawable(drawable);
mViews.add(view);
}
}
}
@Override
public int getCount() {
return mViews.size();
}
@Override
public boolean isViewFromObject(View view, Object object) {
return view == object;
}
@Override
public void destroyItem(ViewGroup container, int position, Object object) {
((ViewPager) container).removeView(mViews.get(position));
}
@Override
public Object instantiateItem(ViewGroup container, int position) {
View view = mViews.get(position);
container.addView(view);
return view;
}
// 实现ViewPager.OnPageChangeListener接口
@Override
public void onPageSelected(int position) {
LogUtils.logi(TAG, "onPageSelected:" + position);
if ( mViews.size() > 1) { //多于1,才会循环跳转
if ( position < 1) { //首位之前,跳转到末尾(N)
position = mList.size(); //注意这里是mList,而不是mViews
mViewPager.setCurrentItem(position, false);
} else if ( position > mList.size()) { //末位之后,跳转到首位(1)
mViewPager.setCurrentItem(1, false); //false:不显示跳转过程的动画
position = 1;
}
}
}
@Override
public void onPageScrolled(int position, float positionOffset,
int positionOffsetPixels) {
// 什么都不干
}
@Override
public void onPageScrollStateChanged(int state) {
// 什么都不干
}
}
//在外层,将mViewPager初始位置设置为1即可
if (mAdapterCycle.getCount() > 1) { //多于1个,才循环并开启定时器
mViewPager.setCurrentItem(1); //将mViewPager初始位置设置为1
startTimer(); //开启定时器,定时切换页面
}
这时,利用定时器,实现循环轮播,需要修改成如下:
int count = adapter.getCount();
if (count > 2) { // 实际上,多于1个,就多于3个
int index = viewPager.getCurrentItem();
index = index % (count - 2) + 1; //这里修改过
viewPager.setCurrentItem(index, true);
}


猜你喜欢
- 公司有一个需求,实现一个多级的树形菜单,并且支持多选功能,实现这个功能之前,我在网上找了找,树形菜单很好找,但是支持多选功能并没有很合适的,
- 本文实例讲述了C#实现简单屏幕监控的方法。分享给大家供大家参考。具体如下:这是一段C#编写的屏幕监控代码,可以自动对屏幕进行截图,软件自身隐
- 1|0构造socket在【客户端/服务端】的通信模式中,客户端需要主动构造与服务器连接的 Socket,构造方法有以下几种重载形式:Sock
- Java的位操作符用来操作整数基本数据类型中的单个“比特”(bit),即代进制位。而我们知道比特就是0和1,那么,位操作就是对这些数据进行基
- 项目中需要实现一个状态显示的悬浮框,要求可以设置两种模式:拖动模式和不可拖动模式。实现效果图如下:实现步骤:1.首先要设置该悬浮框的基本属性
- 使用限定符在平板上面大多数时候采用的双页的模式,程序会在左侧列表上显示一个包含子项列表,右侧的面版会显示详细的内容的因为平板具有足够大的屏幕
- 近来,很多公司的APP都实现了人脸识别登录的功能。今天呢,银鹏带大家从头到尾做一下这个人脸识别登录。首先呢,我们需要采用一个拥有人脸识别算法
- Java8对集合提供了一种流式计算的方式,这种风格将要处理的元素集合看 作一种流, 流在管道中传输, 并且可以在管道的节点上进行处理, 比如
- 在做在线编程题目的时候,需要了解一下数据的输入格式。这样可以对数据处理有比较好的把握,不需要把太多的时间放在这个上面,注重主要的算法逻辑即可
- 前言 找工作的时候,曾经用C语言练习过一段时间的算法题目,也在几个还算出名的OJ平台有过还算靠谱的排名。
- 项目描述: springboot+springcloud+zookeeper+eureka+maven;为多模块多module的分布式架构;
- 概述从今天开始, 小白我将带大家开启 Jave 数据结构 & 算法的新篇章.链表链表 (Linked List) 是一种递归的动态数
- Android WebView常见问题解决方案汇总:就目前而言,如何应对版本的频繁更新呢,又如何灵活多变地展示我们的界面呢,这又涉及到了we
- /// <summary> /// 汉字转拼音缩写 /// </summary> /// <param nam
- 将索引设置为只读可能听起来很奇怪,但在 Elasticsearch 中执行此类操作是可能的。想象一下这样一种情况,你特别需要限制对索引的写入
- 给组件加上 * 定义一个类,这个类继承ActionListenerpubulic class ButListener implements
- 项目场景: 新搭了一个springboot 2.3.7.RELASE的框架,在集成mysql,tkMapper,mybatis的过
- 目录通过切面,实现超灵活的注解式数据校验Spring MVC的校验方式通过切面实现自己的注解式数据校验Spring boot aop注解数据
- 装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构。这种类型的设计模式属于结构型模式,它是
- 最近由于工作要求:前端采用vue开发,后端采用springboot开发,前后端分离开发,最后前端页面又整合到后端来。经历多次采坑,总结以下方