導覽是指使用者在應用程式中移動的方式。使用者通常透過輕觸或點按 UI 元素進行互動,應用程式則會回應並顯示新內容。如果使用者想返回先前的內容,可以使用返回手勢或輕觸返回按鈕。
建立導覽狀態模型
以內容堆疊模擬這類行為是個方便的做法。當使用者向前瀏覽新內容時,新內容會推送到堆疊頂端。當使用者從該內容返回時,該內容會從堆疊中移除,並顯示先前的內容。以導覽來說,這個堆疊通常稱為「返回堆疊」,因為它代表使用者可以返回的內容。

建立返回堆疊
在 Navigation 3 中,返回堆疊實際上不含任何內容。而是包含內容的參照,也就是所謂的「鍵」。鍵可以是任何類型,但通常是簡單的可序列化資料類別。使用參照而非內容有以下優點:
- 只要將鍵推送到返回堆疊,即可輕鬆瀏覽。
- 只要鍵可序列化,返回堆疊就能儲存至永久儲存空間,因此可在設定變更和程序終止後繼續留存。這點非常重要,因為使用者預期離開應用程式後,稍後返回時會看到上次離開時的內容,並從中斷的地方繼續觀看。詳情請參閱「儲存返回堆疊」。
在 Navigation 3 API 中,一個關鍵概念是「您擁有返回堆疊的控制權」。程式庫:
- 預期返回堆疊會是快照狀態支援的
List<T>
,其中T
是返回堆疊keys
的型別。您可以使用Any
,也可以提供自己更嚴格型別的金鑰。如果看到「推入」或「彈出」等字詞,表示底層實作是從清單結尾新增或移除項目。 - 觀察返回堆疊,並使用
NavDisplay
在 UI 中反映其狀態。
以下範例說明如何建立鍵和返回堆疊,以及如何因應使用者導覽事件修改返回堆疊:
// Define keys that will identify content data object ProductList data class ProductDetail(val id: String) @Composable fun MyApp() { // Create a back stack, specifying the key the app should start with val backStack = remember { mutableStateListOf<Any>(ProductList) } // Supply your back stack to a NavDisplay so it can reflect changes in the UI // ...more on this below... // Push a key onto the back stack (navigate forward), the navigation library will reflect the change in state backStack.add(ProductDetail(id = "ABC")) // Pop a key off the back stack (navigate back), the navigation library will reflect the change in state backStack.removeLastOrNull() }
解析內容的鍵
在 Navigation 3 中,內容是使用 NavEntry
建立模型,這個類別包含可組合函式。代表目的地,也就是使用者可前往和返回的單一內容。
NavEntry
也可能包含中繼資料,也就是內容的相關資訊。容器物件 (例如 NavDisplay
) 可以讀取這項中繼資料,決定如何顯示 NavEntry
的內容。舉例來說,中繼資料可用於覆寫特定 NavEntry
的預設動畫。NavEntry
metadata
是 String
鍵到 Any
值的對應,可提供多種資料儲存方式。
如要將 key
轉換為 NavEntry
,請建立項目提供者。這個函式會接受 key
,並傳回該 key
的 NavEntry
。建立 NavDisplay
時,通常會將其定義為 lambda 參數。
建立 Entry Provider 的方法有兩種:直接建立 lambda 函式,或使用 entryProvider
DSL。
直接建立 Entry Provider 函式
您通常會使用 when
陳述式建立 Entry Provider 函式,並為每個鍵建立分支。
entryProvider = { key -> when (key) { is ProductList -> NavEntry(key) { Text("Product List") } is ProductDetail -> NavEntry( key, metadata = mapOf("extraDataKey" to "extraDataValue") ) { Text("Product ${key.id} ") } else -> { NavEntry(Unit) { Text(text = "Invalid Key: $it") } } } }
使用 entryProvider
DSL
entryProvider
DSL 可簡化 lambda 函式,避免需要針對每個鍵類型進行測試,並為每個鍵類型建構 NavEntry
。請使用 entryProvider
建構函式。如果找不到索引鍵,也會包含預設的回溯行為 (擲回錯誤)。
entryProvider = entryProvider { entry<ProductList> { Text("Product List") } entry<ProductDetail>( metadata = mapOf("extraDataKey" to "extraDataValue") ) { key -> Text("Product ${key.id} ") } }
請注意程式碼片段中的下列事項:
entry
用於定義具有指定型別和可組合內容的NavEntry
entry
接受metadata
參數,可設定NavEntry.metadata
顯示返回堆疊
返回堆疊代表應用程式的導覽狀態。每當返回堆疊變更時,應用程式 UI 應反映新的返回堆疊狀態。在 Navigation 3 中,NavDisplay
會觀察返回堆疊,並據此更新 UI。使用下列參數建構:
- 返回堆疊 - 這應該是
SnapshotStateList<T>
類型,其中T
是返回堆疊鍵的類型。這是可觀察的List
,因此變更時會觸發NavDisplay
的重組。 entryProvider
,可將返回堆疊中的鍵轉換為NavEntry
物件。- 您可以選擇為
onBack
參數提供 lambda。使用者觸發返回事件時,系統會呼叫這個函式。
以下範例說明如何建立 NavDisplay
。
data object Home data class Product(val id: String) @Composable fun NavExample() { val backStack = remember { mutableStateListOf<Any>(Home) } NavDisplay( backStack = backStack, onBack = { backStack.removeLastOrNull() }, entryProvider = { key -> when (key) { is Home -> NavEntry(key) { ContentGreen("Welcome to Nav3") { Button(onClick = { backStack.add(Product("123")) }) { Text("Click to navigate") } } } is Product -> NavEntry(key) { ContentBlue("Product ${key.id} ") } else -> NavEntry(Unit) { Text("Unknown route") } } } ) }
根據預設,NavDisplay
會在單一窗格版面配置中,顯示返回堆疊最上層的 NavEntry
。下列錄影畫面顯示這個應用程式的執行情況:

NavDisplay
預設行為,有兩個目的地。平台比一比
下圖顯示 Navigation 3 中各個物件之間的資料流向:

導覽事件會啟動變更。系統會根據使用者互動,在返回堆疊中新增或移除鍵。
返回堆疊狀態變更會觸發內容擷取作業。
NavDisplay
(可組合函式,用於轉譯返回堆疊) 會觀察返回堆疊。在預設設定中,它會在單一窗格版面配置中顯示最頂層的返回堆疊項目。當返回堆疊中的頂端鍵變更時,NavDisplay
會使用這個鍵向項目供應商要求對應內容。登錄供應商提供內容。進入點供應器是將鍵解析為
NavEntry
的函式。收到NavDisplay
傳送的金鑰後,項目提供者會提供相關聯的NavEntry
,其中包含金鑰和內容。顯示內容。
NavDisplay
會接收NavEntry
並顯示內容。