| /* |
| * 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.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.doReturn; |
| import static com.android.dx.mockito.inline.extended.ExtendedMockito.spyOn; |
| |
| 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.mockito.ArgumentMatchers.any; |
| import static org.mockito.ArgumentMatchers.anyInt; |
| import static org.mockito.Mockito.doCallRealMethod; |
| import static org.mockito.Mockito.mock; |
| import static org.mockito.Mockito.never; |
| import static org.mockito.Mockito.verify; |
| |
| import android.annotation.Nullable; |
| 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); |
| } |
| |
| @Override |
| ActivityRecord createActivityRecord(DisplayContent dc, int windowingMode, int activityType) { |
| final ActivityRecord r = super.createActivityRecord(dc, windowingMode, activityType); |
| // Ensure that ActivityRecord#setOccludesParent takes effect. |
| doCallRealMethod().when(r).fillsParent(); |
| return r; |
| } |
| |
| @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.performClearTask("test"); |
| 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); |
| translucentOpening.setOccludesParent(false); |
| 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); |
| translucentClosing.setOccludesParent(false); |
| 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_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 */)); |
| } |
| |
| static class TestRemoteAnimationRunner implements IRemoteAnimationRunner { |
| @Override |
| public void onAnimationStart(int transit, RemoteAnimationTarget[] apps, |
| RemoteAnimationTarget[] wallpapers, RemoteAnimationTarget[] nonApps, |
| IRemoteAnimationFinishedCallback finishedCallback) throws RemoteException { |
| } |
| |
| @Override |
| public void onAnimationCancelled() throws RemoteException { |
| } |
| |
| @Override |
| public IBinder asBinder() { |
| return new Binder(); |
| } |
| } |
| |
| @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 TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final RemoteAnimationAdapter adapter = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| setupTaskFragmentRemoteAnimation(organizer, adapter); |
| |
| // Create a TaskFragment with embedded activity. |
| final TaskFragment taskFragment = createTaskFragmentWithEmbeddedActivity( |
| createTask(mDisplayContent), organizer); |
| final ActivityRecord activity = taskFragment.getTopMostActivity(); |
| activity.allDrawn = true; |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare a transition. |
| prepareAndTriggerAppTransition(activity, null /* closingActivity */, taskFragment); |
| |
| // Should be overridden. |
| verify(mDisplayContent.mAppTransition) |
| .overridePendingAppTransitionRemote(adapter, false /* sync */); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_overrideWithNonEmbeddedActivity() { |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final RemoteAnimationAdapter adapter = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| setupTaskFragmentRemoteAnimation(organizer, adapter); |
| |
| final Task task = createTask(mDisplayContent); |
| // Closing non-embedded activity. |
| final ActivityRecord closingActivity = createActivityRecord(task); |
| closingActivity.allDrawn = true; |
| // Opening TaskFragment with embedded activity. |
| final TaskFragment taskFragment = createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final ActivityRecord openingActivity = taskFragment.getTopMostActivity(); |
| openingActivity.allDrawn = true; |
| task.effectiveUid = openingActivity.getUid(); |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare a transition. |
| prepareAndTriggerAppTransition(openingActivity, closingActivity, taskFragment); |
| |
| // Should be overridden. |
| verify(mDisplayContent.mAppTransition) |
| .overridePendingAppTransitionRemote(adapter, false /* sync */); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_overrideEmbeddedActivityWithDiffUid() { |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final RemoteAnimationAdapter adapter = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| setupTaskFragmentRemoteAnimation(organizer, adapter); |
| |
| final Task task = createTask(mDisplayContent); |
| // Closing TaskFragment with embedded activity. |
| final TaskFragment taskFragment1 = createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final ActivityRecord closingActivity = taskFragment1.getTopMostActivity(); |
| closingActivity.allDrawn = true; |
| closingActivity.info.applicationInfo.uid = 12345; |
| // Opening TaskFragment with embedded activity with different UID. |
| final TaskFragment taskFragment2 = createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final ActivityRecord openingActivity = taskFragment2.getTopMostActivity(); |
| openingActivity.info.applicationInfo.uid = 54321; |
| openingActivity.allDrawn = true; |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare a transition. |
| prepareAndTriggerAppTransition(openingActivity, closingActivity, taskFragment1); |
| |
| // Should be overridden. |
| verify(mDisplayContent.mAppTransition) |
| .overridePendingAppTransitionRemote(adapter, false /* sync */); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_noOverrideWithTwoApps() { |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final RemoteAnimationAdapter adapter = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| setupTaskFragmentRemoteAnimation(organizer, adapter); |
| |
| // Closing activity in Task1. |
| final ActivityRecord closingActivity = createActivityRecord(mDisplayContent); |
| closingActivity.allDrawn = true; |
| // Opening TaskFragment with embedded activity in Task2. |
| final TaskFragment taskFragment = createTaskFragmentWithEmbeddedActivity( |
| createTask(mDisplayContent), organizer); |
| final ActivityRecord openingActivity = taskFragment.getTopMostActivity(); |
| openingActivity.allDrawn = true; |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare a transition for TaskFragment. |
| prepareAndTriggerAppTransition(openingActivity, closingActivity, taskFragment); |
| |
| // Should not be overridden. |
| verify(mDisplayContent.mAppTransition, never()) |
| .overridePendingAppTransitionRemote(adapter, false /* sync */); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_noOverrideNonEmbeddedActivityWithDiffUid() { |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final RemoteAnimationAdapter adapter = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| setupTaskFragmentRemoteAnimation(organizer, adapter); |
| |
| final Task task = createTask(mDisplayContent); |
| // Closing TaskFragment with embedded activity. |
| final TaskFragment taskFragment = createTaskFragmentWithEmbeddedActivity(task, organizer); |
| final ActivityRecord closingActivity = taskFragment.getTopMostActivity(); |
| closingActivity.allDrawn = true; |
| closingActivity.info.applicationInfo.uid = 12345; |
| task.effectiveUid = closingActivity.getUid(); |
| // Opening non-embedded activity with different UID. |
| final ActivityRecord openingActivity = createActivityRecord(task); |
| openingActivity.info.applicationInfo.uid = 54321; |
| openingActivity.allDrawn = true; |
| spyOn(mDisplayContent.mAppTransition); |
| |
| // Prepare a transition. |
| prepareAndTriggerAppTransition(openingActivity, closingActivity, taskFragment); |
| |
| // Should not be overridden |
| verify(mDisplayContent.mAppTransition, never()) |
| .overridePendingAppTransitionRemote(adapter, false /* sync */); |
| } |
| |
| @Test |
| public void testOverrideTaskFragmentAdapter_noOverrideWithWallpaper() { |
| final TaskFragmentOrganizer organizer = new TaskFragmentOrganizer(Runnable::run); |
| final RemoteAnimationAdapter adapter = new RemoteAnimationAdapter( |
| new TestRemoteAnimationRunner(), 10, 1); |
| setupTaskFragmentRemoteAnimation(organizer, adapter); |
| |
| // Create a TaskFragment with embedded activity. |
| final TaskFragment taskFragment = createTaskFragmentWithEmbeddedActivity( |
| createTask(mDisplayContent), organizer); |
| final ActivityRecord activity = taskFragment.getTopMostActivity(); |
| activity.allDrawn = true; |
| // 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 a transition. |
| prepareAndTriggerAppTransition(activity, null /* closingActivity */, taskFragment); |
| |
| // Should not be overridden when there is wallpaper in the transition. |
| verify(mDisplayContent.mAppTransition, never()) |
| .overridePendingAppTransitionRemote(adapter, false /* sync */); |
| } |
| |
| @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(); |
| changeTaskFragment.getTopMostActivity().allDrawn = true; |
| 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(); |
| changeTaskFragment.getTopMostActivity().allDrawn = true; |
| // 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, |
| RemoteAnimationAdapter adapter) { |
| final ITaskFragmentOrganizer iOrganizer = |
| ITaskFragmentOrganizer.Stub.asInterface(organizer.getOrganizerToken().asBinder()); |
| 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, definition); |
| } |
| |
| 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(); |
| } |
| } |