page.title=Testing UI for Multiple Apps page.tags=testing,ui automator trainingnavtop=true @jd:body

Dependencies and Prerequisites

This lesson teaches you to

  1. Set Up UI Automator
  2. Create a UI Automator Test Class
  3. Run UI Automator Tests on a Device or Emulator

You should also read

Try it out

A user interface (UI) test that involves user interactions across multiple apps lets you verify that your app behaves correctly when the user flow crosses into other apps or into the system UI. An example of such a user flow is a messaging app that lets the user enter a text message, launches the Android contact picker so that the users can select recipients to send the message to, and then returns control to the original app for the user to submit the message.

This lesson covers how to write such UI tests using the UI Automator testing framework provided by the Android Testing Support Library. The UI Automator APIs let you interact with visible elements on a device, regardless of which {@link android.app.Activity} is in focus. Your test can look up a UI component by using convenient descriptors such as the text displayed in that component or its content description. UI Automator tests can run on devices running Android 4.3 (API level 18) or higher.

The UI Automator testing framework is an instrumentation-based API and works with the {@code AndroidJUnitRunner} test runner.

Set Up UI Automator

Before building your UI test with UI Automator, make sure to configure your test source code location and project dependencies, as described in Getting Started with Testing.

In the {@code build.gradle} file of your Android app module, you must set a dependency reference to the UI Automator library:

dependencies {
    ...
    androidTestCompile 'com.android.support.test.uiautomator:uiautomator-v18:2.1.1'
}

To optimize your UI Automator testing, you should first inspect the target app’s UI components and ensure that they are accessible. These optimization tips are described in the next two sections.

Inspecting the UI on a device

Before designing your test, inspect the UI components that are visible on the device. To ensure that your UI Automator tests can access these components, check that these components have visible text labels, {@code android:contentDescription} values, or both.

The {@code uiautomatorviewer} tool provides a convenient visual interface to inspect the layout hierarchy and view the properties of UI components that are visible on the foreground of the device. This information lets you create more fine-grained tests using UI Automator. For example, you can create a UI selector that matches a specific visible property.

To launch the {@code uiautomatorviewer} tool:

  1. Launch the target app on a physical device.
  2. Connect the device to your development machine.
  3. Open a terminal window and navigate to the {@code /tools/} directory.
  4. Run the tool with this command:
    $ uiautomatorviewer

To view the UI properties for your application:

  1. In the {@code uiautomatorviewer} interface, click the Device Screenshot button.
  2. Hover over the snapshot in the left-hand panel to see the UI components identified by the {@code uiautomatorviewertool}. The properties are listed in the lower right-hand panel and the layout hierarchy in the upper right-hand panel.
  3. Optionally, click on the Toggle NAF Nodes button to see UI components that are non-accessible to UI Automator. Only limited information may be available for these components.

To learn about the common types of UI components provided by Android, see User Interface.

Ensuring your Activity is accessible

The UI Automator test framework depends on the accessibility features of the Android framework to look up individual UI elements. As a developer, you should implement these minimum optimizations in your {@link android.app.Activity} to support UI Automator:

Generally, app developers get accessibility support for free, courtesy of the {@link android.view.View} and {@link android.view.ViewGroup} classes. However, some apps use custom view elements to provide a richer user experience. Such custom elements won't get the accessibility support that is provided by the standard Android UI elements. If this applies to your app, make sure that it exposes the custom-drawn UI element to Android accessibility services by implementing the {@link android.view.accessibility.AccessibilityNodeProvider} class.

