【Android開發日記】之入門篇(十五)——ViewPager+自定義無限ViewPager

来源:http://www.cnblogs.com/cpacm/archive/2016/06/03/5555221.html
-Advertisement-
Play Games

ViewPager 在 Android 控制項中,ViewPager 一直算是使用率比較高的控制項,包括首頁的banner,tab頁的切換都能見到ViewPager的身影。 來源自 v4 支持包 ( ),用於左右切換界面實現tab的效果。其使用方法與 類似都是搭配一個adapter進行數據適配。 在佈局 ...


ViewPager

在 Android 控制項中,ViewPager 一直算是使用率比較高的控制項,包括首頁的banner,tab頁的切換都能見到ViewPager的身影。
viewpager 來源自 v4 支持包 (android.support.v4.view.ViewPager),用於左右切換界面實現tab的效果。其使用方法與 ListView 類似都是搭配一個adapter進行數據適配。

在佈局文件中添加

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

通常Viewpager直接往佈局文件裡面添加就可以了,雖然ViewPager是一個容器類,繼承自 ViewGroup,但不建議往裡面添加子view。
之後可以在代碼中像調用普通控制項一樣通過 findViewById(R.id.viewpager) 通過id獲取viewpager控制項。

設置 PagerAdapter

PagerAdapter pagerAdapter = new PagerAdapter() {  
  
    //
    @Override  
    public boolean isViewFromObject(View arg0, Object arg1) {  

        return arg0 == arg1;  
    }  

    //返回要滑動的VIew的個數
    @Override  
    public int getCount() {  

        return viewList.size();  
    }  

    //從當前container中刪除指定位置(position)的View;
    @Override  
    public void destroyItem(ViewGroup container, int position,  
            Object object) {  
        container.removeView(viewList.get(position));  

    }  

    @Override  
    public int getItemPosition(Object object) {  

        return super.getItemPosition(object);  
    }  

    @Override  
    public CharSequence getPageTitle(int position) {  

        return titleList.get(position);  
    }  

    //做了兩件事,第一:將當前視圖添加到container中,第二:返回當前View
    @Override  
    public Object instantiateItem(ViewGroup container, int position) {  
        container.addView(viewList.get(position));  
        return viewList.get(position);  
    }  

};

PagerAdapter 支持數據集合的改變,可以調用notifyDataSetChanged方法來進行更新。和BaseAdapter非常相似。

設置 ViewPager

viewPager = (ViewPager) findViewById(R.id.viewpager);  
LayoutInflater inflater=getLayoutInflater();  
view1 = inflater.inflate(R.layout.layout1, null);  
view2 = inflater.inflate(R.layout.layout2,null);  
view3 = inflater.inflate(R.layout.layout3, null);  
  
viewList = new ArrayList<View>();// 將要分頁顯示的View裝入數組中  
viewList.add(view1);  
viewList.add(view2);  
viewList.add(view3); 

//初始化 pagerAdapter 
PagerAdapter pagerAdapter = new PagerAdapter() {...}
viewPager.setAdapter(pagerAdapter);

此時PagerAdapter包含了viewList數據,此時就能顯示到Viewpager中。

ViewPager擴展

PagerTitleStrip 和 PagerTabStrip

這兩個類都屬於v4 support包中的類,兩個都是是ViewPager的一個關於當前頁面、上一個頁面和下一個頁面的指示器。
其作為ViewPager控制項的一個子控制項被被添加在XML佈局文件中,每個頁面的標題是通過適配器的getPageTitle(int)函數提供給ViewPager的。
但兩者又有不同:
1、 PagerTabStrip 是可交互的,點哪滾到哪;
2、 PagerTabStrip 在當前頁面下,會有一個下劃線條來提示當前頁面的Tab是哪個。
這裡以 PagerTabStrip 為例

使用

添加佈局

<android.support.v4.view.ViewPager  
    android:id="@+id/viewpager"  
    android:layout_width="match_parent"  
    android:layout_height="140dp"  
    android:layout_gravity="center">  
      
        <android.support.v4.view.PagerTabStrip  
        android:id="@+id/pagertab"  
        android:layout_width="match_parent"  
        android:layout_height="wrap_content"   
        android:layout_gravity="top"/>  
          
</android.support.v4.view.ViewPager>

添加代碼

    titleList = new ArrayList<String>();// 每個頁面的Title數據  
    titleList.add("1111");  
    titleList.add("2222");  
    titleList.add("3333");

