1 /*
2  * Copyright (C) 2010 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.server.wm;
18 
19 import static com.android.server.wm.WindowManagerDebugConfig.DEBUG_SURFACE_TRACE;
20 import static com.android.server.wm.WindowManagerDebugConfig.SHOW_LIGHT_TRANSACTIONS;
21 import static com.android.server.wm.WindowManagerDebugConfig.SHOW_SURFACE_ALLOC;
22 import static com.android.server.wm.WindowManagerDebugConfig.SHOW_TRANSACTIONS;
23 import static com.android.server.wm.WindowManagerDebugConfig.TAG_WITH_CLASS_NAME;
24 import static com.android.server.wm.WindowManagerDebugConfig.TAG_WM;
25 import static com.android.server.wm.WindowManagerService.TYPE_LAYER_MULTIPLIER;
26 import static com.android.server.wm.WindowStateAnimator.WINDOW_FREEZE_LAYER;
27 import static com.android.server.wm.WindowSurfaceController.SurfaceTrace;
28 
29 import android.content.Context;
30 import android.graphics.Matrix;
31 import android.graphics.PixelFormat;
32 import android.graphics.Rect;
33 import android.util.Slog;
34 import android.view.Display;
35 import android.view.DisplayInfo;
36 import android.view.Surface;
37 import android.view.Surface.OutOfResourcesException;
38 import android.view.SurfaceControl;
39 import android.view.SurfaceSession;
40 import android.view.animation.Animation;
41 import android.view.animation.AnimationUtils;
42 import android.view.animation.Transformation;
43 
44 import java.io.PrintWriter;
45 
46 class ScreenRotationAnimation {
47     static final String TAG = TAG_WITH_CLASS_NAME ? "ScreenRotationAnimation" : TAG_WM;
48     static final boolean DEBUG_STATE = false;
49     static final boolean DEBUG_TRANSFORMS = false;
50     static final boolean TWO_PHASE_ANIMATION = false;
51     static final boolean USE_CUSTOM_BLACK_FRAME = false;
52 
53     /*
54      * Layers for screen rotation animation. We put these layers above
55      * WINDOW_FREEZE_LAYER so that screen freeze will cover all windows.
56      */
57     static final int SCREEN_FREEZE_LAYER_BASE       = WINDOW_FREEZE_LAYER + TYPE_LAYER_MULTIPLIER;
58     static final int SCREEN_FREEZE_LAYER_ENTER      = SCREEN_FREEZE_LAYER_BASE;
59     static final int SCREEN_FREEZE_LAYER_SCREENSHOT = SCREEN_FREEZE_LAYER_BASE + 1;
60     static final int SCREEN_FREEZE_LAYER_EXIT       = SCREEN_FREEZE_LAYER_BASE + 2;
61     static final int SCREEN_FREEZE_LAYER_CUSTOM     = SCREEN_FREEZE_LAYER_BASE + 3;
62 
63     final Context mContext;
64     final DisplayContent mDisplayContent;
65     SurfaceControl mSurfaceControl;
66     BlackFrame mCustomBlackFrame;
67     BlackFrame mExitingBlackFrame;
68     BlackFrame mEnteringBlackFrame;
69     int mWidth, mHeight;
70 
71     int mOriginalRotation;
72     int mOriginalWidth, mOriginalHeight;
73     int mCurRotation;
74     Rect mOriginalDisplayRect = new Rect();
75     Rect mCurrentDisplayRect = new Rect();
76 
77     // For all animations, "exit" is for the UI elements that are going
78     // away (that is the snapshot of the old screen), and "enter" is for
79     // the new UI elements that are appearing (that is the active windows
80     // in their final orientation).
81 
82     // The starting animation for the exiting and entering elements.  This
83     // animation applies a transformation while the rotation is in progress.
84     // It is started immediately, before the new entering UI is ready.
85     Animation mStartExitAnimation;
86     final Transformation mStartExitTransformation = new Transformation();
87     Animation mStartEnterAnimation;
88     final Transformation mStartEnterTransformation = new Transformation();
89     Animation mStartFrameAnimation;
90     final Transformation mStartFrameTransformation = new Transformation();
91 
92     // The finishing animation for the exiting and entering elements.  This
93     // animation needs to undo the transformation of the starting animation.
94     // It starts running once the new rotation UI elements are ready to be
95     // displayed.
96     Animation mFinishExitAnimation;
97     final Transformation mFinishExitTransformation = new Transformation();
98     Animation mFinishEnterAnimation;
99     final Transformation mFinishEnterTransformation = new Transformation();
100     Animation mFinishFrameAnimation;
101     final Transformation mFinishFrameTransformation = new Transformation();
102 
103     // The current active animation to move from the old to the new rotated
104     // state.  Which animation is run here will depend on the old and new
105     // rotations.
106     Animation mRotateExitAnimation;
107     final Transformation mRotateExitTransformation = new Transformation();
108     Animation mRotateEnterAnimation;
109     final Transformation mRotateEnterTransformation = new Transformation();
110     Animation mRotateFrameAnimation;
111     final Transformation mRotateFrameTransformation = new Transformation();
112 
113     // A previously running rotate animation.  This will be used if we need
114     // to switch to a new rotation before finishing the previous one.
115     Animation mLastRotateExitAnimation;
116     final Transformation mLastRotateExitTransformation = new Transformation();
117     Animation mLastRotateEnterAnimation;
118     final Transformation mLastRotateEnterTransformation = new Transformation();
119     Animation mLastRotateFrameAnimation;
120     final Transformation mLastRotateFrameTransformation = new Transformation();
121 
122     // Complete transformations being applied.
123     final Transformation mExitTransformation = new Transformation();
124     final Transformation mEnterTransformation = new Transformation();
125     final Transformation mFrameTransformation = new Transformation();
126 
127     boolean mStarted;
128     boolean mAnimRunning;
129     boolean mFinishAnimReady;
130     long mFinishAnimStartTime;
131     boolean mForceDefaultOrientation;
132 
133     final Matrix mFrameInitialMatrix = new Matrix();
134     final Matrix mSnapshotInitialMatrix = new Matrix();
135     final Matrix mSnapshotFinalMatrix = new Matrix();
136     final Matrix mExitFrameFinalMatrix = new Matrix();
137     final Matrix mTmpMatrix = new Matrix();
138     final float[] mTmpFloats = new float[9];
139     private boolean mMoreRotateEnter;
140     private boolean mMoreRotateExit;
141     private boolean mMoreRotateFrame;
142     private boolean mMoreFinishEnter;
143     private boolean mMoreFinishExit;
144     private boolean mMoreFinishFrame;
145     private boolean mMoreStartEnter;
146     private boolean mMoreStartExit;
147     private boolean mMoreStartFrame;
148     long mHalfwayPoint;
149 
150     private final WindowManagerService mService;
151 
printTo(String prefix, PrintWriter pw)152     public void printTo(String prefix, PrintWriter pw) {
153         pw.print(prefix); pw.print("mSurface="); pw.print(mSurfaceControl);
154                 pw.print(" mWidth="); pw.print(mWidth);
155                 pw.print(" mHeight="); pw.println(mHeight);
156         if (USE_CUSTOM_BLACK_FRAME) {
157             pw.print(prefix); pw.print("mCustomBlackFrame="); pw.println(mCustomBlackFrame);
158             if (mCustomBlackFrame != null) {
159                 mCustomBlackFrame.printTo(prefix + "  ", pw);
160             }
161         }
162         pw.print(prefix); pw.print("mExitingBlackFrame="); pw.println(mExitingBlackFrame);
163         if (mExitingBlackFrame != null) {
164             mExitingBlackFrame.printTo(prefix + "  ", pw);
165         }
166         pw.print(prefix); pw.print("mEnteringBlackFrame="); pw.println(mEnteringBlackFrame);
167         if (mEnteringBlackFrame != null) {
168             mEnteringBlackFrame.printTo(prefix + "  ", pw);
169         }
170         pw.print(prefix); pw.print("mCurRotation="); pw.print(mCurRotation);
171                 pw.print(" mOriginalRotation="); pw.println(mOriginalRotation);
172         pw.print(prefix); pw.print("mOriginalWidth="); pw.print(mOriginalWidth);
173                 pw.print(" mOriginalHeight="); pw.println(mOriginalHeight);
174         pw.print(prefix); pw.print("mStarted="); pw.print(mStarted);
175                 pw.print(" mAnimRunning="); pw.print(mAnimRunning);
176                 pw.print(" mFinishAnimReady="); pw.print(mFinishAnimReady);
177                 pw.print(" mFinishAnimStartTime="); pw.println(mFinishAnimStartTime);
178         pw.print(prefix); pw.print("mStartExitAnimation="); pw.print(mStartExitAnimation);
179                 pw.print(" "); mStartExitTransformation.printShortString(pw); pw.println();
180         pw.print(prefix); pw.print("mStartEnterAnimation="); pw.print(mStartEnterAnimation);
181                 pw.print(" "); mStartEnterTransformation.printShortString(pw); pw.println();
182         pw.print(prefix); pw.print("mStartFrameAnimation="); pw.print(mStartFrameAnimation);
183                 pw.print(" "); mStartFrameTransformation.printShortString(pw); pw.println();
184         pw.print(prefix); pw.print("mFinishExitAnimation="); pw.print(mFinishExitAnimation);
185                 pw.print(" "); mFinishExitTransformation.printShortString(pw); pw.println();
186         pw.print(prefix); pw.print("mFinishEnterAnimation="); pw.print(mFinishEnterAnimation);
187                 pw.print(" "); mFinishEnterTransformation.printShortString(pw); pw.println();
188         pw.print(prefix); pw.print("mFinishFrameAnimation="); pw.print(mFinishFrameAnimation);
189                 pw.print(" "); mFinishFrameTransformation.printShortString(pw); pw.println();
190         pw.print(prefix); pw.print("mRotateExitAnimation="); pw.print(mRotateExitAnimation);
191                 pw.print(" "); mRotateExitTransformation.printShortString(pw); pw.println();
192         pw.print(prefix); pw.print("mRotateEnterAnimation="); pw.print(mRotateEnterAnimation);
193                 pw.print(" "); mRotateEnterTransformation.printShortString(pw); pw.println();
194         pw.print(prefix); pw.print("mRotateFrameAnimation="); pw.print(mRotateFrameAnimation);
195                 pw.print(" "); mRotateFrameTransformation.printShortString(pw); pw.println();
196         pw.print(prefix); pw.print("mExitTransformation=");
197                 mExitTransformation.printShortString(pw); pw.println();
198         pw.print(prefix); pw.print("mEnterTransformation=");
199                 mEnterTransformation.printShortString(pw); pw.println();
200         pw.print(prefix); pw.print("mFrameTransformation=");
201                 mEnterTransformation.printShortString(pw); pw.println();
202         pw.print(prefix); pw.print("mFrameInitialMatrix=");
203                 mFrameInitialMatrix.printShortString(pw);
204                 pw.println();
205         pw.print(prefix); pw.print("mSnapshotInitialMatrix=");
206                 mSnapshotInitialMatrix.printShortString(pw);
207                 pw.print(" mSnapshotFinalMatrix="); mSnapshotFinalMatrix.printShortString(pw);
208                 pw.println();
209         pw.print(prefix); pw.print("mExitFrameFinalMatrix=");
210                 mExitFrameFinalMatrix.printShortString(pw);
211                 pw.println();
212         pw.print(prefix); pw.print("mForceDefaultOrientation="); pw.print(mForceDefaultOrientation);
213         if (mForceDefaultOrientation) {
214             pw.print(" mOriginalDisplayRect="); pw.print(mOriginalDisplayRect.toShortString());
215             pw.print(" mCurrentDisplayRect="); pw.println(mCurrentDisplayRect.toShortString());
216         }
217     }
218 
ScreenRotationAnimation(Context context, DisplayContent displayContent, SurfaceSession session, boolean inTransaction, boolean forceDefaultOrientation, boolean isSecure, WindowManagerService service)219     public ScreenRotationAnimation(Context context, DisplayContent displayContent,
220             SurfaceSession session, boolean inTransaction, boolean forceDefaultOrientation,
221             boolean isSecure, WindowManagerService service) {
222         mService = service;
223         mContext = context;
224         mDisplayContent = displayContent;
225         displayContent.getLogicalDisplayRect(mOriginalDisplayRect);
226 
227         // Screenshot does NOT include rotation!
228         final Display display = displayContent.getDisplay();
229         int originalRotation = display.getRotation();
230         final int originalWidth;
231         final int originalHeight;
232         DisplayInfo displayInfo = displayContent.getDisplayInfo();
233         if (forceDefaultOrientation) {
234             // Emulated orientation.
235             mForceDefaultOrientation = true;
236             originalWidth = displayContent.mBaseDisplayWidth;
237             originalHeight = displayContent.mBaseDisplayHeight;
238         } else {
239             // Normal situation
240             originalWidth = displayInfo.logicalWidth;
241             originalHeight = displayInfo.logicalHeight;
242         }
243         if (originalRotation == Surface.ROTATION_90
244                 || originalRotation == Surface.ROTATION_270) {
245             mWidth = originalHeight;
246             mHeight = originalWidth;
247         } else {
248             mWidth = originalWidth;
249             mHeight = originalHeight;
250         }
251 
252         mOriginalRotation = originalRotation;
253         mOriginalWidth = originalWidth;
254         mOriginalHeight = originalHeight;
255 
256         if (!inTransaction) {
257             if (SHOW_LIGHT_TRANSACTIONS) Slog.i(TAG_WM,
258                     ">>> OPEN TRANSACTION ScreenRotationAnimation");
259             mService.openSurfaceTransaction();
260         }
261 
262         try {
263             try {
264                 int flags = SurfaceControl.HIDDEN;
265                 if (isSecure) {
266                     flags |= SurfaceControl.SECURE;
267                 }
268 
269                 if (DEBUG_SURFACE_TRACE) {
270                     mSurfaceControl = new SurfaceTrace(session, "ScreenshotSurface",
271                             mWidth, mHeight,
272                             PixelFormat.OPAQUE, flags);
273                     Slog.w(TAG, "ScreenRotationAnimation ctor: displayOffset="
274                             + mOriginalDisplayRect.toShortString());
275                 } else {
276                     mSurfaceControl = new SurfaceControl(session, "ScreenshotSurface",
277                             mWidth, mHeight,
278                             PixelFormat.OPAQUE, flags);
279                 }
280                 // capture a screenshot into the surface we just created
281                 Surface sur = new Surface();
282                 sur.copyFrom(mSurfaceControl);
283                 // FIXME: we should use the proper display
284                 SurfaceControl.screenshot(SurfaceControl.getBuiltInDisplay(
285                         SurfaceControl.BUILT_IN_DISPLAY_ID_MAIN), sur);
286                 mSurfaceControl.setLayerStack(display.getLayerStack());
287                 mSurfaceControl.setLayer(SCREEN_FREEZE_LAYER_SCREENSHOT);
288                 mSurfaceControl.setAlpha(0);
289                 mSurfaceControl.show();
290                 sur.destroy();
291             } catch (OutOfResourcesException e) {
292                 Slog.w(TAG, "Unable to allocate freeze surface", e);
293             }
294 
295             if (SHOW_TRANSACTIONS || SHOW_SURFACE_ALLOC) Slog.i(TAG_WM,
296                     "  FREEZE " + mSurfaceControl + ": CREATE");
297 
298             setRotationInTransaction(originalRotation);
299         } finally {
300             if (!inTransaction) {
301                 mService.closeSurfaceTransaction();
302                 if (SHOW_LIGHT_TRANSACTIONS) Slog.i(TAG_WM,
303                         "<<< CLOSE TRANSACTION ScreenRotationAnimation");
304             }
305         }
306     }
307 
hasScreenshot()308     boolean hasScreenshot() {
309         return mSurfaceControl != null;
310     }
311 
setSnapshotTransformInTransaction(Matrix matrix, float alpha)312     private void setSnapshotTransformInTransaction(Matrix matrix, float alpha) {
313         if (mSurfaceControl != null) {
314             matrix.getValues(mTmpFloats);
315             float x = mTmpFloats[Matrix.MTRANS_X];
316             float y = mTmpFloats[Matrix.MTRANS_Y];
317             if (mForceDefaultOrientation) {
318                 mDisplayContent.getLogicalDisplayRect(mCurrentDisplayRect);
319                 x -= mCurrentDisplayRect.left;
320                 y -= mCurrentDisplayRect.top;
321             }
322             mSurfaceControl.setPosition(x, y);
323             mSurfaceControl.setMatrix(
324                     mTmpFloats[Matrix.MSCALE_X], mTmpFloats[Matrix.MSKEW_Y],
325                     mTmpFloats[Matrix.MSKEW_X], mTmpFloats[Matrix.MSCALE_Y]);
326             mSurfaceControl.setAlpha(alpha);
327             if (DEBUG_TRANSFORMS) {
328                 float[] srcPnts = new float[] { 0, 0, mWidth, mHeight };
329                 float[] dstPnts = new float[4];
330                 matrix.mapPoints(dstPnts, srcPnts);
331                 Slog.i(TAG, "Original  : (" + srcPnts[0] + "," + srcPnts[1]
332                         + ")-(" + srcPnts[2] + "," + srcPnts[3] + ")");
333                 Slog.i(TAG, "Transformed: (" + dstPnts[0] + "," + dstPnts[1]
334                         + ")-(" + dstPnts[2] + "," + dstPnts[3] + ")");
335             }
336         }
337     }
338 
createRotationMatrix(int rotation, int width, int height, Matrix outMatrix)339     public static void createRotationMatrix(int rotation, int width, int height,
340             Matrix outMatrix) {
341         switch (rotation) {
342             case Surface.ROTATION_0:
343                 outMatrix.reset();
344                 break;
345             case Surface.ROTATION_90:
346                 outMatrix.setRotate(90, 0, 0);
347                 outMatrix.postTranslate(height, 0);
348                 break;
349             case Surface.ROTATION_180:
350                 outMatrix.setRotate(180, 0, 0);
351                 outMatrix.postTranslate(width, height);
352                 break;
353             case Surface.ROTATION_270:
354                 outMatrix.setRotate(270, 0, 0);
355                 outMatrix.postTranslate(0, width);
356                 break;
357         }
358     }
359 
360     // Must be called while in a transaction.
setRotationInTransaction(int rotation)361     private void setRotationInTransaction(int rotation) {
362         mCurRotation = rotation;
363 
364         // Compute the transformation matrix that must be applied
365         // to the snapshot to make it stay in the same original position
366         // with the current screen rotation.
367         int delta = DisplayContent.deltaRotation(rotation, Surface.ROTATION_0);
368         createRotationMatrix(delta, mWidth, mHeight, mSnapshotInitialMatrix);
369 
370         if (DEBUG_STATE) Slog.v(TAG, "**** ROTATION: " + delta);
371         setSnapshotTransformInTransaction(mSnapshotInitialMatrix, 1.0f);
372     }
373 
374     // Must be called while in a transaction.
setRotationInTransaction(int rotation, SurfaceSession session, long maxAnimationDuration, float animationScale, int finalWidth, int finalHeight)375     public boolean setRotationInTransaction(int rotation, SurfaceSession session,
376             long maxAnimationDuration, float animationScale, int finalWidth, int finalHeight) {
377         setRotationInTransaction(rotation);
378         if (TWO_PHASE_ANIMATION) {
379             return startAnimation(session, maxAnimationDuration, animationScale,
380                     finalWidth, finalHeight, false, 0, 0);
381         }
382 
383         // Don't start animation yet.
384         return false;
385     }
386 
387     /**
388      * Returns true if animating.
389      */
startAnimation(SurfaceSession session, long maxAnimationDuration, float animationScale, int finalWidth, int finalHeight, boolean dismissing, int exitAnim, int enterAnim)390     private boolean startAnimation(SurfaceSession session, long maxAnimationDuration,
391             float animationScale, int finalWidth, int finalHeight, boolean dismissing,
392             int exitAnim, int enterAnim) {
393         if (mSurfaceControl == null) {
394             // Can't do animation.
395             return false;
396         }
397         if (mStarted) {
398             return true;
399         }
400 
401         mStarted = true;
402 
403         boolean firstStart = false;
404 
405         // Figure out how the screen has moved from the original rotation.
406         int delta = DisplayContent.deltaRotation(mCurRotation, mOriginalRotation);
407 
408         if (TWO_PHASE_ANIMATION && mFinishExitAnimation == null
409                 && (!dismissing || delta != Surface.ROTATION_0)) {
410             if (DEBUG_STATE) Slog.v(TAG, "Creating start and finish animations");
411             firstStart = true;
412             mStartExitAnimation = AnimationUtils.loadAnimation(mContext,
413                     com.android.internal.R.anim.screen_rotate_start_exit);
414             mStartEnterAnimation = AnimationUtils.loadAnimation(mContext,
415                     com.android.internal.R.anim.screen_rotate_start_enter);
416             if (USE_CUSTOM_BLACK_FRAME) {
417                 mStartFrameAnimation = AnimationUtils.loadAnimation(mContext,
418                         com.android.internal.R.anim.screen_rotate_start_frame);
419             }
420             mFinishExitAnimation = AnimationUtils.loadAnimation(mContext,
421                     com.android.internal.R.anim.screen_rotate_finish_exit);
422             mFinishEnterAnimation = AnimationUtils.loadAnimation(mContext,
423                     com.android.internal.R.anim.screen_rotate_finish_enter);
424             if (USE_CUSTOM_BLACK_FRAME) {
425                 mFinishFrameAnimation = AnimationUtils.loadAnimation(mContext,
426                         com.android.internal.R.anim.screen_rotate_finish_frame);
427             }
428         }
429 
430         if (DEBUG_STATE) Slog.v(TAG, "Rotation delta: " + delta + " finalWidth="
431                 + finalWidth + " finalHeight=" + finalHeight
432                 + " origWidth=" + mOriginalWidth + " origHeight=" + mOriginalHeight);
433 
434         final boolean customAnim;
435         if (exitAnim != 0 && enterAnim != 0) {
436             customAnim = true;
437             mRotateExitAnimation = AnimationUtils.loadAnimation(mContext, exitAnim);
438             mRotateEnterAnimation = AnimationUtils.loadAnimation(mContext, enterAnim);
439         } else {
440             customAnim = false;
441             switch (delta) {
442                 case Surface.ROTATION_0:
443                     mRotateExitAnimation = AnimationUtils.loadAnimation(mContext,
444                             com.android.internal.R.anim.screen_rotate_0_exit);
445                     mRotateEnterAnimation = AnimationUtils.loadAnimation(mContext,
446                             com.android.internal.R.anim.screen_rotate_0_enter);
447                     if (USE_CUSTOM_BLACK_FRAME) {
448                         mRotateFrameAnimation = AnimationUtils.loadAnimation(mContext,
449                                 com.android.internal.R.anim.screen_rotate_0_frame);
450                     }
451                     break;
452                 case Surface.ROTATION_90:
453                     mRotateExitAnimation = AnimationUtils.loadAnimation(mContext,
454                             com.android.internal.R.anim.screen_rotate_plus_90_exit);
455                     mRotateEnterAnimation = AnimationUtils.loadAnimation(mContext,
456                             com.android.internal.R.anim.screen_rotate_plus_90_enter);
457                     if (USE_CUSTOM_BLACK_FRAME) {
458                         mRotateFrameAnimation = AnimationUtils.loadAnimation(mContext,
459                                 com.android.internal.R.anim.screen_rotate_plus_90_frame);
460                     }
461                     break;
462                 case Surface.ROTATION_180:
463                     mRotateExitAnimation = AnimationUtils.loadAnimation(mContext,
464                             com.android.internal.R.anim.screen_rotate_180_exit);
465                     mRotateEnterAnimation = AnimationUtils.loadAnimation(mContext,
466                             com.android.internal.R.anim.screen_rotate_180_enter);
467                     if (USE_CUSTOM_BLACK_FRAME) {
468                         mRotateFrameAnimation = AnimationUtils.loadAnimation(mContext,
469                                 com.android.internal.R.anim.screen_rotate_180_frame);
470                     }
471                     break;
472                 case Surface.ROTATION_270:
473                     mRotateExitAnimation = AnimationUtils.loadAnimation(mContext,
474                             com.android.internal.R.anim.screen_rotate_minus_90_exit);
475                     mRotateEnterAnimation = AnimationUtils.loadAnimation(mContext,
476                             com.android.internal.R.anim.screen_rotate_minus_90_enter);
477                     if (USE_CUSTOM_BLACK_FRAME) {
478                         mRotateFrameAnimation = AnimationUtils.loadAnimation(mContext,
479                                 com.android.internal.R.anim.screen_rotate_minus_90_frame);
480                     }
481                     break;
482             }
483         }
484 
485         // Initialize the animations.  This is a hack, redefining what "parent"
486         // means to allow supplying the last and next size.  In this definition
487         // "%p" is the original (let's call it "previous") size, and "%" is the
488         // screen's current/new size.
489         if (TWO_PHASE_ANIMATION && firstStart) {
490             // Compute partial steps between original and final sizes.  These
491             // are used for the dimensions of the exiting and entering elements,
492             // so they are never stretched too significantly.
493             final int halfWidth = (finalWidth + mOriginalWidth) / 2;
494             final int halfHeight = (finalHeight + mOriginalHeight) / 2;
495 
496             if (DEBUG_STATE) Slog.v(TAG, "Initializing start and finish animations");
497             mStartEnterAnimation.initialize(finalWidth, finalHeight,
498                     halfWidth, halfHeight);
499             mStartExitAnimation.initialize(halfWidth, halfHeight,
500                     mOriginalWidth, mOriginalHeight);
501             mFinishEnterAnimation.initialize(finalWidth, finalHeight,
502                     halfWidth, halfHeight);
503             mFinishExitAnimation.initialize(halfWidth, halfHeight,
504                     mOriginalWidth, mOriginalHeight);
505             if (USE_CUSTOM_BLACK_FRAME) {
506                 mStartFrameAnimation.initialize(finalWidth, finalHeight,
507                         mOriginalWidth, mOriginalHeight);
508                 mFinishFrameAnimation.initialize(finalWidth, finalHeight,
509                         mOriginalWidth, mOriginalHeight);
510             }
511         }
512         mRotateEnterAnimation.initialize(finalWidth, finalHeight, mOriginalWidth, mOriginalHeight);
513         mRotateExitAnimation.initialize(finalWidth, finalHeight, mOriginalWidth, mOriginalHeight);
514         if (USE_CUSTOM_BLACK_FRAME) {
515             mRotateFrameAnimation.initialize(finalWidth, finalHeight, mOriginalWidth,
516                     mOriginalHeight);
517         }
518         mAnimRunning = false;
519         mFinishAnimReady = false;
520         mFinishAnimStartTime = -1;
521 
522         if (TWO_PHASE_ANIMATION && firstStart) {
523             mStartExitAnimation.restrictDuration(maxAnimationDuration);
524             mStartExitAnimation.scaleCurrentDuration(animationScale);
525             mStartEnterAnimation.restrictDuration(maxAnimationDuration);
526             mStartEnterAnimation.scaleCurrentDuration(animationScale);
527             mFinishExitAnimation.restrictDuration(maxAnimationDuration);
528             mFinishExitAnimation.scaleCurrentDuration(animationScale);
529             mFinishEnterAnimation.restrictDuration(maxAnimationDuration);
530             mFinishEnterAnimation.scaleCurrentDuration(animationScale);
531             if (USE_CUSTOM_BLACK_FRAME) {
532                 mStartFrameAnimation.restrictDuration(maxAnimationDuration);
533                 mStartFrameAnimation.scaleCurrentDuration(animationScale);
534                 mFinishFrameAnimation.restrictDuration(maxAnimationDuration);
535                 mFinishFrameAnimation.scaleCurrentDuration(animationScale);
536             }
537         }
538         mRotateExitAnimation.restrictDuration(maxAnimationDuration);
539         mRotateExitAnimation.scaleCurrentDuration(animationScale);
540         mRotateEnterAnimation.restrictDuration(maxAnimationDuration);
541         mRotateEnterAnimation.scaleCurrentDuration(animationScale);
542         if (USE_CUSTOM_BLACK_FRAME) {
543             mRotateFrameAnimation.restrictDuration(maxAnimationDuration);
544             mRotateFrameAnimation.scaleCurrentDuration(animationScale);
545         }
546 
547         final int layerStack = mDisplayContent.getDisplay().getLayerStack();
548         if (USE_CUSTOM_BLACK_FRAME && mCustomBlackFrame == null) {
549             if (SHOW_LIGHT_TRANSACTIONS || DEBUG_STATE) Slog.i(
550                     TAG_WM,
551                     ">>> OPEN TRANSACTION ScreenRotationAnimation.startAnimation");
552             mService.openSurfaceTransaction();
553 
554             // Compute the transformation matrix that must be applied
555             // the the black frame to make it stay in the initial position
556             // before the new screen rotation.  This is different than the
557             // snapshot transformation because the snapshot is always based
558             // of the native orientation of the screen, not the orientation
559             // we were last in.
560             createRotationMatrix(delta, mOriginalWidth, mOriginalHeight, mFrameInitialMatrix);
561 
562             try {
563                 Rect outer = new Rect(-mOriginalWidth*1, -mOriginalHeight*1,
564                         mOriginalWidth*2, mOriginalHeight*2);
565                 Rect inner = new Rect(0, 0, mOriginalWidth, mOriginalHeight);
566                 mCustomBlackFrame = new BlackFrame(session, outer, inner,
567                         SCREEN_FREEZE_LAYER_CUSTOM, layerStack, false);
568                 mCustomBlackFrame.setMatrix(mFrameInitialMatrix);
569             } catch (OutOfResourcesException e) {
570                 Slog.w(TAG, "Unable to allocate black surface", e);
571             } finally {
572                 mService.closeSurfaceTransaction();
573                 if (SHOW_LIGHT_TRANSACTIONS || DEBUG_STATE) Slog.i(
574                         TAG_WM,
575                         "<<< CLOSE TRANSACTION ScreenRotationAnimation.startAnimation");
576             }
577         }
578 
579         if (!customAnim && mExitingBlackFrame == null) {
580             if (SHOW_LIGHT_TRANSACTIONS || DEBUG_STATE) Slog.i(
581                     TAG_WM,
582                     ">>> OPEN TRANSACTION ScreenRotationAnimation.startAnimation");
583             mService.openSurfaceTransaction();
584             try {
585                 // Compute the transformation matrix that must be applied
586                 // the the black frame to make it stay in the initial position
587                 // before the new screen rotation.  This is different than the
588                 // snapshot transformation because the snapshot is always based
589                 // of the native orientation of the screen, not the orientation
590                 // we were last in.
591                 createRotationMatrix(delta, mOriginalWidth, mOriginalHeight, mFrameInitialMatrix);
592 
593                 final Rect outer;
594                 final Rect inner;
595                 if (mForceDefaultOrientation) {
596                     // Going from a smaller Display to a larger Display, add curtains to sides
597                     // or top and bottom. Going from a larger to smaller display will result in
598                     // no BlackSurfaces being constructed.
599                     outer = mCurrentDisplayRect;
600                     inner = mOriginalDisplayRect;
601                 } else {
602                     outer = new Rect(-mOriginalWidth*1, -mOriginalHeight*1,
603                             mOriginalWidth*2, mOriginalHeight*2);
604                     inner = new Rect(0, 0, mOriginalWidth, mOriginalHeight);
605                 }
606                 mExitingBlackFrame = new BlackFrame(session, outer, inner,
607                         SCREEN_FREEZE_LAYER_EXIT, layerStack, mForceDefaultOrientation);
608                 mExitingBlackFrame.setMatrix(mFrameInitialMatrix);
609             } catch (OutOfResourcesException e) {
610                 Slog.w(TAG, "Unable to allocate black surface", e);
611             } finally {
612                 mService.closeSurfaceTransaction();
613                 if (SHOW_LIGHT_TRANSACTIONS || DEBUG_STATE) Slog.i(
614                         TAG_WM,
615                         "<<< CLOSE TRANSACTION ScreenRotationAnimation.startAnimation");
616             }
617         }
618 
619         if (customAnim && mEnteringBlackFrame == null) {
620             if (SHOW_LIGHT_TRANSACTIONS || DEBUG_STATE) Slog.i(
621                     TAG_WM,
622                     ">>> OPEN TRANSACTION ScreenRotationAnimation.startAnimation");
623             mService.openSurfaceTransaction();
624 
625             try {
626                 Rect outer = new Rect(-finalWidth*1, -finalHeight*1,
627                         finalWidth*2, finalHeight*2);
628                 Rect inner = new Rect(0, 0, finalWidth, finalHeight);
629                 mEnteringBlackFrame = new BlackFrame(session, outer, inner,
630                         SCREEN_FREEZE_LAYER_ENTER, layerStack, false);
631             } catch (OutOfResourcesException e) {
632                 Slog.w(TAG, "Unable to allocate black surface", e);
633             } finally {
634                 mService.closeSurfaceTransaction();
635                 if (SHOW_LIGHT_TRANSACTIONS || DEBUG_STATE) Slog.i(
636                         TAG_WM,
637                         "<<< CLOSE TRANSACTION ScreenRotationAnimation.startAnimation");
638             }
639         }
640 
641         return true;
642     }
643 
644     /**
645      * Returns true if animating.
646      */
dismiss(SurfaceSession session, long maxAnimationDuration, float animationScale, int finalWidth, int finalHeight, int exitAnim, int enterAnim)647     public boolean dismiss(SurfaceSession session, long maxAnimationDuration,
648             float animationScale, int finalWidth, int finalHeight, int exitAnim, int enterAnim) {
649         if (DEBUG_STATE) Slog.v(TAG, "Dismiss!");
650         if (mSurfaceControl == null) {
651             // Can't do animation.
652             return false;
653         }
654         if (!mStarted) {
655             startAnimation(session, maxAnimationDuration, animationScale, finalWidth, finalHeight,
656                     true, exitAnim, enterAnim);
657         }
658         if (!mStarted) {
659             return false;
660         }
661         if (DEBUG_STATE) Slog.v(TAG, "Setting mFinishAnimReady = true");
662         mFinishAnimReady = true;
663         return true;
664     }
665 
kill()666     public void kill() {
667         if (DEBUG_STATE) Slog.v(TAG, "Kill!");
668         if (mSurfaceControl != null) {
669             if (SHOW_TRANSACTIONS ||
670                     SHOW_SURFACE_ALLOC) Slog.i(TAG_WM,
671                             "  FREEZE " + mSurfaceControl + ": DESTROY");
672             mSurfaceControl.destroy();
673             mSurfaceControl = null;
674         }
675         if (mCustomBlackFrame != null) {
676             mCustomBlackFrame.kill();
677             mCustomBlackFrame = null;
678         }
679         if (mExitingBlackFrame != null) {
680             mExitingBlackFrame.kill();
681             mExitingBlackFrame = null;
682         }
683         if (mEnteringBlackFrame != null) {
684             mEnteringBlackFrame.kill();
685             mEnteringBlackFrame = null;
686         }
687         if (TWO_PHASE_ANIMATION) {
688             if (mStartExitAnimation != null) {
689                 mStartExitAnimation.cancel();
690                 mStartExitAnimation = null;
691             }
692             if (mStartEnterAnimation != null) {
693                 mStartEnterAnimation.cancel();
694                 mStartEnterAnimation = null;
695             }
696             if (mFinishExitAnimation != null) {
697                 mFinishExitAnimation.cancel();
698                 mFinishExitAnimation = null;
699             }
700             if (mFinishEnterAnimation != null) {
701                 mFinishEnterAnimation.cancel();
702                 mFinishEnterAnimation = null;
703             }
704         }
705         if (USE_CUSTOM_BLACK_FRAME) {
706             if (mStartFrameAnimation != null) {
707                 mStartFrameAnimation.cancel();
708                 mStartFrameAnimation = null;
709             }
710             if (mRotateFrameAnimation != null) {
711                 mRotateFrameAnimation.cancel();
712                 mRotateFrameAnimation = null;
713             }
714             if (mFinishFrameAnimation != null) {
715                 mFinishFrameAnimation.cancel();
716                 mFinishFrameAnimation = null;
717             }
718         }
719         if (mRotateExitAnimation != null) {
720             mRotateExitAnimation.cancel();
721             mRotateExitAnimation = null;
722         }
723         if (mRotateEnterAnimation != null) {
724             mRotateEnterAnimation.cancel();
725             mRotateEnterAnimation = null;
726         }
727     }
728 
isAnimating()729     public boolean isAnimating() {
730         return hasAnimations() || (TWO_PHASE_ANIMATION && mFinishAnimReady);
731     }
732 
isRotating()733     public boolean isRotating() {
734         return mCurRotation != mOriginalRotation;
735     }
736 
hasAnimations()737     private boolean hasAnimations() {
738         return (TWO_PHASE_ANIMATION &&
739                     (mStartEnterAnimation != null || mStartExitAnimation != null
740                     || mFinishEnterAnimation != null || mFinishExitAnimation != null))
741                 || (USE_CUSTOM_BLACK_FRAME &&
742                         (mStartFrameAnimation != null || mRotateFrameAnimation != null
743                         || mFinishFrameAnimation != null))
744                 || mRotateEnterAnimation != null || mRotateExitAnimation != null;
745     }
746 
stepAnimation(long now)747     private boolean stepAnimation(long now) {
748         if (now > mHalfwayPoint) {
749             mHalfwayPoint = Long.MAX_VALUE;
750         }
751         if (mFinishAnimReady && mFinishAnimStartTime < 0) {
752             if (DEBUG_STATE) Slog.v(TAG, "Step: finish anim now ready");
753             mFinishAnimStartTime = now;
754         }
755 
756         if (TWO_PHASE_ANIMATION) {
757             mMoreStartExit = false;
758             if (mStartExitAnimation != null) {
759                 mMoreStartExit = mStartExitAnimation.getTransformation(now, mStartExitTransformation);
760                 if (DEBUG_TRANSFORMS) Slog.v(TAG, "Stepped start exit: " + mStartExitTransformation);
761             }
762 
763             mMoreStartEnter = false;
764             if (mStartEnterAnimation != null) {
765                 mMoreStartEnter = mStartEnterAnimation.getTransformation(now, mStartEnterTransformation);
766                 if (DEBUG_TRANSFORMS) Slog.v(TAG, "Stepped start enter: " + mStartEnterTransformation);
767             }
768         }
769         if (USE_CUSTOM_BLACK_FRAME) {
770             mMoreStartFrame = false;
771             if (mStartFrameAnimation != null) {
772                 mMoreStartFrame = mStartFrameAnimation.getTransformation(now, mStartFrameTransformation);
773                 if (DEBUG_TRANSFORMS) Slog.v(TAG, "Stepped start frame: " + mStartFrameTransformation);
774             }
775         }
776 
777         long finishNow = mFinishAnimReady ? (now - mFinishAnimStartTime) : 0;
778         if (DEBUG_STATE) Slog.v(TAG, "Step: finishNow=" + finishNow);
779 
780         if (TWO_PHASE_ANIMATION) {
781             mMoreFinishExit = false;
782             if (mFinishExitAnimation != null) {
783                 mMoreFinishExit = mFinishExitAnimation.getTransformation(finishNow, mFinishExitTransformation);
784                 if (DEBUG_TRANSFORMS) Slog.v(TAG, "Stepped finish exit: " + mFinishExitTransformation);
785             }
786 
787             mMoreFinishEnter = false;
788             if (mFinishEnterAnimation != null) {
789                 mMoreFinishEnter = mFinishEnterAnimation.getTransformation(finishNow, mFinishEnterTransformation);
790                 if (DEBUG_TRANSFORMS) Slog.v(TAG, "Stepped finish enter: " + mFinishEnterTransformation);
791             }
792         }
793         if (USE_CUSTOM_BLACK_FRAME) {
794             mMoreFinishFrame = false;
795             if (mFinishFrameAnimation != null) {
796                 mMoreFinishFrame = mFinishFrameAnimation.getTransformation(finishNow, mFinishFrameTransformation);
797                 if (DEBUG_TRANSFORMS) Slog.v(TAG, "Stepped finish frame: " + mFinishFrameTransformation);
798             }
799         }
800 
801         mMoreRotateExit = false;
802         if (mRotateExitAnimation != null) {
803             mMoreRotateExit = mRotateExitAnimation.getTransformation(now, mRotateExitTransformation);
804             if (DEBUG_TRANSFORMS) Slog.v(TAG, "Stepped rotate exit: " + mRotateExitTransformation);
805         }
806 
807         mMoreRotateEnter = false;
808         if (mRotateEnterAnimation != null) {
809             mMoreRotateEnter = mRotateEnterAnimation.getTransformation(now, mRotateEnterTransformation);
810             if (DEBUG_TRANSFORMS) Slog.v(TAG, "Stepped rotate enter: " + mRotateEnterTransformation);
811         }
812 
813         if (USE_CUSTOM_BLACK_FRAME) {
814             mMoreRotateFrame = false;
815             if (mRotateFrameAnimation != null) {
816                 mMoreRotateFrame = mRotateFrameAnimation.getTransformation(now, mRotateFrameTransformation);
817                 if (DEBUG_TRANSFORMS) Slog.v(TAG, "Stepped rotate frame: " + mRotateFrameTransformation);
818             }
819         }
820 
821         if (!mMoreRotateExit && (!TWO_PHASE_ANIMATION || (!mMoreStartExit && !mMoreFinishExit))) {
822             if (TWO_PHASE_ANIMATION) {
823                 if (mStartExitAnimation != null) {
824                     if (DEBUG_STATE) Slog.v(TAG, "Exit animations done, clearing start exit anim!");
825                     mStartExitAnimation.cancel();
826                     mStartExitAnimation = null;
827                     mStartExitTransformation.clear();
828                 }
829                 if (mFinishExitAnimation != null) {
830                     if (DEBUG_STATE) Slog.v(TAG, "Exit animations done, clearing finish exit anim!");
831                     mFinishExitAnimation.cancel();
832                     mFinishExitAnimation = null;
833                     mFinishExitTransformation.clear();
834                 }
835             }
836             if (mRotateExitAnimation != null) {
837                 if (DEBUG_STATE) Slog.v(TAG, "Exit animations done, clearing rotate exit anim!");
838                 mRotateExitAnimation.cancel();
839                 mRotateExitAnimation = null;
840                 mRotateExitTransformation.clear();
841             }
842         }
843 
844         if (!mMoreRotateEnter && (!TWO_PHASE_ANIMATION || (!mMoreStartEnter && !mMoreFinishEnter))) {
845             if (TWO_PHASE_ANIMATION) {
846                 if (mStartEnterAnimation != null) {
847                     if (DEBUG_STATE) Slog.v(TAG, "Enter animations done, clearing start enter anim!");
848                     mStartEnterAnimation.cancel();
849                     mStartEnterAnimation = null;
850                     mStartEnterTransformation.clear();
851                 }
852                 if (mFinishEnterAnimation != null) {
853                     if (DEBUG_STATE) Slog.v(TAG, "Enter animations done, clearing finish enter anim!");
854                     mFinishEnterAnimation.cancel();
855                     mFinishEnterAnimation = null;
856                     mFinishEnterTransformation.clear();
857                 }
858             }
859             if (mRotateEnterAnimation != null) {
860                 if (DEBUG_STATE) Slog.v(TAG, "Enter animations done, clearing rotate enter anim!");
861                 mRotateEnterAnimation.cancel();
862                 mRotateEnterAnimation = null;
863                 mRotateEnterTransformation.clear();
864             }
865         }
866 
867         if (USE_CUSTOM_BLACK_FRAME && !mMoreStartFrame && !mMoreRotateFrame && !mMoreFinishFrame) {
868             if (mStartFrameAnimation != null) {
869                 if (DEBUG_STATE) Slog.v(TAG, "Frame animations done, clearing start frame anim!");
870                 mStartFrameAnimation.cancel();
871                 mStartFrameAnimation = null;
872                 mStartFrameTransformation.clear();
873             }
874             if (mFinishFrameAnimation != null) {
875                 if (DEBUG_STATE) Slog.v(TAG, "Frame animations done, clearing finish frame anim!");
876                 mFinishFrameAnimation.cancel();
877                 mFinishFrameAnimation = null;
878                 mFinishFrameTransformation.clear();
879             }
880             if (mRotateFrameAnimation != null) {
881                 if (DEBUG_STATE) Slog.v(TAG, "Frame animations done, clearing rotate frame anim!");
882                 mRotateFrameAnimation.cancel();
883                 mRotateFrameAnimation = null;
884                 mRotateFrameTransformation.clear();
885             }
886         }
887 
888         mExitTransformation.set(mRotateExitTransformation);
889         mEnterTransformation.set(mRotateEnterTransformation);
890         if (TWO_PHASE_ANIMATION) {
891             mExitTransformation.compose(mStartExitTransformation);
892             mExitTransformation.compose(mFinishExitTransformation);
893 
894             mEnterTransformation.compose(mStartEnterTransformation);
895             mEnterTransformation.compose(mFinishEnterTransformation);
896         }
897 
898         if (DEBUG_TRANSFORMS) Slog.v(TAG, "Final exit: " + mExitTransformation);
899         if (DEBUG_TRANSFORMS) Slog.v(TAG, "Final enter: " + mEnterTransformation);
900 
901         if (USE_CUSTOM_BLACK_FRAME) {
902             //mFrameTransformation.set(mRotateExitTransformation);
903             //mFrameTransformation.compose(mStartExitTransformation);
904             //mFrameTransformation.compose(mFinishExitTransformation);
905             mFrameTransformation.set(mRotateFrameTransformation);
906             mFrameTransformation.compose(mStartFrameTransformation);
907             mFrameTransformation.compose(mFinishFrameTransformation);
908             mFrameTransformation.getMatrix().preConcat(mFrameInitialMatrix);
909             if (DEBUG_TRANSFORMS) Slog.v(TAG, "Final frame: " + mFrameTransformation);
910         }
911 
912         final boolean more = (TWO_PHASE_ANIMATION
913                     && (mMoreStartEnter || mMoreStartExit || mMoreFinishEnter || mMoreFinishExit))
914                 || (USE_CUSTOM_BLACK_FRAME
915                         && (mMoreStartFrame || mMoreRotateFrame || mMoreFinishFrame))
916                 || mMoreRotateEnter || mMoreRotateExit
917                 || !mFinishAnimReady;
918 
919         mSnapshotFinalMatrix.setConcat(mExitTransformation.getMatrix(), mSnapshotInitialMatrix);
920 
921         if (DEBUG_STATE) Slog.v(TAG, "Step: more=" + more);
922 
923         return more;
924     }
925 
updateSurfacesInTransaction()926     void updateSurfacesInTransaction() {
927         if (!mStarted) {
928             return;
929         }
930 
931         if (mSurfaceControl != null) {
932             if (!mMoreStartExit && !mMoreFinishExit && !mMoreRotateExit) {
933                 if (DEBUG_STATE) Slog.v(TAG, "Exit animations done, hiding screenshot surface");
934                 mSurfaceControl.hide();
935             }
936         }
937 
938         if (mCustomBlackFrame != null) {
939             if (!mMoreStartFrame && !mMoreFinishFrame && !mMoreRotateFrame) {
940                 if (DEBUG_STATE) Slog.v(TAG, "Frame animations done, hiding black frame");
941                 mCustomBlackFrame.hide();
942             } else {
943                 mCustomBlackFrame.setMatrix(mFrameTransformation.getMatrix());
944             }
945         }
946 
947         if (mExitingBlackFrame != null) {
948             if (!mMoreStartExit && !mMoreFinishExit && !mMoreRotateExit) {
949                 if (DEBUG_STATE) Slog.v(TAG, "Frame animations done, hiding exiting frame");
950                 mExitingBlackFrame.hide();
951             } else {
952                 mExitFrameFinalMatrix.setConcat(mExitTransformation.getMatrix(), mFrameInitialMatrix);
953                 mExitingBlackFrame.setMatrix(mExitFrameFinalMatrix);
954                 if (mForceDefaultOrientation) {
955                     mExitingBlackFrame.setAlpha(mExitTransformation.getAlpha());
956                 }
957             }
958         }
959 
960         if (mEnteringBlackFrame != null) {
961             if (!mMoreStartEnter && !mMoreFinishEnter && !mMoreRotateEnter) {
962                 if (DEBUG_STATE) Slog.v(TAG, "Frame animations done, hiding entering frame");
963                 mEnteringBlackFrame.hide();
964             } else {
965                 mEnteringBlackFrame.setMatrix(mEnterTransformation.getMatrix());
966             }
967         }
968 
969         setSnapshotTransformInTransaction(mSnapshotFinalMatrix, mExitTransformation.getAlpha());
970     }
971 
stepAnimationLocked(long now)972     public boolean stepAnimationLocked(long now) {
973         if (!hasAnimations()) {
974             if (DEBUG_STATE) Slog.v(TAG, "Step: no animations running");
975             mFinishAnimReady = false;
976             return false;
977         }
978 
979         if (!mAnimRunning) {
980             if (DEBUG_STATE) Slog.v(TAG, "Step: starting start, finish, rotate");
981             if (TWO_PHASE_ANIMATION) {
982                 if (mStartEnterAnimation != null) {
983                     mStartEnterAnimation.setStartTime(now);
984                 }
985                 if (mStartExitAnimation != null) {
986                     mStartExitAnimation.setStartTime(now);
987                 }
988                 if (mFinishEnterAnimation != null) {
989                     mFinishEnterAnimation.setStartTime(0);
990                 }
991                 if (mFinishExitAnimation != null) {
992                     mFinishExitAnimation.setStartTime(0);
993                 }
994             }
995             if (USE_CUSTOM_BLACK_FRAME) {
996                 if (mStartFrameAnimation != null) {
997                     mStartFrameAnimation.setStartTime(now);
998                 }
999                 if (mFinishFrameAnimation != null) {
1000                     mFinishFrameAnimation.setStartTime(0);
1001                 }
1002                 if (mRotateFrameAnimation != null) {
1003                     mRotateFrameAnimation.setStartTime(now);
1004                 }
1005             }
1006             if (mRotateEnterAnimation != null) {
1007                 mRotateEnterAnimation.setStartTime(now);
1008             }
1009             if (mRotateExitAnimation != null) {
1010                 mRotateExitAnimation.setStartTime(now);
1011             }
1012             mAnimRunning = true;
1013             mHalfwayPoint = now + mRotateEnterAnimation.getDuration() / 2;
1014         }
1015 
1016         return stepAnimation(now);
1017     }
1018 
getEnterTransformation()1019     public Transformation getEnterTransformation() {
1020         return mEnterTransformation;
1021     }
1022 }
1023