commonMain.com.arkivanov.decompose.extensions.compose.experimental.stack.ChildStack.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of extensions-compose-experimental Show documentation
Show all versions of extensions-compose-experimental Show documentation
Kotlin Multiplatform lifecycle-aware business logic components
package com.arkivanov.decompose.extensions.compose.experimental.stack
import androidx.compose.animation.AnimatedVisibilityScope
import androidx.compose.runtime.Composable
import androidx.compose.runtime.DisposableEffect
import androidx.compose.runtime.remember
import androidx.compose.runtime.saveable.SaveableStateHolder
import androidx.compose.runtime.saveable.rememberSaveableStateHolder
import androidx.compose.ui.Modifier
import com.arkivanov.decompose.Child
import com.arkivanov.decompose.ExperimentalDecomposeApi
import com.arkivanov.decompose.extensions.compose.experimental.stack.animation.LocalStackAnimationProvider
import com.arkivanov.decompose.extensions.compose.experimental.stack.animation.StackAnimation
import com.arkivanov.decompose.extensions.compose.experimental.stack.animation.emptyStackAnimation
import com.arkivanov.decompose.extensions.compose.subscribeAsState
import com.arkivanov.decompose.keyHashString
import com.arkivanov.decompose.router.stack.ChildStack
import com.arkivanov.decompose.value.Value
/**
* Displays the provided [stack] of child components, taking care of saving and restoring the UI state.
*
* @param stack a [ChildStack] to be displayed.
* @param modifier a [Modifier] to applied to a wrapping container.
* @param animation an optional [StackAnimation] for animating stack changes. If not provided (or `null`),
* then a default [StackAnimation] is obtained from [LocalStackAnimationProvider].
* If that is also `null`, then there is no animation.
* @param content a `Composable` function that displays the provided [Child][Child.Created] component.
* The receiver [AnimatedVisibilityScope] can be used for additional animations, such as
* [Shared Element Transitions](https://developer.android.com/develop/ui/compose/animation/shared-elements).
*/
@ExperimentalDecomposeApi
@Composable
fun ChildStack(
stack: ChildStack,
modifier: Modifier = Modifier,
animation: StackAnimation? = null,
content: @Composable AnimatedVisibilityScope.(child: Child.Created) -> Unit,
) {
val holder = rememberSaveableStateHolder()
holder.retainStates(stack.getKeys())
val animationProvider = LocalStackAnimationProvider.current
val anim = animation ?: remember(animationProvider, animationProvider::provide) ?: emptyStackAnimation()
anim(stack = stack, modifier = modifier) { child ->
holder.SaveableStateProvider(child.keyHashString()) {
content(child)
}
}
}
/**
* Displays the provided [stack] of child components, taking care of saving and restoring the UI state.
*
* @param stack an observable [ChildStack] to be displayed.
* @param modifier a [Modifier] to applied to a wrapping container.
* @param animation an optional [StackAnimation] for animating stack changes. If not provided (or `null`),
* then a default [StackAnimation] is obtained from [LocalStackAnimationProvider].
* If that is also `null`, then there is no animation.
* @param content a `Composable` function that displays the provided [Child][Child.Created] component.
* The receiver [AnimatedVisibilityScope] can be used for additional animations, such as
* [Shared Element Transitions](https://developer.android.com/develop/ui/compose/animation/shared-elements).
*/
@ExperimentalDecomposeApi
@Composable
fun ChildStack(
stack: Value>,
modifier: Modifier = Modifier,
animation: StackAnimation? = null,
content: @Composable AnimatedVisibilityScope.(child: Child.Created) -> Unit,
) {
val state = stack.subscribeAsState()
ChildStack(
stack = state.value,
modifier = modifier,
animation = animation,
content = content
)
}
private fun ChildStack<*, *>.getKeys(): Set =
items.mapTo(HashSet(), Child<*, *>::keyHashString)
@Composable
private fun SaveableStateHolder.retainStates(currentKeys: Set) {
val keys = remember(this) { Keys(currentKeys) }
DisposableEffect(this, currentKeys) {
keys.set.forEach {
if (it !in currentKeys) {
removeState(it)
}
}
keys.set = currentKeys
onDispose {}
}
}
private class Keys(
var set: Set
)
© 2015 - 2025 Weber Informatics LLC | Privacy Policy