Android-ViewPager

本贴最后更新于 3228 天前,其中的信息可能已经时移世异

1. ViewPager 使用示例

2. 源码解析



示例

ViewPager 可以说是 IM 中必用的一个控件了,举例比如 WeChat,Weibo,其用法也非常之简单。

    <android.support.v4.view.ViewPager
        android:id="@+id/viewpager"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent">
    </android.support.v4.view.ViewPager>

xml文件中定义完之后,再看java文件中代码


public class MainActivity extends AppCompatActivity {
private ViewPager viewpager=null;


@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    viewpager=(ViewPager)findViewById(R.id.viewpager);
    MyAdapter adapter=new MyAdapter();
    LayoutInflater lf=getLayoutInflater();
    adapter.addviews(lf.inflate(R.layout.page1,null));
    adapter.addviews(lf.inflate(R.layout.page2,null));
    adapter.addviews(lf.inflate(R.layout.page3,null));
    viewpager.setAdapter(adapter);
}

}




可以看到 ViewPager 的用法和 ListView 的用法是类似的,都需要一个 Adapter 来加载数据,下面看这个自定义的 MyAdapter。


public class MyAdapter extends PagerAdapter {
private ArrayList&lt;View&gt; views;

public void addviews(View v){
    views.add(v);
}
@Override
public int getCount() {
    return views.size();
}


@Override
public boolean isViewFromObject(View view, Object object) {
    return view==object;
}


@Override
public void destroyItem(ViewGroup container, int position, Object object)   {
    container.removeView(views.get(position));
}


@Override
public Object instantiateItem(ViewGroup container,int position){
    container.addView(views.get(position));
    return views.get(position);
}

}


这个MyAdapter继承PagerAdapter,可以看到,需要Override四个方法,方法作用可以从函数名得知。
很简单吧?跟ListView的用法真是太相似了,ViewPager的内部还有一个接口
public interface OnPageChangeListener {
   public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels);
   public void onPageSelected(int position);
   public void onPageScrollStateChanged(int state);
}

接口的作用也很明显,最常用的就是用来加载各个tab的内容。


网上的用法有很多,这里就不细讲了,重点是下一部分。


源码
开始看到ViewPager的滑动效果时,马上联想到了ListView的下拉刷新,想到ViewPager也是采用这种原理来实现的,后来一看源码,果然,其原理就是相当于将多个View平行的摆放在一起,滑动即时调整横向的offset,来实现滑屏效果。
初始化的代码跳过,直接看onTouchEvent函数的代码,
首先介绍一下ViewPager中比较重要的一些变量和类。
private final ArrayList<ItemInfo> mItems = new ArrayList<ItemInfo>();
static class ItemInfo {
        Object object;   //View或者Image 
        int position;    //在mItems中的position,一般等同于offset
        boolean scrolling;
        float widthFactor;
        float offset;
    }

mItems是一个ItemInfo的ArrayList,很重要,是前一个Page,当前Page,后一个Page的集合。
ACTION_DOWN
case MotionEvent.ACTION_DOWN: {
                mScroller.abortAnimation();
                mPopulatePending = false;
                populate();
            // Remember where the motion event started
            mLastMotionX = mInitialMotionX = ev.getX();
            mLastMotionY = mInitialMotionY = ev.getY();
            mActivePointerId = MotionEventCompat.getPointerId(ev, 0);
            break;
        }</pre>

