mirror of
https://github.com/SagerNet/sing-box-for-android.git
synced 2025-04-05 04:47:39 +03:00
Fix group item
This commit is contained in:
parent
f26458ba68
commit
0c76a7479c
10 changed files with 586 additions and 1127 deletions
|
@ -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" />
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -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
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -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()
|
||||||
|
|
|
@ -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>
|
|
@ -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>
|
|
|
@ -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"
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue