1 /*
2  * Copyright (C) 2015 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 android.graphics.drawable;
18 
19 import android.animation.Animator;
20 import android.animation.AnimatorListenerAdapter;
21 import android.animation.ObjectAnimator;
22 import android.animation.TimeInterpolator;
23 import android.graphics.Canvas;
24 import android.graphics.CanvasProperty;
25 import android.graphics.Paint;
26 import android.graphics.RecordingCanvas;
27 import android.graphics.Rect;
28 import android.graphics.animation.RenderNodeAnimator;
29 import android.util.FloatProperty;
30 import android.util.MathUtils;
31 import android.view.animation.AnimationUtils;
32 import android.view.animation.LinearInterpolator;
33 import android.view.animation.PathInterpolator;
34 
35 import java.util.ArrayList;
36 
37 /**
38  * Draws a ripple foreground.
39  */
40 class RippleForeground extends RippleComponent {
41     private static final TimeInterpolator LINEAR_INTERPOLATOR = new LinearInterpolator();
42     // Matches R.interpolator.fast_out_slow_in but as we have no context we can't just import that
43     private static final TimeInterpolator DECELERATE_INTERPOLATOR =
44             new PathInterpolator(0.4f, 0f, 0.2f, 1f);
45 
46     // Time it takes for the ripple to expand
47     private static final int RIPPLE_ENTER_DURATION = 225;
48     // Time it takes for the ripple to slide from the touch to the center point
49     private static final int RIPPLE_ORIGIN_DURATION = 225;
50 
51     private static final int OPACITY_ENTER_DURATION = 75;
52     private static final int OPACITY_EXIT_DURATION = 150;
53     private static final int OPACITY_HOLD_DURATION = OPACITY_ENTER_DURATION + 150;
54 
55     // Parent-relative values for starting position.
56     private float mStartingX;
57     private float mStartingY;
58     private float mClampedStartingX;
59     private float mClampedStartingY;
60 
61     // Hardware rendering properties.
62     private CanvasProperty<Paint> mPropPaint;
63     private CanvasProperty<Float> mPropRadius;
64     private CanvasProperty<Float> mPropX;
65     private CanvasProperty<Float> mPropY;
66 
67     // Target values for tween animations.
68     private float mTargetX = 0;
69     private float mTargetY = 0;
70 
71     // Software rendering properties.
72     private float mOpacity = 0;
73 
74     // Values used to tween between the start and end positions.
75     private float mTweenRadius = 0;
76     private float mTweenX = 0;
77     private float mTweenY = 0;
78 
79     /** Whether this ripple has finished its exit animation. */
80     private boolean mHasFinishedExit;
81 
82     /** Whether we can use hardware acceleration for the exit animation. */
83     private boolean mUsingProperties;
84 
85     private long mEnterStartedAtMillis;
86 
87     private ArrayList<RenderNodeAnimator> mPendingHwAnimators = new ArrayList<>();
88     private ArrayList<RenderNodeAnimator> mRunningHwAnimators = new ArrayList<>();
89 
90     private ArrayList<Animator> mRunningSwAnimators = new ArrayList<>();
91 
92     /**
93      * If set, force all ripple animations to not run on RenderThread, even if it would be
94      * available.
95      */
96     private final boolean mForceSoftware;
97 
98     /**
99      * If we have a bound, don't start from 0. Start from 60% of the max out of width and height.
100      */
101     private float mStartRadius = 0;
102 
RippleForeground(RippleDrawable owner, Rect bounds, float startingX, float startingY, boolean forceSoftware)103     public RippleForeground(RippleDrawable owner, Rect bounds, float startingX, float startingY,
104             boolean forceSoftware) {
105         super(owner, bounds);
106 
107         mForceSoftware = forceSoftware;
108         mStartingX = startingX;
109         mStartingY = startingY;
110 
111         // Take 60% of the maximum of the width and height, then divided half to get the radius.
112         mStartRadius = Math.max(bounds.width(), bounds.height()) * 0.3f;
113         clampStartingPosition();
114     }
115 
116     @Override
onTargetRadiusChanged(float targetRadius)117     protected void onTargetRadiusChanged(float targetRadius) {
118         clampStartingPosition();
119         switchToUiThreadAnimation();
120     }
121 
drawSoftware(Canvas c, Paint p)122     private void drawSoftware(Canvas c, Paint p) {
123         final int origAlpha = p.getAlpha();
124         final int alpha = (int) (origAlpha * mOpacity + 0.5f);
125         final float radius = getCurrentRadius();
126         if (alpha > 0 && radius > 0) {
127             final float x = getCurrentX();
128             final float y = getCurrentY();
129             p.setAlpha(alpha);
130             c.drawCircle(x, y, radius, p);
131             p.setAlpha(origAlpha);
132         }
133     }
134 
startPending(RecordingCanvas c)135     private void startPending(RecordingCanvas c) {
136         if (!mPendingHwAnimators.isEmpty()) {
137             for (int i = 0; i < mPendingHwAnimators.size(); i++) {
138                 RenderNodeAnimator animator = mPendingHwAnimators.get(i);
139                 animator.setTarget(c);
140                 animator.start();
141                 mRunningHwAnimators.add(animator);
142             }
143             mPendingHwAnimators.clear();
144         }
145     }
146 
pruneHwFinished()147     private void pruneHwFinished() {
148         if (!mRunningHwAnimators.isEmpty()) {
149             for (int i = mRunningHwAnimators.size() - 1; i >= 0; i--) {
150                 if (!mRunningHwAnimators.get(i).isRunning()) {
151                     mRunningHwAnimators.remove(i);
152                 }
153             }
154         }
155     }
156 
pruneSwFinished()157     private void pruneSwFinished() {
158         if (!mRunningSwAnimators.isEmpty()) {
159             for (int i = mRunningSwAnimators.size() - 1; i >= 0; i--) {
160                 if (!mRunningSwAnimators.get(i).isRunning()) {
161                     mRunningSwAnimators.remove(i);
162                 }
163             }
164         }
165     }
166 
drawHardware(RecordingCanvas c, Paint p)167     private void drawHardware(RecordingCanvas c, Paint p) {
168         startPending(c);
169         pruneHwFinished();
170         if (mPropPaint != null) {
171             mUsingProperties = true;
172             c.drawCircle(mPropX, mPropY, mPropRadius, mPropPaint);
173         } else {
174             mUsingProperties = false;
175             drawSoftware(c, p);
176         }
177     }
178 
179     /**
180      * Returns the maximum bounds of the ripple relative to the ripple center.
181      */
getBounds(Rect bounds)182     public void getBounds(Rect bounds) {
183         final int outerX = (int) mTargetX;
184         final int outerY = (int) mTargetY;
185         final int r = (int) mTargetRadius + 1;
186         bounds.set(outerX - r, outerY - r, outerX + r, outerY + r);
187     }
188 
189     /**
190      * Specifies the starting position relative to the drawable bounds. No-op if
191      * the ripple has already entered.
192      */
move(float x, float y)193     public void move(float x, float y) {
194         mStartingX = x;
195         mStartingY = y;
196 
197         clampStartingPosition();
198     }
199 
200     /**
201      * @return {@code true} if this ripple has finished its exit animation
202      */
hasFinishedExit()203     public boolean hasFinishedExit() {
204         return mHasFinishedExit;
205     }
206 
computeFadeOutDelay()207     private long computeFadeOutDelay() {
208         long timeSinceEnter = AnimationUtils.currentAnimationTimeMillis() - mEnterStartedAtMillis;
209         if (timeSinceEnter > 0 && timeSinceEnter < OPACITY_HOLD_DURATION) {
210             return OPACITY_HOLD_DURATION - timeSinceEnter;
211         }
212         return 0;
213     }
214 
startSoftwareEnter()215     private void startSoftwareEnter() {
216         for (int i = 0; i < mRunningSwAnimators.size(); i++) {
217             mRunningSwAnimators.get(i).cancel();
218         }
219         mRunningSwAnimators.clear();
220 
221         final ObjectAnimator tweenRadius = ObjectAnimator.ofFloat(this, TWEEN_RADIUS, 1);
222         tweenRadius.setDuration(RIPPLE_ENTER_DURATION);
223         tweenRadius.setInterpolator(DECELERATE_INTERPOLATOR);
224         tweenRadius.start();
225         mRunningSwAnimators.add(tweenRadius);
226 
227         final ObjectAnimator tweenOrigin = ObjectAnimator.ofFloat(this, TWEEN_ORIGIN, 1);
228         tweenOrigin.setDuration(RIPPLE_ORIGIN_DURATION);
229         tweenOrigin.setInterpolator(DECELERATE_INTERPOLATOR);
230         tweenOrigin.start();
231         mRunningSwAnimators.add(tweenOrigin);
232 
233         final ObjectAnimator opacity = ObjectAnimator.ofFloat(this, OPACITY, 1);
234         opacity.setDuration(OPACITY_ENTER_DURATION);
235         opacity.setInterpolator(LINEAR_INTERPOLATOR);
236         opacity.start();
237         mRunningSwAnimators.add(opacity);
238     }
239 
startSoftwareExit()240     private void startSoftwareExit() {
241         final ObjectAnimator opacity = ObjectAnimator.ofFloat(this, OPACITY, 0);
242         opacity.setDuration(OPACITY_EXIT_DURATION);
243         opacity.setInterpolator(LINEAR_INTERPOLATOR);
244         opacity.addListener(mAnimationListener);
245         opacity.setStartDelay(computeFadeOutDelay());
246         opacity.start();
247         mRunningSwAnimators.add(opacity);
248     }
249 
startHardwareEnter()250     private void startHardwareEnter() {
251         if (mForceSoftware) { return; }
252         mPropX = CanvasProperty.createFloat(getCurrentX());
253         mPropY = CanvasProperty.createFloat(getCurrentY());
254         mPropRadius = CanvasProperty.createFloat(getCurrentRadius());
255         final Paint paint = mOwner.updateRipplePaint();
256         mPropPaint = CanvasProperty.createPaint(paint);
257 
258         final RenderNodeAnimator radius = new RenderNodeAnimator(mPropRadius, mTargetRadius);
259         radius.setDuration(RIPPLE_ORIGIN_DURATION);
260         radius.setInterpolator(DECELERATE_INTERPOLATOR);
261         mPendingHwAnimators.add(radius);
262 
263         final RenderNodeAnimator x = new RenderNodeAnimator(mPropX, mTargetX);
264         x.setDuration(RIPPLE_ORIGIN_DURATION);
265         x.setInterpolator(DECELERATE_INTERPOLATOR);
266         mPendingHwAnimators.add(x);
267 
268         final RenderNodeAnimator y = new RenderNodeAnimator(mPropY, mTargetY);
269         y.setDuration(RIPPLE_ORIGIN_DURATION);
270         y.setInterpolator(DECELERATE_INTERPOLATOR);
271         mPendingHwAnimators.add(y);
272 
273         final RenderNodeAnimator opacity = new RenderNodeAnimator(mPropPaint,
274                 RenderNodeAnimator.PAINT_ALPHA, paint.getAlpha());
275         opacity.setDuration(OPACITY_ENTER_DURATION);
276         opacity.setInterpolator(LINEAR_INTERPOLATOR);
277         opacity.setStartValue(0);
278         mPendingHwAnimators.add(opacity);
279 
280         invalidateSelf();
281     }
282 
startHardwareExit()283     private void startHardwareExit() {
284         // Only run a hardware exit if we had a hardware enter to continue from
285         if (mForceSoftware || mPropPaint == null) return;
286 
287         final RenderNodeAnimator opacity = new RenderNodeAnimator(mPropPaint,
288                 RenderNodeAnimator.PAINT_ALPHA, 0);
289         opacity.setDuration(OPACITY_EXIT_DURATION);
290         opacity.setInterpolator(LINEAR_INTERPOLATOR);
291         opacity.addListener(mAnimationListener);
292         opacity.setStartDelay(computeFadeOutDelay());
293         opacity.setStartValue(mOwner.updateRipplePaint().getAlpha());
294         mPendingHwAnimators.add(opacity);
295         invalidateSelf();
296     }
297 
298     /**
299      * Starts a ripple enter animation.
300      */
enter()301     public final void enter() {
302         mEnterStartedAtMillis = AnimationUtils.currentAnimationTimeMillis();
303         startSoftwareEnter();
304         startHardwareEnter();
305     }
306 
307     /**
308      * Starts a ripple exit animation.
309      */
exit()310     public final void exit() {
311         startSoftwareExit();
312         startHardwareExit();
313     }
314 
getCurrentX()315     private float getCurrentX() {
316         return MathUtils.lerp(mClampedStartingX - mBounds.exactCenterX(), mTargetX, mTweenX);
317     }
318 
getCurrentY()319     private float getCurrentY() {
320         return MathUtils.lerp(mClampedStartingY - mBounds.exactCenterY(), mTargetY, mTweenY);
321     }
322 
getCurrentRadius()323     private float getCurrentRadius() {
324         return MathUtils.lerp(mStartRadius, mTargetRadius, mTweenRadius);
325     }
326 
327     /**
328      * Draws the ripple to the canvas, inheriting the paint's color and alpha
329      * properties.
330      *
331      * @param c the canvas to which the ripple should be drawn
332      * @param p the paint used to draw the ripple
333      */
draw(Canvas c, Paint p)334     public void draw(Canvas c, Paint p) {
335         final boolean hasDisplayListCanvas = !mForceSoftware && c instanceof RecordingCanvas;
336 
337         pruneSwFinished();
338         if (hasDisplayListCanvas) {
339             final RecordingCanvas hw = (RecordingCanvas) c;
340             drawHardware(hw, p);
341         } else {
342             drawSoftware(c, p);
343         }
344     }
345 
346     /**
347      * Clamps the starting position to fit within the ripple bounds.
348      */
clampStartingPosition()349     private void clampStartingPosition() {
350         final float cX = mBounds.exactCenterX();
351         final float cY = mBounds.exactCenterY();
352         final float dX = mStartingX - cX;
353         final float dY = mStartingY - cY;
354         final float r = mTargetRadius - mStartRadius;
355         if (dX * dX + dY * dY > r * r) {
356             // Point is outside the circle, clamp to the perimeter.
357             final double angle = Math.atan2(dY, dX);
358             mClampedStartingX = cX + (float) (Math.cos(angle) * r);
359             mClampedStartingY = cY + (float) (Math.sin(angle) * r);
360         } else {
361             mClampedStartingX = mStartingX;
362             mClampedStartingY = mStartingY;
363         }
364     }
365 
366     /**
367      * Ends all animations, jumping values to the end state.
368      */
end()369     public void end() {
370         for (int i = 0; i < mRunningSwAnimators.size(); i++) {
371             mRunningSwAnimators.get(i).end();
372         }
373         mRunningSwAnimators.clear();
374         for (int i = 0; i < mRunningHwAnimators.size(); i++) {
375             mRunningHwAnimators.get(i).end();
376         }
377         mRunningHwAnimators.clear();
378     }
379 
onAnimationPropertyChanged()380     private void onAnimationPropertyChanged() {
381         if (!mUsingProperties) {
382             invalidateSelf();
383         }
384     }
385 
clearHwProps()386     private void clearHwProps() {
387         mPropPaint = null;
388         mPropRadius = null;
389         mPropX = null;
390         mPropY = null;
391         mUsingProperties = false;
392     }
393 
394     private final AnimatorListenerAdapter mAnimationListener = new AnimatorListenerAdapter() {
395         @Override
396         public void onAnimationEnd(Animator animator) {
397             mHasFinishedExit = true;
398             pruneHwFinished();
399             pruneSwFinished();
400 
401             if (mRunningHwAnimators.isEmpty()) {
402                 clearHwProps();
403             }
404         }
405     };
406 
switchToUiThreadAnimation()407     private void switchToUiThreadAnimation() {
408         for (int i = 0; i < mRunningHwAnimators.size(); i++) {
409             Animator animator = mRunningHwAnimators.get(i);
410             animator.removeListener(mAnimationListener);
411             animator.end();
412         }
413         mRunningHwAnimators.clear();
414         clearHwProps();
415         invalidateSelf();
416     }
417 
418     /**
419      * Property for animating radius between its initial and target values.
420      */
421     private static final FloatProperty<RippleForeground> TWEEN_RADIUS =
422             new FloatProperty<RippleForeground>("tweenRadius") {
423         @Override
424         public void setValue(RippleForeground object, float value) {
425             object.mTweenRadius = value;
426             object.onAnimationPropertyChanged();
427         }
428 
429         @Override
430         public Float get(RippleForeground object) {
431             return object.mTweenRadius;
432         }
433     };
434 
435     /**
436      * Property for animating origin between its initial and target values.
437      */
438     private static final FloatProperty<RippleForeground> TWEEN_ORIGIN =
439             new FloatProperty<RippleForeground>("tweenOrigin") {
440         @Override
441         public void setValue(RippleForeground object, float value) {
442             object.mTweenX = value;
443             object.mTweenY = value;
444             object.onAnimationPropertyChanged();
445         }
446 
447         @Override
448         public Float get(RippleForeground object) {
449             return object.mTweenX;
450         }
451     };
452 
453     /**
454      * Property for animating opacity between 0 and its target value.
455      */
456     private static final FloatProperty<RippleForeground> OPACITY =
457             new FloatProperty<RippleForeground>("opacity") {
458         @Override
459         public void setValue(RippleForeground object, float value) {
460             object.mOpacity = value;
461             object.onAnimationPropertyChanged();
462         }
463 
464         @Override
465         public Float get(RippleForeground object) {
466             return object.mOpacity;
467         }
468     };
469 }
470