Тестирование API

Существует три основных способа взаимодействия с элементами пользовательского интерфейса:

  • Функция поиска позволяет выбрать один или несколько элементов (или узлов в семантическом дереве), чтобы сделать над ними утверждения или выполнить над ними действия.
  • Утверждения используются для проверки существования элементов или наличия у них определенных атрибутов.
  • Действия внедряют в элементы имитированные события пользователя, такие как клики или другие жесты.

Некоторые из этих API принимают объект SemanticsMatcher для ссылки на один или несколько узлов в семантическом дереве.

Нашедшие

Вы можете использовать onNode и onAllNodes для выбора одного или нескольких узлов соответственно, но вы также можете использовать удобные средства поиска для наиболее распространенных запросов, таких как onNodeWithText и onNodeWithContentDescription . Полный список можно найти в шпаргалке по тестированию Compose .

Выберите один узел

composeTestRule.onNode(<<SemanticsMatcher>>, useUnmergedTree = false): SemanticsNodeInteraction
// Example
composeTestRule
    .onNode(hasText("Button")) // Equivalent to onNodeWithText("Button")

Выберите несколько узлов

composeTestRule
    .onAllNodes(<<SemanticsMatcher>>): SemanticsNodeInteractionCollection
// Example
composeTestRule
    .onAllNodes(hasText("Button")) // Equivalent to onAllNodesWithText("Button")

Необъединенное дерево

Некоторые узлы объединяют семантическую информацию своих дочерних элементов. Например, кнопка с двумя текстовыми элементами объединяет метки этих текстовых элементов:

MyButton {
    Text("Hello")
    Text("World")
}

В тестовом примере используйте printToLog() для отображения дерева семантики:

composeTestRule.onRoot().printToLog("TAG")

Этот код выводит следующий результат:

Node #1 at (...)px
 |-Node #2 at (...)px
   Role = 'Button'
   Text = '[Hello, World]'
   Actions = [OnClick, GetTextLayoutResult]
   MergeDescendants = 'true'

Если вам нужно сопоставить узел с необъединенным деревом, вы можете установить useUnmergedTree в true :

composeTestRule.onRoot(useUnmergedTree = true).printToLog("TAG")

Этот код выводит следующий результат:

Node #1 at (...)px
 |-Node #2 at (...)px
   OnClick = '...'
   MergeDescendants = 'true'
    |-Node #3 at (...)px
    | Text = '[Hello]'
    |-Node #5 at (83.0, 86.0, 191.0, 135.0)px
      Text = '[World]'

Параметр useUnmergedTree доступен во всех методах поиска. Например, здесь он используется в методе поиска onNodeWithText .

composeTestRule
    .onNodeWithText("World", useUnmergedTree = true).assertIsDisplayed()

Утверждения

Проверьте утверждения, вызвав assert() для объекта SemanticsNodeInteraction , возвращаемого поисковым запросом с одним или несколькими сопоставителями:

// Single matcher:
composeTestRule
    .onNode(matcher)
    .assert(hasText("Button")) // hasText is a SemanticsMatcher

// Multiple matchers can use and / or
composeTestRule
    .onNode(matcher).assert(hasText("Button") or hasText("Button2"))

Для наиболее распространенных проверок можно также использовать вспомогательные функции, такие как assertExists , assertIsDisplayed и assertTextEquals . Полный список можно найти в шпаргалке по Compose Testing .

Также имеются функции для проверки утверждений в наборе узлов:

// Check number of matched nodes
composeTestRule
    .onAllNodesWithContentDescription("Beatle").assertCountEquals(4)
// At least one matches
composeTestRule
    .onAllNodesWithContentDescription("Beatle").assertAny(hasTestTag("Drummer"))
// All of them match
composeTestRule
    .onAllNodesWithContentDescription("Beatle").assertAll(hasClickAction())

Действия

Для внедрения действия в узел вызовите функцию perform…() :

composeTestRule.onNode(...).performClick()

Вот несколько примеров действий:

performClick(),
performSemanticsAction(key),
performKeyPress(keyEvent),
performGesture { swipeLeft() }

Полный список можно посмотреть в шпаргалке по тестированию Compose .

Совпадающие

Для тестирования кода Compose доступен широкий выбор сопоставителей.

Иерархические сопоставители

Иерархические сопоставители позволяют перемещаться вверх или вниз по семантическому дереву и выполнять сопоставление.

fun hasParent(matcher: SemanticsMatcher): SemanticsMatcher
fun hasAnySibling(matcher: SemanticsMatcher): SemanticsMatcher
fun hasAnyAncestor(matcher: SemanticsMatcher): SemanticsMatcher
fun hasAnyDescendant(matcher: SemanticsMatcher):  SemanticsMatcher

Вот несколько примеров использования этих сопоставителей:

composeTestRule.onNode(hasParent(hasText("Button")))
    .assertIsDisplayed()

Селекторы

Альтернативный способ создания тестов — использование селекторов , что может сделать некоторые тесты более читабельными.

composeTestRule.onNode(hasTestTag("Players"))
    .onChildren()
    .filter(hasClickAction())
    .assertCountEquals(4)
    .onFirst()
    .assert(hasText("John"))

Полный список можно посмотреть в шпаргалке по тестированию Compose .

Дополнительные ресурсы

  • Тестирование приложений на Android : Главная страница, посвященная тестированию Android, предоставляет более широкий обзор основ и методов тестирования.
  • Основы тестирования : Узнайте больше об основных концепциях тестирования Android-приложений.
  • Локальные тесты : Вы можете запустить некоторые тесты локально, на своей рабочей станции.
  • Инструментальные тесты : Рекомендуется также запускать инструментальные тесты, то есть тесты, которые выполняются непосредственно на устройстве.
  • Непрерывная интеграция : Непрерывная интеграция позволяет интегрировать ваши тесты в конвейер развертывания.
  • Протестируйте разные размеры экрана : учитывая большое количество доступных пользователям устройств, следует протестировать их работу на экранах разных размеров.
  • Espresso : Хотя этот язык программирования предназначен для пользовательских интерфейсов на основе представлений, знание Espresso может быть полезно и для некоторых аспектов тестирования с помощью Compose.