Home | History | Annotate | Download | only in pageindicators
      1 /*
      2  * Copyright (C) 2016 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package com.android.launcher3.pageindicators;
     18 
     19 import android.animation.Animator;
     20 import android.animation.AnimatorListenerAdapter;
     21 import android.animation.AnimatorSet;
     22 import android.animation.ObjectAnimator;
     23 import android.animation.ValueAnimator;
     24 import android.animation.ValueAnimator.AnimatorUpdateListener;
     25 import android.content.Context;
     26 import android.graphics.Canvas;
     27 import android.graphics.Outline;
     28 import android.graphics.Paint;
     29 import android.graphics.Paint.Style;
     30 import android.graphics.RectF;
     31 import android.util.AttributeSet;
     32 import android.util.Property;
     33 import android.view.View;
     34 import android.view.ViewOutlineProvider;
     35 import android.view.animation.Interpolator;
     36 import android.view.animation.OvershootInterpolator;
     37 
     38 import com.android.launcher3.R;
     39 import com.android.launcher3.Utilities;
     40 import com.android.launcher3.util.Themes;
     41 
     42 /**
     43  * {@link PageIndicator} which shows dots per page. The active page is shown with the current
     44  * accent color.
     45  */
     46 public class PageIndicatorDots extends View implements PageIndicator {
     47 
     48     private static final float SHIFT_PER_ANIMATION = 0.5f;
     49     private static final float SHIFT_THRESHOLD = 0.1f;
     50     private static final long ANIMATION_DURATION = 150;
     51 
     52     private static final int ENTER_ANIMATION_START_DELAY = 300;
     53     private static final int ENTER_ANIMATION_STAGGERED_DELAY = 150;
     54     private static final int ENTER_ANIMATION_DURATION = 400;
     55 
     56     // This value approximately overshoots to 1.5 times the original size.
     57     private static final float ENTER_ANIMATION_OVERSHOOT_TENSION = 4.9f;
     58 
     59     private static final RectF sTempRect = new RectF();
     60 
     61     private static final Property<PageIndicatorDots, Float> CURRENT_POSITION
     62             = new Property<PageIndicatorDots, Float>(float.class, "current_position") {
     63         @Override
     64         public Float get(PageIndicatorDots obj) {
     65             return obj.mCurrentPosition;
     66         }
     67 
     68         @Override
     69         public void set(PageIndicatorDots obj, Float pos) {
     70             obj.mCurrentPosition = pos;
     71             obj.invalidate();
     72             obj.invalidateOutline();
     73         }
     74     };
     75 
     76     private final Paint mCirclePaint;
     77     private final float mDotRadius;
     78     private final int mActiveColor;
     79     private final int mInActiveColor;
     80     private final boolean mIsRtl;
     81 
     82     private int mNumPages;
     83     private int mActivePage;
     84 
     85     /**
     86      * The current position of the active dot including the animation progress.
     87      * For ex:
     88      *   0.0  => Active dot is at position 0
     89      *   0.33 => Active dot is at position 0 and is moving towards 1
     90      *   0.50 => Active dot is at position [0, 1]
     91      *   0.77 => Active dot has left position 0 and is collapsing towards position 1
     92      *   1.0  => Active dot is at position 1
     93      */
     94     private float mCurrentPosition;
     95     private float mFinalPosition;
     96     private ObjectAnimator mAnimator;
     97 
     98     private float[] mEntryAnimationRadiusFactors;
     99 
    100     public PageIndicatorDots(Context context) {
    101         this(context, null);
    102     }
    103 
    104     public PageIndicatorDots(Context context, AttributeSet attrs) {
    105         this(context, attrs, 0);
    106     }
    107 
    108     public PageIndicatorDots(Context context, AttributeSet attrs, int defStyleAttr) {
    109         super(context, attrs, defStyleAttr);
    110 
    111         mCirclePaint = new Paint(Paint.ANTI_ALIAS_FLAG);
    112         mCirclePaint.setStyle(Style.FILL);
    113         mDotRadius = getResources().getDimension(R.dimen.page_indicator_dot_size) / 2;
    114         setOutlineProvider(new MyOutlineProver());
    115 
    116         mActiveColor = Themes.getColorAccent(context);
    117         mInActiveColor = Themes.getAttrColor(context, android.R.attr.colorControlHighlight);
    118 
    119         mIsRtl = Utilities.isRtl(getResources());
    120     }
    121 
    122     @Override
    123     public void setScroll(int currentScroll, int totalScroll) {
    124         if (mNumPages > 1) {
    125             if (mIsRtl) {
    126                 currentScroll = totalScroll - currentScroll;
    127             }
    128             int scrollPerPage = totalScroll / (mNumPages - 1);
    129             int pageToLeft = currentScroll / scrollPerPage;
    130             int pageToLeftScroll = pageToLeft * scrollPerPage;
    131             int pageToRightScroll = pageToLeftScroll + scrollPerPage;
    132 
    133             float scrollThreshold = SHIFT_THRESHOLD * scrollPerPage;
    134             if (currentScroll < pageToLeftScroll + scrollThreshold) {
    135                 // scroll is within the left page's threshold
    136                 animateToPosition(pageToLeft);
    137             } else if (currentScroll > pageToRightScroll - scrollThreshold) {
    138                 // scroll is far enough from left page to go to the right page
    139                 animateToPosition(pageToLeft + 1);
    140             } else {
    141                 // scroll is between left and right page
    142                 animateToPosition(pageToLeft + SHIFT_PER_ANIMATION);
    143             }
    144         }
    145     }
    146 
    147     private void animateToPosition(float position) {
    148         mFinalPosition = position;
    149         if (Math.abs(mCurrentPosition - mFinalPosition) < SHIFT_THRESHOLD) {
    150             mCurrentPosition = mFinalPosition;
    151         }
    152         if (mAnimator == null && Float.compare(mCurrentPosition, mFinalPosition) != 0) {
    153             float positionForThisAnim = mCurrentPosition > mFinalPosition ?
    154                     mCurrentPosition - SHIFT_PER_ANIMATION : mCurrentPosition + SHIFT_PER_ANIMATION;
    155             mAnimator = ObjectAnimator.ofFloat(this, CURRENT_POSITION, positionForThisAnim);
    156             mAnimator.addListener(new AnimationCycleListener());
    157             mAnimator.setDuration(ANIMATION_DURATION);
    158             mAnimator.start();
    159         }
    160     }
    161 
    162     public void stopAllAnimations() {
    163         if (mAnimator != null) {
    164             mAnimator.cancel();
    165             mAnimator = null;
    166         }
    167         mFinalPosition = mActivePage;
    168         CURRENT_POSITION.set(this, mFinalPosition);
    169     }
    170 
    171     /**
    172      * Sets up up the page indicator to play the entry animation.
    173      * {@link #playEntryAnimation()} must be called after this.
    174      */
    175     public void prepareEntryAnimation() {
    176         mEntryAnimationRadiusFactors = new float[mNumPages];
    177         invalidate();
    178     }
    179 
    180     public void playEntryAnimation() {
    181         int count  = mEntryAnimationRadiusFactors.length;
    182         if (count == 0) {
    183             mEntryAnimationRadiusFactors = null;
    184             invalidate();
    185             return;
    186         }
    187 
    188         Interpolator interpolator = new OvershootInterpolator(ENTER_ANIMATION_OVERSHOOT_TENSION);
    189         AnimatorSet animSet = new AnimatorSet();
    190         for (int i = 0; i < count; i++) {
    191             ValueAnimator anim = ValueAnimator.ofFloat(0, 1).setDuration(ENTER_ANIMATION_DURATION);
    192             final int index = i;
    193             anim.addUpdateListener(new AnimatorUpdateListener() {
    194                 @Override
    195                 public void onAnimationUpdate(ValueAnimator animation) {
    196                     mEntryAnimationRadiusFactors[index] = (Float) animation.getAnimatedValue();
    197                     invalidate();
    198                 }
    199             });
    200             anim.setInterpolator(interpolator);
    201             anim.setStartDelay(ENTER_ANIMATION_START_DELAY + ENTER_ANIMATION_STAGGERED_DELAY * i);
    202             animSet.play(anim);
    203         }
    204 
    205         animSet.addListener(new AnimatorListenerAdapter() {
    206 
    207             @Override
    208             public void onAnimationEnd(Animator animation) {
    209                 mEntryAnimationRadiusFactors = null;
    210                 invalidateOutline();
    211                 invalidate();
    212             }
    213         });
    214         animSet.start();
    215     }
    216 
    217     @Override
    218     public void setActiveMarker(int activePage) {
    219         if (mActivePage != activePage) {
    220             mActivePage = activePage;
    221         }
    222     }
    223 
    224     @Override
    225     public void setMarkersCount(int numMarkers) {
    226         mNumPages = numMarkers;
    227         requestLayout();
    228     }
    229 
    230     @Override
    231     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    232         // Add extra spacing of mDotRadius on all sides so than entry animation could be run.
    233         int width = MeasureSpec.getMode(widthMeasureSpec) == MeasureSpec.EXACTLY ?
    234                 MeasureSpec.getSize(widthMeasureSpec) : (int) ((mNumPages * 3 + 2) * mDotRadius);
    235         int height= MeasureSpec.getMode(heightMeasureSpec) == MeasureSpec.EXACTLY ?
    236                 MeasureSpec.getSize(heightMeasureSpec) : (int) (4 * mDotRadius);
    237         setMeasuredDimension(width, height);
    238     }
    239 
    240     @Override
    241     protected void onDraw(Canvas canvas) {
    242         // Draw all page indicators;
    243         float circleGap = 3 * mDotRadius;
    244         float startX = (getWidth() - mNumPages * circleGap + mDotRadius) / 2;
    245 
    246         float x = startX + mDotRadius;
    247         float y = canvas.getHeight() / 2;
    248 
    249         if (mEntryAnimationRadiusFactors != null) {
    250             // During entry animation, only draw the circles
    251             if (mIsRtl) {
    252                 x = getWidth() - x;
    253                 circleGap = -circleGap;
    254             }
    255             for (int i = 0; i < mEntryAnimationRadiusFactors.length; i++) {
    256                 mCirclePaint.setColor(i == mActivePage ? mActiveColor : mInActiveColor);
    257                 canvas.drawCircle(x, y, mDotRadius * mEntryAnimationRadiusFactors[i], mCirclePaint);
    258                 x += circleGap;
    259             }
    260         } else {
    261             mCirclePaint.setColor(mInActiveColor);
    262             for (int i = 0; i < mNumPages; i++) {
    263                 canvas.drawCircle(x, y, mDotRadius, mCirclePaint);
    264                 x += circleGap;
    265             }
    266 
    267             mCirclePaint.setColor(mActiveColor);
    268             canvas.drawRoundRect(getActiveRect(), mDotRadius, mDotRadius, mCirclePaint);
    269         }
    270     }
    271 
    272     private RectF getActiveRect() {
    273         float startCircle = (int) mCurrentPosition;
    274         float delta = mCurrentPosition - startCircle;
    275         float diameter = 2 * mDotRadius;
    276         float circleGap = 3 * mDotRadius;
    277         float startX = (getWidth() - mNumPages * circleGap + mDotRadius) / 2;
    278 
    279         sTempRect.top = getHeight() * 0.5f - mDotRadius;
    280         sTempRect.bottom = getHeight() * 0.5f + mDotRadius;
    281         sTempRect.left = startX + startCircle * circleGap;
    282         sTempRect.right = sTempRect.left + diameter;
    283 
    284         if (delta < SHIFT_PER_ANIMATION) {
    285             // dot is capturing the right circle.
    286             sTempRect.right += delta * circleGap * 2;
    287         } else {
    288             // Dot is leaving the left circle.
    289             sTempRect.right += circleGap;
    290 
    291             delta -= SHIFT_PER_ANIMATION;
    292             sTempRect.left += delta * circleGap * 2;
    293         }
    294 
    295         if (mIsRtl) {
    296             float rectWidth = sTempRect.width();
    297             sTempRect.right = getWidth() - sTempRect.left;
    298             sTempRect.left = sTempRect.right - rectWidth;
    299         }
    300         return sTempRect;
    301     }
    302 
    303     private class MyOutlineProver extends ViewOutlineProvider {
    304 
    305         @Override
    306         public void getOutline(View view, Outline outline) {
    307             if (mEntryAnimationRadiusFactors == null) {
    308                 RectF activeRect = getActiveRect();
    309                 outline.setRoundRect(
    310                         (int) activeRect.left,
    311                         (int) activeRect.top,
    312                         (int) activeRect.right,
    313                         (int) activeRect.bottom,
    314                         mDotRadius
    315                 );
    316             }
    317         }
    318     }
    319 
    320     /**
    321      * Listener for keep running the animation until the final state is reached.
    322      */
    323     private class AnimationCycleListener extends AnimatorListenerAdapter {
    324 
    325         private boolean mCancelled = false;
    326 
    327         @Override
    328         public void onAnimationCancel(Animator animation) {
    329             mCancelled = true;
    330         }
    331 
    332         @Override
    333         public void onAnimationEnd(Animator animation) {
    334             if (!mCancelled) {
    335                 mAnimator = null;
    336                 animateToPosition(mFinalPosition);
    337             }
    338         }
    339     }
    340 }
    341