summaryrefslogtreecommitdiffstats
path: root/src/com/android/messaging/ui/animation/PopupTransitionAnimation.java
diff options
context:
space:
mode:
Diffstat (limited to 'src/com/android/messaging/ui/animation/PopupTransitionAnimation.java')
-rw-r--r--src/com/android/messaging/ui/animation/PopupTransitionAnimation.java302
1 files changed, 0 insertions, 302 deletions
diff --git a/src/com/android/messaging/ui/animation/PopupTransitionAnimation.java b/src/com/android/messaging/ui/animation/PopupTransitionAnimation.java
deleted file mode 100644
index 21529c6..0000000
--- a/src/com/android/messaging/ui/animation/PopupTransitionAnimation.java
+++ /dev/null
@@ -1,302 +0,0 @@
-/*
- * 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();
- final Activity activity = (Activity) viewToAnimate.getRootView().getContext();
- mActionBarView = activity.getWindow().getDecorView().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();
- }
-}