| name | compose-slot-api-pattern |
| description | Use when designing or reviewing a reusable Jetpack Compose component whose visual regions vary by caller, or when primitive content parameters and boolean shape flags are accumulating. Technique-layer skill โ complements the codebase-specific compose-expert. |
Compose: slot API pattern
Core principle
A reusable Compose component's job is to lay things out, not to enumerate what it lays out. The moment you write title: String, subtitle: String?, leadingIcon: ImageVector?, trailingIcon: ImageVector?, trailingText: String?, showSwitch: Boolean, switchValue: Boolean, onSwitchChange: (Boolean) -> Unit?, badge: String?, โฆ, the component has stopped describing a layout and started enumerating call sites โ and the next call site will need a parameter the component doesn't have.
The fix is to delegate content to the caller via @Composable lambda parameters. The component contributes structure (where the leading bit, headline, supporting bit, trailing bit go). The caller contributes everything that goes in those slots.
Material 3's ListItem is the canonical example: every visual piece is a slot (headlineContent, supportingContent, leadingContent, trailingContent, overlineContent), not a primitive. That's not over-engineering โ it's the design that scales to every list-item shape the design system needs without ever editing ListItem again.
When to use this skill
You're designing or reviewing a Compose component intended for reuse (more than one call site, now or planned), its visual content varies by caller, and any of these is true:
- Its signature has
title: String, icon: ImageVector, actionText: String?, etc. โ primitive types describing content.
- It has multiple optional-content parameters that vary by call site (
subtitle: String?, leadingIcon: ImageVector?, trailingText: String?).
- It has boolean flags whose only purpose is to switch between content shapes (
showChevron: Boolean, showSwitch: Boolean, mode: Mode.Text | Mode.Switch | โฆ).
- It accepts a
String parameter where one caller would want a Text with custom style, a second caller a Text with a Badge, a third caller a row of icons.
- It already has one slot (often
trailing or content) and the rest of the parameters are still primitives.
1. Replace primitive content with @Composable slots
Where the component asks for caller-controlled content, prefer a @Composable () -> Unit slot. Where the slot is structurally required, leave it non-nullable with no default. Where it's optional, make it nullable with a null default.
@Composable
fun SettingsRow(
title: String,
onClick: () -> Unit,
modifier: Modifier = Modifier,
subtitle: String? = null,
leadingIcon: ImageVector? = null,
trailing: (@Composable () -> Unit)? = null,
) { โฆ }
This shape seems fine because the call sites today fit (title is always single-line text, leadingIcon is always an ImageVector). The problem is the next call site: a row with a Badge next to the title, a leading slot that's a circular avatar (not an ImageVector), a subtitle that's a row of chips. Each forces either a new parameter, a new flag, or a workaround.
@Composable
fun SettingsRow(
headlineContent: @Composable () -> Unit,
onClick: () -> Unit,
modifier: Modifier = Modifier,
supportingContent: (@Composable () -> Unit)? = null,
leadingContent: (@Composable () -> Unit)? = null,
trailingContent: (@Composable () -> Unit)? = null,
) { โฆ }
Call sites stay short because the typical content is a one-liner:
SettingsRow(
headlineContent = { Text("Account") },
leadingContent = { Icon(Icons.Default.Person, contentDescription = null) },
trailingContent = { SettingsRowDefaults.Chevron() },
onClick = { โฆ },
)
And the awkward cases that would have required new primitive parameters now don't:
SettingsRow(
headlineContent = {
Row(verticalAlignment = Alignment.CenterVertically) {
Text("Inbox")
Spacer(Modifier.width(8.dp))
Badge { Text("3") }
}
},
onClick = { โฆ },
)
Slot naming
- Use
xxxContent for free-form @Composable () -> Unit slots (headlineContent, supportingContent, trailingContent) โ matches Material 3.
- Use a singular noun (
title, icon, actions) when the slot is semantically constrained and the component name disambiguates (Scaffold(topBar = { โฆ }, bottomBar = { โฆ }, floatingActionButton = { โฆ })).
- Don't use
content and other xxxContent slots together โ pick one convention per component.
2. Scope receivers when the slot emits into a layout
If the slot's content will sit inside a Row/Column/Box whose layout features (Modifier.weight, BoxScope.matchParentSize, alignment) should be available to the caller, declare the slot as a receiver lambda: @Composable RowScope.() -> Unit.
@Composable
fun MyTopBar(
title: @Composable () -> Unit,
actions: @Composable () -> Unit = {},
)
@Composable
fun MyTopBar(
title: @Composable () -> Unit,
actions: @Composable RowScope.() -> Unit = {},
)
This is what makes TopAppBar(actions = { IconButton(โฆ); IconButton(โฆ) }) work โ the caller is implicitly inside a RowScope.
Don't bolt a scope receiver onto every slot reflexively. The receiver should match the actual parent layout the slot emits into. If the slot is rendered inside a Box, use BoxScope. If it's inside a Column, use ColumnScope. If the parent is not a standard layout (or none of its scope APIs are useful in slot content), no receiver.
3. Optional slots โ nullable with null default
For slots that may be absent, prefer (@Composable () -> Unit)? = null over @Composable () -> Unit = {}:
leadingContent: @Composable () -> Unit = {}
leadingContent: (@Composable () -> Unit)? = null
Why: with a nullable slot, the component can branch on leadingContent != null and skip the slot's container, spacing, padding entirely. With an empty default, the layout still allocates the slot โ sometimes you see a stray padding or spacer around content that turned out to be nothing. The nullable form makes the "absent" case structurally distinct, which is almost always what you want.
The trade-off: callers who pass an explicit empty {} to silence a slot now have to pass null or omit the argument. That's the right answer either way โ they shouldn't be passing {}.
4. Defaults live in XxxDefaults
When you find yourself documenting "the trailing slot should usually be a chevron" or "pass MaterialTheme.colorScheme.surface for the default background", co-locate the helpers in a XxxDefaults object next to the component:
object SettingsRowDefaults {
@Composable
fun Chevron() = Icon(
imageVector = Icons.AutoMirrored.Filled.KeyboardArrowRight,
contentDescription = null,
)
@Composable
fun TrailingValue(text: String) = Text(
text = text,
style = MaterialTheme.typography.bodyMedium,
color = MaterialTheme.colorScheme.onSurfaceVariant,
)
}
Call sites stay declarative for the common cases and the slot is still fully open for one-offs:
SettingsRow(
headlineContent = { Text("Notifications") },
trailingContent = { SettingsRowDefaults.Chevron() },
onClick = { โฆ },
)
This matches Material 3's ButtonDefaults, TopAppBarDefaults, etc. โ defaults that are themselves composable belong here, not as new component parameters with MaterialTheme.x.y defaults expanded inline.
Quick reference
| Symptom | Diagnosis | Fix |
|---|
title: String, subtitle: String?, leadingIcon: ImageVector? on a reusable component | Primitive content params (ยง1) | Convert to xxxContent: (@Composable () -> Unit)? slots |
Multiple boolean flags (showChevron, showSwitch) selecting trailing shapes | Enumerating shapes (ยง1) | One trailingContent: (@Composable () -> Unit)? slot |
A mode: Mode.Sealed parameter listing variants | Same as flag soup (ยง1) | Slot it |
actions: @Composable () -> Unit = {} inside a Row body | Missing scope receiver (ยง2) | actions: @Composable RowScope.() -> Unit = {} |
slot: @Composable () -> Unit = {} for an optional area | Empty-lambda default (ยง3) | slot: (@Composable () -> Unit)? = null and branch on it |
Component param defaultColor: Color = MaterialTheme.colorScheme.surface | Defaults inlined (ยง4) | Move to XxxDefaults.color and reference it |
| Common trailing content repeats at every call site | Missing default helper (ยง4) | Add XxxDefaults.Chevron() etc. |
When NOT to apply
- Single-use components. A composable used in exactly one place, with no plan to reuse, doesn't benefit from slot flexibility โ and the slot indirection makes the code harder to read for the one reader. Primitive params + inline content is fine. (As soon as a second call site appears, slot it.)
- Design-system primitives where every caller must look identical. A
Heading2(text: String) exists because you want every H2 to look the same; making it headlineContent: @Composable () -> Unit invites callers to break the rule. Keep it primitive. (Conversely: if Heading2 ever needs a badge inline, slot it.)
- Semantic parameters the component intentionally owns. If the component owns typography, iconography, accessibility wording, or product consistency, a primitive parameter may be the constraint you want.
- Constrained-type parameters that genuinely are constrained. A
Switch(checked: Boolean, onCheckedChange: ...) doesn't need its checked indicator to be a slot. Booleans-with-callbacks are not "content."
- Performance-critical fast paths (rare in app code; common in framework primitives). A slot is an allocated lambda. In the deepest LazyList item layer, sometimes primitives win. If you're not writing the framework, this doesn't apply.
Red flags during review
| Thought | Reality |
|---|
| "Title is always a String โ making it a slot is over-engineering" | "Always today" is the trap. Material's ListItem.headlineContent exists because tomorrow someone wants a Text + Badge. The slot is 8 characters of extra wrapping at every call site ({ Text(โฆ) }); the refactor to add a slot later edits every existing call site. |
| "Lambdas are heavier than strings" | At the scale of typical Compose UI, this isn't measurable โ and the framework's own components (Button, ListItem, TopAppBar, Scaffold) all slot. If your component is in the hottest of hot paths, see "When NOT to apply." |
| "I'll add a slot later if someone asks" | The slot turns one parameter into two parameters (the slot itself + maybe an internal flag) and edits every call site. The shape change isn't a "later" change. |
"I'll model the variants with a sealed Trailing type instead" | Sealed enumeration is bounded; slots are unbounded. A sealed type works until the day someone needs a variant you didn't anticipate โ at which point you're back to editing the component. The slot avoids the cycle. |
| "The leading area is always an icon, the trailing area varies โ I'll slot only the trailing" | This is the partial-slot trap. The "always-an-icon" assumption breaks the first time a row needs an avatar or a flag emoji or a coloured shape. Slot leading too. |
| "There's only one call site today" | If there's only one call site, you're probably not designing a reusable component yet. See "When NOT to apply" โ primitives are fine for a true single-use. The moment you copy-paste it, slot it. |
Related
compose-modifier-and-layout-style โ the modifier-parameter rule (ยง1โยง3 there) travels with slot APIs. A reusable component takes a modifier parameter and slots its content; the caller owns both placement and what to place.