| /* |
| * 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.WINDOWING_MODE_FREEFORM; |
| import static android.app.WindowConfiguration.WINDOWING_MODE_MULTI_WINDOW; |
| import static android.app.WindowConfiguration.WINDOWING_MODE_PINNED; |
| import static android.app.WindowConfiguration.WINDOWING_MODE_SPLIT_SCREEN_PRIMARY; |
| import static android.view.InsetsState.ITYPE_IME; |
| import static android.view.InsetsState.ITYPE_NAVIGATION_BAR; |
| import static android.view.InsetsState.ITYPE_STATUS_BAR; |
| import static android.view.ViewRootImpl.NEW_INSETS_MODE_FULL; |
| import static android.view.WindowManager.LayoutParams.FLAG_ALT_FOCUSABLE_IM; |
| import static android.view.WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE; |
| import static android.view.WindowManager.LayoutParams.TYPE_APPLICATION; |
| |
| import static org.junit.Assert.assertEquals; |
| import static org.junit.Assert.assertFalse; |
| import static org.junit.Assert.assertNotNull; |
| import static org.junit.Assert.assertNull; |
| import static org.junit.Assert.assertTrue; |
| import static org.mockito.Mockito.atLeast; |
| import static org.mockito.Mockito.clearInvocations; |
| import static org.mockito.Mockito.spy; |
| import static org.mockito.Mockito.verify; |
| |
| import android.graphics.Rect; |
| import android.platform.test.annotations.Presubmit; |
| import android.view.InsetsSourceControl; |
| import android.view.InsetsState; |
| import android.view.test.InsetsModeSession; |
| |
| import androidx.test.filters.SmallTest; |
| |
| import org.junit.AfterClass; |
| import org.junit.BeforeClass; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| |
| @SmallTest |
| @Presubmit |
| @RunWith(WindowTestRunner.class) |
| public class InsetsStateControllerTest extends WindowTestsBase { |
| private static InsetsModeSession sInsetsModeSession; |
| |
| @BeforeClass |
| public static void setUpOnce() { |
| // To let the insets provider control the insets visibility, the insets mode has to be |
| // NEW_INSETS_MODE_FULL. |
| sInsetsModeSession = new InsetsModeSession(NEW_INSETS_MODE_FULL); |
| } |
| |
| @AfterClass |
| public static void tearDownOnce() { |
| sInsetsModeSession.close(); |
| } |
| |
| @Test |
| public void testStripForDispatch_notOwn() { |
| final WindowState statusBar = createWindow(null, TYPE_APPLICATION, "statusBar"); |
| final WindowState app = createWindow(null, TYPE_APPLICATION, "app"); |
| getController().getSourceProvider(ITYPE_STATUS_BAR).setWindow(statusBar, null, null); |
| statusBar.setControllableInsetProvider(getController().getSourceProvider(ITYPE_STATUS_BAR)); |
| assertNotNull(getController().getInsetsForDispatch(app).peekSource(ITYPE_STATUS_BAR)); |
| } |
| |
| @Test |
| public void testStripForDispatch_own() { |
| final WindowState statusBar = createWindow(null, TYPE_APPLICATION, "statusBar"); |
| mDisplayContent.getInsetsStateController().getSourceProvider(ITYPE_STATUS_BAR) |
| .setWindow(statusBar, null, null); |
| statusBar.setControllableInsetProvider(getController().getSourceProvider(ITYPE_STATUS_BAR)); |
| final InsetsState state = getController().getInsetsForDispatch(statusBar); |
| assertNull(state.peekSource(ITYPE_STATUS_BAR)); |
| } |
| |
| @Test |
| public void testStripForDispatch_navBar() { |
| final WindowState navBar = createWindow(null, TYPE_APPLICATION, "navBar"); |
| final WindowState statusBar = createWindow(null, TYPE_APPLICATION, "statusBar"); |
| final WindowState ime = createWindow(null, TYPE_APPLICATION, "ime"); |
| |
| // IME cannot be the IME target. |
| ime.mAttrs.flags |= FLAG_NOT_FOCUSABLE; |
| |
| getController().getSourceProvider(ITYPE_STATUS_BAR).setWindow(statusBar, null, null); |
| getController().getSourceProvider(ITYPE_NAVIGATION_BAR).setWindow(navBar, null, null); |
| getController().getSourceProvider(ITYPE_IME).setWindow(ime, null, null); |
| assertNull(getController().getInsetsForDispatch(navBar).peekSource(ITYPE_IME)); |
| assertNull(getController().getInsetsForDispatch(navBar).peekSource(ITYPE_STATUS_BAR)); |
| } |
| |
| @Test |
| public void testStripForDispatch_pip() { |
| final WindowState statusBar = createWindow(null, TYPE_APPLICATION, "statusBar"); |
| final WindowState navBar = createWindow(null, TYPE_APPLICATION, "navBar"); |
| final WindowState app = createWindow(null, TYPE_APPLICATION, "app"); |
| |
| getController().getSourceProvider(ITYPE_STATUS_BAR).setWindow(statusBar, null, null); |
| getController().getSourceProvider(ITYPE_NAVIGATION_BAR).setWindow(navBar, null, null); |
| app.setWindowingMode(WINDOWING_MODE_PINNED); |
| |
| assertNull(getController().getInsetsForDispatch(app).peekSource(ITYPE_STATUS_BAR)); |
| assertNull(getController().getInsetsForDispatch(app).peekSource(ITYPE_NAVIGATION_BAR)); |
| } |
| |
| @Test |
| public void testStripForDispatch_freeform() { |
| final WindowState statusBar = createWindow(null, TYPE_APPLICATION, "statusBar"); |
| final WindowState navBar = createWindow(null, TYPE_APPLICATION, "navBar"); |
| final WindowState app = createWindow(null, TYPE_APPLICATION, "app"); |
| |
| getController().getSourceProvider(ITYPE_STATUS_BAR).setWindow(statusBar, null, null); |
| getController().getSourceProvider(ITYPE_NAVIGATION_BAR).setWindow(navBar, null, null); |
| app.setWindowingMode(WINDOWING_MODE_FREEFORM); |
| |
| assertNull(getController().getInsetsForDispatch(app).peekSource(ITYPE_STATUS_BAR)); |
| assertNull(getController().getInsetsForDispatch(app).peekSource(ITYPE_NAVIGATION_BAR)); |
| } |
| |
| @Test |
| public void testStripForDispatch_multiwindow_alwaysOnTop() { |
| final WindowState statusBar = createWindow(null, TYPE_APPLICATION, "statusBar"); |
| final WindowState navBar = createWindow(null, TYPE_APPLICATION, "navBar"); |
| final WindowState app = createWindow(null, TYPE_APPLICATION, "app"); |
| |
| getController().getSourceProvider(ITYPE_STATUS_BAR).setWindow(statusBar, null, null); |
| getController().getSourceProvider(ITYPE_NAVIGATION_BAR).setWindow(navBar, null, null); |
| app.setWindowingMode(WINDOWING_MODE_MULTI_WINDOW); |
| app.setAlwaysOnTop(true); |
| |
| assertNull(getController().getInsetsForDispatch(app).peekSource(ITYPE_STATUS_BAR)); |
| assertNull(getController().getInsetsForDispatch(app).peekSource(ITYPE_NAVIGATION_BAR)); |
| } |
| |
| @Test |
| public void testStripForDispatch_independentSources() { |
| getController().getSourceProvider(ITYPE_IME).setWindow(mImeWindow, null, null); |
| |
| final WindowState app1 = createWindow(null, TYPE_APPLICATION, "app1"); |
| app1.mBehindIme = true; |
| |
| final WindowState app2 = createWindow(null, TYPE_APPLICATION, "app2"); |
| app2.mBehindIme = false; |
| |
| getController().getRawInsetsState().setSourceVisible(ITYPE_IME, true); |
| assertFalse(getController().getInsetsForDispatch(app2).getSource(ITYPE_IME).isVisible()); |
| assertTrue(getController().getInsetsForDispatch(app1).getSource(ITYPE_IME).isVisible()); |
| } |
| |
| @Test |
| public void testStripForDispatch_belowIme() { |
| getController().getSourceProvider(ITYPE_IME).setWindow(mImeWindow, null, null); |
| |
| final WindowState app = createWindow(null, TYPE_APPLICATION, "app"); |
| app.mBehindIme = true; |
| |
| getController().getRawInsetsState().setSourceVisible(ITYPE_IME, true); |
| assertTrue(getController().getInsetsForDispatch(app).getSource(ITYPE_IME).isVisible()); |
| } |
| |
| @Test |
| public void testStripForDispatch_aboveIme() { |
| getController().getSourceProvider(ITYPE_IME).setWindow(mImeWindow, null, null); |
| |
| final WindowState app = createWindow(null, TYPE_APPLICATION, "app"); |
| app.mBehindIme = false; |
| |
| getController().getRawInsetsState().setSourceVisible(ITYPE_IME, true); |
| assertFalse(getController().getInsetsForDispatch(app).getSource(ITYPE_IME).isVisible()); |
| } |
| |
| @Test |
| public void testStripForDispatch_imeOrderChanged() { |
| // This can be the IME z-order target while app cannot be the IME z-order target. |
| // This is also the only IME control target in this test, so IME won't be invisible caused |
| // by the control-target change. |
| mDisplayContent.mInputMethodInputTarget = createWindow(null, TYPE_APPLICATION, "base"); |
| |
| // Make IME and stay visible during the test. |
| mImeWindow.setHasSurface(true); |
| getController().getSourceProvider(ITYPE_IME).setWindow(mImeWindow, null, null); |
| getController().onImeControlTargetChanged(mDisplayContent.mInputMethodInputTarget); |
| final InsetsState requestedState = new InsetsState(); |
| requestedState.getSource(ITYPE_IME).setVisible(true); |
| mDisplayContent.mInputMethodInputTarget.updateRequestedInsetsState(requestedState); |
| getController().onInsetsModified(mDisplayContent.mInputMethodInputTarget, requestedState); |
| |
| // Send our spy window (app) into the system so that we can detect the invocation. |
| final WindowState win = createWindow(null, TYPE_APPLICATION, "app"); |
| win.setHasSurface(true); |
| final WindowToken parent = win.mToken; |
| parent.removeChild(win); |
| final WindowState app = spy(win); |
| parent.addWindow(app); |
| |
| // Adding FLAG_NOT_FOCUSABLE makes app above IME. |
| app.mAttrs.flags |= FLAG_NOT_FOCUSABLE; |
| mDisplayContent.computeImeTarget(true); |
| mDisplayContent.applySurfaceChangesTransaction(); |
| |
| // app won't get visible IME insets while above IME even when IME is visible. |
| assertTrue(getController().getRawInsetsState().getSourceOrDefaultVisibility(ITYPE_IME)); |
| assertFalse(getController().getInsetsForDispatch(app).getSource(ITYPE_IME).isVisible()); |
| |
| // Reset invocation counter. |
| clearInvocations(app); |
| |
| // Removing FLAG_NOT_FOCUSABLE makes app below IME. |
| app.mAttrs.flags &= ~FLAG_NOT_FOCUSABLE; |
| mDisplayContent.computeImeTarget(true); |
| mDisplayContent.applySurfaceChangesTransaction(); |
| |
| // Make sure app got notified. |
| verify(app, atLeast(1)).notifyInsetsChanged(); |
| |
| // app will get visible IME insets while below IME. |
| assertTrue(getController().getInsetsForDispatch(app).getSource(ITYPE_IME).isVisible()); |
| } |
| |
| @Test |
| public void testStripForDispatch_childWindow_altFocusable() { |
| getController().getSourceProvider(ITYPE_IME).setWindow(mImeWindow, null, null); |
| |
| final WindowState app = createWindow(null, TYPE_APPLICATION, "app"); |
| final WindowState child = createWindow(app, TYPE_APPLICATION, "child"); |
| child.mAttrs.flags |= FLAG_ALT_FOCUSABLE_IM; |
| |
| mDisplayContent.computeImeTarget(true); |
| mDisplayContent.setLayoutNeeded(); |
| mDisplayContent.applySurfaceChangesTransaction(); |
| |
| getController().getRawInsetsState().setSourceVisible(ITYPE_IME, true); |
| assertTrue(getController().getInsetsForDispatch(app).getSource(ITYPE_IME).isVisible()); |
| assertFalse(getController().getInsetsForDispatch(child).getSource(ITYPE_IME).isVisible()); |
| } |
| |
| @Test |
| public void testStripForDispatch_childWindow_splitScreen() { |
| getController().getSourceProvider(ITYPE_IME).setWindow(mImeWindow, null, null); |
| |
| final WindowState app = createWindow(null, TYPE_APPLICATION, "app"); |
| final WindowState child = createWindow(app, TYPE_APPLICATION, "child"); |
| child.mAttrs.flags |= FLAG_NOT_FOCUSABLE; |
| child.setWindowingMode(WINDOWING_MODE_SPLIT_SCREEN_PRIMARY); |
| |
| mDisplayContent.computeImeTarget(true); |
| mDisplayContent.setLayoutNeeded(); |
| mDisplayContent.applySurfaceChangesTransaction(); |
| |
| getController().getRawInsetsState().setSourceVisible(ITYPE_IME, true); |
| assertTrue(getController().getInsetsForDispatch(app).getSource(ITYPE_IME).isVisible()); |
| assertFalse(getController().getInsetsForDispatch(child).getSource(ITYPE_IME).isVisible()); |
| } |
| |
| @Test |
| public void testImeForDispatch() { |
| final WindowState statusBar = createWindow(null, TYPE_APPLICATION, "statusBar"); |
| final WindowState ime = createWindow(null, TYPE_APPLICATION, "ime"); |
| |
| // IME cannot be the IME target. |
| ime.mAttrs.flags |= FLAG_NOT_FOCUSABLE; |
| |
| InsetsSourceProvider statusBarProvider = |
| getController().getSourceProvider(ITYPE_STATUS_BAR); |
| statusBarProvider.setWindow(statusBar, null, ((displayFrames, windowState, rect) -> |
| rect.set(0, 1, 2, 3))); |
| getController().getSourceProvider(ITYPE_IME).setWindow(ime, null, null); |
| statusBar.setControllableInsetProvider(statusBarProvider); |
| |
| statusBarProvider.onPostLayout(); |
| |
| final InsetsState state = getController().getInsetsForDispatch(ime); |
| assertEquals(new Rect(0, 1, 2, 3), state.getSource(ITYPE_STATUS_BAR).getFrame()); |
| } |
| |
| @Test |
| public void testBarControllingWinChanged() { |
| final WindowState navBar = createWindow(null, TYPE_APPLICATION, "navBar"); |
| final WindowState statusBar = createWindow(null, TYPE_APPLICATION, "statusBar"); |
| final WindowState app = createWindow(null, TYPE_APPLICATION, "app"); |
| getController().getSourceProvider(ITYPE_STATUS_BAR).setWindow(statusBar, null, null); |
| getController().getSourceProvider(ITYPE_NAVIGATION_BAR).setWindow(navBar, null, null); |
| getController().onBarControlTargetChanged(app, null, app, null); |
| InsetsSourceControl[] controls = getController().getControlsForDispatch(app); |
| assertEquals(2, controls.length); |
| } |
| |
| @Test |
| public void testControlRevoked() { |
| final WindowState statusBar = createWindow(null, TYPE_APPLICATION, "statusBar"); |
| final WindowState app = createWindow(null, TYPE_APPLICATION, "app"); |
| getController().getSourceProvider(ITYPE_STATUS_BAR).setWindow(statusBar, null, null); |
| getController().onBarControlTargetChanged(app, null, null, null); |
| assertNotNull(getController().getControlsForDispatch(app)); |
| getController().onBarControlTargetChanged(null, null, null, null); |
| assertNull(getController().getControlsForDispatch(app)); |
| } |
| |
| @Test |
| public void testControlRevoked_animation() { |
| final WindowState statusBar = createWindow(null, TYPE_APPLICATION, "statusBar"); |
| final WindowState app = createWindow(null, TYPE_APPLICATION, "app"); |
| getController().getSourceProvider(ITYPE_STATUS_BAR).setWindow(statusBar, null, null); |
| getController().onBarControlTargetChanged(app, null, null, null); |
| assertNotNull(getController().getControlsForDispatch(app)); |
| statusBar.cancelAnimation(); |
| assertNull(getController().getControlsForDispatch(app)); |
| } |
| |
| private InsetsStateController getController() { |
| return mDisplayContent.getInsetsStateController(); |
| } |
| } |