This Day In History Jetpack Compose App - Theme Screen

Objective:

Build our Theme Screen and connect it to a mock SettingsViewModel. At the end of this exercise, the Theme Screen will look like below. It will require more work to style it correctly, but that will be done later. We will take care of the functionality first.

Auto Mode: Light Mode: Dark Mode:

Theme Screen functionalities

The Theme screen will allow the user to do the following:

  • Scroll thru available themes: Auto, Light, or Dark.
  • Pick the current theme.

Code

Modifiers

Add a new package, modifiers, under the ui package. Add function Modifier.pagerOffsetAnimation


package com.coroutines.thisdayinhistory.ui.modifiers

@OptIn(ExperimentalFoundationApi::class)
@Composable
fun Modifier.pagerOffsetAnimation(
    pagerState: PagerState,
    page: Int,
): Modifier {
    val columnModifier = this.then(Modifier.graphicsLayer {
        val pageOffset =
            (
                    (pagerState.currentPage - page) + pagerState
                        .currentPageOffsetFraction
                    ).absoluteValue
        lerp(
            start = 0.85f,
            stop = 1f,
            fraction = 1f - pageOffset.coerceIn(0f, 1f)
        ).also { scale ->
            scaleX = scale
            scaleY = scale
        }
        alpha = lerp(
            start = 0.5f,
            stop = 1f,
            fraction = 1f - pageOffset.coerceIn(0f, 1f)
        )
    })
    return columnModifier
}

Theme Composable

Add Theme composable to the ui.screens.uitheme package.


package com.coroutines.thisdayinhistory.ui.screens.uitheme   
//imports


@OptIn(ExperimentalFoundationApi::class)
@Composable
@Suppress("LongMethod", "LongParameterList")
fun ThemeScreen(
    modifier: Modifier = Modifier,
    viewModel: ISettingsViewModel
) {
    Column(
        modifier = modifier
            .fillMaxSize()
            .background(MaterialTheme.colorScheme.background),
        horizontalAlignment = Alignment.CenterHorizontally

    ) {
        val themeState by viewModel.appConfigurationState.collectAsState()

        val pagerState = rememberPagerState(pageCount = { 3 })
        var tintColor = Color.Unspecified

        LaunchedEffect(key1 = true) {
            when (themeState.appTheme) {
                ThisDayInHistoryThemeEnum.Auto -> pagerState.scrollToPage(page = 0)
                ThisDayInHistoryThemeEnum.Light -> pagerState.scrollToPage(page = 1)
                ThisDayInHistoryThemeEnum.Dark -> pagerState.scrollToPage(page = 2)
            }
        }
        Box(modifier = Modifier.fillMaxSize()) {
                HorizontalPager(state = pagerState) { page ->
                    LaunchedEffect(pagerState) {
                        // Collect from the pager state a snapshotFlow reading the currentPage
                        snapshotFlow { pagerState.currentPage }.collect { item ->
                            when (item) {
                                0 -> {
                                    viewModel.setAppTheme(ThisDayInHistoryThemeEnum.Auto)
                                    tintColor = Color.White
                                }

                                1 -> {
                                    viewModel.setAppTheme(ThisDayInHistoryThemeEnum.Light)
                                    tintColor = Color.White
                                }

                                2 -> {
                                    viewModel.setAppTheme(ThisDayInHistoryThemeEnum.Dark)
                                    tintColor = Color.White
                                }
                            }
                        }
                    }
                    Column(
                        Modifier
                            .fillMaxSize()
                            .background(MaterialTheme.colorScheme.background)
                            .padding(0.dp, 50.dp),
                        horizontalAlignment = Alignment.CenterHorizontally,
                        verticalArrangement = Arrangement.Center

                    ) {
                        // Our page content
                        when (page) {
                            0 -> {
                                SetUiTheme(
                                    viewModel,
                                    R.drawable.sun_68,
                                    ThisDayInHistoryThemeEnum.Auto,
                                    tintColor,
                                    "Auto - General Phone Theme",
                                    "Light Theme"
                                )
                            }

                            1 -> {
                                SetUiTheme(
                                    viewModel,
                                    R.drawable.sun_68,
                                    ThisDayInHistoryThemeEnum.Light,
                                    tintColor,
                                    "Light",
                                    "Light Theme"
                                )

                            }

                            2 -> {
                                /*SetUiTheme(viewModel,
                            R.drawable.moon,
                            HistoryCatThemeEnum.Dark,
                            tintColor,
                            "Dark",
                            "Dark Theme")*/
                                Column(horizontalAlignment = Alignment.CenterHorizontally) {
                                    Icon(
                                        painter = painterResource(id = R.drawable.moon),
                                        "Dark Theme",
                                        modifier = Modifier
                                            .size(150.dp)
                                            .padding(30.dp, 35.dp)
                                            .clickable {
                                                viewModel.setAppTheme(ThisDayInHistoryThemeEnum.Dark)
                                            }
                                    )
                                    Text("Dark", textAlign = TextAlign.Center)
                                }
                            }
                        }
                    }
                }
            Row(
                Modifier
                    .wrapContentHeight()
                    .fillMaxWidth()
                    .align(Alignment.BottomCenter)
                    .padding(bottom = 18.dp),
                horizontalArrangement = Arrangement.Center,
            ) {
                repeat(pagerState.pageCount) { iteration ->
                    val color =
                        if (pagerState.currentPage == iteration) Color.DarkGray else Color.LightGray
                    Box(
                        modifier = Modifier
                            .padding(2.dp)
                            .clip(CircleShape)
                            .background(color)
                            .size(6.dp)
                    )
                }
            }
        }
    }
}

@Composable
@Suppress("LongParameterList")
private fun SetUiTheme(
    viewModel: ISettingsViewModel,
    @DrawableRes painterResourceId: Int,
    historyCatThemeEnum: ThisDayInHistoryThemeEnum,
    @ColorRes tintColor: Color,
    text: String,
    iconContentDescription: String,
) {
    Column(horizontalAlignment = Alignment.CenterHorizontally) {
        Icon(
            painter = painterResource(id = painterResourceId),
            contentDescription = iconContentDescription,
            modifier = Modifier
                .size(150.dp)
                .padding(30.dp, 30.dp)
                .clickable {
                    viewModel.setAppTheme(historyCatThemeEnum)
                },
            tint = tintColor
        )
        Text(text)
    }
}

@Preview
@Composable
fun PreviewThemeScreen(@PreviewParameter(ThisDayInHistoryThemeEnumProvider::class) historyCatThemeEnum: ThisDayInHistoryThemeEnum){

    val settingsViewModel = SettingsViewModelMock(historyCatThemeEnum)

    ThisDayInHistoryTheme(
        settingsViewModel
    ) {
        val appThemeColor = MaterialTheme.colorScheme.background
        Surface(
            modifier = Modifier.background(appThemeColor)
        ) {
            ThemeScreen(viewModel = settingsViewModel)
        }
    }
}