The [Navigation Compose API](https://developer.android.com/develop/ui/compose/navigation) lets you navigate between composables in a
Compose app, while taking advantage of the [Jetpack Navigation's](https://developer.android.com/guide/navigation) component,
infrastructure, and features.

This page describes how to migrate from a Fragment-based Jetpack Navigation to
Navigation Compose, as part of the larger, View-based UI migration to Jetpack
Compose.
| **Note:** If you are new to Navigation Compose, see [Navigating with Compose](https://developer.android.com/develop/ui/compose/navigation) first. This doc describes the basics of Navigation Compose, such as how to pass navigation events or complex data and set up composables with arguments.

## Migration prerequisites

You can migrate to Navigation Compose once you're able to *replace all of your
Fragments with corresponding screen composables* . Screen composables can contain
a [mix of Compose and View content](https://developer.android.com/develop/ui/compose/migrate/interoperability-apis/views-in-compose), but *all navigation destinations must be
composables* to enable Navigation Compose migration. Until then, you should
continue using [Fragment-based Navigation component](https://developer.android.com/guide/navigation/navigation-getting-started) in your interop View and
Compose codebase. See the [navigation interop documentation](https://developer.android.com/develop/ui/compose/navigation#interoperability) for more
information.

Using Navigation Compose in a Compose-only app is not a prerequisite. You can
*continue using* [*Fragment-based Navigation component*](https://developer.android.com/guide/navigation/navigation-getting-started), as long as you keep
Fragments for [hosting your composable content](https://developer.android.com/develop/ui/compose/migrate/interoperability-apis/compose-in-views).
| **Note:** Currently, it's not possible to use any single form of the Navigation component to navigate between a mix of Fragment and composable destinations. All destinations must be uniformly either Fragments (using Jetpack Navigation) or composables (using Navigation Compose).

## Migration steps

Whether you are following our [recommended migration strategy](https://developer.android.com/develop/ui/compose/migrate/strategy) or taking
another approach, you'll reach a point where all navigation destinations are
screen composables, with Fragments acting only as composable containers. At this
stage, you can migrate to Navigation Compose.

If your app is already following a [UDF design pattern](https://developer.android.com/develop/ui/compose/architecture#udf) and our [guide to
architecture](https://developer.android.com/topic/architecture), migrating to Jetpack Compose and Navigation Compose shouldn't
require major refactors of other layers of your app, apart from the UI layer.

To migrate to Navigation Compose, follow these steps:

1. Add the [Navigation Compose dependency](https://developer.android.com/develop/ui/compose/navigation#setup) to your app.
2. Create an `App-level` composable and add it to your `Activity` as your
   Compose entry point, replacing the setup of the View layout:


   ```kotlin
   class SampleActivity : ComponentActivity() {

       override fun onCreate(savedInstanceState: Bundle?) {
           super.onCreate(savedInstanceState)
           // setContentView<ActivitySampleBinding>(this, R.layout.activity_sample)
           setContent {
               SampleApp(/* ... */)
           }
       }
   }https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/interop/MigrationCommonScenariosSnippets.kt#L134-L143
   ```

   <br />

3. Create types for each navigation destination. Use a `data object` for
   destinations which don't require any data and `data class` or `class` for
   destinations which require data.


   ```kotlin
   @Serializable data object First
   @Serializable data class Second(val id: String)
   @Serializable data object Third
   https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/interop/MigrationCommonScenariosSnippets.kt#L147-L150
   ```

   <br />

4. Set up the [`NavController`](https://developer.android.com/develop/ui/compose/navigation#getting-started) in a place where all composables that need
   to reference it have access to it (this is usually inside your `App`
   composable). This approach follows the [principles of state hoisting](https://developer.android.com/develop/ui/compose/state#state-hoisting)
   and lets you use the `NavController` as the source of truth for
   navigating between composable screens and maintaining the back stack:


   ```kotlin
   @Composable
   fun SampleApp() {
       val navController = rememberNavController()
       // ...
   }https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/interop/MigrationCommonScenariosSnippets.kt#L155-L159
   ```

   <br />

5. Create your app's [`NavHost`](https://developer.android.com/develop/ui/compose/navigation#create-navhost) inside the `App` composable and pass the
   `navController`:


   ```kotlin
   @Composable
   fun SampleApp() {
       val navController = rememberNavController()

       SampleNavHost(navController = navController)
   }

   @Composable
   fun SampleNavHost(
       navController: NavHostController
   ) {
       NavHost(navController = navController, startDestination = First) {
           // ...
       }
   }https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/interop/MigrationCommonScenariosSnippets.kt#L165-L179
   ```

   <br />

6. Add the `composable` destinations to build your navigation graph. If each
   screen has been previously migrated to Compose, this step only consists of
   extracting these screen composables from your Fragments into the
   `composable` destinations:


   ```kotlin
   class FirstFragment : Fragment() {

       override fun onCreateView(
           inflater: LayoutInflater,
           container: ViewGroup?,
           savedInstanceState: Bundle?
       ): View {
           return ComposeView(requireContext()).apply {
               setContent {
                   // FirstScreen(...) EXTRACT FROM HERE
               }
           }
       }
   }

   @Composable
   fun SampleNavHost(
       navController: NavHostController
   ) {
       NavHost(navController = navController, startDestination = First) {
           composable<First> {
               FirstScreen(/* ... */) // EXTRACT TO HERE
           }
           composable<Second> {
               SecondScreen(/* ... */)
           }
           // ...
       }
   }https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/interop/MigrationCommonScenariosSnippets.kt#L184-L212
   ```

   <br />

7. If you followed the guidance on [architecting your Compose UI](https://developer.android.com/develop/ui/compose/architecture),
   specifically how `ViewModel`s and navigation events should be passed to
   composables, the next step is to change how you provide the `ViewModel` to
   each screen composable. You can often use Hilt injection and its integration
   point with Compose and Navigation via [`hiltViewModel`](https://developer.android.com/develop/ui/compose/libraries#hilt-navigation):


   ```kotlin
   @Composable
   fun FirstScreen(
       // viewModel: FirstViewModel = viewModel(),
       viewModel: FirstViewModel = hiltViewModel(),
       onButtonClick: () -> Unit = {},
   ) {
       // ...
   }https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/interop/MigrationCommonScenariosSnippets.kt#L216-L223
   ```

   <br />

8. Replace all `findNavController()` navigation calls with the `navController`
   ones and pass these as navigation events to each composable screen, rather
   than passing the entire `navController`. This approach follows the [best
   practices](https://developer.android.com/develop/ui/compose/navigation#nav-calls-best-practices) of exposing events from composable functions to callers and
   keeps the `navController` as the single source of truth.

   Data can be passed to a destination by creating an instance of the route
   class defined for that destination. It can then be obtained either directly
   from the back stack entry at the destination or from a `ViewModel` using
   [`SavedStateHandle.toRoute()`](https://developer.android.com/reference/kotlin/androidx/lifecycle/SavedStateHandle#(androidx.lifecycle.SavedStateHandle).toRoute(kotlin.collections.Map)).


   ```kotlin
   @Composable
   fun SampleNavHost(
       navController: NavHostController
   ) {
       NavHost(navController = navController, startDestination = First) {
           composable<First> {
               FirstScreen(
                   onButtonClick = {
                       // findNavController().navigate(firstScreenToSecondScreenAction)
                       navController.navigate(Second(id = "ABC"))
                   }
               )
           }
           composable<Second> { backStackEntry ->
               val secondRoute = backStackEntry.toRoute<Second>()
               SecondScreen(
                   id = secondRoute.id,
                   onIconClick = {
                       // findNavController().navigate(secondScreenToThirdScreenAction)
                       navController.navigate(Third)
                   }
               )
           }
           // ...
       }
   }https://github.com/android/snippets/blob/1da1d9d645cd1a8e693981900e04d6bc32287a5c/compose/snippets/src/main/java/com/example/compose/snippets/interop/MigrationCommonScenariosSnippets.kt#L228-L253
   ```

   <br />

9. Remove all Fragments, relevant XML layouts, unnecessary navigation and other
   resources, and stale Fragment and Jetpack Navigation dependencies.

You can find the same steps with more Navigation Compose-related details in the
[Setup documentation](https://developer.android.com/develop/ui/compose/navigation#setup).

## Common use cases

No matter which Navigation component you're using, the [same principles of
navigation apply](https://developer.android.com/guide/navigation/navigation-principles).

Common use cases when migrating include the following:

- [Navigate to a composable](https://developer.android.com/develop/ui/compose/navigation#nav-to-composable)
- [Navigate with arguments](https://developer.android.com/develop/ui/compose/navigation#nav-with-args)
- [Deep links](https://developer.android.com/develop/ui/compose/navigation#deeplinks)
- [Nested navigation](https://developer.android.com/develop/ui/compose/navigation#nested-nav)
- [Integration with the bottom nav bar](https://developer.android.com/develop/ui/compose/navigation#bottom-nav)
- [Integration with a custom nav component](https://developer.android.com/codelabs/jetpack-compose-navigation#4)

For more detailed information about these use cases, see [Navigating with
Compose](https://developer.android.com/develop/ui/compose/navigation#interoperability).

### Retrieve complex data when navigating

We strongly recommend not passing around complex data objects when navigating.
Instead, pass the minimum necessary information, such as a unique identifier or
other form of ID, as arguments when performing navigation actions. You should
store complex objects as data in a single source of truth, such as the [data
layer](https://developer.android.com/topic/architecture#data-layer). For more information, see [Retrieving complex data when
navigating](https://developer.android.com/develop/ui/compose/navigation#retrieving-complex-data).

If your Fragments are passing complex objects as arguments, consider refactoring
your code first, in a way that allows storing and fetching these objects from
the data layer. See the [Now in Android repository](https://github.com/android/nowinandroid) for
examples.

## Limitations

This section describes current limitations for Navigation Compose.

### Incremental migration to Navigation Compose

Currently, you cannot use Navigation Compose while still using Fragments as
destinations in your code. To start using Navigation Compose, all of your
destinations need to be composables. You can track this [feature request on the
Issue Tracker](https://issuetracker.google.com/issues/265480755).

### Transition animations

Starting with [Navigation 2.7.0-alpha01](https://developer.android.com/jetpack/androidx/releases/navigation#2.7.0-alpha01), support for setting custom
transitions, previously from [`AnimatedNavHost`](https://google.github.io/accompanist/navigation-animation/), is now
directly supported in [`NavHost`](https://developer.android.com/reference/androidx/navigation/NavHost). Read through the [release notes](https://developer.android.com/jetpack/androidx/releases/navigation#2.7.0-alpha01) for
more information.

## Learn more

For more information about migrating to Navigation Compose, see the following
resources:

- [Navigation Compose codelab](https://developer.android.com/codelabs/jetpack-compose-navigation#0): Learn the basics of Navigation Compose with a hands-on codelab.
- [Now in Android repository](https://github.com/android/nowinandroid): A fully functional Android app built entirely with Kotlin and Jetpack Compose, which follows Android design and development best practices and includes Navigation Compose.
- [Migrating Sunflower to Jetpack Compose](https://medium.com/androiddevelopers/migrating-sunflower-to-jetpack-compose-f840fa3b9985): A blog post that documents the migration journey of the Sunflower sample app from Views to Compose, which also includes migration to Navigation Compose.
- [Jetnews for every screen](https://medium.com/androiddevelopers/jetnews-for-every-screen-4d8e7927752): A blog post that documents the refactor and migration of the Jetnews sample to support all screens with Jetpack Compose and Navigation Compose.

## Recommended for you

- Note: link text is displayed when JavaScript is off
- [Navigating with Compose](https://developer.android.com/develop/ui/compose/navigation)
- [Compose and other libraries](https://developer.android.com/develop/ui/compose/libraries)
- [Other considerations](https://developer.android.com/develop/ui/compose/migrate/other-considerations)