java.lang.Object
↳androidx.test.uiautomator.UiObject
Subclasses:
UiScrollable, UiCollection
Gradle dependencies
compile group: 'androidx.test.uiautomator', name: 'uiautomator', version: '2.4.0-alpha01'
- groupId: androidx.test.uiautomator
- artifactId: uiautomator
- version: 2.4.0-alpha01
Artifact androidx.test.uiautomator:uiautomator:2.4.0-alpha01 it located at Google repository (https://maven.google.com/)
Androidx artifact mapping:
androidx.test.uiautomator:uiautomator com.android.support.test.uiautomator:uiautomator
Androidx class mapping:
androidx.test.uiautomator.UiObject android.support.test.uiautomator.UiObject
Overview
A UiObject is a representation of a view. It is not in any way directly bound to a
 view as an object reference. A UiObject contains information to help it
 locate a matching view at runtime based on the UiSelector properties specified in
 its constructor. Once you create an instance of a UiObject, it can
 be reused for different views that match the selector criteria.
Summary
| Constructors | 
|---|
| public | UiObject(UiSelector selector) 
 Constructs a UiObject to represent a view that matches the specified
 selector criteria. | 
| Methods | 
|---|
| public void | clearTextField() 
 Clears the existing text contents in an editable field. | 
| public boolean | click() 
 Performs a click at the center of the visible bounds of the UI element represented
 by this UiObject. | 
| public boolean | clickAndWaitForNewWindow() 
 Waits for window transitions that would typically take longer than the
 usual default timeouts. | 
| public boolean | clickAndWaitForNewWindow(long timeout) 
 Performs a click at the center of the visible bounds of the UI element represented
 by this UiObject and waits for window transitions. | 
| public boolean | clickBottomRight() 
 Clicks the bottom and right corner of the UI element | 
| public boolean | clickTopLeft() 
 Clicks the top and left corner of the UI element | 
| public boolean | dragTo(int destX, int destY, int steps) 
 Drags this object to arbitrary coordinates. | 
| public boolean | dragTo(UiObject destObj, int steps) 
 Drags this object to a destination UiObject. | 
| public boolean | exists() 
 Check if view exists. | 
| protected AccessibilityNodeInfo | findAccessibilityNodeInfo(long timeout) 
 Finds a matching UI element in the accessibility hierarchy, by
 using the selector for this UiObject. | 
| public Rect | getBounds() 
 Returns the view's boundsproperty. | 
| public UiObject | getChild(UiSelector selector) 
 Creates a new UiObject for a child view that is under the present UiObject. | 
| public int | getChildCount() 
 Counts the child views immediately under the present UiObject. | 
| public java.lang.String | getClassName() 
 Retrieves the classNameproperty of the UI element. | 
| public java.lang.String | getContentDescription() 
 Reads the content_descproperty of the UI element | 
| public UiObject | getFromParent(UiSelector selector) 
 Creates a new UiObject for a sibling view or a child of the sibling view,
 relative to the present UiObject. | 
| public java.lang.String | getPackageName() 
 Reads the view's packageproperty | 
| public final UiSelector | getSelector() 
 Debugging helper. | 
| public java.lang.String | getText() 
 Reads the textproperty of the UI element | 
| public Rect | getVisibleBounds() 
 Returns the visible bounds of the view. | 
| public boolean | isCheckable() 
 Checks if the UI element's checkableproperty is currently true. | 
| public boolean | isChecked() 
 Check if the UI element's checkedproperty is currently true | 
| public boolean | isClickable() 
 Checks if the UI element's clickableproperty is currently true. | 
| public boolean | isEnabled() 
 Checks if the UI element's enabledproperty is currently true. | 
| public boolean | isFocusable() 
 Check if the UI element's focusableproperty is currently true. | 
| public boolean | isFocused() 
 Check if the UI element's focusedproperty is currently true | 
| public boolean | isLongClickable() 
 Check if the view's long-clickableproperty is currently true | 
| public boolean | isScrollable() 
 Check if the view's scrollableproperty is currently true | 
| public boolean | isSelected() 
 Checks if the UI element's selectedproperty is currently true. | 
| public boolean | longClick() 
 Long clicks the center of the visible bounds of the UI element | 
| public boolean | longClickBottomRight() 
 Long clicks bottom and right corner of the UI element | 
| public boolean | longClickTopLeft() 
 Long clicks on the top and left corner of the UI element | 
| public boolean | performMultiPointerGesture(PointerCoords touches[][]) 
 Performs a multi-touch gesture. | 
| public boolean | performTwoPointerGesture(Point startPoint1, Point startPoint2, Point endPoint1, Point endPoint2, int steps) 
 Generates a two-pointer gesture with arbitrary starting and ending points. | 
| public boolean | pinchIn(int percent, int steps) 
 Performs a two-pointer gesture, where each pointer moves diagonally
 toward the other, from the edges to the center of this UiObject . | 
| public boolean | pinchOut(int percent, int steps) 
 Performs a two-pointer gesture, where each pointer moves diagonally
 opposite across the other, from the center out towards the edges of the
 this UiObject. | 
| public boolean | setText(java.lang.String text) 
 Sets the text in an editable field, after clearing the field's content. | 
| public boolean | swipeDown(int steps) 
 Performs the swipe down action on the UiObject. | 
| public boolean | swipeLeft(int steps) 
 Performs the swipe left action on the UiObject. | 
| public boolean | swipeRight(int steps) 
 Performs the swipe right action on the UiObject. | 
| public boolean | swipeUp(int steps) 
 Performs the swipe up action on the UiObject. | 
| public boolean | waitForExists(long timeout) 
 Waits a specified length of time for a view to become visible. | 
| public boolean | waitUntilGone(long timeout) 
 Waits a specified length of time for a view to become undetectable. | 
| from java.lang.Object | clone, equals, finalize, getClass, hashCode, notify, notifyAll, toString, wait, wait, wait | 
Fields
protected static final long 
WAIT_FOR_SELECTOR_TIMEOUTDeprecated: use Configurator.setWaitForSelectorTimeout(long)
protected static final long 
WAIT_FOR_SELECTOR_POLLprotected static final long 
WAIT_FOR_WINDOW_TMEOUTprotected static final int 
SWIPE_MARGIN_LIMITprotected static final long 
WAIT_FOR_EVENT_TMEOUTDeprecated: use Configurator.setScrollAcknowledgmentTimeout(long)
protected static final int 
FINGER_TOUCH_HALF_WIDTHConstructors
Deprecated: Use UiDevice.findObject(UiSelector) instead. This version hides
 UiObject's dependency on UiDevice and is prone to misuse.
Constructs a UiObject to represent a view that matches the specified
 selector criteria.
Parameters:
selector: 
Methods
Debugging helper. A test can dump the properties of a selector as a string
 to its logs if needed. getSelector().toString();
Returns:
UiSelector
Creates a new UiObject for a child view that is under the present UiObject.
Parameters:
selector: for child view to match
Returns:
a new UiObject representing the child view
Creates a new UiObject for a sibling view or a child of the sibling view,
 relative to the present UiObject.
Parameters:
selector: for a sibling view or children of the sibling view
Returns:
a new UiObject representing the matched view
public int 
getChildCount()
Counts the child views immediately under the present UiObject.
Returns:
the count of child views.
protected AccessibilityNodeInfo 
findAccessibilityNodeInfo(long timeout)
Finds a matching UI element in the accessibility hierarchy, by
 using the selector for this UiObject.
Parameters:
timeout: in milliseconds
Returns:
AccessibilityNodeInfo if found else null
public boolean 
dragTo(
UiObject destObj, int steps)
Drags this object to a destination UiObject.
 The number of steps specified in your input parameter can influence the
 drag speed, and varying speeds may impact the results. Consider
 evaluating different speeds when using this method in your tests.
Parameters:
destObj: the destination UiObject.
steps: usually 40 steps. You can increase or decrease the steps to change the speed.
Returns:
true if successful
public boolean 
dragTo(int destX, int destY, int steps)
Drags this object to arbitrary coordinates.
 The number of steps specified in your input parameter can influence the
 drag speed, and varying speeds may impact the results. Consider
 evaluating different speeds when using this method in your tests.
Parameters:
destX: the X-axis coordinate.
destY: the Y-axis coordinate.
steps: usually 40 steps. You can increase or decrease the steps to change the speed.
Returns:
true if successful
public boolean 
swipeUp(int steps)
Performs the swipe up action on the UiObject. 
 See also:
 
Parameters:
steps: indicates the number of injected move steps into the system. Steps are
 injected about 5ms apart. So a 100 steps may take about 1/2 second to complete.
Returns:
true of successful
public boolean 
swipeDown(int steps)
Performs the swipe down action on the UiObject. 
 The swipe gesture can be performed over any surface. The targeted
 UI element does not need to be scrollable.
 See also:
 
Parameters:
steps: indicates the number of injected move steps into the system. Steps are
 injected about 5ms apart. So a 100 steps may take about 1/2 second to complete.
Returns:
true if successful
public boolean 
swipeLeft(int steps)
Performs the swipe left action on the UiObject. 
 The swipe gesture can be performed over any surface. The targeted
 UI element does not need to be scrollable.
 See also:
 
Parameters:
steps: indicates the number of injected move steps into the system. Steps are
 injected about 5ms apart. So a 100 steps may take about 1/2 second to complete.
Returns:
true if successful
public boolean 
swipeRight(int steps)
Performs the swipe right action on the UiObject. 
 The swipe gesture can be performed over any surface. The targeted
 UI element does not need to be scrollable.
 See also:
 
Parameters:
steps: indicates the number of injected move steps into the system. Steps are
 injected about 5ms apart. So a 100 steps may take about 1/2 second to complete.
Returns:
true if successful
Performs a click at the center of the visible bounds of the UI element represented
 by this UiObject.
Returns:
true id successful else false
public boolean 
clickAndWaitForNewWindow()
Waits for window transitions that would typically take longer than the
 usual default timeouts.
 See UiObject.clickAndWaitForNewWindow(long)
Returns:
true if the event was triggered, else false
public boolean 
clickAndWaitForNewWindow(long timeout)
Performs a click at the center of the visible bounds of the UI element represented
 by this UiObject and waits for window transitions.
 This method differ from UiObject.click() only in that this method waits for a
 a new window transition as a result of the click. Some examples of a window transition:
 
 - launching a new activity
- bringing up a pop-up menu
- bringing up a dialog
Parameters:
timeout: timeout before giving up on waiting for a new window
Returns:
true if the event was triggered, else false
public boolean 
clickTopLeft()
Clicks the top and left corner of the UI element
Returns:
true on success
public boolean 
longClickBottomRight()
Long clicks bottom and right corner of the UI element
Returns:
true if operation was successful
public boolean 
clickBottomRight()
Clicks the bottom and right corner of the UI element
Returns:
true on success
public boolean 
longClick()
Long clicks the center of the visible bounds of the UI element
Returns:
true if operation was successful
public boolean 
longClickTopLeft()
Long clicks on the top and left corner of the UI element
Returns:
true if operation was successful
public java.lang.String 
getText()
Reads the text property of the UI element
Returns:
text value of the current node represented by this UiObject
public java.lang.String 
getClassName()
Retrieves the className property of the UI element.
Returns:
class name of the current node represented by this UiObject
public java.lang.String 
getContentDescription()
Reads the content_desc property of the UI element
Returns:
value of node attribute "content_desc"
public boolean 
setText(java.lang.String text)
Sets the text in an editable field, after clearing the field's content.
 
 The UiSelector selector of this object must reference a UI element that is editable.
 
 When you call this method, the method sets focus on the editable field, clears its existing
 content, then injects your specified text into the field.
 
 If you want to capture the original contents of the field, call UiObject.getText() first.
 You can then modify the text and use this method to update the field.
 
Improvements: 
 Post API Level 19 (KitKat release), the underlying implementation is updated to a dedicated
 set text accessibility action, and it also now supports Unicode.
Parameters:
text: string to set
Returns:
true if operation is successful
public void 
clearTextField()
Clears the existing text contents in an editable field.
 The UiSelector of this object must reference a UI element that is editable.
 When you call this method, the method sets focus on the editable field, selects all of its
 existing content, and clears it by sending a DELETE key press
public boolean 
isChecked()
Check if the UI element's checked property is currently true
Returns:
true if it is else false
public boolean 
isSelected()
Checks if the UI element's selected property is currently true.
Returns:
true if it is else false
public boolean 
isCheckable()
Checks if the UI element's checkable property is currently true.
Returns:
true if it is else false
public boolean 
isEnabled()
Checks if the UI element's enabled property is currently true.
Returns:
true if it is else false
public boolean 
isClickable()
Checks if the UI element's clickable property is currently true.
Returns:
true if it is else false
public boolean 
isFocused()
Check if the UI element's focused property is currently true
Returns:
true if it is else false
public boolean 
isFocusable()
Check if the UI element's focusable property is currently true.
Returns:
true if it is else false
public boolean 
isScrollable()
Check if the view's scrollable property is currently true
Returns:
true if it is else false
public boolean 
isLongClickable()
Check if the view's long-clickable property is currently true
Returns:
true if it is else false
public java.lang.String 
getPackageName()
Reads the view's package property
Returns:
true if it is else false
public Rect 
getVisibleBounds()
Returns the visible bounds of the view.
 If a portion of the view is visible, only the bounds of the visible portion are
 reported.
Returns:
Rect
See also: UiObject.getBounds()
Returns the view's bounds property. See UiObject.getVisibleBounds()
Returns:
Rect
public boolean 
waitForExists(long timeout)
Waits a specified length of time for a view to become visible.
 This method waits until the view becomes visible on the display, or
 until the timeout has elapsed. You can use this method in situations where
 the content that you want to select is not immediately displayed.
Parameters:
timeout: the amount of time to wait (in milliseconds)
Returns:
true if the view is displayed, else false if timeout elapsed while waiting
public boolean 
waitUntilGone(long timeout)
Waits a specified length of time for a view to become undetectable.
 This method waits until a view is no longer matchable, or until the
 timeout has elapsed.
 A view becomes undetectable when the UiSelector of the object is
 unable to find a match because the element has either changed its state or is no
 longer displayed.
 You can use this method when attempting to wait for some long operation
 to compete, such as downloading a large file or connecting to a remote server.
Parameters:
timeout: time to wait (in milliseconds)
Returns:
true if the element is gone before timeout elapsed, else false if timeout elapsed
 but a matching element is still found.
Check if view exists.
 This methods performs a UiObject.waitForExists(long) with zero timeout. This
 basically returns immediately whether the view represented by this UiObject
 exists or not. If you need to wait longer for this view, then see
 UiObject.waitForExists(long).
Returns:
true if the view represented by this UiObject does exist
public boolean 
pinchOut(int percent, int steps)
Performs a two-pointer gesture, where each pointer moves diagonally
 opposite across the other, from the center out towards the edges of the
 this UiObject.
Parameters:
percent: percentage of the object's diagonal length for the pinch gesture
steps: the number of steps for the gesture. Steps are injected 
 about 5 milliseconds apart, so 100 steps may take around 0.5 seconds to complete.
Returns:
true if all touch events for this gesture are injected successfully,
         false otherwise
public boolean 
pinchIn(int percent, int steps)
Performs a two-pointer gesture, where each pointer moves diagonally
 toward the other, from the edges to the center of this UiObject .
Parameters:
percent: percentage of the object's diagonal length for the pinch gesture
steps: the number of steps for the gesture. Steps are injected 
 about 5 milliseconds apart, so 100 steps may take around 0.5 seconds to complete.
Returns:
true if all touch events for this gesture are injected successfully,
         false otherwise
public boolean 
performTwoPointerGesture(Point startPoint1, Point startPoint2, Point endPoint1, Point endPoint2, int steps)
Generates a two-pointer gesture with arbitrary starting and ending points.
Parameters:
startPoint1: start point of pointer 1
startPoint2: start point of pointer 2
endPoint1: end point of pointer 1
endPoint2: end point of pointer 2
steps: the number of steps for the gesture. Steps are injected 
 about 5 milliseconds apart, so 100 steps may take around 0.5 seconds to complete.
Returns:
true if all touch events for this gesture are injected successfully,
         false otherwise
public boolean 
performMultiPointerGesture(PointerCoords touches[][])
Performs a multi-touch gesture. You must specify touch coordinates for
 at least 2 pointers. Each pointer must have all of its touch steps
 defined in an array of . You can use this method to
 specify complex gestures, like circles and irregular shapes, where each
 pointer may take a different path.
 To create a single point on a pointer's touch path:
 
       PointerCoords p = new PointerCoords();
       p.x = stepX;
       p.y = stepY;
       p.pressure = 1;
       p.size = 1;
 
Parameters:
touches: represents the pointers' paths. Each 
 array represents a different pointer. Each  in an
 array element represents a touch point on a pointer's path.
Returns:
true if all touch events for this gesture are injected successfully,
         false otherwise
Source
/*
 * Copyright (C) 2012 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 androidx.test.uiautomator;
import android.graphics.Point;
import android.graphics.Rect;
import android.os.Bundle;
import android.os.SystemClock;
import android.util.Log;
import android.view.MotionEvent.PointerCoords;
import android.view.accessibility.AccessibilityNodeInfo;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
/**
 * A UiObject is a representation of a view. It is not in any way directly bound to a
 * view as an object reference. A UiObject contains information to help it
 * locate a matching view at runtime based on the {@link UiSelector} properties specified in
 * its constructor. Once you create an instance of a UiObject, it can
 * be reused for different views that match the selector criteria.
 */
public class UiObject {
    private static final String TAG = UiObject.class.getSimpleName();
    /** @deprecated use {@link Configurator#setWaitForSelectorTimeout(long)} */
    @Deprecated
    protected static final long WAIT_FOR_SELECTOR_TIMEOUT = 10 * 1000;
    protected static final long WAIT_FOR_SELECTOR_POLL = 1000;
    // set a default timeout to 5.5s, since ANR threshold is 5s
    protected static final long WAIT_FOR_WINDOW_TMEOUT = 5500;
    protected static final int SWIPE_MARGIN_LIMIT = 5;
    /** @deprecated use {@link Configurator#setScrollAcknowledgmentTimeout(long)} */
    @Deprecated
    protected static final long WAIT_FOR_EVENT_TMEOUT = 3 * 1000;
    protected static final int FINGER_TOUCH_HALF_WIDTH = 20;
    private final UiSelector mUiSelector;
    private final UiDevice mDevice;
    private final Configurator mConfig = Configurator.getInstance();
    /**
     * Constructs a UiObject to represent a view that matches the specified
     * selector criteria.
     *
     * @deprecated Use {@link UiDevice#findObject(UiSelector)} instead. This version hides
     * UiObject's dependency on UiDevice and is prone to misuse.
     * @param selector
     */
    @Deprecated
    public UiObject(UiSelector selector) {
        mUiSelector = selector;
        mDevice = UiDevice.getInstance();
    }
    /**
     * Package-private constructor. Used by {@link UiDevice#findObject(UiSelector)} to construct a
     * UiObject.
     */
    UiObject(UiDevice device, UiSelector selector) {
        mDevice = device;
        mUiSelector = selector;
    }
    /**
     * Debugging helper. A test can dump the properties of a selector as a string
     * to its logs if needed. <code>getSelector().toString();</code>
     *
     * @return {@link UiSelector}
     */
    @NonNull
    public final UiSelector getSelector() {
        if (mUiSelector == null) {
            throw new IllegalStateException("UiSelector not set");
        }
        return mUiSelector;
    }
    UiDevice getDevice() {
        return mDevice;
    }
    /**
     * Retrieves the {@link QueryController} to translate a {@link UiSelector} selector
     * into an {@link AccessibilityNodeInfo}.
     *
     * @return {@link QueryController}
     */
    QueryController getQueryController() {
        return getDevice().getQueryController();
    }
    /**
     * Retrieves the {@link InteractionController} to perform finger actions such as tapping,
     * swiping, or entering text.
     *
     * @return {@link InteractionController}
     */
    InteractionController getInteractionController() {
        return getDevice().getInteractionController();
    }
    /**
     * Creates a new UiObject for a child view that is under the present UiObject.
     *
     * @param selector for child view to match
     * @return a new UiObject representing the child view
     */
    @NonNull
    public UiObject getChild(@NonNull UiSelector selector) throws UiObjectNotFoundException {
        return new UiObject(getSelector().childSelector(selector));
    }
    /**
     * Creates a new UiObject for a sibling view or a child of the sibling view,
     * relative to the present UiObject.
     *
     * @param selector for a sibling view or children of the sibling view
     * @return a new UiObject representing the matched view
     * @throws UiObjectNotFoundException
     */
    @NonNull
    public UiObject getFromParent(@NonNull UiSelector selector) throws UiObjectNotFoundException {
        return new UiObject(getSelector().fromParent(selector));
    }
    /**
     * Counts the child views immediately under the present UiObject.
     *
     * @return the count of child views.
     * @throws UiObjectNotFoundException
     */
    public int getChildCount() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return node.getChildCount();
    }
    /**
     * Finds a matching UI element in the accessibility hierarchy, by
     * using the selector for this UiObject.
     *
     * @param timeout in milliseconds
     * @return AccessibilityNodeInfo if found else null
     */
    @Nullable
    protected AccessibilityNodeInfo findAccessibilityNodeInfo(long timeout) {
        AccessibilityNodeInfo node = null;
        long startMills = SystemClock.uptimeMillis();
        long currentMills = 0;
        while (currentMills <= timeout) {
            node = getQueryController().findAccessibilityNodeInfo(mUiSelector);
            if (node != null) {
                break;
            } else {
                // does nothing if we're reentering another runWatchers()
                getDevice().runWatchers();
            }
            currentMills = SystemClock.uptimeMillis() - startMills;
            if(timeout > 0) {
                SystemClock.sleep(WAIT_FOR_SELECTOR_POLL);
            }
        }
        return node;
    }
    /**
     * Drags this object to a destination UiObject.
     * The number of steps specified in your input parameter can influence the
     * drag speed, and varying speeds may impact the results. Consider
     * evaluating different speeds when using this method in your tests.
     *
     * @param destObj the destination UiObject.
     * @param steps usually 40 steps. You can increase or decrease the steps to change the speed.
     * @return true if successful
     * @throws UiObjectNotFoundException
     */
    public boolean dragTo(@NonNull UiObject destObj, int steps) throws UiObjectNotFoundException {
        Rect srcRect = getVisibleBounds();
        Rect dstRect = destObj.getVisibleBounds();
        Log.d(TAG, String.format("Dragging from (%d, %d) to (%d, %d) in %d steps.",
                srcRect.centerX(), srcRect.centerY(), dstRect.centerX(), dstRect.centerY(), steps));
        return getInteractionController().swipe(srcRect.centerX(), srcRect.centerY(),
                dstRect.centerX(), dstRect.centerY(), steps, true);
    }
    /**
     * Drags this object to arbitrary coordinates.
     * The number of steps specified in your input parameter can influence the
     * drag speed, and varying speeds may impact the results. Consider
     * evaluating different speeds when using this method in your tests.
     *
     * @param destX the X-axis coordinate.
     * @param destY the Y-axis coordinate.
     * @param steps usually 40 steps. You can increase or decrease the steps to change the speed.
     * @return true if successful
     * @throws UiObjectNotFoundException
     */
    public boolean dragTo(int destX, int destY, int steps) throws UiObjectNotFoundException {
        Rect srcRect = getVisibleBounds();
        Log.d(TAG, String.format("Dragging from (%d, %d) to (%d, %d) in %d steps.",
                srcRect.centerX(), srcRect.centerY(), destX, destY, steps));
        return getInteractionController().swipe(srcRect.centerX(), srcRect.centerY(), destX, destY,
                steps, true);
    }
    /**
     * Performs the swipe up action on the UiObject. 
     * See also:
     * <ul>
     * <li>{@link UiScrollable#scrollToBeginning(int)}</li>
     * <li>{@link UiScrollable#scrollToEnd(int)}</li>
     * <li>{@link UiScrollable#scrollBackward()}</li>
     * <li>{@link UiScrollable#scrollForward()}</li>
     * </ul>
     *
     * @param steps indicates the number of injected move steps into the system. Steps are
     * injected about 5ms apart. So a 100 steps may take about 1/2 second to complete.
     * @return true of successful
     * @throws UiObjectNotFoundException
     */
    public boolean swipeUp(int steps) throws UiObjectNotFoundException {
        Rect rect = getVisibleBounds();
        if (rect.height() <= SWIPE_MARGIN_LIMIT * 2) {
            Log.w(TAG, String.format("Cannot swipe. Object height too small (%d < %d).",
                    rect.height(), SWIPE_MARGIN_LIMIT * 2));
            return false;
        }
        Log.d(TAG, String.format("Swiping up from (%d, %d) to (%d, %d) in %d steps.",
                rect.centerX(), rect.bottom - SWIPE_MARGIN_LIMIT, rect.centerX(),
                rect.top + SWIPE_MARGIN_LIMIT, steps));
        return getInteractionController().swipe(rect.centerX(),
                rect.bottom - SWIPE_MARGIN_LIMIT, rect.centerX(), rect.top + SWIPE_MARGIN_LIMIT,
                steps);
    }
    /**
     * Performs the swipe down action on the UiObject. 
     * The swipe gesture can be performed over any surface. The targeted
     * UI element does not need to be scrollable.
     * See also:
     * <ul>
     * <li>{@link UiScrollable#scrollToBeginning(int)}</li>
     * <li>{@link UiScrollable#scrollToEnd(int)}</li>
     * <li>{@link UiScrollable#scrollBackward()}</li>
     * <li>{@link UiScrollable#scrollForward()}</li>
     * </ul>
     *
     * @param steps indicates the number of injected move steps into the system. Steps are
     * injected about 5ms apart. So a 100 steps may take about 1/2 second to complete.
     * @return true if successful
     * @throws UiObjectNotFoundException
     */
    public boolean swipeDown(int steps) throws UiObjectNotFoundException {
        Rect rect = getVisibleBounds();
        if (rect.height() <= SWIPE_MARGIN_LIMIT * 2) {
            Log.w(TAG, String.format("Cannot swipe. Object height too small (%d < %d).",
                    rect.height(), SWIPE_MARGIN_LIMIT * 2));
            return false;
        }
        Log.d(TAG, String.format("Swiping down from (%d, %d) to (%d, %d) in %d steps.",
                rect.centerX(), rect.top + SWIPE_MARGIN_LIMIT, rect.centerX(),
                rect.bottom - SWIPE_MARGIN_LIMIT, steps));
        return getInteractionController().swipe(rect.centerX(),
                rect.top + SWIPE_MARGIN_LIMIT, rect.centerX(),
                rect.bottom - SWIPE_MARGIN_LIMIT, steps);
    }
    /**
     * Performs the swipe left action on the UiObject. 
     * The swipe gesture can be performed over any surface. The targeted
     * UI element does not need to be scrollable.
     * See also:
     * <ul>
     * <li>{@link UiScrollable#scrollToBeginning(int)}</li>
     * <li>{@link UiScrollable#scrollToEnd(int)}</li>
     * <li>{@link UiScrollable#scrollBackward()}</li>
     * <li>{@link UiScrollable#scrollForward()}</li>
     * </ul>
     *
     * @param steps indicates the number of injected move steps into the system. Steps are
     * injected about 5ms apart. So a 100 steps may take about 1/2 second to complete.
     * @return true if successful
     * @throws UiObjectNotFoundException
     */
    public boolean swipeLeft(int steps) throws UiObjectNotFoundException {
        Rect rect = getVisibleBounds();
        if (rect.width() <= SWIPE_MARGIN_LIMIT * 2) {
            Log.w(TAG, String.format("Cannot swipe. Object width too small (%d < %d).",
                    rect.width(), SWIPE_MARGIN_LIMIT * 2));
            return false;
        }
        Log.d(TAG, String.format("Swiping left from (%d, %d) to (%d, %d) in %d steps.",
                rect.right - SWIPE_MARGIN_LIMIT, rect.centerY(), rect.left + SWIPE_MARGIN_LIMIT,
                rect.centerY(), steps));
        return getInteractionController().swipe(rect.right - SWIPE_MARGIN_LIMIT,
                rect.centerY(), rect.left + SWIPE_MARGIN_LIMIT, rect.centerY(), steps);
    }
    /**
     * Performs the swipe right action on the UiObject. 
     * The swipe gesture can be performed over any surface. The targeted
     * UI element does not need to be scrollable.
     * See also:
     * <ul>
     * <li>{@link UiScrollable#scrollToBeginning(int)}</li>
     * <li>{@link UiScrollable#scrollToEnd(int)}</li>
     * <li>{@link UiScrollable#scrollBackward()}</li>
     * <li>{@link UiScrollable#scrollForward()}</li>
     * </ul>
     *
     * @param steps indicates the number of injected move steps into the system. Steps are
     * injected about 5ms apart. So a 100 steps may take about 1/2 second to complete.
     * @return true if successful
     * @throws UiObjectNotFoundException
     */
    public boolean swipeRight(int steps) throws UiObjectNotFoundException {
        Rect rect = getVisibleBounds();
        if (rect.width() <= SWIPE_MARGIN_LIMIT * 2) {
            Log.w(TAG, String.format("Cannot swipe. Object width too small (%d < %d).",
                    rect.width(), SWIPE_MARGIN_LIMIT * 2));
            return false;
        }
        Log.d(TAG, String.format("Swiping right from (%d, %d) to (%d, %d) in %d steps.",
                rect.left + SWIPE_MARGIN_LIMIT, rect.centerY(), rect.right - SWIPE_MARGIN_LIMIT,
                rect.centerY(), steps));
        return getInteractionController().swipe(rect.left + SWIPE_MARGIN_LIMIT,
                rect.centerY(), rect.right - SWIPE_MARGIN_LIMIT, rect.centerY(), steps);
    }
    /**
     * Finds the visible bounds of a partially visible UI element
     *
     * @param node
     * @return null if node is null, else a Rect containing visible bounds
     */
    Rect getVisibleBounds(AccessibilityNodeInfo node) {
        if (node == null) {
            return null;
        }
        // targeted node's bounds
        int w = getDevice().getDisplayWidth();
        int h = getDevice().getDisplayHeight();
        return AccessibilityNodeInfoHelper.getVisibleBoundsInScreen(node, w, h, true);
    }
    /**
     * Performs a click at the center of the visible bounds of the UI element represented
     * by this UiObject.
     *
     * @return true id successful else false
     * @throws UiObjectNotFoundException
     */
    public boolean click() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        Rect rect = getVisibleBounds(node);
        Log.d(TAG, String.format("Clicking on (%d, %d).", rect.centerX(), rect.centerY()));
        return getInteractionController().clickAndSync(rect.centerX(), rect.centerY(),
                mConfig.getActionAcknowledgmentTimeout());
    }
    /**
     * Waits for window transitions that would typically take longer than the
     * usual default timeouts.
     * See {@link #clickAndWaitForNewWindow(long)}
     *
     * @return true if the event was triggered, else false
     * @throws UiObjectNotFoundException
     */
    public boolean clickAndWaitForNewWindow() throws UiObjectNotFoundException {
        return clickAndWaitForNewWindow(WAIT_FOR_WINDOW_TMEOUT);
    }
    /**
     * Performs a click at the center of the visible bounds of the UI element represented
     * by this UiObject and waits for window transitions.
     *
     * This method differ from {@link UiObject#click()} only in that this method waits for a
     * a new window transition as a result of the click. Some examples of a window transition:
     * <ul>
     * <li>launching a new activity</li>
     * <li>bringing up a pop-up menu</li>
     * <li>bringing up a dialog</li>
     * </ul>
     *
     * @param timeout timeout before giving up on waiting for a new window
     * @return true if the event was triggered, else false
     * @throws UiObjectNotFoundException
     */
    public boolean clickAndWaitForNewWindow(long timeout) throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        Rect rect = getVisibleBounds(node);
        Log.d(TAG,
                String.format("Clicking on (%d, %d) and waiting %dms for new window.",
                        rect.centerX(), rect.centerY(), timeout));
        return getInteractionController().clickAndWaitForNewWindow(rect.centerX(), rect.centerY(),
                timeout);
    }
    /**
     * Clicks the top and left corner of the UI element
     *
     * @return true on success
     * @throws UiObjectNotFoundException
     */
    public boolean clickTopLeft() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        Rect rect = getVisibleBounds(node);
        Log.d(TAG, String.format("Clicking on (%d, %d).", rect.left + 5, rect.top + 5));
        return getInteractionController().clickAndSync(rect.left + 5, rect.top + 5,
                mConfig.getActionAcknowledgmentTimeout());
    }
    /**
     * Long clicks bottom and right corner of the UI element
     *
     * @return true if operation was successful
     * @throws UiObjectNotFoundException
     */
    public boolean longClickBottomRight() throws UiObjectNotFoundException  {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        Rect rect = getVisibleBounds(node);
        Log.d(TAG, String.format("Long-clicking on (%d, %d).", rect.right - 5, rect.bottom - 5));
        return getInteractionController().longTapAndSync(rect.right - 5, rect.bottom - 5,
                mConfig.getActionAcknowledgmentTimeout());
    }
    /**
     * Clicks the bottom and right corner of the UI element
     *
     * @return true on success
     * @throws UiObjectNotFoundException
     */
    public boolean clickBottomRight() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        Rect rect = getVisibleBounds(node);
        Log.d(TAG, String.format("Clicking on (%d, %d).", rect.right - 5, rect.bottom - 5));
        return getInteractionController().clickAndSync(rect.right - 5, rect.bottom - 5,
                mConfig.getActionAcknowledgmentTimeout());
    }
    /**
     * Long clicks the center of the visible bounds of the UI element
     *
     * @return true if operation was successful
     * @throws UiObjectNotFoundException
     */
    public boolean longClick() throws UiObjectNotFoundException  {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        Rect rect = getVisibleBounds(node);
        Log.d(TAG, String.format("Long-clicking on (%d, %d).", rect.centerX(), rect.centerY()));
        return getInteractionController().longTapAndSync(rect.centerX(), rect.centerY(),
                mConfig.getActionAcknowledgmentTimeout());
    }
    /**
     * Long clicks on the top and left corner of the UI element
     *
     * @return true if operation was successful
     * @throws UiObjectNotFoundException
     */
    public boolean longClickTopLeft() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        Rect rect = getVisibleBounds(node);
        Log.d(TAG, String.format("Long-clicking on (%d, %d).", rect.left + 5, rect.top + 5));
        return getInteractionController().longTapAndSync(rect.left + 5, rect.top + 5,
                mConfig.getActionAcknowledgmentTimeout());
    }
    /**
     * Reads the <code>text</code> property of the UI element
     *
     * @return text value of the current node represented by this UiObject
     * @throws UiObjectNotFoundException if no match could be found
     */
    @NonNull
    public String getText() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return safeStringReturn(node.getText());
    }
    /**
     * Retrieves the <code>className</code> property of the UI element.
     *
     * @return class name of the current node represented by this UiObject
     * @throws UiObjectNotFoundException if no match was found
     */
    @NonNull
    public String getClassName() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return safeStringReturn(node.getClassName());
    }
    /**
     * Reads the <code>content_desc</code> property of the UI element
     *
     * @return value of node attribute "content_desc"
     * @throws UiObjectNotFoundException
     */
    @NonNull
    public String getContentDescription() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return safeStringReturn(node.getContentDescription());
    }
    /**
     * Sets the text in an editable field, after clearing the field's content.
     *
     * <p>
     * The {@link UiSelector} selector of this object must reference a UI element that is editable.
     *
     * <p>
     * When you call this method, the method sets focus on the editable field, clears its existing
     * content, then injects your specified text into the field.
     *
     * <p>
     * If you want to capture the original contents of the field, call {@link #getText()} first.
     * You can then modify the text and use this method to update the field.
     *
     * <p><strong>Improvements: </strong>
     * Post API Level 19 (KitKat release), the underlying implementation is updated to a dedicated
     * set text accessibility action, and it also now supports Unicode.
     *
     * @param text string to set
     * @return true if operation is successful
     * @throws UiObjectNotFoundException
     */
    public boolean setText(@Nullable String text) throws UiObjectNotFoundException {
        // per framework convention, setText with null means clearing it
        if (text == null) {
            text = "";
        }
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if (node == null) {
            throw new UiObjectNotFoundException(getSelector().toString());
        }
        Log.d(TAG, String.format("Setting text to '%s'.", text));
        Bundle args = new Bundle();
        args.putCharSequence(AccessibilityNodeInfo.ACTION_ARGUMENT_SET_TEXT_CHARSEQUENCE, text);
        return node.performAction(AccessibilityNodeInfo.ACTION_SET_TEXT, args);
    }
    /**
     * Clears the existing text contents in an editable field.
     *
     * The {@link UiSelector} of this object must reference a UI element that is editable.
     *
     * When you call this method, the method sets focus on the editable field, selects all of its
     * existing content, and clears it by sending a DELETE key press
     *
     * @throws UiObjectNotFoundException
     */
    public void clearTextField() throws UiObjectNotFoundException {
        // long click left + center
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        CharSequence text = node.getText();
        // do nothing if already empty
        if (text != null && text.length() > 0) {
            setText("");
        }
    }
    /**
     * Check if the UI element's <code>checked</code> property is currently true
     *
     * @return true if it is else false
     */
    public boolean isChecked() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return node.isChecked();
    }
    /**
     * Checks if the UI element's <code>selected</code> property is currently true.
     *
     * @return true if it is else false
     * @throws UiObjectNotFoundException
     */
    public boolean isSelected() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return node.isSelected();
    }
    /**
     * Checks if the UI element's <code>checkable</code> property is currently true.
     *
     * @return true if it is else false
     * @throws UiObjectNotFoundException
     */
    public boolean isCheckable() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return node.isCheckable();
    }
    /**
     * Checks if the UI element's <code>enabled</code> property is currently true.
     *
     * @return true if it is else false
     * @throws UiObjectNotFoundException
     */
    public boolean isEnabled() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return node.isEnabled();
    }
    /**
     * Checks if the UI element's <code>clickable</code> property is currently true.
     *
     * @return true if it is else false
     * @throws UiObjectNotFoundException
     */
    public boolean isClickable() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return node.isClickable();
    }
    /**
     * Check if the UI element's <code>focused</code> property is currently true
     *
     * @return true if it is else false
     * @throws UiObjectNotFoundException
     */
    public boolean isFocused() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return node.isFocused();
    }
    /**
     * Check if the UI element's <code>focusable</code> property is currently true.
     *
     * @return true if it is else false
     * @throws UiObjectNotFoundException
     */
    public boolean isFocusable() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return node.isFocusable();
    }
    /**
     * Check if the view's <code>scrollable</code> property is currently true
     *
     * @return true if it is else false
     * @throws UiObjectNotFoundException
     */
    public boolean isScrollable() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return node.isScrollable();
    }
    /**
     * Check if the view's <code>long-clickable</code> property is currently true
     *
     * @return true if it is else false
     * @throws UiObjectNotFoundException
     */
    public boolean isLongClickable() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return node.isLongClickable();
    }
    /**
     * Reads the view's <code>package</code> property
     *
     * @return true if it is else false
     * @throws UiObjectNotFoundException
     */
    @NonNull
    public String getPackageName() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return safeStringReturn(node.getPackageName());
    }
    /**
     * Returns the visible bounds of the view.
     *
     * If a portion of the view is visible, only the bounds of the visible portion are
     * reported.
     *
     * @return Rect
     * @throws UiObjectNotFoundException
     * @see #getBounds()
     */
    @NonNull
    public Rect getVisibleBounds() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        return getVisibleBounds(node);
    }
    /**
     * Returns the view's <code>bounds</code> property. See {@link #getVisibleBounds()}
     *
     * @return Rect
     * @throws UiObjectNotFoundException
     */
    @NonNull
    public Rect getBounds() throws UiObjectNotFoundException {
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if(node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        Rect nodeRect = new Rect();
        node.getBoundsInScreen(nodeRect);
        return nodeRect;
    }
    /**
     * Waits a specified length of time for a view to become visible.
     *
     * This method waits until the view becomes visible on the display, or
     * until the timeout has elapsed. You can use this method in situations where
     * the content that you want to select is not immediately displayed.
     *
     * @param timeout the amount of time to wait (in milliseconds)
     * @return true if the view is displayed, else false if timeout elapsed while waiting
     */
    public boolean waitForExists(long timeout) {
        Log.d(TAG, String.format("Waiting %dms for %s.", timeout, mUiSelector));
        return findAccessibilityNodeInfo(timeout) != null;
    }
    /**
     * Waits a specified length of time for a view to become undetectable.
     *
     * This method waits until a view is no longer matchable, or until the
     * timeout has elapsed.
     *
     * A view becomes undetectable when the {@link UiSelector} of the object is
     * unable to find a match because the element has either changed its state or is no
     * longer displayed.
     *
     * You can use this method when attempting to wait for some long operation
     * to compete, such as downloading a large file or connecting to a remote server.
     *
     * @param timeout time to wait (in milliseconds)
     * @return true if the element is gone before timeout elapsed, else false if timeout elapsed
     * but a matching element is still found.
     */
    public boolean waitUntilGone(long timeout) {
        Log.d(TAG, String.format("Waiting %dms for %s to be gone.", timeout, mUiSelector));
        long startMills = SystemClock.uptimeMillis();
        long currentMills = 0;
        while (currentMills <= timeout) {
            if(findAccessibilityNodeInfo(0) == null)
                return true;
            currentMills = SystemClock.uptimeMillis() - startMills;
            if(timeout > 0)
                SystemClock.sleep(WAIT_FOR_SELECTOR_POLL);
        }
        return false;
    }
    /**
     * Check if view exists.
     *
     * This methods performs a {@link #waitForExists(long)} with zero timeout. This
     * basically returns immediately whether the view represented by this UiObject
     * exists or not. If you need to wait longer for this view, then see
     * {@link #waitForExists(long)}.
     *
     * @return true if the view represented by this UiObject does exist
     */
    public boolean exists() {
        return waitForExists(0);
    }
    private String safeStringReturn(CharSequence cs) {
        if(cs == null)
            return "";
        return cs.toString();
    }
    /**
     * Performs a two-pointer gesture, where each pointer moves diagonally
     * opposite across the other, from the center out towards the edges of the
     * this UiObject.
     * @param percent percentage of the object's diagonal length for the pinch gesture
     * @param steps the number of steps for the gesture. Steps are injected 
     * about 5 milliseconds apart, so 100 steps may take around 0.5 seconds to complete.
     * @return <code>true</code> if all touch events for this gesture are injected successfully,
     *         <code>false</code> otherwise
     * @throws UiObjectNotFoundException
     */
    public boolean pinchOut(int percent, int steps) throws UiObjectNotFoundException {
        if (percent < 0 || percent > 100) {
            throw new IllegalArgumentException("Percent must be between 0 and 100");
        }
        float percentage = percent / 100f;
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if (node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        Rect rect = getVisibleBounds(node);
        if (rect.width() <= FINGER_TOUCH_HALF_WIDTH * 2)
            throw new IllegalStateException("Object width is too small for operation");
        // start from the same point at the center of the control
        Point startPoint1 = new Point(rect.centerX() - FINGER_TOUCH_HALF_WIDTH, rect.centerY());
        Point startPoint2 = new Point(rect.centerX() + FINGER_TOUCH_HALF_WIDTH, rect.centerY());
        // End at the top-left and bottom-right corners of the control
        Point endPoint1 = new Point(rect.centerX() - (int)((rect.width()/2) * percentage),
                rect.centerY());
        Point endPoint2 = new Point(rect.centerX() + (int)((rect.width()/2) * percentage),
                rect.centerY());
        return performTwoPointerGesture(startPoint1, startPoint2, endPoint1, endPoint2, steps);
    }
    /**
     * Performs a two-pointer gesture, where each pointer moves diagonally
     * toward the other, from the edges to the center of this UiObject .
     * @param percent percentage of the object's diagonal length for the pinch gesture
     * @param steps the number of steps for the gesture. Steps are injected 
     * about 5 milliseconds apart, so 100 steps may take around 0.5 seconds to complete.
     * @return <code>true</code> if all touch events for this gesture are injected successfully,
     *         <code>false</code> otherwise
     * @throws UiObjectNotFoundException
     */
    public boolean pinchIn(int percent, int steps) throws UiObjectNotFoundException {
        if (percent < 0 || percent > 100) {
            throw new IllegalArgumentException("Percent must be between 0 and 100");
        }
        float percentage = percent / 100f;
        AccessibilityNodeInfo node = findAccessibilityNodeInfo(mConfig.getWaitForSelectorTimeout());
        if (node == null) {
            throw new UiObjectNotFoundException(mUiSelector.toString());
        }
        Rect rect = getVisibleBounds(node);
        if (rect.width() <= FINGER_TOUCH_HALF_WIDTH * 2)
            throw new IllegalStateException("Object width is too small for operation");
        Point startPoint1 = new Point(rect.centerX() - (int)((rect.width()/2) * percentage),
                rect.centerY());
        Point startPoint2 = new Point(rect.centerX() + (int)((rect.width()/2) * percentage),
                rect.centerY());
        Point endPoint1 = new Point(rect.centerX() - FINGER_TOUCH_HALF_WIDTH, rect.centerY());
        Point endPoint2 = new Point(rect.centerX() + FINGER_TOUCH_HALF_WIDTH, rect.centerY());
        return performTwoPointerGesture(startPoint1, startPoint2, endPoint1, endPoint2, steps);
    }
    /**
     * Generates a two-pointer gesture with arbitrary starting and ending points.
     *
     * @param startPoint1 start point of pointer 1
     * @param startPoint2 start point of pointer 2
     * @param endPoint1 end point of pointer 1
     * @param endPoint2 end point of pointer 2
     * @param steps the number of steps for the gesture. Steps are injected 
     * about 5 milliseconds apart, so 100 steps may take around 0.5 seconds to complete.
     * @return <code>true</code> if all touch events for this gesture are injected successfully,
     *         <code>false</code> otherwise
     */
    public boolean performTwoPointerGesture(@NonNull Point startPoint1, @NonNull Point startPoint2,
            @NonNull Point endPoint1, @NonNull Point endPoint2, int steps) {
        // avoid a divide by zero
        if(steps == 0)
            steps = 1;
        final float stepX1 = (endPoint1.x - startPoint1.x) / steps;
        final float stepY1 = (endPoint1.y - startPoint1.y) / steps;
        final float stepX2 = (endPoint2.x - startPoint2.x) / steps;
        final float stepY2 = (endPoint2.y - startPoint2.y) / steps;
        int eventX1, eventY1, eventX2, eventY2;
        eventX1 = startPoint1.x;
        eventY1 = startPoint1.y;
        eventX2 = startPoint2.x;
        eventY2 = startPoint2.y;
        // allocate for steps plus first down and last up
        PointerCoords[] points1 = new PointerCoords[steps + 2];
        PointerCoords[] points2 = new PointerCoords[steps + 2];
        // Include the first and last touch downs in the arrays of steps
        for (int i = 0; i < steps + 1; i++) {
            PointerCoords p1 = new PointerCoords();
            p1.x = eventX1;
            p1.y = eventY1;
            p1.pressure = 1;
            p1.size = 1;
            points1[i] = p1;
            PointerCoords p2 = new PointerCoords();
            p2.x = eventX2;
            p2.y = eventY2;
            p2.pressure = 1;
            p2.size = 1;
            points2[i] = p2;
            eventX1 = (int) (eventX1 + stepX1);
            eventY1 = (int) (eventY1 + stepY1);
            eventX2 = (int) (eventX2 + stepX2);
            eventY2 = (int) (eventY2 + stepY2);
        }
        // ending pointers coordinates
        PointerCoords p1 = new PointerCoords();
        p1.x = endPoint1.x;
        p1.y = endPoint1.y;
        p1.pressure = 1;
        p1.size = 1;
        points1[steps + 1] = p1;
        PointerCoords p2 = new PointerCoords();
        p2.x = endPoint2.x;
        p2.y = endPoint2.y;
        p2.pressure = 1;
        p2.size = 1;
        points2[steps + 1] = p2;
        return performMultiPointerGesture(points1, points2);
    }
    /**
     * Performs a multi-touch gesture. You must specify touch coordinates for
     * at least 2 pointers. Each pointer must have all of its touch steps
     * defined in an array of {@link PointerCoords}. You can use this method to
     * specify complex gestures, like circles and irregular shapes, where each
     * pointer may take a different path.
     *
     * To create a single point on a pointer's touch path:
     * <code>
     *       PointerCoords p = new PointerCoords();
     *       p.x = stepX;
     *       p.y = stepY;
     *       p.pressure = 1;
     *       p.size = 1;
     * </code>
     * @param touches represents the pointers' paths. Each {@link PointerCoords}
     * array represents a different pointer. Each {@link PointerCoords} in an
     * array element represents a touch point on a pointer's path.
     * @return <code>true</code> if all touch events for this gesture are injected successfully,
     *         <code>false</code> otherwise
     */
    public boolean performMultiPointerGesture(@NonNull PointerCoords[]... touches) {
        Log.d(TAG, String.format("Performing multi-point gesture %s.", touchesToString(touches)));
        return getInteractionController().performMultiPointerGesture(touches);
    }
    private static String touchesToString(@NonNull PointerCoords[]... touches) {
        StringBuilder result = new StringBuilder();
        result.append("[");
        for (int i = 0; i < touches.length; i++) {
            result.append("[");
            for (int j = 0; j < touches[i].length; j++) {
                PointerCoords point = touches[i][j];
                result.append(String.format("(%f, %f)", point.x, point.y));
                if (j + 1 < touches[i].length) result.append(", ");
            }
            result.append("]");
            if (i + 1 < touches.length) result.append(", ");
        }
        result.append("]");
        return result.toString();
    }
}