1 /*
2  * Copyright (C) 2020 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 com.android.cts.helpers.aosp;
18 
19 import android.app.Instrumentation;
20 import android.app.UiAutomation;
21 import android.os.RemoteException;
22 import android.os.SystemClock;
23 import android.platform.helpers.exceptions.TestHelperException;
24 import android.util.Log;
25 
26 import androidx.test.uiautomator.By;
27 import androidx.test.uiautomator.UiDevice;
28 import androidx.test.uiautomator.UiObject;
29 import androidx.test.uiautomator.UiObject2;
30 import androidx.test.uiautomator.UiObjectNotFoundException;
31 import androidx.test.uiautomator.UiSelector;
32 import androidx.test.uiautomator.Until;
33 
34 import com.android.cts.helpers.ICtsPrintHelper;
35 
36 import java.io.ByteArrayOutputStream;
37 import java.io.IOException;
38 
39 public class DefaultCtsPrintHelper implements ICtsPrintHelper {
40     private static final String LOG_TAG = DefaultCtsPrintHelper.class.getSimpleName();
41 
42     protected static final long OPERATION_TIMEOUT_MILLIS = 60000;
43     private static final long GET_UIAUTOMATION_TIMEOUT_MS = 60000;
44 
45     protected Instrumentation mInstrumentation;
46     protected UiDevice mDevice;
47     protected UiAutomation mAutomation;
48 
DefaultCtsPrintHelper(Instrumentation instrumentation)49     public DefaultCtsPrintHelper(Instrumentation instrumentation) {
50         mInstrumentation = instrumentation;
51         mDevice = UiDevice.getInstance(mInstrumentation);
52 
53         long start = SystemClock.uptimeMillis();
54         while (SystemClock.uptimeMillis() - start < GET_UIAUTOMATION_TIMEOUT_MS) {
55             UiAutomation ui = mInstrumentation.getUiAutomation();
56             if (ui != null) {
57                 mAutomation = ui;
58                 break;
59             }
60         }
61         if (mAutomation == null) {
62             throw new AssertionError("Failed to get UiAutomation");
63         }
64     }
65 
dumpWindowHierarchy()66     protected void dumpWindowHierarchy() throws TestHelperException {
67         try {
68             ByteArrayOutputStream os = new ByteArrayOutputStream();
69             mDevice.dumpWindowHierarchy(os);
70 
71             Log.w(LOG_TAG, "Window hierarchy:");
72             for (String line : os.toString("UTF-8").split("\n")) {
73                 Log.w(LOG_TAG, line);
74             }
75         } catch (IOException e) {
76             throw new TestHelperException(e);
77         }
78     }
79 
80     @Override
setUp()81     public void setUp() throws TestHelperException {
82         try {
83             // Prevent rotation
84             mDevice.freezeRotation();
85             while (!mDevice.isNaturalOrientation()) {
86                 mDevice.setOrientationNatural();
87                 mDevice.waitForIdle();
88             }
89         } catch (RemoteException e) {
90             throw new TestHelperException("Failed to freeze device rotation", e);
91         }
92     }
93 
94     @Override
tearDown()95     public void tearDown() throws TestHelperException {
96         try {
97             // Allow rotation
98             mDevice.unfreezeRotation();
99         } catch (RemoteException e) {
100             throw new TestHelperException("Failed to unfreeze device rotation", e);
101         }
102     }
103 
104     @Override
submitPrintJob()105     public void submitPrintJob() throws TestHelperException {
106         Log.d(LOG_TAG, "Clicking print button");
107 
108         mDevice.waitForIdle();
109 
110         UiObject2 printButton =
111                 mDevice.wait(
112                         Until.findObject(By.res("com.android.printspooler:id/print_button")),
113                         OPERATION_TIMEOUT_MILLIS);
114         if (printButton == null) {
115             dumpWindowHierarchy();
116             throw new TestHelperException("print button not found");
117         }
118 
119         printButton.click();
120     }
121 
122     @Override
retryPrintJob()123     public void retryPrintJob() throws TestHelperException {
124         try {
125             UiObject retryButton = mDevice.findObject(new UiSelector().resourceId(
126                     "com.android.printspooler:id/action_button"));
127             retryButton.click();
128         } catch (UiObjectNotFoundException e) {
129             dumpWindowHierarchy();
130             throw new TestHelperException("Retry button not found", e);
131         }
132     }
133 
134     @Override
canSubmitJob()135     public boolean canSubmitJob() {
136         return mDevice.hasObject(By.res("com.android.printspooler:id/print_button"));
137     }
138 
139     @Override
answerPrintServicesWarning(boolean confirm)140     public void answerPrintServicesWarning(boolean confirm) throws TestHelperException {
141         try {
142             mDevice.waitForIdle();
143             UiObject button;
144             if (confirm) {
145                 button = mDevice.findObject(new UiSelector().resourceId("android:id/button1"));
146             } else {
147                 button = mDevice.findObject(new UiSelector().resourceId("android:id/button2"));
148             }
149             button.click();
150         } catch (UiObjectNotFoundException e) {
151             dumpWindowHierarchy();
152             throw new TestHelperException("Unable to find print service dialog button", e);
153         }
154     }
155 
156     @Override
selectPrinter(String printerName, long timeout)157     public void selectPrinter(String printerName, long timeout) throws TestHelperException {
158         Log.d(LOG_TAG, "Selecting printer " + printerName);
159         try {
160             UiObject2 destinationSpinner =
161                     mDevice.wait(
162                             Until.findObject(
163                                     By.res("com.android.printspooler:id/destination_spinner")),
164                             timeout);
165 
166             if (destinationSpinner != null) {
167                 destinationSpinner.click();
168                 mDevice.waitForIdle();
169             }
170 
171             UiObject2 printerOption = mDevice.wait(Until.findObject(By.text(printerName)), timeout);
172             if (printerOption == null) {
173                 throw new UiObjectNotFoundException(printerName + " not found");
174             }
175 
176             printerOption.click();
177             mDevice.waitForIdle();
178         } catch (Exception e) {
179             dumpWindowHierarchy();
180             throw new TestHelperException("Failed to select printer", e);
181         }
182     }
183 
184     @Override
selectPrinterWhenAvailable(String printerName)185     public void selectPrinterWhenAvailable(String printerName) throws TestHelperException {
186         try {
187             while (true) {
188                 UiObject printerItem = mDevice.findObject(
189                         new UiSelector().text(printerName));
190 
191                 if (printerItem.isEnabled()) {
192                     printerItem.click();
193                     break;
194                 } else {
195                     Thread.sleep(100);
196                 }
197             }
198         } catch (UiObjectNotFoundException e) {
199             throw new TestHelperException("Failed to find printer label", e);
200         } catch (InterruptedException e) {
201             throw new TestHelperException("Interruped while waiting for printer", e);
202         }
203     }
204 
205     @Override
setPageOrientation(String orientation)206     public void setPageOrientation(String orientation) throws TestHelperException {
207         try {
208             UiObject orientationSpinner = mDevice.findObject(new UiSelector().resourceId(
209                     "com.android.printspooler:id/orientation_spinner"));
210             orientationSpinner.click();
211             UiObject orientationOption = mDevice.findObject(new UiSelector().text(orientation));
212             orientationOption.click();
213         } catch (UiObjectNotFoundException e) {
214             dumpWindowHierarchy();
215             throw new TestHelperException("Failed to set page orientation to " + orientation, e);
216         }
217     }
218 
219     @Override
getPageOrientation()220     public String getPageOrientation() throws TestHelperException {
221         try {
222             UiObject orientationSpinner = mDevice.findObject(new UiSelector().resourceId(
223                     "com.android.printspooler:id/orientation_spinner"));
224             return orientationSpinner.getText();
225         } catch (UiObjectNotFoundException e) {
226             dumpWindowHierarchy();
227             throw new TestHelperException("Failed to get page orientation", e);
228         }
229     }
230 
231     @Override
setMediaSize(String mediaSize)232     public void setMediaSize(String mediaSize) throws TestHelperException {
233         try {
234             UiObject mediaSizeSpinner = mDevice.findObject(new UiSelector().resourceId(
235                     "com.android.printspooler:id/paper_size_spinner"));
236             mediaSizeSpinner.click();
237             UiObject mediaSizeOption = mDevice.findObject(new UiSelector().text(mediaSize));
238             mediaSizeOption.click();
239         } catch (UiObjectNotFoundException e) {
240             dumpWindowHierarchy();
241             throw new TestHelperException("Unable to set media size to " + mediaSize, e);
242         }
243     }
244 
245     @Override
setColorMode(String color)246     public void setColorMode(String color) throws TestHelperException {
247         try {
248             UiObject colorSpinner = mDevice.findObject(new UiSelector().resourceId(
249                     "com.android.printspooler:id/color_spinner"));
250             colorSpinner.click();
251             UiObject colorOption = mDevice.findObject(new UiSelector().text(color));
252             colorOption.click();
253         } catch (UiObjectNotFoundException e) {
254             dumpWindowHierarchy();
255             throw new TestHelperException("Unable to set color mode to " + color, e);
256         }
257     }
258 
259     @Override
getColorMode()260     public String getColorMode() throws TestHelperException {
261         try {
262             UiObject colorSpinner = mDevice.findObject(new UiSelector().resourceId(
263                     "com.android.printspooler:id/color_spinner"));
264             return colorSpinner.getText();
265         } catch (UiObjectNotFoundException e) {
266             dumpWindowHierarchy();
267             throw new TestHelperException("Unable to get color mode", e);
268         }
269     }
270 
271     @Override
setDuplexMode(String duplex)272     public void setDuplexMode(String duplex) throws TestHelperException {
273         try {
274             UiObject duplexSpinner = mDevice.findObject(new UiSelector().resourceId(
275                     "com.android.printspooler:id/duplex_spinner"));
276             duplexSpinner.click();
277             UiObject duplexOption = mDevice.findObject(new UiSelector().text(duplex));
278             duplexOption.click();
279         } catch (UiObjectNotFoundException e) {
280             dumpWindowHierarchy();
281             throw new TestHelperException("Unable to set duplex mode to " + duplex, e);
282         }
283     }
284 
285     @Override
setCopies(int newCopies)286     public void setCopies(int newCopies) throws TestHelperException {
287         try {
288             UiObject copies = mDevice.findObject(new UiSelector().resourceId(
289                     "com.android.printspooler:id/copies_edittext"));
290             copies.setText(Integer.valueOf(newCopies).toString());
291         } catch (UiObjectNotFoundException e) {
292             dumpWindowHierarchy();
293             throw new TestHelperException("Unable to set copies to " + newCopies, e);
294         }
295     }
296 
297     @Override
getCopies()298     public int getCopies() throws TestHelperException {
299         try {
300             UiObject copies = mDevice.findObject(new UiSelector().resourceId(
301                     "com.android.printspooler:id/copies_edittext"));
302             return Integer.parseInt(copies.getText());
303         } catch (UiObjectNotFoundException e) {
304             dumpWindowHierarchy();
305             throw new TestHelperException("Unable to get number of copies", e);
306         }
307     }
308 
309     @Override
setPageRange(String pages, int expectedPages)310     public void setPageRange(String pages, int expectedPages) throws TestHelperException {
311         try {
312             mDevice.waitForIdle();
313             UiObject pagesSpinner = mDevice.findObject(new UiSelector().resourceId(
314                     "com.android.printspooler:id/range_options_spinner"));
315             pagesSpinner.click();
316 
317             mDevice.waitForIdle();
318             UiObject rangeOption = mDevice.findObject(new UiSelector().textContains("Range of "
319                     + expectedPages));
320             rangeOption.click();
321 
322             mDevice.waitForIdle();
323             UiObject pagesEditText = mDevice.findObject(new UiSelector().resourceId(
324                     "com.android.printspooler:id/page_range_edittext"));
325             pagesEditText.setText(pages);
326 
327             mDevice.waitForIdle();
328             // Hide the keyboard.
329             mDevice.pressBack();
330         } catch (UiObjectNotFoundException e) {
331             dumpWindowHierarchy();
332             throw new TestHelperException("Unable to set page range", e);
333         }
334     }
335 
336     @Override
getPageRange(int docPages)337     public String getPageRange(int docPages) throws TestHelperException {
338         final String fullRange = "All " + docPages;
339 
340         try {
341             if (mDevice.hasObject(By.text(fullRange))) {
342                 return fullRange;
343             }
344 
345             UiObject pagesEditText = mDevice.findObject(new UiSelector().resourceId(
346                     "com.android.printspooler:id/page_range_edittext"));
347 
348             return pagesEditText.getText();
349         } catch (UiObjectNotFoundException e) {
350             dumpWindowHierarchy();
351             throw new TestHelperException("Unable to get page range", e);
352         }
353     }
354 
355     @Override
getStatusMessage()356     public String getStatusMessage() throws TestHelperException {
357         UiObject2 message = mDevice.wait(Until.findObject(
358                 By.res("com.android.printspooler:id/message")), OPERATION_TIMEOUT_MILLIS);
359 
360         if (message == null) {
361             dumpWindowHierarchy();
362             throw new TestHelperException("Cannot find status message");
363         }
364 
365         return message.getText();
366     }
367 
368     @Override
openPrintOptions()369     public void openPrintOptions() throws TestHelperException {
370         try {
371             UiObject expandHandle = mDevice.findObject(new UiSelector().resourceId(
372                     "com.android.printspooler:id/expand_collapse_handle"));
373             expandHandle.click();
374         } catch (UiObjectNotFoundException e) {
375             dumpWindowHierarchy();
376             throw new TestHelperException("Unable to find print options handle", e);
377         }
378     }
379 
380     @Override
openCustomPrintOptions()381     public void openCustomPrintOptions() throws TestHelperException {
382         try {
383             UiObject expandHandle = mDevice.findObject(new UiSelector().resourceId(
384                     "com.android.printspooler:id/more_options_button"));
385             expandHandle.click();
386         } catch (UiObjectNotFoundException e) {
387             dumpWindowHierarchy();
388             throw new TestHelperException("Unable to find print options handle", e);
389         }
390     }
391 
392     @Override
displayPrinterList()393     public void displayPrinterList() throws TestHelperException {
394         try {
395             // Open destination spinner
396             UiObject destinationSpinner = mDevice.findObject(new UiSelector().resourceId(
397                     "com.android.printspooler:id/destination_spinner"));
398             destinationSpinner.click();
399 
400             // Wait until spinner is opened
401             mDevice.waitForIdle();
402         } catch (UiObjectNotFoundException e) {
403             throw new TestHelperException("Unable to find destination spinner", e);
404         }
405     }
406 
407     @Override
displayMoreInfo()408     public void displayMoreInfo() throws TestHelperException {
409         mDevice.waitForIdle();
410         mDevice.wait(
411                 Until.findObject(By.res("com.android.printspooler:id/more_info")),
412                 OPERATION_TIMEOUT_MILLIS).click();
413     }
414 
415     @Override
closePrinterList()416     public void closePrinterList() {
417         mDevice.pressBack();
418     }
419 
420     @Override
closeCustomPrintOptions()421     public void closeCustomPrintOptions() {
422         mDevice.pressBack();
423     }
424 
425     @Override
closePrintOptions()426     public void closePrintOptions() {
427         mDevice.pressBack();
428     }
429 
430     @Override
cancelPrinting()431     public void cancelPrinting() throws TestHelperException {
432         try {
433             mDevice.wakeUp();
434             mDevice.pressBack();
435             mDevice.waitForIdle();
436         } catch (RemoteException e) {
437             throw new TestHelperException("Failed to cancel printing", e);
438         }
439     }
440 }
441