| /* |
| * Copyright (C) 2018 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.server.wm; |
| |
| import static android.app.WindowConfiguration.ACTIVITY_TYPE_STANDARD; |
| import static android.app.WindowConfiguration.WINDOWING_MODE_FREEFORM; |
| import static android.app.WindowConfiguration.WINDOWING_MODE_FULLSCREEN; |
| import static android.app.WindowConfiguration.WINDOWING_MODE_MULTI_WINDOW; |
| import static android.view.WindowManager.LayoutParams.FLAG_SHOW_WALLPAPER; |
| import static android.view.WindowManager.LayoutParams.TYPE_BASE_APPLICATION; |
| import static android.view.WindowManager.TRANSIT_CHANGE; |
| import static android.view.WindowManager.TRANSIT_CLOSE; |
| import static android.view.WindowManager.TRANSIT_OLD_ACTIVITY_OPEN; |
| import static android.view.WindowManager.TRANSIT_OLD_KEYGUARD_UNOCCLUDE; |
| import static android.view.WindowManager.TRANSIT_OLD_TASK_CHANGE_WINDOWING_MODE; |
| import static android.view.WindowManager.TRANSIT_OLD_TASK_FRAGMENT_CHANGE; |
| import static android.view.WindowManager.TRANSIT_OLD_TASK_FRAGMENT_CLOSE; |
| import static android.view.WindowManager.TRANSIT_OLD_TASK_FRAGMENT_OPEN; |
| import static android.view.WindowManager.TRANSIT_OLD_TASK_OPEN; |
| import static android.view.WindowManager.TRANSIT_OPEN; |
| import static android.view.WindowManager.TRANSIT_TO_FRONT; |
| |
| import static com.android.dx.mockito.inline.extended.ExtendedMockito.doNothing; |
| import static com.android.dx.mockito.inline.extended.ExtendedMockito.doReturn; |
| import static com.android.dx.mockito.inline.extended.ExtendedMockito.spyOn; |
| import static com.android.server.wm.WindowContainer.POSITION_BOTTOM; |
| |
| import static org.junit.Assert.assertEquals; |
| import static org.junit.Assert.assertFalse; |
| import static org.junit.Assert.assertNull; |
| import static org.junit.Assert.assertTrue; |
| import static org.junit.Assert.fail; |
| import static org.mockito.ArgumentMatchers.any; |
| import static org.mockito.ArgumentMatchers.anyInt; |
| import static org.mockito.Mockito.clearInvocations; |
| import static org.mockito.Mockito.mock; |
| import static org.mockito.Mockito.never; |
| import static org.mockito.Mockito.verify; |
| |
| import android.annotation.Nullable; |
| import android.gui.DropInputMode; |
| import android.os.Binder; |
| import android.os.IBinder; |
| import android.os.RemoteException; |
| import android.platform.test.annotations.Presubmit; |
| import android.util.ArraySet; |
| import android.view.IRemoteAnimationFinishedCallback; |
| import android.view.IRemoteAnimationRunner; |
| import android.view.RemoteAnimationAdapter; |
| import android.view.RemoteAnimationDefinition; |
| import android.view.RemoteAnimationTarget; |
| import android.view.WindowManager; |
| import android.window.ITaskFragmentOrganizer; |
| import android.window.TaskFragmentOrganizer; |
| |
| import androidx.test.filters.SmallTest; |
| |
| import org.junit.Before; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| |
| /** |
| * Build/Install/Run: |
| * atest WmTests:AppTransitionControllerTest |
| */ |
| @SmallTest |
| @Presubmit |
| @RunWith(WindowTestRunner.class) |
| public class AppTransitionControllerTest extends WindowTestsBase { |
| |
| private AppTransitionController mAppTransitionController; |
| |
| @Before |
| public void setUp() throws Exception { |
| mAppTransitionController = new AppTransitionController(mWm, mDisplayContent); |
| } |
| |
| @Test |
| public void testSkipOccludedActivityCloseTransition() { |
| final ActivityRecord behind = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| final ActivityRecord topOpening = createActivityRecord(behind.getTask()); |
| topOpening.setOccludesParent(true); |
| topOpening.setVisible(true); |
| |
| mDisplayContent.prepareAppTransition(TRANSIT_OPEN); |
| mDisplayContent.prepareAppTransition(TRANSIT_CLOSE); |
| mDisplayContent.mClosingApps.add(behind); |
| |
| assertEquals(WindowManager.TRANSIT_OLD_UNSET, |
| AppTransitionController.getTransitCompatType(mDisplayContent.mAppTransition, |
| mDisplayContent.mOpeningApps, mDisplayContent.mClosingApps, |
| mDisplayContent.mChangingContainers, null, null, false)); |
| } |
| |
| @Test |
| public void testClearTaskSkipAppExecuteTransition() { |
| final ActivityRecord behind = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| final Task task = behind.getTask(); |
| final ActivityRecord top = createActivityRecord(task); |
| top.setState(ActivityRecord.State.RESUMED, "test"); |
| behind.setState(ActivityRecord.State.STARTED, "test"); |
| behind.mVisibleRequested = true; |
| |
| task.removeActivities("test", false /* excludingTaskOverlay */); |
| assertFalse(mDisplayContent.mAppTransition.isReady()); |
| } |
| |
| @Test |
| public void testTranslucentOpen() { |
| final ActivityRecord behind = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| final ActivityRecord translucentOpening = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| doReturn(false).when(translucentOpening).fillsParent(); |
| translucentOpening.setVisible(false); |
| mDisplayContent.prepareAppTransition(TRANSIT_OPEN); |
| mDisplayContent.mOpeningApps.add(behind); |
| mDisplayContent.mOpeningApps.add(translucentOpening); |
| |
| assertEquals(WindowManager.TRANSIT_OLD_TRANSLUCENT_ACTIVITY_OPEN, |
| AppTransitionController.getTransitCompatType(mDisplayContent.mAppTransition, |
| mDisplayContent.mOpeningApps, mDisplayContent.mClosingApps, |
| mDisplayContent.mChangingContainers, null, null, false)); |
| } |
| |
| @Test |
| public void testTranslucentClose() { |
| final ActivityRecord behind = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| final ActivityRecord translucentClosing = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| doReturn(false).when(translucentClosing).fillsParent(); |
| mDisplayContent.prepareAppTransition(TRANSIT_CLOSE); |
| mDisplayContent.mClosingApps.add(translucentClosing); |
| assertEquals(WindowManager.TRANSIT_OLD_TRANSLUCENT_ACTIVITY_CLOSE, |
| AppTransitionController.getTransitCompatType(mDisplayContent.mAppTransition, |
| mDisplayContent.mOpeningApps, mDisplayContent.mClosingApps, |
| mDisplayContent.mChangingContainers, null, null, false)); |
| } |
| |
| @Test |
| public void testChangeIsNotOverwritten() { |
| final ActivityRecord behind = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| final ActivityRecord translucentOpening = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| translucentOpening.setOccludesParent(false); |
| translucentOpening.setVisible(false); |
| mDisplayContent.prepareAppTransition(TRANSIT_CHANGE); |
| mDisplayContent.mOpeningApps.add(behind); |
| mDisplayContent.mOpeningApps.add(translucentOpening); |
| mDisplayContent.mChangingContainers.add(translucentOpening.getTask()); |
| assertEquals(TRANSIT_OLD_TASK_CHANGE_WINDOWING_MODE, |
| AppTransitionController.getTransitCompatType(mDisplayContent.mAppTransition, |
| mDisplayContent.mOpeningApps, mDisplayContent.mClosingApps, |
| mDisplayContent.mChangingContainers, null, null, false)); |
| } |
| |
| @Test |
| public void testTransitWithinTask() { |
| final ActivityRecord opening = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FREEFORM, ACTIVITY_TYPE_STANDARD); |
| opening.setOccludesParent(false); |
| final ActivityRecord closing = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FREEFORM, ACTIVITY_TYPE_STANDARD); |
| closing.setOccludesParent(false); |
| final Task task = opening.getTask(); |
| mDisplayContent.mOpeningApps.add(opening); |
| mDisplayContent.mClosingApps.add(closing); |
| assertFalse(mAppTransitionController.isTransitWithinTask(TRANSIT_OLD_ACTIVITY_OPEN, task)); |
| closing.getTask().removeChild(closing); |
| task.addChild(closing, 0); |
| assertTrue(mAppTransitionController.isTransitWithinTask(TRANSIT_OLD_ACTIVITY_OPEN, task)); |
| assertFalse(mAppTransitionController.isTransitWithinTask(TRANSIT_OLD_TASK_OPEN, task)); |
| } |
| |
| |
| @Test |
| public void testIntraWallpaper_open() { |
| final ActivityRecord opening = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| opening.setVisible(false); |
| final WindowManager.LayoutParams attrOpening = new WindowManager.LayoutParams( |
| TYPE_BASE_APPLICATION); |
| attrOpening.setTitle("WallpaperOpening"); |
| attrOpening.flags |= FLAG_SHOW_WALLPAPER; |
| final TestWindowState appWindowOpening = createWindowState(attrOpening, opening); |
| opening.addWindow(appWindowOpening); |
| |
| final ActivityRecord closing = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| final WindowManager.LayoutParams attrClosing = new WindowManager.LayoutParams( |
| TYPE_BASE_APPLICATION); |
| attrOpening.setTitle("WallpaperClosing"); |
| attrClosing.flags |= FLAG_SHOW_WALLPAPER; |
| final TestWindowState appWindowClosing = createWindowState(attrClosing, closing); |
| closing.addWindow(appWindowClosing); |
| |
| mDisplayContent.prepareAppTransition(TRANSIT_OPEN); |
| mDisplayContent.mOpeningApps.add(opening); |
| mDisplayContent.mClosingApps.add(closing); |
| |
| assertEquals(WindowManager.TRANSIT_OLD_WALLPAPER_INTRA_OPEN, |
| AppTransitionController.getTransitCompatType(mDisplayContent.mAppTransition, |
| mDisplayContent.mOpeningApps, mDisplayContent.mClosingApps, |
| mDisplayContent.mChangingContainers, appWindowClosing, null, false)); |
| } |
| |
| @Test |
| public void testIntraWallpaper_toFront() { |
| final ActivityRecord opening = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| opening.setVisible(false); |
| final WindowManager.LayoutParams attrOpening = new WindowManager.LayoutParams( |
| TYPE_BASE_APPLICATION); |
| attrOpening.setTitle("WallpaperOpening"); |
| attrOpening.flags |= FLAG_SHOW_WALLPAPER; |
| final TestWindowState appWindowOpening = createWindowState(attrOpening, opening); |
| opening.addWindow(appWindowOpening); |
| |
| final ActivityRecord closing = createActivityRecord(mDisplayContent, |
| WINDOWING_MODE_FULLSCREEN, ACTIVITY_TYPE_STANDARD); |
| final WindowManager.LayoutParams attrClosing = new WindowManager.LayoutParams( |
| TYPE_BASE_APPLICATION); |
| attrOpening.setTitle("WallpaperClosing"); |
| attrClosing.flags |= FLAG_SHOW_WALLPAPER; |
| final TestWindowState appWindowClosing = createWindowState(attrClosing, closing); |
| closing.addWindow(appWindowClosing); |
| |
| mDisplayContent.prepareAppTransition(TRANSIT_TO_FRONT); |
| mDisplayContent.mOpeningApps.add(opening); |
| mDisplayContent.mClosingApps.add(closing); |
| |
| assertEquals(WindowManager.TRANSIT_OLD_WALLPAPER_INTRA_OPEN, |
| AppTransitionController.getTransitCompatType(mDisplayContent.mAppTransition, |
| mDisplayContent.mOpeningApps, mDisplayContent.mClosingApps, |
| mDisplayContent.mChangingContainers, appWindowClosing, null, false)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_visibilityAlreadyUpdated() { |
| // [DisplayContent] -+- [Task1] - [ActivityRecord1] (opening, visible) |
| // +- [Task2] - [ActivityRecord2] (closing, invisible) |
| final ActivityRecord activity1 = createActivityRecord(mDisplayContent); |
| |
| final ActivityRecord activity2 = createActivityRecord(mDisplayContent); |
| activity2.setVisible(false); |
| activity2.mVisibleRequested = false; |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity2); |
| |
| // No animation, since visibility of the opening and closing apps are already updated |
| // outside of AppTransition framework. |
| assertEquals( |
| new ArraySet<>(), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals( |
| new ArraySet<>(), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_visibilityAlreadyUpdated_butForcedTransitionRequested() { |
| // [DisplayContent] -+- [Task1] - [ActivityRecord1] (closing, invisible) |
| // +- [Task2] - [ActivityRecord2] (opening, visible) |
| final ActivityRecord activity1 = createActivityRecord(mDisplayContent); |
| activity1.setVisible(true); |
| activity1.mVisibleRequested = true; |
| activity1.mRequestForceTransition = true; |
| |
| final ActivityRecord activity2 = createActivityRecord(mDisplayContent); |
| activity2.setVisible(false); |
| activity2.mVisibleRequested = false; |
| activity2.mRequestForceTransition = true; |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity2); |
| |
| // The visibility are already updated, but since forced transition is requested, it will |
| // be included. |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity1.getRootTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity2.getRootTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_exitingBeforeTransition() { |
| // Create another non-empty task so the animation target won't promote to task display area. |
| createActivityRecord(mDisplayContent); |
| final ActivityRecord activity = createActivityRecord(mDisplayContent); |
| activity.setVisible(false); |
| activity.mIsExiting = true; |
| |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity); |
| |
| // Animate closing apps even if it's not visible when it is exiting before we had a chance |
| // to play the transition animation. |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity.getRootTask()}), |
| AppTransitionController.getAnimationTargets( |
| new ArraySet<>(), closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testExitAnimationDone_beforeAppTransition() { |
| final Task task = createTask(mDisplayContent); |
| final WindowState win = createAppWindow(task, ACTIVITY_TYPE_STANDARD, "Win"); |
| spyOn(win); |
| win.mAnimatingExit = true; |
| mDisplayContent.mAppTransition.setTimeout(); |
| mDisplayContent.mAppTransitionController.handleAppTransitionReady(); |
| |
| verify(win).onExitAnimationDone(); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_windowsAreBeingReplaced() { |
| // [DisplayContent] -+- [Task1] - [ActivityRecord1] (opening, visible) |
| // +- [AppWindow1] (being-replaced) |
| // +- [Task2] - [ActivityRecord2] (closing, invisible) |
| // +- [AppWindow2] (being-replaced) |
| final ActivityRecord activity1 = createActivityRecord(mDisplayContent); |
| final WindowManager.LayoutParams attrs = new WindowManager.LayoutParams( |
| TYPE_BASE_APPLICATION); |
| attrs.setTitle("AppWindow1"); |
| final TestWindowState appWindow1 = createWindowState(attrs, activity1); |
| appWindow1.mWillReplaceWindow = true; |
| activity1.addWindow(appWindow1); |
| |
| final ActivityRecord activity2 = createActivityRecord(mDisplayContent); |
| activity2.setVisible(false); |
| activity2.mVisibleRequested = false; |
| attrs.setTitle("AppWindow2"); |
| final TestWindowState appWindow2 = createWindowState(attrs, activity2); |
| appWindow2.mWillReplaceWindow = true; |
| activity2.addWindow(appWindow2); |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity2); |
| |
| // Animate opening apps even if it's already visible in case its windows are being replaced. |
| // Don't animate closing apps if it's already invisible even though its windows are being |
| // replaced. |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity1.getRootTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_openingClosingInDifferentTask() { |
| // [DisplayContent] -+- [Task1] -+- [ActivityRecord1] (opening, invisible) |
| // | +- [ActivityRecord2] (invisible) |
| // | |
| // +- [Task2] -+- [ActivityRecord3] (closing, visible) |
| // +- [ActivityRecord4] (invisible) |
| final ActivityRecord activity1 = createActivityRecord(mDisplayContent); |
| activity1.setVisible(false); |
| activity1.mVisibleRequested = true; |
| final ActivityRecord activity2 = createActivityRecord(mDisplayContent, |
| activity1.getTask()); |
| activity2.setVisible(false); |
| activity2.mVisibleRequested = false; |
| |
| final ActivityRecord activity3 = createActivityRecord(mDisplayContent); |
| final ActivityRecord activity4 = createActivityRecord(mDisplayContent, |
| activity3.getTask()); |
| activity4.setVisible(false); |
| activity4.mVisibleRequested = false; |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity3); |
| |
| // Promote animation targets to root Task level. Invisible ActivityRecords don't affect |
| // promotion decision. |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity1.getRootTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity3.getRootTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_openingClosingInSameTask() { |
| // [DisplayContent] - [Task] -+- [ActivityRecord1] (opening, invisible) |
| // +- [ActivityRecord2] (closing, visible) |
| final ActivityRecord activity1 = createActivityRecord(mDisplayContent); |
| activity1.setVisible(false); |
| activity1.mVisibleRequested = true; |
| final ActivityRecord activity2 = createActivityRecord(mDisplayContent, |
| activity1.getTask()); |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity2); |
| |
| // Don't promote an animation target to Task level, since the same task contains both |
| // opening and closing app. |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity1}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity2}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_animateOnlyTranslucentApp() { |
| // [DisplayContent] -+- [Task1] -+- [ActivityRecord1] (opening, invisible) |
| // | +- [ActivityRecord2] (visible) |
| // | |
| // +- [Task2] -+- [ActivityRecord3] (closing, visible) |
| // +- [ActivityRecord4] (visible) |
| |
| final ActivityRecord activity1 = createActivityRecord(mDisplayContent); |
| activity1.setVisible(false); |
| activity1.mVisibleRequested = true; |
| activity1.setOccludesParent(false); |
| |
| final ActivityRecord activity2 = createActivityRecord(mDisplayContent, |
| activity1.getTask()); |
| |
| final ActivityRecord activity3 = createActivityRecord(mDisplayContent); |
| activity3.setOccludesParent(false); |
| final ActivityRecord activity4 = createActivityRecord(mDisplayContent, |
| activity3.getTask()); |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity3); |
| |
| // Don't promote an animation target to Task level, since opening (closing) app is |
| // translucent and is displayed over other non-animating app. |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity1}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity3}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_animateTranslucentAndOpaqueApps() { |
| // [DisplayContent] -+- [Task1] -+- [ActivityRecord1] (opening, invisible) |
| // | +- [ActivityRecord2] (opening, invisible) |
| // | |
| // +- [Task2] -+- [ActivityRecord3] (closing, visible) |
| // +- [ActivityRecord4] (closing, visible) |
| |
| final ActivityRecord activity1 = createActivityRecord(mDisplayContent); |
| activity1.setVisible(false); |
| activity1.mVisibleRequested = true; |
| activity1.setOccludesParent(false); |
| |
| final ActivityRecord activity2 = createActivityRecord(mDisplayContent, |
| activity1.getTask()); |
| activity2.setVisible(false); |
| activity2.mVisibleRequested = true; |
| |
| final ActivityRecord activity3 = createActivityRecord(mDisplayContent); |
| activity3.setOccludesParent(false); |
| final ActivityRecord activity4 = createActivityRecord(mDisplayContent, |
| activity3.getTask()); |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| opening.add(activity2); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity3); |
| closing.add(activity4); |
| |
| // Promote animation targets to TaskStack level even though opening (closing) app is |
| // translucent as long as all visible siblings animate at the same time. |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity1.getRootTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity3.getRootTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_taskContainsMultipleTasks() { |
| // [DisplayContent] - [Task] -+- [Task1] - [ActivityRecord1] (opening, invisible) |
| // +- [Task2] - [ActivityRecord2] (closing, visible) |
| final Task parentTask = createTask(mDisplayContent); |
| final ActivityRecord activity1 = createActivityRecordWithParentTask(parentTask); |
| activity1.setVisible(false); |
| activity1.mVisibleRequested = true; |
| final ActivityRecord activity2 = createActivityRecordWithParentTask(parentTask); |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity2); |
| |
| // Promote animation targets up to Task level, not beyond. |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity1.getTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity2.getTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_splitScreenOpening() { |
| // [DisplayContent] - [Task] -+- [split task 1] -+- [Task1] - [AR1] (opening, invisible) |
| // +- [split task 2] -+- [Task2] - [AR2] (opening, invisible) |
| final Task singleTopRoot = createTask(mDisplayContent); |
| final TaskBuilder builder = new TaskBuilder(mSupervisor) |
| .setWindowingMode(WINDOWING_MODE_MULTI_WINDOW) |
| .setParentTaskFragment(singleTopRoot) |
| .setCreatedByOrganizer(true); |
| final Task splitRoot1 = builder.build(); |
| final Task splitRoot2 = builder.build(); |
| splitRoot1.setAdjacentTaskFragment(splitRoot2, false /* moveTogether */); |
| final ActivityRecord activity1 = createActivityRecordWithParentTask(splitRoot1); |
| activity1.setVisible(false); |
| activity1.mVisibleRequested = true; |
| final ActivityRecord activity2 = createActivityRecordWithParentTask(splitRoot2); |
| activity2.setVisible(false); |
| activity2.mVisibleRequested = true; |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| opening.add(activity2); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| |
| // Promote animation targets up to Task level, not beyond. |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{splitRoot1, splitRoot2}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_openingClosingTaskFragment() { |
| // [DefaultTDA] - [Task] -+- [TaskFragment1] - [ActivityRecord1] (opening, invisible) |
| // +- [TaskFragment2] - [ActivityRecord2] (closing, visible) |
| final Task parentTask = createTask(mDisplayContent); |
| final TaskFragment taskFragment1 = createTaskFragmentWithParentTask(parentTask, |
| false /* createEmbeddedTask */); |
| final ActivityRecord activity1 = taskFragment1.getTopMostActivity(); |
| activity1.setVisible(false); |
| activity1.mVisibleRequested = true; |
| |
| final TaskFragment taskFragment2 = createTaskFragmentWithParentTask(parentTask, |
| false /* createEmbeddedTask */); |
| final ActivityRecord activity2 = taskFragment2.getTopMostActivity(); |
| activity2.setVisible(true); |
| activity2.mVisibleRequested = false; |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity2); |
| |
| // Promote animation targets up to TaskFragment level, not beyond. |
| assertEquals(new ArraySet<>(new WindowContainer[]{taskFragment1}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals(new ArraySet<>(new WindowContainer[]{taskFragment2}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_openingClosingTaskFragmentWithEmbeddedTask() { |
| // [DefaultTDA] - [Task] -+- [TaskFragment1] - [ActivityRecord1] (opening, invisible) |
| // +- [TaskFragment2] - [ActivityRecord2] (closing, visible) |
| final Task parentTask = createTask(mDisplayContent); |
| final TaskFragment taskFragment1 = createTaskFragmentWithParentTask(parentTask, |
| true /* createEmbeddedTask */); |
| final ActivityRecord activity1 = taskFragment1.getTopMostActivity(); |
| activity1.setVisible(false); |
| activity1.mVisibleRequested = true; |
| |
| final TaskFragment taskFragment2 = createTaskFragmentWithParentTask(parentTask, |
| true /* createEmbeddedTask */); |
| final ActivityRecord activity2 = taskFragment2.getTopMostActivity(); |
| activity2.setVisible(true); |
| activity2.mVisibleRequested = false; |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity2); |
| |
| // Promote animation targets up to TaskFragment level, not beyond. |
| assertEquals(new ArraySet<>(new WindowContainer[]{taskFragment1}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals(new ArraySet<>(new WindowContainer[]{taskFragment2}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_openingTheOnlyTaskFragmentInTask() { |
| // [DefaultTDA] -+- [Task1] - [TaskFragment1] - [ActivityRecord1] (opening, invisible) |
| // +- [Task2] - [ActivityRecord2] (closing, visible) |
| final Task task1 = createTask(mDisplayContent); |
| final TaskFragment taskFragment1 = createTaskFragmentWithParentTask(task1, |
| false /* createEmbeddedTask */); |
| final ActivityRecord activity1 = taskFragment1.getTopMostActivity(); |
| activity1.setVisible(false); |
| activity1.mVisibleRequested = true; |
| |
| final ActivityRecord activity2 = createActivityRecord(mDisplayContent); |
| activity2.setVisible(true); |
| activity2.mVisibleRequested = false; |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity2); |
| |
| // Promote animation targets up to leaf Task level because there's only one TaskFragment in |
| // the Task. |
| assertEquals(new ArraySet<>(new WindowContainer[]{task1}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals(new ArraySet<>(new WindowContainer[]{activity2.getTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_closingTheOnlyTaskFragmentInTask() { |
| // [DefaultTDA] -+- [Task1] - [TaskFragment1] - [ActivityRecord1] (closing, visible) |
| // +- [Task2] - [ActivityRecord2] (opening, invisible) |
| final Task task1 = createTask(mDisplayContent); |
| final TaskFragment taskFragment1 = createTaskFragmentWithParentTask(task1, |
| false /* createEmbeddedTask */); |
| final ActivityRecord activity1 = taskFragment1.getTopMostActivity(); |
| activity1.setVisible(true); |
| activity1.mVisibleRequested = false; |
| |
| final ActivityRecord activity2 = createActivityRecord(mDisplayContent); |
| activity2.setVisible(false); |
| activity2.mVisibleRequested = true; |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity2); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity1); |
| |
| // Promote animation targets up to leaf Task level because there's only one TaskFragment in |
| // the Task. |
| assertEquals(new ArraySet<>(new WindowContainer[]{activity2.getTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals(new ArraySet<>(new WindowContainer[]{task1}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| @Test |
| public void testGetAnimationTargets_embeddedTask() { |
| // [DisplayContent] -+- [Task1] - [ActivityRecord1] (opening, invisible) |
| // +- [Task2] (embedded) - [ActivityRecord2] (opening, invisible) |
| final ActivityRecord activity1 = createActivityRecord(mDisplayContent); |
| activity1.setVisible(false); |
| activity1.mVisibleRequested = true; |
| |
| final Task task2 = createTask(mDisplayContent); |
| task2.mRemoveWithTaskOrganizer = true; |
| final ActivityRecord activity2 = createActivityRecord(task2); |
| activity2.setVisible(false); |
| activity2.mVisibleRequested = true; |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| opening.add(activity2); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| |
| // No animation on the embedded task. |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity1.getTask()}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals( |
| new ArraySet<>(), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| |
| @Test |
| public void testGetAnimationTargets_activityInEmbeddedTask() { |
| // [DisplayContent] - [Task] (embedded)-+- [ActivityRecord1] (opening, invisible) |
| // +- [ActivityRecord2] (closing, visible) |
| final Task task = createTask(mDisplayContent); |
| task.mRemoveWithTaskOrganizer = true; |
| |
| final ActivityRecord activity1 = createActivityRecord(task); |
| activity1.setVisible(false); |
| activity1.mVisibleRequested = true; |
| final ActivityRecord activity2 = createActivityRecord(task); |
| |
| final ArraySet<ActivityRecord> opening = new ArraySet<>(); |
| opening.add(activity1); |
| final ArraySet<ActivityRecord> closing = new ArraySet<>(); |
| closing.add(activity2); |
| |
| // Even though embedded task itself doesn't animate, activities in an embedded task |
| // animate. |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity1}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, true /* visible */)); |
| assertEquals( |
| new ArraySet<>(new WindowContainer[]{activity2}), |
| AppTransitionController.getAnimationTargets( |
| opening, closing, false /* visible */)); |
| } |
| |
| static class TestRemoteAnimationRunner implements IRemoteAnimationRunner { |
| private IRemoteAnimationFinishedCallback mFinishedCallback; |
| |
| @Override |
| public void onAnimationStart(int transit, RemoteAnimationTarget[] apps, |
| RemoteAnimationTarget[] wallpapers, RemoteAnimationTarget[] nonApps, |
| IRemoteAnimationFinishedCallback finishedCallback) throws RemoteException { |
| mFinishedCallback = finishedCallback; |
| } |
| |
| @Override |
| public void onAnimationCancelled(boolean isKeyguardOccluded) throws RemoteException { |
| mFinishedCallback = null; |
| } |
| |
| @Override |
| public IBinder asBinder() { |
| return new Binder(); |
| } |
| |
| boolean isAnimationStarted() { |
| return mFinishedCallback != null; |
| } |
| |
| void finishAnimation() { |
| try { |
| mFinishedCallback.onAnimationFinished(); |
| } catch (RemoteException e) { |
| fail(); |
| } |
| } |
| } |
| |
| @Test |
| public void testGetRemoteAnimationOverrideEmpty() { |
| final ActivityRecord activity = createActivityRecord(mDisplayContent); |
| assertNull(mAppTransitionController.getRemoteAnimationOverride(activity, |
| TRANSIT_OLD_ACTIVITY_OPEN, new ArraySet<Integer>())); |
| } |
| |
| @Test |
| public void testGetRemoteAnimationOverrideWindowContainer() { |
| final ActivityRecord activity = createActivityRecord(mDisplayContent); |
| final RemoteAnimationDefinition definition = new RemoteAnimationDefinition(); |
| final RemoteAnimationAdapter adapter = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| definition.addRemoteAnimation(TRANSIT_OLD_ACTIVITY_OPEN, adapter); |
| activity.registerRemoteAnimations(definition); |
| |
| assertEquals(adapter, |
| mAppTransitionController.getRemoteAnimationOverride( |
| activity, TRANSIT_OLD_ACTIVITY_OPEN, new ArraySet<Integer>())); |
| assertNull(mAppTransitionController.getRemoteAnimationOverride( |
| null, TRANSIT_OLD_ACTIVITY_OPEN, new ArraySet<Integer>())); |
| } |
| |
| @Test |
| public void testGetRemoteAnimationOverrideTransitionController() { |
| final ActivityRecord activity = createActivityRecord(mDisplayContent); |
| final RemoteAnimationDefinition definition = new RemoteAnimationDefinition(); |
| final RemoteAnimationAdapter adapter = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| definition.addRemoteAnimation(TRANSIT_OLD_ACTIVITY_OPEN, adapter); |
| mAppTransitionController.registerRemoteAnimations(definition); |
| |
| assertEquals(adapter, |
| mAppTransitionController.getRemoteAnimationOverride( |
| activity, TRANSIT_OLD_ACTIVITY_OPEN, new ArraySet<Integer>())); |
| assertEquals(adapter, |
| mAppTransitionController.getRemoteAnimationOverride( |
| null, TRANSIT_OLD_ACTIVITY_OPEN, new ArraySet<Integer>())); |
| } |
| |
| @Test |
| public void testGetRemoteAnimationOverrideBoth() { |
| final ActivityRecord activity = createActivityRecord(mDisplayContent); |
| final RemoteAnimationDefinition definition1 = new RemoteAnimationDefinition(); |
| final RemoteAnimationAdapter adapter1 = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| definition1.addRemoteAnimation(TRANSIT_OLD_ACTIVITY_OPEN, adapter1); |
| activity.registerRemoteAnimations(definition1); |
| |
| final RemoteAnimationDefinition definition2 = new RemoteAnimationDefinition(); |
| final RemoteAnimationAdapter adapter2 = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| definition2.addRemoteAnimation(TRANSIT_OLD_KEYGUARD_UNOCCLUDE, adapter2); |
| mAppTransitionController.registerRemoteAnimations(definition2); |
| |
| assertEquals(adapter2, |
| mAppTransitionController.getRemoteAnimationOverride( |
| activity, TRANSIT_OLD_KEYGUARD_UNOCCLUDE, new ArraySet<Integer>())); |
| assertEquals(adapter2, |
| mAppTransitionController.getRemoteAnimationOverride( |
| null, TRANSIT_OLD_KEYGUARD_UNOCCLUDE, new ArraySet<Integer>())); |
| } |
| |
| @Test |
| public void testGetRemoteAnimationOverrideWindowContainerHasPriority() { |
| final ActivityRecord activity = createActivityRecord(mDisplayContent); |
| final RemoteAnimationDefinition definition1 = new RemoteAnimationDefinition(); |
| final RemoteAnimationAdapter adapter1 = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| definition1.addRemoteAnimation(TRANSIT_OLD_ACTIVITY_OPEN, adapter1); |
| activity.registerRemoteAnimations(definition1); |
| |
| final RemoteAnimationDefinition definition2 = new RemoteAnimationDefinition(); |
| final RemoteAnimationAdapter adapter2 = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| definition2.addRemoteAnimation(TRANSIT_OLD_ACTIVITY_OPEN, adapter2); |
| mAppTransitionController.registerRemoteAnimations(definition2); |
| |
| assertEquals(adapter1, |
| mAppTransitionController.getRemoteAnimationOverride( |
| activity, TRANSIT_OLD_ACTIVITY_OPEN, new ArraySet<Integer>())); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_overrideWithEmbeddedActivity() { |
| final Task task = createTask(mDisplayContent); |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final TestRemoteAnimationRunner remoteAnimationRunner = new TestRemoteAnimationRunner(); |
| setupTaskFragmentRemoteAnimation(organizer, task.mTaskId, remoteAnimationRunner); |
| |
| // Create a TaskFragment with embedded activity. |
| final TaskFragment taskFragment = createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final ActivityRecord activity = taskFragment.getTopMostActivity(); |
| prepareActivityForAppTransition(activity); |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare and start transition. |
| prepareAndTriggerAppTransition(activity, null /* closingActivity */, taskFragment); |
| mWm.mAnimator.executeAfterPrepareSurfacesRunnables(); |
| |
| // Animation run by the remote handler. |
| assertTrue(remoteAnimationRunner.isAnimationStarted()); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_overrideWithNonEmbeddedActivity() { |
| final Task task = createTask(mDisplayContent); |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final TestRemoteAnimationRunner remoteAnimationRunner = new TestRemoteAnimationRunner(); |
| setupTaskFragmentRemoteAnimation(organizer, task.mTaskId, remoteAnimationRunner); |
| |
| // Closing non-embedded activity. |
| final ActivityRecord closingActivity = createActivityRecord(task); |
| prepareActivityForAppTransition(closingActivity); |
| // Opening TaskFragment with embedded activity. |
| final TaskFragment taskFragment = createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final ActivityRecord openingActivity = taskFragment.getTopMostActivity(); |
| prepareActivityForAppTransition(openingActivity); |
| task.effectiveUid = openingActivity.getUid(); |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare and start transition. |
| prepareAndTriggerAppTransition(openingActivity, closingActivity, taskFragment); |
| mWm.mAnimator.executeAfterPrepareSurfacesRunnables(); |
| |
| // Animation run by the remote handler. |
| assertTrue(remoteAnimationRunner.isAnimationStarted()); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_overrideEmbeddedActivityWithDiffUid() { |
| final Task task = createTask(mDisplayContent); |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final TestRemoteAnimationRunner remoteAnimationRunner = new TestRemoteAnimationRunner(); |
| setupTaskFragmentRemoteAnimation(organizer, task.mTaskId, remoteAnimationRunner); |
| |
| // Closing TaskFragment with embedded activity. |
| final TaskFragment taskFragment1 = createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final ActivityRecord closingActivity = taskFragment1.getTopMostActivity(); |
| prepareActivityForAppTransition(closingActivity); |
| closingActivity.info.applicationInfo.uid = 12345; |
| // Opening TaskFragment with embedded activity with different UID. |
| final TaskFragment taskFragment2 = createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final ActivityRecord openingActivity = taskFragment2.getTopMostActivity(); |
| prepareActivityForAppTransition(openingActivity); |
| openingActivity.info.applicationInfo.uid = 54321; |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare and start transition. |
| prepareAndTriggerAppTransition(openingActivity, closingActivity, taskFragment1); |
| mWm.mAnimator.executeAfterPrepareSurfacesRunnables(); |
| |
| // Animation run by the remote handler. |
| assertTrue(remoteAnimationRunner.isAnimationStarted()); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_noOverrideWithTwoApps() { |
| final Task task = createTask(mDisplayContent); |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final TestRemoteAnimationRunner remoteAnimationRunner = new TestRemoteAnimationRunner(); |
| setupTaskFragmentRemoteAnimation(organizer, task.mTaskId, remoteAnimationRunner); |
| |
| // Closing activity in Task1. |
| final ActivityRecord closingActivity = createActivityRecord(mDisplayContent); |
| prepareActivityForAppTransition(closingActivity); |
| // Opening TaskFragment with embedded activity in Task2. |
| final TaskFragment taskFragment = createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final ActivityRecord openingActivity = taskFragment.getTopMostActivity(); |
| prepareActivityForAppTransition(openingActivity); |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare and start transition. |
| prepareAndTriggerAppTransition(openingActivity, closingActivity, taskFragment); |
| mWm.mAnimator.executeAfterPrepareSurfacesRunnables(); |
| |
| // Animation not run by the remote handler. |
| assertFalse(remoteAnimationRunner.isAnimationStarted()); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_noOverrideNonEmbeddedActivityWithDiffUid() { |
| final Task task = createTask(mDisplayContent); |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final TestRemoteAnimationRunner remoteAnimationRunner = new TestRemoteAnimationRunner(); |
| setupTaskFragmentRemoteAnimation(organizer, task.mTaskId, remoteAnimationRunner); |
| |
| // Closing TaskFragment with embedded activity. |
| final TaskFragment taskFragment = createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final ActivityRecord closingActivity = taskFragment.getTopMostActivity(); |
| prepareActivityForAppTransition(closingActivity); |
| closingActivity.info.applicationInfo.uid = 12345; |
| task.effectiveUid = closingActivity.getUid(); |
| // Opening non-embedded activity with different UID. |
| final ActivityRecord openingActivity = createActivityRecord(task); |
| prepareActivityForAppTransition(openingActivity); |
| openingActivity.info.applicationInfo.uid = 54321; |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare and start transition. |
| prepareAndTriggerAppTransition(openingActivity, closingActivity, taskFragment); |
| mWm.mAnimator.executeAfterPrepareSurfacesRunnables(); |
| |
| // Animation should not run by the remote handler when there are non-embedded activities of |
| // different UID. |
| assertFalse(remoteAnimationRunner.isAnimationStarted()); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_noOverrideWithWallpaper() { |
| final Task task = createTask(mDisplayContent); |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final TestRemoteAnimationRunner remoteAnimationRunner = new TestRemoteAnimationRunner(); |
| setupTaskFragmentRemoteAnimation(organizer, task.mTaskId, remoteAnimationRunner); |
| |
| // Create a TaskFragment with embedded activity. |
| final TaskFragment taskFragment = createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final ActivityRecord activity = taskFragment.getTopMostActivity(); |
| prepareActivityForAppTransition(activity); |
| // Set wallpaper as visible. |
| final WallpaperWindowToken wallpaperWindowToken = new WallpaperWindowToken(mWm, |
| mock(IBinder.class), true, mDisplayContent, true /* ownerCanManageAppTokens */); |
| spyOn(mDisplayContent.mWallpaperController); |
| doReturn(true).when(mDisplayContent.mWallpaperController).isWallpaperVisible(); |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare and start transition. |
| prepareAndTriggerAppTransition(activity, null /* closingActivity */, taskFragment); |
| mWm.mAnimator.executeAfterPrepareSurfacesRunnables(); |
| |
| // Animation should not run by the remote handler when there is wallpaper in the transition. |
| assertFalse(remoteAnimationRunner.isAnimationStarted()); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_inputProtectedForUntrustedAnimation() { |
| final Task task = createTask(mDisplayContent); |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final TestRemoteAnimationRunner remoteAnimationRunner = new TestRemoteAnimationRunner(); |
| setupTaskFragmentRemoteAnimation(organizer, task.mTaskId, remoteAnimationRunner); |
| |
| // Create a TaskFragment with embedded activities, one is trusted embedded, and the other |
| // one is untrusted embedded. |
| final TaskFragment taskFragment = new TaskFragmentBuilder(mAtm) |
| .setParentTask(task) |
| .createActivityCount(2) |
| .setOrganizer(organizer) |
| .build(); |
| final ActivityRecord activity0 = taskFragment.getChildAt(0).asActivityRecord(); |
| final ActivityRecord activity1 = taskFragment.getChildAt(1).asActivityRecord(); |
| // Also create a non-embedded activity in the Task. |
| final ActivityRecord activity2 = new ActivityBuilder(mAtm).build(); |
| task.addChild(activity2, POSITION_BOTTOM); |
| prepareActivityForAppTransition(activity0); |
| prepareActivityForAppTransition(activity1); |
| prepareActivityForAppTransition(activity2); |
| doReturn(false).when(taskFragment).isAllowedToEmbedActivityInTrustedMode(activity0); |
| doReturn(true).when(taskFragment).isAllowedToEmbedActivityInTrustedMode(activity1); |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare and start transition. |
| prepareAndTriggerAppTransition(activity1, null /* closingActivity */, taskFragment); |
| mWm.mAnimator.executeAfterPrepareSurfacesRunnables(); |
| |
| // The animation will be animated remotely by client and all activities are input disabled |
| // for untrusted animation. |
| assertTrue(remoteAnimationRunner.isAnimationStarted()); |
| verify(activity0).setDropInputForAnimation(true); |
| verify(activity1).setDropInputForAnimation(true); |
| verify(activity2).setDropInputForAnimation(true); |
| verify(activity0).setDropInputMode(DropInputMode.ALL); |
| verify(activity1).setDropInputMode(DropInputMode.ALL); |
| verify(activity2).setDropInputMode(DropInputMode.ALL); |
| |
| // Reset input after animation is finished. |
| clearInvocations(activity0); |
| clearInvocations(activity1); |
| clearInvocations(activity2); |
| remoteAnimationRunner.finishAnimation(); |
| |
| verify(activity0).setDropInputForAnimation(false); |
| verify(activity1).setDropInputForAnimation(false); |
| verify(activity2).setDropInputForAnimation(false); |
| verify(activity0).setDropInputMode(DropInputMode.OBSCURED); |
| verify(activity1).setDropInputMode(DropInputMode.NONE); |
| verify(activity2).setDropInputMode(DropInputMode.NONE); |
| } |
| |
| /** |
| * Since we don't have any use case to rely on handling input during animation, disable it even |
| * if it is trusted embedding so that it could cover some edge-cases when a previously trusted |
| * host starts doing something bad. |
| */ |
| @Test |
| public void testOverrideTaskFragmentAdapter_inputProtectedForTrustedAnimation() { |
| final Task task = createTask(mDisplayContent); |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final TestRemoteAnimationRunner remoteAnimationRunner = new TestRemoteAnimationRunner(); |
| setupTaskFragmentRemoteAnimation(organizer, task.mTaskId, remoteAnimationRunner); |
| |
| // Create a TaskFragment with only trusted embedded activity |
| final TaskFragment taskFragment = new TaskFragmentBuilder(mAtm) |
| .setParentTask(task) |
| .createActivityCount(1) |
| .setOrganizer(organizer) |
| .build(); |
| final ActivityRecord activity = taskFragment.getChildAt(0).asActivityRecord(); |
| prepareActivityForAppTransition(activity); |
| doReturn(true).when(taskFragment).isAllowedToEmbedActivityInTrustedMode(activity); |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare and start transition. |
| prepareAndTriggerAppTransition(activity, null /* closingActivity */, taskFragment); |
| mWm.mAnimator.executeAfterPrepareSurfacesRunnables(); |
| |
| // The animation will be animated remotely by client and all activities are input disabled |
| // for untrusted animation. |
| assertTrue(remoteAnimationRunner.isAnimationStarted()); |
| verify(activity).setDropInputForAnimation(true); |
| verify(activity).setDropInputMode(DropInputMode.ALL); |
| |
| // Reset input after animation is finished. |
| clearInvocations(activity); |
| remoteAnimationRunner.finishAnimation(); |
| |
| verify(activity).setDropInputForAnimation(false); |
| verify(activity).setDropInputMode(DropInputMode.NONE); |
| } |
| |
| /** |
| * We don't need to drop input for fully trusted embedding (system app, and embedding in the |
| * same app). This will allow users to do fast tapping. |
| */ |
| @Test |
| public void testOverrideTaskFragmentAdapter_noInputProtectedForFullyTrustedAnimation() { |
| final Task task = createTask(mDisplayContent); |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final TestRemoteAnimationRunner remoteAnimationRunner = new TestRemoteAnimationRunner(); |
| setupTaskFragmentRemoteAnimation(organizer, task.mTaskId, remoteAnimationRunner); |
| |
| // Create a TaskFragment with only trusted embedded activity |
| final TaskFragment taskFragment = new TaskFragmentBuilder(mAtm) |
| .setParentTask(task) |
| .createActivityCount(1) |
| .setOrganizer(organizer) |
| .build(); |
| final ActivityRecord activity = taskFragment.getChildAt(0).asActivityRecord(); |
| prepareActivityForAppTransition(activity); |
| final int uid = mAtm.mTaskFragmentOrganizerController.getTaskFragmentOrganizerUid( |
| getITaskFragmentOrganizer(organizer)); |
| doReturn(true).when(task).isFullyTrustedEmbedding(uid); |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare and start transition. |
| prepareAndTriggerAppTransition(activity, null /* closingActivity */, taskFragment); |
| mWm.mAnimator.executeAfterPrepareSurfacesRunnables(); |
| |
| // The animation will be animated remotely by client, but input should not be dropped for |
| // fully trusted. |
| assertTrue(remoteAnimationRunner.isAnimationStarted()); |
| verify(activity, never()).setDropInputForAnimation(true); |
| verify(activity, never()).setDropInputMode(DropInputMode.ALL); |
| } |
| |
| @Test |
| public void testTransitionGoodToGoForTaskFragments() { |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final Task task = createTask(mDisplayContent); |
| final TaskFragment changeTaskFragment = |
| createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final TaskFragment emptyTaskFragment = new TaskFragmentBuilder(mAtm) |
| .setParentTask(task) |
| .setOrganizer(organizer) |
| .build(); |
| prepareActivityForAppTransition(changeTaskFragment.getTopMostActivity()); |
| spyOn(mDisplayContent.mAppTransition); |
| spyOn(emptyTaskFragment); |
| |
| prepareAndTriggerAppTransition( |
| null /* openingActivity */, null /* closingActivity*/, changeTaskFragment); |
| |
| // Transition not ready because there is an empty non-finishing TaskFragment. |
| verify(mDisplayContent.mAppTransition, never()).goodToGo(anyInt(), any()); |
| |
| doReturn(true).when(emptyTaskFragment).hasChild(); |
| emptyTaskFragment.remove(false /* withTransition */, "test"); |
| |
| mDisplayContent.mAppTransitionController.handleAppTransitionReady(); |
| |
| // Transition ready because the empty (no running activity) TaskFragment is requested to be |
| // removed. |
| verify(mDisplayContent.mAppTransition).goodToGo(anyInt(), any()); |
| } |
| |
| @Test |
| public void testTransitionGoodToGoForTaskFragments_detachedApp() { |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final Task task = createTask(mDisplayContent); |
| final TaskFragment changeTaskFragment = |
| createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final TaskFragment emptyTaskFragment = new TaskFragmentBuilder(mAtm) |
| .setParentTask(task) |
| .setOrganizer(organizer) |
| .build(); |
| prepareActivityForAppTransition(changeTaskFragment.getTopMostActivity()); |
| // To make sure that having a detached activity won't cause any issue. |
| final ActivityRecord detachedActivity = createActivityRecord(task); |
| detachedActivity.removeImmediately(); |
| assertNull(detachedActivity.getRootTask()); |
| spyOn(mDisplayContent.mAppTransition); |
| spyOn(emptyTaskFragment); |
| |
| prepareAndTriggerAppTransition( |
| null /* openingActivity */, detachedActivity, changeTaskFragment); |
| |
| // Transition not ready because there is an empty non-finishing TaskFragment. |
| verify(mDisplayContent.mAppTransition, never()).goodToGo(anyInt(), any()); |
| |
| doReturn(true).when(emptyTaskFragment).hasChild(); |
| emptyTaskFragment.remove(false /* withTransition */, "test"); |
| |
| mDisplayContent.mAppTransitionController.handleAppTransitionReady(); |
| |
| // Transition ready because the empty (no running activity) TaskFragment is requested to be |
| // removed. |
| verify(mDisplayContent.mAppTransition).goodToGo(anyInt(), any()); |
| } |
| |
| /** Registers remote animation for the organizer. */ |
| private void setupTaskFragmentRemoteAnimation(TaskFragmentOrganizer organizer, int taskId, |
| TestRemoteAnimationRunner remoteAnimationRunner) { |
| final RemoteAnimationAdapter adapter = new RemoteAnimationAdapter( |
| remoteAnimationRunner, 10, 1); |
| final ITaskFragmentOrganizer iOrganizer = getITaskFragmentOrganizer(organizer); |
| final RemoteAnimationDefinition definition = new RemoteAnimationDefinition(); |
| definition.addRemoteAnimation(TRANSIT_OLD_TASK_FRAGMENT_CHANGE, adapter); |
| definition.addRemoteAnimation(TRANSIT_OLD_TASK_FRAGMENT_OPEN, adapter); |
| definition.addRemoteAnimation(TRANSIT_OLD_TASK_FRAGMENT_CLOSE, adapter); |
| mAtm.mTaskFragmentOrganizerController.registerOrganizer(iOrganizer); |
| mAtm.mTaskFragmentOrganizerController.registerRemoteAnimations(iOrganizer, taskId, |
| definition); |
| } |
| |
| private static ITaskFragmentOrganizer getITaskFragmentOrganizer( |
| TaskFragmentOrganizer organizer) { |
| return ITaskFragmentOrganizer.Stub.asInterface(organizer.getOrganizerToken().asBinder()); |
| } |
| |
| private void prepareAndTriggerAppTransition(@Nullable ActivityRecord openingActivity, |
| @Nullable ActivityRecord closingActivity, @Nullable TaskFragment changingTaskFragment) { |
| if (openingActivity != null) { |
| mDisplayContent.mAppTransition.prepareAppTransition(TRANSIT_OPEN, 0); |
| mDisplayContent.mOpeningApps.add(openingActivity); |
| } |
| if (closingActivity != null) { |
| mDisplayContent.mAppTransition.prepareAppTransition(TRANSIT_CLOSE, 0); |
| mDisplayContent.mClosingApps.add(closingActivity); |
| } |
| if (changingTaskFragment != null) { |
| mDisplayContent.mAppTransition.prepareAppTransition(TRANSIT_CHANGE, 0); |
| mDisplayContent.mChangingContainers.add(changingTaskFragment); |
| } |
| mDisplayContent.mAppTransitionController.handleAppTransitionReady(); |
| } |
| |
| private static void prepareActivityForAppTransition(ActivityRecord activity) { |
| // Transition will wait until all participated activities to be drawn. |
| activity.allDrawn = true; |
| // Skip manipulate the SurfaceControl. |
| doNothing().when(activity).setDropInputMode(anyInt()); |
| // Make sure activity can create remote animation target. |
| doReturn(mock(RemoteAnimationTarget.class)).when(activity).createRemoteAnimationTarget( |
| any()); |
| } |
| } |