Compose 和其他程式庫

您可以在 Compose 中使用自己偏好的程式庫。本節將說明如何加入一些最為實用的程式庫。

Activity

如要在活動中使用 Compose,您必須使用 ComponentActivity、 是 Activity 的子類別,提供適當的 LifecycleOwner 和 部署至 Compose並提供其他 API 可將程式碼分離 在您的活動類別中覆寫方法 Activity Compose 會將向可組合項公開這些 API,這樣您就無需覆寫可組合項之外的方法,或是擷取明確的 Activity 執行個體。此外,這些 API 可確保只需將其初始化一次,在重構後仍有效,並且可在從組合內容中移除可組合項後妥善清理。

Activity 結果

rememberLauncherForActivityResult()敬上 API 可讓您 取得活動結果

@Composable
fun GetContentExample() {
    var imageUri by remember { mutableStateOf<Uri?>(null) }
    val launcher = rememberLauncherForActivityResult(ActivityResultContracts.GetContent()) { uri: Uri? ->
        imageUri = uri
    }
    Column {
        Button(onClick = { launcher.launch("image/*") }) {
            Text(text = "Load Image")
        }
        Image(
            painter = rememberAsyncImagePainter(imageUri),
            contentDescription = "My Image"
        )
    }
}

本範例說明了簡單的 GetContent() 合約。輕觸按鈕即可啟動要求。使用者選取圖片並返回正在啟動的 Activity 後,系統就會叫用 rememberLauncherForActivityResult() 的結尾 lambda。這個指令會使用 Coil 的 rememberImagePainter() 函式載入所選圖片。

的所有子類別 ActivityResultContract敬上 可以做為 rememberLauncherForActivityResult()。 也就是說,您可以利用這項技術向架構和其他常見模式要求內容。您也可以自行建立 自訂合約,並搭配

要求執行階段權限

您可以使用上述的 Activity Result API 和 rememberLauncherForActivityResult() 要求執行階段權限,使用 RequestPermission 合約可用於請求單個權限,使用 RequestMultiplePermissions 合約可用於請求多個權限。

Accompanist 權限程式庫 也可以使用這些 API 上方的圖層,對應目前授予的狀態 授予 Compose UI 可使用的權限。

處理系統返回按鈕

如要提供自訂返回導覽功能並覆寫可組合項的系統返回按鈕的預設行為,您的可組合項可以使用 BackHandler 以攔截該事件:

var backHandlingEnabled by remember { mutableStateOf(true) }
BackHandler(backHandlingEnabled) {
    // Handle back press
}

第一個引數可控制目前是否要啟用 BackHandler;您可以根據元件狀態使用此引數暫時停用處理常式。如果使用者觸發系統返回事件,且目前已啟用 BackHandler,則系統會叫用結尾 lambda。

ViewModel

如果您使用架構元件 ViewModel 程式庫,您可以存取 從任何可組合函式中的 ViewModel 呼叫 viewModel() 函式。將以下依附元件新增至 Gradle 檔案:

Groovy

dependencies {
    implementation 'androidx.lifecycle:lifecycle-viewmodel-compose:2.6.1'
}

Kotlin

dependencies {
    implementation("androidx.lifecycle:lifecycle-viewmodel-compose:2.6.1")
}

接著,您就可以在程式碼中使用 viewModel() 函式。

class MyViewModel : ViewModel() { /*...*/ }

// import androidx.lifecycle.viewmodel.compose.viewModel
@Composable
fun MyScreen(
    viewModel: MyViewModel = viewModel()
) {
    // use viewModel here
}

viewModel() 會傳回現有的 ViewModel 或建立新的。根據預設 傳回的 ViewModel 範圍僅限於封閉的活動、片段或 導覽目的地,且只要範圍維持有效,就會保留下來。

舉例來說,如果在活動中使用可組合函式,viewModel() 會傳回 相同的執行個體,直到活動完成或程序終止為止。

class MyViewModel : ViewModel() { /*...*/ }
// import androidx.lifecycle.viewmodel.compose.viewModel
@Composable
fun MyScreen(
    // Returns the same instance as long as the activity is alive,
    // just as if you grabbed the instance from an Activity or Fragment
    viewModel: MyViewModel = viewModel()
) { /* ... */ }

@Composable
fun MyScreen2(
    viewModel: MyViewModel = viewModel() // Same instance as in MyScreen
) { /* ... */ }

使用指南

您通常會在畫面層級存取 ViewModel 執行個體 可組合元件,也就是接近從活動呼叫的根可組合項 片段或導覽圖的目的地。這是因為 ViewModel 秒 預設為這些畫面層級物件。進一步瞭解 ViewModel生命週期和範圍

請盡量避免排隊 將 ViewModel 執行個體傳送至其他可組合項,因為這樣可讓這些可組合項 更難以進行測試 預覽。反之,只要傳遞資料即可 以及所需的函式

可以使用 ViewModel 例項來 管理子畫面層級可組合函式的狀態。不過,請留意 ViewModel生命週期和範圍。如果 可組合項是獨立的,建議您使用 Hilt 插入 ViewModel,以避免從父項傳遞依附元件 可組合函式

如果您的 ViewModel 有依附元件,viewModel() 會選用依附元件 ViewModelProvider.Factory敬上 做為參數

進一步瞭解 Compose 中的 ViewModel 和執行個體的使用方式 搭配使用 Navigation Compose 程式庫、活動和片段 請參閱互通性說明文件

資料串

