1 /* 2 * Copyright (C) 2017 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 17 package android.system.helpers; 18 19 import android.app.Instrumentation; 20 import android.content.ComponentName; 21 import android.content.Intent; 22 import android.graphics.Point; 23 import android.graphics.Rect; 24 import android.support.test.uiautomator.By; 25 import android.support.test.uiautomator.UiDevice; 26 import android.support.test.uiautomator.UiObject2; 27 import android.support.test.uiautomator.UiObjectNotFoundException; 28 import android.support.test.uiautomator.UiScrollable; 29 import android.support.test.uiautomator.UiSelector; 30 import android.support.test.uiautomator.Until; 31 import android.util.Log; 32 33 import java.util.List; 34 35 /** 36 * Implement common helper functions for Accessibility scanner. 37 */ 38 public class AccessibilityScannerHelper { 39 public static final String ACCESSIBILITY_SCANNER_PACKAGE 40 = "com.google.android.apps.accessibility.auditor"; 41 public static final String MAIN_ACTIVITY_CLASS = "%s.ui.MainActivity"; 42 public static final String CHECK_BUTTON_RES_ID = "accessibilibutton"; 43 private static final int SCANNER_WAIT_TIME = 5000; 44 private static final int SHORT_TIMEOUT = 2000; 45 private static final String LOG_TAG = AccessibilityScannerHelper.class.getSimpleName(); 46 private static final String RESULT_TAG = "A11Y_SCANNER_RESULT"; 47 public static AccessibilityScannerHelper sInstance = null; 48 private UiDevice mDevice = null; 49 private ActivityHelper mActivityHelper = null; 50 private PackageHelper mPackageHelper = null; 51 private AccessibilityHelper mAccessibilityHelper = null; 52 53 private AccessibilityScannerHelper(Instrumentation instr) { 54 mDevice = UiDevice.getInstance(instr); 55 mActivityHelper = ActivityHelper.getInstance(); 56 mPackageHelper = PackageHelper.getInstance(instr); 57 mAccessibilityHelper = AccessibilityHelper.getInstance(instr); 58 } 59 60 public static AccessibilityScannerHelper getInstance(Instrumentation instr) { 61 if (sInstance == null) { 62 sInstance = new AccessibilityScannerHelper(instr); 63 } 64 return sInstance; 65 } 66 67 /** 68 * If accessibility scanner installed. 69 * 70 * @return true/false 71 */ 72 public boolean scannerInstalled() { 73 return mPackageHelper.isPackageInstalled(ACCESSIBILITY_SCANNER_PACKAGE); 74 } 75 76 /** 77 * Click scanner check button and parse and log results. 78 * 79 * @param resultPrefix 80 * @throws Exception 81 */ 82 public void runScanner(String resultPrefix) throws Exception { 83 int tries = 3; // retries 84 while (tries-- > 0) { 85 try { 86 clickScannerCheck(); 87 logScannerResult(resultPrefix); 88 break; 89 } catch (UiObjectNotFoundException e) { 90 continue; 91 } catch (Exception e) { 92 throw e; 93 } 94 } 95 } 96 97 /** 98 * Click scanner check button and open share app in the share menu. 99 * 100 * @param resultPrefix 101 * @param shareAppTag 102 * @throws Exception 103 */ 104 public void runScannerAndOpenShareApp(String resultPrefix, String shareAppTag) 105 throws Exception { 106 runScanner(resultPrefix); 107 UiObject2 shareApp = getShareApp(shareAppTag); 108 if (shareApp != null) { 109 shareApp.click(); 110 } 111 } 112 113 /** 114 * Set Accessibility Scanner setting ON/OFF. 115 * 116 * @throws Exception 117 */ 118 public void setAccessibilityScannerSetting(AccessibilityHelper.SwitchStatus value) 119 throws Exception { 120 if (!scannerInstalled()) { 121 throw new Exception("Accessibility Scanner not installed."); 122 } 123 mAccessibilityHelper.launchSpecificAccessibilitySetting("Accessibility Scanner"); 124 for (int tries = 0; tries < 2; tries++) { 125 UiObject2 swt = mDevice.wait(Until.findObject( 126 By.res(AccessibilityHelper.SETTINGS_PACKAGE, "switch_widget")), 127 SHORT_TIMEOUT * 2); 128 if (swt.getText().equals(value.toString())) { 129 break; 130 } else if (tries == 1) { 131 throw new Exception(String.format("Fail to set scanner to: %s.", value.toString())); 132 } else { 133 swt.click(); 134 UiObject2 okBtn = mDevice.wait(Until.findObject(By.text("OK")), SHORT_TIMEOUT); 135 if (okBtn != null) { 136 okBtn.click(); 137 } 138 if (initialSetups()) { 139 mDevice.pressBack(); 140 } 141 grantPermissions(); 142 } 143 } 144 } 145 146 /** 147 * Click through all permission pop ups for scanner. Grant all necessary permissions. 148 */ 149 private void grantPermissions() { 150 UiObject2 auth1 = mDevice.wait(Until.findObject( 151 By.text("BEGIN AUTHORIZATION")), SHORT_TIMEOUT); 152 if (auth1 != null) { 153 auth1.click(); 154 } 155 UiObject2 chk = mDevice.wait(Until.findObject( 156 By.clazz(AccessibilityHelper.CHECK_BOX)), SHORT_TIMEOUT); 157 if (chk != null) { 158 chk.click(); 159 mDevice.findObject(By.text("START NOW")).click(); 160 } 161 UiObject2 auth2 = mDevice.wait(Until.findObject( 162 By.text("BEGIN AUTHORIZATION")), SHORT_TIMEOUT); 163 if (auth2 != null) { 164 auth2.click(); 165 } 166 UiObject2 tapOk = mDevice.wait(Until.findObject( 167 By.pkg(ACCESSIBILITY_SCANNER_PACKAGE).text("OK")), SHORT_TIMEOUT); 168 if (tapOk != null) { 169 tapOk.click(); 170 } 171 } 172 173 /** 174 * Launch accessibility scanner. 175 * 176 * @throws UiObjectNotFoundException 177 */ 178 public void launchScannerApp() throws Exception { 179 Intent intent = new Intent(Intent.ACTION_MAIN); 180 ComponentName settingComponent = new ComponentName(ACCESSIBILITY_SCANNER_PACKAGE, 181 String.format(MAIN_ACTIVITY_CLASS, ACCESSIBILITY_SCANNER_PACKAGE)); 182 intent.setComponent(settingComponent); 183 mActivityHelper.launchIntent(intent); 184 initialSetups(); 185 } 186 187 /** 188 * Steps for first time launching scanner app. 189 * 190 * @return true/false return false immediately, if initial setup screen doesn't show up. 191 * @throws Exception 192 */ 193 private boolean initialSetups() throws Exception { 194 UiObject2 getStartBtn = mDevice.wait( 195 Until.findObject(By.text("GET STARTED")), SHORT_TIMEOUT); 196 if (getStartBtn != null) { 197 getStartBtn.click(); 198 UiObject2 msg = mDevice.wait(Until.findObject( 199 By.text("Turn on Accessibility Scanner")), SHORT_TIMEOUT); 200 if (msg != null) { 201 mDevice.findObject(By.text("OK")).click(); 202 setAccessibilityScannerSetting(AccessibilityHelper.SwitchStatus.ON); 203 } 204 mDevice.wait(Until.findObject(By.text("OK, GOT IT")), SCANNER_WAIT_TIME).click(); 205 mDevice.wait(Until.findObject(By.text("DISMISS")), SHORT_TIMEOUT).click(); 206 return true; 207 } else { 208 return false; 209 } 210 } 211 212 /** 213 * Clear history of accessibility scanner. 214 * 215 * @throws InterruptedException 216 */ 217 public void clearHistory() throws Exception { 218 launchScannerApp(); 219 int maxTry = 20; 220 while (maxTry > 0) { 221 List<UiObject2> historyItemList = mDevice.findObjects( 222 By.res(ACCESSIBILITY_SCANNER_PACKAGE, "history_item_row")); 223 if (historyItemList.size() == 0) { 224 break; 225 } 226 historyItemList.get(0).click(); 227 Thread.sleep(SHORT_TIMEOUT); 228 deleteHistory(); 229 Thread.sleep(SHORT_TIMEOUT); 230 maxTry--; 231 } 232 } 233 234 /** 235 * Log results of accessibility scanner. 236 * 237 * @param pageName 238 * @throws Exception 239 */ 240 public void logScannerResult(String pageName) throws Exception { 241 int res = getNumberOfSuggestions(); 242 if (res > 0) { 243 Log.i(RESULT_TAG, String.format("%s: %s suggestions!", pageName, res)); 244 } else if (res == 0) { 245 Log.i(RESULT_TAG, String.format("%s: Pass.", pageName)); 246 } else { 247 throw new UiObjectNotFoundException("Fail to get number of suggestions."); 248 } 249 } 250 251 /** 252 * Move scanner button to avoid blocking the object. 253 * 254 * @param avoidObj object to move the check button away from 255 */ 256 public void adjustScannerButton(UiObject2 avoidObj) 257 throws UiObjectNotFoundException, InterruptedException { 258 Rect origBounds = getScannerCheckBtn().getVisibleBounds(); 259 Rect avoidBounds = avoidObj.getVisibleBounds(); 260 if (origBounds.intersect(avoidBounds)) { 261 Point dest = calculateDest(origBounds, avoidBounds); 262 moveScannerCheckButton(dest.x, dest.y); 263 } 264 } 265 266 /** 267 * Move scanner check button back to the middle of the screen. 268 */ 269 public void resetScannerCheckButton() throws UiObjectNotFoundException, InterruptedException { 270 int midY = (int) Math.ceil(mDevice.getDisplayHeight() * 0.5); 271 int midX = (int) Math.ceil(mDevice.getDisplayWidth() * 0.5); 272 moveScannerCheckButton(midX, midY); 273 } 274 275 /** 276 * Move scanner check button to a target location. 277 * 278 * @param locX target location x-axis 279 * @param locY target location y-axis 280 * @throws UiObjectNotFoundException 281 */ 282 public void moveScannerCheckButton(int locX, int locY) 283 throws UiObjectNotFoundException, InterruptedException { 284 int tries = 2; 285 while (tries-- > 0) { 286 UiObject2 btn = getScannerCheckBtn(); 287 Rect bounds = btn.getVisibleBounds(); 288 int origX = bounds.centerX(); 289 int origY = bounds.centerY(); 290 int buttonWidth = bounds.width(); 291 int buttonHeight = bounds.height(); 292 if (Math.abs(locX - origX) > buttonWidth || Math.abs(locY - origY) > buttonHeight) { 293 btn.drag(new Point(locX, locY)); 294 } 295 Thread.sleep(SCANNER_WAIT_TIME); 296 // drag cause a click on the scanner button, bring the UI into scanner app 297 if (getScannerCheckBtn() == null 298 && mDevice.findObject(By.pkg(ACCESSIBILITY_SCANNER_PACKAGE)) != null) { 299 mDevice.pressBack(); 300 } else { 301 break; 302 } 303 } 304 } 305 306 /** 307 * Calculate the moving destination of check button. 308 * 309 * @param origRect original bounds of the check button 310 * @param avoidRect bounds to move away from 311 * @return destination of check button center point. 312 */ 313 private Point calculateDest(Rect origRect, Rect avoidRect) { 314 int bufferY = (int)Math.ceil(mDevice.getDisplayHeight() * 0.1); 315 int destY = avoidRect.bottom + bufferY + origRect.height()/2; 316 if (destY >= mDevice.getDisplayHeight()) { 317 destY = avoidRect.top - bufferY - origRect.height()/2; 318 } 319 return new Point(origRect.centerX(), destY); 320 } 321 322 /** 323 * Return scanner check button. 324 * 325 * @return UiObject2 326 */ 327 private UiObject2 getScannerCheckBtn() { 328 return mDevice.findObject(By.res(ACCESSIBILITY_SCANNER_PACKAGE, CHECK_BUTTON_RES_ID)); 329 } 330 331 private void clickScannerCheck() throws UiObjectNotFoundException, InterruptedException { 332 UiObject2 accessibilityScannerButton = getScannerCheckBtn(); 333 if (accessibilityScannerButton != null) { 334 accessibilityScannerButton.click(); 335 } else { 336 // TODO: check if app crash error, restart scanner service 337 Log.i(LOG_TAG, "Fail to find accessibility scanner check button."); 338 throw new UiObjectNotFoundException( 339 "Fail to find accessibility scanner check button."); 340 } 341 Thread.sleep(SCANNER_WAIT_TIME); 342 } 343 344 /** 345 * Check if no suggestion. 346 * @deprecated Use {@link #getNumberOfSuggestions} instead 347 */ 348 @Deprecated 349 private Boolean testPass() throws UiObjectNotFoundException { 350 UiObject2 txtView = getToolBarTextView(); 351 return txtView.getText().equals("No suggestions"); 352 } 353 354 /** 355 * Return accessibility scanner tool bar text view. 356 * 357 * @return UiObject2 358 * @throws UiObjectNotFoundException 359 */ 360 private UiObject2 getToolBarTextView() throws UiObjectNotFoundException { 361 UiObject2 toolBar = mDevice.wait(Until.findObject( 362 By.res(ACCESSIBILITY_SCANNER_PACKAGE, "toolbar")), SHORT_TIMEOUT); 363 if (toolBar != null) { 364 return toolBar.findObject(By.clazz(AccessibilityHelper.TEXT_VIEW)); 365 } else { 366 throw new UiObjectNotFoundException( 367 "Failed to find Scanner tool bar. Scanner app might not be active."); 368 } 369 } 370 371 /** 372 * Delete active scanner history. 373 */ 374 private void deleteHistory() { 375 UiObject2 moreBtn = mDevice.wait(Until.findObject(By.desc("More options")), SHORT_TIMEOUT); 376 if (moreBtn != null) { 377 moreBtn.click(); 378 mDevice.wait(Until.findObject( 379 By.clazz(AccessibilityHelper.TEXT_VIEW).text("Delete")), SHORT_TIMEOUT).click(); 380 } 381 } 382 383 /** 384 * Return number suggestions. 385 * 386 * @return number of suggestions 387 * @throws UiObjectNotFoundException 388 */ 389 private int getNumberOfSuggestions() throws UiObjectNotFoundException { 390 int tries = 2; // retries 391 while (tries-- > 0) { 392 UiObject2 txtView = getToolBarTextView(); 393 if (txtView != null) { 394 String result = txtView.getText(); 395 if (result.equals("No suggestions")) { 396 return 0; 397 } else { 398 String str = result.split("\\s+")[0]; 399 return Integer.parseInt(str); 400 } 401 } 402 } 403 Log.i(LOG_TAG, String.format("Error in getting number of suggestions.")); 404 return -1; 405 } 406 407 /** 408 * Return share app UiObject2 409 * 410 * @param appName 411 * @return 412 */ 413 private UiObject2 getShareApp(String appName) throws UiObjectNotFoundException { 414 UiObject2 shareBtn = mDevice.wait(Until.findObject(By.res(ACCESSIBILITY_SCANNER_PACKAGE, 415 "action_share_results")), SHORT_TIMEOUT); 416 if (shareBtn != null) { 417 shareBtn.click(); 418 mDevice.wait(Until.hasObject(By.res("android:id/resolver_list")), SHORT_TIMEOUT * 3); 419 UiScrollable scrollable = new UiScrollable( 420 new UiSelector().className("android.widget.ScrollView")); 421 int tries = 3; 422 while (!mDevice.hasObject(By.text(appName)) && tries-- > 0) { 423 scrollable.scrollForward(); 424 } 425 return mDevice.findObject(By.text(appName)); 426 } 427 return null; 428 } 429 430 /** 431 * Return if scanner enabled by check if home screen has check button. 432 * 433 * @return true/false 434 */ 435 public boolean ifScannerEnabled() throws InterruptedException { 436 mDevice.pressHome(); 437 Thread.sleep(SHORT_TIMEOUT); 438 mDevice.waitForIdle(); 439 return getScannerCheckBtn() != null; 440 } 441 } 442