1 /* 2 * Copyright (C) 2015 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 14 * limitations under the License. 15 */ 16 package com.android.tv.testing.uihelper; 17 18 import static com.android.tv.testing.uihelper.Constants.FOCUSED_VIEW; 19 import static junit.framework.Assert.assertEquals; 20 import static junit.framework.Assert.assertNotNull; 21 import static junit.framework.Assert.assertTrue; 22 23 import androidx.test.uiautomator.By; 24 import androidx.test.uiautomator.BySelector; 25 import androidx.test.uiautomator.Direction; 26 import androidx.test.uiautomator.SearchCondition; 27 import androidx.test.uiautomator.UiDevice; 28 import androidx.test.uiautomator.UiObject2; 29 import androidx.test.uiautomator.Until; 30 import junit.framework.Assert; 31 32 /** Asserts for {@link UiDevice}s. */ 33 public final class UiDeviceAsserts { 34 assertHas(UiDevice uiDevice, BySelector bySelector, boolean expected)35 public static void assertHas(UiDevice uiDevice, BySelector bySelector, boolean expected) { 36 assertEquals("Has " + bySelector, expected, uiDevice.hasObject(bySelector)); 37 } 38 assertWaitUntilFocused(UiDevice uiDevice, BySelector bySelector)39 public static void assertWaitUntilFocused(UiDevice uiDevice, BySelector bySelector) { 40 UiObject2 uiObject = uiDevice.findObject(bySelector); 41 assertNotNull(uiObject); 42 assertTrue(uiObject.wait(Until.focused(true), Constants.MAX_FOCUSED_DELAY_MILLIS)); 43 } 44 45 /** 46 * Assert that {@code searchCondition} becomes true within {@value 47 * Constants#MAX_SHOW_DELAY_MILLIS} milliseconds. 48 * 49 * @param uiDevice the device under test. 50 * @param searchCondition the condition to wait for. 51 */ assertWaitForCondition( UiDevice uiDevice, SearchCondition<Boolean> searchCondition)52 public static void assertWaitForCondition( 53 UiDevice uiDevice, SearchCondition<Boolean> searchCondition) { 54 assertWaitForCondition(uiDevice, searchCondition, Constants.MAX_SHOW_DELAY_MILLIS); 55 } 56 57 /** 58 * Assert that {@code searchCondition} becomes true within {@code timeout} milliseconds. 59 * 60 * @param uiDevice the device under test. 61 * @param searchCondition the condition to wait for. 62 */ assertWaitForCondition( UiDevice uiDevice, SearchCondition<Boolean> searchCondition, long timeout)63 public static void assertWaitForCondition( 64 UiDevice uiDevice, SearchCondition<Boolean> searchCondition, long timeout) { 65 boolean result = waitForCondition(uiDevice, searchCondition, timeout); 66 assertTrue(searchCondition + " not true after " + timeout / 1000.0 + " seconds.", result); 67 } 68 69 /** 70 * Wait until {@code searchCondition} becomes true. 71 * 72 * @param uiDevice The device under test. 73 * @param searchCondition The condition to wait for. 74 * @return {@code true} if the condition is met, otherwise {@code false}. 75 */ waitForCondition( UiDevice uiDevice, SearchCondition<Boolean> searchCondition)76 public static boolean waitForCondition( 77 UiDevice uiDevice, SearchCondition<Boolean> searchCondition) { 78 return waitForCondition(uiDevice, searchCondition, Constants.MAX_SHOW_DELAY_MILLIS); 79 } 80 waitForCondition( UiDevice uiDevice, SearchCondition<Boolean> searchCondition, long timeout)81 private static boolean waitForCondition( 82 UiDevice uiDevice, SearchCondition<Boolean> searchCondition, long timeout) { 83 long adjustedTimeout = 84 timeout 85 + Math.max( 86 Constants.MIN_EXTRA_TIMEOUT, 87 (long) (timeout * Constants.EXTRA_TIMEOUT_PERCENT)); 88 return uiDevice.wait(searchCondition, adjustedTimeout); 89 } 90 91 /** 92 * Navigates through the focus items in a container returning the container child that has a 93 * descendant matching the {@code selector}. 94 * 95 * <p>The navigation starts in the {@code direction} specified and {@link 96 * Direction#reverse(Direction) reverses} once if needed. Fails if there is not a focused 97 * descendant, or if after completing both directions no focused child has a descendant matching 98 * {@code selector}. 99 * 100 * <p>Fails if the menu item can not be navigated to. 101 * 102 * @param uiDevice the device under test. 103 * @param container contains children to navigate over. 104 * @param selector the selector for the object to navigate to. 105 * @param direction the direction to start navigating. 106 * @return the object navigated to. 107 */ assertNavigateTo( UiDevice uiDevice, UiObject2 container, BySelector selector, Direction direction)108 public static UiObject2 assertNavigateTo( 109 UiDevice uiDevice, UiObject2 container, BySelector selector, Direction direction) { 110 int count = 0; 111 while (count < 2) { 112 BySelector hasFocusedDescendant = By.hasDescendant(FOCUSED_VIEW); 113 UiObject2 focusedChild = null; 114 SearchCondition<Boolean> untilHasFocusedDescendant = 115 Until.hasObject(hasFocusedDescendant); 116 117 boolean result = 118 container.wait( 119 untilHasFocusedDescendant, 120 UiObject2Asserts.getAdjustedTimeout(Constants.MAX_SHOW_DELAY_MILLIS)); 121 if (!result) { 122 // HACK: Try direction anyways because play control does not always have a 123 // focused item. 124 UiDeviceUtils.pressDpad(uiDevice, direction); 125 UiObject2Asserts.assertWaitForCondition(container, untilHasFocusedDescendant); 126 } 127 128 for (UiObject2 c : container.getChildren()) { 129 if (c.isFocused() || c.hasObject(hasFocusedDescendant)) { 130 focusedChild = c; 131 break; 132 } 133 } 134 if (focusedChild == null) { 135 Assert.fail("No focused item found in container " + container); 136 } 137 if (focusedChild.hasObject(selector)) { 138 return focusedChild; 139 } 140 if (!UiObject2Utils.hasSiblingInDirection(focusedChild, direction)) { 141 direction = Direction.reverse(direction); 142 count++; 143 } 144 UiDeviceUtils.pressDpad(uiDevice, direction); 145 } 146 Assert.fail("Could not find item with " + selector); 147 return null; 148 } 149 UiDeviceAsserts()150 private UiDeviceAsserts() {} 151 } 152