Add AppOpenByDefaultPreference for Spa
The "Open by Default" in App Info page. Bug: 236346018 Test: Manual with App Info page Test: Settings Unit tests Change-Id: I20f827241ff46bca28440b56fd32a0712ee439f9
This commit is contained in:
@@ -92,7 +92,12 @@ private fun AppInfoSettings(packageInfoPresenter: PackageInfoPresenter) {
|
|||||||
|
|
||||||
AppPermissionPreference(app)
|
AppPermissionPreference(app)
|
||||||
AppStoragePreference(app)
|
AppStoragePreference(app)
|
||||||
|
// TODO: instant_app_launch_supported_domain_urls
|
||||||
|
// TODO: data_settings
|
||||||
AppTimeSpentPreference(app)
|
AppTimeSpentPreference(app)
|
||||||
|
// TODO: battery
|
||||||
|
// TODO: app_language_setting
|
||||||
|
AppOpenByDefaultPreference(app)
|
||||||
|
|
||||||
Category(title = stringResource(R.string.advanced_apps)) {
|
Category(title = stringResource(R.string.advanced_apps)) {
|
||||||
DisplayOverOtherAppsAppListProvider.InfoPageEntryItem(app)
|
DisplayOverOtherAppsAppListProvider.InfoPageEntryItem(app)
|
||||||
|
@@ -33,7 +33,7 @@ class AppInstallButton(private val packageInfoPresenter: PackageInfoPresenter) {
|
|||||||
val app = packageInfo.applicationInfo
|
val app = packageInfo.applicationInfo
|
||||||
if (!app.isInstantApp) return null
|
if (!app.isInstantApp) return null
|
||||||
|
|
||||||
return AppStoreUtil.getAppStoreLink(packageInfoPresenter.contextAsUser, app.packageName)
|
return AppStoreUtil.getAppStoreLink(packageInfoPresenter.userContext, app.packageName)
|
||||||
?.let { intent -> installButton(intent, app) }
|
?.let { intent -> installButton(intent, app) }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -0,0 +1,92 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2022 The Android Open Source Project
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.android.settings.spa.app.appinfo
|
||||||
|
|
||||||
|
import android.app.settings.SettingsEnums
|
||||||
|
import android.content.Context
|
||||||
|
import android.content.pm.ApplicationInfo
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.runtime.livedata.observeAsState
|
||||||
|
import androidx.compose.runtime.remember
|
||||||
|
import androidx.compose.ui.platform.LocalContext
|
||||||
|
import androidx.compose.ui.res.stringResource
|
||||||
|
import androidx.lifecycle.liveData
|
||||||
|
import com.android.settings.R
|
||||||
|
import com.android.settings.applications.appinfo.AppInfoDashboardFragment
|
||||||
|
import com.android.settings.applications.intentpicker.AppLaunchSettings
|
||||||
|
import com.android.settings.applications.intentpicker.IntentPickerUtils
|
||||||
|
import com.android.settingslib.applications.AppUtils
|
||||||
|
import com.android.settingslib.spa.framework.compose.stateOf
|
||||||
|
import com.android.settingslib.spa.widget.preference.Preference
|
||||||
|
import com.android.settingslib.spa.widget.preference.PreferenceModel
|
||||||
|
import com.android.settingslib.spaprivileged.framework.common.asUser
|
||||||
|
import com.android.settingslib.spaprivileged.framework.common.domainVerificationManager
|
||||||
|
import com.android.settingslib.spaprivileged.model.app.hasFlag
|
||||||
|
import com.android.settingslib.spaprivileged.model.app.userHandle
|
||||||
|
import com.android.settingslib.spaprivileged.model.app.userId
|
||||||
|
import kotlinx.coroutines.Dispatchers
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun AppOpenByDefaultPreference(app: ApplicationInfo) {
|
||||||
|
val context = LocalContext.current
|
||||||
|
val presenter = remember { AppOpenByDefaultPresenter(context, app) }
|
||||||
|
if (!presenter.isAvailable()) return
|
||||||
|
|
||||||
|
Preference(object : PreferenceModel {
|
||||||
|
override val title = stringResource(R.string.launch_by_default)
|
||||||
|
override val summary = presenter.summaryLiveData.observeAsState(
|
||||||
|
initial = stringResource(R.string.summary_placeholder),
|
||||||
|
)
|
||||||
|
override val enabled = stateOf(presenter.isEnabled())
|
||||||
|
override val onClick = presenter::startActivity
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
private class AppOpenByDefaultPresenter(
|
||||||
|
private val context: Context,
|
||||||
|
private val app: ApplicationInfo,
|
||||||
|
) {
|
||||||
|
private val domainVerificationManager = context.asUser(app.userHandle).domainVerificationManager
|
||||||
|
|
||||||
|
fun isAvailable() =
|
||||||
|
!app.isInstantApp && !AppUtils.isBrowserApp(context, app.packageName, app.userId)
|
||||||
|
|
||||||
|
fun isEnabled() = app.hasFlag(ApplicationInfo.FLAG_INSTALLED) && app.enabled
|
||||||
|
|
||||||
|
val summaryLiveData = liveData(Dispatchers.IO) {
|
||||||
|
emit(context.getString(when {
|
||||||
|
isLinkHandlingAllowed() -> R.string.app_link_open_always
|
||||||
|
else -> R.string.app_link_open_never
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
|
fun isLinkHandlingAllowed(): Boolean {
|
||||||
|
val userState = IntentPickerUtils.getDomainVerificationUserState(
|
||||||
|
domainVerificationManager, app.packageName
|
||||||
|
)
|
||||||
|
return userState?.isLinkHandlingAllowed ?: false
|
||||||
|
}
|
||||||
|
|
||||||
|
fun startActivity() {
|
||||||
|
AppInfoDashboardFragment.startAppInfoFragment(
|
||||||
|
AppLaunchSettings::class.java,
|
||||||
|
app,
|
||||||
|
context,
|
||||||
|
SettingsEnums.APPLICATIONS_INSTALLED_APP_DETAILS,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
@@ -24,6 +24,7 @@ import androidx.lifecycle.LiveData
|
|||||||
import com.android.settings.R
|
import com.android.settings.R
|
||||||
import com.android.settingslib.applications.PermissionsSummaryHelper
|
import com.android.settingslib.applications.PermissionsSummaryHelper
|
||||||
import com.android.settingslib.applications.PermissionsSummaryHelper.PermissionsResultCallback
|
import com.android.settingslib.applications.PermissionsSummaryHelper.PermissionsResultCallback
|
||||||
|
import com.android.settingslib.spaprivileged.framework.common.asUser
|
||||||
import com.android.settingslib.spaprivileged.model.app.userHandle
|
import com.android.settingslib.spaprivileged.model.app.userHandle
|
||||||
|
|
||||||
data class AppPermissionSummaryState(
|
data class AppPermissionSummaryState(
|
||||||
@@ -35,8 +36,8 @@ class AppPermissionSummaryLiveData(
|
|||||||
private val context: Context,
|
private val context: Context,
|
||||||
private val app: ApplicationInfo,
|
private val app: ApplicationInfo,
|
||||||
) : LiveData<AppPermissionSummaryState>() {
|
) : LiveData<AppPermissionSummaryState>() {
|
||||||
private val contextAsUser = context.createContextAsUser(app.userHandle, 0)
|
private val userContext = context.asUser(app.userHandle)
|
||||||
private val packageManager = contextAsUser.packageManager
|
private val packageManager = userContext.packageManager
|
||||||
|
|
||||||
private val onPermissionsChangedListener = OnPermissionsChangedListener { uid ->
|
private val onPermissionsChangedListener = OnPermissionsChangedListener { uid ->
|
||||||
if (uid == app.uid) update()
|
if (uid == app.uid) update()
|
||||||
@@ -53,7 +54,7 @@ class AppPermissionSummaryLiveData(
|
|||||||
|
|
||||||
private fun update() {
|
private fun update() {
|
||||||
PermissionsSummaryHelper.getPermissionSummary(
|
PermissionsSummaryHelper.getPermissionSummary(
|
||||||
contextAsUser, app.packageName, permissionsCallback
|
userContext, app.packageName, permissionsCallback
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -29,6 +29,7 @@ import android.util.Log
|
|||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
import com.android.settings.overlay.FeatureFactory
|
import com.android.settings.overlay.FeatureFactory
|
||||||
import com.android.settingslib.spa.framework.compose.LocalNavController
|
import com.android.settingslib.spa.framework.compose.LocalNavController
|
||||||
|
import com.android.settingslib.spaprivileged.framework.common.asUser
|
||||||
import com.android.settingslib.spaprivileged.framework.compose.DisposableBroadcastReceiverAsUser
|
import com.android.settingslib.spaprivileged.framework.compose.DisposableBroadcastReceiverAsUser
|
||||||
import com.android.settingslib.spaprivileged.model.app.PackageManagers
|
import com.android.settingslib.spaprivileged.model.app.PackageManagers
|
||||||
import kotlinx.coroutines.CoroutineScope
|
import kotlinx.coroutines.CoroutineScope
|
||||||
@@ -49,8 +50,8 @@ class PackageInfoPresenter(
|
|||||||
private val coroutineScope: CoroutineScope,
|
private val coroutineScope: CoroutineScope,
|
||||||
) {
|
) {
|
||||||
private val metricsFeatureProvider = FeatureFactory.getFactory(context).metricsFeatureProvider
|
private val metricsFeatureProvider = FeatureFactory.getFactory(context).metricsFeatureProvider
|
||||||
val contextAsUser by lazy { context.createContextAsUser(UserHandle.of(userId), 0) }
|
val userContext by lazy { context.asUser(UserHandle.of(userId)) }
|
||||||
val packageManagerAsUser: PackageManager by lazy { contextAsUser.packageManager }
|
val packageManagerAsUser: PackageManager by lazy { userContext.packageManager }
|
||||||
private val _flow: MutableStateFlow<PackageInfo?> = MutableStateFlow(null)
|
private val _flow: MutableStateFlow<PackageInfo?> = MutableStateFlow(null)
|
||||||
|
|
||||||
val flow: StateFlow<PackageInfo?> = _flow
|
val flow: StateFlow<PackageInfo?> = _flow
|
||||||
|
@@ -35,9 +35,12 @@ android_test {
|
|||||||
"androidx.compose.ui_ui-test-manifest",
|
"androidx.compose.ui_ui-test-manifest",
|
||||||
"androidx.test.ext.junit",
|
"androidx.test.ext.junit",
|
||||||
"androidx.test.runner",
|
"androidx.test.runner",
|
||||||
"mockito-target-minus-junit4",
|
"mockito-target-inline-minus-junit4",
|
||||||
"truth-prebuilt",
|
"truth-prebuilt",
|
||||||
],
|
],
|
||||||
|
jni_libs: [
|
||||||
|
"libdexmakerjvmtiagent",
|
||||||
|
],
|
||||||
kotlincflags: [
|
kotlincflags: [
|
||||||
"-Xjvm-default=all",
|
"-Xjvm-default=all",
|
||||||
"-opt-in=kotlin.RequiresOptIn",
|
"-opt-in=kotlin.RequiresOptIn",
|
||||||
|
@@ -19,7 +19,7 @@
|
|||||||
xmlns:tools="http://schemas.android.com/tools"
|
xmlns:tools="http://schemas.android.com/tools"
|
||||||
package="com.android.settings.tests.spa_unit">
|
package="com.android.settings.tests.spa_unit">
|
||||||
|
|
||||||
<application>
|
<application android:debuggable="true">
|
||||||
<provider android:name="com.android.settings.slices.SettingsSliceProvider"
|
<provider android:name="com.android.settings.slices.SettingsSliceProvider"
|
||||||
android:authorities="${applicationId}.slices"
|
android:authorities="${applicationId}.slices"
|
||||||
tools:replace="android:authorities"/>
|
tools:replace="android:authorities"/>
|
||||||
|
@@ -0,0 +1,202 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2022 The Android Open Source Project
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.android.settings.spa.app.appinfo
|
||||||
|
|
||||||
|
import android.content.Context
|
||||||
|
import android.content.pm.ActivityInfo
|
||||||
|
import android.content.pm.ApplicationInfo
|
||||||
|
import android.content.pm.PackageManager
|
||||||
|
import android.content.pm.ResolveInfo
|
||||||
|
import android.content.pm.verify.domain.DomainVerificationManager
|
||||||
|
import android.content.pm.verify.domain.DomainVerificationUserState
|
||||||
|
import androidx.compose.runtime.CompositionLocalProvider
|
||||||
|
import androidx.compose.ui.platform.LocalContext
|
||||||
|
import androidx.compose.ui.test.assertIsDisplayed
|
||||||
|
import androidx.compose.ui.test.assertIsEnabled
|
||||||
|
import androidx.compose.ui.test.assertIsNotDisplayed
|
||||||
|
import androidx.compose.ui.test.assertIsNotEnabled
|
||||||
|
import androidx.compose.ui.test.junit4.createComposeRule
|
||||||
|
import androidx.compose.ui.test.onNodeWithText
|
||||||
|
import androidx.compose.ui.test.onRoot
|
||||||
|
import androidx.test.core.app.ApplicationProvider
|
||||||
|
import androidx.test.ext.junit.runners.AndroidJUnit4
|
||||||
|
import com.android.settings.R
|
||||||
|
import com.android.settingslib.spaprivileged.framework.common.domainVerificationManager
|
||||||
|
import org.junit.Before
|
||||||
|
import org.junit.Rule
|
||||||
|
import org.junit.Test
|
||||||
|
import org.junit.runner.RunWith
|
||||||
|
import org.mockito.Mock
|
||||||
|
import org.mockito.Mockito.any
|
||||||
|
import org.mockito.Mockito.anyInt
|
||||||
|
import org.mockito.Mockito.doReturn
|
||||||
|
import org.mockito.Spy
|
||||||
|
import org.mockito.junit.MockitoJUnit
|
||||||
|
import org.mockito.junit.MockitoRule
|
||||||
|
import org.mockito.Mockito.`when` as whenever
|
||||||
|
|
||||||
|
@RunWith(AndroidJUnit4::class)
|
||||||
|
class AppOpenByDefaultPreferenceTest {
|
||||||
|
@JvmField
|
||||||
|
@Rule
|
||||||
|
val mockito: MockitoRule = MockitoJUnit.rule()
|
||||||
|
|
||||||
|
@get:Rule
|
||||||
|
val composeTestRule = createComposeRule()
|
||||||
|
|
||||||
|
@Spy
|
||||||
|
private val context: Context = ApplicationProvider.getApplicationContext()
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
private lateinit var packageManager: PackageManager
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
private lateinit var domainVerificationManager: DomainVerificationManager
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
private lateinit var allowedUserState: DomainVerificationUserState
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
private lateinit var notAllowedUserState: DomainVerificationUserState
|
||||||
|
|
||||||
|
@Before
|
||||||
|
fun setUp() {
|
||||||
|
whenever(context.packageManager).thenReturn(packageManager)
|
||||||
|
doReturn(context).`when`(context).createContextAsUser(any(), anyInt())
|
||||||
|
whenever(context.domainVerificationManager).thenReturn(domainVerificationManager)
|
||||||
|
whenever(allowedUserState.isLinkHandlingAllowed).thenReturn(true)
|
||||||
|
whenever(notAllowedUserState.isLinkHandlingAllowed).thenReturn(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun instantApp_notDisplay() {
|
||||||
|
val instantApp = ApplicationInfo().apply {
|
||||||
|
packageName = PACKAGE_NAME
|
||||||
|
privateFlags = ApplicationInfo.PRIVATE_FLAG_INSTANT
|
||||||
|
}
|
||||||
|
|
||||||
|
composeTestRule.setContent {
|
||||||
|
CompositionLocalProvider(LocalContext provides context) {
|
||||||
|
AppOpenByDefaultPreference(instantApp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
composeTestRule.onRoot().assertIsNotDisplayed()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun browserApp_notDisplay() {
|
||||||
|
val browserApp = ApplicationInfo().apply {
|
||||||
|
packageName = PACKAGE_NAME
|
||||||
|
privateFlags = ApplicationInfo.PRIVATE_FLAG_INSTANT
|
||||||
|
}
|
||||||
|
val resolveInfo = ResolveInfo().apply {
|
||||||
|
activityInfo = ActivityInfo()
|
||||||
|
handleAllWebDataURI = true
|
||||||
|
}
|
||||||
|
whenever(packageManager.queryIntentActivitiesAsUser(any(), anyInt(), anyInt()))
|
||||||
|
.thenReturn(listOf(resolveInfo))
|
||||||
|
|
||||||
|
composeTestRule.setContent {
|
||||||
|
CompositionLocalProvider(LocalContext provides context) {
|
||||||
|
AppOpenByDefaultPreference(browserApp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
composeTestRule.onRoot().assertIsNotDisplayed()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun allowedUserState_alwaysOpen() {
|
||||||
|
whenever(domainVerificationManager.getDomainVerificationUserState(PACKAGE_NAME))
|
||||||
|
.thenReturn(allowedUserState)
|
||||||
|
|
||||||
|
composeTestRule.setContent {
|
||||||
|
CompositionLocalProvider(LocalContext provides context) {
|
||||||
|
AppOpenByDefaultPreference(INSTALLED_ENABLED_APP)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
composeTestRule.onNodeWithText(context.getString(R.string.launch_by_default))
|
||||||
|
.assertIsDisplayed()
|
||||||
|
.assertIsEnabled()
|
||||||
|
composeTestRule.onNodeWithText(context.getString(R.string.app_link_open_always))
|
||||||
|
.assertIsDisplayed()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun notAllowedUserState_neverOpen() {
|
||||||
|
whenever(domainVerificationManager.getDomainVerificationUserState(PACKAGE_NAME))
|
||||||
|
.thenReturn(notAllowedUserState)
|
||||||
|
|
||||||
|
composeTestRule.setContent {
|
||||||
|
CompositionLocalProvider(LocalContext provides context) {
|
||||||
|
AppOpenByDefaultPreference(INSTALLED_ENABLED_APP)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
composeTestRule.onNodeWithText(context.getString(R.string.launch_by_default))
|
||||||
|
.assertIsDisplayed()
|
||||||
|
.assertIsEnabled()
|
||||||
|
composeTestRule.onNodeWithText(context.getString(R.string.app_link_open_never))
|
||||||
|
.assertIsDisplayed()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun notInstalledApp_disabled() {
|
||||||
|
val notInstalledApp = ApplicationInfo().apply {
|
||||||
|
packageName = PACKAGE_NAME
|
||||||
|
}
|
||||||
|
|
||||||
|
composeTestRule.setContent {
|
||||||
|
CompositionLocalProvider(LocalContext provides context) {
|
||||||
|
AppOpenByDefaultPreference(notInstalledApp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
composeTestRule.onNodeWithText(context.getString(R.string.launch_by_default))
|
||||||
|
.assertIsNotEnabled()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun notEnabledApp_disabled() {
|
||||||
|
val notEnabledApp = ApplicationInfo().apply {
|
||||||
|
packageName = PACKAGE_NAME
|
||||||
|
flags = ApplicationInfo.FLAG_INSTALLED
|
||||||
|
enabled = false
|
||||||
|
}
|
||||||
|
|
||||||
|
composeTestRule.setContent {
|
||||||
|
CompositionLocalProvider(LocalContext provides context) {
|
||||||
|
AppOpenByDefaultPreference(notEnabledApp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
composeTestRule.onNodeWithText(context.getString(R.string.launch_by_default))
|
||||||
|
.assertIsNotEnabled()
|
||||||
|
}
|
||||||
|
|
||||||
|
private companion object {
|
||||||
|
const val PACKAGE_NAME = "package name"
|
||||||
|
|
||||||
|
val INSTALLED_ENABLED_APP = ApplicationInfo().apply {
|
||||||
|
packageName = PACKAGE_NAME
|
||||||
|
flags = ApplicationInfo.FLAG_INSTALLED
|
||||||
|
enabled = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -68,12 +68,12 @@ class AppStoragePreferenceTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun uninstalledApp_notDisplayed() {
|
fun notInstalledApp_notDisplayed() {
|
||||||
val uninstalledApp = ApplicationInfo()
|
val notInstalledApp = ApplicationInfo()
|
||||||
|
|
||||||
composeTestRule.setContent {
|
composeTestRule.setContent {
|
||||||
CompositionLocalProvider(LocalContext provides context) {
|
CompositionLocalProvider(LocalContext provides context) {
|
||||||
AppStoragePreference(uninstalledApp)
|
AppStoragePreference(notInstalledApp)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -120,15 +120,15 @@ class AppTimeSpentPreferenceTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun uninstalledApp_disabled() {
|
fun notInstalledApp_disabled() {
|
||||||
mockActivitiesQueryResult(listOf(MATCHED_RESOLVE_INFO))
|
mockActivitiesQueryResult(listOf(MATCHED_RESOLVE_INFO))
|
||||||
val uninstalledApp = ApplicationInfo().apply {
|
val notInstalledApp = ApplicationInfo().apply {
|
||||||
packageName = PACKAGE_NAME
|
packageName = PACKAGE_NAME
|
||||||
}
|
}
|
||||||
|
|
||||||
composeTestRule.setContent {
|
composeTestRule.setContent {
|
||||||
CompositionLocalProvider(LocalContext provides context) {
|
CompositionLocalProvider(LocalContext provides context) {
|
||||||
AppTimeSpentPreference(uninstalledApp)
|
AppTimeSpentPreference(notInstalledApp)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user