通過PagerAdaper的getPageTitle(int position)來設置標題

@Override  
public CharSequence getPageTitle(int position) {  
    return titleList.get(position);  
}

缺陷

但由於 PagerTabStrip 的局限性,大家一般都會自定義一個指示器,通過ViewPager.OnPageChangeListener綁定。
同時安利一下 google design 包裡面的 TabLayout 和 最新推出的Bottom navigation

開源項目:無限 viewpager——一款可以高度自定義的 slider

這是一款我剛寫的開源庫,起初寫這個是因為我想要一個可以高度自定義化的輪播器,它包括了一下幾個特點:

使用自己項目中的ImageLoader,而不是被迫使用庫中的圖片載入器;
庫中內置幾款常用的指示器,也可以創建自己想要的指示器;
豐富多彩的轉場動畫,亦可以發揮你的創意創造新的特效;
可以為每一頁建立動畫效果。

Github地址

關於無限輪播

一般的ViewPager播放到最後一個位置時,若要返回到第一個位置則必須從右向左,作為Banner時這是不自然的過渡效果。
我們知道ViewPager顯示的個數是由PagerAdaper的getCount()方法決定的,既然如此那我們可以給它賦予一個極大的值使ViewPager不斷向右輪播,這也是整個無限輪播的核心思想。

public class InfinitePagerAdapter extends PagerAdapter {

    private static final String TAG = "InfinitePagerAdapter";
    private static final boolean DEBUG = false;

    private BaseSliderAdapter adapter;

    public InfinitePagerAdapter(BaseSliderAdapter adapter) {
        this.adapter = adapter;
        adapter.registerDataSetObserver(new DataSetObserver() {
            @Override
            public void onChanged() {
                notifyDataSetChanged();
                super.onChanged();
            }
        });
    }

    public BaseSliderAdapter getRealAdapter() {
        return this.adapter;
    }

    @Override
    public CharSequence getPageTitle(int position) {
        return adapter.getPageTitle(position % getRealCount());
    }

    @Override
    public int getCount() {
        // warning: scrolling to very high values (1,000,000+) results in
        // strange drawing behaviour
        if (getRealCount() == 0) return 0;
        if (getRealCount() == 1) return 1;
        return Integer.MAX_VALUE;
    }

    /**
     * @return the {@link #getCount()} result of the wrapped adapter
     */
    public int getRealCount() {
        return adapter.getCount();
    }

    public BaseSliderView getSliderView(int position) {
        return adapter.getSliderView(position % getRealCount());
    }

    @Override
    public Object instantiateItem(ViewGroup container, int position) {
        if (getRealCount() == 0) {
            return null;
        }
        int virtualPosition = position % getRealCount();
        debug("instantiateItem: real position: " + position);
        debug("instantiateItem: virtual position: " + virtualPosition);

        // only expose virtual position to the inner adapter
        return adapter.instantiateItem(container, virtualPosition);
    }

    @Override
    public void destroyItem(ViewGroup container, int position, Object object) {
        if (getRealCount() == 0) {
            return;
        }
        int virtualPosition = position % getRealCount();
        debug("destroyItem: real position: " + position);
        debug("destroyItem: virtual position: " + virtualPosition);

        // only expose virtual position to the inner adapter
        adapter.destroyItem(container, virtualPosition, object);
    }

    /*
     * Delegate rest of methods directly to the inner adapter.
     */

    @Override
    public void finishUpdate(ViewGroup container) {
        adapter.finishUpdate(container);
    }

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

    @Override
    public void restoreState(Parcelable bundle, ClassLoader classLoader) {
        adapter.restoreState(bundle, classLoader);
    }

    @Override
    public Parcelable saveState() {
        return adapter.saveState();
    }

    @Override
    public void startUpdate(ViewGroup container) {
        adapter.startUpdate(container);
    }

    /*
     * End delegation
     */

    private void debug(String message) {
        if (DEBUG) {
            Log.d(TAG, message);
        }
    }
}

可以看到在代碼中還存在著 BaseSliderAdapter 另一個 PagerAdapter ,這是一個正常的PagerAdapter,它既是數據的真正載體同時也起著維護viewpager真正個數的作用。

