Testowanie interfejsów API

Elementy interfejsu można obsługiwać na 3 główne sposoby:

  • Wyszukiwarki umożliwiają wybieranie co najmniej 1 elementu (lub węzła w drzewie semantycznym) w celu tworzenia asercji lub wykonywania na nich działań.
  • Asercje służą do sprawdzania, czy elementy istnieją lub mają określone atrybuty.
  • Działania wstawiają w elementach symulowane zdarzenia użytkownika, takie jak kliknięcia lub inne gesty.

Niektóre z tych interfejsów API akceptują SemanticsMatcher, aby odwoływać się do co najmniej jednego węzła w drzewie semantycznym.

Finders

Możesz użyć onNodeonAllNodes, aby wybrać odpowiednio jeden lub kilka węzłów, ale możesz też użyć wygodnych wyszukiwarek do najczęstszych wyszukiwań, takich jak onNodeWithTextonNodeWithContentDescription. Pełną listę znajdziesz w arkuszu z przydatnymi informacjami o testowaniu w Compose.

Wybieranie pojedynczego węzła

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

Wybieranie wielu węzłów

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

Niezłączone drzewo

Niektóre węzły łączą informacje semantyczne swoich elementów podrzędnych. Na przykład przycisk z 2 elementami tekstowymi łączy etykiety elementów tekstowych:

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

W teście użyj printToLog(), aby wyświetlić drzewo semantyczne:

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

Ten kod wyświetla te dane wyjściowe:

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

Jeśli chcesz dopasować węzeł do drzewa niepołączonego, możesz ustawić useUnmergedTree na true:

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

Ten kod wyświetla te dane wyjściowe:

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]'

Parametr useUnmergedTree jest dostępny we wszystkich wyszukiwarkach. Na przykład tutaj jest używana w onNodeWithText.

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

Asercje

Sprawdź asercje, wywołując assert() na obiekcie SemanticsNodeInteraction zwróconym przez wyszukiwarkę z co najmniej 1 elementem dopasowującym:

// 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"))

Możesz też używać funkcji ułatwiających tworzenie najczęstszych asercji, takich jak assertExists, assertIsDisplayedassertTextEquals. Pełną listę znajdziesz w arkuszu z przydatnymi informacjami o testowaniu w Compose.

Dostępne są też funkcje sprawdzania asercji w kolekcji węzłów:

// 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())

Działania

Aby wstawić działanie w węźle, wywołaj funkcję perform…():

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

Oto kilka przykładów działań:

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

Pełną listę znajdziesz w arkuszu z przydatnymi informacjami o testowaniu w Compose.

Dopasowywanie

Do testowania kodu Compose dostępnych jest wiele dopasowań.

Dopasowania hierarchiczne

Dopasowywanie hierarchiczne umożliwia poruszanie się w górę i w dół drzewa semantycznego oraz przeprowadzanie dopasowywania.

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

Oto kilka przykładów użycia tych dopasowań:

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

Selektory

Alternatywnym sposobem tworzenia testów jest używanie selektorów, które mogą sprawić, że niektóre testy będą bardziej czytelne.

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

Pełną listę znajdziesz w arkuszu z przydatnymi informacjami o testowaniu w Compose.

Dodatkowe materiały

  • Testowanie aplikacji na Androida: główna strona docelowa dotycząca testowania na Androidzie zawiera szersze omówienie podstaw testowania i technik testowania.
  • Podstawy testowania: dowiedz się więcej o podstawowych koncepcjach związanych z testowaniem aplikacji na Androida.
  • Testy lokalne: niektóre testy możesz przeprowadzać lokalnie, na własnej stacji roboczej.
  • Testy z użyciem instrumentacji: warto też przeprowadzać testy z użyciem instrumentacji. Są to testy, które są przeprowadzane bezpośrednio na urządzeniu.
  • Ciągła integracja: Ciągła integracja umożliwia zintegrowanie testów z potokiem wdrażania.
  • Testowanie różnych rozmiarów ekranu: użytkownicy mają do dyspozycji wiele urządzeń, dlatego warto testować różne rozmiary ekranu.
  • Espresso: chociaż Espresso jest przeznaczony do interfejsów opartych na widokach, wiedza na jego temat może być przydatna w przypadku niektórych aspektów testowania Compose.