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.
| |
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:
| |
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:
| |
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:
| |
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:
| |
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 animatesanimateFloatAsState+graphicsLayer— animate a float, apply in graphicsLayer
| |
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:
| |
Modifier Extensions
Prefer plain Modifier extension functions or Modifier.Node-based factories. Do not make modifier extensions @Composable:
| |
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:
| |
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:
| |
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:
| |
For decorative elements that convey no information, set contentDescription = null.
TextField
Text input state lives in the ViewModel. Every keystroke dispatches an Action:
| |
The ViewModel updates state (and optionally persists to SavedStateHandle) in response to the Action — see the android-presentation-mvi skill for the full pattern.