但用這個方法實現的無限輪播存在著一個很大的缺陷,在 page 頁少於3個的情況下會出現問題,這是由於ViewPager本身的機制導致的。
簡單來說,ViewPager在顯示的時候會同時存在3個page頁,當前顯示頁,當前顯示的前一頁,當前顯示的後一頁。當我們輪播到下一頁時,原來的前一頁會被回收,原來的當前頁變成前一頁,原來的後一頁變成當前頁,同時會載入出新的一頁作為後一頁。
而在無限輪播小於等於3頁時,由於前一頁可能沒有被回收就被當做新的一頁載入到後一頁中,這就導致了view重覆被ViewGroup添加,而在Android中View只能被允許擁有一個ParentView,這裡就出現了問題。
所以在page頁少於等於3頁的時候還是要使用基礎的 PagerAdapter。

關於指示器

指示器可以通過繼承 ViewPager.OnPageChangeListener與ViewPager保持聯動,其監聽器包括以下方法:

@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
    // 滑動時的參數變化
}

@Override
public void onPageSelected(int position) {
    // 滑動到第幾頁
}

@Override
public void onPageScrollStateChanged(int state) {
   // 滑動時的狀態變化
}

剩下的就是看你怎麼定義指示器的界面,附上指示器演示圖和一個自定義的指示器代碼
indicator

public class CirclePageIndicator extends View implements PageIndicator {
    private static final int INVALID_POINTER = -1;

    private float mRadius;
    private final Paint mPaintPageFill = new Paint(ANTI_ALIAS_FLAG);
    private final Paint mPaintStroke = new Paint(ANTI_ALIAS_FLAG);
    private final Paint mPaintFill = new Paint(ANTI_ALIAS_FLAG);
    private ViewPager mViewPager;
    private ViewPager.OnPageChangeListener mListener;
    private int mCurrentPage;
    private int mSnapPage;
    private float mPageOffset;
    private int mScrollState;
    private int mOrientation;
    private boolean mCentered;
    private boolean mSnap;

    private int mTouchSlop;
    private float mLastMotionX = -1;
    private int mActivePointerId = INVALID_POINTER;
    private boolean mIsDragging;


    public CirclePageIndicator(Context context) {
        this(context, null);
    }

    public CirclePageIndicator(Context context, AttributeSet attrs) {
        this(context, attrs, R.attr.vpiCirclePageIndicatorStyle);
    }

    public CirclePageIndicator(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        if (isInEditMode()) return;

        //Load defaults from resources
        final Resources res = getResources();
        final int defaultPageColor = res.getColor(R.color.default_circle_indicator_page_color);
        final int defaultFillColor = res.getColor(R.color.default_circle_indicator_fill_color);
        final int defaultOrientation = res.getInteger(R.integer.default_circle_indicator_orientation);
        final int defaultStrokeColor = res.getColor(R.color.default_circle_indicator_stroke_color);
        final float defaultStrokeWidth = res.getDimension(R.dimen.default_circle_indicator_stroke_width);
        final float defaultRadius = res.getDimension(R.dimen.default_circle_indicator_radius);
        final boolean defaultCentered = res.getBoolean(R.bool.default_circle_indicator_centered);
        final boolean defaultSnap = res.getBoolean(R.bool.default_circle_indicator_snap);

        //Retrieve styles attributes
        TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.CirclePageIndicator, defStyle, 0);

        mCentered = a.getBoolean(R.styleable.CirclePageIndicator_centered, defaultCentered);
        mOrientation = a.getInt(R.styleable.CirclePageIndicator_android_orientation, defaultOrientation);
        mPaintPageFill.setStyle(Style.FILL);
        mPaintPageFill.setColor(a.getColor(R.styleable.CirclePageIndicator_pageColor, defaultPageColor));
        mPaintStroke.setStyle(Style.STROKE);
        mPaintStroke.setColor(a.getColor(R.styleable.CirclePageIndicator_strokeColor, defaultStrokeColor));
        mPaintStroke.setStrokeWidth(a.getDimension(R.styleable.CirclePageIndicator_strokeWidth, defaultStrokeWidth));
        mPaintFill.setStyle(Style.FILL);
        mPaintFill.setColor(a.getColor(R.styleable.CirclePageIndicator_fillColor, defaultFillColor));
        mRadius = a.getDimension(R.styleable.CirclePageIndicator_radius, defaultRadius);
        mSnap = a.getBoolean(R.styleable.CirclePageIndicator_snap, defaultSnap);

        Drawable background = a.getDrawable(R.styleable.CirclePageIndicator_android_background);
        if (background != null) {
            setBackgroundDrawable(background);
        }

        a.recycle();