If the custom view element contains a single element, make it accessible by implementing accessibility API methods. If the custom view contains elements that are not views themselves (for example, a {@link android.webkit.WebView}, make sure it implements the {@link android.view.accessibility.AccessibilityNodeProvider} class. For container views that extend an existing container implementation (for example, a {@link android.widget.ListView}), implementing {@link android.view.accessibility.AccessibilityNodeProvider} is not necessary.

For more information about implementing and testing accessibility, see Making Applications Accessible.

Create a UI Automator Test Class

Your UI Automator test class should be written the same way as a JUnit 4 test class. To learn more about creating JUnit 4 test classes and using JUnit 4 assertions and annotations, see Create an Instrumented Unit Test Class.

Add the {@code @RunWith(AndroidJUnit4.class)} annotation at the beginning of your test class definition. You also need to specify the {@code AndroidJUnitRunner} class provided in the Android Testing Support Library as your default test runner. This step is described in more detail in Run UI Automator Tests on a Device or Emulator.

Implement the following programming model in your UI Automator test class:

  1. Get a {@code UiDevice} object to access the device you want to test, by calling the {@code getInstance()} method and passing it an {@link android.app.Instrumentation} object as the argument.
  2. Get a {@code UiObject} object to access a UI component that is displayed on the device (for example, the current view in the foreground), by calling the {@code findObject()} method.
  3. Simulate a specific user interaction to perform on that UI component, by calling a {@code UiObject} method; for example, call {@code performMultiPointerGesture()} to simulate a multi-touch gesture, and {@code setText()} to edit a text field. You can call on the APIs in steps 2 and 3 repeatedly as necessary to test more complex user interactions that involve multiple UI components or sequences of user actions.
  4. Check that the UI reflects the expected state or behavior, after these user interactions are performed.

These steps are covered in more detail in the sections below.

Accessing UI Components

The {@code UiDevice} object is the primary way you access and manipulate the state of the device. In your tests, you can call {@code UiDevice} methods to check for the state of various properties, such as current orientation or display size. Your test can use the {@code UiDevice} object to perform device-level actions, such as forcing the device into a specific rotation, pressing D-pad hardware buttons, and pressing the Home and Menu buttons.

It’s good practice to start your test from the Home screen of the device. From the Home screen (or some other starting location you’ve chosen in the device), you can call the methods provided by the UI Automator API to select and interact with specific UI elements.

The following code snippet shows how your test might get an instance of {@code UiDevice} and simulate a Home button press:

import org.junit.Before;
import android.support.test.runner.AndroidJUnit4;
import android.support.test.uiautomator.UiDevice;
import android.support.test.uiautomator.By;
import android.support.test.uiautomator.Until;
...

@RunWith(AndroidJUnit4.class)
@SdkSuppress(minSdkVersion = 18)
public class ChangeTextBehaviorTest {

    private static final String BASIC_SAMPLE_PACKAGE
            = "com.example.android.testing.uiautomator.BasicSample";
    private static final int LAUNCH_TIMEOUT = 5000;
    private static final String STRING_TO_BE_TYPED = "UiAutomator";
    private UiDevice mDevice;

    @Before
    public void startMainActivityFromHomeScreen() {
        // Initialize UiDevice instance
        mDevice = UiDevice.getInstance(InstrumentationRegistry.getInstrumentation());

        // Start from the home screen
        mDevice.pressHome();

        // Wait for launcher
        final String launcherPackage = mDevice.getLauncherPackageName();
        assertThat(launcherPackage, notNullValue());
        mDevice.wait(Until.hasObject(By.pkg(launcherPackage).depth(0)),
                LAUNCH_TIMEOUT);

        // Launch the app
        Context context = InstrumentationRegistry.getContext();
        final Intent intent = context.getPackageManager()
                .getLaunchIntentForPackage(BASIC_SAMPLE_PACKAGE);
        // Clear out any previous instances
        intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK);
        context.startActivity(intent);

        // Wait for the app to appear
        mDevice.wait(Until.hasObject(By.pkg(BASIC_SAMPLE_PACKAGE).depth(0)),
                LAUNCH_TIMEOUT);
    }
}

In the example, the {@code @SdkSuppress(minSdkVersion = 18)} statement helps to ensure that tests will only run on devices with Android 4.3 (API level 18) or higher, as required by the UI Automator framework.

Use the {@code findObject()} method to retrieve a {@code UiObject} which represents a view that matches a given selector criteria. You can reuse the {@code UiObject} instances that you have created in other parts of your app testing, as needed. Note that the UI Automator test framework searches the current display for a match every time your test uses a {@code UiObject} instance to click on a UI element or query a property.

The following snippet shows how your test might construct {@code UiObject} instances that represent a Cancel button and a OK button in an app.

UiObject cancelButton = mDevice.findObject(new UiSelector()
        .text("Cancel"))
        .className("android.widget.Button"));
UiObject okButton = mDevice.findObject(new UiSelector()
        .text("OK"))
        .className("android.widget.Button"));

// Simulate a user-click on the OK button, if found.
if(okButton.exists() && okButton.isEnabled()) {
    okButton.click();
}

Specifying a selector

If you want to access a specific UI component in an app, use the {@code UiSelector} class. This class represents a query for specific elements in the currently displayed UI.

If more than one matching element is found, the first matching element in the layout hierarchy is returned as the target {@code UiObject}. When constructing a {@code UiSelector}, you can chain together multiple properties to refine your search. If no matching UI element is found, a {@code UiAutomatorObjectNotFoundException} is thrown.

You can use the {@code childSelector()} method to nest multiple {@code UiSelector} instances. For example, the following code example shows how your test might specify a search to find the first {@link android.widget.ListView} in the currently displayed UI, then search within that {@link android.widget.ListView} to find a UI element with the text property Apps.

