Padrões comuns

Você pode testar o app do Compose com abordagens e padrões bem estabelecidos.

Testar de forma isolada

O ComposeTestRule permite iniciar uma atividade mostrando qualquer elemento combinável, como seu aplicativo completo, uma única tela ou um elemento pequeno. Também é uma prática recomendada verificar se os elementos combináveis estão encapsulados corretamente e funcionam de forma independente, permitindo testes de interface mais fáceis e focados.

Isso não significa que você pode criar testes de IU de unidade. Os testes voltados para partes maiores da IU também são muito importantes.

Acessar a atividade e os recursos depois de definir seu conteúdo

Muitas vezes, você precisa definir o conteúdo em teste usando composeTestRule.setContent e também acessar recursos da atividade, por exemplo, para declarar que um texto mostrado corresponde a um recurso de string. No entanto, não é possível chamar setContent em uma regra criada com createAndroidComposeRule() se a atividade já a chama.

Um padrão comum para fazer isso é criar um AndroidComposeTestRule usando uma atividade vazia, como ComponentActivity.

class MyComposeTest {

    @get:Rule
    val composeTestRule = createAndroidComposeRule<ComponentActivity>()

    @Test
    fun myTest() {
        // Start the app
        composeTestRule.setContent {
            MyAppTheme {
                MainScreen(uiState = exampleUiState, /*...*/)
            }
        }
        val continueLabel = composeTestRule.activity.getString(R.string.next)
        composeTestRule.onNodeWithText(continueLabel).performClick()
    }
}

Observe que a ComponentActivity precisa ser adicionada ao arquivo AndroidManifest.xml do app. Para ativar, adicione esta dependência ao módulo:

debugImplementation("androidx.compose.ui:ui-test-manifest:$compose_version")

Propriedades de semântica personalizadas

Você pode criar propriedades semantics personalizadas para expor informações a testes. Para fazer isso, defina um novo SemanticsPropertyKey e disponibilize-o usando a SemanticsPropertyReceiver.

// Creates a semantics property of type Long.
val PickedDateKey = SemanticsPropertyKey<Long>("PickedDate")
var SemanticsPropertyReceiver.pickedDate by PickedDateKey

Agora, use essa propriedade no modificador semantics:

val datePickerValue by remember { mutableStateOf(0L) }
MyCustomDatePicker(
    modifier = Modifier.semantics { pickedDate = datePickerValue }
)

Nos testes, use SemanticsMatcher.expectValue para declarar o valor da propriedade:

composeTestRule
    .onNode(SemanticsMatcher.expectValue(PickedDateKey, 1445378400)) // 2015-10-21
    .assertExists()

Verificar a restauração do estado

Verifique se o estado dos elementos do Compose é restaurado corretamente quando a atividade ou processo é recriado. Faça essas verificações sem depender da recriação de atividades com a classe StateRestorationTester.

Essa classe permite simular a recriação de um elemento combinável. É especialmente útil verificar a implementação de rememberSaveable.


class MyStateRestorationTests {

    @get:Rule
    val composeTestRule = createComposeRule()

    @Test
    fun onRecreation_stateIsRestored() {
        val restorationTester = StateRestorationTester(composeTestRule)

        restorationTester.setContent { MainScreen() }

        // TODO: Run actions that modify the state

        // Trigger a recreation
        restorationTester.emulateSavedInstanceStateRestore()

        // TODO: Verify that state has been correctly restored.
    }
}

Testar diferentes configurações de dispositivos

Os apps Android precisam se adaptar a muitas condições em mudança: tamanhos de janela, localidades, tamanhos de fonte, temas escuros e claros e muito mais. A maioria dessas condições é derivada de valores no nível do dispositivo controlados pelo usuário e expostas com a instância atual do Configuration. Testar configurações diferentes diretamente em um teste é difícil, já que o teste precisa configurar propriedades no nível do dispositivo.

DeviceConfigurationOverride é uma API somente para teste que permite simular diferentes configurações de dispositivos de maneira localizada para o conteúdo @Composable em teste.

O objeto complementar de DeviceConfigurationOverride tem as seguintes funções de extensão, que substituem as propriedades de configuração no nível do dispositivo:

Para aplicar uma substituição específica, envolva o conteúdo em teste em uma chamada para a função de nível superior DeviceConfigurationOverride(), transmitindo a substituição para ser aplicada como um parâmetro.

Por exemplo, o código abaixo aplica a substituição DeviceConfigurationOverride.ForcedSize() para mudar a densidade localmente, forçando o elemento combinável MyScreen a ser renderizado em uma janela grande no modo paisagem, mesmo que o dispositivo em que o teste está sendo executado não ofereça suporte direto a esse tamanho de janela:

composeTestRule.setContent {
    DeviceConfigurationOverride(
        DeviceConfigurationOverride.ForcedSize(DpSize(1280.dp, 800.dp))
    ) {
        MyScreen() // will be rendered in the space for 1280dp by 800dp without clipping
    }
}

Para aplicar várias substituições juntas, use DeviceConfigurationOverride.then():

composeTestRule.setContent {
    DeviceConfigurationOverride(
        DeviceConfigurationOverride.FontScale(1.5f) then
            DeviceConfigurationOverride.FontWeightAdjustment(200)
    ) {
        Text(text = "text with increased scale and weight")
    }
}

Outros recursos

  • Testar apps no Android: a página de destino principal de testes do Android oferece uma visão mais ampla dos princípios básicos e das técnicas de teste.
  • Conceitos básicos de testes:saiba mais sobre os principais conceitos por trás de testes de apps Android.
  • Testes locais: é possível executar alguns testes localmente na sua própria estação de trabalho.
  • Testes instrumentados:é uma prática recomendada executar testes instrumentados. Ou seja, testes executados diretamente no dispositivo.
  • Integração contínua: a integração contínua permite integrar seus testes ao pipeline de implantação.
  • Teste diferentes tamanhos de tela:com alguns dispositivos disponíveis para os usuários, teste diferentes tamanhos de tela.
  • Espresso: embora seja destinado a interfaces baseadas em visualização, o conhecimento do Espresso ainda pode ser útil para alguns aspectos dos testes do Compose.