Compose 帶有擴充功能,適用於 Android 最熱門的串流式解決方案。 這些擴充功能均由不同的構件提供:

這些構件會註冊為事件監聽器,並顯示值為 State。每當有新值時 發出後,Compose 會重組 UI 中該 state.value 的位置 例如,在下列程式碼中,ShowData 每次都會重組 exampleLiveData 會發出新的值。

// import androidx.lifecycle.viewmodel.compose.viewModel
@Composable
fun MyScreen(
    viewModel: MyViewModel = viewModel()
) {
    val dataExample = viewModel.exampleLiveData.observeAsState()

    // Because the state is read here,
    // MyScreen recomposes whenever dataExample changes.
    dataExample.value?.let {
        ShowData(dataExample)
    }
}

Compose 中的非同步作業

Jetpack Compose 可助您透過可組合項中的協同程式執行非同步作業。

詳情請參閱副作用說明文件中的 LaunchedEffectproduceStaterememberCoroutineScope API。

Navigation 元件支援 Jetpack Compose 應用程式。 請參閱「使用 Compose 進行導覽」和 詳情請參閱「將 Jetpack Navigation 遷移至 Navigation Compose」。

Hilt

我們建議您在 Android 應用程式中插入依附元件時採用 Hilt 解決方案。 可與 Compose 完美搭配運作

ViewModel 一節中提及的 viewModel() 函式 會自動使用 Hilt 透過 @HiltViewModel 建構的 ViewModel 註解。我們在說明文件中提供了 Hilt 的 ViewModel 相關資訊 整合

@HiltViewModel
class MyViewModel @Inject constructor(
    private val savedStateHandle: SavedStateHandle,
    private val repository: ExampleRepository
) : ViewModel() { /* ... */ }

// import androidx.lifecycle.viewmodel.compose.viewModel
@Composable
fun MyScreen(
    viewModel: MyViewModel = viewModel()
) { /* ... */ }

Hilt 與 Navigation

Hilt 也整合了 Navigation Compose 程式庫。將以下額外依附元件新增至 Gradle 檔案:

Groovy

dependencies {
    implementation 'androidx.hilt:hilt-navigation-compose:1.0.0'
}

Kotlin

dependencies {
    implementation("androidx.hilt:hilt-navigation-compose:1.0.0")
}

使用 Navigation Compose 程式庫時,請一律使用 hiltViewModel 可組合函式來取得帶有 @HiltViewModel 註解的 ViewModel 的執行個體。該函式可與帶有 @AndroidEntryPoint 註解的 Fragment 或 Activity 搭配使用。

例如,如果 ExampleScreen 是導覽圖表中的目的地,請呼叫 hiltViewModel() 來取得限定為目的地的 ExampleViewModel 執行個體,如以下程式碼片段所示:

// import androidx.hilt.navigation.compose.hiltViewModel

@Composable
fun MyApp() {
    val navController = rememberNavController()
    val startRoute = "example"
    NavHost(navController, startDestination = startRoute) {
        composable("example") { backStackEntry ->
            // Creates a ViewModel from the current BackStackEntry
            // Available in the androidx.hilt:hilt-navigation-compose artifact
            val viewModel = hiltViewModel<MyViewModel>()
            MyScreen(viewModel)
        }
        /* ... */
    }
}

如果您需要改為擷取範圍為導覽路徑導覽圖表ViewModel 執行個體,請使用 hiltViewModel 可組合函式,並將對應的 backStackEntry 做為參數傳遞:

// import androidx.hilt.navigation.compose.hiltViewModel
// import androidx.navigation.compose.getBackStackEntry

@Composable
fun MyApp() {
    val navController = rememberNavController()
    val startRoute = "example"
    val innerStartRoute = "exampleWithRoute"
    NavHost(navController, startDestination = startRoute) {
        navigation(startDestination = innerStartRoute, route = "Parent") {
            // ...
            composable("exampleWithRoute") { backStackEntry ->
                val parentEntry = remember(backStackEntry) {
                    navController.getBackStackEntry("Parent")
                }
                val parentViewModel = hiltViewModel<ParentViewModel>(parentEntry)
                ExampleWithRouteScreen(parentViewModel)
            }
        }
    }
}

分頁

分頁程式庫可助您更輕鬆地以漸進方式載入資料,並且 Compose 亦支援此程式庫。分頁發布頁面包含需要新增至專案及其版本的額外 paging-compose 依附元件的資訊。

以下是分頁程式庫的 Compose API 範例:

@Composable
fun MyScreen(flow: Flow<PagingData<String>>) {
    val lazyPagingItems = flow.collectAsLazyPagingItems()
    LazyColumn {
        items(
            lazyPagingItems.itemCount,
            key = lazyPagingItems.itemKey { it }
        ) { index ->
            val item = lazyPagingItems[index]
            Text("Item is $item")
        }
    }
}

請參閱清單和格線文件,瞭解關於在 Compose 中使用分頁的更多資訊。

Maps

您可以使用 Maps Compose 程式庫,以在應用程式中提供 Google 地圖。以下是使用範例:

@Composable
fun MapsExample() {
    val singapore = LatLng(1.35, 103.87)
    val cameraPositionState = rememberCameraPositionState {
        position = CameraPosition.fromLatLngZoom(singapore, 10f)
    }
    GoogleMap(
        modifier = Modifier.fillMaxSize(),
        cameraPositionState = cameraPositionState
    ) {
        Marker(
            state = MarkerState(position = singapore),
            title = "Singapore",
            snippet = "Marker in Singapore"
        )
    }
}