1 /* 2 * Copyright (C) 2018 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not 5 * use this file except in compliance with the License. You may obtain a copy of 6 * the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT 12 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the 13 * License for the specific language governing permissions and limitations under 14 * the License. 15 */ 16 17 package com.android.quickstep; 18 19 import static androidx.test.InstrumentationRegistry.getInstrumentation; 20 21 import static com.android.quickstep.NavigationModeSwitchRule.Mode.ALL; 22 import static com.android.quickstep.NavigationModeSwitchRule.Mode.THREE_BUTTON; 23 import static com.android.quickstep.NavigationModeSwitchRule.Mode.ZERO_BUTTON; 24 import static com.android.systemui.shared.system.QuickStepContract.NAV_BAR_MODE_3BUTTON_OVERLAY; 25 import static com.android.systemui.shared.system.QuickStepContract.NAV_BAR_MODE_GESTURAL_OVERLAY; 26 27 import android.content.Context; 28 import android.content.pm.PackageManager; 29 import android.util.Log; 30 31 import androidx.test.uiautomator.UiDevice; 32 33 import com.android.launcher3.tapl.LauncherInstrumentation; 34 import com.android.launcher3.tapl.TestHelpers; 35 import com.android.launcher3.ui.AbstractLauncherUiTest; 36 import com.android.launcher3.util.DisplayController; 37 import com.android.launcher3.util.Wait; 38 import com.android.launcher3.util.rule.FailureWatcher; 39 import com.android.systemui.shared.system.QuickStepContract; 40 41 import org.junit.rules.TestRule; 42 import org.junit.runner.Description; 43 import org.junit.runners.model.Statement; 44 45 import java.lang.annotation.ElementType; 46 import java.lang.annotation.Retention; 47 import java.lang.annotation.RetentionPolicy; 48 import java.lang.annotation.Target; 49 import java.util.concurrent.CountDownLatch; 50 import java.util.concurrent.TimeUnit; 51 52 /** 53 * Test rule that allows executing a test with Quickstep on and then Quickstep off. 54 * The test should be annotated with @NavigationModeSwitch. 55 */ 56 public class NavigationModeSwitchRule implements TestRule { 57 58 static final String TAG = "QuickStepOnOffRule"; 59 60 public static final int WAIT_TIME_MS = 10000; 61 62 public enum Mode { 63 THREE_BUTTON, ZERO_BUTTON, ALL 64 } 65 66 // Annotation for tests that need to be run with quickstep enabled and disabled. 67 @Retention(RetentionPolicy.RUNTIME) 68 @Target(ElementType.METHOD) 69 public @interface NavigationModeSwitch { mode()70 Mode mode() default ALL; 71 } 72 73 private final LauncherInstrumentation mLauncher; 74 75 static final DisplayController DISPLAY_CONTROLLER = 76 DisplayController.INSTANCE.get(getInstrumentation().getTargetContext()); 77 NavigationModeSwitchRule(LauncherInstrumentation launcher)78 public NavigationModeSwitchRule(LauncherInstrumentation launcher) { 79 mLauncher = launcher; 80 } 81 82 @Override apply(Statement base, Description description)83 public Statement apply(Statement base, Description description) { 84 if (TestHelpers.isInLauncherProcess() && 85 description.getAnnotation(NavigationModeSwitch.class) != null) { 86 Mode mode = description.getAnnotation(NavigationModeSwitch.class).mode(); 87 return new Statement() { 88 @Override 89 public void evaluate() throws Throwable { 90 mLauncher.enableDebugTracing(); 91 final Context context = getInstrumentation().getContext(); 92 final int currentInteractionMode = 93 LauncherInstrumentation.getCurrentInteractionMode(context); 94 final String prevOverlayPkg = getCurrentOverlayPackage(currentInteractionMode); 95 final LauncherInstrumentation.NavigationModel originalMode = 96 mLauncher.getNavigationModel(); 97 try { 98 if (mode == ZERO_BUTTON || mode == ALL) { 99 evaluateWithZeroButtons(); 100 } 101 if (mode == THREE_BUTTON || mode == ALL) { 102 evaluateWithThreeButtons(); 103 } 104 } catch (Throwable e) { 105 Log.e(TAG, "Error", e); 106 throw e; 107 } finally { 108 Log.d(TAG, "In Finally block"); 109 assertTrue(mLauncher, "Couldn't set overlay", 110 setActiveOverlay(mLauncher, prevOverlayPkg, originalMode, 111 description), description); 112 } 113 } 114 115 private void evaluateWithThreeButtons() throws Throwable { 116 if (setActiveOverlay(mLauncher, NAV_BAR_MODE_3BUTTON_OVERLAY, 117 LauncherInstrumentation.NavigationModel.THREE_BUTTON, description)) { 118 base.evaluate(); 119 } 120 } 121 122 private void evaluateWithZeroButtons() throws Throwable { 123 if (setActiveOverlay(mLauncher, NAV_BAR_MODE_GESTURAL_OVERLAY, 124 LauncherInstrumentation.NavigationModel.ZERO_BUTTON, description)) { 125 base.evaluate(); 126 } 127 } 128 }; 129 } else { 130 return base; 131 } 132 } 133 134 public static String getCurrentOverlayPackage(int currentInteractionMode) { 135 return QuickStepContract.isGesturalMode(currentInteractionMode) 136 ? NAV_BAR_MODE_GESTURAL_OVERLAY 137 : NAV_BAR_MODE_3BUTTON_OVERLAY; 138 } 139 140 private static LauncherInstrumentation.NavigationModel currentSysUiNavigationMode() { 141 return LauncherInstrumentation.getNavigationModel( 142 DisplayController.getNavigationMode( 143 getInstrumentation(). 144 getTargetContext()). 145 resValue); 146 } 147 148 public static boolean setActiveOverlay(LauncherInstrumentation launcher, String overlayPackage, 149 LauncherInstrumentation.NavigationModel expectedMode, Description description) 150 throws Exception { 151 if (!packageExists(overlayPackage)) { 152 Log.d(TAG, "setActiveOverlay: " + overlayPackage + " pkg does not exist"); 153 return false; 154 } 155 156 Log.d(TAG, "setActiveOverlay: " + overlayPackage + "..."); 157 UiDevice.getInstance(getInstrumentation()).executeShellCommand( 158 "cmd overlay enable-exclusive --category " + overlayPackage); 159 160 if (currentSysUiNavigationMode() != expectedMode) { 161 final CountDownLatch latch = new CountDownLatch(1); 162 final Context targetContext = getInstrumentation().getTargetContext(); 163 final DisplayController.DisplayInfoChangeListener listener = 164 (context, info, flags) -> { 165 if (LauncherInstrumentation.getNavigationModel(info.navigationMode.resValue) 166 == expectedMode) { 167 latch.countDown(); 168 } 169 }; 170 targetContext.getMainExecutor().execute(() -> 171 DISPLAY_CONTROLLER.addChangeListener(listener)); 172 latch.await(60, TimeUnit.SECONDS); 173 targetContext.getMainExecutor().execute(() -> 174 DISPLAY_CONTROLLER.removeChangeListener(listener)); 175 176 assertTrue(launcher, "Navigation mode didn't change to " + expectedMode, 177 currentSysUiNavigationMode() == expectedMode, description); 178 179 } 180 181 Wait.atMost("Couldn't switch to " + overlayPackage, 182 () -> launcher.getNavigationModel() == expectedMode, WAIT_TIME_MS, launcher); 183 184 Wait.atMost(() -> "Switching nav mode: " 185 + launcher.getNavigationModeMismatchError(false), 186 () -> launcher.getNavigationModeMismatchError(false) == null, 187 WAIT_TIME_MS, launcher); 188 AbstractLauncherUiTest.checkDetectedLeaks(launcher); 189 return true; 190 } 191 192 private static boolean packageExists(String packageName) { 193 try { 194 PackageManager pm = getInstrumentation().getContext().getPackageManager(); 195 if (pm.getApplicationInfo(packageName, 0 /* flags */) == null) { 196 return false; 197 } 198 } catch (PackageManager.NameNotFoundException e) { 199 return false; 200 } 201 return true; 202 } 203 204 private static void assertTrue(LauncherInstrumentation launcher, String message, 205 boolean condition, Description description) { 206 launcher.checkForAnomaly(true, true); 207 if (!condition) { 208 final AssertionError assertionError = new AssertionError(message); 209 if (description != null) { 210 FailureWatcher.onError(launcher, description, assertionError); 211 } 212 throw assertionError; 213 } 214 } 215 } 216