From chrisbanes-skills
Guides Jetpack Compose side-effect API selection including LaunchedEffect, DisposableEffect, SideEffect, rememberCoroutineScope, rememberUpdatedState, and snapshotFlow with correct key and lifecycle patterns.
How this skill is triggered — by the user, by Claude, or both
Slash command
/chrisbanes-skills:compose-side-effectsThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
Composable bodies describe UI. They can be recomposed, skipped, or abandoned. Work that changes the outside world belongs in an effect API whose lifecycle matches the work.
Composable bodies describe UI. They can be recomposed, skipped, or abandoned. Work that changes the outside world belongs in an effect API whose lifecycle matches the work.
| Need | API |
|---|---|
| Publish Compose state to non-Compose code after every successful recomposition | SideEffect |
| Register/unregister a listener, callback, observer, or resource | DisposableEffect(keys...) |
| Run suspending, deferred, or keyed one-shot work | LaunchedEffect(keys...) |
| Launch suspending work from a user event callback | rememberCoroutineScope() |
| Convert Compose snapshot reads into a Flow inside a coroutine | snapshotFlow { ... } inside LaunchedEffect |
Keys define restart identity. When any key changes, the old effect is cancelled/disposed and a new one starts.
// ✅ Restart collection when userId changes
LaunchedEffect(userId) {
repository.events(userId).collect { event -> handle(event) }
}
// ❌ Unit hides a changing input; collection keeps using the first userId
LaunchedEffect(Unit) {
repository.events(userId).collect { event -> handle(event) }
}
Use stable, semantic keys:
userId, screenId, lifecycleOwner, focusRequester.state, viewModel) when only one property matters.For long-running effects that should not restart but need the latest callback or value, use rememberUpdatedState.
@Composable
fun Timeout(onTimeout: () -> Unit) {
val latestOnTimeout by rememberUpdatedState(onTimeout)
LaunchedEffect(Unit) {
delay(1_000)
latestOnTimeout()
}
}
Use this when the lifecycle is "start once" but the invoked lambda should stay fresh. Common cases:
onTimeout changes, but it should call the latest callback.onStart / onStop lambdas.Do not use rememberUpdatedState to avoid choosing proper keys. If the changed value should restart the work, make it a key instead:
// BAD: userId changes should restart the collection, not update a captured value.
val latestUserId by rememberUpdatedState(userId)
LaunchedEffect(Unit) {
repository.events(latestUserId).collect { event -> handle(event) }
}
// GOOD: the collection lifecycle follows userId.
LaunchedEffect(userId) {
repository.events(userId).collect { event -> handle(event) }
}
rememberUpdatedState values are stale inside remember {} blocksrememberUpdatedState returns a State object whose .value is updated on every recomposition. The "latest" behavior only helps when the State is read lazily — inside an effect body or a lambda that runs later — not when the value is captured eagerly.
Inside a remember {} block the producer lambda runs once. Reading the delegate there snapshots the current .value into the remembered object — future State updates never reach it:
val latestChannelId by rememberUpdatedState(channelId)
// ❌ BAD — channelId is read once when remember's lambda executes;
// the destination holds the initial value forever
val destination = remember {
Destination(channelId = latestChannelId)
}
// ✅ GOOD — skip rememberUpdatedState; key remember on the changing value
val destination = remember(channelId) {
Destination(channelId = channelId)
}
// ✅ ALSO GOOD — wrapping lambda defers the read to each invocation
val destination = remember {
Destination(channelId = { latestChannelId })
}
The same trap applies anywhere a rememberUpdatedState delegate is read eagerly rather than deferred behind a lambda or effect body: data classes constructed in remember, objects built once in DisposableEffect's setup block, or any expression evaluated at creation time.
When the captured value should trigger recreation of the remembered object, make it a remember key and skip rememberUpdatedState entirely. Reserve rememberUpdatedState for values that must stay fresh inside a long-lived scope (effect coroutine, event callback) without restarting that scope.
rememberUpdatedState also does not make render state "non-recomposing." If the UI needs to display a changing value, read normal State in composition or use the deferred-read patterns in compose-state-deferred-reads for frame-rate values.
Use LaunchedEffect for side-effect/event flows: snackbars, navigation events, analytics events, focus commands, or other streams where each emission triggers imperative work.
LaunchedEffect(events) {
events.collect { event ->
snackbarHostState.showSnackbar(event.message)
}
}
Do not collect render state imperatively just to mutate local state. For UI state, collect near the state holder and pass plain values into the UI composable—the state-holder vs UI split, collectAsStateWithLifecycle() / collectAsState(), and preview-friendly wiring are covered in compose-state-holder-ui-split. Do not duplicate that architecture here.
On Android, prefer lifecycle-aware collection where available; use collectAsState() on targets without lifecycle-aware APIs.
For Compose state reads, use snapshotFlow:
LaunchedEffect(listState) {
snapshotFlow { listState.firstVisibleItemIndex }
.distinctUntilChanged()
.collect { index -> analytics.visibleIndex(index) }
}
snapshotFlow { ... }.map { ... } without a terminal collect does nothing.
Use rememberCoroutineScope() when a click or gesture starts suspending work:
@Composable
fun SaveButton(snackbarHostState: SnackbarHostState) {
val scope = rememberCoroutineScope()
Button(
onClick = {
scope.launch {
snackbarHostState.showSnackbar("Saved")
}
},
) {
Text("Save")
}
}
Avoid "event flag" state just to trigger a LaunchedEffect. The click already is the event.
Use DisposableEffect for paired setup/teardown:
@Composable
fun ObserveLifecycle(owner: LifecycleOwner, observer: LifecycleObserver) {
DisposableEffect(owner, observer) {
owner.lifecycle.addObserver(observer)
onDispose {
owner.lifecycle.removeObserver(observer)
}
}
}
Every registration path should have a matching onDispose cleanup path.
| Mistake | Diagnosis | Fix |
|---|---|---|
| Network request directly in the composable body | Side work in composition | Usually move to a ViewModel/state holder; use LaunchedEffect only for UI-owned keyed work |
| Analytics property written from the composable body | Side work in composition | Use SideEffect when it should publish after every successful recomposition |
| Impression/event logged from the composable body | Side work in composition | Use LaunchedEffect(key) when it should run once for that key |
LaunchedEffect(Unit) captures changing id | Missing key | Key by id, or use rememberUpdatedState if it must not restart |
rememberUpdatedState(id) used so LaunchedEffect(Unit) keeps running after id changes | Hidden lifecycle bug | Key the effect by id |
| Long-lived effect invokes an old callback after recomposition | Stale capture | Wrap the callback with rememberUpdatedState and call the wrapper inside the effect |
rememberUpdatedState delegate read directly in remember {} (e.g. Destination(id = latestId)) | Value captured once, never refreshed | Make the value a remember key: remember(id) { Destination(id = id) } |
LaunchedEffect(state) { ... } restarts too often | Overly broad key | Key by the specific property |
LaunchedEffect(...) { nonSuspendSetter() } | Wrong effect type | Usually SideEffect; keep LaunchedEffect only for keyed one-shot/deferred work |
Listener added in LaunchedEffect with no cleanup | Missing disposal | Use DisposableEffect |
Launching from click by setting shouldShowSnackbar = true | Event flag anti-pattern | Use rememberCoroutineScope() in the click callback |
if (isFocused) { … } or focus read in composable body for side work | Side work during composition | LaunchedEffect(focused) { … } or snapshotFlow |
onSizeChanged { heightState = it.height } on measured composable | Layout → composition back-write if a sibling reads heightState in composition | Siblings must consume height in measure phase, not Modifier.height(state.dp) in composition |
Focus: Reading focus in the composable body to drive side work (preloading, analytics, toasts) runs that work during composition. Observe focus in an effect instead:
// ❌ BAD — side work runs during composition every time `focused` is true,
// including transient focus passes; `SideEffect` re-runs after every successful recomposition
@Composable
fun Preloader(interactionSource: MutableInteractionSource) {
val focused by interactionSource.collectIsFocusedAsState()
if (focused) {
preloadImages()
}
}
// ✅ GOOD — side work in a keyed effect
@Composable
fun Preloader(interactionSource: MutableInteractionSource) {
val focused by interactionSource.collectIsFocusedAsState()
LaunchedEffect(focused) {
if (focused) preloadImages()
}
}
Use snapshotFlow { … } inside LaunchedEffect when you need to sample multiple snapshot reads or debounce rapid changes without keying the effect on every derived value. For TV/D-pad focus navigation semantics, see compose-focus-navigation.
Measurement: onSizeChanged / onGloballyPositioned are valid callbacks, but they fire during the layout phase. Writing snapshot state there is only safe if no earlier phase reads it. If a sibling reads that state in composition, layout is back-writing into composition and the sibling will recompose every measure pass. Apply captured dimensions in Modifier.layout (see compose-modifier-and-layout-style §7 and compose-state-deferred-reads).
LaunchedEffect(Unit) in a function with changing parameters.rememberUpdatedState.rememberUpdatedState delegate read eagerly inside a remember {} block or object constructor — the value is captured once and never refreshes.npx claudepluginhub chrisbanes/skills --plugin chrisbanes-skillsTeaches correct state patterns for Jetpack Compose: State-backed vars, remember/mutableStateOf, mutableStateListOf/mutableStateMapOf, and @ReadOnlyComposable. Debugs disappearing state.
Guides building native Android UIs with Jetpack Compose, including state management via remember/mutableStateOf, state hoisting, and ViewModel integration.
Guides building production-quality Android UIs with Jetpack Compose, including state management (ViewModel/StateFlow), type-safe navigation, and performance optimization.