diff --git a/app/build.gradle.kts b/app/build.gradle.kts
index d7b4fbc..764ea2b 100644
--- a/app/build.gradle.kts
+++ b/app/build.gradle.kts
@@ -73,6 +73,7 @@ dependencies {
implementation("com.takisoft.preferencex:preferencex:1.1.0")
implementation("com.google.android.material:material:1.12.0")
implementation("androidx.constraintlayout:constraintlayout:2.1.4")
+ implementation("androidx.recyclerview:recyclerview:1.3.2")
implementation("androidx.lifecycle:lifecycle-runtime-ktx:2.8.4")
implementation("androidx.lifecycle:lifecycle-service:2.8.4")
testImplementation("junit:junit:4.13.2")
diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml
index 3dd4840..71e714b 100644
--- a/app/src/main/AndroidManifest.xml
+++ b/app/src/main/AndroidManifest.xml
@@ -6,6 +6,9 @@
+
+
,
+ private val onChange: (Model, Boolean) -> Unit,
+) : RecyclerView.Adapter() {
+
+ private val comparator = compareBy({ !checker.test(it) }, { it?.label })
+
+ private val list: SortedList = SortedList(Model::class.java, SortedList.BatchedCallback(object :
+ SortedList.Callback() {
+
+ override fun compare(o1: Model?, o2: Model?): Int {
+ return comparator.compare(o1, o2)
+ }
+
+ override fun onInserted(position: Int, count: Int) {
+ notifyItemRangeInserted(position, count)
+ }
+
+ override fun onRemoved(position: Int, count: Int) {
+ notifyItemRangeRemoved(position, count)
+ }
+
+ override fun onMoved(fromPosition: Int, toPosition: Int) {
+ notifyItemMoved(fromPosition, toPosition)
+ }
+
+ override fun onChanged(position: Int, count: Int) {
+ notifyItemRangeChanged(position, count)
+ }
+
+ override fun areItemsTheSame(item1: Model?, item2: Model?): Boolean =
+ item1?.packageName == item2?.packageName
+
+ override fun areContentsTheSame(oldItem: Model?, newItem: Model?): Boolean =
+ oldItem == newItem
+
+ }))
+
+ override fun onCreateViewHolder(
+ parent: ViewGroup,
+ viewType: Int
+ ): ViewHolder = ViewHolder(ItemAppBinding.inflate(LayoutInflater.from(context), parent, false))
+
+ override fun getItemCount(): Int {
+ return list.size()
+ }
+
+ override fun onBindViewHolder(holder: ViewHolder, position: Int) {
+ val model = list[position]
+
+ holder.binding.apply {
+ appLabel.text = model.label
+ appPackage.text = model.packageName
+ icon.setImageDrawable(model.icon)
+
+ checkbox.isChecked = checker.test(model)
+
+ root.setOnClickListener {
+ val checked = !checkbox.isChecked
+ checkbox.isChecked = checked
+ onChange(model, checked)
+ }
+ }
+ }
+
+ fun setList(newList: List) {
+ list.beginBatchedUpdates()
+ list.clear()
+ list.addAll(newList)
+ list.endBatchedUpdates()
+ }
+
+ class ViewHolder(
+ val binding: ItemAppBinding
+ ) : RecyclerView.ViewHolder(binding.root)
+
+ data class Model(
+ val label: String,
+ val packageName: String,
+ val icon: Drawable
+ )
+}
\ No newline at end of file
diff --git a/app/src/main/java/io/github/dovecoteescapee/byedpi/fragments/VpnAppsFilterFragment.kt b/app/src/main/java/io/github/dovecoteescapee/byedpi/fragments/VpnAppsFilterFragment.kt
new file mode 100644
index 0000000..5fc1119
--- /dev/null
+++ b/app/src/main/java/io/github/dovecoteescapee/byedpi/fragments/VpnAppsFilterFragment.kt
@@ -0,0 +1,151 @@
+package io.github.dovecoteescapee.byedpi.fragments
+
+import android.os.Bundle
+import android.view.LayoutInflater
+import android.view.Menu
+import android.view.MenuInflater
+import android.view.MenuItem
+import android.view.View
+import android.view.ViewGroup
+import androidx.appcompat.widget.SearchView
+import androidx.appcompat.widget.SearchView.OnQueryTextListener
+import androidx.core.content.edit
+import androidx.core.view.MenuProvider
+import androidx.fragment.app.Fragment
+import androidx.lifecycle.Lifecycle
+import androidx.lifecycle.lifecycleScope
+import androidx.recyclerview.widget.DividerItemDecoration
+import androidx.recyclerview.widget.LinearLayoutManager
+import androidx.recyclerview.widget.RecyclerView
+import io.github.dovecoteescapee.byedpi.adapters.AppItemAdapter
+import io.github.dovecoteescapee.byedpi.R
+import io.github.dovecoteescapee.byedpi.databinding.FragmentFilterBinding
+import io.github.dovecoteescapee.byedpi.utility.getPreferences
+import kotlinx.coroutines.Dispatchers
+import kotlinx.coroutines.Job
+import kotlinx.coroutines.launch
+import kotlinx.coroutines.withContext
+
+class VpnAppsFilterFragment : Fragment(), MenuProvider {
+
+ private lateinit var binding: FragmentFilterBinding
+ private lateinit var adapter: AppItemAdapter
+
+ private lateinit var apps: List
+ private lateinit var checked: MutableSet
+
+ private var updateJob: Job? = null
+
+ override fun onCreateView(
+ inflater: LayoutInflater,
+ container: ViewGroup?,
+ savedInstanceState: Bundle?
+ ): View {
+ binding = FragmentFilterBinding.inflate(inflater, container, false)
+
+ val activity = requireActivity()
+
+ checked = activity.getPreferences().getStringSet("vpn_filtered_apps", emptySet())!!
+ .toMutableSet()
+ adapter = AppItemAdapter(
+ activity,
+ { checked.contains(it.packageName) },
+ { app, status ->
+ checked.apply {
+ if (status) {
+ add(app.packageName)
+ } else {
+ remove(app.packageName)
+ }
+ }
+ activity.getPreferences().edit {
+ putStringSet("vpn_filtered_apps", checked.toSet())
+ }
+ })
+
+
+ binding.appList.layoutManager = LinearLayoutManager(activity)
+ binding.appList.addItemDecoration(
+ DividerItemDecoration(
+ binding.appList.context,
+ RecyclerView.VERTICAL
+ )
+ )
+
+ binding.appList.adapter = adapter
+
+ updateJob?.cancel()
+ updateJob = lifecycleScope.launch {
+ apps = withContext(Dispatchers.IO) { collectApps() }
+ adapter.setList(apps)
+
+ binding.progressCircular.visibility = View.GONE
+ binding.appList.visibility = View.VISIBLE
+ }
+
+ requireActivity().addMenuProvider(this, viewLifecycleOwner, Lifecycle.State.RESUMED)
+
+ return binding.root
+ }
+
+ private fun collectApps(): List {
+ val context = requireContext()
+ val packageManager = context.packageManager
+
+ return packageManager.getInstalledApplications(0)
+ .filter { it.packageName != context.packageName }
+ .map {
+ AppItemAdapter.Model(
+ label = packageManager.getApplicationLabel(it).toString(),
+ packageName = it.packageName,
+ icon = packageManager.getApplicationIcon(it),
+ )
+ }
+ }
+
+ private fun updateFilter(query: String?) {
+ if (!::apps.isInitialized) {
+ return
+ }
+
+ if (query == null) {
+ adapter.setList(apps)
+ return
+ }
+
+ val lQuery = query.lowercase()
+ adapter.setList(apps.filter { it.label.lowercase().contains(lQuery) })
+ }
+
+ override fun onCreateMenu(menu: Menu, menuInflater: MenuInflater) {
+ menuInflater.inflate(R.menu.menu_filter, menu)
+
+ val searchItem = menu.findItem(R.id.action_search)
+ (searchItem.actionView as SearchView).setOnQueryTextListener(object : OnQueryTextListener {
+ override fun onQueryTextSubmit(query: String): Boolean {
+ updateFilter(query)
+ return true
+ }
+
+ override fun onQueryTextChange(query: String?): Boolean {
+ updateFilter(query)
+ return true
+ }
+ })
+
+ searchItem.setOnActionExpandListener(object : MenuItem.OnActionExpandListener {
+ override fun onMenuItemActionExpand(item: MenuItem): Boolean {
+ return true
+ }
+
+ override fun onMenuItemActionCollapse(item: MenuItem): Boolean {
+ updateFilter(null)
+ return true
+ }
+ })
+ }
+
+ override fun onMenuItemSelected(menuItem: MenuItem): Boolean {
+ return false
+ }
+}
diff --git a/app/src/main/java/io/github/dovecoteescapee/byedpi/services/ByeDpiVpnService.kt b/app/src/main/java/io/github/dovecoteescapee/byedpi/services/ByeDpiVpnService.kt
index d2abdfd..feb7241 100644
--- a/app/src/main/java/io/github/dovecoteescapee/byedpi/services/ByeDpiVpnService.kt
+++ b/app/src/main/java/io/github/dovecoteescapee/byedpi/services/ByeDpiVpnService.kt
@@ -3,6 +3,7 @@ package io.github.dovecoteescapee.byedpi.services
import android.app.Notification
import android.app.PendingIntent
import android.content.Intent
+import android.content.pm.PackageManager
import android.content.pm.ServiceInfo
import android.os.Build
import android.os.ParcelFileDescriptor
@@ -297,7 +298,29 @@ class ByeDpiVpnService : LifecycleVpnService() {
builder.setMetered(false)
}
- builder.addDisallowedApplication(applicationContext.packageName)
+ val filter = getPreferences().getStringSet("vpn_filtered_apps", emptySet())!!
+ when (val filterMode = getPreferences().getString("vpn_filter_mode", "blacklist")) {
+ "blacklist" -> {
+ filter.forEach {
+ try {
+ builder.addDisallowedApplication(it)
+ } catch (ignore: PackageManager.NameNotFoundException) {}
+ }
+ builder.addDisallowedApplication(applicationContext.packageName)
+ }
+
+ "whitelist" -> {
+ filter.forEach {
+ try {
+ builder.addAllowedApplication(it)
+ } catch (ignore: PackageManager.NameNotFoundException) {}
+ }
+ }
+
+ else -> {
+ Log.w(TAG, "Invalid VPN filter mode: $filterMode")
+ }
+ }
return builder
}
diff --git a/app/src/main/res/drawable-anydpi-v24/ic_search.xml b/app/src/main/res/drawable-anydpi-v24/ic_search.xml
new file mode 100644
index 0000000..79129dd
--- /dev/null
+++ b/app/src/main/res/drawable-anydpi-v24/ic_search.xml
@@ -0,0 +1,15 @@
+
+
+
+
+
diff --git a/app/src/main/res/drawable-hdpi/ic_search.png b/app/src/main/res/drawable-hdpi/ic_search.png
new file mode 100644
index 0000000..a2ec243
Binary files /dev/null and b/app/src/main/res/drawable-hdpi/ic_search.png differ
diff --git a/app/src/main/res/drawable-mdpi/ic_search.png b/app/src/main/res/drawable-mdpi/ic_search.png
new file mode 100644
index 0000000..c650250
Binary files /dev/null and b/app/src/main/res/drawable-mdpi/ic_search.png differ
diff --git a/app/src/main/res/drawable-xhdpi/ic_search.png b/app/src/main/res/drawable-xhdpi/ic_search.png
new file mode 100644
index 0000000..9e7ce1e
Binary files /dev/null and b/app/src/main/res/drawable-xhdpi/ic_search.png differ
diff --git a/app/src/main/res/drawable-xxhdpi/ic_search.png b/app/src/main/res/drawable-xxhdpi/ic_search.png
new file mode 100644
index 0000000..64e56b9
Binary files /dev/null and b/app/src/main/res/drawable-xxhdpi/ic_search.png differ
diff --git a/app/src/main/res/layout/fragment_filter.xml b/app/src/main/res/layout/fragment_filter.xml
new file mode 100644
index 0000000..788271d
--- /dev/null
+++ b/app/src/main/res/layout/fragment_filter.xml
@@ -0,0 +1,20 @@
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/layout/item_app.xml b/app/src/main/res/layout/item_app.xml
new file mode 100644
index 0000000..ec1b05e
--- /dev/null
+++ b/app/src/main/res/layout/item_app.xml
@@ -0,0 +1,56 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app/src/main/res/menu/menu_filter.xml b/app/src/main/res/menu/menu_filter.xml
new file mode 100644
index 0000000..e89c3fb
--- /dev/null
+++ b/app/src/main/res/menu/menu_filter.xml
@@ -0,0 +1,11 @@
+
+
\ No newline at end of file
diff --git a/app/src/main/res/values/arrays.xml b/app/src/main/res/values/arrays.xml
index 0b1744c..69f0bc8 100644
--- a/app/src/main/res/values/arrays.xml
+++ b/app/src/main/res/values/arrays.xml
@@ -47,4 +47,13 @@
- blacklist
- whitelist
+
+
+ - Blacklist apps
+ - Whilelist apps
+
+
+ - blacklist
+ - whitelist
+
diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml
index dc26369..ad9ac8b 100644
--- a/app/src/main/res/values/strings.xml
+++ b/app/src/main/res/values/strings.xml
@@ -72,4 +72,8 @@
HTTPS
Protocols
Uncheck all to desync all traffic
+ VPN
+ Apps filtering mode
+ Filtered applications
+ Search
diff --git a/app/src/main/res/xml/main_settings.xml b/app/src/main/res/xml/main_settings.xml
index e81fc6b..d08cd05 100644
--- a/app/src/main/res/xml/main_settings.xml
+++ b/app/src/main/res/xml/main_settings.xml
@@ -58,6 +58,25 @@
+
+
+
+
+
+
+
+