PagerSlidingTabStrip源码解析
项目简介
项目地址https://github.com/astuetz/PagerSlidingTabStrip
该项目是一个配合ViewPager使用的指示器控件,这里的ViewPager的adapter必须是继承FragmentPagerAdapter,且需要重写getPageIconResId(int position)或者getPageTitle(int position)以便指示器显示内容。
使用方法
首先在布局文件中包含PagerSlidingTabTrip和ViewPager
1 |
|
然后绑定ViewPager
1 | indicator = (PagerSlidingTabStrip) findViewById(R.id.indicator); |
这里需要注意的是如果要为ViewPager设置OnPageChangeListener应该设置在indicator里,而不是直接为ViewPager设置,至于为什么下面会解释。
源码解析
从构造方法开始
1 | public PagerSlidingTabStrip(Context context, AttributeSet attrs, int defStyle) { |
在构造方法中先是new了一个LinearLayout作为tab的容器,然后取得一些属性,完成了一些初始化操作。
接下来看一下setViewPager(ViewPager pager)方法
1 | public void setViewPager(ViewPager pager) { |
这里将viewpager保存起来,然后设置listener,不过这里的listener是一个内部类,等下再来分析这个类,在该方法的最后调用方法notifyDataSetChanged(),我们来看一下
1 | public void notifyDataSetChanged() { |
这里主要完成tab的初始化,没有什么难度。接下来就是比较重要的PageListener类了
1 | private class PageListener implements OnPageChangeListener { |
该控件提供一个setOnPageChangeListener()方法允许用户设置自己的Listener,然后在PageListener类中每个方法都会在listener非空 的情况下调用相应方法。
在ViewPager滑动的时候调用scrollToChild方法滑动自身,然后通过invalidata触发onDraw绘制indicator。
绘制的主要代码为
1 | ...... |
改变一下就可以发现
lineLeft = lineLeft + (nextTabLeft - lineLeft) * currentPositionOffset
lineRight = lineRight + (nextTabRight - lineRight) * currentPositionOffset
而且nextTabLeft - lineLeft就是当前tab的width,nextTabRight - lineRight是下一个tab的width
所以最后就是
lineLeft = lineLeft + currentWidth * currentPositionOffset
lineright = lineRight + nextWidth * currentPositionOffset
这样的写法可以动态改变indicator的width。
值得注意的是我们发现在无论是scrollToChild()还是onDraw()中都用的是tab.getLeft(),那么我们一个一个来分析。
在scrollToChild()中,是使用scrollTo()滑动scrollview,tab.getLeft()得到的是相对于父控件的距离,也就是相对于LinearLayout的距离,这样无论怎么滚动,任何tab的left都是不会变的,因为是相对于LinearLayout的距离,而LinearLayout是不会变的。然后根据该left加上偏移量去scrollTo(),就会正好向左滑动让tab靠在最左边。
onDraw()中主要是为了绘制indicator,canvas的绘制应该是绘制在本身的,也就是绘制在HorizontalScrollView上的,经过试验发现如果我们draw的left为0,那么会显示在LinearLayout的最左边,也就是说left为0的位置并不是在显示的最左端,而是实际上内容的最左端,可以把整个控件想象为完全展开的,left就是最左边,虽然有可能因为滑动而被挡住,没有显示出来。