重点是记录X,Y,PointerId,和populate()函数,前面的变量都好理解,这个populate函数还是比较复杂的,我们看看。
       
 final int pageLimit = mOffscreenPageLimit;//偏离屏幕page限制,一般左右各一个
        final int startPos = Math.max(0, mCurItem - pageLimit);左边的page
        final int N = mAdapter.getCount();
        final int endPos = Math.min(N-1, mCurItem + pageLimit);右边的page
        ............
        int curIndex = -1;
        ItemInfo curItem = null;
        for (curIndex = 0; curIndex < mItems.size(); curIndex++) {
            final ItemInfo ii = mItems.get(curIndex);
            if (ii.position >= mCurItem) {
                if (ii.position == mCurItem) curItem = ii;
                break;
            }
        }      //找到curItem
        .............
        if (curItem != null) {
            float extraWidthLeft = 0.f;
            int itemIndex = curIndex - 1;
            ItemInfo ii = itemIndex >= 0 ? mItems.get(itemIndex) : null;
            final int clientWidth = getClientWidth();
            final float leftWidthNeeded = clientWidth <= 0 ? 0 :
                    2.f - curItem.widthFactor + (float) getPaddingLeft() / (float) clientWidth;
            for (int pos = mCurItem - 1; pos >= 0; pos--) {
                if (extraWidthLeft >= leftWidthNeeded && pos < startPos) {
                    if (ii == null) {
                        break;
                    }
                    if (pos == ii.position && !ii.scrolling) {
                        mItems.remove(itemIndex);
                        mAdapter.destroyItem(this, pos, ii.object);
                        if (DEBUG) {
                            Log.i(TAG, "populate() - destroyItem() with pos: " + pos +
                                    " view: " + ((View) ii.object));
                        }
                        itemIndex--;
                        curIndex--;
                        ii = itemIndex >= 0 ? mItems.get(itemIndex) : null;
                    }
                } else if (ii != null && pos == ii.position) {
                    extraWidthLeft += ii.widthFactor;
                    itemIndex--;
                    ii = itemIndex >= 0 ? mItems.get(itemIndex) : null;
                } else {
                    ii = addNewItem(pos, itemIndex + 1);
                    extraWidthLeft += ii.widthFactor;
                    curIndex++;
                    ii = itemIndex >= 0 ? mItems.get(itemIndex) : null;
                }
            }
        float extraWidthRight = curItem.widthFactor;
        itemIndex = curIndex + 1;
        if (extraWidthRight &lt; 2.f) {
            ii = itemIndex &lt; mItems.size() ? mItems.get(itemIndex) : null;
            final float rightWidthNeeded = clientWidth &lt;= 0 ? 0 :
                    (float) getPaddingRight() / (float) clientWidth + 2.f;
            for (int pos = mCurItem + 1; pos &lt; N; pos++) {
                if (extraWidthRight &gt;= rightWidthNeeded &amp;&amp; pos &gt; endPos) {
                    if (ii == null) {
                        break;
                    }
                    if (pos == ii.position &amp;&amp; !ii.scrolling) {
                        mItems.remove(itemIndex);
                        mAdapter.destroyItem(this, pos, ii.object);
                        if (DEBUG) {
                            Log.i(TAG, "populate() - destroyItem() with pos: " + pos +
                                    " view: " + ((View) ii.object));
                        }
                        ii = itemIndex &lt; mItems.size() ? mItems.get(itemIndex) : null;
                    }
                } else if (ii != null &amp;&amp; pos == ii.position) {
                    extraWidthRight += ii.widthFactor;
                    itemIndex++;
                    ii = itemIndex &lt; mItems.size() ? mItems.get(itemIndex) : null;
                } else {
                    ii = addNewItem(pos, itemIndex);
                    itemIndex++;
                    extraWidthRight += ii.widthFactor;
                    ii = itemIndex &lt; mItems.size() ? mItems.get(itemIndex) : null;
                }
            }
        }</pre>


这是核心代码,代码有点长,主要作用就是产生一个mItems,使得mItems中含有左右以及当前的Page,产生的依据是extraWidth和WidthNeeded的关系,这里注意,mItems有多个地方会产生调用,比如onMeasure中,所以一次touch滑屏,会分别调用多次populate(),一般调用两次吧。
extraWidth和WidthNeeded分Left和Right,现只拿Right来做讲解。
首先float extraWidthRight=curItem.widthFactor;
widthFactor宽度因子,一般为1,代表一个page的宽度。
final float rightWidthNeeded=(float)getPaddingRight()/(float)clientWidth+2.f;
rightWidthNeeded=2,很明显,不用计算。
进入for循环,先假设总共有3个Page,现在从2滑到1
先看第2次调用吧,即屏幕已经从2变成了1,但是mItems里的值还没有发生改变,但是curItem是1了。
pos=mCurItem+1=1,此时pos=1,endPos=1,ii代表的是itemIndex所对应的mItem,现在是page2,进入else if语句,extraWidthRight+=ii.widthFactor,此时extraWidthRight=2,即curItem和右边Page的宽度总和。

itemIndex++;得到page3所对应的mItem,pos=2>endPos,直接进入if语句,我们可以看到


 mItems.remove(itemIndex);
 mAdapter.destroyItem(this, pos, ii.object);


成功删除了page3这个对象,mItems的大小变为了2。
以上就是ViewPager的核心--mItems,基本由populate掌控。
后面的就比较简单了,上面的代码看了比较久,主要是在观察各种width和mItems之间的关系,还有千万不要忽略了滑屏过后onMeasure方法会重新调用populate()。

ACTION_MOVE


           
 if (mIsBeingDragged) {
                final int activePointerIndex = MotionEventCompat.findPointerIndex(
                        ev, mActivePointerId);
                final float x = MotionEventCompat.getX(ev, activePointerIndex);
                needsInvalidate |= performDrag(x);
            }

核心函数是performDrag(x),作用顾名思义,就是产生drag的效果。


   
 private boolean performDrag(float x) {
        boolean needsInvalidate = false;
    final float deltaX = mLastMotionX - x;
    mLastMotionX = x;


    float oldScrollX = getScrollX();
    float scrollX = oldScrollX + deltaX;
    final int width = getClientWidth();


    float leftBound = width * mFirstOffset;
    float rightBound = width * mLastOffset;
    boolean leftAbsolute = true;
    boolean rightAbsolute = true;


    final ItemInfo firstItem = mItems.get(0);
    final ItemInfo lastItem = mItems.get(mItems.size() - 1);
    if (firstItem.position != 0) {
        leftAbsolute = false;
        leftBound = firstItem.offset * width;
    }
    if (lastItem.position != mAdapter.getCount() - 1) {
        rightAbsolute = false;
        rightBound = lastItem.offset * width;
    }


    if (scrollX &lt; leftBound) {
        if (leftAbsolute) {
            float over = leftBound - scrollX;
            needsInvalidate = mLeftEdge.onPull(Math.abs(over) / width);
        }
        scrollX = leftBound;
    } else if (scrollX &gt; rightBound) {
        if (rightAbsolute) {
            float over = scrollX - rightBound;
            needsInvalidate = mRightEdge.onPull(Math.abs(over) / width);
        }
        scrollX = rightBound;
    }
    // Don't lose the rounded component
    mLastMotionX += scrollX - (int) scrollX;
    scrollTo((int) scrollX, getScrollY());
    pageScrolled((int) scrollX);


    return needsInvalidate;
}

别看代码很乱,主要就是leftBound,RightBound,leftAbsolute,rightAbsolute的关系,这里还要注意First和Last,千万不要以为是第一个Page和最后一个Page,这个是相对于mItems来说的,所以前面的mItems概念理解错了,这里也会理解错。
leftAbsolute=true表示,左边没有Page了,RightAbsolute也一样。
从上到下的这些if分别是判断,firstItem和lastItem是否在边缘,socrllX的判断,则是在leftAbsolute或者RightAbsolute为true时起作用,即是超过边缘,无效的滑动。最后scrollTo实现drag效果。
ACTION_UP
其实弹起后的代码就很好想了,无非是根据fling这类的手势或者offset来判断是否要跳到下一个page,代码也很简单了,大家自己看看吧。


其实我们也可以看到,android系统自带的ViewPager中mItems的使用是非常节约内存的。为了锻炼自己对View和ViewGroup的理解,决定下星期自己写一个ScrollLayou试试。





  • Android

    Android 是一种以 Linux 为基础的开放源码操作系统,主要使用于便携设备。2005 年由 Google 收购注资,并拉拢多家制造商组成开放手机联盟开发改良,逐渐扩展到到平板电脑及其他领域上。

    334 引用 • 323 回帖 • 25 关注
  • ViewPager
    3 引用

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • 微信

    腾讯公司 2011 年 1 月 21 日推出的一款手机通讯软件。用户可以通过摇一摇、搜索号码、扫描二维码等添加好友和关注公众平台,同时可以将自己看到的精彩内容分享到微信朋友圈。

    130 引用 • 793 回帖
  • IPFS

    IPFS(InterPlanetary File System,星际文件系统)是永久的、去中心化保存和共享文件的方法,这是一种内容可寻址、版本化、点对点超媒体的分布式协议。请浏览 IPFS 入门笔记了解更多细节。

    20 引用 • 245 回帖 • 234 关注
  • abitmean

    有点意思就行了

    29 关注
  • WebClipper

    Web Clipper 是一款浏览器剪藏扩展,它可以帮助你把网页内容剪藏到本地。

    3 引用 • 9 回帖 • 2 关注
  • JVM

    JVM(Java Virtual Machine)Java 虚拟机是一个微型操作系统,有自己的硬件构架体系,还有相应的指令系统。能够识别 Java 独特的 .class 文件(字节码),能够将这些文件中的信息读取出来,使得 Java 程序只需要生成 Java 虚拟机上的字节码后就能在不同操作系统平台上进行运行。

    180 引用 • 120 回帖 • 1 关注
  • React

    React 是 Facebook 开源的一个用于构建 UI 的 JavaScript 库。

    192 引用 • 291 回帖 • 433 关注
  • 架构

    我们平时所说的“架构”主要是指软件架构,这是有关软件整体结构与组件的抽象描述,用于指导软件系统各个方面的设计。另外还有“业务架构”、“网络架构”、“硬件架构”等细分领域。

    140 引用 • 441 回帖 • 1 关注
  • Chrome

    Chrome 又称 Google 浏览器,是一个由谷歌公司开发的网页浏览器。该浏览器是基于其他开源软件所编写,包括 WebKit,目标是提升稳定性、速度和安全性,并创造出简单且有效率的使用者界面。

    60 引用 • 287 回帖
  • 反馈

    Communication channel for makers and users.

    124 引用 • 907 回帖 • 210 关注
  • 电影

    这是一个不能说的秘密。

    120 引用 • 597 回帖
  • danl
    89 关注
  • Laravel

    Laravel 是一套简洁、优雅的 PHP Web 开发框架。它采用 MVC 设计,是一款崇尚开发效率的全栈框架。

    19 引用 • 23 回帖 • 699 关注
  • Maven

    Maven 是基于项目对象模型(POM)、通过一小段描述信息来管理项目的构建、报告和文档的软件项目管理工具。

    186 引用 • 318 回帖 • 336 关注
  • BND

    BND(Baidu Netdisk Downloader)是一款图形界面的百度网盘不限速下载器,支持 Windows、Linux 和 Mac,详细介绍请看这里

    107 引用 • 1281 回帖 • 31 关注
  • Angular

    AngularAngularJS 的新版本。

    26 引用 • 66 回帖 • 531 关注
  • 知乎

    知乎是网络问答社区,连接各行各业的用户。用户分享着彼此的知识、经验和见解,为中文互联网源源不断地提供多种多样的信息。

    10 引用 • 66 回帖
  • 学习

    “梦想从学习开始,事业从实践起步” —— 习近平

    163 引用 • 473 回帖
  • GitLab

    GitLab 是利用 Ruby 一个开源的版本管理系统,实现一个自托管的 Git 项目仓库,可通过 Web 界面操作公开或私有项目。

    46 引用 • 72 回帖
  • 30Seconds

    📙 前端知识精选集,包含 HTML、CSS、JavaScript、React、Node、安全等方面,每天仅需 30 秒。

    • 精选常见面试题,帮助您准备下一次面试
    • 精选常见交互,帮助您拥有简洁酷炫的站点
    • 精选有用的 React 片段,帮助你获取最佳实践
    • 精选常见代码集,帮助您提高打码效率
    • 整理前端界的最新资讯,邀您一同探索新世界
    488 引用 • 383 回帖 • 4 关注
  • WordPress

    WordPress 是一个使用 PHP 语言开发的博客平台,用户可以在支持 PHP 和 MySQL 数据库的服务器上架设自己的博客。也可以把 WordPress 当作一个内容管理系统(CMS)来使用。WordPress 是一个免费的开源项目,在 GNU 通用公共许可证(GPLv2)下授权发布。

    45 引用 • 113 回帖 • 284 关注
  • 前端

    前端技术一般分为前端设计和前端开发,前端设计可以理解为网站的视觉设计,前端开发则是网站的前台代码实现,包括 HTML、CSS 以及 JavaScript 等。

    247 引用 • 1347 回帖
  • Flume

    Flume 是一套分布式的、可靠的,可用于有效地收集、聚合和搬运大量日志数据的服务架构。

    9 引用 • 6 回帖 • 608 关注
  • PHP

    PHP(Hypertext Preprocessor)是一种开源脚本语言。语法吸收了 C 语言、 Java 和 Perl 的特点,主要适用于 Web 开发领域,据说是世界上最好的编程语言。

    165 引用 • 407 回帖 • 514 关注
  • Solo

    Solo 是一款小而美的开源博客系统,专为程序员设计。Solo 有着非常活跃的社区,可将文章作为帖子推送到社区,来自社区的回帖将作为博客评论进行联动(具体细节请浏览 B3log 构思 - 分布式社区网络)。

    这是一种全新的网络社区体验,让热爱记录和分享的你不再感到孤单!

    1425 引用 • 10043 回帖 • 474 关注
  • Scala

    Scala 是一门多范式的编程语言,集成面向对象编程和函数式编程的各种特性。

    13 引用 • 11 回帖 • 111 关注
  • 一些有用的避坑指南。

    69 引用 • 93 回帖
  • RESTful

    一种软件架构设计风格而不是标准,提供了一组设计原则和约束条件,主要用于客户端和服务器交互类的软件。基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存等机制。

    30 引用 • 114 回帖