mirror of
https://github.com/SagerNet/sing-box-for-android.git
synced 2025-04-04 20:37:40 +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)
|
||||||
}
|
}
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -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
|
<com.google.android.material.appbar.MaterialToolbar
|
||||||
style="?attr/collapsingToolbarLayoutLargeStyle"
|
android:id="@+id/toolbar"
|
||||||
|
style="?attr/toolbarStyle"
|
||||||
android:layout_width="match_parent"
|
android:layout_width="match_parent"
|
||||||
android:layout_height="?attr/collapsingToolbarLayoutLargeSize"
|
android:layout_height="?actionBarSize" />
|
||||||
app:layout_scrollFlags="scroll|enterAlways|snap">
|
|
||||||
|
|
||||||
<LinearLayout
|
|
||||||
android:layout_width="match_parent"
|
<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:layout_height="wrap_content"
|
||||||
android:orientation="vertical">
|
android:text="@string/per_app_proxy_mode_include" />
|
||||||
|
|
||||||
<com.google.android.material.appbar.MaterialToolbar
|
</LinearLayout>
|
||||||
android:id="@+id/toolbar"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="?actionBarSize" />
|
|
||||||
|
|
||||||
|
|
||||||
<RadioGroup
|
|
||||||
android:id="@+id/radio_group_per_app_mode"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:paddingHorizontal="16dp">
|
|
||||||
|
|
||||||
<RadioButton
|
|
||||||
android:id="@+id/radio_per_app_exclude"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:text="@string/per_app_proxy_mode_exclude" />
|
|
||||||
|
|
||||||
<RadioButton
|
|
||||||
android:id="@+id/radio_per_app_include"
|
|
||||||
android:layout_width="match_parent"
|
|
||||||
android:layout_height="wrap_content"
|
|
||||||
android:text="@string/per_app_proxy_mode_include" />
|
|
||||||
|
|
||||||
</RadioGroup>
|
|
||||||
|
|
||||||
</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