ProAndroidDev

The latest posts from Android Professionals and Google Developer Experts.

Follow publication

Jetpack Compose Tips and Conventions for the @Composables to Make Them Better

Tomáš Repčík
ProAndroidDev
Published in
6 min readMar 3, 2024

Simplicity over complexity

// LOW-LEVEL BLOCS

// custom body text
@Composable
fun BodyText(...) {
}

// icon with custom modifier for whole app
@Composable
fun AppIcon(...) {
}

/// HIGHER-LEVEL BLOCS
@Composable
fun IconTextButton(@StringRes val title: Int, @DrawableRes val drawableId: Int, modifier: Modifier) =
Surface(
modifier = modifier,
) {
Row() {
AppIcon(
painter = painterResource(id = item.drawableId),
contentDescription = null
)
Spacer()
BodyText(
text = stringResource(id = title),
)
}
}

/// SCREEN-LEVEL
@Composable
fun NavigationScreen() {
Column {
IconTextButton(...)
IconTextButton(...)
IconTextButton(...)
IconTextButton(...)
}
}

Consistency over customisations

// DON'T
@Composable
fun Button(style: ButtonStyle, ...) {
...
}
// create your own versions of the text as independent composables
@Composable
fun TitleText(@StringRes id: int, ...) {
Text(
text = stringResource(id = id),
style = MaterialTheme.typography.titleLarge
)
}

@Composable
fun BodyText(@StringRes id: int, ...) {
Text(
text = stringResource(id = id),
style = MaterialTheme.typography.bodyMedium
)
}

// in need of modification, modify already existing one
// without adding more styles - add more explicit inputs with defaults
@Composable
fun OtherTitleText(@StringRes id: int, textAlign = TextAlign.Center, ... ) {
TitleText(id = id, textAlign = textAlign)
}

Input parameters

Modifier

Order

@Composable
fun AppButton(
// required might not be even needed,
// because modifier has most of the required parameters
// such as clickable {}

// modifier
modifier: Modifier = Modifier,
// optional
enabled: Boolean = true,
// composable trailing
text: @Composable () -> Unit,
) {}
// usage
AppButton(
modifier = Modifier.padding(bottom = 30.dp).clickable {},
) {
Text(...)
}

State

data class SomeData(val text: String) 

// pass the data directly to the composable
@Composable
fun TitleText(text: SomeData) {
Text(text, ...)
}

// delay getter by passing function
@Composable
fun MainScreen(text: () -> String){
}

// invocation with function
MainScreen({ someData.text })

Slot inputs

@Composable
fun TextButton(
text: String,
)
{
...
}
@Composable
fun BoldBodyText(
text: String
)
{
...
}

@Composable
fun PrimaryButton(
content: @Composable () -> Unit
)
{
...
content()
...
}
// usage / composable which could be extracted as individual component
PrimaryButton {
BoldBodyText("My text")
}

Semantics

Conclusion

Resources

For more about Android development:

Android development

23 stories

Free

Distraction-free reading. No ads.

Organize your knowledge with lists and highlights.

Tell your story. Find your audience.

Membership

Read member-only stories

Support writers you read most

Earn money for your writing

Listen to audio narrations

Read offline with the Medium app

Published in ProAndroidDev

The latest posts from Android Professionals and Google Developer Experts.

Written by Tomáš Repčík

https://tomasrepcik.dev/ - Flutter app developer with experience in native app development and degree in biomedical engineering.