diff options
| author | 2020-03-20 18:28:11 +0800 | |
|---|---|---|
| committer | 2020-03-31 10:59:55 +0000 | |
| commit | fc246d16466053eac5ef83ca310f40265843ef9c (patch) | |
| tree | 4530a4780d024c51829c5df2ac62ec7497b6f443 | |
| parent | 884aa6afe156ec88cfc89e9d5dac4480223c9495 (diff) | |
Cancel animation in previous rotation
The local animations are cancelled if fixed rotation transform is
going to be finished to perform seamless display rotation. That
avoids a case: a closing animation in landscape moved outside the
bottom of the screen, and it still has unfinished alpha animation
(e.g. wallpaper_open_exit is 250+167=417ms which is longer than
than wallpaper_open_enter 225ms). Then if the display is rotated
to portrait, the animation out of view becomes visible.
The reason not waiting for all animations to finish is that will
increase the latency to rotate display, and touching won't work
in the waiting state because the rotation of top activity (rotated
surface) is different than the display.
Bug: 151709552
Test: atest DisplayContentTests#testApplyTopFixedRotationTransform
Test: Enable fixed_rotation_transform, return from a task in
different orientation to home with default animation.
Change-Id: I7c268de1f9061a7ac11fc42a70f02000faba7106
| -rw-r--r-- | services/core/java/com/android/server/wm/DisplayContent.java | 39 | ||||
| -rw-r--r-- | services/tests/wmtests/src/com/android/server/wm/DisplayContentTests.java | 10 |
2 files changed, 41 insertions, 8 deletions
diff --git a/services/core/java/com/android/server/wm/DisplayContent.java b/services/core/java/com/android/server/wm/DisplayContent.java index 4a7edee7beac..4ccc07efba8e 100644 --- a/services/core/java/com/android/server/wm/DisplayContent.java +++ b/services/core/java/com/android/server/wm/DisplayContent.java @@ -5176,14 +5176,7 @@ class DisplayContent extends WindowContainer<DisplayContent.DisplayChildWindowCo final int currRotation = currOverrideConfig.windowConfiguration.getRotation(); final int overrideRotation = overrideConfiguration.windowConfiguration.getRotation(); if (currRotation != ROTATION_UNDEFINED && currRotation != overrideRotation) { - if (mFixedRotationLaunchingApp != null) { - mFixedRotationLaunchingApp.clearFixedRotationTransform( - () -> applyRotation(currRotation, overrideRotation)); - // Clear the record because the display will sync to current rotation. - mFixedRotationLaunchingApp = null; - } else { - applyRotation(currRotation, overrideRotation); - } + applyRotationAndClearFixedRotation(currRotation, overrideRotation); } mCurrentOverrideConfigurationChanges = currOverrideConfig.diff(overrideConfiguration); super.onRequestedOverrideConfigurationChanged(overrideConfiguration); @@ -5193,6 +5186,36 @@ class DisplayContent extends WindowContainer<DisplayContent.DisplayChildWindowCo ActivityTaskManagerService.LAYOUT_REASON_CONFIG_CHANGED); } + /** + * If the launching rotated activity ({@link #mFixedRotationLaunchingApp}) is null, it simply + * applies the rotation to display. Otherwise because the activity has shown as rotated, the + * fixed rotation transform also needs to be cleared to make sure the rotated activity fits + * the display naturally. + */ + private void applyRotationAndClearFixedRotation(int oldRotation, int newRotation) { + if (mFixedRotationLaunchingApp == null) { + applyRotation(oldRotation, newRotation); + return; + } + + // The display may be about to rotate seamlessly, and the animation of closing apps may + // still animate in old rotation. So make sure the outdated animation won't show on the + // rotated display. + mTaskContainers.forAllActivities(a -> { + if (a.nowVisible && a != mFixedRotationLaunchingApp + && a.getWindowConfiguration().getRotation() != newRotation) { + final WindowContainer<?> w = a.getAnimatingContainer(); + if (w != null) { + w.cancelAnimation(); + } + } + }); + + mFixedRotationLaunchingApp.clearFixedRotationTransform( + () -> applyRotation(oldRotation, newRotation)); + mFixedRotationLaunchingApp = null; + } + /** Checks whether the given activity is in size compatibility mode and notifies the change. */ void handleActivitySizeCompatModeIfNeeded(ActivityRecord r) { if (!r.isState(RESUMED) || r.getWindowingMode() != WINDOWING_MODE_FULLSCREEN) { diff --git a/services/tests/wmtests/src/com/android/server/wm/DisplayContentTests.java b/services/tests/wmtests/src/com/android/server/wm/DisplayContentTests.java index 5b96c4372abc..c1b5be2ef039 100644 --- a/services/tests/wmtests/src/com/android/server/wm/DisplayContentTests.java +++ b/services/tests/wmtests/src/com/android/server/wm/DisplayContentTests.java @@ -56,6 +56,7 @@ import static com.android.dx.mockito.inline.extended.ExtendedMockito.same; import static com.android.dx.mockito.inline.extended.ExtendedMockito.spyOn; import static com.android.dx.mockito.inline.extended.ExtendedMockito.times; import static com.android.dx.mockito.inline.extended.ExtendedMockito.verify; +import static com.android.server.wm.SurfaceAnimator.ANIMATION_TYPE_APP_TRANSITION; import static com.android.server.wm.WindowContainer.POSITION_TOP; import static com.android.server.wm.WindowManagerService.UPDATE_FOCUS_NORMAL; @@ -1006,6 +1007,13 @@ public class DisplayContentTests extends WindowTestsBase { mDisplayContent.computeScreenConfiguration(config); mDisplayContent.onRequestedOverrideConfigurationChanged(config); + final ActivityRecord closingApp = new ActivityTestsBase.StackBuilder(mWm.mRoot) + .setDisplay(mDisplayContent).setOnTop(false).build().getTopMostActivity(); + closingApp.nowVisible = true; + closingApp.startAnimation(closingApp.getPendingTransaction(), mock(AnimationAdapter.class), + false /* hidden */, ANIMATION_TYPE_APP_TRANSITION); + assertTrue(closingApp.isAnimating()); + final ActivityRecord app = mAppWindow.mActivityRecord; mDisplayContent.prepareAppTransition(WindowManager.TRANSIT_ACTIVITY_OPEN, false /* alwaysKeepCurrent */); @@ -1033,6 +1041,8 @@ public class DisplayContentTests extends WindowTestsBase { mDisplayContent.mAppTransition.notifyAppTransitionFinishedLocked(app.token); + // The animation in old rotation should be cancelled. + assertFalse(closingApp.isAnimating()); // The display should be rotated after the launch is finished. assertFalse(app.hasFixedRotationTransform()); assertEquals(config90.orientation, mDisplayContent.getConfiguration().orientation); |