现在的情况

不改变的源代码,什么时候ViewPager滑动到最后item的时候,他就无法再往右滑动;当ViewPager滑动到第一个item的时候,他也无法再往前滑动。

(以上全是废话)

设想

我们能够这样想。当滑动到最后一个的时候,我们让他跳转到第一个,这样他就能够继续往后滑动了,这样就达到了我们想要的循环滑动。

尽管功能上是循环了,可是实际显示的时候会在最后一个和第一个之间自己主动跳转。

优化

我们能够在原来的链表中首尾各添加一个假的item。用多余的两个item来作跳转的动作,这样就能够避免出现自己主动跳转的错误画面了。

我们要显示的是以下A、B、C画面。位置各自是0、1、2.

实际上,我们加入数据的时候,多加入了2个。

在位置0加入了最后一个界面C,在位置4加入了第一个界面A。

当界面滑动到位置3的时候,他还能够往右滑动,这样给人的感觉就是循环的。但,当滑动到位置4的时候。他右边没有了,这样岂不是露馅了?所以,当滑动到位置4的时候。立马跳转到位置1。

由于他们是相同的数据,所以从显示效果是看不出跳转了的。这样实际上我们就变成了位置1,这样就又能够继续往右滑动了。

反复上面条件的推断。这样就实现了往右的循环。往左也是相同的道理。

代码分析

在onPageSelected里面做条件推断,在onPageScrollStateChanged里面做跳转。

关键代码例如以下:

初始化。首尾各添加一个item。

// 添加第1个界面,实际上他显示的是最后一个界面
addTextView(POINT_LENGTH - 1);
// 添加实际显示的2、3、4界面
for (int i = 0; i < 3; i++) {
addTextView(i);
addPoint(i);
}
// 添加最后的第5个界面,实际上他显示的是第一个界面
addTextView(0);

条件推断:

	@Override
public void onPageSelected(int pPosition) {
mIsChanged = true;
if (pPosition > POINT_LENGTH) {
mCurrentPagePosition = FIRST_ITEM_INDEX;
} else if (pPosition < FIRST_ITEM_INDEX) {
mCurrentPagePosition = POINT_LENGTH;
} else {
mCurrentPagePosition = pPosition;
}
Log.i(TAG,"当前的位置是"+mCurrentPagePosition);
setCurrentDot(mCurrentPagePosition);
}

跳转:

	@Override
public void onPageScrollStateChanged(int pState) {
if (ViewPager.SCROLL_STATE_IDLE == pState) {
if (mIsChanged) {
mIsChanged = false;
mViewPager.setCurrentItem(mCurrentPagePosition, false);
}
}
}

完整的逻辑例如以下:

package com.ahacool.circleviewpager;

import java.util.ArrayList;

