1 /* 2 * Copyright (C) 2022 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.settings.spa.app.appinfo 18 19 import android.content.pm.ApplicationInfo 20 import android.os.Bundle 21 import androidx.compose.runtime.Composable 22 import androidx.compose.runtime.remember 23 import androidx.compose.runtime.rememberCoroutineScope 24 import androidx.compose.ui.platform.LocalContext 25 import androidx.compose.ui.res.stringResource 26 import androidx.lifecycle.compose.ExperimentalLifecycleComposeApi 27 import androidx.lifecycle.compose.collectAsStateWithLifecycle 28 import androidx.navigation.NavType 29 import androidx.navigation.navArgument 30 import com.android.settings.R 31 import com.android.settingslib.spa.framework.common.SettingsPageProvider 32 import com.android.settingslib.spa.framework.compose.LifecycleEffect 33 import com.android.settingslib.spa.widget.scaffold.RegularScaffold 34 import com.android.settingslib.spaprivileged.model.app.toRoute 35 import com.android.settingslib.spaprivileged.template.app.AppInfoProvider 36 37 private const val PACKAGE_NAME = "packageName" 38 private const val USER_ID = "userId" 39 40 object CloneAppInfoSettingsProvider : SettingsPageProvider { 41 override val name = "CloneAppInfoSettingsProvider" 42 43 override val parameter = listOf( <lambda>null44 navArgument(PACKAGE_NAME) { type = NavType.StringType }, <lambda>null45 navArgument(USER_ID) { type = NavType.IntType }, 46 ) 47 48 @Composable Pagenull49 override fun Page(arguments: Bundle?) { 50 val packageName = arguments!!.getString(PACKAGE_NAME)!! 51 val userId = arguments.getInt(USER_ID) 52 val context = LocalContext.current 53 val coroutineScope = rememberCoroutineScope() 54 val packageInfoPresenter = remember { 55 PackageInfoPresenter(context, packageName, userId, coroutineScope) 56 } 57 CloneAppInfoSettings(packageInfoPresenter) 58 packageInfoPresenter.PackageRemoveDetector() 59 } 60 61 @Composable navigatornull62 fun navigator(app: ApplicationInfo) = com.android.settingslib.spa.framework.compose.navigator(route = "$name/${app.toRoute()}") 63 64 /** 65 * Gets the route to the App Info Settings page. 66 * 67 * Expose route to enable enter from non-SPA pages. 68 */ 69 fun getRoute(packageName: String, userId: Int): String = "$name/$packageName/$userId" 70 } 71 72 @OptIn(ExperimentalLifecycleComposeApi::class) 73 @Composable 74 private fun CloneAppInfoSettings(packageInfoPresenter: PackageInfoPresenter) { 75 LifecycleEffect(onStart = { packageInfoPresenter.reloadPackageInfo() }) 76 val packageInfo = packageInfoPresenter.flow.collectAsStateWithLifecycle().value ?: return 77 RegularScaffold( 78 title = stringResource(R.string.application_info_label), 79 ) { 80 val appInfoProvider = remember { AppInfoProvider(packageInfo) } 81 82 appInfoProvider.AppInfo(isClonedAppPage = true) 83 ClonePageAppButtons(packageInfoPresenter) 84 } 85 } 86