• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 com.android.messaging.ui.animation;
18 
19 import android.animation.TypeEvaluator;
20 import android.app.Activity;
21 import android.graphics.Canvas;
22 import android.graphics.Color;
23 import android.graphics.Rect;
24 import android.view.Gravity;
25 import android.view.View;
26 import android.view.ViewGroup;
27 import android.view.animation.Animation;
28 import android.view.animation.Transformation;
29 import android.widget.PopupWindow;
30 
31 import com.android.messaging.util.LogUtil;
32 import com.android.messaging.util.ThreadUtil;
33 import com.android.messaging.util.UiUtils;
34 
35 /**
36  * Animates viewToAnimate from startRect to the place where it is in the layout,  viewToAnimate
37  * should be in its final destination location before startAfterLayoutComplete is called.
38  * viewToAnimate will be drawn scaled and offset in a popupWindow.
39  * This class handles the case where the viewToAnimate moves during the animation
40  */
41 public class PopupTransitionAnimation extends Animation {
42     /** The view we're animating */
43     private final View mViewToAnimate;
44 
45     /** The rect to start the slide in animation from */
46     private final Rect mStartRect;
47 
48     /** The rect of the currently animated view */
49     private Rect mCurrentRect;
50 
51     /** The rect that we're animating to.  This can change during the animation */
52     private final Rect mDestRect;
53 
54     /** The bounds of the popup in window coordinates.  Does not include notification bar */
55     private final Rect mPopupRect;
56 
57     /** The bounds of the action bar in window coordinates.  We clip the popup to below this */
58     private final Rect mActionBarRect;
59 
60     /** Interpolates between the start and end rect for every animation tick */
61     private final TypeEvaluator<Rect> mRectEvaluator;
62 
63     /** The popup window that holds contains the animating view */
64     private PopupWindow mPopupWindow;
65 
66     /** The layout root for the popup which is where the animated view is rendered */
67     private View mPopupRoot;
68 
69     /** The action bar's view */
70     private final View mActionBarView;
71 
72     private Runnable mOnStartCallback;
73     private Runnable mOnStopCallback;
74 
PopupTransitionAnimation(final Rect startRect, final View viewToAnimate)75     public PopupTransitionAnimation(final Rect startRect, final View viewToAnimate) {
76         mViewToAnimate = viewToAnimate;
77         mStartRect = startRect;
78         mCurrentRect = new Rect(mStartRect);
79         mDestRect = new Rect();
80         mPopupRect = new Rect();
81         mActionBarRect = new Rect();
82         final Activity activity = (Activity) viewToAnimate.getRootView().getContext();
83         mActionBarView = activity.getWindow().getDecorView().findViewById(
84                 android.support.v7.appcompat.R.id.action_bar);
85         mRectEvaluator = RectEvaluatorCompat.create();
86         setDuration(UiUtils.MEDIAPICKER_TRANSITION_DURATION);
87         setInterpolator(UiUtils.DEFAULT_INTERPOLATOR);
88         setAnimationListener(new AnimationListener() {
89             @Override
90             public void onAnimationStart(final Animation animation) {
91                 if (mOnStartCallback != null) {
92                     mOnStartCallback.run();
93                 }
94                 mEvents.append("oAS,");
95             }
96 
97             @Override
98             public void onAnimationEnd(final Animation animation) {
99                 if (mOnStopCallback != null) {
100                     mOnStopCallback.run();
101                 }
102                 dismiss();
103                 mEvents.append("oAE,");
104             }
105 
106             @Override
107             public void onAnimationRepeat(final Animation animation) {
108             }
109         });
110     }
111 
112     private final StringBuilder mEvents = new StringBuilder();
113     private final Runnable mCleanupRunnable = new Runnable() {
114         @Override
115         public void run() {
116             LogUtil.w(LogUtil.BUGLE_TAG, "PopupTransitionAnimation: " + mEvents);
117         }
118     };
119 
120     /**
121      * Ensures the animation is ready before starting the animation.
122      * viewToAnimate must first be layed out so we know where we will animate to
123      */
startAfterLayoutComplete()124     public void startAfterLayoutComplete() {
125         // We want layout to occur, and then we immediately animate it in, so hide it initially to
126         // reduce jank on the first frame
127         mViewToAnimate.setVisibility(View.INVISIBLE);
128         mViewToAnimate.setAlpha(0);
129 
130         final Runnable startAnimation = new Runnable() {
131             boolean mRunComplete = false;
132             boolean mFirstTry = true;
133 
134             @Override
135             public void run() {
136                 if (mRunComplete) {
137                     return;
138                 }
139 
140                 mViewToAnimate.getGlobalVisibleRect(mDestRect);
141                 // In Android views which are visible but haven't computed their size yet have a
142                 // size of 1x1 because anything with a size of 0x0 is considered hidden.  We can't
143                 // start the animation until after the size is greater than 1x1
144                 if (mDestRect.width() <= 1 || mDestRect.height() <= 1) {
145                     // Layout hasn't occurred yet
146                     if (!mFirstTry) {
147                         // Give up if this is not the first try, since layout change still doesn't
148                         // yield a size for the view. This is likely because the media picker is
149                         // full screen so there's no space left for the animated view. We give up
150                         // on animation, but need to make sure the view that was initially
151                         // hidden is re-shown.
152                         mViewToAnimate.setAlpha(1);
153                         mViewToAnimate.setVisibility(View.VISIBLE);
154                     } else {
155                         mFirstTry = false;
156                         UiUtils.doOnceAfterLayoutChange(mViewToAnimate, this);
157                     }
158                     return;
159                 }
160 
161                 mRunComplete = true;
162                 mViewToAnimate.startAnimation(PopupTransitionAnimation.this);
163                 mViewToAnimate.invalidate();
164                 // http://b/20856505: The PopupWindow sometimes does not get dismissed.
165                 ThreadUtil.getMainThreadHandler().postDelayed(mCleanupRunnable, getDuration() * 2);
166             }
167         };
168 
169         startAnimation.run();
170     }
171 
setOnStartCallback(final Runnable onStart)172     public PopupTransitionAnimation setOnStartCallback(final Runnable onStart) {
173         mOnStartCallback = onStart;
174         return this;
175     }
176 
setOnStopCallback(final Runnable onStop)177     public PopupTransitionAnimation setOnStopCallback(final Runnable onStop) {
178         mOnStopCallback = onStop;
179         return this;
180     }
181 
182     @Override
applyTransformation(final float interpolatedTime, final Transformation t)183     protected void applyTransformation(final float interpolatedTime, final Transformation t) {
184         if (mPopupWindow == null) {
185             initPopupWindow();
186         }
187         // Update mDestRect as it may have moved during the animation
188         mPopupRect.set(UiUtils.getMeasuredBoundsOnScreen(mPopupRoot));
189         mActionBarRect.set(UiUtils.getMeasuredBoundsOnScreen(mActionBarView));
190         computeDestRect();
191 
192         // Update currentRect to the new animated coordinates, and request mPopupRoot to redraw
193         // itself at the new coordinates
194         mCurrentRect = mRectEvaluator.evaluate(interpolatedTime, mStartRect, mDestRect);
195         mPopupRoot.invalidate();
196 
197         if (interpolatedTime >= 0.98) {
198             mEvents.append("aT").append(interpolatedTime).append(',');
199         }
200         if (interpolatedTime == 1) {
201             dismiss();
202         }
203     }
204 
dismiss()205     private void dismiss() {
206         mEvents.append("d,");
207         mViewToAnimate.setAlpha(1);
208         mViewToAnimate.setVisibility(View.VISIBLE);
209         // Delay dismissing the popup window to let mViewToAnimate draw under it and reduce the
210         // flash
211         ThreadUtil.getMainThreadHandler().post(new Runnable() {
212             @Override
213             public void run() {
214                 try {
215                     mPopupWindow.dismiss();
216                 } catch (IllegalArgumentException e) {
217                     // PopupWindow.dismiss() will fire an IllegalArgumentException if the activity
218                     // has already ended while we were animating
219                 }
220                 ThreadUtil.getMainThreadHandler().removeCallbacks(mCleanupRunnable);
221             }
222         });
223     }
224 
225     @Override
willChangeBounds()226     public boolean willChangeBounds() {
227         return false;
228     }
229 
230     /**
231      * Computes mDestRect (the position in window space of the placeholder view that we should
232      * animate to).  Some frames during the animation fail to compute getGlobalVisibleRect, so use
233      * the last known values in that case
234      */
computeDestRect()235     private void computeDestRect() {
236         final int prevTop = mDestRect.top;
237         final int prevLeft = mDestRect.left;
238         final int prevRight = mDestRect.right;
239         final int prevBottom = mDestRect.bottom;
240 
241         if (!getViewScreenMeasureRect(mViewToAnimate, mDestRect)) {
242             mDestRect.top = prevTop;
243             mDestRect.left = prevLeft;
244             mDestRect.bottom = prevBottom;
245             mDestRect.right = prevRight;
246         }
247     }
248 
249     /**
250      * Sets up the PopupWindow that the view will animate in.  Animating the size and position of a
251      * popup can be choppy, so instead we make the popup fill the entire space of the screen, and
252      * animate the position of viewToAnimate within the popup using a Transformation
253      */
initPopupWindow()254     private void initPopupWindow() {
255         mPopupRoot = new View(mViewToAnimate.getContext()) {
256             @Override
257             protected void onDraw(final Canvas canvas) {
258                 canvas.save();
259                 canvas.clipRect(getLeft(), mActionBarRect.bottom - mPopupRect.top, getRight(),
260                         getBottom());
261                 canvas.drawColor(Color.TRANSPARENT);
262                 final float previousAlpha = mViewToAnimate.getAlpha();
263                 mViewToAnimate.setAlpha(1);
264                 // The view's global position includes the notification bar height, but
265                 // the popup window may or may not cover the notification bar (depending on screen
266                 // rotation, IME status etc.), so we need to compensate for this difference by
267                 // offseting vertically.
268                 canvas.translate(mCurrentRect.left, mCurrentRect.top - mPopupRect.top);
269 
270                 final float viewWidth = mViewToAnimate.getWidth();
271                 final float viewHeight = mViewToAnimate.getHeight();
272                 if (viewWidth > 0 && viewHeight > 0) {
273                     canvas.scale(mCurrentRect.width() / viewWidth,
274                             mCurrentRect.height() / viewHeight);
275                 }
276                 canvas.clipRect(0, 0, mCurrentRect.width(), mCurrentRect.height());
277                 if (!mPopupRect.isEmpty()) {
278                     // HACK: Layout is unstable until mPopupRect is non-empty.
279                     mViewToAnimate.draw(canvas);
280                 }
281                 mViewToAnimate.setAlpha(previousAlpha);
282                 canvas.restore();
283             }
284         };
285         mPopupWindow = new PopupWindow(mViewToAnimate.getContext());
286         mPopupWindow.setBackgroundDrawable(null);
287         mPopupWindow.setContentView(mPopupRoot);
288         mPopupWindow.setWidth(ViewGroup.LayoutParams.MATCH_PARENT);
289         mPopupWindow.setHeight(ViewGroup.LayoutParams.MATCH_PARENT);
290         mPopupWindow.setTouchable(false);
291         // We must pass a non-zero value for the y offset, or else the system resets the status bar
292         // color to black (M only) during the animation. The actual position of the window (and
293         // the animated view inside it) are still correct, regardless of what we pass for the y
294         // parameter (e.g. 1 and 100 both work). Not entirely sure why this works.
295         mPopupWindow.showAtLocation(mViewToAnimate, Gravity.TOP, 0, 1);
296     }
297 
getViewScreenMeasureRect(final View view, final Rect outRect)298     private static boolean getViewScreenMeasureRect(final View view, final Rect outRect) {
299         outRect.set(UiUtils.getMeasuredBoundsOnScreen(view));
300         return !outRect.isEmpty();
301     }
302 }
303