| /* |
| * Copyright (C) 2022 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.systemui.accessibility.floatingmenu; |
| |
| import static android.view.View.GONE; |
| import static android.view.View.VISIBLE; |
| import static android.view.WindowInsets.Type.displayCutout; |
| import static android.view.WindowInsets.Type.ime; |
| import static android.view.WindowInsets.Type.systemBars; |
| |
| import static com.android.systemui.accessibility.floatingmenu.MenuViewLayer.LayerIndex; |
| |
| import static com.google.common.truth.Truth.assertThat; |
| |
| import static org.mockito.Mockito.doReturn; |
| import static org.mockito.Mockito.spy; |
| import static org.mockito.Mockito.verify; |
| import static org.mockito.Mockito.when; |
| |
| import android.accessibilityservice.AccessibilityServiceInfo; |
| import android.content.ComponentName; |
| import android.content.pm.ApplicationInfo; |
| import android.content.pm.ResolveInfo; |
| import android.content.pm.ServiceInfo; |
| import android.graphics.Insets; |
| import android.graphics.PointF; |
| import android.graphics.Rect; |
| import android.os.Build; |
| import android.os.UserHandle; |
| import android.provider.Settings; |
| import android.testing.AndroidTestingRunner; |
| import android.testing.TestableLooper; |
| import android.view.View; |
| import android.view.WindowInsets; |
| import android.view.WindowManager; |
| import android.view.WindowMetrics; |
| import android.view.accessibility.AccessibilityManager; |
| |
| import androidx.test.filters.SmallTest; |
| |
| import com.android.systemui.SysuiTestCase; |
| import com.android.systemui.util.settings.SecureSettings; |
| |
| import org.junit.After; |
| import org.junit.Before; |
| import org.junit.Rule; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| import org.mockito.Mock; |
| import org.mockito.junit.MockitoJUnit; |
| import org.mockito.junit.MockitoRule; |
| |
| import java.util.ArrayList; |
| import java.util.List; |
| |
| /** Tests for {@link MenuViewLayer}. */ |
| @RunWith(AndroidTestingRunner.class) |
| @TestableLooper.RunWithLooper(setAsMainLooper = true) |
| @SmallTest |
| public class MenuViewLayerTest extends SysuiTestCase { |
| private static final String SELECT_TO_SPEAK_PACKAGE_NAME = "com.google.android.marvin.talkback"; |
| private static final String SELECT_TO_SPEAK_SERVICE_NAME = |
| "com.google.android.accessibility.selecttospeak.SelectToSpeakService"; |
| private static final ComponentName TEST_SELECT_TO_SPEAK_COMPONENT_NAME = new ComponentName( |
| SELECT_TO_SPEAK_PACKAGE_NAME, SELECT_TO_SPEAK_SERVICE_NAME); |
| |
| private static final int DISPLAY_WINDOW_WIDTH = 1080; |
| private static final int DISPLAY_WINDOW_HEIGHT = 2340; |
| private static final int STATUS_BAR_HEIGHT = 75; |
| private static final int NAVIGATION_BAR_HEIGHT = 125; |
| private static final int IME_HEIGHT = 350; |
| private static final int IME_TOP = |
| DISPLAY_WINDOW_HEIGHT - STATUS_BAR_HEIGHT - NAVIGATION_BAR_HEIGHT - IME_HEIGHT; |
| |
| private MenuViewLayer mMenuViewLayer; |
| private String mLastAccessibilityButtonTargets; |
| private String mLastEnabledAccessibilityServices; |
| private WindowMetrics mWindowMetrics; |
| private MenuView mMenuView; |
| private MenuAnimationController mMenuAnimationController; |
| |
| @Rule |
| public MockitoRule mockito = MockitoJUnit.rule(); |
| |
| @Mock |
| private IAccessibilityFloatingMenu mFloatingMenu; |
| |
| @Mock |
| private SecureSettings mSecureSettings; |
| |
| @Mock |
| private WindowManager mStubWindowManager; |
| |
| @Mock |
| private AccessibilityManager mStubAccessibilityManager; |
| |
| @Before |
| public void setUp() throws Exception { |
| final Rect mDisplayBounds = new Rect(); |
| mDisplayBounds.set(/* left= */ 0, /* top= */ 0, DISPLAY_WINDOW_WIDTH, |
| DISPLAY_WINDOW_HEIGHT); |
| mWindowMetrics = spy(new WindowMetrics(mDisplayBounds, fakeDisplayInsets())); |
| doReturn(mWindowMetrics).when(mStubWindowManager).getCurrentWindowMetrics(); |
| |
| mMenuViewLayer = new MenuViewLayer(mContext, mStubWindowManager, mStubAccessibilityManager, |
| mFloatingMenu, mSecureSettings); |
| mMenuView = (MenuView) mMenuViewLayer.getChildAt(LayerIndex.MENU_VIEW); |
| mMenuAnimationController = mMenuView.getMenuAnimationController(); |
| |
| mLastAccessibilityButtonTargets = |
| Settings.Secure.getStringForUser(mContext.getContentResolver(), |
| Settings.Secure.ACCESSIBILITY_BUTTON_TARGETS, UserHandle.USER_CURRENT); |
| mLastEnabledAccessibilityServices = |
| Settings.Secure.getStringForUser(mContext.getContentResolver(), |
| Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES, UserHandle.USER_CURRENT); |
| |
| mMenuViewLayer.onAttachedToWindow(); |
| Settings.Secure.putStringForUser(mContext.getContentResolver(), |
| Settings.Secure.ACCESSIBILITY_BUTTON_TARGETS, "", UserHandle.USER_CURRENT); |
| Settings.Secure.putStringForUser(mContext.getContentResolver(), |
| Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES, "", UserHandle.USER_CURRENT); |
| } |
| |
| @After |
| public void tearDown() throws Exception { |
| Settings.Secure.putStringForUser(mContext.getContentResolver(), |
| Settings.Secure.ACCESSIBILITY_BUTTON_TARGETS, mLastAccessibilityButtonTargets, |
| UserHandle.USER_CURRENT); |
| Settings.Secure.putStringForUser(mContext.getContentResolver(), |
| Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES, mLastEnabledAccessibilityServices, |
| UserHandle.USER_CURRENT); |
| |
| mMenuView.updateMenuMoveToTucked(/* isMoveToTucked= */ false); |
| mMenuViewLayer.onDetachedFromWindow(); |
| } |
| |
| @Test |
| public void onAttachedToWindow_menuIsVisible() { |
| mMenuViewLayer.onAttachedToWindow(); |
| final View menuView = mMenuViewLayer.getChildAt(LayerIndex.MENU_VIEW); |
| |
| assertThat(menuView.getVisibility()).isEqualTo(VISIBLE); |
| } |
| |
| @Test |
| public void onAttachedToWindow_menuIsGone() { |
| mMenuViewLayer.onDetachedFromWindow(); |
| final View menuView = mMenuViewLayer.getChildAt(LayerIndex.MENU_VIEW); |
| |
| assertThat(menuView.getVisibility()).isEqualTo(GONE); |
| } |
| |
| @Test |
| public void tiggerDismissMenuAction_hideFloatingMenu() { |
| mMenuViewLayer.mDismissMenuAction.run(); |
| |
| verify(mFloatingMenu).hide(); |
| } |
| |
| @Test |
| public void tiggerDismissMenuAction_matchA11yButtonTargetsResult() { |
| mMenuViewLayer.mDismissMenuAction.run(); |
| verify(mSecureSettings).putStringForUser( |
| Settings.Secure.ACCESSIBILITY_BUTTON_TARGETS, /* value= */ "", |
| UserHandle.USER_CURRENT); |
| } |
| |
| @Test |
| public void tiggerDismissMenuAction_matchEnabledA11yServicesResult() { |
| Settings.Secure.putString(mContext.getContentResolver(), |
| Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES, |
| TEST_SELECT_TO_SPEAK_COMPONENT_NAME.flattenToString()); |
| final ResolveInfo resolveInfo = new ResolveInfo(); |
| final ServiceInfo serviceInfo = new ServiceInfo(); |
| final ApplicationInfo applicationInfo = new ApplicationInfo(); |
| resolveInfo.serviceInfo = serviceInfo; |
| serviceInfo.applicationInfo = applicationInfo; |
| applicationInfo.targetSdkVersion = Build.VERSION_CODES.R; |
| final AccessibilityServiceInfo accessibilityServiceInfo = new AccessibilityServiceInfo(); |
| accessibilityServiceInfo.setResolveInfo(resolveInfo); |
| accessibilityServiceInfo.flags = AccessibilityServiceInfo.FLAG_REQUEST_ACCESSIBILITY_BUTTON; |
| final List<AccessibilityServiceInfo> serviceInfoList = new ArrayList<>(); |
| accessibilityServiceInfo.setComponentName(TEST_SELECT_TO_SPEAK_COMPONENT_NAME); |
| serviceInfoList.add(accessibilityServiceInfo); |
| when(mStubAccessibilityManager.getEnabledAccessibilityServiceList( |
| AccessibilityServiceInfo.FEEDBACK_ALL_MASK)).thenReturn(serviceInfoList); |
| |
| mMenuViewLayer.mDismissMenuAction.run(); |
| final String value = Settings.Secure.getString(mContext.getContentResolver(), |
| Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES); |
| |
| assertThat(value).isEqualTo(""); |
| } |
| |
| @Test |
| public void showingImeInsetsChange_notOverlapOnIme_menuKeepOriginalPosition() { |
| final float menuTop = STATUS_BAR_HEIGHT + 100; |
| mMenuAnimationController.moveAndPersistPosition(new PointF(0, menuTop)); |
| |
| dispatchShowingImeInsets(); |
| |
| assertThat(mMenuView.getTranslationX()).isEqualTo(0); |
| assertThat(mMenuView.getTranslationY()).isEqualTo(menuTop); |
| } |
| |
| @Test |
| public void showingImeInsetsChange_overlapOnIme_menuShownAboveIme() { |
| final float menuTop = IME_TOP + 100; |
| mMenuAnimationController.moveAndPersistPosition(new PointF(0, menuTop)); |
| |
| dispatchShowingImeInsets(); |
| |
| final float menuBottom = mMenuView.getTranslationY() + mMenuView.getMenuHeight(); |
| assertThat(mMenuView.getTranslationX()).isEqualTo(0); |
| assertThat(menuBottom).isLessThan(IME_TOP); |
| } |
| |
| @Test |
| public void hidingImeInsetsChange_overlapOnIme_menuBackToOriginalPosition() { |
| final float menuTop = IME_TOP + 200; |
| mMenuAnimationController.moveAndPersistPosition(new PointF(0, menuTop)); |
| dispatchShowingImeInsets(); |
| |
| dispatchHidingImeInsets(); |
| |
| assertThat(mMenuView.getTranslationX()).isEqualTo(0); |
| assertThat(mMenuView.getTranslationY()).isEqualTo(menuTop); |
| } |
| |
| private void dispatchShowingImeInsets() { |
| final WindowInsets fakeShowingImeInsets = fakeImeInsets(/* isImeVisible= */ true); |
| doReturn(fakeShowingImeInsets).when(mWindowMetrics).getWindowInsets(); |
| mMenuViewLayer.dispatchApplyWindowInsets(fakeShowingImeInsets); |
| } |
| |
| private void dispatchHidingImeInsets() { |
| final WindowInsets fakeHidingImeInsets = fakeImeInsets(/* isImeVisible= */ false); |
| doReturn(fakeHidingImeInsets).when(mWindowMetrics).getWindowInsets(); |
| mMenuViewLayer.dispatchApplyWindowInsets(fakeHidingImeInsets); |
| } |
| |
| private WindowInsets fakeDisplayInsets() { |
| return new WindowInsets.Builder() |
| .setVisible(systemBars() | displayCutout(), /* visible= */ true) |
| .setInsets(systemBars() | displayCutout(), |
| Insets.of(/* left= */ 0, STATUS_BAR_HEIGHT, /* right= */ 0, |
| NAVIGATION_BAR_HEIGHT)) |
| .build(); |
| } |
| |
| private WindowInsets fakeImeInsets(boolean isImeVisible) { |
| final int bottom = isImeVisible ? (IME_HEIGHT + NAVIGATION_BAR_HEIGHT) : 0; |
| return new WindowInsets.Builder() |
| .setVisible(ime(), isImeVisible) |
| .setInsets(ime(), |
| Insets.of(/* left= */ 0, /* top= */ 0, /* right= */ 0, bottom)) |
| .build(); |
| } |
| } |