Fix group item

This commit is contained in:
世界 2024-03-15 11:15:07 +08:00
parent f26458ba68
commit 0c76a7479c
No known key found for this signature in database
GPG key ID: CD109927C34A63C4
10 changed files with 586 additions and 1127 deletions

View file

@ -119,9 +119,6 @@
<activity <activity
android:name="io.nekohasekai.sfa.ui.profileoverride.PerAppProxyActivity" android:name="io.nekohasekai.sfa.ui.profileoverride.PerAppProxyActivity"
android:exported="false" /> android:exported="false" />
<activity
android:name="io.nekohasekai.sfa.ui.profileoverride.PerAppProxyActivity0"
android:exported="false" />
<activity <activity
android:name="io.nekohasekai.sfa.ui.debug.DebugActivity" android:name="io.nekohasekai.sfa.ui.debug.DebugActivity"
android:exported="false" /> android:exported="false" />

View file

@ -5,7 +5,7 @@ import android.content.Context
import android.content.Intent import android.content.Intent
import android.util.Log import android.util.Log
import io.nekohasekai.sfa.database.Settings import io.nekohasekai.sfa.database.Settings
import io.nekohasekai.sfa.ui.profileoverride.PerAppProxyActivity0 import io.nekohasekai.sfa.ui.profileoverride.PerAppProxyActivity
class AppChangeReceiver : BroadcastReceiver() { class AppChangeReceiver : BroadcastReceiver() {
@ -37,7 +37,7 @@ class AppChangeReceiver : BroadcastReceiver() {
Log.d(TAG, "missing package name in intent") Log.d(TAG, "missing package name in intent")
return return
} }
val isChinaApp = PerAppProxyActivity0.scanChinaPackage(packageName) val isChinaApp = PerAppProxyActivity.scanChinaPackage(packageName)
Log.d(TAG, "scan china app result for $packageName: $isChinaApp") Log.d(TAG, "scan china app result for $packageName: $isChinaApp")
if ((perAppProxyUpdateOnChange == Settings.PER_APP_PROXY_INCLUDE) xor !isChinaApp) { if ((perAppProxyUpdateOnChange == Settings.PER_APP_PROXY_INCLUDE) xor !isChinaApp) {
Settings.perAppProxyList += packageName Settings.perAppProxyList += packageName

View file

@ -19,7 +19,9 @@ import androidx.lifecycle.lifecycleScope
import androidx.navigation.NavController import androidx.navigation.NavController
import androidx.navigation.NavDestination import androidx.navigation.NavDestination
import androidx.navigation.findNavController import androidx.navigation.findNavController
import androidx.navigation.fragment.NavHostFragment
import androidx.navigation.ui.AppBarConfiguration import androidx.navigation.ui.AppBarConfiguration
import androidx.navigation.ui.navigateUp
import androidx.navigation.ui.setupActionBarWithNavController import androidx.navigation.ui.setupActionBarWithNavController
import androidx.navigation.ui.setupWithNavController import androidx.navigation.ui.setupWithNavController
import androidx.preference.Preference import androidx.preference.Preference
@ -61,6 +63,10 @@ class MainActivity : AbstractActivity<ActivityMainBinding>(),
private const val TAG = "MainActivity" private const val TAG = "MainActivity"
} }
private lateinit var navHostFragment: NavHostFragment
private lateinit var navController: NavController
private lateinit var appBarConfiguration: AppBarConfiguration
private val connection = ServiceConnection(this, this) private val connection = ServiceConnection(this, this)
val logList = LinkedList<String>() val logList = LinkedList<String>()
@ -70,11 +76,13 @@ class MainActivity : AbstractActivity<ActivityMainBinding>(),
override fun onCreate(savedInstanceState: Bundle?) { override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState) super.onCreate(savedInstanceState)
val navController = findNavController(R.id.nav_host_fragment_activity_my) navHostFragment =
supportFragmentManager.findFragmentById(R.id.nav_host_fragment_activity_my) as NavHostFragment
navController = navHostFragment.navController
navController.setGraph(R.navigation.mobile_navigation) navController.setGraph(R.navigation.mobile_navigation)
navController.navigate(R.id.navigation_dashboard) navController.navigate(R.id.navigation_dashboard)
navController.addOnDestinationChangedListener(::onDestinationChanged) navController.addOnDestinationChangedListener(::onDestinationChanged)
val appBarConfiguration = appBarConfiguration =
AppBarConfiguration( AppBarConfiguration(
setOf( setOf(
R.id.navigation_dashboard, R.id.navigation_dashboard,
@ -85,13 +93,15 @@ class MainActivity : AbstractActivity<ActivityMainBinding>(),
) )
setupActionBarWithNavController(navController, appBarConfiguration) setupActionBarWithNavController(navController, appBarConfiguration)
binding.navView.setupWithNavController(navController) binding.navView.setupWithNavController(navController)
reconnect() reconnect()
startIntegration() startIntegration()
onNewIntent(intent) onNewIntent(intent)
} }
override fun onSupportNavigateUp(): Boolean {
return navController.navigateUp(appBarConfiguration)
}
private fun onDestinationChanged( private fun onDestinationChanged(
navController: NavController, navController: NavController,

View file

@ -12,6 +12,7 @@ import androidx.core.view.isInvisible
import androidx.core.view.isVisible import androidx.core.view.isVisible
import androidx.fragment.app.Fragment import androidx.fragment.app.Fragment
import androidx.lifecycle.lifecycleScope import androidx.lifecycle.lifecycleScope
import androidx.recyclerview.widget.GridLayoutManager
import androidx.recyclerview.widget.LinearLayoutManager import androidx.recyclerview.widget.LinearLayoutManager
import androidx.recyclerview.widget.RecyclerView import androidx.recyclerview.widget.RecyclerView
import androidx.recyclerview.widget.SimpleItemAnimator import androidx.recyclerview.widget.SimpleItemAnimator
@ -175,7 +176,7 @@ class GroupsFragment : Fragment(), CommandClient.Handler {
binding.itemList.adapter = adapter binding.itemList.adapter = adapter
(binding.itemList.itemAnimator as SimpleItemAnimator).supportsChangeAnimations = (binding.itemList.itemAnimator as SimpleItemAnimator).supportsChangeAnimations =
false false
binding.itemList.layoutManager = LinearLayoutManager(binding.root.context) binding.itemList.layoutManager = GridLayoutManager(binding.root.context, 2)
} else { } else {
adapter.setItems(items) adapter.setItems(items)
} }

View file

@ -2,23 +2,20 @@ package io.nekohasekai.sfa.ui.profileoverride
import android.Manifest import android.Manifest
import android.annotation.SuppressLint import android.annotation.SuppressLint
import android.app.Dialog
import android.content.ClipboardManager
import android.content.pm.ApplicationInfo import android.content.pm.ApplicationInfo
import android.content.pm.PackageInfo
import android.content.pm.PackageManager import android.content.pm.PackageManager
import android.content.pm.PackageManager.PackageInfoFlags
import android.graphics.drawable.Drawable
import android.os.Build import android.os.Build
import android.os.Bundle import android.os.Bundle
import android.util.Log import android.view.Gravity
import android.view.LayoutInflater import android.view.LayoutInflater
import android.view.Menu import android.view.Menu
import android.view.MenuItem import android.view.MenuItem
import android.view.ViewGroup import android.view.ViewGroup
import android.widget.TextView
import android.widget.Toast import android.widget.Toast
import androidx.appcompat.widget.PopupMenu
import androidx.appcompat.widget.SearchView import androidx.appcompat.widget.SearchView
import androidx.core.content.getSystemService import androidx.core.view.isVisible
import androidx.lifecycle.lifecycleScope import androidx.lifecycle.lifecycleScope
import androidx.recyclerview.widget.RecyclerView import androidx.recyclerview.widget.RecyclerView
import com.google.android.material.dialog.MaterialAlertDialogBuilder import com.google.android.material.dialog.MaterialAlertDialogBuilder
@ -26,10 +23,12 @@ import io.nekohasekai.sfa.Application
import io.nekohasekai.sfa.R import io.nekohasekai.sfa.R
import io.nekohasekai.sfa.database.Settings import io.nekohasekai.sfa.database.Settings
import io.nekohasekai.sfa.databinding.ActivityPerAppProxyBinding import io.nekohasekai.sfa.databinding.ActivityPerAppProxyBinding
import io.nekohasekai.sfa.databinding.DialogProgressbarBinding
import io.nekohasekai.sfa.databinding.ViewAppListItemBinding import io.nekohasekai.sfa.databinding.ViewAppListItemBinding
import io.nekohasekai.sfa.ktx.clipboardText import io.nekohasekai.sfa.ktx.clipboardText
import io.nekohasekai.sfa.ui.shared.AbstractActivity import io.nekohasekai.sfa.ui.shared.AbstractActivity
import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.delay
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import kotlinx.coroutines.withContext import kotlinx.coroutines.withContext
import org.jf.dexlib2.dexbacked.DexBackedDexFile import org.jf.dexlib2.dexbacked.DexBackedDexFile
@ -37,114 +36,266 @@ import java.io.File
import java.util.zip.ZipFile import java.util.zip.ZipFile
class PerAppProxyActivity : AbstractActivity<ActivityPerAppProxyBinding>() { class PerAppProxyActivity : AbstractActivity<ActivityPerAppProxyBinding>() {
enum class SortMode {
NAME, PACKAGE_NAME, UID, INSTALL_TIME, UPDATE_TIME,
}
private var proxyMode = Settings.PER_APP_PROXY_INCLUDE
private var sortMode = SortMode.NAME
private var sortReverse = false
private var hideSystemApps = false
private var hideOfflineApps = true
private var hideDisabledApps = true
private lateinit var adapter: AppListAdapter inner class PackageCache(
private val packageInfo: PackageInfo,
) {
private val perAppProxyList = mutableSetOf<String>() val packageName: String get() = packageInfo.packageName
private val appList = mutableListOf<AppItem>()
private var hideSystem = false val uid get() = packageInfo.applicationInfo.uid
private var searchKeyword = ""
private val filteredAppList = mutableListOf<AppItem>() val installTime get() = packageInfo.firstInstallTime
val updateTime get() = packageInfo.lastUpdateTime
val isSystem get() = packageInfo.applicationInfo.flags and ApplicationInfo.FLAG_SYSTEM == 1
val isOffline get() = packageInfo.requestedPermissions?.contains(Manifest.permission.INTERNET) != true
val isDisabled get() = packageInfo.applicationInfo.flags and ApplicationInfo.FLAG_INSTALLED == 0
val applicationIcon by lazy {
packageInfo.applicationInfo.loadIcon(packageManager)
}
val applicationLabel by lazy {
packageInfo.applicationInfo.loadLabel(packageManager).toString()
}
}
private lateinit var adapter: ApplicationAdapter
private var packages = listOf<PackageCache>()
private var displayPackages = listOf<PackageCache>()
private var currentPackages = listOf<PackageCache>()
private var selectedUIDs = mutableSetOf<Int>()
override fun onCreate(savedInstanceState: Bundle?) { override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState) super.onCreate(savedInstanceState)
setTitle(R.string.title_per_app_proxy) setTitle(R.string.title_per_app_proxy)
val proxyMode = Settings.perAppProxyMode
if (proxyMode == Settings.PER_APP_PROXY_INCLUDE) {
binding.radioPerAppInclude.isChecked = true
} else {
binding.radioPerAppExclude.isChecked = true
}
binding.radioGroupPerAppMode.setOnCheckedChangeListener { _, checkedId ->
if (checkedId == R.id.radio_per_app_include) {
Settings.perAppProxyMode = Settings.PER_APP_PROXY_INCLUDE
} else {
Settings.perAppProxyMode = Settings.PER_APP_PROXY_EXCLUDE
}
}
perAppProxyList.addAll(Settings.perAppProxyList.toMutableSet())
adapter = AppListAdapter(filteredAppList) {
val item = filteredAppList[it]
if (item.selected) {
perAppProxyList.add(item.packageName)
} else {
perAppProxyList.remove(item.packageName)
}
Settings.perAppProxyList = perAppProxyList
}
binding.recyclerViewAppList.adapter = adapter
loadAppList()
}
@SuppressLint("NotifyDataSetChanged")
private fun loadAppList() {
// binding.recyclerViewAppList.isGone = true
// binding.layoutProgress.isGone = false
lifecycleScope.launch { lifecycleScope.launch {
val list = withContext(Dispatchers.IO) { proxyMode = if (Settings.perAppProxyMode == Settings.PER_APP_PROXY_EXCLUDE) {
val flag = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { Settings.PER_APP_PROXY_EXCLUDE
} else {
Settings.PER_APP_PROXY_INCLUDE
}
withContext(Dispatchers.Main) {
if (proxyMode != Settings.PER_APP_PROXY_EXCLUDE) {
binding.perAppProxyMode.setText(R.string.per_app_proxy_mode_include_description)
} else {
binding.perAppProxyMode.setText(R.string.per_app_proxy_mode_exclude_description)
}
}
reloadApplicationList()
filterApplicationList()
withContext(Dispatchers.Main) {
adapter = ApplicationAdapter(displayPackages)
binding.appList.adapter = adapter
delay(500L)
binding.progress.isVisible = false
}
}
}
private fun reloadApplicationList() {
val packageManagerFlags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
PackageManager.GET_PERMISSIONS or PackageManager.MATCH_UNINSTALLED_PACKAGES PackageManager.GET_PERMISSIONS or PackageManager.MATCH_UNINSTALLED_PACKAGES
} else { } else {
@Suppress("DEPRECATION") @Suppress("DEPRECATION") PackageManager.GET_PERMISSIONS or PackageManager.GET_UNINSTALLED_PACKAGES
PackageManager.GET_PERMISSIONS or PackageManager.GET_UNINSTALLED_PACKAGES
} }
val installedPackages = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) { val installedPackages = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
packageManager.getInstalledPackages(PackageInfoFlags.of(flag.toLong())) packageManager.getInstalledPackages(
PackageManager.PackageInfoFlags.of(
packageManagerFlags.toLong()
)
)
} else { } else {
@Suppress("DEPRECATION") @Suppress("DEPRECATION") packageManager.getInstalledPackages(packageManagerFlags)
packageManager.getInstalledPackages(flag)
} }
val list = mutableListOf<AppItem>() val packages = mutableListOf<PackageCache>()
installedPackages.forEach { for (packageInfo in installedPackages) {
if (it.packageName != packageName && if (packageInfo.packageName == packageName) continue
(it.requestedPermissions?.contains(Manifest.permission.INTERNET) == true packages.add(PackageCache(packageInfo))
|| it.packageName == "android") }
val selectedPackageNames = Settings.perAppProxyList.toMutableSet()
val selectedUIDs = mutableSetOf<Int>()
for (packageCache in packages) {
if (selectedPackageNames.contains(packageCache.packageName)) {
selectedUIDs.add(packageCache.uid)
}
}
this.packages = packages
this.selectedUIDs = selectedUIDs
}
private fun filterApplicationList(selectedUIDs: Set<Int> = this.selectedUIDs) {
val displayPackages = mutableListOf<PackageCache>()
for (packageCache in packages) {
if (hideSystemApps && packageCache.isSystem) continue
if (hideOfflineApps && packageCache.isOffline) continue
if (hideDisabledApps && packageCache.isDisabled) continue
displayPackages.add(packageCache)
}
displayPackages.sortWith(compareBy<PackageCache> {
!selectedUIDs.contains(it.uid)
}.let {
if (!sortReverse) it.thenBy {
when (sortMode) {
SortMode.NAME -> it.applicationLabel
SortMode.PACKAGE_NAME -> it.packageName
SortMode.UID -> it.uid
SortMode.INSTALL_TIME -> it.installTime
SortMode.UPDATE_TIME -> it.updateTime
}
} else it.thenByDescending {
when (sortMode) {
SortMode.NAME -> it.applicationLabel
SortMode.PACKAGE_NAME -> it.packageName
SortMode.UID -> it.uid
SortMode.INSTALL_TIME -> it.installTime
SortMode.UPDATE_TIME -> it.updateTime
}
}
})
this.displayPackages = displayPackages
this.currentPackages = displayPackages
}
private fun updateApplicationSelection(packageCache: PackageCache, selected: Boolean) {
val performed = if (selected) {
selectedUIDs.add(packageCache.uid)
} else {
selectedUIDs.remove(packageCache.uid)
}
if (!performed) return
currentPackages.forEachIndexed { index, it ->
if (it.uid == packageCache.uid) {
adapter.notifyItemChanged(index, PayloadUpdateSelection(selected))
}
}
saveSelectedApplications()
}
data class PayloadUpdateSelection(val selected: Boolean)
inner class ApplicationAdapter(private var applicationList: List<PackageCache>) :
RecyclerView.Adapter<ApplicationViewHolder>() {
@SuppressLint("NotifyDataSetChanged")
fun setApplicationList(applicationList: List<PackageCache>) {
this.applicationList = applicationList
notifyDataSetChanged()
}
override fun onCreateViewHolder(
parent: ViewGroup, viewType: Int
): ApplicationViewHolder {
return ApplicationViewHolder(
ViewAppListItemBinding.inflate(
LayoutInflater.from(parent.context), parent, false
)
)
}
override fun getItemCount(): Int {
return applicationList.size
}
override fun onBindViewHolder(
holder: ApplicationViewHolder, position: Int
) { ) {
list.add( holder.bind(applicationList[position])
AppItem(
it.packageName,
it.applicationInfo.loadLabel(packageManager).toString(),
it.applicationInfo.loadIcon(packageManager),
it.applicationInfo.flags and ApplicationInfo.FLAG_SYSTEM == 1,
perAppProxyList.contains(it.packageName)
)
)
} }
}
list.sortedWith(compareBy({ !it.selected }, { it.name }))
}
appList.clear()
appList.addAll(list)
perAppProxyList.toSet().forEach { override fun onBindViewHolder(
if (appList.find { app -> app.packageName == it } == null) { holder: ApplicationViewHolder, position: Int, payloads: MutableList<Any>
perAppProxyList.remove(it) ) {
if (payloads.isEmpty()) {
onBindViewHolder(holder, position)
return
}
payloads.forEach {
when (it) {
is PayloadUpdateSelection -> holder.updateSelection(it.selected)
}
}
} }
} }
Settings.perAppProxyList = perAppProxyList
filteredAppList.clear() inner class ApplicationViewHolder(
if (hideSystem) { private val binding: ViewAppListItemBinding
filteredAppList.addAll(appList.filter { !it.isSystemApp }) ) : RecyclerView.ViewHolder(binding.root) {
@SuppressLint("SetTextI18n")
fun bind(packageCache: PackageCache) {
binding.appIcon.setImageDrawable(packageCache.applicationIcon)
binding.applicationLabel.text = packageCache.applicationLabel
binding.packageName.text = "${packageCache.packageName} (${packageCache.uid})"
binding.selected.isChecked = selectedUIDs.contains(packageCache.uid)
binding.root.setOnClickListener {
updateApplicationSelection(packageCache, !binding.selected.isChecked)
}
binding.root.setOnLongClickListener {
val popup = PopupMenu(it.context, it)
popup.setForceShowIcon(true)
popup.gravity = Gravity.END
popup.menuInflater.inflate(R.menu.app_menu, popup.menu)
popup.setOnMenuItemClickListener {
when (it.itemId) {
R.id.action_copy_application_label -> {
clipboardText = packageCache.applicationLabel
true
}
R.id.action_copy_package_name -> {
clipboardText = packageCache.packageName
true
}
R.id.action_copy_uid -> {
clipboardText = packageCache.uid.toString()
true
}
else -> false
}
}
popup.show()
true
}
}
fun updateSelection(selected: Boolean) {
binding.selected.isChecked = selected
}
}
private fun searchApplications(searchText: String) {
currentPackages = if (searchText.isEmpty()) {
displayPackages
} else { } else {
filteredAppList.addAll(appList) displayPackages.filter {
it.applicationLabel.contains(
searchText, ignoreCase = true
) || it.packageName.contains(
searchText, ignoreCase = true
) || it.uid.toString().contains(searchText)
} }
adapter.notifyDataSetChanged()
binding.recyclerViewAppList.scrollToPosition(0)
// binding.layoutProgress.isGone = true
// binding.recyclerViewAppList.isGone = false
} }
adapter.setApplicationList(currentPackages)
} }
override fun onCreateOptionsMenu(menu: Menu?): Boolean { override fun onCreateOptionsMenu(menu: Menu?): Boolean {
menuInflater.inflate(R.menu.per_app_menu, menu) menuInflater.inflate(R.menu.per_app_menu0, menu)
if (menu != null) { if (menu != null) {
val searchView = menu.findItem(R.id.action_search).actionView as SearchView val searchView = menu.findItem(R.id.action_search).actionView as SearchView
@ -154,225 +305,336 @@ class PerAppProxyActivity : AbstractActivity<ActivityPerAppProxyBinding>() {
} }
override fun onQueryTextChange(newText: String): Boolean { override fun onQueryTextChange(newText: String): Boolean {
searchKeyword = newText searchApplications(newText)
filterApps()
return true return true
} }
}) })
searchView.setOnCloseListener {
searchApplications("")
true
}
when (proxyMode) {
Settings.PER_APP_PROXY_INCLUDE -> {
menu.findItem(R.id.action_mode_include).isChecked = true
}
Settings.PER_APP_PROXY_EXCLUDE -> {
menu.findItem(R.id.action_mode_exclude).isChecked = true
}
}
when (sortMode) {
SortMode.NAME -> {
menu.findItem(R.id.action_sort_by_name).isChecked = true
}
SortMode.PACKAGE_NAME -> {
menu.findItem(R.id.action_sort_by_package_name).isChecked = true
}
SortMode.UID -> {
menu.findItem(R.id.action_sort_by_uid).isChecked = true
}
SortMode.INSTALL_TIME -> {
menu.findItem(R.id.action_sort_by_install_time).isChecked = true
}
SortMode.UPDATE_TIME -> {
menu.findItem(R.id.action_sort_by_update_time).isChecked = true
}
}
menu.findItem(R.id.action_sort_reverse).isChecked = sortReverse
menu.findItem(R.id.action_hide_system_apps).isChecked = hideSystemApps
menu.findItem(R.id.action_hide_offline_apps).isChecked = hideOfflineApps
menu.findItem(R.id.action_hide_disabled_apps).isChecked = hideDisabledApps
} }
return super.onCreateOptionsMenu(menu) return super.onCreateOptionsMenu(menu)
} }
@SuppressLint("NotifyDataSetChanged")
override fun onOptionsItemSelected(item: MenuItem): Boolean { override fun onOptionsItemSelected(item: MenuItem): Boolean {
when (item.itemId) { when (item.itemId) {
R.id.action_hide_system -> { R.id.action_mode_include -> {
hideSystem = !hideSystem item.isChecked = true
if (hideSystem) { proxyMode = Settings.PER_APP_PROXY_INCLUDE
item.setTitle(R.string.menu_show_system) binding.perAppProxyMode.setText(R.string.per_app_proxy_mode_include_description)
} else { lifecycleScope.launch {
item.setTitle(R.string.menu_hide_system) Settings.perAppProxyMode = Settings.PER_APP_PROXY_INCLUDE
} }
filterApps()
return true
} }
R.id.action_import -> { R.id.action_mode_exclude -> {
MaterialAlertDialogBuilder(this) item.isChecked = true
.setTitle(R.string.per_app_proxy_import) proxyMode = Settings.PER_APP_PROXY_EXCLUDE
.setMessage(R.string.message_import_from_clipboard) binding.perAppProxyMode.setText(R.string.per_app_proxy_mode_exclude_description)
.setPositiveButton(R.string.ok) { _, _ -> lifecycleScope.launch {
importFromClipboard() Settings.perAppProxyMode = Settings.PER_APP_PROXY_EXCLUDE
} }
.setNegativeButton(android.R.string.cancel, null) }
.show()
return true R.id.action_sort_by_name -> {
item.isChecked = true
sortMode = SortMode.NAME
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_sort_by_package_name -> {
item.isChecked = true
sortMode = SortMode.PACKAGE_NAME
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_sort_by_uid -> {
item.isChecked = true
sortMode = SortMode.UID
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_sort_by_install_time -> {
item.isChecked = true
sortMode = SortMode.INSTALL_TIME
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_sort_by_update_time -> {
item.isChecked = true
sortMode = SortMode.UPDATE_TIME
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_sort_reverse -> {
item.isChecked = !item.isChecked
sortReverse = item.isChecked
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_hide_system_apps -> {
item.isChecked = !item.isChecked
hideSystemApps = item.isChecked
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_hide_offline_apps -> {
item.isChecked = !item.isChecked
hideOfflineApps = item.isChecked
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_hide_disabled_apps -> {
item.isChecked = !item.isChecked
hideDisabledApps = item.isChecked
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_select_all -> {
val selectedUIDs = mutableSetOf<Int>()
for (packageCache in packages) {
selectedUIDs.add(packageCache.uid)
}
this.selectedUIDs = selectedUIDs
saveSelectedApplications()
}
R.id.action_deselect_all -> {
selectedUIDs = mutableSetOf()
saveSelectedApplications()
} }
R.id.action_export -> { R.id.action_export -> {
exportToClipboard() lifecycleScope.launch {
val packageList = mutableListOf<String>()
for (packageCache in packages) {
if (selectedUIDs.contains(packageCache.uid)) {
packageList.add(packageCache.packageName)
}
}
clipboardText = packageList.joinToString("\n")
withContext(Dispatchers.Main) {
Toast.makeText(
this@PerAppProxyActivity,
R.string.toast_copied_to_clipboard,
Toast.LENGTH_SHORT
).show()
}
}
}
R.id.action_import -> {
val packageNames = clipboardText?.split("\n")?.distinct()
?.takeIf { it.isNotEmpty() && it[0].isNotEmpty() }
if (packageNames.isNullOrEmpty()) {
Toast.makeText(
this@PerAppProxyActivity,
R.string.toast_clipboard_empty,
Toast.LENGTH_SHORT
).show()
return true return true
} }
val selectedUIDs = mutableSetOf<Int>()
for (packageCache in packages) {
if (packageNames.contains(packageCache.packageName)) {
selectedUIDs.add(packageCache.uid)
}
}
lifecycleScope.launch {
postSaveSelectedApplications(selectedUIDs)
withContext(Dispatchers.Main) {
Toast.makeText(
this@PerAppProxyActivity,
R.string.toast_imported_from_clipboard,
Toast.LENGTH_SHORT
).show()
}
}
}
R.id.action_scan_china_apps -> { R.id.action_scan_china_apps -> {
scanChinaApps() scanChinaApps()
}
}
return true return true
} }
}
return super.onOptionsItemSelected(item)
}
@SuppressLint("NotifyDataSetChanged") @SuppressLint("NotifyDataSetChanged")
private fun filterApps() {
filteredAppList.clear()
if (searchKeyword.isNotEmpty()) {
filteredAppList.addAll(appList.filter {
(!hideSystem || !it.isSystemApp) &&
(it.name.contains(searchKeyword, true)
|| it.packageName.contains(searchKeyword, true))
})
adapter.notifyDataSetChanged()
} else if (hideSystem) {
filteredAppList.addAll(appList.filter { !it.isSystemApp })
} else {
filteredAppList.addAll(appList)
}
adapter.notifyDataSetChanged()
}
private fun importFromClipboard() {
val clipboardManager = getSystemService<ClipboardManager>()!!
if (!clipboardManager.hasPrimaryClip()) {
Toast.makeText(this, R.string.toast_clipboard_empty, Toast.LENGTH_SHORT).show()
return
}
val content = clipboardManager.primaryClip?.getItemAt(0)?.text?.split("\n")?.distinct()
if (content.isNullOrEmpty()) {
Toast.makeText(this, R.string.toast_clipboard_empty, Toast.LENGTH_SHORT).show()
return
}
perAppProxyList.clear()
perAppProxyList.addAll(content)
loadAppList()
Toast.makeText(this, R.string.toast_imported_from_clipboard, Toast.LENGTH_SHORT).show()
}
private fun exportToClipboard() {
if (perAppProxyList.isEmpty()) {
Toast.makeText(this, R.string.toast_app_list_empty, Toast.LENGTH_SHORT).show()
return
}
val content = perAppProxyList.joinToString("\n")
clipboardText = content
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.TIRAMISU) {
Toast.makeText(this, R.string.toast_copied_to_clipboard, Toast.LENGTH_SHORT).show()
}
}
private fun scanChinaApps() { private fun scanChinaApps() {
val progressDialog = MaterialAlertDialogBuilder(this) val binding = DialogProgressbarBinding.inflate(layoutInflater)
.setView(R.layout.dialog_progress) binding.progress.max = currentPackages.size
.setCancelable(false) binding.message.setText(R.string.message_scanning)
.create() val progress = MaterialAlertDialogBuilder(
progressDialog.setOnShowListener { this, com.google.android.material.R.style.Theme_MaterialComponents_Dialog
val dialog = it as Dialog ).setView(binding.root).setCancelable(false).create()
dialog.findViewById<TextView>(R.id.text_message).setText(R.string.message_scanning) progress.show()
}
progressDialog.show()
lifecycleScope.launch { lifecycleScope.launch {
val scanResult = withContext(Dispatchers.IO) { val foundApps = withContext(Dispatchers.IO) {
val appNameMap = mutableMapOf<String, String>() mutableMapOf<String, PackageCache>().also { foundApps ->
appList.forEach { currentPackages.forEachIndexed { index, it ->
appNameMap[it.packageName] = it.name if (scanChinaPackage(it.packageName)) {
foundApps[it.packageName] = it
} }
val foundChinaApps = mutableMapOf<String, String>() withContext(Dispatchers.Main) {
scanChinaApps(appList.map { it.packageName }).forEach { packageName -> binding.progress.progress = index + 1
foundChinaApps[packageName] = appNameMap[packageName] ?: "Unknown"
} }
foundChinaApps
} }
progressDialog.dismiss() }
}
if (scanResult.isEmpty()) { withContext(Dispatchers.Main) {
MaterialAlertDialogBuilder(this@PerAppProxyActivity) progress.dismiss()
.setTitle(R.string.title_scan_result) if (foundApps.isEmpty()) {
MaterialAlertDialogBuilder(this@PerAppProxyActivity).setTitle(R.string.title_scan_result)
.setMessage(R.string.message_scan_app_no_apps_found) .setMessage(R.string.message_scan_app_no_apps_found)
.setPositiveButton(R.string.ok, null) .setPositiveButton(R.string.ok, null).show()
.show() return@withContext
return@launch
} }
val dialogContent =
val dialogContent = getString(R.string.message_scan_app_found) + "\n\n" + getString(R.string.message_scan_app_found) + "\n\n" + foundApps.entries.joinToString(
scanResult.entries.joinToString("\n") { "\n"
"${it.value} (${it.key})" ) {
"${it.value.applicationLabel} (${it.key})"
} }
MaterialAlertDialogBuilder(this@PerAppProxyActivity) MaterialAlertDialogBuilder(this@PerAppProxyActivity).setTitle(R.string.title_scan_result)
.setTitle(R.string.title_scan_result)
.setMessage(dialogContent) .setMessage(dialogContent)
.setPositiveButton(R.string.action_select) { dialog, _ -> .setPositiveButton(R.string.action_select) { dialog, _ ->
perAppProxyList.addAll(scanResult.keys)
Settings.perAppProxyList = perAppProxyList
loadAppList()
dialog.dismiss() dialog.dismiss()
lifecycleScope.launch {
val selectedUIDs = selectedUIDs.toMutableSet()
foundApps.values.forEach {
selectedUIDs.add(it.uid)
} }
.setNegativeButton(R.string.action_deselect) { dialog, _ -> postSaveSelectedApplications(selectedUIDs)
perAppProxyList.removeAll(scanResult.keys) }
Settings.perAppProxyList = perAppProxyList }.setNegativeButton(R.string.action_deselect) { dialog, _ ->
loadAppList()
dialog.dismiss() dialog.dismiss()
lifecycleScope.launch {
val selectedUIDs = selectedUIDs.toMutableSet()
foundApps.values.forEach {
selectedUIDs.remove(it.uid)
} }
.setNeutralButton(android.R.string.cancel, null) postSaveSelectedApplications(selectedUIDs)
.show() }
}.setNeutralButton(android.R.string.cancel, null).show()
}
}
}
@SuppressLint("NotifyDataSetChanged")
private suspend fun postSaveSelectedApplications(newUIDs: MutableSet<Int>) {
filterApplicationList(newUIDs)
withContext(Dispatchers.Main) {
selectedUIDs = newUIDs
adapter.notifyDataSetChanged()
}
val packageList = selectedUIDs.mapNotNull { uid ->
packages.find { it.uid == uid }?.packageName
}
Settings.perAppProxyList = packageList.toSet()
}
private fun saveSelectedApplications() {
lifecycleScope.launch {
val packageList = selectedUIDs.mapNotNull { uid ->
packages.find { it.uid == uid }?.packageName
}
Settings.perAppProxyList = packageList.toSet()
} }
} }
companion object { companion object {
private const val TAG = "PerAppProxyActivity"
fun scanChinaApps(packageNameList: List<String>): List<String> { private val chinaAppPrefixList by lazy {
val chinaAppPrefixList = try { runCatching {
Application.application.assets.open("prefix-china-apps.txt").reader().readLines() Application.application.assets.open("prefix-china-apps.txt").reader().readLines()
} catch (e: Exception) { }.getOrNull() ?: emptyList()
Log.w(
TAG,
"scan china apps: failed to load prefix from assets, error = ${e.message}"
)
null
} }
if (chinaAppPrefixList.isNullOrEmpty()) {
return listOf() private val chinaAppRegex by lazy {
}
val chinaAppRegex =
("(" + chinaAppPrefixList.joinToString("|").replace(".", "\\.") + ").*").toRegex() ("(" + chinaAppPrefixList.joinToString("|").replace(".", "\\.") + ").*").toRegex()
val packageManagerFlags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
PackageManager.MATCH_UNINSTALLED_PACKAGES or
PackageManager.GET_ACTIVITIES or
PackageManager.GET_SERVICES or
PackageManager.GET_RECEIVERS or
PackageManager.GET_PROVIDERS
} else {
@Suppress("DEPRECATION")
PackageManager.GET_UNINSTALLED_PACKAGES or
PackageManager.GET_ACTIVITIES or
PackageManager.GET_SERVICES or
PackageManager.GET_RECEIVERS or
PackageManager.GET_PROVIDERS
} }
val foundChinaApps = mutableListOf<String>()
for (packageName in packageNameList) { fun scanChinaPackage(packageName: String): Boolean {
if (packageName == "android") { val packageManagerFlags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
continue PackageManager.MATCH_UNINSTALLED_PACKAGES or PackageManager.GET_ACTIVITIES or PackageManager.GET_SERVICES or PackageManager.GET_RECEIVERS or PackageManager.GET_PROVIDERS
} else {
@Suppress("DEPRECATION") PackageManager.GET_UNINSTALLED_PACKAGES or PackageManager.GET_ACTIVITIES or PackageManager.GET_SERVICES or PackageManager.GET_RECEIVERS or PackageManager.GET_PROVIDERS
} }
if (packageName.matches(chinaAppRegex)) { if (packageName.matches(chinaAppRegex)) {
foundChinaApps.add(packageName) return true
continue
} }
try { try {
val packageInfo = val packageInfo = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
Application.packageManager.getPackageInfo( Application.packageManager.getPackageInfo(
packageName, packageName,
PackageInfoFlags.of(packageManagerFlags.toLong()) PackageManager.PackageInfoFlags.of(packageManagerFlags.toLong())
) )
} else { } else {
@Suppress("DEPRECATION") @Suppress("DEPRECATION") Application.packageManager.getPackageInfo(
Application.packageManager.getPackageInfo( packageName, packageManagerFlags
packageName,
packageManagerFlags
) )
} }
if (packageInfo.services?.find { it.name.matches(chinaAppRegex) } != null || packageInfo.activities?.find {
if (packageInfo.services?.find { it.name.matches(chinaAppRegex) } != null it.name.matches(
|| packageInfo.activities?.find { it.name.matches(chinaAppRegex) } != null chinaAppRegex
|| packageInfo.receivers?.find { it.name.matches(chinaAppRegex) } != null )
|| packageInfo.providers?.find { it.name.matches(chinaAppRegex) } != null) { } != null || packageInfo.receivers?.find { it.name.matches(chinaAppRegex) } != null || packageInfo.providers?.find {
foundChinaApps.add(packageName) it.name.matches(
continue chinaAppRegex
)
} != null) {
return true
} }
val packageFile = ZipFile(File(packageInfo.applicationInfo.publicSourceDir)) ZipFile(File(packageInfo.applicationInfo.publicSourceDir)).use {
for (packageEntry in packageFile.entries()) { for (packageEntry in it.entries()) {
if (!(packageEntry.name.startsWith("classes") && packageEntry.name.endsWith( if (!(packageEntry.name.startsWith("classes") && packageEntry.name.endsWith(
".dex" ".dex"
)) ))
@ -380,105 +642,28 @@ class PerAppProxyActivity : AbstractActivity<ActivityPerAppProxyBinding>() {
continue continue
} }
if (packageEntry.size > 15000000) { if (packageEntry.size > 15000000) {
foundChinaApps.add(packageName) return true
break
} }
val input = packageFile.getInputStream(packageEntry).buffered() val input = it.getInputStream(packageEntry).buffered()
val dexFile = try { val dexFile = try {
DexBackedDexFile.fromInputStream(null, input) DexBackedDexFile.fromInputStream(null, input)
} catch (e: Exception) { } catch (e: Exception) {
foundChinaApps.add(packageName) return false
Log.w(
TAG,
"scan china apps: failed to read dex file, error = ${e.message}"
)
break
} }
for (clazz in dexFile.classes) { for (clazz in dexFile.classes) {
val clazzName = clazz.type.substring(1, clazz.type.length - 1) val clazzName =
.replace("/", ".") clazz.type.substring(1, clazz.type.length - 1).replace("/", ".")
.replace("$", ".") .replace("$", ".")
if (clazzName.matches(chinaAppRegex)) { if (clazzName.matches(chinaAppRegex)) {
foundChinaApps.add(packageName) return true
break
} }
} }
} }
packageFile.close()
} catch (e: Exception) {
Log.w(
TAG,
"scan china apps: something went wrong when scanning package ${packageName}, error = ${e.message}"
)
continue
} }
System.gc() } catch (ignored: Exception) {
} }
return foundChinaApps return false
}
}
data class AppItem(
val packageName: String,
val name: String,
val icon: Drawable,
val isSystemApp: Boolean,
var selected: Boolean
)
class AppListAdapter(
private val list: List<AppItem>,
private val onSelectChange: (Int) -> Unit
) :
RecyclerView.Adapter<AppListViewHolder>() {
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): AppListViewHolder {
val binding =
ViewAppListItemBinding.inflate(LayoutInflater.from(parent.context), parent, false)
return AppListViewHolder(binding)
}
override fun getItemCount(): Int {
return list.size
}
override fun onBindViewHolder(holder: AppListViewHolder, position: Int) {
val item = list[position]
holder.bind(item)
holder.itemView.setOnClickListener {
item.selected = !item.selected
onSelectChange.invoke(position)
notifyItemChanged(position, "check")
}
}
override fun onBindViewHolder(
holder: AppListViewHolder,
position: Int,
payloads: MutableList<Any>
) {
if (payloads.contains("check")) {
holder.bindCheck(list[position])
} else {
super.onBindViewHolder(holder, position, payloads)
} }
} }
} }
class AppListViewHolder(
private val binding: ViewAppListItemBinding
) : RecyclerView.ViewHolder(binding.root) {
fun bind(item: AppItem) {
binding.appIcon.setImageDrawable(item.icon)
binding.applicationLabel.text = item.name
binding.packageName.text = item.packageName
binding.selected.isChecked = item.selected
}
fun bindCheck(item: AppItem) {
binding.selected.isChecked = item.selected
}
}
}

View file

@ -1,669 +0,0 @@
package io.nekohasekai.sfa.ui.profileoverride
import android.Manifest
import android.annotation.SuppressLint
import android.content.pm.ApplicationInfo
import android.content.pm.PackageInfo
import android.content.pm.PackageManager
import android.os.Build
import android.os.Bundle
import android.view.Gravity
import android.view.LayoutInflater
import android.view.Menu
import android.view.MenuItem
import android.view.ViewGroup
import android.widget.Toast
import androidx.appcompat.widget.PopupMenu
import androidx.appcompat.widget.SearchView
import androidx.core.view.isVisible
import androidx.lifecycle.lifecycleScope
import androidx.recyclerview.widget.RecyclerView
import com.google.android.material.dialog.MaterialAlertDialogBuilder
import io.nekohasekai.sfa.Application
import io.nekohasekai.sfa.R
import io.nekohasekai.sfa.database.Settings
import io.nekohasekai.sfa.databinding.ActivityPerAppProxy0Binding
import io.nekohasekai.sfa.databinding.DialogProgressbarBinding
import io.nekohasekai.sfa.databinding.ViewAppListItemBinding
import io.nekohasekai.sfa.ktx.clipboardText
import io.nekohasekai.sfa.ui.shared.AbstractActivity
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.delay
import kotlinx.coroutines.launch
import kotlinx.coroutines.withContext
import org.jf.dexlib2.dexbacked.DexBackedDexFile
import java.io.File
import java.util.zip.ZipFile
class PerAppProxyActivity0 : AbstractActivity<ActivityPerAppProxy0Binding>() {
enum class SortMode {
NAME, PACKAGE_NAME, UID, INSTALL_TIME, UPDATE_TIME,
}
private var proxyMode = Settings.PER_APP_PROXY_INCLUDE
private var sortMode = SortMode.NAME
private var sortReverse = false
private var hideSystemApps = false
private var hideOfflineApps = true
private var hideDisabledApps = true
inner class PackageCache(
private val packageInfo: PackageInfo,
) {
val packageName: String get() = packageInfo.packageName
val uid get() = packageInfo.applicationInfo.uid
val installTime get() = packageInfo.firstInstallTime
val updateTime get() = packageInfo.lastUpdateTime
val isSystem get() = packageInfo.applicationInfo.flags and ApplicationInfo.FLAG_SYSTEM == 1
val isOffline get() = packageInfo.requestedPermissions?.contains(Manifest.permission.INTERNET) != true
val isDisabled get() = packageInfo.applicationInfo.flags and ApplicationInfo.FLAG_INSTALLED == 0
val applicationIcon by lazy {
packageInfo.applicationInfo.loadIcon(packageManager)
}
val applicationLabel by lazy {
packageInfo.applicationInfo.loadLabel(packageManager).toString()
}
}
private lateinit var adapter: ApplicationAdapter
private var packages = listOf<PackageCache>()
private var displayPackages = listOf<PackageCache>()
private var currentPackages = listOf<PackageCache>()
private var selectedUIDs = mutableSetOf<Int>()
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setTitle(R.string.title_per_app_proxy)
lifecycleScope.launch {
proxyMode = if (Settings.perAppProxyMode == Settings.PER_APP_PROXY_EXCLUDE) {
Settings.PER_APP_PROXY_EXCLUDE
} else {
Settings.PER_APP_PROXY_INCLUDE
}
withContext(Dispatchers.Main) {
if (proxyMode != Settings.PER_APP_PROXY_EXCLUDE) {
binding.perAppProxyMode.setText(R.string.per_app_proxy_mode_include_description)
} else {
binding.perAppProxyMode.setText(R.string.per_app_proxy_mode_exclude_description)
}
}
reloadApplicationList()
filterApplicationList()
withContext(Dispatchers.Main) {
adapter = ApplicationAdapter(displayPackages)
binding.appList.adapter = adapter
delay(500L)
binding.progress.isVisible = false
}
}
}
private fun reloadApplicationList() {
val packageManagerFlags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
PackageManager.GET_PERMISSIONS or PackageManager.MATCH_UNINSTALLED_PACKAGES
} else {
@Suppress("DEPRECATION") PackageManager.GET_PERMISSIONS or PackageManager.GET_UNINSTALLED_PACKAGES
}
val installedPackages = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
packageManager.getInstalledPackages(
PackageManager.PackageInfoFlags.of(
packageManagerFlags.toLong()
)
)
} else {
@Suppress("DEPRECATION") packageManager.getInstalledPackages(packageManagerFlags)
}
val packages = mutableListOf<PackageCache>()
for (packageInfo in installedPackages) {
if (packageInfo.packageName == packageName) continue
packages.add(PackageCache(packageInfo))
}
val selectedPackageNames = Settings.perAppProxyList.toMutableSet()
val selectedUIDs = mutableSetOf<Int>()
for (packageCache in packages) {
if (selectedPackageNames.contains(packageCache.packageName)) {
selectedUIDs.add(packageCache.uid)
}
}
this.packages = packages
this.selectedUIDs = selectedUIDs
}
private fun filterApplicationList(selectedUIDs: Set<Int> = this.selectedUIDs) {
val displayPackages = mutableListOf<PackageCache>()
for (packageCache in packages) {
if (hideSystemApps && packageCache.isSystem) continue
if (hideOfflineApps && packageCache.isOffline) continue
if (hideDisabledApps && packageCache.isDisabled) continue
displayPackages.add(packageCache)
}
displayPackages.sortWith(compareBy<PackageCache> {
!selectedUIDs.contains(it.uid)
}.let {
if (!sortReverse) it.thenBy {
when (sortMode) {
SortMode.NAME -> it.applicationLabel
SortMode.PACKAGE_NAME -> it.packageName
SortMode.UID -> it.uid
SortMode.INSTALL_TIME -> it.installTime
SortMode.UPDATE_TIME -> it.updateTime
}
} else it.thenByDescending {
when (sortMode) {
SortMode.NAME -> it.applicationLabel
SortMode.PACKAGE_NAME -> it.packageName
SortMode.UID -> it.uid
SortMode.INSTALL_TIME -> it.installTime
SortMode.UPDATE_TIME -> it.updateTime
}
}
})
this.displayPackages = displayPackages
this.currentPackages = displayPackages
}
private fun updateApplicationSelection(packageCache: PackageCache, selected: Boolean) {
val performed = if (selected) {
selectedUIDs.add(packageCache.uid)
} else {
selectedUIDs.remove(packageCache.uid)
}
if (!performed) return
currentPackages.forEachIndexed { index, it ->
if (it.uid == packageCache.uid) {
adapter.notifyItemChanged(index, PayloadUpdateSelection(selected))
}
}
saveSelectedApplications()
}
data class PayloadUpdateSelection(val selected: Boolean)
inner class ApplicationAdapter(private var applicationList: List<PackageCache>) :
RecyclerView.Adapter<ApplicationViewHolder>() {
@SuppressLint("NotifyDataSetChanged")
fun setApplicationList(applicationList: List<PackageCache>) {
this.applicationList = applicationList
notifyDataSetChanged()
}
override fun onCreateViewHolder(
parent: ViewGroup, viewType: Int
): ApplicationViewHolder {
return ApplicationViewHolder(
ViewAppListItemBinding.inflate(
LayoutInflater.from(parent.context), parent, false
)
)
}
override fun getItemCount(): Int {
return applicationList.size
}
override fun onBindViewHolder(
holder: ApplicationViewHolder, position: Int
) {
holder.bind(applicationList[position])
}
override fun onBindViewHolder(
holder: ApplicationViewHolder, position: Int, payloads: MutableList<Any>
) {
if (payloads.isEmpty()) {
onBindViewHolder(holder, position)
return
}
payloads.forEach {
when (it) {
is PayloadUpdateSelection -> holder.updateSelection(it.selected)
}
}
}
}
inner class ApplicationViewHolder(
private val binding: ViewAppListItemBinding
) : RecyclerView.ViewHolder(binding.root) {
@SuppressLint("SetTextI18n")
fun bind(packageCache: PackageCache) {
binding.appIcon.setImageDrawable(packageCache.applicationIcon)
binding.applicationLabel.text = packageCache.applicationLabel
binding.packageName.text = "${packageCache.packageName} (${packageCache.uid})"
binding.selected.isChecked = selectedUIDs.contains(packageCache.uid)
binding.root.setOnClickListener {
updateApplicationSelection(packageCache, !binding.selected.isChecked)
}
binding.root.setOnLongClickListener {
val popup = PopupMenu(it.context, it)
popup.setForceShowIcon(true)
popup.gravity = Gravity.END
popup.menuInflater.inflate(R.menu.app_menu, popup.menu)
popup.setOnMenuItemClickListener {
when (it.itemId) {
R.id.action_copy_application_label -> {
clipboardText = packageCache.applicationLabel
true
}
R.id.action_copy_package_name -> {
clipboardText = packageCache.packageName
true
}
R.id.action_copy_uid -> {
clipboardText = packageCache.uid.toString()
true
}
else -> false
}
}
popup.show()
true
}
}
fun updateSelection(selected: Boolean) {
binding.selected.isChecked = selected
}
}
private fun searchApplications(searchText: String) {
currentPackages = if (searchText.isEmpty()) {
displayPackages
} else {
displayPackages.filter {
it.applicationLabel.contains(
searchText, ignoreCase = true
) || it.packageName.contains(
searchText, ignoreCase = true
) || it.uid.toString().contains(searchText)
}
}
adapter.setApplicationList(currentPackages)
}
override fun onCreateOptionsMenu(menu: Menu?): Boolean {
menuInflater.inflate(R.menu.per_app_menu0, menu)
if (menu != null) {
val searchView = menu.findItem(R.id.action_search).actionView as SearchView
searchView.setOnQueryTextListener(object : SearchView.OnQueryTextListener {
override fun onQueryTextSubmit(query: String): Boolean {
return true
}
override fun onQueryTextChange(newText: String): Boolean {
searchApplications(newText)
return true
}
})
searchView.setOnCloseListener {
searchApplications("")
true
}
when (proxyMode) {
Settings.PER_APP_PROXY_INCLUDE -> {
menu.findItem(R.id.action_mode_include).isChecked = true
}
Settings.PER_APP_PROXY_EXCLUDE -> {
menu.findItem(R.id.action_mode_exclude).isChecked = true
}
}
when (sortMode) {
SortMode.NAME -> {
menu.findItem(R.id.action_sort_by_name).isChecked = true
}
SortMode.PACKAGE_NAME -> {
menu.findItem(R.id.action_sort_by_package_name).isChecked = true
}
SortMode.UID -> {
menu.findItem(R.id.action_sort_by_uid).isChecked = true
}
SortMode.INSTALL_TIME -> {
menu.findItem(R.id.action_sort_by_install_time).isChecked = true
}
SortMode.UPDATE_TIME -> {
menu.findItem(R.id.action_sort_by_update_time).isChecked = true
}
}
menu.findItem(R.id.action_sort_reverse).isChecked = sortReverse
menu.findItem(R.id.action_hide_system_apps).isChecked = hideSystemApps
menu.findItem(R.id.action_hide_offline_apps).isChecked = hideOfflineApps
menu.findItem(R.id.action_hide_disabled_apps).isChecked = hideDisabledApps
}
return super.onCreateOptionsMenu(menu)
}
@SuppressLint("NotifyDataSetChanged")
override fun onOptionsItemSelected(item: MenuItem): Boolean {
when (item.itemId) {
R.id.action_mode_include -> {
item.isChecked = true
proxyMode = Settings.PER_APP_PROXY_INCLUDE
binding.perAppProxyMode.setText(R.string.per_app_proxy_mode_include_description)
lifecycleScope.launch {
Settings.perAppProxyMode = Settings.PER_APP_PROXY_INCLUDE
}
}
R.id.action_mode_exclude -> {
item.isChecked = true
proxyMode = Settings.PER_APP_PROXY_EXCLUDE
binding.perAppProxyMode.setText(R.string.per_app_proxy_mode_exclude_description)
lifecycleScope.launch {
Settings.perAppProxyMode = Settings.PER_APP_PROXY_EXCLUDE
}
}
R.id.action_sort_by_name -> {
item.isChecked = true
sortMode = SortMode.NAME
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_sort_by_package_name -> {
item.isChecked = true
sortMode = SortMode.PACKAGE_NAME
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_sort_by_uid -> {
item.isChecked = true
sortMode = SortMode.UID
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_sort_by_install_time -> {
item.isChecked = true
sortMode = SortMode.INSTALL_TIME
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_sort_by_update_time -> {
item.isChecked = true
sortMode = SortMode.UPDATE_TIME
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_sort_reverse -> {
item.isChecked = !item.isChecked
sortReverse = item.isChecked
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_hide_system_apps -> {
item.isChecked = !item.isChecked
hideSystemApps = item.isChecked
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_hide_offline_apps -> {
item.isChecked = !item.isChecked
hideOfflineApps = item.isChecked
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_hide_disabled_apps -> {
item.isChecked = !item.isChecked
hideDisabledApps = item.isChecked
filterApplicationList()
adapter.setApplicationList(currentPackages)
}
R.id.action_select_all -> {
val selectedUIDs = mutableSetOf<Int>()
for (packageCache in packages) {
selectedUIDs.add(packageCache.uid)
}
this.selectedUIDs = selectedUIDs
saveSelectedApplications()
}
R.id.action_deselect_all -> {
selectedUIDs = mutableSetOf()
saveSelectedApplications()
}
R.id.action_export -> {
lifecycleScope.launch {
val packageList = mutableListOf<String>()
for (packageCache in packages) {
if (selectedUIDs.contains(packageCache.uid)) {
packageList.add(packageCache.packageName)
}
}
clipboardText = packageList.joinToString("\n")
withContext(Dispatchers.Main) {
Toast.makeText(
this@PerAppProxyActivity0,
R.string.toast_copied_to_clipboard,
Toast.LENGTH_SHORT
).show()
}
}
}
R.id.action_import -> {
val packageNames = clipboardText?.split("\n")?.distinct()
?.takeIf { it.isNotEmpty() && it[0].isNotEmpty() }
if (packageNames.isNullOrEmpty()) {
Toast.makeText(
this@PerAppProxyActivity0,
R.string.toast_clipboard_empty,
Toast.LENGTH_SHORT
).show()
return true
}
val selectedUIDs = mutableSetOf<Int>()
for (packageCache in packages) {
if (packageNames.contains(packageCache.packageName)) {
selectedUIDs.add(packageCache.uid)
}
}
lifecycleScope.launch {
postSaveSelectedApplications(selectedUIDs)
withContext(Dispatchers.Main) {
Toast.makeText(
this@PerAppProxyActivity0,
R.string.toast_imported_from_clipboard,
Toast.LENGTH_SHORT
).show()
}
}
}
R.id.action_scan_china_apps -> {
scanChinaApps()
}
}
return true
}
@SuppressLint("NotifyDataSetChanged")
private fun scanChinaApps() {
val binding = DialogProgressbarBinding.inflate(layoutInflater)
binding.progress.max = currentPackages.size
binding.message.setText(R.string.message_scanning)
val progress = MaterialAlertDialogBuilder(
this, com.google.android.material.R.style.Theme_MaterialComponents_Dialog
).setView(binding.root).setCancelable(false).create()
progress.show()
lifecycleScope.launch {
val foundApps = withContext(Dispatchers.IO) {
mutableMapOf<String, PackageCache>().also { foundApps ->
currentPackages.forEachIndexed { index, it ->
if (scanChinaPackage(it.packageName)) {
foundApps[it.packageName] = it
}
withContext(Dispatchers.Main) {
binding.progress.progress = index + 1
}
}
}
}
withContext(Dispatchers.Main) {
progress.dismiss()
if (foundApps.isEmpty()) {
MaterialAlertDialogBuilder(this@PerAppProxyActivity0).setTitle(R.string.title_scan_result)
.setMessage(R.string.message_scan_app_no_apps_found)
.setPositiveButton(R.string.ok, null).show()
return@withContext
}
val dialogContent =
getString(R.string.message_scan_app_found) + "\n\n" + foundApps.entries.joinToString(
"\n"
) {
"${it.value.applicationLabel} (${it.key})"
}
MaterialAlertDialogBuilder(this@PerAppProxyActivity0).setTitle(R.string.title_scan_result)
.setMessage(dialogContent)
.setPositiveButton(R.string.action_select) { dialog, _ ->
dialog.dismiss()
lifecycleScope.launch {
val selectedUIDs = selectedUIDs.toMutableSet()
foundApps.values.forEach {
selectedUIDs.add(it.uid)
}
postSaveSelectedApplications(selectedUIDs)
}
}.setNegativeButton(R.string.action_deselect) { dialog, _ ->
dialog.dismiss()
lifecycleScope.launch {
val selectedUIDs = selectedUIDs.toMutableSet()
foundApps.values.forEach {
selectedUIDs.remove(it.uid)
}
postSaveSelectedApplications(selectedUIDs)
}
}.setNeutralButton(android.R.string.cancel, null).show()
}
}
}
@SuppressLint("NotifyDataSetChanged")
private suspend fun postSaveSelectedApplications(newUIDs: MutableSet<Int>) {
filterApplicationList(newUIDs)
withContext(Dispatchers.Main) {
selectedUIDs = newUIDs
adapter.notifyDataSetChanged()
}
val packageList = selectedUIDs.mapNotNull { uid ->
packages.find { it.uid == uid }?.packageName
}
Settings.perAppProxyList = packageList.toSet()
}
private fun saveSelectedApplications() {
lifecycleScope.launch {
val packageList = selectedUIDs.mapNotNull { uid ->
packages.find { it.uid == uid }?.packageName
}
Settings.perAppProxyList = packageList.toSet()
}
}
companion object {
private val chinaAppPrefixList by lazy {
runCatching {
Application.application.assets.open("prefix-china-apps.txt").reader().readLines()
}.getOrNull() ?: emptyList()
}
private val chinaAppRegex by lazy {
("(" + chinaAppPrefixList.joinToString("|").replace(".", "\\.") + ").*").toRegex()
}
fun scanChinaPackage(packageName: String): Boolean {
val packageManagerFlags = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
PackageManager.MATCH_UNINSTALLED_PACKAGES or PackageManager.GET_ACTIVITIES or PackageManager.GET_SERVICES or PackageManager.GET_RECEIVERS or PackageManager.GET_PROVIDERS
} else {
@Suppress("DEPRECATION") PackageManager.GET_UNINSTALLED_PACKAGES or PackageManager.GET_ACTIVITIES or PackageManager.GET_SERVICES or PackageManager.GET_RECEIVERS or PackageManager.GET_PROVIDERS
}
if (packageName.matches(chinaAppRegex)) {
return true
}
try {
val packageInfo = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
Application.packageManager.getPackageInfo(
packageName,
PackageManager.PackageInfoFlags.of(packageManagerFlags.toLong())
)
} else {
@Suppress("DEPRECATION") Application.packageManager.getPackageInfo(
packageName, packageManagerFlags
)
}
if (packageInfo.services?.find { it.name.matches(chinaAppRegex) } != null || packageInfo.activities?.find {
it.name.matches(
chinaAppRegex
)
} != null || packageInfo.receivers?.find { it.name.matches(chinaAppRegex) } != null || packageInfo.providers?.find {
it.name.matches(
chinaAppRegex
)
} != null) {
return true
}
ZipFile(File(packageInfo.applicationInfo.publicSourceDir)).use {
for (packageEntry in it.entries()) {
if (!(packageEntry.name.startsWith("classes") && packageEntry.name.endsWith(
".dex"
))
) {
continue
}
if (packageEntry.size > 15000000) {
return true
}
val input = it.getInputStream(packageEntry).buffered()
val dexFile = try {
DexBackedDexFile.fromInputStream(null, input)
} catch (e: Exception) {
return false
}
for (clazz in dexFile.classes) {
val clazzName =
clazz.type.substring(1, clazz.type.length - 1).replace("/", ".")
.replace("$", ".")
if (clazzName.matches(chinaAppRegex)) {
return true
}
}
}
}
} catch (ignored: Exception) {
}
return false
}
}
}

View file

@ -38,7 +38,7 @@ class ProfileOverrideActivity :
} }
binding.configureAppListButton.setOnClickListener { binding.configureAppListButton.setOnClickListener {
startActivity(Intent(this, PerAppProxyActivity0::class.java)) startActivity(Intent(this, PerAppProxyActivity::class.java))
} }
lifecycleScope.launch(Dispatchers.IO) { lifecycleScope.launch(Dispatchers.IO) {
reloadSettings() reloadSettings()

View file

@ -1,6 +1,7 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android" <androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="match_parent"> android:layout_height="match_parent">
@ -11,56 +12,44 @@
android:background="?colorSurfaceContainer" android:background="?colorSurfaceContainer"
android:fitsSystemWindows="true"> android:fitsSystemWindows="true">
<com.google.android.material.appbar.CollapsingToolbarLayout
style="?attr/collapsingToolbarLayoutLargeStyle"
android:layout_width="match_parent"
android:layout_height="?attr/collapsingToolbarLayoutLargeSize"
app:layout_scrollFlags="scroll|enterAlways|snap">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical">
<com.google.android.material.appbar.MaterialToolbar <com.google.android.material.appbar.MaterialToolbar
android:id="@+id/toolbar" android:id="@+id/toolbar"
style="?attr/toolbarStyle"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="?actionBarSize" /> android:layout_height="?actionBarSize" />
<RadioGroup <com.google.android.material.progressindicator.LinearProgressIndicator
android:id="@+id/radio_group_per_app_mode" android:id="@+id/progress"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:paddingHorizontal="16dp"> android:indeterminate="true" />
<RadioButton <LinearLayout
android:id="@+id/radio_per_app_exclude"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_height="40dp"
android:text="@string/per_app_proxy_mode_exclude" /> android:background="?colorSurfaceContainerLow"
android:gravity="center_vertical"
android:paddingLeft="16dp">
<RadioButton <TextView
android:id="@+id/radio_per_app_include" android:id="@+id/per_app_proxy_mode"
android:layout_width="match_parent" android:layout_width="wrap_content"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:text="@string/per_app_proxy_mode_include" /> android:text="@string/per_app_proxy_mode_include" />
</RadioGroup>
</LinearLayout> </LinearLayout>
</com.google.android.material.appbar.CollapsingToolbarLayout>
</com.google.android.material.appbar.AppBarLayout> </com.google.android.material.appbar.AppBarLayout>
<androidx.recyclerview.widget.RecyclerView <androidx.recyclerview.widget.RecyclerView
android:id="@+id/recycler_view_app_list" android:id="@+id/app_list"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="match_parent" android:layout_height="match_parent"
android:scrollbars="vertical" android:scrollbars="vertical"
app:layoutManager="androidx.recyclerview.widget.LinearLayoutManager" app:layoutManager="androidx.recyclerview.widget.LinearLayoutManager"
app:layout_behavior="@string/appbar_scrolling_view_behavior" /> app:layout_behavior="@string/appbar_scrolling_view_behavior"
tools:listitem="@layout/view_app_list_item" />
</androidx.coordinatorlayout.widget.CoordinatorLayout> </androidx.coordinatorlayout.widget.CoordinatorLayout>

View file

@ -1,55 +0,0 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent">
<com.google.android.material.appbar.AppBarLayout
android:id="@+id/appbar_layout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="?colorSurfaceContainer"
android:fitsSystemWindows="true">
<com.google.android.material.appbar.MaterialToolbar
android:id="@+id/toolbar"
style="?attr/toolbarStyle"
android:layout_width="match_parent"
android:layout_height="?actionBarSize" />
<com.google.android.material.progressindicator.LinearProgressIndicator
android:id="@+id/progress"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:indeterminate="true" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="40dp"
android:background="?colorSurfaceContainerLow"
android:gravity="center_vertical"
android:paddingLeft="16dp">
<TextView
android:id="@+id/per_app_proxy_mode"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/per_app_proxy_mode_include" />
</LinearLayout>
</com.google.android.material.appbar.AppBarLayout>
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/app_list"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:scrollbars="vertical"
app:layoutManager="androidx.recyclerview.widget.LinearLayoutManager"
app:layout_behavior="@string/appbar_scrolling_view_behavior"
tools:listitem="@layout/view_app_list_item" />
</androidx.coordinatorlayout.widget.CoordinatorLayout>

View file

@ -6,9 +6,10 @@
style="?materialCardViewElevatedStyle" style="?materialCardViewElevatedStyle"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:layout_margin="4dp"
app:cardBackgroundColor="?colorSurfaceContainer" app:cardBackgroundColor="?colorSurfaceContainer"
app:cardCornerRadius="0dp" app:cardCornerRadius="0dp"
app:cardElevation="0dp"> app:cardElevation="4dp">
<LinearLayout <LinearLayout
android:layout_width="match_parent" android:layout_width="match_parent"