Существует три основных способа взаимодействия с элементами пользовательского интерфейса:
- Поисковики позволяют выбирать один или несколько элементов (или узлов в дереве семантики) для создания утверждений или выполнения действий над ними.
- Утверждения используются для проверки того, что элементы существуют или имеют определенные атрибуты.
- Действия внедряют в элементы имитированные пользовательские события, такие как щелчки или другие жесты.
Некоторые из этих 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 .
Существуют также функции для проверки утверждений на коллекции узлов:
// 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 и предназначено для пользовательских интерфейсов на основе View, оно все равно может быть полезно для некоторых аспектов тестирования Compose.