1 /*
2  * Copyright (C) 2013 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.app;
18 
19 import android.accessibilityservice.AccessibilityService.Callbacks;
20 import android.accessibilityservice.AccessibilityService.IAccessibilityServiceClientWrapper;
21 import android.accessibilityservice.AccessibilityServiceInfo;
22 import android.accessibilityservice.IAccessibilityServiceClient;
23 import android.accessibilityservice.IAccessibilityServiceConnection;
24 import android.graphics.Bitmap;
25 import android.graphics.Canvas;
26 import android.graphics.Point;
27 import android.hardware.display.DisplayManagerGlobal;
28 import android.os.IBinder;
29 import android.os.Looper;
30 import android.os.ParcelFileDescriptor;
31 import android.os.RemoteException;
32 import android.os.SystemClock;
33 import android.util.Log;
34 import android.view.Display;
35 import android.view.InputEvent;
36 import android.view.KeyEvent;
37 import android.view.Surface;
38 import android.view.WindowAnimationFrameStats;
39 import android.view.WindowContentFrameStats;
40 import android.view.accessibility.AccessibilityEvent;
41 import android.view.accessibility.AccessibilityInteractionClient;
42 import android.view.accessibility.AccessibilityNodeInfo;
43 import android.view.accessibility.AccessibilityWindowInfo;
44 import android.view.accessibility.IAccessibilityInteractionConnection;
45 import libcore.io.IoUtils;
46 
47 import java.io.IOException;
48 import java.util.ArrayList;
49 import java.util.List;
50 import java.util.concurrent.TimeoutException;
51 
52 /**
53  * Class for interacting with the device's UI by simulation user actions and
54  * introspection of the screen content. It relies on the platform accessibility
55  * APIs to introspect the screen and to perform some actions on the remote view
56  * tree. It also allows injecting of arbitrary raw input events simulating user
57  * interaction with keyboards and touch devices. One can think of a UiAutomation
58  * as a special type of {@link android.accessibilityservice.AccessibilityService}
59  * which does not provide hooks for the service life cycle and exposes other
60  * APIs that are useful for UI test automation.
61  * <p>
62  * The APIs exposed by this class are low-level to maximize flexibility when
63  * developing UI test automation tools and libraries. Generally, a UiAutomation
64  * client should be using a higher-level library or implement high-level functions.
65  * For example, performing a tap on the screen requires construction and injecting
66  * of a touch down and up events which have to be delivered to the system by a
67  * call to {@link #injectInputEvent(InputEvent, boolean)}.
68  * </p>
69  * <p>
70  * The APIs exposed by this class operate across applications enabling a client
71  * to write tests that cover use cases spanning over multiple applications. For
72  * example, going to the settings application to change a setting and then
73  * interacting with another application whose behavior depends on that setting.
74  * </p>
75  */
76 public final class UiAutomation {
77 
78     private static final String LOG_TAG = UiAutomation.class.getSimpleName();
79 
80     private static final boolean DEBUG = false;
81 
82     private static final int CONNECTION_ID_UNDEFINED = -1;
83 
84     private static final long CONNECT_TIMEOUT_MILLIS = 5000;
85 
86     /** Rotation constant: Unfreeze rotation (rotating the device changes its rotation state). */
87     public static final int ROTATION_UNFREEZE = -2;
88 
89     /** Rotation constant: Freeze rotation to its current state. */
90     public static final int ROTATION_FREEZE_CURRENT = -1;
91 
92     /** Rotation constant: Freeze rotation to 0 degrees (natural orientation) */
93     public static final int ROTATION_FREEZE_0 = Surface.ROTATION_0;
94 
95     /** Rotation constant: Freeze rotation to 90 degrees . */
96     public static final int ROTATION_FREEZE_90 = Surface.ROTATION_90;
97 
98     /** Rotation constant: Freeze rotation to 180 degrees . */
99     public static final int ROTATION_FREEZE_180 = Surface.ROTATION_180;
100 
101     /** Rotation constant: Freeze rotation to 270 degrees . */
102     public static final int ROTATION_FREEZE_270 = Surface.ROTATION_270;
103 
104     private final Object mLock = new Object();
105 
106     private final ArrayList<AccessibilityEvent> mEventQueue = new ArrayList<AccessibilityEvent>();
107 
108     private final IAccessibilityServiceClient mClient;
109 
110     private final IUiAutomationConnection mUiAutomationConnection;
111 
112     private int mConnectionId = CONNECTION_ID_UNDEFINED;
113 
114     private OnAccessibilityEventListener mOnAccessibilityEventListener;
115 
116     private boolean mWaitingForEventDelivery;
117 
118     private long mLastEventTimeMillis;
119 
120     private boolean mIsConnecting;
121 
122     /**
123      * Listener for observing the {@link AccessibilityEvent} stream.
124      */
125     public static interface OnAccessibilityEventListener {
126 
127         /**
128          * Callback for receiving an {@link AccessibilityEvent}.
129          * <p>
130          * <strong>Note:</strong> This method is <strong>NOT</strong> executed
131          * on the main test thread. The client is responsible for proper
132          * synchronization.
133          * </p>
134          * <p>
135          * <strong>Note:</strong> It is responsibility of the client
136          * to recycle the received events to minimize object creation.
137          * </p>
138          *
139          * @param event The received event.
140          */
onAccessibilityEvent(AccessibilityEvent event)141         public void onAccessibilityEvent(AccessibilityEvent event);
142     }
143 
144     /**
145      * Listener for filtering accessibility events.
146      */
147     public static interface AccessibilityEventFilter {
148 
149         /**
150          * Callback for determining whether an event is accepted or
151          * it is filtered out.
152          *
153          * @param event The event to process.
154          * @return True if the event is accepted, false to filter it out.
155          */
accept(AccessibilityEvent event)156         public boolean accept(AccessibilityEvent event);
157     }
158 
159     /**
160      * Creates a new instance that will handle callbacks from the accessibility
161      * layer on the thread of the provided looper and perform requests for privileged
162      * operations on the provided connection.
163      *
164      * @param looper The looper on which to execute accessibility callbacks.
165      * @param connection The connection for performing privileged operations.
166      *
167      * @hide
168      */
UiAutomation(Looper looper, IUiAutomationConnection connection)169     public UiAutomation(Looper looper, IUiAutomationConnection connection) {
170         if (looper == null) {
171             throw new IllegalArgumentException("Looper cannot be null!");
172         }
173         if (connection == null) {
174             throw new IllegalArgumentException("Connection cannot be null!");
175         }
176         mUiAutomationConnection = connection;
177         mClient = new IAccessibilityServiceClientImpl(looper);
178     }
179 
180     /**
181      * Connects this UiAutomation to the accessibility introspection APIs.
182      *
183      * @hide
184      */
connect()185     public void connect() {
186         synchronized (mLock) {
187             throwIfConnectedLocked();
188             if (mIsConnecting) {
189                 return;
190             }
191             mIsConnecting = true;
192         }
193 
194         try {
195             // Calling out without a lock held.
196             mUiAutomationConnection.connect(mClient);
197         } catch (RemoteException re) {
198             throw new RuntimeException("Error while connecting UiAutomation", re);
199         }
200 
201         synchronized (mLock) {
202             final long startTimeMillis = SystemClock.uptimeMillis();
203             try {
204                 while (true) {
205                     if (isConnectedLocked()) {
206                         break;
207                     }
208                     final long elapsedTimeMillis = SystemClock.uptimeMillis() - startTimeMillis;
209                     final long remainingTimeMillis = CONNECT_TIMEOUT_MILLIS - elapsedTimeMillis;
210                     if (remainingTimeMillis <= 0) {
211                         throw new RuntimeException("Error while connecting UiAutomation");
212                     }
213                     try {
214                         mLock.wait(remainingTimeMillis);
215                     } catch (InterruptedException ie) {
216                         /* ignore */
217                     }
218                 }
219             } finally {
220                 mIsConnecting = false;
221             }
222         }
223     }
224 
225     /**
226      * Disconnects this UiAutomation from the accessibility introspection APIs.
227      *
228      * @hide
229      */
disconnect()230     public void disconnect() {
231         synchronized (mLock) {
232             if (mIsConnecting) {
233                 throw new IllegalStateException(
234                         "Cannot call disconnect() while connecting!");
235             }
236             throwIfNotConnectedLocked();
237             mConnectionId = CONNECTION_ID_UNDEFINED;
238         }
239         try {
240             // Calling out without a lock held.
241             mUiAutomationConnection.disconnect();
242         } catch (RemoteException re) {
243             throw new RuntimeException("Error while disconnecting UiAutomation", re);
244         }
245     }
246 
247     /**
248      * The id of the {@link IAccessibilityInteractionConnection} for querying
249      * the screen content. This is here for legacy purposes since some tools use
250      * hidden APIs to introspect the screen.
251      *
252      * @hide
253      */
getConnectionId()254     public int getConnectionId() {
255         synchronized (mLock) {
256             throwIfNotConnectedLocked();
257             return mConnectionId;
258         }
259     }
260 
261     /**
262      * Sets a callback for observing the stream of {@link AccessibilityEvent}s.
263      *
264      * @param listener The callback.
265      */
setOnAccessibilityEventListener(OnAccessibilityEventListener listener)266     public void setOnAccessibilityEventListener(OnAccessibilityEventListener listener) {
267         synchronized (mLock) {
268             mOnAccessibilityEventListener = listener;
269         }
270     }
271 
272     /**
273      * Performs a global action. Such an action can be performed at any moment
274      * regardless of the current application or user location in that application.
275      * For example going back, going home, opening recents, etc.
276      *
277      * @param action The action to perform.
278      * @return Whether the action was successfully performed.
279      *
280      * @see android.accessibilityservice.AccessibilityService#GLOBAL_ACTION_BACK
281      * @see android.accessibilityservice.AccessibilityService#GLOBAL_ACTION_HOME
282      * @see android.accessibilityservice.AccessibilityService#GLOBAL_ACTION_NOTIFICATIONS
283      * @see android.accessibilityservice.AccessibilityService#GLOBAL_ACTION_RECENTS
284      */
performGlobalAction(int action)285     public final boolean performGlobalAction(int action) {
286         final IAccessibilityServiceConnection connection;
287         synchronized (mLock) {
288             throwIfNotConnectedLocked();
289             connection = AccessibilityInteractionClient.getInstance()
290                     .getConnection(mConnectionId);
291         }
292         // Calling out without a lock held.
293         if (connection != null) {
294             try {
295                 return connection.performGlobalAction(action);
296             } catch (RemoteException re) {
297                 Log.w(LOG_TAG, "Error while calling performGlobalAction", re);
298             }
299         }
300         return false;
301     }
302 
303     /**
304      * Find the view that has the specified focus type. The search is performed
305      * across all windows.
306      * <p>
307      * <strong>Note:</strong> In order to access the windows you have to opt-in
308      * to retrieve the interactive windows by setting the
309      * {@link AccessibilityServiceInfo#FLAG_RETRIEVE_INTERACTIVE_WINDOWS} flag.
310      * Otherwise, the search will be performed only in the active window.
311      * </p>
312      *
313      * @param focus The focus to find. One of {@link AccessibilityNodeInfo#FOCUS_INPUT} or
314      *         {@link AccessibilityNodeInfo#FOCUS_ACCESSIBILITY}.
315      * @return The node info of the focused view or null.
316      *
317      * @see AccessibilityNodeInfo#FOCUS_INPUT
318      * @see AccessibilityNodeInfo#FOCUS_ACCESSIBILITY
319      */
findFocus(int focus)320     public AccessibilityNodeInfo findFocus(int focus) {
321         return AccessibilityInteractionClient.getInstance().findFocus(mConnectionId,
322                 AccessibilityNodeInfo.ANY_WINDOW_ID, AccessibilityNodeInfo.ROOT_NODE_ID, focus);
323     }
324 
325     /**
326      * Gets the an {@link AccessibilityServiceInfo} describing this UiAutomation.
327      * This method is useful if one wants to change some of the dynamically
328      * configurable properties at runtime.
329      *
330      * @return The accessibility service info.
331      *
332      * @see AccessibilityServiceInfo
333      */
getServiceInfo()334     public final AccessibilityServiceInfo getServiceInfo() {
335         final IAccessibilityServiceConnection connection;
336         synchronized (mLock) {
337             throwIfNotConnectedLocked();
338             connection = AccessibilityInteractionClient.getInstance()
339                     .getConnection(mConnectionId);
340         }
341         // Calling out without a lock held.
342         if (connection != null) {
343             try {
344                 return connection.getServiceInfo();
345             } catch (RemoteException re) {
346                 Log.w(LOG_TAG, "Error while getting AccessibilityServiceInfo", re);
347             }
348         }
349         return null;
350     }
351 
352     /**
353      * Sets the {@link AccessibilityServiceInfo} that describes how this
354      * UiAutomation will be handled by the platform accessibility layer.
355      *
356      * @param info The info.
357      *
358      * @see AccessibilityServiceInfo
359      */
setServiceInfo(AccessibilityServiceInfo info)360     public final void setServiceInfo(AccessibilityServiceInfo info) {
361         final IAccessibilityServiceConnection connection;
362         synchronized (mLock) {
363             throwIfNotConnectedLocked();
364             AccessibilityInteractionClient.getInstance().clearCache();
365             connection = AccessibilityInteractionClient.getInstance()
366                     .getConnection(mConnectionId);
367         }
368         // Calling out without a lock held.
369         if (connection != null) {
370             try {
371                 connection.setServiceInfo(info);
372             } catch (RemoteException re) {
373                 Log.w(LOG_TAG, "Error while setting AccessibilityServiceInfo", re);
374             }
375         }
376     }
377 
378     /**
379      * Gets the windows on the screen. This method returns only the windows
380      * that a sighted user can interact with, as opposed to all windows.
381      * For example, if there is a modal dialog shown and the user cannot touch
382      * anything behind it, then only the modal window will be reported
383      * (assuming it is the top one). For convenience the returned windows
384      * are ordered in a descending layer order, which is the windows that
385      * are higher in the Z-order are reported first.
386      * <p>
387      * <strong>Note:</strong> In order to access the windows you have to opt-in
388      * to retrieve the interactive windows by setting the
389      * {@link AccessibilityServiceInfo#FLAG_RETRIEVE_INTERACTIVE_WINDOWS} flag.
390      * </p>
391      *
392      * @return The windows if there are windows such, otherwise an empty list.
393      */
getWindows()394     public List<AccessibilityWindowInfo> getWindows() {
395         final int connectionId;
396         synchronized (mLock) {
397             throwIfNotConnectedLocked();
398             connectionId = mConnectionId;
399         }
400         // Calling out without a lock held.
401         return AccessibilityInteractionClient.getInstance()
402                 .getWindows(connectionId);
403     }
404 
405     /**
406      * Gets the root {@link AccessibilityNodeInfo} in the active window.
407      *
408      * @return The root info.
409      */
getRootInActiveWindow()410     public AccessibilityNodeInfo getRootInActiveWindow() {
411         final int connectionId;
412         synchronized (mLock) {
413             throwIfNotConnectedLocked();
414             connectionId = mConnectionId;
415         }
416         // Calling out without a lock held.
417         return AccessibilityInteractionClient.getInstance()
418                 .getRootInActiveWindow(connectionId);
419     }
420 
421     /**
422      * A method for injecting an arbitrary input event.
423      * <p>
424      * <strong>Note:</strong> It is caller's responsibility to recycle the event.
425      * </p>
426      * @param event The event to inject.
427      * @param sync Whether to inject the event synchronously.
428      * @return Whether event injection succeeded.
429      */
injectInputEvent(InputEvent event, boolean sync)430     public boolean injectInputEvent(InputEvent event, boolean sync) {
431         synchronized (mLock) {
432             throwIfNotConnectedLocked();
433         }
434         try {
435             if (DEBUG) {
436                 Log.i(LOG_TAG, "Injecting: " + event + " sync: " + sync);
437             }
438             // Calling out without a lock held.
439             return mUiAutomationConnection.injectInputEvent(event, sync);
440         } catch (RemoteException re) {
441             Log.e(LOG_TAG, "Error while injecting input event!", re);
442         }
443         return false;
444     }
445 
446     /**
447      * Sets the device rotation. A client can freeze the rotation in
448      * desired state or freeze the rotation to its current state or
449      * unfreeze the rotation (rotating the device changes its rotation
450      * state).
451      *
452      * @param rotation The desired rotation.
453      * @return Whether the rotation was set successfully.
454      *
455      * @see #ROTATION_FREEZE_0
456      * @see #ROTATION_FREEZE_90
457      * @see #ROTATION_FREEZE_180
458      * @see #ROTATION_FREEZE_270
459      * @see #ROTATION_FREEZE_CURRENT
460      * @see #ROTATION_UNFREEZE
461      */
setRotation(int rotation)462     public boolean setRotation(int rotation) {
463         synchronized (mLock) {
464             throwIfNotConnectedLocked();
465         }
466         switch (rotation) {
467             case ROTATION_FREEZE_0:
468             case ROTATION_FREEZE_90:
469             case ROTATION_FREEZE_180:
470             case ROTATION_FREEZE_270:
471             case ROTATION_UNFREEZE:
472             case ROTATION_FREEZE_CURRENT: {
473                 try {
474                     // Calling out without a lock held.
475                     mUiAutomationConnection.setRotation(rotation);
476                     return true;
477                 } catch (RemoteException re) {
478                     Log.e(LOG_TAG, "Error while setting rotation!", re);
479                 }
480             } return false;
481             default: {
482                 throw new IllegalArgumentException("Invalid rotation.");
483             }
484         }
485     }
486 
487     /**
488      * Executes a command and waits for a specific accessibility event up to a
489      * given wait timeout. To detect a sequence of events one can implement a
490      * filter that keeps track of seen events of the expected sequence and
491      * returns true after the last event of that sequence is received.
492      * <p>
493      * <strong>Note:</strong> It is caller's responsibility to recycle the returned event.
494      * </p>
495      * @param command The command to execute.
496      * @param filter Filter that recognizes the expected event.
497      * @param timeoutMillis The wait timeout in milliseconds.
498      *
499      * @throws TimeoutException If the expected event is not received within the timeout.
500      */
executeAndWaitForEvent(Runnable command, AccessibilityEventFilter filter, long timeoutMillis)501     public AccessibilityEvent executeAndWaitForEvent(Runnable command,
502             AccessibilityEventFilter filter, long timeoutMillis) throws TimeoutException {
503         // Acquire the lock and prepare for receiving events.
504         synchronized (mLock) {
505             throwIfNotConnectedLocked();
506             mEventQueue.clear();
507             // Prepare to wait for an event.
508             mWaitingForEventDelivery = true;
509         }
510 
511         // Note: We have to release the lock since calling out with this lock held
512         // can bite. We will correctly filter out events from other interactions,
513         // so starting to collect events before running the action is just fine.
514 
515         // We will ignore events from previous interactions.
516         final long executionStartTimeMillis = SystemClock.uptimeMillis();
517         // Execute the command *without* the lock being held.
518         command.run();
519 
520         // Acquire the lock and wait for the event.
521         synchronized (mLock) {
522             try {
523                 // Wait for the event.
524                 final long startTimeMillis = SystemClock.uptimeMillis();
525                 while (true) {
526                     // Drain the event queue
527                     while (!mEventQueue.isEmpty()) {
528                         AccessibilityEvent event = mEventQueue.remove(0);
529                         // Ignore events from previous interactions.
530                         if (event.getEventTime() < executionStartTimeMillis) {
531                             continue;
532                         }
533                         if (filter.accept(event)) {
534                             return event;
535                         }
536                         event.recycle();
537                     }
538                     // Check if timed out and if not wait.
539                     final long elapsedTimeMillis = SystemClock.uptimeMillis() - startTimeMillis;
540                     final long remainingTimeMillis = timeoutMillis - elapsedTimeMillis;
541                     if (remainingTimeMillis <= 0) {
542                         throw new TimeoutException("Expected event not received within: "
543                                 + timeoutMillis + " ms.");
544                     }
545                     try {
546                         mLock.wait(remainingTimeMillis);
547                     } catch (InterruptedException ie) {
548                         /* ignore */
549                     }
550                 }
551             } finally {
552                 mWaitingForEventDelivery = false;
553                 mEventQueue.clear();
554                 mLock.notifyAll();
555             }
556         }
557     }
558 
559     /**
560      * Waits for the accessibility event stream to become idle, which is not to
561      * have received an accessibility event within <code>idleTimeoutMillis</code>.
562      * The total time spent to wait for an idle accessibility event stream is bounded
563      * by the <code>globalTimeoutMillis</code>.
564      *
565      * @param idleTimeoutMillis The timeout in milliseconds between two events
566      *            to consider the device idle.
567      * @param globalTimeoutMillis The maximal global timeout in milliseconds in
568      *            which to wait for an idle state.
569      *
570      * @throws TimeoutException If no idle state was detected within
571      *            <code>globalTimeoutMillis.</code>
572      */
waitForIdle(long idleTimeoutMillis, long globalTimeoutMillis)573     public void waitForIdle(long idleTimeoutMillis, long globalTimeoutMillis)
574             throws TimeoutException {
575         synchronized (mLock) {
576             throwIfNotConnectedLocked();
577 
578             final long startTimeMillis = SystemClock.uptimeMillis();
579             if (mLastEventTimeMillis <= 0) {
580                 mLastEventTimeMillis = startTimeMillis;
581             }
582 
583             while (true) {
584                 final long currentTimeMillis = SystemClock.uptimeMillis();
585                 // Did we get idle state within the global timeout?
586                 final long elapsedGlobalTimeMillis = currentTimeMillis - startTimeMillis;
587                 final long remainingGlobalTimeMillis =
588                         globalTimeoutMillis - elapsedGlobalTimeMillis;
589                 if (remainingGlobalTimeMillis <= 0) {
590                     throw new TimeoutException("No idle state with idle timeout: "
591                             + idleTimeoutMillis + " within global timeout: "
592                             + globalTimeoutMillis);
593                 }
594                 // Did we get an idle state within the idle timeout?
595                 final long elapsedIdleTimeMillis = currentTimeMillis - mLastEventTimeMillis;
596                 final long remainingIdleTimeMillis = idleTimeoutMillis - elapsedIdleTimeMillis;
597                 if (remainingIdleTimeMillis <= 0) {
598                     return;
599                 }
600                 try {
601                      mLock.wait(remainingIdleTimeMillis);
602                 } catch (InterruptedException ie) {
603                      /* ignore */
604                 }
605             }
606         }
607     }
608 
609     /**
610      * Takes a screenshot.
611      *
612      * @return The screenshot bitmap on success, null otherwise.
613      */
takeScreenshot()614     public Bitmap takeScreenshot() {
615         synchronized (mLock) {
616             throwIfNotConnectedLocked();
617         }
618         Display display = DisplayManagerGlobal.getInstance()
619                 .getRealDisplay(Display.DEFAULT_DISPLAY);
620         Point displaySize = new Point();
621         display.getRealSize(displaySize);
622         final int displayWidth = displaySize.x;
623         final int displayHeight = displaySize.y;
624 
625         final float screenshotWidth;
626         final float screenshotHeight;
627 
628         final int rotation = display.getRotation();
629         switch (rotation) {
630             case ROTATION_FREEZE_0: {
631                 screenshotWidth = displayWidth;
632                 screenshotHeight = displayHeight;
633             } break;
634             case ROTATION_FREEZE_90: {
635                 screenshotWidth = displayHeight;
636                 screenshotHeight = displayWidth;
637             } break;
638             case ROTATION_FREEZE_180: {
639                 screenshotWidth = displayWidth;
640                 screenshotHeight = displayHeight;
641             } break;
642             case ROTATION_FREEZE_270: {
643                 screenshotWidth = displayHeight;
644                 screenshotHeight = displayWidth;
645             } break;
646             default: {
647                 throw new IllegalArgumentException("Invalid rotation: "
648                         + rotation);
649             }
650         }
651 
652         // Take the screenshot
653         Bitmap screenShot = null;
654         try {
655             // Calling out without a lock held.
656             screenShot = mUiAutomationConnection.takeScreenshot((int) screenshotWidth,
657                     (int) screenshotHeight);
658             if (screenShot == null) {
659                 return null;
660             }
661         } catch (RemoteException re) {
662             Log.e(LOG_TAG, "Error while taking screnshot!", re);
663             return null;
664         }
665 
666         // Rotate the screenshot to the current orientation
667         if (rotation != ROTATION_FREEZE_0) {
668             Bitmap unrotatedScreenShot = Bitmap.createBitmap(displayWidth, displayHeight,
669                     Bitmap.Config.ARGB_8888);
670             Canvas canvas = new Canvas(unrotatedScreenShot);
671             canvas.translate(unrotatedScreenShot.getWidth() / 2,
672                     unrotatedScreenShot.getHeight() / 2);
673             canvas.rotate(getDegreesForRotation(rotation));
674             canvas.translate(- screenshotWidth / 2, - screenshotHeight / 2);
675             canvas.drawBitmap(screenShot, 0, 0, null);
676             canvas.setBitmap(null);
677             screenShot.recycle();
678             screenShot = unrotatedScreenShot;
679         }
680 
681         // Optimization
682         screenShot.setHasAlpha(false);
683 
684         return screenShot;
685     }
686 
687     /**
688      * Sets whether this UiAutomation to run in a "monkey" mode. Applications can query whether
689      * they are executed in a "monkey" mode, i.e. run by a test framework, and avoid doing
690      * potentially undesirable actions such as calling 911 or posting on public forums etc.
691      *
692      * @param enable whether to run in a "monkey" mode or not. Default is not.
693      * @see {@link android.app.ActivityManager#isUserAMonkey()}
694      */
setRunAsMonkey(boolean enable)695     public void setRunAsMonkey(boolean enable) {
696         synchronized (mLock) {
697             throwIfNotConnectedLocked();
698         }
699         try {
700             ActivityManagerNative.getDefault().setUserIsMonkey(enable);
701         } catch (RemoteException re) {
702             Log.e(LOG_TAG, "Error while setting run as monkey!", re);
703         }
704     }
705 
706     /**
707      * Clears the frame statistics for the content of a given window. These
708      * statistics contain information about the most recently rendered content
709      * frames.
710      *
711      * @param windowId The window id.
712      * @return Whether the window is present and its frame statistics
713      *         were cleared.
714      *
715      * @see android.view.WindowContentFrameStats
716      * @see #getWindowContentFrameStats(int)
717      * @see #getWindows()
718      * @see AccessibilityWindowInfo#getId() AccessibilityWindowInfo.getId()
719      */
clearWindowContentFrameStats(int windowId)720     public boolean clearWindowContentFrameStats(int windowId) {
721         synchronized (mLock) {
722             throwIfNotConnectedLocked();
723         }
724         try {
725             if (DEBUG) {
726                 Log.i(LOG_TAG, "Clearing content frame stats for window: " + windowId);
727             }
728             // Calling out without a lock held.
729             return mUiAutomationConnection.clearWindowContentFrameStats(windowId);
730         } catch (RemoteException re) {
731             Log.e(LOG_TAG, "Error clearing window content frame stats!", re);
732         }
733         return false;
734     }
735 
736     /**
737      * Gets the frame statistics for a given window. These statistics contain
738      * information about the most recently rendered content frames.
739      * <p>
740      * A typical usage requires clearing the window frame statistics via {@link
741      * #clearWindowContentFrameStats(int)} followed by an interaction with the UI and
742      * finally getting the window frame statistics via calling this method.
743      * </p>
744      * <pre>
745      * // Assume we have at least one window.
746      * final int windowId = getWindows().get(0).getId();
747      *
748      * // Start with a clean slate.
749      * uiAutimation.clearWindowContentFrameStats(windowId);
750      *
751      * // Do stuff with the UI.
752      *
753      * // Get the frame statistics.
754      * WindowContentFrameStats stats = uiAutomation.getWindowContentFrameStats(windowId);
755      * </pre>
756      *
757      * @param windowId The window id.
758      * @return The window frame statistics, or null if the window is not present.
759      *
760      * @see android.view.WindowContentFrameStats
761      * @see #clearWindowContentFrameStats(int)
762      * @see #getWindows()
763      * @see AccessibilityWindowInfo#getId() AccessibilityWindowInfo.getId()
764      */
getWindowContentFrameStats(int windowId)765     public WindowContentFrameStats getWindowContentFrameStats(int windowId) {
766         synchronized (mLock) {
767             throwIfNotConnectedLocked();
768         }
769         try {
770             if (DEBUG) {
771                 Log.i(LOG_TAG, "Getting content frame stats for window: " + windowId);
772             }
773             // Calling out without a lock held.
774             return mUiAutomationConnection.getWindowContentFrameStats(windowId);
775         } catch (RemoteException re) {
776             Log.e(LOG_TAG, "Error getting window content frame stats!", re);
777         }
778         return null;
779     }
780 
781     /**
782      * Clears the window animation rendering statistics. These statistics contain
783      * information about the most recently rendered window animation frames, i.e.
784      * for window transition animations.
785      *
786      * @see android.view.WindowAnimationFrameStats
787      * @see #getWindowAnimationFrameStats()
788      * @see android.R.styleable#WindowAnimation
789      */
clearWindowAnimationFrameStats()790     public void clearWindowAnimationFrameStats() {
791         synchronized (mLock) {
792             throwIfNotConnectedLocked();
793         }
794         try {
795             if (DEBUG) {
796                 Log.i(LOG_TAG, "Clearing window animation frame stats");
797             }
798             // Calling out without a lock held.
799             mUiAutomationConnection.clearWindowAnimationFrameStats();
800         } catch (RemoteException re) {
801             Log.e(LOG_TAG, "Error clearing window animation frame stats!", re);
802         }
803     }
804 
805     /**
806      * Gets the window animation frame statistics. These statistics contain
807      * information about the most recently rendered window animation frames, i.e.
808      * for window transition animations.
809      *
810      * <p>
811      * A typical usage requires clearing the window animation frame statistics via
812      * {@link #clearWindowAnimationFrameStats()} followed by an interaction that causes
813      * a window transition which uses a window animation and finally getting the window
814      * animation frame statistics by calling this method.
815      * </p>
816      * <pre>
817      * // Start with a clean slate.
818      * uiAutimation.clearWindowAnimationFrameStats();
819      *
820      * // Do stuff to trigger a window transition.
821      *
822      * // Get the frame statistics.
823      * WindowAnimationFrameStats stats = uiAutomation.getWindowAnimationFrameStats();
824      * </pre>
825      *
826      * @return The window animation frame statistics.
827      *
828      * @see android.view.WindowAnimationFrameStats
829      * @see #clearWindowAnimationFrameStats()
830      * @see android.R.styleable#WindowAnimation
831      */
getWindowAnimationFrameStats()832     public WindowAnimationFrameStats getWindowAnimationFrameStats() {
833         synchronized (mLock) {
834             throwIfNotConnectedLocked();
835         }
836         try {
837             if (DEBUG) {
838                 Log.i(LOG_TAG, "Getting window animation frame stats");
839             }
840             // Calling out without a lock held.
841             return mUiAutomationConnection.getWindowAnimationFrameStats();
842         } catch (RemoteException re) {
843             Log.e(LOG_TAG, "Error getting window animation frame stats!", re);
844         }
845         return null;
846     }
847 
848     /**
849      * Executes a shell command. This method returs a file descriptor that points
850      * to the standard output stream. The command execution is similar to running
851      * "adb shell <command>" from a host connected to the device.
852      * <p>
853      * <strong>Note:</strong> It is your responsibility to close the retunred file
854      * descriptor once you are done reading.
855      * </p>
856      *
857      * @param command The command to execute.
858      * @return A file descriptor to the standard output stream.
859      */
executeShellCommand(String command)860     public ParcelFileDescriptor executeShellCommand(String command) {
861         synchronized (mLock) {
862             throwIfNotConnectedLocked();
863         }
864 
865         ParcelFileDescriptor source = null;
866         ParcelFileDescriptor sink = null;
867 
868         try {
869             ParcelFileDescriptor[] pipe = ParcelFileDescriptor.createPipe();
870             source = pipe[0];
871             sink = pipe[1];
872 
873             // Calling out without a lock held.
874             mUiAutomationConnection.executeShellCommand(command, sink);
875         } catch (IOException ioe) {
876             Log.e(LOG_TAG, "Error executing shell command!", ioe);
877         } catch (RemoteException re) {
878             Log.e(LOG_TAG, "Error executing shell command!", re);
879         } finally {
880             IoUtils.closeQuietly(sink);
881         }
882 
883         return source;
884     }
885 
getDegreesForRotation(int value)886     private static float getDegreesForRotation(int value) {
887         switch (value) {
888             case Surface.ROTATION_90: {
889                 return 360f - 90f;
890             }
891             case Surface.ROTATION_180: {
892                 return 360f - 180f;
893             }
894             case Surface.ROTATION_270: {
895                 return 360f - 270f;
896             } default: {
897                 return 0;
898             }
899         }
900     }
901 
isConnectedLocked()902     private boolean isConnectedLocked() {
903         return mConnectionId != CONNECTION_ID_UNDEFINED;
904     }
905 
throwIfConnectedLocked()906     private void throwIfConnectedLocked() {
907         if (mConnectionId != CONNECTION_ID_UNDEFINED) {
908             throw new IllegalStateException("UiAutomation not connected!");
909         }
910     }
911 
throwIfNotConnectedLocked()912     private void throwIfNotConnectedLocked() {
913         if (!isConnectedLocked()) {
914             throw new IllegalStateException("UiAutomation not connected!");
915         }
916     }
917 
918     private class IAccessibilityServiceClientImpl extends IAccessibilityServiceClientWrapper {
919 
IAccessibilityServiceClientImpl(Looper looper)920         public IAccessibilityServiceClientImpl(Looper looper) {
921             super(null, looper, new Callbacks() {
922                 @Override
923                 public void init(int connectionId, IBinder windowToken) {
924                     synchronized (mLock) {
925                         mConnectionId = connectionId;
926                         mLock.notifyAll();
927                     }
928                 }
929 
930                 @Override
931                 public void onServiceConnected() {
932                     /* do nothing */
933                 }
934 
935                 @Override
936                 public void onInterrupt() {
937                     /* do nothing */
938                 }
939 
940                 @Override
941                 public boolean onGesture(int gestureId) {
942                     /* do nothing */
943                     return false;
944                 }
945 
946                 @Override
947                 public void onAccessibilityEvent(AccessibilityEvent event) {
948                     synchronized (mLock) {
949                         mLastEventTimeMillis = event.getEventTime();
950                         if (mWaitingForEventDelivery) {
951                             mEventQueue.add(AccessibilityEvent.obtain(event));
952                         }
953                         mLock.notifyAll();
954                     }
955                     // Calling out only without a lock held.
956                     final OnAccessibilityEventListener listener = mOnAccessibilityEventListener;
957                     if (listener != null) {
958                         listener.onAccessibilityEvent(AccessibilityEvent.obtain(event));
959                     }
960                 }
961 
962                 @Override
963                 public boolean onKeyEvent(KeyEvent event) {
964                     return false;
965                 }
966             });
967         }
968     }
969 }
970