import android.app.Activity;
import android.os.Bundle;
import android.support.v4.view.PagerAdapter;
import android.support.v4.view.ViewPager;
import android.support.v4.view.ViewPager.OnPageChangeListener;
import android.util.Log;
import android.view.Gravity;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.LinearLayout.LayoutParams;
import android.widget.TextView; /**
* @ClassName MainActivity
* @Description 循环滑动viewpager的一种方法,滑动非常流畅。实现方法:在实际显示的界面头和尾分别添加一个界面。 * @author Moto
* @date 2014 2014-7-18
*
*/
public class MainActivity extends Activity implements OnPageChangeListener { private ViewPager mViewPager;
private ViewGroup mPointViewGroup;
private ArrayList<View> mViewPagerList;
private boolean mIsChanged = false;
private int mCurrentPagePosition = FIRST_ITEM_INDEX;
private int mCurrentIndex;
private static final int POINT_LENGTH = 3;
private static final int FIRST_ITEM_INDEX = 1;
private static final String TAG = "MOTO"; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initUI();
} private void initUI() {
mViewPager = (ViewPager) findViewById(R.id.viewpager);
mPointViewGroup = (ViewGroup) findViewById(R.id.point_layout); mViewPagerList = new ArrayList<View>();
// 添加第1个界面,实际上他显示的是最后一个界面
addTextView(POINT_LENGTH - 1);
// 添加实际显示的2、3、4界面
for (int i = 0; i < 3; i++) {
addTextView(i);
addPoint(i);
}
// 添加最后的第5个界面,实际上他显示的是第一个界面
addTextView(0); PagerAdapter pagerAdapter = new CustomPagerAdapter(mViewPagerList);
mViewPager.setAdapter(pagerAdapter);
mViewPager.setOnPageChangeListener(this);
mViewPager.setCurrentItem(mCurrentPagePosition, false);
} private void addTextView(int pIndex) {
TextView textview = new TextView(this);
textview.setLayoutParams(new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT));
textview.setGravity(Gravity.CENTER);
textview.setText("这是第" + (pIndex + 1) + "个页面");
textview.setTextSize(50);
mViewPagerList.add(textview);
} private void addPoint(int pIndex) {
ImageView pointImageView = new ImageView(this);
LayoutParams layoutParams = new LayoutParams(20, 20);
layoutParams.setMargins(10, 0, 10, 0);
pointImageView.setLayoutParams(layoutParams);
pointImageView.setBackgroundResource(R.drawable.point_style);
if (0 == pIndex) {
pointImageView.setEnabled(false);
}
mPointViewGroup.addView(pointImageView);
} private void setCurrentDot(int positon) {
// 界面实际显示的序号是第1, 2, 3。而点的序号应该是0, 1, 2.所以减1.
positon = positon - 1;
if (positon < 0 || positon > mViewPagerList.size() - 1 || mCurrentIndex == positon) {
return;
}
mPointViewGroup.getChildAt(positon).setEnabled(false);
mPointViewGroup.getChildAt(mCurrentIndex).setEnabled(true);
mCurrentIndex = positon;
} @Override
public void onPageScrollStateChanged(int pState) {
if (ViewPager.SCROLL_STATE_IDLE == pState) {
if (mIsChanged) {
mIsChanged = false;
mViewPager.setCurrentItem(mCurrentPagePosition, false);
}
}
} @Override
public void onPageScrolled(int arg0, float arg1, int arg2) { } @Override
public void onPageSelected(int pPosition) {
mIsChanged = true;
if (pPosition > POINT_LENGTH) {
mCurrentPagePosition = FIRST_ITEM_INDEX;
} else if (pPosition < FIRST_ITEM_INDEX) {
mCurrentPagePosition = POINT_LENGTH;
} else {
mCurrentPagePosition = pPosition;
}
Log.i(TAG,"当前的位置是"+mCurrentPagePosition);
setCurrentDot(mCurrentPagePosition);
} }

源代码下在地址:https://github.com/bird7310/Demos.git

总结

希望对大家有帮助。多提意见。

近段时间项目非常赶,非常长时间没看书写博客了。

赶项目赶得都麻,放松。偷偷懒。写博客是。

版权声明:本文博主原创文章,博客,未经同意不得转载。

