Android / KMP Architecture Skills for Claude Code[android-compose-ui]


name: android-compose-ui

description: | Compose UI patterns for Android/KMP - stability, recomposition, side effects, lazy lists, animations, previews, accessibility, modifier extensions, and design system composables. Use this skill whenever writing or reviewing composables, optimizing recomposition, adding animations, creating previews, writing custom modifiers, structuring a design system, or making any Compose UI decision beyond the MVI/ViewModel layer. Trigger on phrases like “composable”, “recomposition”, “LaunchedEffect”, “Modifier”, “LazyColumn”, “preview”, “animation”, “design system”, “stability”, “contentDescription”, “graphicsLayer”, “slot API”, or “Compose performance”.


Android / KMP Compose UI Patterns

Core Principle

The UI is dumb. Composables render state and forward user actions — nothing more. All state lives in the ViewModel. All logic lives in the ViewModel, domain, or data layer. Compose code should contain zero business logic, zero data transformation, and minimal side effects.


Stability & Recomposition

Strong skipping mode is enabled by default in modern Compose — no explicit opt-in needed.

Only annotate a state data class with @Stable when it contains fields the Compose compiler considers unstable (e.g., List, Map, Set, interfaces, or abstract types). If all fields are primitive types, String, or other stable types, no annotation is needed.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
// Needs @Stable — contains a List (unstable by default)
@Stable
data class NoteListState(
    val notes: List<NoteUi> = emptyList(),
    val isLoading: Boolean = false
)

// No annotation needed — all fields are stable
data class NoteDetailState(
    val title: String = "",
    val body: String = "",
    val isSaving: Boolean = false
)

State Ownership

All state lives in the ViewModel. Do not use remember or rememberSaveable for application state — that belongs in the ViewModel’s StateFlow and is surfaced via collectAsStateWithLifecycle().

The only exception is Compose-internal state that the framework requires you to hold in composition, such as LazyListState, ScrollState, or PagerState. For these, use remember* as needed:

1
2
3
4
5
6
7
// Acceptable — Compose-owned UI state
val lazyListState = rememberLazyListState()

// Reacting to Compose-owned state with derivedStateOf
val showScrollToTop by remember {
    derivedStateOf { lazyListState.firstVisibleItemIndex > 5 }
}

derivedStateOf should only be used in these rare scenarios where Compose-internal state drives a derived value. If the derivation can happen in the ViewModel, it should.

Always collect ViewModel state with lifecycle awareness:

1
val state by viewModel.state.collectAsStateWithLifecycle()

Side Effects

Side effects should be avoided when possible. If something can be handled by the ViewModel through an Action, do that instead of using a side effect in a composable.

When a side effect is truly necessary (e.g., interacting with Android lifecycle APIs that have no ViewModel equivalent), extract it into a dedicated composable to keep the Screen composable clean:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
// Extracted into its own composable
@Composable
fun ObserveLifecycle(onStart: () -> Unit, onStop: () -> Unit) {
    val lifecycleOwner = LocalLifecycleOwner.current
    DisposableEffect(lifecycleOwner) {
        val observer = LifecycleEventObserver { _, event ->
            when (event) {
                Lifecycle.Event.ON_START -> onStart()
                Lifecycle.Event.ON_STOP -> onStop()
                else -> Unit
            }
        }
        lifecycleOwner.lifecycle.addObserver(observer)
        onDispose { lifecycleOwner.lifecycle.removeObserver(observer) }
    }
}

LaunchedEffect is acceptable when genuinely needed, but question whether the work belongs in the ViewModel first. Do not use custom CompositionLocals.


Lazy Layouts

Add key to lazy list items when there is an obvious unique identifier available. Don’t force it if it’s unclear which property is unique:

1
2
3
4
5
6
7
8
LazyColumn {
    items(
        items = state.notes,
        key = { it.id }  // id is clearly unique
    ) { note ->
        NoteItem(note = note, onClick = { onAction(OnNoteClick(note.id)) })
    }
}

Animations

Avoid animations that cause recompositions. Prefer approaches that animate below the recomposition layer:

  • graphicsLayer — for alpha, scale, rotation, translation
  • Offset lambda — for position changes (offset { ... })
  • Canvas — for custom drawing that animates
  • animateFloatAsState + graphicsLayer — animate a float, apply in graphicsLayer
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
// Good — animates without recomposition
val alpha by animateFloatAsState(if (state.isVisible) 1f else 0f)
Box(
    modifier = Modifier.graphicsLayer { this.alpha = alpha }
)

// Bad — causes recomposition on every frame
Box(
    modifier = Modifier.alpha(animatedAlpha)
)

Deferred state reads: When a value drives an animation, pass it as a lambda rather than reading it directly. This defers the state read to the layout/draw phase and avoids recomposition:

1
2
3
4
5
6
7
// Good — deferred read
fun Modifier.animatedOffset(offsetProvider: () -> IntOffset) =
    offset { offsetProvider() }

// Bad — immediate read causes recomposition
fun Modifier.animatedOffset(offset: IntOffset) =
    offset(x = offset.x.dp, y = offset.y.dp)

Modifier Extensions

Prefer plain Modifier extension functions or Modifier.Node-based factories. Do not make modifier extensions @Composable:

1
2
3
4
5
6
7
8
// Good — plain extension
fun Modifier.shimmerEffect(): Modifier = composed {
    // shimmer implementation
}

// Better — Modifier factory (no composition needed)
fun Modifier.roundedBackground(color: Color, radius: Dp) =
    background(color, RoundedCornerShape(radius))

Design System & Slot APIs

The design system lives in :core:design-system and contains reusable Compose components, colors, theme, and typography.

Use slot APIs (passing @Composable lambdas) primarily for design system components that need flexible content areas:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
// Slot API — design system component
@Composable
fun AppCard(
    modifier: Modifier = Modifier,
    header: @Composable () -> Unit,
    content: @Composable () -> Unit
) {
    Card(modifier = modifier) {
        header()
        content()
    }
}

Feature-level composables should prefer typed parameters over slots for clarity.


Previews

Every Screen composable should have at least one meaningful @Preview that shows a realistic state:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
@Preview
@Composable
private fun NoteListScreenPreview() {
    AppTheme {
        NoteListScreen(
            state = NoteListState(
                notes = listOf(
                    NoteUi("1", "Meeting notes", "Mar 15"),
                    NoteUi("2", "Shopping list", "Mar 14")
                )
            ),
            onAction = {}
        )
    }
}

Wrap previews in the app theme so they reflect real appearance. Use realistic sample data, not empty states (unless previewing the empty state specifically).


Accessibility

Use meaningful contentDescription on all interactive or informational visual elements. Always use string resources to allow localization:

1
2
3
4
5
6
7
8
9
Icon(
    imageVector = Icons.Default.Delete,
    contentDescription = stringResource(R.string.cd_delete_note)
)

Image(
    painter = painterResource(R.drawable.profile),
    contentDescription = stringResource(R.string.cd_profile_picture)
)

For decorative elements that convey no information, set contentDescription = null.


TextField

Text input state lives in the ViewModel. Every keystroke dispatches an Action:

1
2
3
4
5
// In the Screen composable
TextField(
    value = state.title,
    onValueChange = { onAction(NoteEditorAction.OnTitleChange(it)) }
)

The ViewModel updates state (and optionally persists to SavedStateHandle) in response to the Action — see the android-presentation-mvi skill for the full pattern.

0%