Платформа тестирования эспрессо - AdapterView
AdapterView - это особый вид представления, специально разработанный для отображения набора аналогичной информации, например списка продуктов и контактов пользователей, полученных из базового источника данных с помощью адаптера . Источником данных может быть простой список для сложных записей базы данных. Некоторые представления, производные от AdapterView, - это ListView , GridView и Spinner .
AdapterView отображает пользовательский интерфейс динамически в зависимости от количества данных, доступных в базовом источнике данных. Кроме того, AdapterView отображает только минимально необходимые данные, которые можно отобразить в доступной видимой области экрана. AdapterView делает это для экономии памяти и для того, чтобы пользовательский интерфейс выглядел гладко, даже если базовые данные велики.
После анализа природа архитектуры AdapterView делает параметр onView и его сопоставители представлений неактуальными, поскольку конкретное представление, которое нужно протестировать, может вообще не отображаться. К счастью, espresso предоставляет метод onData ( ), который принимает сопоставители hamcrest (относящиеся к типу данных базовых данных) для сопоставления с базовыми данными и возвращает объект типа DataInteraction, соответствующий представлению сопоставленных данных. Пример кода выглядит следующим образом:
onData(allOf(is(instanceOf(String.class)), startsWith("Apple"))).perform(click())
Здесь onData () соответствует записи «Apple», если она доступна в базовых данных (список массивов), и возвращает объект DataInteraction для взаимодействия с согласованным представлением (TextView соответствует записи «Apple»).
Методы
DataInteraction предоставляет следующие методы для взаимодействия с представлением,
выполнить ()
Это принимает действия просмотра и запускает переданные действия просмотра.
onData(allOf(is(instanceOf(String.class)), startsWith("Apple"))).perform(click())
чек()
Это принимает утверждения представления и проверяет переданные утверждения представления.
onData(allOf(is(instanceOf(String.class)), startsWith("Apple")))
.check(matches(withText("Apple")))
inAdapterView ()
Это принимает сопоставители представлений. Он выбирает конкретный AdapterView на основе переданных сопоставителей представления и возвращает объект DataInteraction для взаимодействия с сопоставленным AdapterView.
onData(allOf())
.inAdapterView(withId(R.id.adapter_view))
.atPosition(5)
.perform(click())
atPosition ()
Это принимает аргумент целочисленного типа и ссылается на позицию элемента в базовых данных. Он выбирает представление, соответствующее переданному позиционному значению данных, и возвращает объект DataInteraction для взаимодействия с согласованным представлением. Это будет полезно, если мы знаем правильный порядок лежащих в основе данных.
onData(allOf())
.inAdapterView(withId(R.id.adapter_view))
.atPosition(5)
.perform(click())
onChildView ()
Это принимает сопоставители представлений и сопоставляет представление внутри определенного дочернего представления. Например, мы можем взаимодействовать с определенными элементами, такими как кнопка « Купить» , в списке продуктов на основе AdapterView .
onData(allOf(is(instanceOf(String.class)), startsWith("Apple")))
.onChildView(withId(R.id.buy_button))
.perform(click())
Напишите образец заявления
Выполните шаги, показанные ниже, чтобы написать простое приложение на основе AdapterView и написать тестовый пример с помощью метода onData () .
Запустить студию Android.
Создайте новый проект, как обсуждалось ранее, и назовите его MyFruitApp .
Перенесите приложение в платформу AndroidX с помощью меню параметров Refactor → Migrate to AndroidX .
Удалите дизайн по умолчанию в основном действии и добавьте ListView . Содержимое activity_main.xml выглядит следующим образом:
<?xml version = "1.0" encoding = "utf-8"?>
<RelativeLayout xmlns:android = "http://schemas.android.com/apk/res/android"
xmlns:app = "http://schemas.android.com/apk/res-auto"
xmlns:tools = "http://schemas.android.com/tools"
android:layout_width = "match_parent"
android:layout_height = "match_parent"
tools:context = ".MainActivity">
<ListView
android:id = "@+id/listView"
android:layout_width = "wrap_content"
android:layout_height = "wrap_content" />
</RelativeLayout>
Добавьте новый ресурс макета, item.xml, чтобы указать шаблон элемента представления списка. Содержимое item.xml выглядит следующим образом:
<?xml version = "1.0" encoding = "utf-8"?>
<TextView xmlns:android = "http://schemas.android.com/apk/res/android"
android:id = "@+id/name"
android:layout_width = "fill_parent"
android:layout_height = "fill_parent"
android:padding = "8dp"
/>
Теперь создайте адаптер с массивом фруктов в качестве базовых данных и установите его в виде списка. Это должно быть сделано в OnCreate () из MainActivity , как указано ниже,
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// Find fruit list view
final ListView listView = (ListView) findViewById(R.id.listView);
// Initialize fruit data
String[] fruits = new String[]{
"Apple",
"Banana",
"Cherry",
"Dates",
"Elderberry",
"Fig",
"Grapes",
"Grapefruit",
"Guava",
"Jack fruit",
"Lemon",
"Mango",
"Orange",
"Papaya",
"Pears",
"Peaches",
"Pineapple",
"Plums",
"Raspberry",
"Strawberry",
"Watermelon"
};
// Create array list of fruits
final ArrayList<String> fruitList = new ArrayList<String>();
for (int i = 0; i < fruits.length; ++i) {
fruitList.add(fruits[i]);
}
// Create Array adapter
final ArrayAdapter adapter = new ArrayAdapter(this, R.layout.item, fruitList);
// Set adapter in list view
listView.setAdapter(adapter);
}
Теперь скомпилируйте код и запустите приложение. Снимок экрана приложения « Мои фрукты» выглядит следующим образом:
Теперь откройте файл ExampleInstrumentedTest.java и добавьте ActivityTestRule, как указано ниже,
@Rule
public ActivityTestRule<MainActivity> mActivityRule =
new ActivityTestRule<MainActivity>(MainActivity.class);
Также убедитесь, что тестовая конфигурация выполнена в app / build.gradle -
dependencies {
testImplementation 'junit:junit:4.12'
androidTestImplementation 'androidx.test:runner:1.1.1'
androidTestImplementation 'androidx.test:rules:1.1.1'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.1'
}
Добавьте новый тестовый пример, чтобы проверить представление списка, как показано ниже,
@Test
public void listView_isCorrect() {
// check list view is visible
onView(withId(R.id.listView)).check(matches(isDisplayed()));
onData(allOf(is(instanceOf(String.class)), startsWith("Apple"))).perform(click());
onData(allOf(is(instanceOf(String.class)), startsWith("Apple")))
.check(matches(withText("Apple")));
// click a child item
onData(allOf())
.inAdapterView(withId(R.id.listView))
.atPosition(10)
.perform(click());
}
Наконец, запустите тестовый пример с помощью контекстного меню студии Android и проверьте, все ли тестовые примеры выполнены успешно.