Skip to content

Latest commit

 

History

History
118 lines (93 loc) · 3.66 KB

05_using_navigation_arguments.md

File metadata and controls

118 lines (93 loc) · 3.66 KB

5. Launching new destination with arguments

In this step we will open a detail view from the ListScreen and pass the id of the clicked item as a launch argument to the new screen.

Let's start by creating new screen in .ui.detail package:

.ui.detail.DetailScreen.kt:

import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.padding
import androidx.compose.material.*
import androidx.compose.runtime.Composable
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.unit.dp

@Composable
fun DetailScreen(itemId: Int) {
   Box(
      modifier = Modifier
         .fillMaxSize()
         .padding(16.dp)
   ) {
      Text(text = "Opened item with id $itemId", modifier = Modifier.align(Alignment.Center))
   }
}

Now we need to add a special destination for this screen to TutorialGraph. You have 2 options here:

  1. using next on already declared destination to build on top of it. This is helpful if you want to avoid repeating same common parts of the paths.
  2. creating a destination from scratch

.nav.TutorialGraph.kt

import com.adamkobus.compose.navigation.destination.NavStackEntry
import com.adamkobus.compose.navigation.destination.getInt

object TutorialGraph : NavGraph("tutorialGraph") {
    const val PARAM_ITEM_ID = "itemId"

    // (...)

    val List = navDestination("list")

    // Option 1: will create "list/{itemId}" destination
    val Detail = List.next { 
        param(PARAM_ITEM_ID)
    }

    // or
    
    // Option 2: will create "detail/{itemId}" destination
    val Detail = navDestination("detail") {
        param(PARAM_ITEM_ID)
    }
}

// I recommend adding extensions like this for more convenient access to param's value
fun NavStackEntry?.getItemId() = getInt(TutorialGraph.PARAM_ITEM_ID)

@ExperimentalAnimationApi
fun NavGraphBuilder.tutorialGraph() {
    composableNavigation(TutorialGraph) {
       // (...)
       
        composableDestination(TutorialGraph.Detail) { navStackEntry ->
            DetailScreen(itemId = navStackEntry.getItemId()) // getItemId() is the extension we declared above
        }
    }
}

Now all that's left is defining navigation action in TutorialNavActions and handling list element click in the ListScreen:

.nav.TutorialNavActions.kt

object TutorialNavActions {
    // (...)
    fun fromListToDetail(itemId: Int) = TutorialGraph.List goTo TutorialGraph.Detail arg itemId
}

.ui.list.ListScreen.kt

@HiltViewModel
class ListScreenVM @Inject constructor(
    private val navigationConsumer: NavigationConsumer // Added
) : ViewModel() {
    val listContent = mutableStateOf(
        // will generate 50 list elements numbered from 1 to 50
        generateSequence(1) { it + 1 }.take(50).map { ListItemData(it, "List item #$it") }.toList()
    )
    val interactions = ListScreenInteractions(
        onItemSelected = {
            viewModelScope.launch {
                navigationConsumer.offer(TutorialNavActions.fromListToDetail(it.id)) // changed, previously it was TODO
            }
        }
    )
}

And that's all. Clicking on a list element should open a new screen that displays the id of the list item that you clicked. Also notice that thanks to the verifier which we implemented in the previous step it's not possible to open multiple detail screens by pressing and releasing multiple list elements at once.