/responsive-scaffold

A modified Scaffold layout from the androidx.compose.material3 package with a support for Side Rail content.

Primary LanguageKotlinApache License 2.0Apache-2.0

Responsive Scaffold

API GitHub last commit GitHub issues

A modified Scaffold layout from the androidx.compose.material3 package with a support for Side Rail content.


Table of Contents


Key Features

  • Side Rail Support: The Responsive Scaffold layout is thoughtfully designed to seamlessly incorporate Side Rail content, making it ideal for implementing components like a NavigationRail on larger screen devices or compact devices in landscape mode. This creates a natural and efficient user navigation experience.

  • Adaptive Window Sizing: Leveraging the capabilities of the WindowSizeClass from the androidx.compose.material3.windowsizeclass package, the layout dynamically adjusts its hierarchy based on the window width size. It enables the layout to smoothly switch between displaying a Bottom Bar content for compact window width size and displaying a Side Rail content for medium or expanded window width size.

  • When the window width size is medium or expanded, the Floating Action Button content disappears, in that case, the FAB should be included at the upper left of the screen, as stated in the Material Design FAB guidelines. For example by placing it inside the header content of a NavigationRail component.


Requirements

  • Project minSdk version - 23
  • Project compileSdk version - 34
  • Jetpack Compose version - 1.5.4

Implementation

Before implementing the Responsive Scaffold, it's recommended (but not necessarily required) to provide a top level WindowSizeClass instance, simply by wrapping the activity content inside a ProvideWindowSizeClass composable for consumption down the composition and to avoid calculating and recreating multiple WindowSizeClass instances.

The default WindowSizeClass instance provided by the ProvideWindowSizeClass composable is calculated via a modified calculateWindowSizeClass() method, which is also used by the Responsive Scaffold to create an instance if not provided.

@Composable
fun ProvideWindowSizeClass(
    windowSizeClass: WindowSizeClass = calculateWindowSizeClass(),
    content: @Composable () -> Unit
) = CompositionLocalProvider(
    LocalWindowSizeClass provides windowSizeClass,
    content = content
)

If you don't have a top level WindowSizeClass instance, the ProvideWindowSizeClass composable will provide it under the hood:

setContent {

    ProvideWindowSizeClass {

        // The activity UI content

    }

}

If you already have a top level activity WindowSizeClass instance, you can pass it as an argument:

setContent {

    val activityWindowSizeClass = calculateWindowSizeClass(activity = this)

    ProvideWindowSizeClass(
        windowSizeClass = activityWindowSizeClass
    ) {

        // The activity UI content

    }

}

Here is an example of how to consume the provided WindowSizeClass instance whenever you need it:

val windowSizeClass: WindowSizeClass? = LocalWindowSizeClass.current

Responsive Scaffold Parameters:

  • modifier - a Modifier to be applied to the scaffold.
  • topBar - a top app bar component, typically a TopAppBar.
  • bottomBar - a bottom bar component, typically a NavigationBar.
  • sideRail - a side rail component, typically a NavigationRail.
  • snackbarHost - a component to host Snackbars.
  • floatingActionButton - the main action button of the screen, typically a FloatingActionButton.
  • floatingActionButtonPosition - position of the FAB on the screen.
  • containerColor - a color used for the background of the scaffold.
  • contentColor - a preferred color for the content inside the scaffold.
  • contentWindowInsets - a window insets to be passed to the content slot via PaddingValues params.
  • windowSizeClass - used for under the hood calculations from a WindowSizeClass instance, with a support for optional overriding.
  • content - the main content of the screen, e.g. a LazyColumn.

ResponsiveScaffold(
    topBar = { TopAppBar() },
    bottomBar = { NavigationBar() },
    sideRail = { NavigationRail() },
    floatingActionButton = { FAB() },
    // ...
) { paddingValues ->
    
    LazyColumn(
        modifier = Modifier.fillMaxSize(),
        contentPadding = paddingValues,
        // ...
    ) {
        // ...
    }
    
}


Gradle Kotlin DSL Setup

Step 1

  • Add the Jitpack maven repository in your settings.gradle.kts file.
repositories {
    maven(url = "https://jitpack.io")
}

Step 2

  • Add the Responsive Scaffold dependency in your module build.gradle.kts file.
  • Latest version:
implementation("com.github.stoyan-vuchev:responsive-scaffold:<version>")
  • Or if you're using a libs.versions.toml catalog, declare it in the catalog instead.
[versions]
responsive-scaffold = "<version>"

[libraries]
responsive-scaffold = { group = "com.github.stoyan-vuchev", name = "responsive-scaffold", version.ref = "responsive-scaffold" }
  • Then include the dependency in your module build.gradle.kts file.
implementation(libs.responsive.scaffold)

Step 3

  • Sync and rebuild the project.

Gradle Groovy Setup

Step 1

  • Add the Jitpack maven repository in your project (root) level build.gradle file.
allprojects {
    repositories {
        maven { url 'https://jitpack.io' }
    }
}

Step 2

  • Add the Responsive Scaffold dependency in your module build.gradle file.
  • Latest version:
implementation 'com.github.stoyan-vuchev:responsive-scaffold:<version>'

Step 3

  • Sync and rebuild the project.

Notice

This project includes software components that are subject to the Apache License, Version 2.0.

Portions of this software were originally developed by The Android Open Source Project.

Copyright 2021-2022 The Android Open Source Project

Modifications and enhancements to this software have been made by Stoyan Vuchev, licensed under the Apache License, Version 2.0.

For more information, please refer to the NOTICE file.

Attributions:


License

This project is open source and available under the Apache License, Version 2.0. For more information, please refer to the LICENSE file.