This Day In History Jetpack Compose App - Languages Screen UI Tests

Objective:

Add android UI tests for the Welcome Screen. At the end of this exercise, all individual tests within the Welcome Screen scenario should succeed when executed againt a phone, while 2 of them should fail when executed against a tablet. Yes, our design flaw will be revealed by running android unit test. We will address it at a later stage.

Components built

We will add/modify the following components:

  • testExtensions.kt: Add more helper methods
  • LanguageSelectionTest.kt: Class for testing individual language selection composable
  • LanguageScreenTest.kt: Class for testing the Language Screen

Code

testExtensions.kt

Modify testExtensions.kt in the com.coroutines.thisdayinhistory.helpers package in the androidTest source set:


// add:    
fun SemanticsNodeInteractionCollection.assertAreDisplayed(): SemanticsNodeInteractionCollection {
    fetchSemanticsNodes().forEachIndexed { index, _ ->
        get(index).assertIsDisplayed()
    }
    return this
}

Language Selection Test

Add LanguageSelectionTest.kt to the com.coroutines.thisdayinhistory.screens.language package in the androidTest sourceset:


package com.coroutines.thisdayinhistory.screens.language

import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent
import androidx.compose.ui.test.assertCountEquals
import androidx.compose.ui.test.assertHasClickAction
import androidx.compose.ui.test.assertIsDisplayed
import androidx.compose.ui.test.junit4.createEmptyComposeRule
import androidx.compose.ui.test.onAllNodesWithTag
import androidx.compose.ui.test.onNodeWithText
import androidx.compose.ui.test.performClick
import androidx.navigation.compose.rememberNavController
import androidx.test.core.app.ActivityScenario
import assertAreDisplayed
import com.coroutines.data.models.LangEnum
import com.coroutines.thisdayinhistory.ui.constants.LANGUAGE_SELECTION_TEXT_TAG
import com.coroutines.thisdayinhistory.ui.screens.language.LanguageScreen
import com.coroutines.thisdayinhistory.ui.theme.ThisDayInHistoryTheme
import com.coroutines.thisdayinhistory.ui.viewmodels.SettingsViewModelMock
import org.junit.After
import org.junit.Before
import org.junit.Rule
import org.junit.Test

class LanguageScreenTest {

    @get:Rule
    val composeTestRule = createEmptyComposeRule()
    private var wasCalled = false
    private lateinit var scenario: ActivityScenario<ComponentActivity>

    @Before
    fun setup() {
        scenario = ActivityScenario.launch(ComponentActivity::class.java)
    }

    @After
    fun tearDown() {
        scenario.close()
    }

    private fun initComposable() {
        scenario.onActivity { activity ->
            val settingsViewModel = SettingsViewModelMock()
            activity.setContent {
                val navController = rememberNavController()
                ThisDayInHistoryTheme(viewModel = settingsViewModel ) {
                    LanguageScreen(
                        navController = navController,
                        viewModel = settingsViewModel
                    )
                }
            }
        }
    }

    @Test
    fun allLanguagesCountTest(){
        initComposable()
        val languageCount = LangEnum.entries.toTypedArray().count()
        composeTestRule
            .onAllNodesWithTag(LANGUAGE_SELECTION_TEXT_TAG)
            .assertCountEquals(languageCount)
    }

    @Test
    fun allLanguagesDisplayedTest(){
        initComposable()
        composeTestRule
            .onAllNodesWithTag(LANGUAGE_SELECTION_TEXT_TAG)
            .assertAreDisplayed()

    }
    @Test
    fun languageNativeNameIsDisplayedTest() {
        initComposable()
        for (entry in LangEnum.entries) {
            composeTestRule.onNodeWithText(entry.langNativeName).assertIsDisplayed()
        }

    }

    @Test
    fun languageClickableTest() {
        initComposable()
        for (entry in LangEnum.entries) {
            composeTestRule.onNodeWithText(entry.langNativeName).assertHasClickAction()
        }
    }
}
    

Language Screen Test

Add LanguageSreenTest.kt to the com.coroutines.thisdayinhistory.screens.language package in the androidTest sourceset:


package com.coroutines.thisdayinhistory.screens.language

import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent
import androidx.compose.ui.test.assertCountEquals
import androidx.compose.ui.test.assertHasClickAction
import androidx.compose.ui.test.assertIsDisplayed
import androidx.compose.ui.test.junit4.createEmptyComposeRule
import androidx.compose.ui.test.onAllNodesWithTag
import androidx.compose.ui.test.onNodeWithText
import androidx.compose.ui.test.performClick
import androidx.navigation.compose.rememberNavController
import androidx.test.core.app.ActivityScenario
import assertAreDisplayed
import com.coroutines.data.models.LangEnum
import com.coroutines.thisdayinhistory.ui.constants.LANGUAGE_SELECTION_TEXT_TAG
import com.coroutines.thisdayinhistory.ui.screens.language.LanguageScreen
import com.coroutines.thisdayinhistory.ui.theme.ThisDayInHistoryTheme
import com.coroutines.thisdayinhistory.ui.viewmodels.SettingsViewModelMock
import org.junit.After
import org.junit.Before
import org.junit.Rule
import org.junit.Test

class LanguageScreenTest {

    @get:Rule
    val composeTestRule = createEmptyComposeRule()
    private var wasCalled = false
    private lateinit var scenario: ActivityScenario<ComponentActivity>

    @Before
    fun setup() {
        scenario = ActivityScenario.launch(ComponentActivity::class.java)
    }

    @After
    fun tearDown() {
        scenario.close()
    }

    private fun initComposable() {
        scenario.onActivity { activity ->
            val settingsViewModel = SettingsViewModelMock()
            activity.setContent {
                val navController = rememberNavController()
                ThisDayInHistoryTheme(viewModel = settingsViewModel ) {
                    LanguageScreen(
                        navController = navController,
                        viewModel = settingsViewModel
                    )
                }
            }
        }
    }

    @Test
    fun allLanguagesCountTest(){
        initComposable()
        val languageCount = LangEnum.entries.toTypedArray().count()
        composeTestRule
            .onAllNodesWithTag(LANGUAGE_SELECTION_TEXT_TAG)
            .assertCountEquals(languageCount)
    }

    @Test
    fun allLanguagesDisplayedTest(){
        initComposable()
        composeTestRule
            .onAllNodesWithTag(LANGUAGE_SELECTION_TEXT_TAG)
            .assertAreDisplayed()

    }
    @Test
    fun languageNativeNameIsDisplayedTest() {
        initComposable()
        for (entry in LangEnum.entries) {
            composeTestRule.onNodeWithText(entry.langNativeName).assertIsDisplayed()
        }

    }

    @Test
    fun languageClickableTest() {
        initComposable()
        for (entry in LangEnum.entries) {
            composeTestRule.onNodeWithText(entry.langNativeName).assertHasClickAction()
        }
    }


}