无尽的循环ViewPager的更多相关文章

  1. Android循环ViewPager(二)

    上午没事写了一篇,下午有事,晚上回来看看感觉写的差点意思,上篇文章大概的关于循环是自己添加了两个空的View,看到网上还有一种就是在自定义的Adapter中getCount中返回最大值,然后destr ...

  2. 循环viewpager

    如果viewpager listadapter小于三个.用这个移除异常. for (View view : viewList) {             ViewGroup p = (ViewGro ...

  3. ViewPager 无限循环

    Overview 我们在使用ViewPager来制作图片轮播的时候,常常为ViewPager不能一直无限循环的问题所苦恼.对于这个问题,目前从网上找到了两个思路来解决: 将 ViewPager 的Co ...

  4. 【NodeJS线程】Boss和他的职员们

    >>>[说明]还是一如既往的,这篇文章是从我的个人博客里挪过来的.原文参见:http://www.jscon.co/coding/frontend/nodejs_fork_child ...

  5. node 学习笔记 - Modules 模块加载系统 (1)

    本文同步自我的个人博客:http://www.52cik.com/2015/12/11/learn-node-modules-path.html 用了这么久的 require,但却没有系统的学习过 n ...

  6. [转载] Python的GIL是什么鬼,多线程性能究竟如何

    原文: http://cenalulu.github.io/python/gil-in-python/ GIL是什么 首先需要明确的一点是GIL并不是Python的特性,它是在实现Python解析器( ...

  7. lintcode:快乐数

    快乐数 写一个算法来判断一个数是不是"快乐数". 一个数是不是快乐是这么定义的:对于一个正整数,每一次将该数替换为他每个位置上的数字的平方和,然后重复这个过程直到这个数变为1,或是 ...

  8. Project Euler 92:Square digit chains 平方数字链

    题目 Square digit chains A number chain is created by continuously adding the square of the digits in ...

  9. Happy Number——LeetCode

    Write an algorithm to determine if a number is "happy". A happy number is a number defined ...

随机推荐

  1. 从尾到头打印链表--《剑指offer》

    题目:非常easy,就是题目,将链表从尾到头打印出来. 可能我们首先想到的是将链表进行遍历,将之前的訪问的数据进行保存,最后进行反向输出,但是保存数据的空间是个问题:或者是我们将整个链表进行反向操作, ...

  2. ESXI主机打开shell后主机警告处理

    昨天为了配置snmp监控,将几台ESXI 5.5主机的shell 在控制台上从disable状态修改为enable状态后,登陆vcenter后,发现所有的主机都有警告. 处理过程如下: 选中有警告标志 ...

  3. Jetty:开发指导Handlers

    Rewrite Handler RewriteHandler匹配一个基于该请求的规则集合,然后根据匹配规则的变更请求. 最常见的要求是改写URI.但不限于:规则可以被配置为重定向响应.设置cookie ...

  4. Learning Cocos2d-x for WP8(1)——创建首个项目

    原文:Learning Cocos2d-x for WP8(1)--创建首个项目 Cocos2d-x for WP8开发语言是C++,系列文章将参考兄弟篇Learning Cocos2d-x for ...

  5. WPF界面设计技巧(3)—实现不规则动画按钮

    原文:WPF界面设计技巧(3)-实现不规则动画按钮 发布了定义WPF按钮的教程后,有朋友问能否实现不规则形状的按钮,今天我们就来讲一下不规则按钮的制作. 不规则按钮的做法实际上和先前我们做不规则窗体的 ...

  6. C# 通信学习笔记

    C# 通信学习笔记 DNS 是域名系统 (Domain Name System) 的缩写,是因特网的一项核心服务,它作为可以将域名和IP地址相互映射的一个分布式数据库,能够使人更方便的访问互联网,而不 ...

  7. 《UNIX编程环境》的源代码的第二个版本Ubuntu下编

    1.在http://www.apuebook.com下载源代码 2. 视图READ root@ubuntu:/home/wl/mywork/apue.2e# cat -n README 1 Read ...

  8. Bitmap

    Bitmap篇   在前一篇中介绍了使用API做Distinct Count,但是计算精确结果的API都较慢,那有没有能更快的优化解决方案呢? 1. Bitmap介绍 <编程珠玑>上是这样 ...

  9. html中滚动栏的样式

    DIV滚动栏设置 (CSS)2008/09/26 03:07div 中滚动栏的控制2008年01月06日 星期日 01:181)隐藏滚动栏<body style="overflow-x ...

  10. SignalR技术

    Asp.net SignalR快速入门 一.前言 之前半年时间感觉自己有点浮躁,导致停顿了半年多的时间没有更新博客,今天重新开始记录博文,希望自己可以找回初心,继续沉淀.由于最近做的项目中用到Sign ...