Jetpack Compose App - Bottom Sheet
Objective:
Previously, we added a Date Picker
to let users switch dates. Today, we will extend the main screen with a
Bottom Sheet
. We will use to display an expanded image when the user clicks on the thumbnail.
Dependencies
Add the following dependency for material (the original). We use material3 throughout the app and this will be the only exception to the rule, as
ModalBottomSheetLayout
has not been ported to material3 yet.
- androidx.compose.material:material
Components built
Add or modify the following components:
- BottomSheetContent.kt: Content to display within the modal bottom sheet.
- HistoryEventList.kt: Modify the HistoryEventList to include ModalBottomSheetLayout
Code
Bottom Sheet Content
Add BottomSheetContent.kt to the com.coroutines.thisdayinhistory.ui.screens.main package in the app module:
@file:Suppress("MagicNumber")
package com.coroutines.thisdayinhistory.ui.screens.main
import androidx.compose.foundation.isSystemInDarkTheme
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.WindowInsets
import androidx.compose.foundation.layout.WindowInsetsSides
import androidx.compose.foundation.layout.fillMaxHeight
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.navigationBars
import androidx.compose.foundation.layout.only
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.layout.windowInsetsPadding
import androidx.compose.material3.Surface
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.text.style.TextAlign
import androidx.compose.ui.unit.Dp
import androidx.compose.ui.unit.dp
import androidx.compose.ui.unit.sp
import com.coroutines.thisdayinhistory.ui.components.DominantColorState
import com.coroutines.thisdayinhistory.ui.components.DynamicThemePrimaryColorsFromImage
import com.coroutines.thisdayinhistory.ui.components.ExpandedImage
import com.coroutines.thisdayinhistory.ui.components.verticalGradientScrim
import com.coroutines.thisdayinhistory.ui.viewmodels.IHistoryViewModel
@Composable
fun BottomSheetContent(viewModel: IHistoryViewModel, readyState: Boolean, dominantColorState: DominantColorState) {
fun getPadding(): Dp {
val originalImage = viewModel.selectedItem.originalImage
return if (readyState) {
if (originalImage?.height!! > originalImage.width) 0.dp else 25.dp
} else {
0.dp
}
}
val sel = viewModel.selectedItem.description != "No Events"
DynamicThemePrimaryColorsFromImage(dominantColorState) {
Surface(
modifier = Modifier
.fillMaxHeight(getSheetContentSize(viewModel, sel))
.verticalGradientScrim(
color = dominantColorState.color.copy(alpha = if (isSystemInDarkTheme()) 0.89f else 0.99f),
startYPercentage = 1f,
endYPercentage = 0f
)
) {
Column(
modifier = Modifier
.fillMaxSize()
.windowInsetsPadding(
WindowInsets.navigationBars.only(WindowInsetsSides.Bottom)
)
.verticalGradientScrim(
color = dominantColorState.color.copy(alpha = if (isSystemInDarkTheme()) 0.89f else 0.99f),
startYPercentage = 1f,
endYPercentage = 0f
)
.padding(getPadding()),
horizontalAlignment = Alignment.CenterHorizontally
) {
if (readyState) {
Text(
text = viewModel.selectedItem.shortTitle!!.replace("_", " "),
fontSize = 20.sp,
modifier = Modifier
.padding(10.dp)
.padding(bottom = 10.dp),
color = dominantColorState.onColor,
textAlign = TextAlign.Center
)
ExpandedImage(viewModel.selectedItem)
}
}
}
}
}
fun getSheetContentSize(viewModel: IHistoryViewModel, readyState: Boolean): Float {
val originalImage = viewModel.selectedItem.originalImage
return if (readyState) {
if (originalImage?.height!! > originalImage.width) 0.9f else 0.5f
} else {
0.9f
}
}
HistoryEventList
Modify HistoryEventList.kt in the com.coroutines.thisdayinhistory.ui.screens.main package in the app module to display ModalBottomSheetLayout:
package com.coroutines.thisdayinhistory.ui.screens.main
import android.annotation.SuppressLint
import androidx.compose.animation.AnimatedContent
import androidx.compose.animation.AnimatedVisibility
import androidx.compose.animation.core.tween
import androidx.compose.animation.fadeIn
import androidx.compose.animation.fadeOut
import androidx.compose.animation.slideInHorizontally
import androidx.compose.animation.slideOutHorizontally
import androidx.compose.animation.togetherWith
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.PaddingValues
import androidx.compose.foundation.layout.Spacer
import androidx.compose.foundation.layout.height
import androidx.compose.foundation.lazy.LazyColumn
import androidx.compose.foundation.lazy.items
import androidx.compose.foundation.lazy.rememberLazyListState
import androidx.compose.foundation.shape.RoundedCornerShape
import androidx.compose.material.ExperimentalMaterialApi
import androidx.compose.material.ModalBottomSheetLayout
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.windowsizeclass.ExperimentalMaterial3WindowSizeClassApi
import androidx.compose.material3.windowsizeclass.WindowSizeClass
import androidx.compose.runtime.Composable
import androidx.compose.runtime.DisposableEffect
import androidx.compose.runtime.derivedStateOf
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.graphics.Color
import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.tooling.preview.PreviewParameterProvider
import androidx.compose.ui.unit.DpSize
import androidx.compose.ui.unit.dp
import androidx.lifecycle.compose.collectAsStateWithLifecycle
import androidx.navigation.NavController
import com.coroutines.data.models.HistoricalEvent
import com.coroutines.thisdayinhistory.components.NAV_ARGUMENT_DOMINANT_COLOR
import com.coroutines.thisdayinhistory.components.NAV_ARGUMENT_HISTORY_EVENT
import com.coroutines.thisdayinhistory.graph.MainNavOption
import com.coroutines.thisdayinhistory.ui.components.DominantColorState
import com.coroutines.thisdayinhistory.ui.components.MinContrastOfPrimaryVsSurface
import com.coroutines.thisdayinhistory.ui.components.ScrollToTopButton
import com.coroutines.thisdayinhistory.ui.components.contrastAgainst
import com.coroutines.thisdayinhistory.ui.components.rememberDominantColorState
import com.coroutines.thisdayinhistory.ui.theme.BabyPowder
import com.coroutines.thisdayinhistory.ui.viewmodels.HistoryViewModelMock
import com.coroutines.thisdayinhistory.ui.viewmodels.IHistoryViewModel
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch
//noinspection UsingMaterialAndMaterial3Libraries
import androidx.compose.material.ModalBottomSheetValue
@OptIn(ExperimentalMaterialApi::class)
@SuppressLint("UnusedContentLambdaTargetStateParameter")
@Suppress("LongMethod")
@Composable
fun HistoryEventList(
viewModel: IHistoryViewModel,
windowSizeClass: WindowSizeClass,
navController: NavController,
onItemImageStateChanged: (readyState: Boolean) -> Unit
) {
val context = LocalContext.current
val uiState by viewModel.uiState.collectAsStateWithLifecycle()
val selectedCategory = uiState.selectedCategory
val previousCategory = uiState.previousCategory
val listState = rememberLazyListState()
val coroutineScope = rememberCoroutineScope()
val data = viewModel.historyData
val backgroundColor = MaterialTheme.colorScheme.background
val dominantColorState = rememberDominantColorState { color ->
color.contrastAgainst(backgroundColor) >= MinContrastOfPrimaryVsSurface
}
val skipHalfExpanded by remember { mutableStateOf(true) }
val modalBottomSheetState = androidx.compose.material.rememberModalBottomSheetState(
initialValue = ModalBottomSheetValue.Hidden,
skipHalfExpanded = skipHalfExpanded
)
var showModalBottomSheet by remember { mutableStateOf(false) }
if (modalBottomSheetState.currentValue != ModalBottomSheetValue.Hidden) {
DisposableEffect(Unit) {
onDispose {
showModalBottomSheet = false
onItemImageStateChanged (false)
}
}
}
ModalBottomSheetLayout(
sheetState = modalBottomSheetState,
sheetShape = RoundedCornerShape(topEnd = 35.dp, topStart = 35.dp),
sheetBackgroundColor = Color.Transparent,
scrimColor = Color.Transparent,
sheetElevation = 0.dp,
sheetContent = {
BottomSheetContent(viewModel, showModalBottomSheet, dominantColorState)
}
) {
AnimatedContent(
targetState = Pair<String, String>(
selectedCategory,
previousCategory
),
transitionSpec = {
if (selectedCategory != previousCategory) {
(slideInHorizontally(
animationSpec = tween(
durationMillis = 200
)
)
{ width -> width }
+ fadeIn()).togetherWith(slideOutHorizontally
{ width -> -width } + fadeOut())
} else {
fadeIn(animationSpec = tween(200)) togetherWith
fadeOut(animationSpec = tween(200))
}
}, label = ""
) {
LazyColumn(
state = listState,
contentPadding = PaddingValues(bottom = 65.dp),
verticalArrangement = Arrangement.Top
) {
item { Spacer(Modifier.height(8.dp)) }
items(data) { item: HistoricalEvent ->
HistoryListItem(
historyEvent = item,
windowSizeClass = windowSizeClass,
onClick = { selectedEvent ->
viewModel.selectedItem = selectedEvent
navController.currentBackStackEntry?.savedStateHandle?.set(
NAV_ARGUMENT_HISTORY_EVENT, selectedEvent
)
navigateToDetail(
backgroundColor,
coroutineScope,
viewModel,
dominantColorState,
navController
)
},
onImageClick = { selectedEvent ->
viewModel.selectedItem = selectedEvent
coroutineScope.launch {
if (viewModel.selectedItem.imageUrl.isNotEmpty()) {
dominantColorState.updateColorsFromImageUrl(viewModel.selectedItem.imageUrl)
} else {
dominantColorState.reset()
}
showModalBottomSheet = true
onItemImageStateChanged(true)
modalBottomSheetState.show()
}
},
onShare = {
}
)
}
item { Spacer(Modifier.height(20.dp)) }
}
val showButton by remember {
derivedStateOf {
listState.firstVisibleItemIndex > 0
}
}
viewModel.isScrolled.value = showButton
AnimatedVisibility(
visible = showButton,
enter = fadeIn(),
exit = fadeOut()
) {
ScrollToTopButton(onClick = {
coroutineScope.launch {
listState.scrollToItem(0)
}
})
}
}
}
}
private fun navigateToDetail(
backgroundColor: Color,
coroutineScope: CoroutineScope,
viewModel: IHistoryViewModel,
dominantColorState: DominantColorState,
navController: NavController
) {
if (backgroundColor != BabyPowder) {
coroutineScope.launch {
val job = launch {
if (viewModel.selectedItem.imageUrl.isNotEmpty()) {
dominantColorState.updateColorsFromImageUrl(viewModel.selectedItem.imageUrl)
} else {
dominantColorState.reset()
}
}
job.join()
navController.currentBackStackEntry?.savedStateHandle?.set(
NAV_ARGUMENT_DOMINANT_COLOR,
dominantColorState.color.value.toString()
)
navController.navigate(MainNavOption.DetailScreen.name) {
popUpTo(MainNavOption.HistoryScreen.name)
}
}
} else {
navController.navigate(MainNavOption.DetailScreen.name) {
popUpTo(MainNavOption.HistoryScreen.name)
}
}
}