| /* |
| * Copyright (C) 2015 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| package com.android.messaging.ui.animation; |
| |
| import android.animation.TypeEvaluator; |
| import android.app.Activity; |
| import android.graphics.Canvas; |
| import android.graphics.Color; |
| import android.graphics.Rect; |
| import android.view.Gravity; |
| import android.view.View; |
| import android.view.ViewGroup; |
| import android.view.animation.Animation; |
| import android.view.animation.Transformation; |
| import android.widget.PopupWindow; |
| |
| import com.android.messaging.util.LogUtil; |
| import com.android.messaging.util.ThreadUtil; |
| import com.android.messaging.util.UiUtils; |
| |
| /** |
| * Animates viewToAnimate from startRect to the place where it is in the layout, viewToAnimate |
| * should be in its final destination location before startAfterLayoutComplete is called. |
| * viewToAnimate will be drawn scaled and offset in a popupWindow. |
| * This class handles the case where the viewToAnimate moves during the animation |
| */ |
| public class PopupTransitionAnimation extends Animation { |
| /** The view we're animating */ |
| private final View mViewToAnimate; |
| |
| /** The rect to start the slide in animation from */ |
| private final Rect mStartRect; |
| |
| /** The rect of the currently animated view */ |
| private Rect mCurrentRect; |
| |
| /** The rect that we're animating to. This can change during the animation */ |
| private final Rect mDestRect; |
| |
| /** The bounds of the popup in window coordinates. Does not include notification bar */ |
| private final Rect mPopupRect; |
| |
| /** The bounds of the action bar in window coordinates. We clip the popup to below this */ |
| private final Rect mActionBarRect; |
| |
| /** Interpolates between the start and end rect for every animation tick */ |
| private final TypeEvaluator<Rect> mRectEvaluator; |
| |
| /** The popup window that holds contains the animating view */ |
| private PopupWindow mPopupWindow; |
| |
| /** The layout root for the popup which is where the animated view is rendered */ |
| private View mPopupRoot; |
| |
| /** The action bar's view */ |
| private final View mActionBarView; |
| |
| private Runnable mOnStartCallback; |
| private Runnable mOnStopCallback; |
| |
| public PopupTransitionAnimation(final Rect startRect, final View viewToAnimate) { |
| mViewToAnimate = viewToAnimate; |
| mStartRect = startRect; |
| mCurrentRect = new Rect(mStartRect); |
| mDestRect = new Rect(); |
| mPopupRect = new Rect(); |
| mActionBarRect = new Rect(); |
| mActionBarView = viewToAnimate.getRootView().findViewById( |
| android.support.v7.appcompat.R.id.action_bar); |
| mRectEvaluator = RectEvaluatorCompat.create(); |
| setDuration(UiUtils.MEDIAPICKER_TRANSITION_DURATION); |
| setInterpolator(UiUtils.DEFAULT_INTERPOLATOR); |
| setAnimationListener(new AnimationListener() { |
| @Override |
| public void onAnimationStart(final Animation animation) { |
| if (mOnStartCallback != null) { |
| mOnStartCallback.run(); |
| } |
| mEvents.append("oAS,"); |
| } |
| |
| @Override |
| public void onAnimationEnd(final Animation animation) { |
| if (mOnStopCallback != null) { |
| mOnStopCallback.run(); |
| } |
| dismiss(); |
| mEvents.append("oAE,"); |
| } |
| |
| @Override |
| public void onAnimationRepeat(final Animation animation) { |
| } |
| }); |
| } |
| |
| private final StringBuilder mEvents = new StringBuilder(); |
| private final Runnable mCleanupRunnable = new Runnable() { |
| @Override |
| public void run() { |
| LogUtil.w(LogUtil.BUGLE_TAG, "PopupTransitionAnimation: " + mEvents); |
| } |
| }; |
| |
| /** |
| * Ensures the animation is ready before starting the animation. |
| * viewToAnimate must first be layed out so we know where we will animate to |
| */ |
| public void startAfterLayoutComplete() { |
| // We want layout to occur, and then we immediately animate it in, so hide it initially to |
| // reduce jank on the first frame |
| mViewToAnimate.setVisibility(View.INVISIBLE); |
| mViewToAnimate.setAlpha(0); |
| |
| final Runnable startAnimation = new Runnable() { |
| boolean mRunComplete = false; |
| boolean mFirstTry = true; |
| |
| @Override |
| public void run() { |
| if (mRunComplete) { |
| return; |
| } |
| |
| mViewToAnimate.getGlobalVisibleRect(mDestRect); |
| // In Android views which are visible but haven't computed their size yet have a |
| // size of 1x1 because anything with a size of 0x0 is considered hidden. We can't |
| // start the animation until after the size is greater than 1x1 |
| if (mDestRect.width() <= 1 || mDestRect.height() <= 1) { |
| // Layout hasn't occurred yet |
| if (!mFirstTry) { |
| // Give up if this is not the first try, since layout change still doesn't |
| // yield a size for the view. This is likely because the media picker is |
| // full screen so there's no space left for the animated view. We give up |
| // on animation, but need to make sure the view that was initially |
| // hidden is re-shown. |
| mViewToAnimate.setAlpha(1); |
| mViewToAnimate.setVisibility(View.VISIBLE); |
| } else { |
| mFirstTry = false; |
| UiUtils.doOnceAfterLayoutChange(mViewToAnimate, this); |
| } |
| return; |
| } |
| |
| mRunComplete = true; |
| mViewToAnimate.startAnimation(PopupTransitionAnimation.this); |
| mViewToAnimate.invalidate(); |
| // http://b/20856505: The PopupWindow sometimes does not get dismissed. |
| ThreadUtil.getMainThreadHandler().postDelayed(mCleanupRunnable, getDuration() * 2); |
| } |
| }; |
| |
| startAnimation.run(); |
| } |
| |
| public PopupTransitionAnimation setOnStartCallback(final Runnable onStart) { |
| mOnStartCallback = onStart; |
| return this; |
| } |
| |
| public PopupTransitionAnimation setOnStopCallback(final Runnable onStop) { |
| mOnStopCallback = onStop; |
| return this; |
| } |
| |
| @Override |
| protected void applyTransformation(final float interpolatedTime, final Transformation t) { |
| if (mPopupWindow == null) { |
| initPopupWindow(); |
| } |
| // Update mDestRect as it may have moved during the animation |
| mPopupRect.set(UiUtils.getMeasuredBoundsOnScreen(mPopupRoot)); |
| mActionBarRect.set(UiUtils.getMeasuredBoundsOnScreen(mActionBarView)); |
| computeDestRect(); |
| |
| // Update currentRect to the new animated coordinates, and request mPopupRoot to redraw |
| // itself at the new coordinates |
| mCurrentRect = mRectEvaluator.evaluate(interpolatedTime, mStartRect, mDestRect); |
| mPopupRoot.invalidate(); |
| |
| if (interpolatedTime >= 0.98) { |
| mEvents.append("aT").append(interpolatedTime).append(','); |
| } |
| if (interpolatedTime == 1) { |
| dismiss(); |
| } |
| } |
| |
| private void dismiss() { |
| mEvents.append("d,"); |
| mViewToAnimate.setAlpha(1); |
| mViewToAnimate.setVisibility(View.VISIBLE); |
| // Delay dismissing the popup window to let mViewToAnimate draw under it and reduce the |
| // flash |
| ThreadUtil.getMainThreadHandler().post(new Runnable() { |
| @Override |
| public void run() { |
| try { |
| mPopupWindow.dismiss(); |
| } catch (IllegalArgumentException e) { |
| // PopupWindow.dismiss() will fire an IllegalArgumentException if the activity |
| // has already ended while we were animating |
| } |
| ThreadUtil.getMainThreadHandler().removeCallbacks(mCleanupRunnable); |
| } |
| }); |
| } |
| |
| @Override |
| public boolean willChangeBounds() { |
| return false; |
| } |
| |
| /** |
| * Computes mDestRect (the position in window space of the placeholder view that we should |
| * animate to). Some frames during the animation fail to compute getGlobalVisibleRect, so use |
| * the last known values in that case |
| */ |
| private void computeDestRect() { |
| final int prevTop = mDestRect.top; |
| final int prevLeft = mDestRect.left; |
| final int prevRight = mDestRect.right; |
| final int prevBottom = mDestRect.bottom; |
| |
| if (!getViewScreenMeasureRect(mViewToAnimate, mDestRect)) { |
| mDestRect.top = prevTop; |
| mDestRect.left = prevLeft; |
| mDestRect.bottom = prevBottom; |
| mDestRect.right = prevRight; |
| } |
| } |
| |
| /** |
| * Sets up the PopupWindow that the view will animate in. Animating the size and position of a |
| * popup can be choppy, so instead we make the popup fill the entire space of the screen, and |
| * animate the position of viewToAnimate within the popup using a Transformation |
| */ |
| private void initPopupWindow() { |
| mPopupRoot = new View(mViewToAnimate.getContext()) { |
| @Override |
| protected void onDraw(final Canvas canvas) { |
| canvas.save(); |
| canvas.clipRect(getLeft(), mActionBarRect.bottom - mPopupRect.top, getRight(), |
| getBottom()); |
| canvas.drawColor(Color.TRANSPARENT); |
| final float previousAlpha = mViewToAnimate.getAlpha(); |
| mViewToAnimate.setAlpha(1); |
| // The view's global position includes the notification bar height, but |
| // the popup window may or may not cover the notification bar (depending on screen |
| // rotation, IME status etc.), so we need to compensate for this difference by |
| // offseting vertically. |
| canvas.translate(mCurrentRect.left, mCurrentRect.top - mPopupRect.top); |
| |
| final float viewWidth = mViewToAnimate.getWidth(); |
| final float viewHeight = mViewToAnimate.getHeight(); |
| if (viewWidth > 0 && viewHeight > 0) { |
| canvas.scale(mCurrentRect.width() / viewWidth, |
| mCurrentRect.height() / viewHeight); |
| } |
| canvas.clipRect(0, 0, mCurrentRect.width(), mCurrentRect.height()); |
| if (!mPopupRect.isEmpty()) { |
| // HACK: Layout is unstable until mPopupRect is non-empty. |
| mViewToAnimate.draw(canvas); |
| } |
| mViewToAnimate.setAlpha(previousAlpha); |
| canvas.restore(); |
| } |
| }; |
| mPopupWindow = new PopupWindow(mViewToAnimate.getContext()); |
| mPopupWindow.setBackgroundDrawable(null); |
| mPopupWindow.setContentView(mPopupRoot); |
| mPopupWindow.setWidth(ViewGroup.LayoutParams.MATCH_PARENT); |
| mPopupWindow.setHeight(ViewGroup.LayoutParams.MATCH_PARENT); |
| mPopupWindow.setTouchable(false); |
| // We must pass a non-zero value for the y offset, or else the system resets the status bar |
| // color to black (M only) during the animation. The actual position of the window (and |
| // the animated view inside it) are still correct, regardless of what we pass for the y |
| // parameter (e.g. 1 and 100 both work). Not entirely sure why this works. |
| mPopupWindow.showAtLocation(mViewToAnimate, Gravity.TOP, 0, 1); |
| } |
| |
| private static boolean getViewScreenMeasureRect(final View view, final Rect outRect) { |
| outRect.set(UiUtils.getMeasuredBoundsOnScreen(view)); |
| return !outRect.isEmpty(); |
| } |
| } |