Loading...
Loading...
Use this skill to migrate your Jetpack Compose app to add adaptive edge-to-edge support and troubleshoot common issues. Use this skill to fix UI components (like buttons or lists) that are obscured by or overlapping with the navigation bar or status bar, fix IME insets, and fix system bar legibility.
npx skill4agent add android/skills edge-to-edgeTextFieldOutlinedTextFieldBasicTextFieldenableEdgeToEdgesetContentonCreateenableEdgeToEdgeandroid:windowSoftInputMode="adjustResize"ScaffoldPaddingValuesScaffold { innerPadding ->
// innerPadding accounts for system bars and any Scaffold components
LazyColumn(
modifier = Modifier
.fillMaxSize()
.consumeWindowInsets(innerPadding),
contentPadding = innerPadding
) { /* Content */ }
}TopAppBarSmallTopAppBarCenterAlignedTopAppBarMediumTopAppBarLargeTopAppBarBottomAppBarModalDrawerSheetDismissibleDrawerSheetPermanentDrawerSheetModalBottomSheetNavigationBarNavigationRailwindowInsetsBottomAppBarTopAppBarBottomNavigationTopAppBarTopAppBar(windowInsets = AppBarDefaults.topAppBarWindowInsets)TopAppBar(windowInsets = WindowInsets.systemBars.exclude(WindowInsets.navigationBars))TopAppBar(windowInsets = WindowInsets.systemBars.add(WindowInsets.captionBar))Modifier.safeDrawingPadding()Modifier.windowInsetsPadding(WindowInsets.safeDrawing)Box(
modifier = Modifier
.fillMaxSize()
.safeDrawingPadding()
) {
Button(
onClick = {},
modifier = Modifier.align(Alignment.BottomCenter)
) {
Text("Login")
}
}WindowInsetsRulersModifier.fitInside(WindowInsetsRulers.SafeDrawing.current)Modifier.windowInsetsTopHeight(WindowInsets.systemBars)NavigationSuiteScaffoldNavigationRailNavigationBarNavigationSuiteScaffoldListDetailPaneScaffoldcontentPaddingsafeDrawingPaddingNavigationSuiteScaffoldandroid:windowSoftInputMode="adjustResize"SOFT_INPUT_ADJUST_RESIZEModifier.fitInside(WindowInsetsRulers.Ime.current)imePadding()imePaddingModifier.verticalScroll()Modifier.imePadding()contentWindowInsetscontentWindowInsets = WindowInsets.safeDrawingcontentWindowInsetsinnerPadding// RIGHT
Scaffold(contentWindowInsets = WindowInsets.safeDrawing) { innerPadding ->
Column(
modifier = Modifier
.padding(innerPadding)
.consumeWindowInsets(innerPadding)
.verticalScroll(rememberScrollState())
) { /* Content */ }
}fitInsidecontentWindowInsets// RIGHT
Scaffold() { innerPadding ->
Column(
modifier = Modifier
.padding(innerPadding)
.consumeWindowInsets(innerPadding)
.fitInside(WindowInsetsRulers.Ime.current)
.verticalScroll(rememberScrollState())
) { /* Content */ }
}contentWindowInsetsimePadding()// RIGHT
Scaffold() { innerPadding ->
Column(
modifier = Modifier
.padding(innerPadding)
.consumeWindowInsets(innerPadding)
.imePadding()
.verticalScroll(rememberScrollState())
) { /* Content */ }
}contentWindowInsetsimePadding// WRONG
Scaffold( contentWindowInsets = WindowInsets.safeDrawing ) { innerPadding ->
Column(
modifier = Modifier
.padding(innerPadding)
.imePadding()
.verticalScroll(rememberScrollState())
) { /* Content */ }
}contentWindowInsets// WRONG
Scaffold() { innerPadding ->
Column(
modifier = Modifier
.padding(innerPadding)
.verticalScroll(rememberScrollState())
) { /* Content */ }
}// RIGHT
Box(
// Insets consumed
modifier = Modifier.safeDrawingPadding() // or imePadding(), safeContentPadding(), safeGesturesPadding()
) {
Column(
modifier = Modifier.imePadding()
) { /* Content */ }
}// RIGHT
Box(
// Insets consumed
modifier = Modifier.windowInsetsPadding(WindowInsets.safeDrawing) // or WindowInsets.ime, WindowInsets.safeContent, WindowInsets.safeGestures
) {
Column(
modifier = Modifier.imePadding()
) { /* Content */ }
}// RIGHT
Box(
// Insets not consumed, but irrelevant due to fitInside
modifier = Modifier.padding(WindowInsets.safeDrawing.asPaddingValues()) // or WindowInsets.ime.asPaddingValues(), WindowInsets.safeContent.asPaddingValues(), WindowInsets.safeGestures.asPaddingValues()
) {
Column(
modifier = Modifier
.fillMaxSize()
.fitInside(WindowInsetsRulers.Ime.current)
) { /* Content */ }
}// WRONG
Box(
// Insets not consumed
modifier = Modifier.padding(WindowInsets.safeDrawing.asPaddingValues()) // or WindowInsets.ime.asPaddingValues(), WindowInsets.safeContent.asPaddingValues(), WindowInsets.safeGestures.asPaddingValues()
) {
Column(
modifier = Modifier.imePadding()
) { /* Content */ }
}enableEdgeToEdgeWindowCompatisAppearanceLightNavigationBarsisAppearanceLightStatusBarsenableEdgeToEdgeComponentActivity// Only use if calling `enableEdgeToEdge` from `WindowCompat`.
// Apply to your theme file.
@Composable
fun MyTheme(
darkTheme: Boolean = isSystemInDarkTheme(),
content: @Composable () -> Unit
) {
val view = LocalView.current
if (!view.isInEditMode) {
SideEffect {
val window = (view.context as? Activity)?.window ?: return@SideEffect
val controller = WindowCompat.getInsetsController(window, view)
// Dark icons for Light Mode (!darkTheme), Light icons for Dark Mode
controller.isAppearanceLightStatusBars = !darkTheme
controller.isAppearanceLightNavigationBars = !darkTheme
}
}
MaterialTheme(content = content)
}ScaffoldNavigationSuiteScaffoldBottomAppBarNavigationBarwindow.isNavigationBarContrastEnforced = falseScaffoldinnerPaddingcontentPaddingLazyColumnLazyRowModifier.padding()class SystemBarProtectionSnippets : ComponentActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
// enableEdgeToEdge sets window.isNavigationBarContrastEnforced = true
// which is used to add a translucent scrim to three-button navigation
enableEdgeToEdge()
setContent {
MyTheme {
// Main content
MyContent()
// After drawing main content, draw status bar protection
StatusBarProtection()
}
}
}
}
@Composable
private fun StatusBarProtection(
color: Color = MaterialTheme.colorScheme.surfaceContainer,
) {
Spacer(
modifier = Modifier
.fillMaxWidth()
.height(
with(LocalDensity.current) {
(WindowInsets.statusBars.getTop(this) * 1.2f).toDp()
}
)
.background(
brush = Brush.verticalGradient(
colors = listOf(
color.copy(alpha = 1f),
color.copy(alpha = 0.8f),
Color.Transparent
)
)
)
)
}DialogPropertiesusePlatformDefaultWidth = falseModifier.fillMaxSize()decorFitsSystemWindows = falseDialogPropertiesDialog(
onDismissRequest = { /* Handle dismiss */ },
properties = DialogProperties(
// 1. Allows the dialog to span the full width of the screen
usePlatformDefaultWidth = false,
// 2. Allows the dialog to draw behind status and navigation bars
decorFitsSystemWindows = false
)
) { /* Content */ }ActivityenableEdgeToEdge()adjustResizeAndroidManifest.xmlTextFieldOutlinedTextFieldBasicTextFieldimePadding()fitInsideModifier.safeDrawingPadding()Modifier.safeContentPadding()Modifier.safeGesturesPadding()contentWindowInsetsWindowInsets.safeDrawingWindowInsets.imecontentPaddingModifier.safeDrawingPadding()./gradlew build