Skip to content

Commit

Permalink
perf: save filter settings
Browse files Browse the repository at this point in the history
  • Loading branch information
lisonge committed Oct 2, 2024
1 parent 76b3049 commit 8892dd3
Show file tree
Hide file tree
Showing 10 changed files with 110 additions and 69 deletions.
43 changes: 25 additions & 18 deletions app/src/main/kotlin/li/songe/gkd/ui/AppConfigPage.kt
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,7 @@ import li.songe.gkd.util.ResolvedGroup
import li.songe.gkd.util.RuleSortOption
import li.songe.gkd.util.appInfoCacheFlow
import li.songe.gkd.util.launchTry
import li.songe.gkd.util.storeFlow
import li.songe.gkd.util.throttle

@Destination<RootGraph>(style = ProfileTransitions::class)
Expand Down Expand Up @@ -148,12 +149,12 @@ fun AppConfigPage(appId: String) {
RadioButton(
selected = ruleSortType == s,
onClick = {
vm.ruleSortTypeFlow.update { s }
storeFlow.update { it.copy(appRuleSortType = s.value) }
}
)
},
onClick = {
vm.ruleSortTypeFlow.update { s }
storeFlow.update { it.copy(appRuleSortType = s.value) }
},
)
}
Expand All @@ -164,7 +165,8 @@ fun AppConfigPage(appId: String) {
floatingActionButton = {
FloatingActionButton(
onClick = throttle {
navController.toDestinationsNavigator().navigate(AppItemPageDestination(LOCAL_SUBS_ID, appId))
navController.toDestinationsNavigator()
.navigate(AppItemPageDestination(LOCAL_SUBS_ID, appId))
},
content = {
Icon(
Expand Down Expand Up @@ -295,7 +297,6 @@ private fun AppGroupCard(
onClick: () -> Unit,
onCheckedChange: ((Boolean) -> Unit)?,
) {
val context = LocalContext.current as MainActivity
Row(
modifier = Modifier
.clickable(onClick = onClick)
Expand Down Expand Up @@ -349,20 +350,26 @@ private fun AppGroupCard(
if (checked != null) {
Switch(checked = checked, onCheckedChange = onCheckedChange)
} else {
Switch(
checked = false,
enabled = false,
onCheckedChange = null,
modifier = Modifier.clickable(
interactionSource = remember { MutableInteractionSource() },
indication = null,
) {
context.mainVm.dialogFlow.updateDialogOptions(
title = "内置禁用",
text = "此规则组已经在其 apps 字段中配置对当前应用的禁用, 因此无法手动开启规则组\n\n提示: 这种情况一般在此全局规则无法适配/跳过适配/单独适配当前应用时出现",
)
}
)
InnerDisableSwitch()
}
}
}

@Composable
fun InnerDisableSwitch() {
val context = LocalContext.current as MainActivity
Switch(
checked = false,
enabled = false,
onCheckedChange = null,
modifier = Modifier.clickable(
interactionSource = remember { MutableInteractionSource() },
indication = null,
) {
context.mainVm.dialogFlow.updateDialogOptions(
title = "内置禁用",
text = "此规则组已经在其 apps 字段中配置对当前应用的禁用, 因此无法手动开启规则组\n\n提示: 这种情况一般在此全局规则无法适配/跳过适配/单独适配当前应用时出现",
)
}
)
}
9 changes: 6 additions & 3 deletions app/src/main/kotlin/li/songe/gkd/ui/AppConfigVm.kt
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@ import androidx.lifecycle.SavedStateHandle
import androidx.lifecycle.ViewModel
import androidx.lifecycle.viewModelScope
import com.ramcosta.composedestinations.generated.destinations.AppConfigPageDestination
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.SharingStarted
import kotlinx.coroutines.flow.combine
import kotlinx.coroutines.flow.flatMapLatest
Expand All @@ -16,11 +15,14 @@ import li.songe.gkd.util.ResolvedAppGroup
import li.songe.gkd.util.ResolvedGlobalGroup
import li.songe.gkd.util.RuleSortOption
import li.songe.gkd.util.collator
import li.songe.gkd.util.findOption
import li.songe.gkd.util.getGroupRawEnable
import li.songe.gkd.util.map
import li.songe.gkd.util.storeFlow
import li.songe.gkd.util.subsIdToRawFlow
import li.songe.gkd.util.subsItemsFlow

class AppConfigVm (stateHandle: SavedStateHandle) : ViewModel() {
class AppConfigVm(stateHandle: SavedStateHandle) : ViewModel() {
private val args = AppConfigPageDestination.argsFrom(stateHandle)

private val latestGlobalLogsFlow = DbSet.clickLogDao.queryAppLatest(
Expand All @@ -33,7 +35,8 @@ class AppConfigVm (stateHandle: SavedStateHandle) : ViewModel() {
SubsConfig.AppGroupType
)

val ruleSortTypeFlow = MutableStateFlow<RuleSortOption>(RuleSortOption.Default)
val ruleSortTypeFlow =
storeFlow.map(viewModelScope) { RuleSortOption.allSubObject.findOption(it.appRuleSortType) }

private val subsFlow = combine(subsIdToRawFlow, subsItemsFlow) { subsIdToRaw, subsItems ->
subsItems.mapNotNull { if (it.enable && subsIdToRaw[it.id] != null) it to subsIdToRaw[it.id]!! else null }
Expand Down
2 changes: 1 addition & 1 deletion app/src/main/kotlin/li/songe/gkd/ui/AuthA11yPage.kt
Original file line number Diff line number Diff line change
Expand Up @@ -259,7 +259,7 @@ fun AuthA11yPage() {
}
}

private val commandText by lazy { "adb pm grant ${META.appId} android.permission.WRITE_SECURE_SETTINGS" }
private val commandText by lazy { "adb shell pm grant ${META.appId} android.permission.WRITE_SECURE_SETTINGS" }

private suspend fun MainActivity.grantPermissionByShizuku() {
if (shizukuOkState.stateFlow.value) {
Expand Down
35 changes: 20 additions & 15 deletions app/src/main/kotlin/li/songe/gkd/ui/GlobalRuleExcludePage.kt
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,7 @@ import androidx.lifecycle.viewmodel.compose.viewModel
import com.google.accompanist.drawablepainter.rememberDrawablePainter
import com.ramcosta.composedestinations.annotation.Destination
import com.ramcosta.composedestinations.annotation.RootGraph
import kotlinx.coroutines.flow.update
import li.songe.gkd.data.AppInfo
import li.songe.gkd.data.ExcludeData
import li.songe.gkd.data.RawSubscription
Expand All @@ -79,6 +80,8 @@ import li.songe.gkd.util.LocalNavController
import li.songe.gkd.util.ProfileTransitions
import li.songe.gkd.util.SortTypeOption
import li.songe.gkd.util.launchTry
import li.songe.gkd.util.mapHashCode
import li.songe.gkd.util.storeFlow
import li.songe.gkd.util.toast

@Destination<RootGraph>(style = ProfileTransitions::class)
Expand Down Expand Up @@ -109,9 +112,16 @@ fun GlobalRuleExcludePage(subsItemId: Long, groupKey: Int) {
})
val scrollBehavior = TopAppBarDefaults.enterAlwaysScrollBehavior()
val listState = rememberLazyListState()
LaunchedEffect(key1 = showAppInfos, block = {
listState.animateScrollToItem(0)
})
var isFirstVisit by remember { mutableStateOf(true) }
LaunchedEffect(
key1 = showAppInfos.mapHashCode { it.id },
) {
if (isFirstVisit) {
isFirstVisit = false
} else {
listState.scrollToItem(0)
}
}
var expanded by remember { mutableStateOf(false) }
Scaffold(modifier = Modifier.nestedScroll(scrollBehavior.nestedScrollConnection), topBar = {
TopAppBar(scrollBehavior = scrollBehavior, navigationIcon = {
Expand Down Expand Up @@ -190,12 +200,12 @@ fun GlobalRuleExcludePage(subsItemId: Long, groupKey: Int) {
trailingIcon = {
RadioButton(selected = sortType == sortOption,
onClick = {
vm.sortTypeFlow.value = sortOption
storeFlow.update { it.copy(subsExcludeSortType = sortOption.value) }
}
)
},
onClick = {
vm.sortTypeFlow.value = sortOption
storeFlow.update { it.copy(subsExcludeSortType = sortOption.value) }
},
)
}
Expand All @@ -213,11 +223,11 @@ fun GlobalRuleExcludePage(subsItemId: Long, groupKey: Int) {
Checkbox(
checked = showSystemApp,
onCheckedChange = {
vm.showSystemAppFlow.value = !vm.showSystemAppFlow.value
storeFlow.update { it.copy(subsExcludeShowSystemApp = !it.subsExcludeShowSystemApp) }
})
},
onClick = {
vm.showSystemAppFlow.value = !vm.showSystemAppFlow.value
storeFlow.update { it.copy(subsExcludeShowSystemApp = !it.subsExcludeShowSystemApp) }
},
)
DropdownMenuItem(
Expand All @@ -228,12 +238,11 @@ fun GlobalRuleExcludePage(subsItemId: Long, groupKey: Int) {
Checkbox(
checked = showHiddenApp,
onCheckedChange = {
vm.showHiddenAppFlow.value =
!vm.showHiddenAppFlow.value
storeFlow.update { it.copy(subsExcludeShowHiddenApp = !it.subsExcludeShowHiddenApp) }
})
},
onClick = {
vm.showHiddenAppFlow.value = !vm.showHiddenAppFlow.value
storeFlow.update { it.copy(subsExcludeShowHiddenApp = !it.subsExcludeShowHiddenApp) }
},
)
}
Expand Down Expand Up @@ -334,11 +343,7 @@ fun GlobalRuleExcludePage(subsItemId: Long, groupKey: Int) {
},
)
} else {
Switch(
enabled = false,
checked = false,
onCheckedChange = {},
)
InnerDisableSwitch()
}
}
}
Expand Down
12 changes: 8 additions & 4 deletions app/src/main/kotlin/li/songe/gkd/ui/GlobalRuleExcludeVm.kt
Original file line number Diff line number Diff line change
Expand Up @@ -13,11 +13,13 @@ import kotlinx.coroutines.flow.stateIn
import li.songe.gkd.data.ExcludeData
import li.songe.gkd.db.DbSet
import li.songe.gkd.util.SortTypeOption
import li.songe.gkd.util.findOption
import li.songe.gkd.util.map
import li.songe.gkd.util.orderedAppInfosFlow
import li.songe.gkd.util.storeFlow
import li.songe.gkd.util.subsIdToRawFlow

class GlobalRuleExcludeVm (stateHandle: SavedStateHandle) : ViewModel() {
class GlobalRuleExcludeVm(stateHandle: SavedStateHandle) : ViewModel() {
private val args = GlobalRuleExcludePageDestination.argsFrom(stateHandle)

val rawSubsFlow = subsIdToRawFlow.map(viewModelScope) { it[args.subsItemId] }
Expand All @@ -40,9 +42,11 @@ class GlobalRuleExcludeVm (stateHandle: SavedStateHandle) : ViewModel() {
DbSet.clickLogDao.queryLatestUniqueAppIds(args.subsItemId, args.groupKey).map { appIds ->
appIds.mapIndexed { index, appId -> appId to index }.toMap()
}
val sortTypeFlow = MutableStateFlow<SortTypeOption>(SortTypeOption.SortByName)
val showSystemAppFlow = MutableStateFlow(true)
val showHiddenAppFlow = MutableStateFlow(false)
val sortTypeFlow = storeFlow.map(viewModelScope) {
SortTypeOption.allSubObject.findOption(it.subsExcludeSortType)
}
val showSystemAppFlow = storeFlow.map(viewModelScope) { it.subsExcludeShowSystemApp }
val showHiddenAppFlow = storeFlow.map(viewModelScope) { it.subsExcludeShowHiddenApp }
val showAppInfosFlow =
combine(orderedAppInfosFlow.combine(showHiddenAppFlow) { appInfos, showHiddenApp ->
if (showHiddenApp) {
Expand Down
36 changes: 20 additions & 16 deletions app/src/main/kotlin/li/songe/gkd/ui/SubsPage.kt
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,7 @@ import com.ramcosta.composedestinations.annotation.Destination
import com.ramcosta.composedestinations.annotation.RootGraph
import com.ramcosta.composedestinations.generated.destinations.AppItemPageDestination
import com.ramcosta.composedestinations.utils.toDestinationsNavigator
import kotlinx.coroutines.flow.update
import li.songe.gkd.MainActivity
import li.songe.gkd.data.RawSubscription
import li.songe.gkd.data.SubsConfig
Expand All @@ -70,6 +71,8 @@ import li.songe.gkd.util.appInfoCacheFlow
import li.songe.gkd.util.json
import li.songe.gkd.util.launchAsFn
import li.songe.gkd.util.launchTry
import li.songe.gkd.util.mapHashCode
import li.songe.gkd.util.storeFlow
import li.songe.gkd.util.throttle
import li.songe.gkd.util.toast
import li.songe.gkd.util.updateSubscription
Expand Down Expand Up @@ -117,16 +120,14 @@ fun SubsPage(
val showUninstallApp by vm.showUninstallAppFlow.collectAsState()
val sortType by vm.sortTypeFlow.collectAsState()
val listState = rememberLazyListState()
var isFirstVisit by remember { mutableStateOf(false) }
var isFirstVisit by remember { mutableStateOf(true) }
LaunchedEffect(
appAndConfigs.size,
sortType.value,
appAndConfigs.fold(0) { acc, t -> 31 * acc + t.t0.id.hashCode() }
key1 = appAndConfigs.mapHashCode { it.t0.id }
) {
if (isFirstVisit) {
listState.scrollToItem(0)
isFirstVisit = false
} else {
isFirstVisit = true
listState.scrollToItem(0)
}
}

Expand Down Expand Up @@ -197,11 +198,11 @@ fun SubsPage(
RadioButton(
selected = sortType == sortOption,
onClick = {
vm.sortTypeFlow.value = sortOption
storeFlow.update { s -> s.copy(subsAppSortType = sortOption.value) }
})
},
onClick = {
vm.sortTypeFlow.value = sortOption
storeFlow.update { s -> s.copy(subsAppSortType = sortOption.value) }
},
)
}
Expand All @@ -217,11 +218,11 @@ fun SubsPage(
},
trailingIcon = {
Checkbox(checked = showUninstallApp, onCheckedChange = {
vm.showUninstallAppFlow.value = it
storeFlow.update { s -> s.copy(subsAppShowUninstallApp = it) }
})
},
onClick = {
vm.showUninstallAppFlow.value = !showUninstallApp
storeFlow.update { s -> s.copy(subsAppShowUninstallApp = !showUninstallApp) }
},
)
}
Expand Down Expand Up @@ -252,7 +253,8 @@ fun SubsPage(
subsConfig = subsConfig,
enableSize = enableSize,
onClick = throttle {
navController.toDestinationsNavigator().navigate(AppItemPageDestination(subsItemId, appRaw.id))
navController.toDestinationsNavigator()
.navigate(AppItemPageDestination(subsItemId, appRaw.id))
},
onValueChange = throttle(fn = vm.viewModelScope.launchAsFn { enable ->
val newItem = subsConfig?.copy(
Expand Down Expand Up @@ -284,11 +286,13 @@ fun SubsPage(
item {
Spacer(modifier = Modifier.height(EmptyHeight))
if (appAndConfigs.isEmpty()) {
EmptyText(text = if (searchStr.isNotEmpty()) {
if (showUninstallApp) "暂无搜索结果" else "暂无搜索结果,请尝试修改筛选条件"
} else {
"暂无规则"
})
EmptyText(
text = if (searchStr.isNotEmpty()) {
if (showUninstallApp) "暂无搜索结果" else "暂无搜索结果,请尝试修改筛选条件"
} else {
"暂无规则"
}
)
} else if (editable) {
Spacer(modifier = Modifier.height(EmptyHeight))
}
Expand Down
8 changes: 5 additions & 3 deletions app/src/main/kotlin/li/songe/gkd/ui/SubsVm.kt
Original file line number Diff line number Diff line change
Expand Up @@ -17,12 +17,14 @@ import li.songe.gkd.db.DbSet
import li.songe.gkd.util.SortTypeOption
import li.songe.gkd.util.appInfoCacheFlow
import li.songe.gkd.util.collator
import li.songe.gkd.util.findOption
import li.songe.gkd.util.getGroupRawEnable
import li.songe.gkd.util.map
import li.songe.gkd.util.storeFlow
import li.songe.gkd.util.subsIdToRawFlow
import li.songe.gkd.util.subsItemsFlow

class SubsVm (stateHandle: SavedStateHandle) : ViewModel() {
class SubsVm(stateHandle: SavedStateHandle) : ViewModel() {
private val args = SubsPageDestination.argsFrom(stateHandle)

val subsItemFlow =
Expand All @@ -43,9 +45,9 @@ class SubsVm (stateHandle: SavedStateHandle) : ViewModel() {
DbSet.clickLogDao.queryLatestUniqueAppIds(args.subsItemId).map { appIds ->
appIds.mapIndexed { index, appId -> appId to index }.toMap()
}
val sortTypeFlow = MutableStateFlow<SortTypeOption>(SortTypeOption.SortByName)
val sortTypeFlow = storeFlow.map(viewModelScope) { SortTypeOption.allSubObject.findOption(it.subsAppSortType) }

val showUninstallAppFlow = MutableStateFlow(false)
val showUninstallAppFlow = storeFlow.map(viewModelScope) { it.subsAppShowUninstallApp }
private val sortAppsFlow =
combine(combine((subsRawFlow.combine(appInfoCacheFlow) { subs, appInfoCache ->
(subs?.apps ?: emptyList()).sortedWith { a, b ->
Expand Down
Loading

0 comments on commit 8892dd3

Please sign in to comment.