        final ViewConfiguration configuration = ViewConfiguration.get(context);
        mTouchSlop = ViewConfigurationCompat.getScaledPagingTouchSlop(configuration);
    }


    public void setCentered(boolean centered) {
        mCentered = centered;
        invalidate();
    }

    public boolean isCentered() {
        return mCentered;
    }

    public void setPageColor(int pageColor) {
        mPaintPageFill.setColor(pageColor);
        invalidate();
    }

    public int getPageColor() {
        return mPaintPageFill.getColor();
    }

    public void setFillColor(int fillColor) {
        mPaintFill.setColor(fillColor);
        invalidate();
    }

    public int getFillColor() {
        return mPaintFill.getColor();
    }

    public void setOrientation(int orientation) {
        switch (orientation) {
            case HORIZONTAL:
            case VERTICAL:
                mOrientation = orientation;
                requestLayout();
                break;

            default:
                throw new IllegalArgumentException("Orientation must be either HORIZONTAL or VERTICAL.");
        }
    }

    public int getOrientation() {
        return mOrientation;
    }

    public void setStrokeColor(int strokeColor) {
        mPaintStroke.setColor(strokeColor);
        invalidate();
    }

    public int getStrokeColor() {
        return mPaintStroke.getColor();
    }

    public void setStrokeWidth(float strokeWidth) {
        mPaintStroke.setStrokeWidth(strokeWidth);
        invalidate();
    }

    public float getStrokeWidth() {
        return mPaintStroke.getStrokeWidth();
    }

    public void setRadius(float radius) {
        mRadius = radius;
        invalidate();
    }

    public float getRadius() {
        return mRadius;
    }

    public void setSnap(boolean snap) {
        mSnap = snap;
        invalidate();
    }

    public boolean isSnap() {
        return mSnap;
    }

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

        if (mViewPager == null) {
            return;
        }
        int count = mViewPager.getAdapter().getCount();
        if (mViewPager.getAdapter() instanceof InfinitePagerAdapter) {
            count = ((InfinitePagerAdapter) mViewPager.getAdapter()).getRealCount();
            mCurrentPage = mCurrentPage % count;
        }
        if (count == 0) {
            return;
        }

        if (mCurrentPage >= count) {
            setCurrentItem(count - 1);
            return;
        }

        int longSize;
        int longPaddingBefore;
        int longPaddingAfter;
        int shortPaddingBefore;
        if (mOrientation == HORIZONTAL) {
            longSize = getWidth();
            longPaddingBefore = getPaddingLeft();
            longPaddingAfter = getPaddingRight();
            shortPaddingBefore = getPaddingTop();
        } else {
            longSize = getHeight();
            longPaddingBefore = getPaddingTop();
            longPaddingAfter = getPaddingBottom();
            shortPaddingBefore = getPaddingLeft();
        }

        final float threeRadius = mRadius * 3;
        final float shortOffset = shortPaddingBefore + mRadius;
        float longOffset = longPaddingBefore + mRadius;
        if (mCentered) {
            longOffset += ((longSize - longPaddingBefore - longPaddingAfter) / 2.0f) - ((count * threeRadius-mRadius) / 2.0f);
        }

        float dX;
        float dY;

        float pageFillRadius = mRadius;
        if (mPaintStroke.getStrokeWidth() > 0) {
            pageFillRadius -= mPaintStroke.getStrokeWidth() / 2.0f;
        }

        //Draw stroked circles
        for (int iLoop = 0; iLoop < count; 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);
            }
        }

        //Draw the filled circle according to the current scroll
        float cx = (mSnap ? mSnapPage : mCurrentPage) * threeRadius;
        if (!mSnap && mCurrentPage < count - 1) {
            cx += mPageOffset * threeRadius;
        }
        if (mOrientation == HORIZONTAL) {
            dX = longOffset + cx;
            dY = shortOffset;
        } else {
            dX = shortOffset;
            dY = longOffset + cx;
        }
        canvas.drawCircle(dX, dY, mRadius, mPaintFill);
    }

    @Override
    public boolean onTouchEvent(MotionEvent ev) {
        if (super.onTouchEvent(ev)) {
            return true;
        }
        if ((mViewPager == null) || (mViewPager.getAdapter().getCount() == 0)) {
            return false;
        }

        final int action = ev.getAction() & MotionEventCompat.ACTION_MASK;
        switch (action) {
            case MotionEvent.ACTION_DOWN:
                mActivePointerId = MotionEventCompat.getPointerId(ev, 0);
                mLastMotionX = ev.getX();
                break;
            case MotionEvent.ACTION_MOVE: {
                final int activePointerIndex = MotionEventCompat.findPointerIndex(ev, mActivePointerId);
                final float x = MotionEventCompat.getX(ev, activePointerIndex);
                final float deltaX = x - mLastMotionX;

                if (!mIsDragging) {
                    if (Math.abs(deltaX) > mTouchSlop) {
                        mIsDragging = true;
                    }
                }

                if (mIsDragging) {
                    mLastMotionX = x;
                    if (mViewPager.isFakeDragging() || mViewPager.beginFakeDrag()) {
                        mViewPager.fakeDragBy(deltaX);
                    }
                }
                break;
            }
            case MotionEvent.ACTION_CANCEL:
            case MotionEvent.ACTION_UP:
                if (!mIsDragging) {
                    int count = mViewPager.getAdapter().getCount();
                    if (mViewPager.getAdapter() instanceof InfinitePagerAdapter) {
                        count = ((InfinitePagerAdapter) mViewPager.getAdapter()).getRealCount();
                    }
                    final int width = getWidth();
                    final float halfWidth = width / 2f;
                    final float sixthWidth = width / 6f;

                    if ((mCurrentPage > 0) && (ev.getX() < halfWidth - sixthWidth)) {
                        if (action != MotionEvent.ACTION_CANCEL) {
                            mViewPager.setCurrentItem(mCurrentPage - 1);
                        }
                        return true;
                    } else if ((mCurrentPage < count - 1) && (ev.getX() > halfWidth + sixthWidth)) {
                        if (action != MotionEvent.ACTION_CANCEL) {
                            mViewPager.setCurrentItem(mCurrentPage + 1);
                        }
                        return true;
                    }
                }
                mIsDragging = false;
                mActivePointerId = INVALID_POINTER;
                if (mViewPager.isFakeDragging()) mViewPager.endFakeDrag();
                break;
            case MotionEventCompat.ACTION_POINTER_DOWN: {
                final int index = MotionEventCompat.getActionIndex(ev);
                mLastMotionX = MotionEventCompat.getX(ev, index);
                mActivePointerId = MotionEventCompat.getPointerId(ev, index);
                break;
            }
            case MotionEventCompat.ACTION_POINTER_UP:
                final int pointerIndex = MotionEventCompat.getActionIndex(ev);
                final int pointerId = MotionEventCompat.getPointerId(ev, pointerIndex);
                if (pointerId == mActivePointerId) {
                    final int newPointerIndex = pointerIndex == 0 ? 1 : 0;
                    mActivePointerId = MotionEventCompat.getPointerId(ev, newPointerIndex);
                }
                mLastMotionX = MotionEventCompat.getX(ev, MotionEventCompat.findPointerIndex(ev, mActivePointerId));
                break;
        }
        return true;
    }

    @Override
    public void setViewPager(ViewPager view) {
        if (mViewPager == view) {
            return;
        }
        if (view.getAdapter() == null) {
            throw new IllegalStateException("ViewPager does not have adapter instance.");
        }
        mViewPager = view;
        mViewPager.addOnPageChangeListener(this);
        invalidate();
    }

    @Override
    public void setViewPager(ViewPager view, int initialPosition) {
        setViewPager(view);
        setCurrentItem(initialPosition);
    }

    @Override
    public void setCurrentItem(int item) {
        if (mViewPager == null) {
            throw new IllegalStateException("ViewPager has not been bound.");
        }
        mViewPager.setCurrentItem(item);
        mCurrentPage = item;
        invalidate();
    }

    @Override
    public void notifyDataSetChanged() {
        invalidate();
    }

    @Override
    public void onPageScrollStateChanged(int state) {
        mScrollState = state;

        if (mListener != null) {
            mListener.onPageScrollStateChanged(state);
        }
    }

    @Override
    public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
        if (mViewPager.getAdapter() instanceof InfinitePagerAdapter) {
            position = position % ((InfinitePagerAdapter) mViewPager.getAdapter()).getRealCount();
        }
        mCurrentPage = position;
        mPageOffset = positionOffset;
        invalidate();

        if (mListener != null) {
            mListener.onPageScrolled(position, positionOffset, positionOffsetPixels);
        }
    }

    @Override
    public void onPageSelected(int position) {
        if (mViewPager.getAdapter() instanceof InfinitePagerAdapter) {
            position = position % ((InfinitePagerAdapter) mViewPager.getAdapter()).getRealCount();
        }
        if (mSnap || mScrollState == ViewPager.SCROLL_STATE_IDLE) {
            mCurrentPage = position;
            mSnapPage = position;
            invalidate();
        }

        if (mListener != null) {
            mListener.onPageSelected(position);
        }
    }

    @Override
    public void addOnPageChangeListener(ViewPager.OnPageChangeListener listener) {
        mListener = listener;
    }

    /*
     * (non-Javadoc)
     *
     * @see android.view.View#onMeasure(int, int)
     */
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        if (mOrientation == HORIZONTAL) {
            setMeasuredDimension(measureLong(widthMeasureSpec), measureShort(heightMeasureSpec));
        } else {
            setMeasuredDimension(measureShort(widthMeasureSpec), measureLong(heightMeasureSpec));
        }
    }

    /**
     * Determines the width of this view
     *
     * @param measureSpec A measureSpec packed into an int
     * @return The width of the view, honoring constraints from measureSpec
     */
    private int measureLong(int measureSpec) {
        int result;
        int specMode = MeasureSpec.getMode(measureSpec);
        int specSize = MeasureSpec.getSize(measureSpec);

        if ((specMode == MeasureSpec.EXACTLY) || (mViewPager == null)) {
            //We were told how big to be
            result = specSize;
        } else {
            //Calculate the width according the views count
            int count = mViewPager.getAdapter().getCount();
            if (mViewPager.getAdapter() instanceof InfinitePagerAdapter) {
                count = ((InfinitePagerAdapter) mViewPager.getAdapter()).getRealCount();
            }
            result = (int) (getPaddingLeft() + getPaddingRight()
                    + (count * 2 * mRadius) + (count - 1) * mRadius + 1);
            //Respect AT_MOST value if that was what is called for by measureSpec
            if (specMode == MeasureSpec.AT_MOST) {
                result = Math.min(result, specSize);
            }
        }
        return result;
    }

    /**
     * Determines the height of this view
     *
     * @param measureSpec A measureSpec packed into an int
     * @return The height of the view, honoring constraints from measureSpec
     */
    private int measureShort(int measureSpec) {
        int result;
        int specMode = MeasureSpec.getMode(measureSpec);
        int specSize = MeasureSpec.getSize(measureSpec);

        if (specMode == MeasureSpec.EXACTLY) {
            //We were told how big to be
            result = specSize;
        } else {
            //Measure the height
            result = (int) (2 * mRadius + getPaddingTop() + getPaddingBottom() + 1);
            //Respect AT_MOST value if that was what is called for by measureSpec
            if (specMode == MeasureSpec.AT_MOST) {
                result = Math.min(result, specSize);
            }
        }
        return result;
    }

    @Override
    public void onRestoreInstanceState(Parcelable state) {
        SavedState savedState = (SavedState) state;
        super.onRestoreInstanceState(savedState.getSuperState());
        mCurrentPage = savedState.currentPage;
        mSnapPage = savedState.currentPage;
        requestLayout();
    }

    @Override
    public Parcelable onSaveInstanceState() {
        Parcelable superState = super.onSaveInstanceState();
        SavedState savedState = new SavedState(superState);
        savedState.currentPage = mCurrentPage;
        return savedState;
    }

    static class SavedState extends BaseSavedState {
        int currentPage;

        public SavedState(Parcelable superState) {
            super(superState);
        }

        private SavedState(Parcel in) {
            super(in);
            currentPage = in.readInt();
        }

        @Override
        public void writeToParcel(Parcel dest, int flags) {
            super.writeToParcel(dest, flags);
            dest.writeInt(currentPage);
        }

        @SuppressWarnings("UnusedDeclaration")
        public static final Creator<SavedState> CREATOR = new Creator<SavedState>() {
            @Override
            public SavedState createFromParcel(Parcel in) {
                return new SavedState(in);
            }

            @Override
            public SavedState[] newArray(int size) {
                return new SavedState[size];
            }
        };
    }
}

