Jetpack Compose & Compose Multiplatform
This skill covers the full Compose app development lifecycle — from architecture and state management through UI, networking, persistence, performance, accessibility, cross-platform sharing, build configuration, and distribution. Jetpack Compose and Compose Multiplatform share the same core APIs and mental model. Not all Jetpack libraries work in commonMain — many remain Android-only. A subset of AndroidX libraries now publish multiplatform artifacts (e.g., lifecycle-viewmodel, lifecycle-runtime-compose, datastore-preferences), but availability and API surface vary by version. Before adding any Jetpack/AndroidX dependency to commonMain, verify the artifact is published for all required targets by checking Maven Central or the library's official documentation. CMP uses expect/actual or interfaces for platform-specific code. MVI (Model-View-Intent) is the recommended architecture, but the skill adapts to existing project conventions.
Existing Project Policy
Do not force migration. If a project already follows MVI with its own conventions (different base class, different naming, different file layout), respect that. Adapt to the project's existing patterns. The architecture pattern — unidirectional data flow with Event, State, and Effect — is what matters, not a specific base class or framework. Only suggest structural changes when the user asks for them or when the existing code has clear architectural violations (business logic in composables, scattered state mutations, etc.).
Workflow
When helping with Jetpack Compose or Compose Multiplatform code, follow this process:
- Read the existing code first for context — check conventions, base classes, and layout. For small UI or logic asks, restrict your reading to the immediately relevant files to save time. Do not map out the entire project architecture unless a structural refactor is requested.
- Identify the concern — is this architecture, state modeling, performance, navigation, DI, animation, cross-platform, or testing?
- Apply the core rules below — the decision heuristics and defaults in this file cover most cases.
- Consult the right reference — load the relevant file from
references/only when deeper guidance is needed. Use the Quick Routing in the Detailed References section to pick the right file. - Verify dependencies before recommending — before adding or upgrading any dependency, verify coordinates, target support, and API shape via a documentation MCP tool or official docs (see Dependency Verification Rule).
- Flag anti-patterns contextually — if the user's code violates best practices, call it out for production code. For quick prototypes or minor UI tweaks, prioritize answering their specific question over lecturing them on strict rules.
- Write the minimal correct solution — do not over-engineer. Prefer feature-specific code over generic frameworks.
Dependency Verification Rule
Before recommending any new dependency or version upgrade, verify:
- Coordinates — Confirm the exact Maven coordinates (
group:artifact:version) exist and are current. - Target support — Confirm the artifact supports the project's targets (Android, iOS, Desktop,
commonMain). Do not assume a Jetpack library works incommonMainunless verified. - API shape — Confirm the API you plan to use actually exists in that version. Function signatures, parameter names, and return types change between major versions.
How to verify:
- Documentation MCP tool (preferred) — If a documentation MCP server is available (e.g., Context7), verify exact tool names and schemas first, then use it to fetch current official documentation for the library.
- Official docs — Search the library's official documentation or release notes.
- Maven Central / Google Maven — Check artifact availability and supported platforms.
If verification is not possible (no documentation tool, no network access, docs unavailable), provide the standard or latest known dependency snippet anyway. Add a brief comment (e.g., // Verify latest version) so the user isn't blocked.
Fetching Up-to-Date Documentation
When adding a new dependency, upgrading major versions, or verifying latest API patterns, use a documentation MCP tool (e.g., Context7) if available. Before invoking, verify the tool's exact name and parameter schema — tool names vary across environments.
- Resolve library ID — if the tool requires a resolution step, call it first.
- Query docs — call with the resolved ID and a specific question.
Alternative: Users can add use context7 (or equivalent) to their prompt. Bundled references remain the primary source for architectural patterns and MVI guidance; use documentation tools for API-specific and version-specific queries.
Core Architecture: MVI or MVVM
Both MVI and MVVM use unidirectional data flow: UI renders state → user acts → ViewModel updates state → UI re-renders. The difference is how UI actions reach the ViewModel.
- MVI:
sealed interface Event+ singleonEvent()entry point - MVVM: Named public functions (
onTitleChanged(),save())
Both patterns use:
- State — immutable data class that fully describes the screen, owned via
StateFlow - Effect — one-shot commands (navigate, snackbar, share) delivered via
Channel
Default recommendation: Preserve the project's existing pattern when it is coherent. For new projects, choose based on team preference and screen complexity. See Architecture & State Management for the decision guide, then mvi.md or mvvm.md for implementation details.
UI Rendering Boundary
These boundaries apply to both MVI and MVVM:
- Route composable: obtains ViewModel, collects state via
collectAsStateWithLifecycle(), collects effects viaCollectEffect(see compose-essentials.md), binds navigation/snackbar/platform APIs - Screen composable: stateless renderer — receives state and callbacks (MVI:
onEvent, MVVM: individual callbacks), renders the screen, adapts callbacks for leaf composables - Leaf composables: render sub-state, emit specific callbacks, keep only tiny visual-local state (focus, scroll, animation)
Decision Heuristics
- Composable functions render state and emit events, never decide business rules
- If a value can be derived from state, do not store it redundantly unless async/persistence/performance justifies it
- Event handling in the ViewModel owns state transitions; composables do not mutate state
- UI-local state is acceptable only for ephemeral visual concerns: focus, scroll, animation progress, expansion toggles
- Do not push animation-only flags into global screen state unless business logic depends on them
- Pass the narrowest possible state to leaf composables
- MVI: implement
onEvent()as the single entry point; MVVM: implement named functions for user actions - Do not introduce a use case for every repository call
- Cross-platform sharing prioritizes business logic and presentation state before platform behavior
- Least recomposition is achieved by state shape and read boundaries first, Compose APIs second
- When a project has an existing MVI base class or pattern, use it — don't introduce a competing abstraction
State Modeling
For calculator/form screens, split state into four buckets:
- Editable input — raw text and choice values as the user edits them
- Derived display/business — parsed, validated, calculated values
- Persisted domain snapshot — saved entity for dirty tracking or reset
- Transient UI-only — purely visual, not business-significant
| Concern | Where | Example |
|---|---|---|
| Raw field text | state fields | "12", "12.", "" |
| Parsed/ |