Jetpack Compose App - Date Picker

Objective:

Previously, we added the primary App Bar to make the app display a hamburger menu on the left (to display the navigation drawer) and a calendar icon on the right. We also connected the hamburger menu to an action to open the navigation drawer, but the calendar icon was just a placeholder - until now. We will add a Date Picker @Composable and connect it to the calendar icon on the app bar. We will also make sure it works in both Light and Dark modes, and is properly internationalized (we added support for internationalization earlier) by switching the in-app language from English to another language - for example, Italian.

Dependencies

Add the following dependency for kotlinx-datetime:

  • org.jetbrains.kotlinx:kotlinx-datetime:0.6.0

Components built

Add or modify the following components:

  • HistoryDatePicker.kt: Data picker widget selectable via calendar icon on the top App Bar.
  • AppBar.kt: Modify the AppBar to include HistoryDatePicker

Code

Date Picker

Add HistoryDatePicker.kt to the com.coroutines.thisdayinhistory.ui.components package in the app module:


package com.coroutines.thisdayinhistory.ui.components


import androidx.annotation.StringRes
import androidx.compose.material3.Button
import androidx.compose.material3.DatePicker
import androidx.compose.material3.DatePickerDialog
import androidx.compose.material3.DatePickerState
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.Text
import androidx.compose.material3.DatePickerDefaults
import androidx.compose.material3.MaterialTheme
import androidx.compose.runtime.Composable
import androidx.compose.runtime.remember
import androidx.compose.ui.res.stringResource
import com.coroutines.thisdayinhistory.components.ADJUSTED_COLOR_LIGHTER_FACTOR
import com.coroutines.thisdayinhistory.ui.utils.lighter
import com.coroutines.thisdayinhistory.ui.viewmodels.IHistoryViewModel
import kotlinx.datetime.toInstant
import kotlinx.datetime.toKotlinLocalDateTime
import kotlinx.datetime.toKotlinTimeZone
import java.time.Instant
import java.time.LocalDateTime
import java.time.ZoneId
import java.time.ZoneOffset


@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun HistoryDatePickerDialog(
    viewModel: IHistoryViewModel,
    @StringRes resIdCancel: Int,
    onDateSelected: (LocalDateTime?) -> Unit,
    onDismiss: () -> Unit
) {


    val time = kotlinx.datetime.Clock.System.now().toEpochMilliseconds()

    val datePickerState = remember { DatePickerState(
        locale = androidx.compose.material3.CalendarLocale.FRANCE,
        time)
    }

    val selectedDate = datePickerState.selectedDateMillis?.let {
        Instant.ofEpochMilli(it).atOffset(ZoneOffset.UTC)
    }

    DatePickerDialog(
        colors = DatePickerDefaults.colors().copy(
            containerColor = MaterialTheme.colorScheme.background.lighter(
                ADJUSTED_COLOR_LIGHTER_FACTOR),
        ),
        onDismissRequest = { onDismiss() },
        confirmButton = {
            Button(onClick = {
                onDateSelected(selectedDate?.toLocalDateTime())
                onDismiss()
            }

            ) {
                Text(text = "OK")
            }
        },
        dismissButton = {
            Button(onClick = {
                onDismiss()
            }) {
                Text(text = stringResource(id = resIdCancel))
            }
        }
    ) {
        DatePicker(
            state = datePickerState
        )
    }
}

AppBar

Modify AppBar.kt in the com.coroutines.thisdayinhistory.ui.appbar package in the app module to include HistoryDatePicker:



package com.coroutines.thisdayinhistory.ui.appbar

import androidx.compose.material3.LocalContentColor
import com.coroutines.thisdayinhistory.R
import com.coroutines.thisdayinhistory.ui.viewmodels.IHistoryViewModel
import androidx.annotation.StringRes
import androidx.compose.foundation.background
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.filled.DateRange
import androidx.compose.material.icons.rounded.Menu
import androidx.compose.material3.DrawerState
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.Icon
import androidx.compose.material3.IconButton
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.TopAppBar
import androidx.compose.material3.TopAppBarDefaults
import androidx.compose.material3.TopAppBarScrollBehavior
import androidx.compose.runtime.Composable
import androidx.compose.runtime.CompositionLocalProvider
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberCoroutineScope
import androidx.compose.runtime.setValue
import androidx.compose.ui.Modifier
import androidx.compose.ui.res.stringResource
import com.coroutines.thisdayinhistory.ui.components.HistoryDatePickerDialog
import kotlinx.coroutines.launch

@OptIn(ExperimentalMaterial3Api::class)
@Suppress("LongParameterList")
@Composable
fun AppBar(
    drawerState: DrawerState? = null,
    navigationIcon: (@Composable () -> Unit)? = null,
    historyViewModel: IHistoryViewModel,
    @StringRes title: Int? = null,
    @StringRes cancelButtonText: Int,
    scrollBehavior: TopAppBarScrollBehavior = TopAppBarDefaults.enterAlwaysScrollBehavior(),
) {
    TopAppBar(
        modifier = Modifier.background(MaterialTheme.colorScheme.background),
        colors = TopAppBarDefaults.topAppBarColors(
            containerColor = MaterialTheme.colorScheme.background
        ),
        title = {}/* {
            title?.let {
                Text(
                    text = stringResource(R.string.title),
                    style = MaterialTheme.typography.titleMedium
                )
            }
        }*/,
        actions = {
            /*appBarActions?.let {
                for (appBarAction in it) {
                    AppBarAction(appBarAction)
                }
            }*/

            CompositionLocalProvider(LocalContentColor provides MaterialTheme.colorScheme.onSurface) {
                var showDatePicker by remember {
                    mutableStateOf(false)
                }

                if (showDatePicker) {
                    HistoryDatePickerDialog(
                        historyViewModel,
                        resIdCancel = cancelButtonText,
                        onDateSelected = {
                            historyViewModel.onDateChanged(it!!)
                        },
                        onDismiss = { showDatePicker = false }
                    )
                }

                IconButton(
                    onClick = {
                        showDatePicker = true
                    },
                    ) {
                    Icon(
                        imageVector = Icons.Filled.DateRange,
                        contentDescription = stringResource(R.string.appbar_calendar)
                    )
                }
            }
        },
        navigationIcon = {
            if (drawerState != null && navigationIcon == null){
                DrawerIcon(drawerState = drawerState)
            } else {
                navigationIcon?.invoke()
            }
        },
        scrollBehavior = scrollBehavior
    )
}


@Composable
private fun DrawerIcon(drawerState: DrawerState) {
    val coroutineScope = rememberCoroutineScope()
    IconButton(onClick = {
        coroutineScope.launch {
            drawerState.open()
        }
    }) {
        Icon(
            Icons.Rounded.Menu,
            tint = MaterialTheme.colorScheme.onBackground,
            contentDescription = "hello"
        )
    }
}