關於轉場動畫

ViewPager有個方法叫做setPageTransformer(boolean reverseDrawingOrder, PageTransformer transformer)用於設置ViewPager切換時的動畫效果
但註意這隻能在3.0及其以後使用。因為View的動畫使用的是屬性動畫,而屬性動畫是3.0才推出。當然這個問題可以剋服,首先先使用nineoldandroids讓動畫在3.0之前也能跑起來,然後再去修改ViewPager的源碼

    public void setPageTransformer(boolean reverseDrawingOrder, PageTransformer transformer) {  
       if (Build.VERSION.SDK_INT >= 11) {  
           final boolean hasTransformer = transformer != null;  
           final boolean needsPopulate = hasTransformer != (mPageTransformer != null);  
           mPageTransformer = transformer;  
           setChildrenDrawingOrderEnabledCompat(hasTransformer);  
           if (hasTransformer) {  
               mDrawingOrder = reverseDrawingOrder ? DRAW_ORDER_REVERSE : DRAW_ORDER_FORWARD;  
           } else {  
               mDrawingOrder = DRAW_ORDER_DEFAULT;  
           }  
           if (needsPopulate) populate();  
       }  
   } 

去除if (Build.VERSION.SDK_INT >= 11)這個if判斷就行了。
所有的PageTransformer都要繼承ViewPager.PageTransformer介面,其中包含的方法只有一個

    @Override
    public void transformPage(View view, float position) {

    }

