blob: 2452ef0425e1dba66629c5211a8253114f54dc26 [file] [log] [blame]
Wale Ogunwaleb783fd82016-11-04 09:51:54 -07001/*
2 * Copyright (C) 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -070014 * limitations under the License.
Wale Ogunwaleb783fd82016-11-04 09:51:54 -070015 */
16
17package com.android.server.wm;
18
Andrii Kulian92c9a942017-10-10 00:41:41 -070019import static android.content.pm.ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE;
20import static android.content.pm.ActivityInfo.SCREEN_ORIENTATION_PORTRAIT;
21import static android.content.pm.ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED;
Tiger Huang51c5a1d2018-12-11 20:24:51 +080022import static android.os.Build.VERSION_CODES.P;
23import static android.os.Build.VERSION_CODES.Q;
Wale Ogunwale34247952017-02-19 11:57:53 -080024import static android.view.Display.DEFAULT_DISPLAY;
Issei Suzuki43190bd2018-08-20 17:28:41 +020025import static android.view.DisplayCutout.BOUNDS_POSITION_LEFT;
26import static android.view.DisplayCutout.BOUNDS_POSITION_TOP;
Adrian Roos24264212018-02-19 16:26:15 +010027import static android.view.DisplayCutout.fromBoundingRect;
Wale Ogunwale34247952017-02-19 11:57:53 -080028import static android.view.WindowManager.LayoutParams.TYPE_APPLICATION;
29import static android.view.WindowManager.LayoutParams.TYPE_APPLICATION_ATTACHED_DIALOG;
30import static android.view.WindowManager.LayoutParams.TYPE_BASE_APPLICATION;
Andrii Kulian92c9a942017-10-10 00:41:41 -070031import static android.view.WindowManager.LayoutParams.TYPE_STATUS_BAR;
Wale Ogunwale34247952017-02-19 11:57:53 -080032import static android.view.WindowManager.LayoutParams.TYPE_VOICE_INTERACTION;
Brett Chabota26eda92018-07-23 13:08:30 -070033
Garfield Tan90b04282018-12-11 14:04:42 -080034import static com.android.dx.mockito.inline.extended.ExtendedMockito.any;
35import static com.android.dx.mockito.inline.extended.ExtendedMockito.anyBoolean;
Tadashi G. Takaokabf0d57b2018-11-19 16:09:58 +090036import static com.android.dx.mockito.inline.extended.ExtendedMockito.doNothing;
Garfield Tan90b04282018-12-11 14:04:42 -080037import static com.android.dx.mockito.inline.extended.ExtendedMockito.eq;
38import static com.android.dx.mockito.inline.extended.ExtendedMockito.mock;
39import static com.android.dx.mockito.inline.extended.ExtendedMockito.never;
40import static com.android.dx.mockito.inline.extended.ExtendedMockito.same;
Tadashi G. Takaokabf0d57b2018-11-19 16:09:58 +090041import static com.android.dx.mockito.inline.extended.ExtendedMockito.spy;
42import static com.android.dx.mockito.inline.extended.ExtendedMockito.times;
43import static com.android.dx.mockito.inline.extended.ExtendedMockito.verify;
David Stevens46939562017-03-24 13:04:00 -070044import static com.android.server.wm.WindowContainer.POSITION_TOP;
Tiger Huang1e5b10a2018-07-30 20:19:51 +080045import static com.android.server.wm.WindowManagerService.UPDATE_FOCUS_NORMAL;
chaviwebcbc342018-02-07 13:19:00 -080046
Adrian Roos5251b1d2018-03-23 18:57:43 +010047import static org.hamcrest.Matchers.is;
Wale Ogunwale34247952017-02-19 11:57:53 -080048import static org.junit.Assert.assertEquals;
Andrii Kulianf0379de2018-03-14 16:24:07 -070049import static org.junit.Assert.assertFalse;
Arthur Hungbe5ce212018-09-13 18:41:56 +080050import static org.junit.Assert.assertNotNull;
lumarkff0ab692018-11-05 20:32:30 +080051import static org.junit.Assert.assertNull;
Adrian Roos5251b1d2018-03-23 18:57:43 +010052import static org.junit.Assert.assertThat;
Wale Ogunwale34247952017-02-19 11:57:53 -080053import static org.junit.Assert.assertTrue;
Tarandeep Singha6f35612019-01-11 19:50:46 -080054import static org.mockito.ArgumentMatchers.eq;
Wale Ogunwale34247952017-02-19 11:57:53 -080055
Andrii Kulian92c9a942017-10-10 00:41:41 -070056import android.annotation.SuppressLint;
Tarandeep Singha6f35612019-01-11 19:50:46 -080057import android.app.WindowConfiguration;
Andrii Kuliand68501e2017-01-10 22:57:27 -080058import android.content.res.Configuration;
Adrian Roos1cf585052018-01-03 18:43:27 +010059import android.graphics.Rect;
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -070060import android.os.SystemClock;
Wale Ogunwaleb783fd82016-11-04 09:51:54 -070061import android.platform.test.annotations.Presubmit;
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -070062import android.util.DisplayMetrics;
Adrian Roos1cf585052018-01-03 18:43:27 +010063import android.view.DisplayCutout;
Riddle Hsua4d6fa22018-08-11 00:50:39 +080064import android.view.Gravity;
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -070065import android.view.MotionEvent;
Adrian Roos1cf585052018-01-03 18:43:27 +010066import android.view.Surface;
Tarandeep Singha6f35612019-01-11 19:50:46 -080067import android.view.ViewRootImpl;
68import android.view.test.InsetsModeSession;
Wale Ogunwaleb783fd82016-11-04 09:51:54 -070069
Brett Chabota26eda92018-07-23 13:08:30 -070070import androidx.test.filters.FlakyTest;
71import androidx.test.filters.SmallTest;
Brett Chabota26eda92018-07-23 13:08:30 -070072
Garfield Tan90b04282018-12-11 14:04:42 -080073import com.android.dx.mockito.inline.extended.ExtendedMockito;
Adrian Roos6a4fa0e2018-03-05 19:50:16 +010074import com.android.server.wm.utils.WmDisplayCutout;
75
Brett Chabota26eda92018-07-23 13:08:30 -070076import org.junit.Test;
Garfield Tan90b04282018-12-11 14:04:42 -080077import org.mockito.ArgumentCaptor;
Brett Chabota26eda92018-07-23 13:08:30 -070078
Adrian Roos0f9368c2018-04-08 10:59:08 -070079import java.util.ArrayList;
Wale Ogunwale34247952017-02-19 11:57:53 -080080import java.util.Arrays;
Adrian Roos0f9368c2018-04-08 10:59:08 -070081import java.util.Collections;
Wale Ogunwale34247952017-02-19 11:57:53 -080082import java.util.LinkedList;
83import java.util.List;
Wale Ogunwaleb783fd82016-11-04 09:51:54 -070084
85/**
86 * Tests for the {@link DisplayContent} class.
87 *
88 * Build/Install/Run:
Yunfan Chen6dd9a622019-02-18 15:12:33 +090089 * atest WmTests:DisplayContentTests
Wale Ogunwaleb783fd82016-11-04 09:51:54 -070090 */
91@SmallTest
92@Presubmit
Wale Ogunwale44fbdf52016-11-16 10:18:45 -080093public class DisplayContentTests extends WindowTestsBase {
Wale Ogunwaleb783fd82016-11-04 09:51:54 -070094
95 @Test
Yunfan Chen6dd9a622019-02-18 15:12:33 +090096 @FlakyTest(detail = "Promote to presubmit when shown to be stable.")
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -070097 public void testForAllWindows() {
Wale Ogunwale3c1170d2016-12-02 14:44:52 -080098 final WindowState exitingAppWindow = createWindow(null, TYPE_BASE_APPLICATION,
Wale Ogunwale11cc5162017-04-25 20:29:13 -070099 mDisplayContent, "exiting app");
Wale Ogunwaleb783fd82016-11-04 09:51:54 -0700100 final AppWindowToken exitingAppToken = exitingAppWindow.mAppToken;
Yunfan Chen6dd9a622019-02-18 15:12:33 +0900101 // Wait until everything in animation handler get executed to prevent the exiting window
102 // from being removed during WindowSurfacePlacer Traversal.
103 waitUntilHandlersIdle();
104
Wale Ogunwaleb783fd82016-11-04 09:51:54 -0700105 exitingAppToken.mIsExiting = true;
Bryce Lee6d410262017-02-28 15:30:17 -0800106 exitingAppToken.getTask().mStack.mExitingAppTokens.add(exitingAppToken);
Wale Ogunwaleb783fd82016-11-04 09:51:54 -0700107
Wale Ogunwale34247952017-02-19 11:57:53 -0800108 assertForAllWindowsOrder(Arrays.asList(
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700109 mWallpaperWindow,
Wale Ogunwale34247952017-02-19 11:57:53 -0800110 exitingAppWindow,
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700111 mChildAppWindowBelow,
112 mAppWindow,
113 mChildAppWindowAbove,
114 mDockedDividerWindow,
115 mStatusBarWindow,
116 mNavBarWindow,
117 mImeWindow,
118 mImeDialogWindow));
Wale Ogunwale3c1170d2016-12-02 14:44:52 -0800119 }
120
121 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700122 public void testForAllWindows_WithAppImeTarget() {
Wale Ogunwale3c1170d2016-12-02 14:44:52 -0800123 final WindowState imeAppTarget =
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700124 createWindow(null, TYPE_BASE_APPLICATION, mDisplayContent, "imeAppTarget");
Wale Ogunwale3c1170d2016-12-02 14:44:52 -0800125
lumarkff0ab692018-11-05 20:32:30 +0800126 mDisplayContent.mInputMethodTarget = imeAppTarget;
Wale Ogunwale3c1170d2016-12-02 14:44:52 -0800127
Wale Ogunwale34247952017-02-19 11:57:53 -0800128 assertForAllWindowsOrder(Arrays.asList(
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700129 mWallpaperWindow,
130 mChildAppWindowBelow,
131 mAppWindow,
132 mChildAppWindowAbove,
Wale Ogunwale34247952017-02-19 11:57:53 -0800133 imeAppTarget,
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700134 mImeWindow,
135 mImeDialogWindow,
136 mDockedDividerWindow,
137 mStatusBarWindow,
138 mNavBarWindow));
Wale Ogunwale34247952017-02-19 11:57:53 -0800139 }
Wale Ogunwale3c1170d2016-12-02 14:44:52 -0800140
Wale Ogunwale34247952017-02-19 11:57:53 -0800141 @Test
lumarkff0ab692018-11-05 20:32:30 +0800142 public void testForAllWindows_WithChildWindowImeTarget() throws Exception {
143 mDisplayContent.mInputMethodTarget = mChildAppWindowAbove;
Wale Ogunwale3c1170d2016-12-02 14:44:52 -0800144
Wale Ogunwale34247952017-02-19 11:57:53 -0800145 assertForAllWindowsOrder(Arrays.asList(
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700146 mWallpaperWindow,
147 mChildAppWindowBelow,
148 mAppWindow,
149 mChildAppWindowAbove,
150 mImeWindow,
151 mImeDialogWindow,
152 mDockedDividerWindow,
153 mStatusBarWindow,
154 mNavBarWindow));
Wale Ogunwaleb783fd82016-11-04 09:51:54 -0700155 }
Wale Ogunwale5d7e7f12016-12-12 14:47:05 -0800156
157 @Test
lumarkff0ab692018-11-05 20:32:30 +0800158 public void testForAllWindows_WithStatusBarImeTarget() throws Exception {
159 mDisplayContent.mInputMethodTarget = mStatusBarWindow;
Wale Ogunwale6ce0fb82016-12-13 14:24:00 -0800160
Wale Ogunwale34247952017-02-19 11:57:53 -0800161 assertForAllWindowsOrder(Arrays.asList(
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700162 mWallpaperWindow,
163 mChildAppWindowBelow,
164 mAppWindow,
165 mChildAppWindowAbove,
166 mDockedDividerWindow,
167 mStatusBarWindow,
168 mImeWindow,
169 mImeDialogWindow,
170 mNavBarWindow));
Wale Ogunwale6ce0fb82016-12-13 14:24:00 -0800171 }
172
173 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700174 public void testForAllWindows_WithInBetweenWindowToken() {
Wale Ogunwale5d7e7f12016-12-12 14:47:05 -0800175 // This window is set-up to be z-ordered between some windows that go in the same token like
176 // the nav bar and status bar.
177 final WindowState voiceInteractionWindow = createWindow(null, TYPE_VOICE_INTERACTION,
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700178 mDisplayContent, "voiceInteractionWindow");
Wale Ogunwale5d7e7f12016-12-12 14:47:05 -0800179
Wale Ogunwale34247952017-02-19 11:57:53 -0800180 assertForAllWindowsOrder(Arrays.asList(
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700181 mWallpaperWindow,
182 mChildAppWindowBelow,
183 mAppWindow,
184 mChildAppWindowAbove,
185 mDockedDividerWindow,
Wale Ogunwale34247952017-02-19 11:57:53 -0800186 voiceInteractionWindow,
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700187 mStatusBarWindow,
188 mNavBarWindow,
189 mImeWindow,
190 mImeDialogWindow));
Wale Ogunwale34247952017-02-19 11:57:53 -0800191 }
Wale Ogunwale5d7e7f12016-12-12 14:47:05 -0800192
Wale Ogunwale34247952017-02-19 11:57:53 -0800193 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700194 public void testComputeImeTarget() {
Wale Ogunwale34247952017-02-19 11:57:53 -0800195 // Verify that an app window can be an ime target.
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700196 final WindowState appWin = createWindow(null, TYPE_APPLICATION, mDisplayContent, "appWin");
Wale Ogunwale34247952017-02-19 11:57:53 -0800197 appWin.setHasSurface(true);
198 assertTrue(appWin.canBeImeTarget());
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700199 WindowState imeTarget = mDisplayContent.computeImeTarget(false /* updateImeTarget */);
Wale Ogunwale34247952017-02-19 11:57:53 -0800200 assertEquals(appWin, imeTarget);
chaviwebcbc342018-02-07 13:19:00 -0800201 appWin.mHidden = false;
Wale Ogunwale5d7e7f12016-12-12 14:47:05 -0800202
Wale Ogunwale34247952017-02-19 11:57:53 -0800203 // Verify that an child window can be an ime target.
204 final WindowState childWin = createWindow(appWin,
205 TYPE_APPLICATION_ATTACHED_DIALOG, "childWin");
206 childWin.setHasSurface(true);
207 assertTrue(childWin.canBeImeTarget());
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700208 imeTarget = mDisplayContent.computeImeTarget(false /* updateImeTarget */);
Wale Ogunwale34247952017-02-19 11:57:53 -0800209 assertEquals(childWin, imeTarget);
Wale Ogunwale5d7e7f12016-12-12 14:47:05 -0800210 }
Andrii Kulian6cc1a1d2016-12-27 23:52:59 -0800211
Andrii Kuliand68501e2017-01-10 22:57:27 -0800212 /**
213 * This tests stack movement between displays and proper stack's, task's and app token's display
214 * container references updates.
215 */
Andrii Kulian6cc1a1d2016-12-27 23:52:59 -0800216 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700217 public void testMoveStackBetweenDisplays() {
Andrii Kulian367ff7f2017-01-25 19:45:34 -0800218 // Create a second display.
219 final DisplayContent dc = createNewDisplay();
Andrii Kulian6cc1a1d2016-12-27 23:52:59 -0800220
221 // Add stack with activity.
222 final TaskStack stack = createTaskStackOnDisplay(dc);
223 assertEquals(dc.getDisplayId(), stack.getDisplayContent().getDisplayId());
224 assertEquals(dc, stack.getParent().getParent());
225 assertEquals(dc, stack.getDisplayContent());
226
227 final Task task = createTaskInStack(stack, 0 /* userId */);
chaviw97d28202018-02-27 16:23:53 -0800228 final WindowTestUtils.TestAppWindowToken token = WindowTestUtils.createTestAppWindowToken(
229 dc);
Andrii Kulian6cc1a1d2016-12-27 23:52:59 -0800230 task.addChild(token, 0);
231 assertEquals(dc, task.getDisplayContent());
232 assertEquals(dc, token.getDisplayContent());
233
234 // Move stack to first display.
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700235 mDisplayContent.moveStackToDisplay(stack, true /* onTop */);
236 assertEquals(mDisplayContent.getDisplayId(), stack.getDisplayContent().getDisplayId());
237 assertEquals(mDisplayContent, stack.getParent().getParent());
238 assertEquals(mDisplayContent, stack.getDisplayContent());
239 assertEquals(mDisplayContent, task.getDisplayContent());
240 assertEquals(mDisplayContent, token.getDisplayContent());
Andrii Kulian6cc1a1d2016-12-27 23:52:59 -0800241 }
Andrii Kuliand68501e2017-01-10 22:57:27 -0800242
243 /**
244 * This tests override configuration updates for display content.
245 */
246 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700247 public void testDisplayOverrideConfigUpdate() {
Evan Roskydfe3da72018-10-26 17:21:06 -0700248 final Configuration currentOverrideConfig =
249 mDisplayContent.getRequestedOverrideConfiguration();
Andrii Kuliand68501e2017-01-10 22:57:27 -0800250
251 // Create new, slightly changed override configuration and apply it to the display.
252 final Configuration newOverrideConfig = new Configuration(currentOverrideConfig);
253 newOverrideConfig.densityDpi += 120;
254 newOverrideConfig.fontScale += 0.3;
255
Evan Roskye747c3e2018-10-30 20:06:41 -0700256 mWm.setNewDisplayOverrideConfiguration(newOverrideConfig, mDisplayContent);
Andrii Kuliand68501e2017-01-10 22:57:27 -0800257
258 // Check that override config is applied.
Evan Roskydfe3da72018-10-26 17:21:06 -0700259 assertEquals(newOverrideConfig, mDisplayContent.getRequestedOverrideConfiguration());
Andrii Kulian367ff7f2017-01-25 19:45:34 -0800260 }
261
262 /**
263 * This tests global configuration updates when default display config is updated.
264 */
265 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700266 public void testDefaultDisplayOverrideConfigUpdate() {
Evan Roskye747c3e2018-10-30 20:06:41 -0700267 DisplayContent defaultDisplay = mWm.mRoot.getDisplayContent(DEFAULT_DISPLAY);
268 final Configuration currentConfig = defaultDisplay.getConfiguration();
Andrii Kulian367ff7f2017-01-25 19:45:34 -0800269
270 // Create new, slightly changed override configuration and apply it to the display.
Andrii Kuliana95bfff2017-03-30 19:00:41 -0700271 final Configuration newOverrideConfig = new Configuration(currentConfig);
Andrii Kulian367ff7f2017-01-25 19:45:34 -0800272 newOverrideConfig.densityDpi += 120;
273 newOverrideConfig.fontScale += 0.3;
274
Evan Roskye747c3e2018-10-30 20:06:41 -0700275 mWm.setNewDisplayOverrideConfiguration(newOverrideConfig, defaultDisplay);
Andrii Kuliand68501e2017-01-10 22:57:27 -0800276
277 // Check that global configuration is updated, as we've updated default display's config.
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700278 Configuration globalConfig = mWm.mRoot.getConfiguration();
Andrii Kuliand68501e2017-01-10 22:57:27 -0800279 assertEquals(newOverrideConfig.densityDpi, globalConfig.densityDpi);
280 assertEquals(newOverrideConfig.fontScale, globalConfig.fontScale, 0.1 /* delta */);
Andrii Kulian367ff7f2017-01-25 19:45:34 -0800281
282 // Return back to original values.
Evan Roskye747c3e2018-10-30 20:06:41 -0700283 mWm.setNewDisplayOverrideConfiguration(currentConfig, defaultDisplay);
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700284 globalConfig = mWm.mRoot.getConfiguration();
Andrii Kuliana95bfff2017-03-30 19:00:41 -0700285 assertEquals(currentConfig.densityDpi, globalConfig.densityDpi);
286 assertEquals(currentConfig.fontScale, globalConfig.fontScale, 0.1 /* delta */);
Andrii Kuliand68501e2017-01-10 22:57:27 -0800287 }
Wale Ogunwale34247952017-02-19 11:57:53 -0800288
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700289 /**
290 * Tests tapping on a stack in different display results in window gaining focus.
291 */
292 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700293 public void testInputEventBringsCorrectDisplayInFocus() {
294 DisplayContent dc0 = mWm.getDefaultDisplayContentLocked();
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700295 // Create a second display
296 final DisplayContent dc1 = createNewDisplay();
297
298 // Add stack with activity.
299 final TaskStack stack0 = createTaskStackOnDisplay(dc0);
300 final Task task0 = createTaskInStack(stack0, 0 /* userId */);
301 final WindowTestUtils.TestAppWindowToken token =
chaviw97d28202018-02-27 16:23:53 -0800302 WindowTestUtils.createTestAppWindowToken(dc0);
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700303 task0.addChild(token, 0);
Arthur Hungbe5ce212018-09-13 18:41:56 +0800304 dc0.configureDisplayPolicy();
305 assertNotNull(dc0.mTapDetector);
306
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700307 final TaskStack stack1 = createTaskStackOnDisplay(dc1);
308 final Task task1 = createTaskInStack(stack1, 0 /* userId */);
309 final WindowTestUtils.TestAppWindowToken token1 =
chaviw97d28202018-02-27 16:23:53 -0800310 WindowTestUtils.createTestAppWindowToken(dc0);
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700311 task1.addChild(token1, 0);
Arthur Hungbe5ce212018-09-13 18:41:56 +0800312 dc1.configureDisplayPolicy();
313 assertNotNull(dc1.mTapDetector);
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700314
Arthur Hungbe5ce212018-09-13 18:41:56 +0800315 // tap on primary display.
316 tapOnDisplay(dc0);
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700317 // Check focus is on primary display.
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700318 assertEquals(mWm.mRoot.getTopFocusedDisplayContent().mCurrentFocus,
Tiger Huang1e5b10a2018-07-30 20:19:51 +0800319 dc0.findFocusedWindow());
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700320
Arthur Hungbe5ce212018-09-13 18:41:56 +0800321 // Tap on secondary display.
322 tapOnDisplay(dc1);
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700323 // Check focus is on secondary.
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700324 assertEquals(mWm.mRoot.getTopFocusedDisplayContent().mCurrentFocus,
Tiger Huang1e5b10a2018-07-30 20:19:51 +0800325 dc1.findFocusedWindow());
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700326 }
327
David Stevens46939562017-03-24 13:04:00 -0700328 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700329 public void testFocusedWindowMultipleDisplays() {
Tiger Huang51c5a1d2018-12-11 20:24:51 +0800330 doTestFocusedWindowMultipleDisplays(false /* perDisplayFocusEnabled */, Q);
331 }
332
333 @Test
334 public void testFocusedWindowMultipleDisplaysPerDisplayFocusEnabled() {
335 doTestFocusedWindowMultipleDisplays(true /* perDisplayFocusEnabled */, Q);
336 }
337
338 @Test
339 public void testFocusedWindowMultipleDisplaysPerDisplayFocusEnabledLegacyApp() {
340 doTestFocusedWindowMultipleDisplays(true /* perDisplayFocusEnabled */, P);
341 }
342
343 private void doTestFocusedWindowMultipleDisplays(boolean perDisplayFocusEnabled,
344 int targetSdk) {
345 mWm.mPerDisplayFocusEnabled = perDisplayFocusEnabled;
346
Andrii Kulian0214ed92017-05-16 13:44:05 -0700347 // Create a focusable window and check that focus is calculated correctly
David Stevens46939562017-03-24 13:04:00 -0700348 final WindowState window1 =
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700349 createWindow(null, TYPE_BASE_APPLICATION, mDisplayContent, "window1");
Tiger Huang51c5a1d2018-12-11 20:24:51 +0800350 window1.mAppToken.mTargetSdk = targetSdk;
Tiger Huang1e5b10a2018-07-30 20:19:51 +0800351 updateFocusedWindow();
352 assertTrue(window1.isFocused());
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700353 assertEquals(window1, mWm.mRoot.getTopFocusedDisplayContent().mCurrentFocus);
David Stevens46939562017-03-24 13:04:00 -0700354
355 // Check that a new display doesn't affect focus
356 final DisplayContent dc = createNewDisplay();
Tiger Huang1e5b10a2018-07-30 20:19:51 +0800357 updateFocusedWindow();
358 assertTrue(window1.isFocused());
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700359 assertEquals(window1, mWm.mRoot.getTopFocusedDisplayContent().mCurrentFocus);
David Stevens46939562017-03-24 13:04:00 -0700360
361 // Add a window to the second display, and it should be focused
362 final WindowState window2 = createWindow(null, TYPE_BASE_APPLICATION, dc, "window2");
Tiger Huang51c5a1d2018-12-11 20:24:51 +0800363 window2.mAppToken.mTargetSdk = targetSdk;
Tiger Huang1e5b10a2018-07-30 20:19:51 +0800364 updateFocusedWindow();
Tiger Huang1e5b10a2018-07-30 20:19:51 +0800365 assertTrue(window2.isFocused());
Tiger Huang51c5a1d2018-12-11 20:24:51 +0800366 assertEquals(perDisplayFocusEnabled && targetSdk >= Q, window1.isFocused());
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700367 assertEquals(window2, mWm.mRoot.getTopFocusedDisplayContent().mCurrentFocus);
David Stevens46939562017-03-24 13:04:00 -0700368
Tiger Huang51c5a1d2018-12-11 20:24:51 +0800369 // Move the first window to top including parents, and make sure focus is updated
David Stevens46939562017-03-24 13:04:00 -0700370 window1.getParent().positionChildAt(POSITION_TOP, window1, true);
Tiger Huang1e5b10a2018-07-30 20:19:51 +0800371 updateFocusedWindow();
372 assertTrue(window1.isFocused());
Tiger Huang51c5a1d2018-12-11 20:24:51 +0800373 assertEquals(perDisplayFocusEnabled && targetSdk >= Q, window2.isFocused());
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700374 assertEquals(window1, mWm.mRoot.getTopFocusedDisplayContent().mCurrentFocus);
Adrian Roos4163d622018-05-22 16:56:35 +0200375 }
376
Bryce Lee27cec322017-03-21 09:41:37 -0700377 /**
378 * This tests setting the maximum ui width on a display.
379 */
380 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700381 public void testMaxUiWidth() {
Riddle Hsu654a6f92018-07-13 22:59:36 +0800382 // Prevent base display metrics for test from being updated to the value of real display.
383 final DisplayContent displayContent = createDisplayNoUpdateDisplayInfo();
Bryce Lee27cec322017-03-21 09:41:37 -0700384 final int baseWidth = 1440;
385 final int baseHeight = 2560;
386 final int baseDensity = 300;
387
Riddle Hsu654a6f92018-07-13 22:59:36 +0800388 displayContent.updateBaseDisplayMetrics(baseWidth, baseHeight, baseDensity);
Bryce Lee27cec322017-03-21 09:41:37 -0700389
390 final int maxWidth = 300;
391 final int resultingHeight = (maxWidth * baseHeight) / baseWidth;
392 final int resultingDensity = (maxWidth * baseDensity) / baseWidth;
393
Riddle Hsu654a6f92018-07-13 22:59:36 +0800394 displayContent.setMaxUiWidth(maxWidth);
395 verifySizes(displayContent, maxWidth, resultingHeight, resultingDensity);
Bryce Lee27cec322017-03-21 09:41:37 -0700396
397 // Assert setting values again does not change;
Riddle Hsu654a6f92018-07-13 22:59:36 +0800398 displayContent.updateBaseDisplayMetrics(baseWidth, baseHeight, baseDensity);
399 verifySizes(displayContent, maxWidth, resultingHeight, resultingDensity);
Bryce Lee27cec322017-03-21 09:41:37 -0700400
401 final int smallerWidth = 200;
402 final int smallerHeight = 400;
403 final int smallerDensity = 100;
404
405 // Specify smaller dimension, verify that it is honored
Riddle Hsu654a6f92018-07-13 22:59:36 +0800406 displayContent.updateBaseDisplayMetrics(smallerWidth, smallerHeight, smallerDensity);
407 verifySizes(displayContent, smallerWidth, smallerHeight, smallerDensity);
Bryce Lee27cec322017-03-21 09:41:37 -0700408
409 // Verify that setting the max width to a greater value than the base width has no effect
Riddle Hsu654a6f92018-07-13 22:59:36 +0800410 displayContent.setMaxUiWidth(maxWidth);
411 verifySizes(displayContent, smallerWidth, smallerHeight, smallerDensity);
Bryce Lee27cec322017-03-21 09:41:37 -0700412 }
413
Andrii Kulian92c9a942017-10-10 00:41:41 -0700414 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700415 public void testDisplayCutout_rot0() {
416 synchronized (mWm.getWindowManagerLock()) {
Adrian Roos1cf585052018-01-03 18:43:27 +0100417 final DisplayContent dc = createNewDisplay();
418 dc.mInitialDisplayWidth = 200;
419 dc.mInitialDisplayHeight = 400;
Adrian Roos24264212018-02-19 16:26:15 +0100420 Rect r = new Rect(80, 0, 120, 10);
Adrian Roos6a4fa0e2018-03-05 19:50:16 +0100421 final DisplayCutout cutout = new WmDisplayCutout(
Issei Suzuki43190bd2018-08-20 17:28:41 +0200422 fromBoundingRect(r.left, r.top, r.right, r.bottom, BOUNDS_POSITION_TOP), null)
Adrian Roos6a4fa0e2018-03-05 19:50:16 +0100423 .computeSafeInsets(200, 400).getDisplayCutout();
Adrian Roos1cf585052018-01-03 18:43:27 +0100424
425 dc.mInitialDisplayCutout = cutout;
426 dc.setRotation(Surface.ROTATION_0);
427 dc.computeScreenConfiguration(new Configuration()); // recomputes dc.mDisplayInfo.
428
429 assertEquals(cutout, dc.getDisplayInfo().displayCutout);
430 }
431 }
432
433 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700434 public void testDisplayCutout_rot90() {
435 synchronized (mWm.getWindowManagerLock()) {
Riddle Hsu654a6f92018-07-13 22:59:36 +0800436 // Prevent mInitialDisplayCutout from being updated from real display (e.g. null
437 // if the device has no cutout).
438 final DisplayContent dc = createDisplayNoUpdateDisplayInfo();
439 // Rotation may use real display info to compute bound, so here also uses the
440 // same width and height.
441 final int displayWidth = dc.mInitialDisplayWidth;
442 final int displayHeight = dc.mInitialDisplayHeight;
443 final int cutoutWidth = 40;
444 final int cutoutHeight = 10;
445 final int left = (displayWidth - cutoutWidth) / 2;
446 final int top = 0;
447 final int right = (displayWidth + cutoutWidth) / 2;
448 final int bottom = cutoutHeight;
449
450 final Rect r1 = new Rect(left, top, right, bottom);
Adrian Roos6a4fa0e2018-03-05 19:50:16 +0100451 final DisplayCutout cutout = new WmDisplayCutout(
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700452 fromBoundingRect(r1.left, r1.top, r1.right, r1.bottom, BOUNDS_POSITION_TOP),
453 null)
Riddle Hsu654a6f92018-07-13 22:59:36 +0800454 .computeSafeInsets(displayWidth, displayHeight).getDisplayCutout();
Adrian Roos1cf585052018-01-03 18:43:27 +0100455
456 dc.mInitialDisplayCutout = cutout;
457 dc.setRotation(Surface.ROTATION_90);
458 dc.computeScreenConfiguration(new Configuration()); // recomputes dc.mDisplayInfo.
459
Riddle Hsu654a6f92018-07-13 22:59:36 +0800460 // ----o---------- -------------
461 // | | | | |
462 // | ------o | o---
463 // | | | |
464 // | | -> | |
465 // | | ---o
466 // | | |
467 // | | -------------
468 final Rect r = new Rect(top, left, bottom, right);
Adrian Roos6a4fa0e2018-03-05 19:50:16 +0100469 assertEquals(new WmDisplayCutout(
Issei Suzuki43190bd2018-08-20 17:28:41 +0200470 fromBoundingRect(r.left, r.top, r.right, r.bottom, BOUNDS_POSITION_LEFT), null)
Riddle Hsu654a6f92018-07-13 22:59:36 +0800471 .computeSafeInsets(displayHeight, displayWidth)
472 .getDisplayCutout(), dc.getDisplayInfo().displayCutout);
Adrian Roos1cf585052018-01-03 18:43:27 +0100473 }
474 }
475
476 @Test
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700477 public void testLayoutSeq_assignedDuringLayout() {
478 synchronized (mWm.getWindowManagerLock()) {
Adrian Roos5251b1d2018-03-23 18:57:43 +0100479
480 final DisplayContent dc = createNewDisplay();
481 final WindowState win = createWindow(null /* parent */, TYPE_BASE_APPLICATION, dc, "w");
482
483 dc.setLayoutNeeded();
484 dc.performLayout(true /* initial */, false /* updateImeWindows */);
485
486 assertThat(win.mLayoutSeq, is(dc.mLayoutSeq));
487 }
488 }
489
490 @Test
Andrii Kulian92c9a942017-10-10 00:41:41 -0700491 @SuppressLint("InlinedApi")
492 public void testOrientationDefinedByKeyguard() {
493 final DisplayContent dc = createNewDisplay();
Garfield Tan90b04282018-12-11 14:04:42 -0800494
495 // When display content is created its configuration is not yet initialized, which could
496 // cause unnecessary configuration propagation, so initialize it here.
497 final Configuration config = new Configuration();
498 dc.computeScreenConfiguration(config);
499 dc.onRequestedOverrideConfigurationChanged(config);
500
Andrii Kulian92c9a942017-10-10 00:41:41 -0700501 // Create a window that requests landscape orientation. It will define device orientation
502 // by default.
503 final WindowState window = createWindow(null /* parent */, TYPE_BASE_APPLICATION, dc, "w");
504 window.mAppToken.setOrientation(SCREEN_ORIENTATION_LANDSCAPE);
505
506 final WindowState keyguard = createWindow(null, TYPE_STATUS_BAR, dc, "keyguard");
507 keyguard.mHasSurface = true;
508 keyguard.mAttrs.screenOrientation = SCREEN_ORIENTATION_UNSPECIFIED;
509
510 assertEquals("Screen orientation must be defined by the app window by default",
511 SCREEN_ORIENTATION_LANDSCAPE, dc.getOrientation());
512
513 keyguard.mAttrs.screenOrientation = SCREEN_ORIENTATION_PORTRAIT;
514 assertEquals("Visible keyguard must influence device orientation",
515 SCREEN_ORIENTATION_PORTRAIT, dc.getOrientation());
516
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700517 mWm.setKeyguardGoingAway(true);
Andrii Kulian92c9a942017-10-10 00:41:41 -0700518 assertEquals("Keyguard that is going away must not influence device orientation",
519 SCREEN_ORIENTATION_LANDSCAPE, dc.getOrientation());
520 }
521
Andrii Kulianf0379de2018-03-14 16:24:07 -0700522 @Test
523 public void testDisableDisplayInfoOverrideFromWindowManager() {
524 final DisplayContent dc = createNewDisplay();
525
526 assertTrue(dc.mShouldOverrideDisplayConfiguration);
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700527 mWm.dontOverrideDisplayInfo(dc.getDisplayId());
Andrii Kulianf0379de2018-03-14 16:24:07 -0700528
529 assertFalse(dc.mShouldOverrideDisplayConfiguration);
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700530 verify(mWm.mDisplayManagerInternal, times(1))
Andrii Kulianf0379de2018-03-14 16:24:07 -0700531 .setDisplayInfoOverrideFromWindowManager(dc.getDisplayId(), null);
532 }
533
Riddle Hsua4d6fa22018-08-11 00:50:39 +0800534 @Test
Riddle Hsu85bd04b2018-11-17 00:34:36 +0800535 public void testClearLastFocusWhenReparentingFocusedWindow() {
536 final DisplayContent defaultDisplay = mWm.getDefaultDisplayContentLocked();
537 final WindowState window = createWindow(null /* parent */, TYPE_BASE_APPLICATION,
538 defaultDisplay, "window");
539 defaultDisplay.mLastFocus = window;
540 mDisplayContent.mCurrentFocus = window;
541 mDisplayContent.reParentWindowToken(window.mToken);
542
543 assertNull(defaultDisplay.mLastFocus);
544 }
545
546 @Test
Riddle Hsua4d6fa22018-08-11 00:50:39 +0800547 public void testGetPreferredOptionsPanelGravityFromDifferentDisplays() {
548 final DisplayContent portraitDisplay = createNewDisplay();
549 portraitDisplay.mInitialDisplayHeight = 2000;
550 portraitDisplay.mInitialDisplayWidth = 1000;
551
552 portraitDisplay.setRotation(Surface.ROTATION_0);
553 assertFalse(isOptionsPanelAtRight(portraitDisplay.getDisplayId()));
554 portraitDisplay.setRotation(Surface.ROTATION_90);
555 assertTrue(isOptionsPanelAtRight(portraitDisplay.getDisplayId()));
556
557 final DisplayContent landscapeDisplay = createNewDisplay();
558 landscapeDisplay.mInitialDisplayHeight = 1000;
559 landscapeDisplay.mInitialDisplayWidth = 2000;
560
561 landscapeDisplay.setRotation(Surface.ROTATION_0);
562 assertTrue(isOptionsPanelAtRight(landscapeDisplay.getDisplayId()));
563 landscapeDisplay.setRotation(Surface.ROTATION_90);
564 assertFalse(isOptionsPanelAtRight(landscapeDisplay.getDisplayId()));
565 }
566
lumarkff0ab692018-11-05 20:32:30 +0800567 @Test
568 public void testInputMethodTargetUpdateWhenSwitchingOnDisplays() {
569 final DisplayContent newDisplay = createNewDisplay();
570
571 final WindowState appWin = createWindow(null, TYPE_APPLICATION, mDisplayContent, "appWin");
572 final WindowState appWin1 = createWindow(null, TYPE_APPLICATION, newDisplay, "appWin1");
573 appWin.setHasSurface(true);
574 appWin1.setHasSurface(true);
575
576 // Set current input method window on default display, make sure the input method target
577 // is appWin & null on the other display.
578 mDisplayContent.setInputMethodWindowLocked(mImeWindow);
579 newDisplay.setInputMethodWindowLocked(null);
580 assertTrue("appWin should be IME target window",
581 appWin.equals(mDisplayContent.mInputMethodTarget));
582 assertNull("newDisplay Ime target: ", newDisplay.mInputMethodTarget);
583
584 // Switch input method window on new display & make sure the input method target also
585 // switched as expected.
586 newDisplay.setInputMethodWindowLocked(mImeWindow);
587 mDisplayContent.setInputMethodWindowLocked(null);
588 assertTrue("appWin1 should be IME target window",
589 appWin1.equals(newDisplay.mInputMethodTarget));
590 assertNull("default display Ime target: ", mDisplayContent.mInputMethodTarget);
591 }
592
Garfield Tan90b04282018-12-11 14:04:42 -0800593 @Test
594 public void testOnDescendantOrientationRequestChanged() {
Garfield Tan49dae102019-02-04 09:51:59 -0800595 final DisplayContent dc = createNewDisplay();
Garfield Tan90b04282018-12-11 14:04:42 -0800596 mWm.mAtmService.mRootActivityContainer = mock(RootActivityContainer.class);
Garfield Tan49dae102019-02-04 09:51:59 -0800597 final int newOrientation = dc.getLastOrientation() == SCREEN_ORIENTATION_LANDSCAPE
598 ? SCREEN_ORIENTATION_PORTRAIT
599 : SCREEN_ORIENTATION_LANDSCAPE;
Garfield Tan90b04282018-12-11 14:04:42 -0800600
601 final WindowState window = createWindow(null /* parent */, TYPE_BASE_APPLICATION, dc, "w");
602 window.getTask().mTaskRecord = mock(TaskRecord.class, ExtendedMockito.RETURNS_DEEP_STUBS);
Garfield Tan49dae102019-02-04 09:51:59 -0800603 window.mAppToken.setOrientation(newOrientation);
Garfield Tan90b04282018-12-11 14:04:42 -0800604
605 ActivityRecord activityRecord = mock(ActivityRecord.class);
606
607 assertTrue("Display should rotate to handle orientation request by default.",
608 dc.onDescendantOrientationChanged(window.mToken.token, activityRecord));
609
610 final ArgumentCaptor<Configuration> captor = ArgumentCaptor.forClass(Configuration.class);
611 verify(mWm.mAtmService).updateDisplayOverrideConfigurationLocked(captor.capture(),
612 same(activityRecord), anyBoolean(), eq(dc.getDisplayId()));
613 final Configuration newDisplayConfig = captor.getValue();
Garfield Tan49dae102019-02-04 09:51:59 -0800614 assertEquals(Configuration.ORIENTATION_PORTRAIT, newDisplayConfig.orientation);
Garfield Tan90b04282018-12-11 14:04:42 -0800615 }
616
617 @Test
618 public void testOnDescendantOrientationRequestChanged_FrozenToUserRotation() {
Garfield Tan49dae102019-02-04 09:51:59 -0800619 final DisplayContent dc = createNewDisplay();
Garfield Tan90b04282018-12-11 14:04:42 -0800620 dc.getDisplayRotation().setFixedToUserRotation(true);
621 mWm.mAtmService.mRootActivityContainer = mock(RootActivityContainer.class);
Garfield Tan49dae102019-02-04 09:51:59 -0800622 final int newOrientation = dc.getLastOrientation() == SCREEN_ORIENTATION_LANDSCAPE
623 ? SCREEN_ORIENTATION_PORTRAIT
624 : SCREEN_ORIENTATION_LANDSCAPE;
Garfield Tan90b04282018-12-11 14:04:42 -0800625
626 final WindowState window = createWindow(null /* parent */, TYPE_BASE_APPLICATION, dc, "w");
627 window.getTask().mTaskRecord = mock(TaskRecord.class, ExtendedMockito.RETURNS_DEEP_STUBS);
Garfield Tan49dae102019-02-04 09:51:59 -0800628 window.mAppToken.setOrientation(newOrientation);
Garfield Tan90b04282018-12-11 14:04:42 -0800629
630 ActivityRecord activityRecord = mock(ActivityRecord.class);
631
632 assertFalse("Display shouldn't rotate to handle orientation request if fixed to"
633 + " user rotation.",
634 dc.onDescendantOrientationChanged(window.mToken.token, activityRecord));
635 verify(mWm.mAtmService, never()).updateDisplayOverrideConfigurationLocked(any(),
636 eq(activityRecord), anyBoolean(), eq(dc.getDisplayId()));
637 }
638
Tarandeep Singha6f35612019-01-11 19:50:46 -0800639 @Test
640 public void testComputeImeParent_app() throws Exception {
641 try (final InsetsModeSession session =
642 new InsetsModeSession(ViewRootImpl.NEW_INSETS_MODE_IME)) {
643 final DisplayContent dc = createNewDisplay();
644 dc.mInputMethodTarget = createWindow(null, TYPE_BASE_APPLICATION, "app");
645 assertEquals(dc.mInputMethodTarget.mAppToken.getSurfaceControl(),
646 dc.computeImeParent());
647 }
648 }
649
650 @Test
651 public void testComputeImeParent_app_notFullscreen() throws Exception {
652 try (final InsetsModeSession session =
653 new InsetsModeSession(ViewRootImpl.NEW_INSETS_MODE_IME)) {
654 final DisplayContent dc = createNewDisplay();
655 dc.mInputMethodTarget = createWindow(null, TYPE_STATUS_BAR, "app");
656 dc.mInputMethodTarget.setWindowingMode(
657 WindowConfiguration.WINDOWING_MODE_SPLIT_SCREEN_PRIMARY);
658 assertEquals(dc.getWindowingLayer(), dc.computeImeParent());
659 }
660 }
661
662 @Test
663 public void testComputeImeParent_noApp() throws Exception {
664 try (final InsetsModeSession session =
665 new InsetsModeSession(ViewRootImpl.NEW_INSETS_MODE_IME)) {
666 final DisplayContent dc = createNewDisplay();
667 dc.mInputMethodTarget = createWindow(null, TYPE_STATUS_BAR, "statusBar");
668 assertEquals(dc.getWindowingLayer(), dc.computeImeParent());
669 }
670 }
671
Riddle Hsua4d6fa22018-08-11 00:50:39 +0800672 private boolean isOptionsPanelAtRight(int displayId) {
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700673 return (mWm.getPreferredOptionsPanelGravity(displayId) & Gravity.RIGHT) == Gravity.RIGHT;
Riddle Hsua4d6fa22018-08-11 00:50:39 +0800674 }
675
Bryce Lee27cec322017-03-21 09:41:37 -0700676 private static void verifySizes(DisplayContent displayContent, int expectedBaseWidth,
677 int expectedBaseHeight, int expectedBaseDensity) {
678 assertEquals(displayContent.mBaseDisplayWidth, expectedBaseWidth);
679 assertEquals(displayContent.mBaseDisplayHeight, expectedBaseHeight);
680 assertEquals(displayContent.mBaseDisplayDensity, expectedBaseDensity);
681 }
682
Tiger Huang1e5b10a2018-07-30 20:19:51 +0800683 private void updateFocusedWindow() {
Tadashi G. Takaokab6e148c2018-11-03 02:59:06 -0700684 synchronized (mWm.mGlobalLock) {
685 mWm.updateFocusedWindowLocked(UPDATE_FOCUS_NORMAL, false);
Tiger Huang1e5b10a2018-07-30 20:19:51 +0800686 }
687 }
688
Riddle Hsu654a6f92018-07-13 22:59:36 +0800689 /**
690 * Create DisplayContent that does not update display base/initial values from device to keep
691 * the values set by test.
692 */
693 private DisplayContent createDisplayNoUpdateDisplayInfo() {
694 final DisplayContent displayContent = spy(createNewDisplay());
695 doNothing().when(displayContent).updateDisplayInfo();
696 return displayContent;
697 }
698
Adrian Roos0f9368c2018-04-08 10:59:08 -0700699 private void assertForAllWindowsOrder(List<WindowState> expectedWindowsBottomToTop) {
700 final LinkedList<WindowState> actualWindows = new LinkedList<>();
Wale Ogunwale34247952017-02-19 11:57:53 -0800701
702 // Test forward traversal.
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700703 mDisplayContent.forAllWindows(actualWindows::addLast, false /* traverseTopToBottom */);
Adrian Roos0f9368c2018-04-08 10:59:08 -0700704 assertThat("bottomToTop", actualWindows, is(expectedWindowsBottomToTop));
705
706 actualWindows.clear();
Wale Ogunwale34247952017-02-19 11:57:53 -0800707
708 // Test backward traversal.
Wale Ogunwale11cc5162017-04-25 20:29:13 -0700709 mDisplayContent.forAllWindows(actualWindows::addLast, true /* traverseTopToBottom */);
Adrian Roos0f9368c2018-04-08 10:59:08 -0700710 assertThat("topToBottom", actualWindows, is(reverseList(expectedWindowsBottomToTop)));
711 }
712
713 private static List<WindowState> reverseList(List<WindowState> list) {
714 final ArrayList<WindowState> result = new ArrayList<>(list);
715 Collections.reverse(result);
716 return result;
Wale Ogunwale34247952017-02-19 11:57:53 -0800717 }
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700718
Arthur Hungbe5ce212018-09-13 18:41:56 +0800719 private void tapOnDisplay(final DisplayContent dc) {
720 final DisplayMetrics dm = dc.getDisplayMetrics();
721 final float x = dm.widthPixels / 2;
722 final float y = dm.heightPixels / 2;
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700723 final long downTime = SystemClock.uptimeMillis();
724 final long eventTime = SystemClock.uptimeMillis() + 100;
Arthur Hungbe5ce212018-09-13 18:41:56 +0800725 // sending ACTION_DOWN
726 final MotionEvent downEvent = MotionEvent.obtain(
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700727 downTime,
Arthur Hungbe5ce212018-09-13 18:41:56 +0800728 downTime,
729 MotionEvent.ACTION_DOWN,
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700730 x,
731 y,
Arthur Hungbe5ce212018-09-13 18:41:56 +0800732 0 /*metaState*/);
733 downEvent.setDisplayId(dc.getDisplayId());
734 dc.mTapDetector.onPointerEvent(downEvent);
735
736 // sending ACTION_UP
737 final MotionEvent upEvent = MotionEvent.obtain(
738 downTime,
739 eventTime,
740 MotionEvent.ACTION_UP,
741 x,
742 y,
743 0 /*metaState*/);
744 upEvent.setDisplayId(dc.getDisplayId());
745 dc.mTapDetector.onPointerEvent(upEvent);
Tarandeep Singhe1cfcf42017-07-10 18:50:00 -0700746 }
Wale Ogunwaleb783fd82016-11-04 09:51:54 -0700747}