UiObject appItem = new UiObject(new UiSelector()
        .className("android.widget.ListView")
        .instance(1)
        .childSelector(new UiSelector()
        .text("Apps")));

As a best practice, when specifying a selector, you should use a Resource ID (if one is assigned to a UI element) instead of a text element or content-descriptor. Not all elements have a text element (for example, icons in a toolbar). Text selectors are brittle and can lead to test failures if there are minor changes to the UI. They may also not scale across different languages; your text selectors may not match translated strings.

It can be useful to specify the object state in your selector criteria. For example, if you want to select a list of all checked elements so that you can uncheck them, call the {@code checked()} method with the argument set to {@code true}.

Performing Actions

Once your test has obtained a {@code UiObject} object, you can call the methods in the {@code UiObject} class to perform user interactions on the UI component represented by that object. You can specify such actions as:

The UI Automator testing framework allows you to send an {@link android.content.Intent} or launch an {@link android.app.Activity} without using shell commands, by getting a {@link android.content.Context} object through {@link android.app.Instrumentation#getContext() getContext()}.

The following snippet shows how your test can use an {@link android.content.Intent} to launch the app under test. This approach is useful when you are only interested in testing the calculator app, and don't care about the launcher.

public void setUp() {
    ...

    // Launch a simple calculator app
    Context context = getInstrumentation().getContext();
    Intent intent = context.getPackageManager()
            .getLaunchIntentForPackage(CALC_PACKAGE);
    intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK);
            // Clear out any previous instances
    context.startActivity(intent);
    mDevice.wait(Until.hasObject(By.pkg(CALC_PACKAGE).depth(0)), TIMEOUT);
}

Performing actions on collections

Use the {@code UiCollection} class if you want to simulate user interactions on a collection of items (for example, songs in a music album or a list of emails in an Inbox). To create a {@code UiCollection} object, specify a {@code UiSelector} that searches for a UI container or a wrapper of other child UI elements, such as a layout view that contains child UI elements.

The following code snippet shows how your test might construct a {@code UiCollection} to represent a video album that is displayed within a {@link android.widget.FrameLayout}:

UiCollection videos = new UiCollection(new UiSelector()
        .className("android.widget.FrameLayout"));

// Retrieve the number of videos in this collection:
int count = videos.getChildCount(new UiSelector()
        .className("android.widget.LinearLayout"));

// Find a specific video and simulate a user-click on it
UiObject video = videos.getChildByText(new UiSelector()
        .className("android.widget.LinearLayout"), "Cute Baby Laughing");
video.click();

// Simulate selecting a checkbox that is associated with the video
UiObject checkBox = video.getChild(new UiSelector()
        .className("android.widget.Checkbox"));
if(!checkBox.isSelected()) checkbox.click();

Performing actions on scrollable views

Use the {@code UiScrollable} class to simulate vertical or horizontal scrolling across a display. This technique is helpful when a UI element is positioned off-screen and you need to scroll to bring it into view.

The following code snippet shows how to simulate scrolling down the Settings menu and clicking on an About tablet option:

UiScrollable settingsItem = new UiScrollable(new UiSelector()
        .className("android.widget.ListView"));
UiObject about = settingsItem.getChildByText(new UiSelector()
        .className("android.widget.LinearLayout"), "About tablet");
about.click();

Verifying Results

The {@link android.test.InstrumentationTestCase} extends {@link junit.framework.TestCase}, so you can use standard JUnit {@code Assert} methods to test that UI components in the app return the expected results.

The following snippet shows how your test can locate several buttons in a calculator app, click on them in order, then verify that the correct result is displayed.

private static final String CALC_PACKAGE = "com.myexample.calc";

public void testTwoPlusThreeEqualsFive() {
    // Enter an equation: 2 + 3 = ?
    mDevice.findObject(new UiSelector()
            .packageName(CALC_PACKAGE).resourceId("two")).click();
    mDevice.findObject(new UiSelector()
            .packageName(CALC_PACKAGE).resourceId("plus")).click();
    mDevice.findObject(new UiSelector()
            .packageName(CALC_PACKAGE).resourceId("three")).click();
    mDevice.findObject(new UiSelector()
            .packageName(CALC_PACKAGE).resourceId("equals")).click();

    // Verify the result = 5
    UiObject result = mDevice.findObject(By.res(CALC_PACKAGE, "result"));
    assertEquals("5", result.getText());
}

Run UI Automator Tests on a Device or Emulator

You can run UI Automator tests from Android Studio or from the command-line. Make sure to specify {@code AndroidJUnitRunner} as the default instrumentation runner in your project.

To run your UI Automator test, follow the steps for running instrumented tests described in Getting Started with Testing.