其中position反映的是view的位置變化。
假設現在ViewPager在A頁現在滑出B頁,則:
A頁的position變化就是( 0, -1]
B頁的position變化就是[ 1 , 0 ]
根據這個position就可以做出多樣的轉場變化了。
Transform

關於pager動畫

同樣要利用 ViewPager.OnPageChangeListener 監聽ViewPager的變化以實現每個界面的變化。
Animation介面

public interface OnAnimationListener {
    void onNextAnimationStart(BaseSliderView slider);

    void onNextAnimationEnd(BaseSliderView slider);

    void onPreAnimationStart(BaseSliderView slider);

    void onPreAnimationEnd(BaseSliderView slider);
}

監聽變化

/**
 * A {@link ViewPager} that allows define custom animation
 */
public class AnimationViewPager extends ViewPager {

    private int position = 0, prePositon = 0;
    private OnAnimationListener animationListener;
    private BaseSliderView slider, preSlider;
    private boolean animating = false;

    public AnimationViewPager(Context context) {
        super(context);
        initViewPager();
    }

    public AnimationViewPager(Context context, AttributeSet attrs) {
        super(context, attrs);
        initViewPager();
    }

    private void initViewPager() {
        addOnPageChangeListener(new OnPageChangeListener() {
            @Override
            public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
                if (position == 0 && positionOffset == 0) {
                    animateSliderEnd(position);
                    if (slider != null)
                        animationListener.onNextAnimationEnd(slider);
                }
            }

            @Override
            public void onPageSelected(int position) {
                prePositon = AnimationViewPager.this.position;
                AnimationViewPager.this.position = position;
                if (prePositon > position)
                    animateSliderStart(position, position + 1);
                if (prePositon < position)
                    animateSliderStart(position, position - 1);
            }

            @Override
            public void onPageScrollStateChanged(int state) {
                if (state == SCROLL_STATE_IDLE) {
                    animateSliderEnd(position);
                }
            }
        });
    }

    private void animateSliderStart(int position, int prePositon) {
        if (animationListener == null) return;
        BaseSliderView slider = null, preSlider = null;
        if (getAdapter() instanceof InfinitePagerAdapter) {
            InfinitePagerAdapter pagerAdapter = (InfinitePagerAdapter) getAdapter();
            slider = pagerAdapter.getSliderView(position);
            preSlider = pagerAdapter.getSliderView(prePositon);
        } else if (getAdapter() instanceof BaseSliderAdapter) {
            BaseSliderAdapter sliderAdapter = (BaseSliderAdapter) getAdapter();
            slider = sliderAdapter.getSliderView(position);
            preSlider = sliderAdapter.getSliderView(prePositon);
        }
        if (slider != null)
            animationListener.onNextAnimationStart(slider);
        if (preSlider != null)
            animationListener.onPreAnimationStart(preSlider);

    }

    private void animateSliderEnd(int position) {
        if (animationListener == null) return;
        if (getAdapter() instanceof InfinitePagerAdapter) {
            InfinitePagerAdapter pagerAdapter = (InfinitePagerAdapter) getAdapter();
            if (slider == null) slider = pagerAdapter.getSliderView(position);
            else if (slider != pagerAdapter.getSliderView(position)) {
                preSlider = slider;
                slider = pagerAdapter.getSliderView(position);
            }
        } else if (getAdapter() instanceof BaseSliderAdapter) {
            BaseSliderAdapter sliderAdapter = (BaseSliderAdapter) getAdapter();
            if (slider == null) slider = sliderAdapter.getSliderView(position);
            else if (slider != sliderAdapter.getSliderView(position)) {
                preSlider = slider;
                slider = sliderAdapter.getSliderView(position);
            }
        }
        if (slider != null)
            animationListener.onNextAnimationEnd(slider);
        if (preSlider != null)
            animationListener.onPreAnimationEnd(preSlider);
        animating = false;
    }

    public OnAnimationListener getAnimationListener() {
        return animationListener;
    }

    public void setAnimationListener(OnAnimationListener animationListener) {
        this.animationListener = animationListener;
    }
}

