This Day In History Jetpack Compose App - Welcome Screen
Objective:
Build our Welcome Screen and connect it to a mock WelcomeViewModel. At the end of this exercise, the Welcome Screen will look and behave like below.
Welcome Screen functionalities
The welcome screen will do the following:
- Display initial welcome message in English.
- Request translation of the welcome message into the device language via translation API.
- Transition into the Language Selection screen and display the translated version of "Choose Your Language".
Code
Tag Constants
Add the following constants to the ui.constants package. We will need them for unit tests.
package com.coroutines.thisdayinhistory.ui.constants
//add:
const val WELCOME_MESSAGE_TEXT_TAG = "welcomeMessageText"
Data Models
And data class TranslateRequestParams to the com.coroutines.data.models in the Models project.
package com.coroutines.data.models
data class TranslateRequestParams(
val language: String,
val welcomeText: String,
val languagePrompt: String
)
Welcome ViewModel (mock viewmodel for now)
Add IWelcomeViewModel interface to the com.coroutines.thisdayinhistory.ui.viewmodels in the app project.
package com.coroutines.thisdayinhistory.ui.viewmodels
import com.coroutines.data.models.TranslateRequestParams
import com.coroutines.thisdayinhistory.ui.state.WelcomeScreenUiState
import kotlinx.coroutines.flow.StateFlow
interface IWelcomeViewModel {
val uiScreenState: StateFlow
val prompt: String
fun setDefaultLanguageWelcomeMessage(welcomeMessage: String)
fun translate(translateRequestParams: TranslateRequestParams)
}
Welcome ViewModel (mock viewmodel for now)
Add WelcomeViewModelMock class to the com.coroutines.thisdayinhistory.ui.viewmodels in the app project.
package com.coroutines.thisdayinhistory.ui.viewmodels
import androidx.lifecycle.ViewModel
import androidx.lifecycle.viewModelScope
import com.coroutines.data.models.TranslateRequestParams
import com.coroutines.thisdayinhistory.ui.state.WelcomeScreenUiState
import kotlinx.coroutines.delay
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.launch
class WelcomeViewModelMock(
welcomeScreenUiState: WelcomeScreenUiState =
WelcomeScreenUiState.Initial(DEFAULT_WELCOME_MESSAGE)
): IWelcomeViewModel, ViewModel()
{
private var _prompt: String = DEFAULT_LANGUAGE_PROMPT
private val _screenState = MutableStateFlow(
value = welcomeScreenUiState
)
override val uiScreenState: StateFlow<WelcomeScreenUiState>
get() = _screenState
override val prompt by lazy {
_prompt
}
override fun setDefaultLanguageWelcomeMessage(welcomeMessage: String){
_screenState.value = WelcomeScreenUiState.Initial(welcomeMessage)
}
override fun translate(translateRequestParams: TranslateRequestParams){
viewModelScope.launch {
_screenState.value = WelcomeScreenUiState.WaitingForTranslation(translateRequestParams.welcomeText)
delay(2000)
_prompt = "Choose your language:"
_screenState.value = WelcomeScreenUiState.Success("this text is translated")
}
}
companion object {
private const val DEFAULT_WELCOME_MESSAGE = "History Cat"
private const val DEFAULT_LANGUAGE_PROMPT = "Please choose your language"
}
}
Welcome Graph
Update the Languages Screen code in the Welcome Graph to accept the prompt parameter and retrieve it from the backstack:
package com.coroutines.thisdayinhistory.graph
import androidx.compose.runtime.getValue
import androidx.lifecycle.compose.collectAsStateWithLifecycle
import androidx.navigation.NavController
import androidx.navigation.NavGraphBuilder
import androidx.navigation.NavType
import androidx.navigation.navigation
import androidx.navigation.compose.composable
import androidx.navigation.navArgument
import com.coroutines.thisdayinhistory.ui.screens.language.LanguageScreen
import com.coroutines.thisdayinhistory.ui.screens.welcome.WelcomeScreen
import com.coroutines.thisdayinhistory.ui.viewmodels.ISettingsViewModel
import com.coroutines.thisdayinhistory.ui.viewmodels.WelcomeViewModelMock
const val langPrompt = "prompt"
fun NavGraphBuilder.introGraph(navController: NavController, settingsViewModel: ISettingsViewModel) {
navigation(startDestination = IntroNavOption.WelcomeScreen.name, route = NavRoutes.IntroRoute.name) {
composable(IntroNavOption.WelcomeScreen.name){
val appConfigState by settingsViewModel.appConfigurationState.collectAsStateWithLifecycle()
WelcomeScreen(navController, appConfigState, WelcomeViewModelMock())
}
/*composable(IntroNavOption.LanguagesScreen.name){
LanguageScreen(navController = navController, viewModel = settingsViewModel)
}*/
composable(IntroNavOption.LanguagesScreen.name + "/{$langPrompt}",
arguments = listOf(navArgument(langPrompt) { type = NavType.StringType })){ backStackEntry ->
val prompt = backStackEntry.arguments?.getString(langPrompt) ?: "Please choose your language"
LanguageScreen(
navController= navController,
viewModel = settingsViewModel,
languagePrompt = prompt
)
}
}
}
MainContent
Update the MainContent function to have isOnboarded parameter set to false to trigger the Welcome Workflow.
@Composable
fun MainContent(settingsViewModel: ISettingsViewModel, appConfigState: AppConfigurationState,){
val navController = rememberNavController()
ThisDayInHistoryTheme(
viewModel = settingsViewModel
) {
val appThemeColor = MaterialTheme.colorScheme.background
Surface(
modifier = Modifier.background(appThemeColor)
) {
AppNavHost(
navController = navController,
settingsViewModel = settingsViewModel,
isOnboarded = false
)
}
}
}