1 /*
2  * Copyright (C) 2014 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.systemui.statusbar.phone;
18 
19 import com.android.systemui.R;
20 
21 import android.animation.Animator;
22 import android.animation.AnimatorListenerAdapter;
23 import android.animation.AnimatorSet;
24 import android.animation.ValueAnimator;
25 import android.content.Context;
26 import android.content.res.Resources;
27 import android.graphics.Canvas;
28 import android.graphics.Color;
29 import android.graphics.ColorFilter;
30 import android.graphics.Paint;
31 import android.graphics.PixelFormat;
32 import android.graphics.Rect;
33 import android.graphics.drawable.Drawable;
34 import android.view.animation.AccelerateDecelerateInterpolator;
35 import android.view.animation.AnimationUtils;
36 import android.view.animation.Interpolator;
37 
38 public class TrustDrawable extends Drawable {
39 
40     private static final long ENTERING_FROM_UNSET_START_DELAY = 200;
41     private static final long VISIBLE_DURATION = 1000;
42     private static final long EXIT_DURATION = 500;
43     private static final long ENTER_DURATION = 500;
44 
45     private static final int ALPHA_VISIBLE_MIN = 0x26;
46     private static final int ALPHA_VISIBLE_MAX = 0x4c;
47 
48     private static final int STATE_UNSET = -1;
49     private static final int STATE_GONE = 0;
50     private static final int STATE_ENTERING = 1;
51     private static final int STATE_VISIBLE = 2;
52     private static final int STATE_EXITING = 3;
53 
54     private int mAlpha;
55     private boolean mAnimating;
56 
57     private int mCurAlpha;
58     private float mCurInnerRadius;
59     private Animator mCurAnimator;
60     private int mState = STATE_UNSET;
61     private Paint mPaint;
62     private boolean mTrustManaged;
63 
64     private final float mInnerRadiusVisibleMin;
65     private final float mInnerRadiusVisibleMax;
66     private final float mInnerRadiusExit;
67     private final float mInnerRadiusEnter;
68     private final float mThickness;
69 
70     private final Animator mVisibleAnimator;
71 
72     private final Interpolator mLinearOutSlowInInterpolator;
73     private final Interpolator mFastOutSlowInInterpolator;
74     private final Interpolator mAccelerateDecelerateInterpolator;
75 
TrustDrawable(Context context)76     public TrustDrawable(Context context) {
77         Resources r = context.getResources();
78         mInnerRadiusVisibleMin = r.getDimension(R.dimen.trust_circle_inner_radius_visible_min);
79         mInnerRadiusVisibleMax = r.getDimension(R.dimen.trust_circle_inner_radius_visible_max);
80         mInnerRadiusExit = r.getDimension(R.dimen.trust_circle_inner_radius_exit);
81         mInnerRadiusEnter = r.getDimension(R.dimen.trust_circle_inner_radius_enter);
82         mThickness = r.getDimension(R.dimen.trust_circle_thickness);
83 
84         mCurInnerRadius = mInnerRadiusEnter;
85 
86         mLinearOutSlowInInterpolator = AnimationUtils.loadInterpolator(
87                 context, android.R.interpolator.linear_out_slow_in);
88         mFastOutSlowInInterpolator = AnimationUtils.loadInterpolator(
89                 context, android.R.interpolator.fast_out_slow_in);
90         mAccelerateDecelerateInterpolator = new AccelerateDecelerateInterpolator();
91 
92         mVisibleAnimator = makeVisibleAnimator();
93 
94         mPaint = new Paint();
95         mPaint.setStyle(Paint.Style.STROKE);
96         mPaint.setColor(Color.WHITE);
97         mPaint.setAntiAlias(true);
98         mPaint.setStrokeWidth(mThickness);
99     }
100 
101     @Override
draw(Canvas canvas)102     public void draw(Canvas canvas) {
103         int newAlpha = (mCurAlpha * mAlpha) / 256;
104         if (newAlpha == 0) {
105             return;
106         }
107         final Rect r = getBounds();
108         mPaint.setAlpha(newAlpha);
109         canvas.drawCircle(r.exactCenterX(), r.exactCenterY(), mCurInnerRadius, mPaint);
110     }
111 
112     @Override
setAlpha(int alpha)113     public void setAlpha(int alpha) {
114         mAlpha = alpha;
115     }
116 
117     @Override
getAlpha()118     public int getAlpha() {
119         return mAlpha;
120     }
121 
122     @Override
setColorFilter(ColorFilter cf)123     public void setColorFilter(ColorFilter cf) {
124         throw new UnsupportedOperationException("not implemented");
125     }
126 
127     @Override
getOpacity()128     public int getOpacity() {
129         return PixelFormat.TRANSLUCENT;
130     }
131 
start()132     public void start() {
133         if (!mAnimating) {
134             mAnimating = true;
135             updateState(true);
136             invalidateSelf();
137         }
138     }
139 
stop()140     public void stop() {
141         if (mAnimating) {
142             mAnimating = false;
143             if (mCurAnimator != null) {
144                 mCurAnimator.cancel();
145                 mCurAnimator = null;
146             }
147             mState = STATE_UNSET;
148             mCurAlpha = 0;
149             mCurInnerRadius = mInnerRadiusEnter;
150             invalidateSelf();
151         }
152     }
153 
setTrustManaged(boolean trustManaged)154     public void setTrustManaged(boolean trustManaged) {
155         if (trustManaged == mTrustManaged && mState != STATE_UNSET) return;
156         mTrustManaged = trustManaged;
157         updateState(true);
158     }
159 
updateState(boolean allowTransientState)160     private void updateState(boolean allowTransientState) {
161         if (!mAnimating) {
162             return;
163         }
164 
165         int nextState = mState;
166         if (mState == STATE_UNSET) {
167             nextState = mTrustManaged ? STATE_ENTERING : STATE_GONE;
168         } else if (mState == STATE_GONE) {
169             if (mTrustManaged) nextState = STATE_ENTERING;
170         } else if (mState == STATE_ENTERING) {
171             if (!mTrustManaged) nextState = STATE_EXITING;
172         } else if (mState == STATE_VISIBLE) {
173             if (!mTrustManaged) nextState = STATE_EXITING;
174         } else if (mState == STATE_EXITING) {
175             if (mTrustManaged) nextState = STATE_ENTERING;
176         }
177         if (!allowTransientState) {
178             if (nextState == STATE_ENTERING) nextState = STATE_VISIBLE;
179             if (nextState == STATE_EXITING) nextState = STATE_GONE;
180         }
181 
182         if (nextState != mState) {
183             if (mCurAnimator != null) {
184                 mCurAnimator.cancel();
185                 mCurAnimator = null;
186             }
187 
188             if (nextState == STATE_GONE) {
189                 mCurAlpha = 0;
190                 mCurInnerRadius = mInnerRadiusEnter;
191             } else if (nextState == STATE_ENTERING) {
192                 mCurAnimator = makeEnterAnimator(mCurInnerRadius, mCurAlpha);
193                 if (mState == STATE_UNSET) {
194                     mCurAnimator.setStartDelay(ENTERING_FROM_UNSET_START_DELAY);
195                 }
196             } else if (nextState == STATE_VISIBLE) {
197                 mCurAlpha = ALPHA_VISIBLE_MAX;
198                 mCurInnerRadius = mInnerRadiusVisibleMax;
199                 mCurAnimator = mVisibleAnimator;
200             } else if (nextState == STATE_EXITING) {
201                 mCurAnimator = makeExitAnimator(mCurInnerRadius, mCurAlpha);
202             }
203 
204             mState = nextState;
205             if (mCurAnimator != null) {
206                 mCurAnimator.start();
207             }
208             invalidateSelf();
209         }
210     }
211 
makeVisibleAnimator()212     private Animator makeVisibleAnimator() {
213         return makeAnimators(mInnerRadiusVisibleMax, mInnerRadiusVisibleMin,
214                 ALPHA_VISIBLE_MAX, ALPHA_VISIBLE_MIN, VISIBLE_DURATION,
215                 mAccelerateDecelerateInterpolator,
216                 true /* repeating */, false /* stateUpdateListener */);
217     }
218 
makeEnterAnimator(float radius, int alpha)219     private Animator makeEnterAnimator(float radius, int alpha) {
220         return makeAnimators(radius, mInnerRadiusVisibleMax,
221                 alpha, ALPHA_VISIBLE_MAX, ENTER_DURATION, mLinearOutSlowInInterpolator,
222                 false /* repeating */, true /* stateUpdateListener */);
223     }
224 
makeExitAnimator(float radius, int alpha)225     private Animator makeExitAnimator(float radius, int alpha) {
226         return makeAnimators(radius, mInnerRadiusExit,
227                 alpha, 0, EXIT_DURATION, mFastOutSlowInInterpolator,
228                 false /* repeating */, true /* stateUpdateListener */);
229     }
230 
makeAnimators(float startRadius, float endRadius, int startAlpha, int endAlpha, long duration, Interpolator interpolator, boolean repeating, boolean stateUpdateListener)231     private Animator makeAnimators(float startRadius, float endRadius,
232             int startAlpha, int endAlpha, long duration, Interpolator interpolator,
233             boolean repeating, boolean stateUpdateListener) {
234         ValueAnimator alphaAnimator = configureAnimator(
235                 ValueAnimator.ofInt(startAlpha, endAlpha),
236                 duration, mAlphaUpdateListener, interpolator, repeating);
237         ValueAnimator sizeAnimator = configureAnimator(
238                 ValueAnimator.ofFloat(startRadius, endRadius),
239                 duration, mRadiusUpdateListener, interpolator, repeating);
240 
241         AnimatorSet set = new AnimatorSet();
242         set.playTogether(alphaAnimator, sizeAnimator);
243         if (stateUpdateListener) {
244             set.addListener(new StateUpdateAnimatorListener());
245         }
246         return set;
247     }
248 
configureAnimator(ValueAnimator animator, long duration, ValueAnimator.AnimatorUpdateListener updateListener, Interpolator interpolator, boolean repeating)249     private ValueAnimator configureAnimator(ValueAnimator animator, long duration,
250             ValueAnimator.AnimatorUpdateListener updateListener, Interpolator interpolator,
251             boolean repeating) {
252         animator.setDuration(duration);
253         animator.addUpdateListener(updateListener);
254         animator.setInterpolator(interpolator);
255         if (repeating) {
256             animator.setRepeatCount(ValueAnimator.INFINITE);
257             animator.setRepeatMode(ValueAnimator.REVERSE);
258         }
259         return animator;
260     }
261 
262     private final ValueAnimator.AnimatorUpdateListener mAlphaUpdateListener =
263             new ValueAnimator.AnimatorUpdateListener() {
264         @Override
265         public void onAnimationUpdate(ValueAnimator animation) {
266             mCurAlpha = (int) animation.getAnimatedValue();
267             invalidateSelf();
268         }
269     };
270 
271     private final ValueAnimator.AnimatorUpdateListener mRadiusUpdateListener =
272             new ValueAnimator.AnimatorUpdateListener() {
273         @Override
274         public void onAnimationUpdate(ValueAnimator animation) {
275             mCurInnerRadius = (float) animation.getAnimatedValue();
276             invalidateSelf();
277         }
278     };
279 
280     private class StateUpdateAnimatorListener extends AnimatorListenerAdapter {
281         boolean mCancelled;
282 
283         @Override
onAnimationStart(Animator animation)284         public void onAnimationStart(Animator animation) {
285             mCancelled = false;
286         }
287 
288         @Override
onAnimationCancel(Animator animation)289         public void onAnimationCancel(Animator animation) {
290             mCancelled = true;
291         }
292 
293         @Override
onAnimationEnd(Animator animation)294         public void onAnimationEnd(Animator animation) {
295             if (!mCancelled) {
296                 updateState(false);
297             }
298         }
299     }
300 }
301