Multiple NavGraphs with Compose Navigation

Multiple NavGraphs with Compose Navigation

Update: This article was written before compose-navigation reached alpha01 and has been not updated since to reflect the changes in alpha01.

Update 2: I strongly recommend following the official Jetpack Compose Navigation codelab instead of following this article.

In part 1, I wrote about how to get started with the Jetpack Compose Navigation library. I discussed how to create a simple navigation graph and how to obtain information about the graph outside it. Now, let’s explore how we can use multiple navigation graphs in a Bottom Navigation-driven UI.

Setup instructions

For instructions on how to setup your project to import the Jetpack Compose Navigation library, please refer to part 1 of this series.

BottomNavigation

Let’s explore a scenario where we use BottomNavigation in our app but we have one tab inside which we need hierarchical navigation. Google’s Clock app does this for the Timer tab.

Implementation

Let’s create a Composable that shows a screen based on which tab is selected:

@Composable
fun TabContent(screen: Screen) {
  when (screen) {
    Screen.Profile -> ProfileTab()
    Screen.Dashboard -> DashboardTab()
    Screen.Phrases -> Phrases()
    else -> ProfileTab()
  }
}

@Composable
fun DashboardTab() {
  val navController = rememberNavController()

  NavHost(
    navController = navController,
    startDestination = "Dashboard"
  ) {
    composable("Dashboard") {
        Dashboard(navController)
    }
    composable("DashboardDetail") {
        Text("Some Dashboard detail")
    }
  }
}

We pass the selected screen to this Composable, it shows it. Our Dashboard tab has changed though. That’s the tab we need navigation inside, DashboardTab just creates a NavHost and defines a NavGraph. Dashboard now has a Button that navigates to the DashboardDetail screen when clicked. We’ve also add it to the previously defined Screen sealed class.

Figure 1 - Bottom Navigation with back press issue

That looks good, right? Looks like there are a few issues though.

Back button doesn’t seem to work properly

It seems that back button taps get intercepted by the NavHostController even if we leave the tab containing the NavHost (i.e. the Dashboard Screen). I’m not sure why this is happening and it looks like a bug but I could be wrong. I have to confirm this before I file a bug.

In the meantime, I have found that disabling the NavHostController’s OnBackPressed functionality in onDispose seems to fix this. When DashboardTab is recomposed, this resets back to true.

@Composable
fun DashboardTab() {
  // ... existing code

  onDispose {
    // workaround for issue where back press is intercepted
    // outside this tab, even after this Composable is disposed
    navController.enableOnBackPressed(false)
  }
}

Works properly now! Pressing the back button closes the app now instead of doing nothing.

Figure 2 - Bottom Navigation with back press issue fixed

Dashboard’s backstack gets cleared on switching tabs

We can see in figure 1 that Dashboard screen’s backstack gets cleared when we switch tabs away from it. This is the default behaviour but it’s easy to save and restore the backstack state. Let’s create a MutableState to hold the navigation state and pass it to DashboardTab:

@Composable
fun TabContent(screen: Screen) {
  val navState = remember { mutableStateOf(Bundle()) }
  when (screen) {
    Screen.Dashboard -> DashboardTab(navState)
    // .. other screens
  }
}

@Composable
fun DashboardTab(navState: MutableState<Bundle>) {
  val navController = rememberNavController()

  onCommit {
    navController.addOnDestinationChangedListener { navController, _, _ ->
      navState.value = navController.saveState() ?: Bundle()
    }
    navController.restoreState(navState.value)

    onDispose {
      navController.removeOnDestinationChangedListener(callback)
      // workaround for issue where back press is intercepted
      // outside this tab, even after this Composable is disposed
      navController.enableOnBackPressed(false)
    }
  }

  // .. NavHost stuff
}

NavHostController provides us the saveState() and the restoreState(Bundle) functions to manually handle the state of the backstack. On every destination change, we save the state to navState and every time DashboardTab gets recomposed, we restore its state from navState. We keep this navState in TabContent so it survives even when DashboardTab is disposed. We also remove the OnDestinationChangedListener from the navController when this Composable is disposed to avoid creating unnecessary listeners on every recomposition.

Okay, this is working now. We’re able to keep Dashboard screen’s backstack, even as we switch away and return to it!

Figure 3 - Bottom Navigation with backstack restoration

It’s not surviving process death anymore!

One downside of manually handling the backstack state is that we lose NavHostController’s built-in ability to survive process death. Let’s add this ability back:

@Composable
fun TabContent(screen: Screen) {
  val navState = rememberSavedInstanceState(saver = NavStateSaver()) { mutableStateOf(Bundle()) }
  // .. show screen
}

fun NavStateSaver(): Saver<MutableState<Bundle>, out Any> = Saver(
  save = { it.value },
  restore = { mutableStateOf(it) }
)

rememberSavedInstanceState allows us to persist and restore mutable data beyond process death. It uses a Saver object to handle the save and restore operations. Since our navState is just a MutableState<Bundle>, for saving, we just get the value: Bundle from it. For restoring, we recreate the MutableState object from that Bundle.

The backstack state will now persist across process death.

The currently selected screen in TabContent doesn’t survive process death. The code on GitHub shows how to do this as well.

Bottom Navigation code on GitHub

Multiple NavGraphs within BottomNavigation

What if we want to keep multiple backstacks, within multiple tabs? Actually, it’s not very different from keeping one NavGraph. Let’s create a Phrases screen to have its own backstack. It will navigate to a new screen called PhraseDetail.

@Composable
fun TabContent(screen: Screen) {
  val dashboardNavState = rememberSavedInstanceState(saver = NavStateSaver()) { mutableStateOf(Bundle()) }
  val phrasesNavState = rememberSavedInstanceState(saver = NavStateSaver()) { mutableStateOf(Bundle()) }
  when (screen) {
    Screen.Dashboard -> DashboardTab(dashboardNavState)
    Screen.Phrases -> PhrasesTab(phrasesNavState)
    // .. other screens
  }
}

@Composable
fun PhrasesTab(navState: MutableState<Bundle>) {
  val navController = rememberNavController()

  onCommit {
    val callback = NavController.OnDestinationChangedListener { navController, _, _ ->
      navState.value = navController.saveState() ?: Bundle()
    }
    navController.addOnDestinationChangedListener(callback)
    navController.restoreState(navState.value)
    onDispose {
      navController.removeOnDestinationChangedListener(callback)
      // workaround for issue where back press is intercepted
      // outside this tab, even after this Composable is disposed
      navController.enableOnBackPressed(false)
    }
  }

  NavHost(
    navController = navController,
    startDestination = "Phrases"
  ) {
    composable("Phrases") {
        Phrases(navController)
    }
    composable("PhraseDetail") {
        PhraseDetail()
    }
  }
}

PhrasesTab ends up looking very similar to DashboardTab. At this point, we could probably create a RestorableNavHost that just contains this functionality, takes a MutableState<Bundle> and a NavGraphBuilder function.

The Result

Figure 4 - Bottom Navigation with multiple Nav Graphs

GitHub repository

All of the code discussed in this blog post is available here:

Multiple Nav Graphs code on GitHub

Conclusion

While it was still relatively simple to achieve this, there were a few things we had to be careful of. Since this is pre-alpha/alpha stage, I’m sure some (if not all) of the issues mentioned here will be addressed by the time this is production ready. I’m really excited for Compose and how it shapes the future of Android development and the quality of apps that we will be able to create.

Found this article interesting, or better yet, found a bug in the article? Please comment and let me know!

Thanks to the Compose Navigation samples by the Androidx Team, and thanks to Neal Manaktola for reviewing this article.


© Afzal Najam