繼承OnAnimationListener介面實現後的例子

public class DefaultDescriptionAnimation implements OnAnimationListener {

    private final long DURATION = 300;

    @Override
    public void onNextAnimationStart(BaseSliderView slider) {
        Log.d("simpleSlider", "onNextAnimationStart:" + slider.getPageTitle());
    }

    @Override
    public void onNextAnimationEnd(BaseSliderView slider) {
        Log.d("simpleSlider", "onNextAnimationEnd:" + slider.getPageTitle());
        DescriptionSliderView sliderView = (DescriptionSliderView) slider;
        if (sliderView.getTitleLayout().getVisi

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在實際開發中,jQuery的實踐性非常強大。上一篇本人已經整理出了一部分基礎知識點,該文即是對以上的補充和擴展。 1、表單值的操作 2、屬性操作 3、特性操作 【註意】checked、selected、disabled要使用.prop()方法。 prop方法通常用來影響DOM元素的動態狀態,而不是改 ...
  • 今天朋友共用了一道js中經典的面試題,需求是這樣的 給定你任意一個字元串,讓你寫出一個演算法,求算出該字元串中出現次數最多的一個字元,並將其結果輸出 剛拿到這道題的第一感覺便是定義一個count計時器,然後通過for迴圈 裡面嵌套if判斷,但是這樣的話貌似並不是一個很好的解決方案 因為你並不知道該字元 ...
  • 俗話說欲善其功,必先利其器。 作為目前新型的Web Server開發棧倍受開發者關註的Nodejs來說,調試技術是學習開發的基石,所以對於開始學習Nodejs童鞋來說,Nodejs的調試工具使用是必不可少的,總的來說Nodejs的調試方法主要分類三類: 1、基於Nodejs內建的調試器 2、基於V8 ...
  • Sass 是 CSS 的擴展,增加了嵌套規則,變數,混入功能等很多更多。它簡化了組織和維護 CSS 代碼的成本。Compass 是一個開源的 CSS 框架,使得使用 CSS3 和流行的設計模式比以往任何時候都更容易。在這篇文章中,我們已經收集了一組有用的 Sass 和 Compass 工具,將幫助您... ...
  • 學習要點: 1.什麼是 jQuery EasyUI 2.學習 jQuery EasyUI 的條件 3.jQuery EasyUI 的功能和優勢 4.其他的 UI 插件 5.是否相容低版本 IE 6.下載及運行 jQuery EasyUI 在正式瞭解 jQuery EasyUI 之前,我們先瞭解一下什 ...
  • 說到水平居中,大家可能覺得很簡單啊,text-align:center 就OK了。 但是,有時候會發現這樣寫了也沒出效果。原因是什麼呢? 請往下看。 水平居中:分為塊級元素居中和行元素居中 行內元素: 行內元素就是內聯元素。例如<span>、<a>、<label>、<em>、<img>等。。 直接構 ...
  • 學習要點: 1.底部區域 2.說明區域 3.版權及證件區 主講教師:李炎恢 本章主要開始使用學慣用 HTML5 和 CSS3 來構建 Web 頁面,第一個項目採用 PC 端固定佈局來實現。 一.底部區域 本節課,我們將探討一下首頁中最底部的區域。這部分區域由兩個部分組成,一個是說明內容,有:合作伙伴 ...
  • 題目:編寫如下頁面 當用戶點擊”統計“按鈕時,在視窗中彈出文本框中出現次數最多的字元並顯示其出現的次數 點擊統計按鈕時效果如圖所示: 實現代碼: 本題的主要知識點 js對象屬性可以後期添加的特點、對象屬性的遍歷等js對象的綜合運用。js相關的知識可以參考我之前的博客 javascript對象的相關操 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...