Compare commits

..

135 Commits

Author SHA1 Message Date
Prefill add-on d43a219dce Translated using Weblate (Italian)
Currently translated at 78.2% (368 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/it/
2026-06-05 04:51:18 +00:00
Prefill add-on 7e98691181 Translated using Weblate (Chinese (Traditional Han script))
Currently translated at 76.1% (358 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/zh_Hant/
2026-06-05 04:51:18 +00:00
Prefill add-on b9becd433c Translated using Weblate (Slovak)
Currently translated at 30.6% (144 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/sk/
2026-06-05 04:51:17 +00:00
Prefill add-on 2120810f5f Translated using Weblate (Thai (Northern))
Currently translated at 13.1% (62 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/nod/
2026-06-05 04:51:16 +00:00
Prefill add-on d8c2c24023 Translated using Weblate (Estonian)
Currently translated at 79.1% (372 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/et/
2026-06-05 04:51:16 +00:00
Prefill add-on 043865dcdc Translated using Weblate (Vietnamese)
Currently translated at 18.5% (87 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/vi/
2026-06-05 04:51:15 +00:00
Prefill add-on 94d0227c42 Translated using Weblate (Ukrainian)
Currently translated at 79.1% (372 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/uk/
2026-06-05 04:51:14 +00:00
Prefill add-on c56d314f32 Translated using Weblate (Arabic)
Currently translated at 13.1% (62 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ar/
2026-06-05 04:51:14 +00:00
Prefill add-on 795d18a44a Translated using Weblate (Polish)
Currently translated at 99.3% (467 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/pl/
2026-06-05 04:51:13 +00:00
Prefill add-on 1b987a8c80 Translated using Weblate (Japanese)
Currently translated at 25.1% (118 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ja/
2026-06-05 04:51:12 +00:00
Prefill add-on 4c647bd5c7 Translated using Weblate (Hungarian)
Currently translated at 79.1% (372 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/hu/
2026-06-05 04:51:12 +00:00
Prefill add-on 511b96dbc7 Translated using Weblate (Russian)
Currently translated at 79.1% (372 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ru/
2026-06-05 04:51:11 +00:00
Prefill add-on acba25cbb2 Translated using Weblate (Urdu)
Currently translated at 65.7% (309 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ur/
2026-06-05 04:51:10 +00:00
Prefill add-on 87bb514da5 Translated using Weblate (Thai)
Currently translated at 13.4% (63 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/th/
2026-06-05 04:51:09 +00:00
Prefill add-on d169385b5c Translated using Weblate (Tamil)
Currently translated at 44.6% (210 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ta/
2026-06-05 04:51:08 +00:00
Prefill add-on bcb010e3fd Translated using Weblate (Persian)
Currently translated at 15.7% (74 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/fa/
2026-06-05 04:51:08 +00:00
Prefill add-on 66101ed515 Translated using Weblate (Turkish)
Currently translated at 38.7% (182 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/tr/
2026-06-05 04:51:07 +00:00
Prefill add-on 9e81a4d283 Translated using Weblate (Portuguese (Brazil))
Currently translated at 35.3% (166 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/pt_BR/
2026-06-05 04:51:06 +00:00
Prefill add-on d9ff97f83e Translated using Weblate (Czech)
Currently translated at 78.0% (367 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/cs/
2026-06-05 04:51:05 +00:00
Prefill add-on cf975a71d8 Translated using Weblate (Alemannic)
Currently translated at 13.1% (62 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/gsw/
2026-06-05 04:51:05 +00:00
Prefill add-on ca555805b3 Translated using Weblate (Dutch)
Currently translated at 76.5% (360 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/nl/
2026-06-05 04:51:04 +00:00
Prefill add-on 9c10e7c7b2 Translated using Weblate (Norwegian Bokmål)
Currently translated at 13.1% (62 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/nb_NO/
2026-06-05 04:51:03 +00:00
Prefill add-on f3fb4ab77c Translated using Weblate (German)
Currently translated at 79.1% (372 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/de/
2026-06-05 04:51:03 +00:00
Prefill add-on 7a648fd930 Translated using Weblate (Chinese (Simplified Han script))
Currently translated at 79.1% (372 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/zh_Hans/
2026-06-05 04:51:02 +00:00
Prefill add-on 2266cfc771 Translated using Weblate (French)
Currently translated at 79.1% (372 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/fr/
2026-06-05 04:51:01 +00:00
Prefill add-on 7166752d50 Translated using Weblate (Spanish)
Currently translated at 79.1% (372 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/es/
2026-06-05 04:51:01 +00:00
Prefill add-on f75e91fdc0 Translated using Weblate (Korean)
Currently translated at 79.1% (372 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ko/
2026-06-05 04:51:00 +00:00
Prefill add-on 2b7d2d3da6 Translated using Weblate (Indonesian)
Currently translated at 79.1% (372 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/id/
2026-06-05 04:50:59 +00:00
Prefill add-on 277068f6fb Translated using Weblate (Finnish)
Currently translated at 30.0% (141 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/fi/
2026-06-05 04:50:58 +00:00
Prefill add-on 097b86db9b Translated using Weblate (Portuguese (Portugal))
Currently translated at 36.5% (172 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/pt_PT/
2026-06-05 04:50:58 +00:00
Prefill add-on a8ec89f59e Translated using Weblate (Georgian)
Currently translated at 28.0% (132 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ka/
2026-06-05 04:50:57 +00:00
Prefill add-on db993759e4 Translated using Weblate (Danish)
Currently translated at 13.4% (63 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/da/
2026-06-05 04:50:56 +00:00
Prefill add-on 00a9f5ba8f Translated using Weblate (Serbian)
Currently translated at 13.1% (62 of 470 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/sr/
2026-06-05 04:50:55 +00:00
Matthaiks b3284339c1 Translated using Weblate (Polish)
Currently translated at 100.0% (472 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/pl/
2026-06-05 04:49:42 +00:00
Matthaiks d67d524e7b Translated using Weblate (Polish)
Currently translated at 100.0% (23 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/pl/
2026-06-05 04:49:41 +00:00
Fill read-only add-on 6f1a1bdc48 Translated using Weblate (Portuguese (Brazil))
Currently translated at 36.0% (170 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/pt_BR/
2026-06-05 04:49:41 +00:00
Fill read-only add-on e5790f1fa0 Translated using Weblate (Polish)
Currently translated at 80.0% (378 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/pl/
2026-06-05 04:49:40 +00:00
Fill read-only add-on 525a12b2f2 Translated using Weblate (Georgian)
Currently translated at 28.3% (134 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ka/
2026-06-05 04:49:40 +00:00
Fill read-only add-on f083441370 Translated using Weblate (Portuguese (Portugal))
Currently translated at 37.2% (176 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/pt_PT/
2026-06-05 04:49:39 +00:00
Fill read-only add-on 2945fcbf75 Translated using Weblate (Dutch)
Currently translated at 77.3% (365 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/nl/
2026-06-05 04:49:39 +00:00
Fill read-only add-on d379bf8960 Translated using Weblate (Chinese (Traditional Han script))
Currently translated at 76.9% (363 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/zh_Hant/
2026-06-05 04:49:38 +00:00
Fill read-only add-on 4421d31f69 Translated using Weblate (French)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/fr/
2026-06-05 04:49:37 +00:00
Fill read-only add-on 294081cec4 Translated using Weblate (Vietnamese)
Currently translated at 18.6% (88 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/vi/
2026-06-05 04:49:37 +00:00
Fill read-only add-on 47617012c2 Translated using Weblate (Italian)
Currently translated at 79.0% (373 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/it/
2026-06-05 04:49:36 +00:00
Fill read-only add-on 8cad5c6a14 Translated using Weblate (Persian)
Currently translated at 15.8% (75 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/fa/
2026-06-05 04:49:35 +00:00
Fill read-only add-on 8ac54c7a97 Translated using Weblate (German)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/de/
2026-06-05 04:49:35 +00:00
Fill read-only add-on 642f98fdaf Translated using Weblate (Hungarian)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/hu/
2026-06-05 04:49:34 +00:00
Fill read-only add-on 91cb569585 Translated using Weblate (Ukrainian)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/uk/
2026-06-05 04:49:33 +00:00
Fill read-only add-on 4368bfdc62 Translated using Weblate (Slovak)
Currently translated at 31.1% (147 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/sk/
2026-06-05 04:49:33 +00:00
Fill read-only add-on 9ac510eeff Translated using Weblate (Indonesian)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/id/
2026-06-05 04:49:32 +00:00
Fill read-only add-on ce08a5c308 Translated using Weblate (Czech)
Currently translated at 78.8% (372 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/cs/
2026-06-05 04:49:31 +00:00
Fill read-only add-on 5db281ba3e Translated using Weblate (Serbian)
Currently translated at 13.1% (62 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/sr/
2026-06-05 04:49:31 +00:00
Fill read-only add-on 48917ad0b8 Translated using Weblate (Thai)
Currently translated at 13.5% (64 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/th/
2026-06-05 04:49:30 +00:00
Fill read-only add-on d5c3983a5e Translated using Weblate (Alemannic)
Currently translated at 13.1% (62 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/gsw/
2026-06-05 04:49:30 +00:00
Fill read-only add-on e0a75d8dad Translated using Weblate (Spanish)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/es/
2026-06-05 04:49:29 +00:00
Fill read-only add-on eaf52b1476 Translated using Weblate (Finnish)
Currently translated at 30.7% (145 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/fi/
2026-06-05 04:49:28 +00:00
Fill read-only add-on b1b906d863 Translated using Weblate (Japanese)
Currently translated at 25.6% (121 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ja/
2026-06-05 04:49:28 +00:00
Fill read-only add-on 351f8ba573 Translated using Weblate (Turkish)
Currently translated at 39.4% (186 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/tr/
2026-06-05 04:49:27 +00:00
Fill read-only add-on 38168814b4 Translated using Weblate (Danish)
Currently translated at 13.3% (63 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/da/
2026-06-05 04:49:27 +00:00
Fill read-only add-on 4214aeea37 Translated using Weblate (Russian)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ru/
2026-06-05 04:49:26 +00:00
Fill read-only add-on 85a0ca695a Translated using Weblate (Thai (Northern))
Currently translated at 13.1% (62 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/nod/
2026-06-05 04:49:25 +00:00
Fill read-only add-on 0a281bb257 Translated using Weblate (Chinese (Simplified Han script))
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/zh_Hans/
2026-06-05 04:49:25 +00:00
Fill read-only add-on d71f307052 Translated using Weblate (Arabic)
Currently translated at 13.1% (62 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ar/
2026-06-05 04:49:24 +00:00
Fill read-only add-on de8f85ae9e Translated using Weblate (Korean)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ko/
2026-06-05 04:49:23 +00:00
Fill read-only add-on 402e258fcf Translated using Weblate (Tamil)
Currently translated at 45.5% (215 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ta/
2026-06-05 04:49:23 +00:00
Fill read-only add-on 271f1e5df3 Translated using Weblate (Norwegian Bokmål)
Currently translated at 13.1% (62 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/nb_NO/
2026-06-05 04:49:22 +00:00
Fill read-only add-on 0842364f74 Translated using Weblate (Estonian)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/et/
2026-06-05 04:49:22 +00:00
Fill read-only add-on 427baab115 Translated using Weblate (Urdu)
Currently translated at 66.5% (314 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ur/
2026-06-05 04:49:21 +00:00
Prefill add-on ef636fdafa Translated using Weblate (Slovak)
Currently translated at 31.1% (147 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/sk/
2026-06-05 04:49:20 +00:00
Prefill add-on e5e4bca5fa Translated using Weblate (Urdu)
Currently translated at 66.5% (314 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ur/
2026-06-05 04:49:19 +00:00
Prefill add-on cb3eeee9f8 Translated using Weblate (French)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/fr/
2026-06-05 04:49:19 +00:00
Prefill add-on 47a84985f5 Translated using Weblate (Korean)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ko/
2026-06-05 04:49:18 +00:00
Prefill add-on 57a86709ec Translated using Weblate (Chinese (Simplified Han script))
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/zh_Hans/
2026-06-05 04:49:18 +00:00
Prefill add-on d37c1cc139 Translated using Weblate (Tamil)
Currently translated at 45.5% (215 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ta/
2026-06-05 04:49:17 +00:00
Prefill add-on fcdcc07668 Translated using Weblate (Serbian)
Currently translated at 13.1% (62 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/sr/
2026-06-05 04:49:16 +00:00
Prefill add-on 9bfda12147 Translated using Weblate (Estonian)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/et/
2026-06-05 04:49:16 +00:00
Prefill add-on 84366f9642 Translated using Weblate (Persian)
Currently translated at 15.8% (75 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/fa/
2026-06-05 04:49:15 +00:00
Prefill add-on a213dba276 Translated using Weblate (Japanese)
Currently translated at 25.6% (121 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ja/
2026-06-05 04:49:14 +00:00
Prefill add-on 831723dfc2 Translated using Weblate (Vietnamese)
Currently translated at 18.6% (88 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/vi/
2026-06-05 04:49:14 +00:00
Prefill add-on 0ddfb278df Translated using Weblate (Norwegian Bokmål)
Currently translated at 13.1% (62 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/nb_NO/
2026-06-05 04:49:13 +00:00
Prefill add-on c475b5c702 Translated using Weblate (Finnish)
Currently translated at 30.7% (145 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/fi/
2026-06-05 04:49:12 +00:00
Prefill add-on 4b669ce8ec Translated using Weblate (Alemannic)
Currently translated at 13.1% (62 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/gsw/
2026-06-05 04:49:12 +00:00
Prefill add-on 00f63ff914 Translated using Weblate (Czech)
Currently translated at 78.8% (372 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/cs/
2026-06-05 04:49:11 +00:00
Prefill add-on 2341d7a229 Translated using Weblate (Ukrainian)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/uk/
2026-06-05 04:49:10 +00:00
Prefill add-on 02bb01de76 Translated using Weblate (German)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/de/
2026-06-05 04:49:10 +00:00
Prefill add-on a0e42fd77e Translated using Weblate (Georgian)
Currently translated at 28.3% (134 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ka/
2026-06-05 04:49:09 +00:00
Prefill add-on 356fba697c Translated using Weblate (Thai (Northern))
Currently translated at 13.1% (62 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/nod/
2026-06-05 04:49:08 +00:00
Prefill add-on 1fad725e87 Translated using Weblate (Spanish)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/es/
2026-06-05 04:49:07 +00:00
Prefill add-on 4ade7b2e64 Translated using Weblate (Polish)
Currently translated at 80.0% (378 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/pl/
2026-06-05 04:49:07 +00:00
Prefill add-on 8f60cf90ea Translated using Weblate (Hungarian)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/hu/
2026-06-05 04:49:06 +00:00
Prefill add-on 3bb16c307f Translated using Weblate (Italian)
Currently translated at 79.0% (373 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/it/
2026-06-05 04:49:05 +00:00
Prefill add-on 0d4c5a2731 Translated using Weblate (Danish)
Currently translated at 13.3% (63 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/da/
2026-06-05 04:49:05 +00:00
Prefill add-on 3a968cf23c Translated using Weblate (Portuguese (Brazil))
Currently translated at 36.0% (170 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/pt_BR/
2026-06-05 04:49:04 +00:00
Prefill add-on d18798ab56 Translated using Weblate (Russian)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ru/
2026-06-05 04:49:03 +00:00
Prefill add-on fe32e32783 Translated using Weblate (Indonesian)
Currently translated at 79.8% (377 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/id/
2026-06-05 04:49:02 +00:00
Prefill add-on d7b01f8b36 Translated using Weblate (Turkish)
Currently translated at 39.4% (186 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/tr/
2026-06-05 04:49:02 +00:00
Prefill add-on 9702210303 Translated using Weblate (Thai)
Currently translated at 13.5% (64 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/th/
2026-06-05 04:49:01 +00:00
Prefill add-on 403425dab7 Translated using Weblate (Portuguese (Portugal))
Currently translated at 37.2% (176 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/pt_PT/
2026-06-05 04:49:00 +00:00
Prefill add-on f5e71157f4 Translated using Weblate (Chinese (Traditional Han script))
Currently translated at 76.9% (363 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/zh_Hant/
2026-06-05 04:49:00 +00:00
Prefill add-on 999a269fcb Translated using Weblate (Dutch)
Currently translated at 77.3% (365 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/nl/
2026-06-05 04:48:59 +00:00
Prefill add-on eaf6421e82 Translated using Weblate (Arabic)
Currently translated at 13.1% (62 of 472 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/ar/
2026-06-05 04:48:59 +00:00
Prefill add-on 219c036e8b Translated using Weblate (Turkish)
Currently translated at 4.3% (1 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/tr/
2026-06-05 04:48:58 +00:00
Prefill add-on bd3268dcbc Translated using Weblate (Czech)
Currently translated at 39.1% (9 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/cs/
2026-06-05 04:48:57 +00:00
Prefill add-on c22deeee96 Translated using Weblate (Japanese)
Currently translated at 4.3% (1 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/ja/
2026-06-05 04:48:57 +00:00
Prefill add-on 0966b724d6 Translated using Weblate (Alemannic)
Currently translated at 0.0% (0 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/gsw/
2026-06-05 04:48:56 +00:00
Prefill add-on f9b73591e4 Translated using Weblate (Hungarian)
Currently translated at 82.6% (19 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/hu/
2026-06-05 04:48:56 +00:00
Prefill add-on 60dc43ebd5 Translated using Weblate (Persian)
Currently translated at 69.5% (16 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/fa/
2026-06-05 04:48:55 +00:00
Prefill add-on 69b7fd98de Translated using Weblate (Spanish)
Currently translated at 82.6% (19 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/es/
2026-06-05 04:48:55 +00:00
Prefill add-on 3b7c0880fc Translated using Weblate (French)
Currently translated at 82.6% (19 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/fr/
2026-06-05 04:48:55 +00:00
Prefill add-on ac69ff85d4 Translated using Weblate (Vietnamese)
Currently translated at 17.3% (4 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/vi/
2026-06-05 04:48:54 +00:00
Prefill add-on 031bdba9dd Translated using Weblate (Italian)
Currently translated at 82.6% (19 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/it/
2026-06-05 04:48:54 +00:00
Prefill add-on 51b3330154 Translated using Weblate (Finnish)
Currently translated at 4.3% (1 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/fi/
2026-06-05 04:48:53 +00:00
Prefill add-on 71e66380b4 Translated using Weblate (Thai)
Currently translated at 8.6% (2 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/th/
2026-06-05 04:48:53 +00:00
Prefill add-on aaad893d95 Translated using Weblate (Portuguese (Brazil))
Currently translated at 17.3% (4 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/pt_BR/
2026-06-05 04:48:52 +00:00
Prefill add-on dd40b166bc Translated using Weblate (Arabic)
Currently translated at 4.3% (1 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/ar/
2026-06-05 04:48:52 +00:00
Prefill add-on 3e0cdb0b11 Translated using Weblate (Slovak)
Currently translated at 8.6% (2 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/sk/
2026-06-05 04:48:51 +00:00
Prefill add-on 6a8fbdda5c Translated using Weblate (Danish)
Currently translated at 4.3% (1 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/da/
2026-06-05 04:48:51 +00:00
Prefill add-on c6e023e217 Translated using Weblate (Ukrainian)
Currently translated at 82.6% (19 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/uk/
2026-06-05 04:48:50 +00:00
Prefill add-on 154355c071 Translated using Weblate (German)
Currently translated at 82.6% (19 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/de/
2026-06-05 04:48:50 +00:00
Prefill add-on 4eaa9fb481 Translated using Weblate (Chinese (Simplified Han script))
Currently translated at 4.3% (1 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/zh_Hans/
2026-06-05 04:48:49 +00:00
Prefill add-on 255f1b3c23 Translated using Weblate (Polish)
Currently translated at 82.6% (19 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/pl/
2026-06-05 04:48:49 +00:00
Prefill add-on eaa32eb8cb Translated using Weblate (Tamil)
Currently translated at 4.3% (1 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/ta/
2026-06-05 04:48:48 +00:00
Prefill add-on 0a2b57782f Translated using Weblate (Estonian)
Currently translated at 82.6% (19 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/et/
2026-06-05 04:48:48 +00:00
Prefill add-on 32d7305297 Translated using Weblate (Russian)
Currently translated at 21.7% (5 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/ru/
2026-06-05 04:48:47 +00:00
Prefill add-on 38d1ee76ce Translated using Weblate (Indonesian)
Currently translated at 82.6% (19 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/id/
2026-06-05 04:48:46 +00:00
Prefill add-on c08f92f798 Translated using Weblate (Dutch)
Currently translated at 13.0% (3 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/nl/
2026-06-05 04:48:46 +00:00
Prefill add-on d35adeb2dd Translated using Weblate (Portuguese (Portugal))
Currently translated at 39.1% (9 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/pt_PT/
2026-06-05 04:48:45 +00:00
Prefill add-on 97935d6ff9 Translated using Weblate (Korean)
Currently translated at 82.6% (19 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/ko/
2026-06-05 04:48:45 +00:00
Prefill add-on fb0bb144f9 Translated using Weblate (Serbian)
Currently translated at 4.3% (1 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/sr/
2026-06-05 04:48:44 +00:00
Prefill add-on d264eee841 Translated using Weblate (Thai (Northern))
Currently translated at 0.0% (0 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/nod/
2026-06-05 04:48:44 +00:00
Prefill add-on 6cef71bf37 Translated using Weblate (Georgian)
Currently translated at 4.3% (1 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/ka/
2026-06-05 04:48:43 +00:00
Prefill add-on 1a26f6f68c Translated using Weblate (Norwegian Bokmål)
Currently translated at 4.3% (1 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/nb_NO/
2026-06-05 04:48:43 +00:00
Matthaiks 0795a20e9c Translated using Weblate (Polish)
Currently translated at 100.0% (426 of 426 strings)

Translation: WG Tunnel/strings
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/strings/pl/
2026-06-05 04:48:42 +00:00
Prefill add-on 076bd7f6df Translated using Weblate (Chinese (Traditional Han script))
Currently translated at 13.0% (3 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/zh_Hant/
2026-06-05 04:48:42 +00:00
Prefill add-on 2f2864dade Translated using Weblate (Urdu)
Currently translated at 21.7% (5 of 23 strings)

Translation: WG Tunnel/fastlane
Translate-URL: https://hosted.weblate.org/projects/wg-tunnel/fastlane/ur/
2026-06-05 04:48:41 +00:00
356 changed files with 4436 additions and 6209 deletions
+3 -3
View File
@@ -70,16 +70,16 @@ jobs:
outputs:
UPLOAD_DIR_ANDROID: ${{ env.UPLOAD_DIR_ANDROID }}
steps:
- uses: actions/checkout@v7
- uses: actions/checkout@v6
with:
fetch-depth: 0
submodules: recursive
- name: Set up JDK 21
- name: Set up JDK 17
uses: actions/setup-java@v5
with:
distribution: 'temurin'
java-version: '21'
java-version: '17'
cache: gradle
- name: Grant execute permission for gradlew
+3 -6
View File
@@ -72,15 +72,15 @@ jobs:
outputs:
UPLOAD_DIR_ANDROID: ${{ env.UPLOAD_DIR_ANDROID }}
steps:
- uses: actions/checkout@v7
- uses: actions/checkout@v6
with:
fetch-depth: 0
submodules: recursive
- name: Set up JDK 21
- name: Set up JDK 17
uses: actions/setup-java@v5
with:
distribution: 'temurin'
java-version: '21'
java-version: '17'
cache: gradle
- name: Grant execute permission for gradlew
run: chmod +x gradlew
@@ -112,9 +112,6 @@ jobs:
./gradlew :app:assemble${flavor^}Debug --stacktrace
;;
esac
env:
GITHUB_SHA: ${{ github.sha }}
GITHUB_RUN_NUMBER: ${{ github.run_number }}
- name: Get release apk path
id: apk-path
run: echo "path=$(find . -regex '^.*/build/outputs/apk/${{ inputs.flavor }}/${{ inputs.build_type }}/.*\.apk$' -type f | head -1 | tail -c+2)" >> $GITHUB_OUTPUT
+2 -2
View File
@@ -16,7 +16,7 @@ jobs:
has_new_commits: ${{ steps.check.outputs.new_commits }}
steps:
- name: Checkout Repository
uses: actions/checkout@v7
uses: actions/checkout@v6
- name: Check for new commits
id: check
env:
@@ -43,7 +43,7 @@ jobs:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v7
- uses: actions/checkout@v6
with:
submodules: recursive
+6 -11
View File
@@ -1,30 +1,25 @@
name: on-pr
permissions:
contents: read
on:
workflow_dispatch:
pull_request:
workflow_dispatch:
pull_request:
jobs:
format_check:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v7
- name: Verify Gradle Wrapper
uses: gradle/actions/wrapper-validation@v6
- name: Set up JDK 21
- uses: actions/checkout@v6
- name: Set up JDK 17
uses: actions/setup-java@v5
with:
distribution: 'temurin'
java-version: '21'
java-version: '17'
cache: gradle
- name: Grant execute permission for gradlew
run: chmod +x gradlew
- name: Run ktfmt
run: ./gradlew ktfmtCheck
run: ./gradlew ktfmtCheck
+35 -39
View File
@@ -78,7 +78,7 @@ jobs:
name: publish-github
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v7
- uses: actions/checkout@v6
with:
ref: ${{ github.event_name == 'push' && github.ref || 'master' }}
- name: Install system dependencies
@@ -187,61 +187,57 @@ jobs:
repository: wgtunnel/fdroid
event-type: fdroid-update
build-google-aab:
if: >-
${{
github.event_name == 'push' ||
inputs.track != 'none'
}}
uses: ./.github/workflows/build-aab.yml
secrets: inherit
with:
build_type: release
flavor: google
publish-play:
if: ${{ github.event_name == 'push' || inputs.track != 'none' }}
name: Publish to Google Play
runs-on: ubuntu-latest
needs: build-google-aab
env:
SIGNING_KEY_ALIAS: ${{ secrets.SIGNING_KEY_ALIAS }}
SIGNING_KEY_PASSWORD: ${{ secrets.SIGNING_KEY_PASSWORD }}
SIGNING_STORE_PASSWORD: ${{ secrets.SIGNING_STORE_PASSWORD }}
KEY_STORE_FILE: 'android_keystore.jks'
KEY_STORE_LOCATION: ${{ github.workspace }}/app/keystore/
steps:
- uses: actions/checkout@v7
- uses: actions/checkout@v6
- name: Set up JDK 17
uses: actions/setup-java@v5
with:
ref: ${{ github.event_name == 'push' && github.ref || 'master' }}
distribution: 'temurin'
java-version: '17'
cache: gradle
- name: Download AAB artifact
uses: actions/download-artifact@v8
- name: Grant execute permission for gradlew
run: chmod +x gradlew
# Here we need to decode keystore.jks from base64 string and place it
# in the folder specified in the release signing configuration
- name: Decode Keystore
id: decode_keystore
uses: timheuer/base64-to-file@v2.0
with:
name: google-play-aab
path: ${{ github.workspace }}/aab
fileName: ${{ env.KEY_STORE_FILE }}
fileDir: ${{ env.KEY_STORE_LOCATION }}
encodedString: ${{ secrets.KEYSTORE }}
- name: Find exact AAB file path
id: find-aab
# create keystore path for gradle to read
- name: Create keystore path env var
run: |
AAB_PATH=$(find "${{ github.workspace }}/aab" -name "*.aab" -type f | head -1)
if [ -z "$AAB_PATH" ]; then
echo "ERROR: No .aab file found after download!"
find "${{ github.workspace }}/aab" -type f
exit 1
fi
echo "Found AAB: $AAB_PATH"
echo "aab_path=$AAB_PATH" >> $GITHUB_OUTPUT
store_path=${{ env.KEY_STORE_LOCATION }}${{ env.KEY_STORE_FILE }}
echo "KEY_STORE_PATH=$store_path" >> $GITHUB_ENV
- name: Create service_account.json
id: createServiceAccount
run: echo '${{ secrets.SERVICE_ACCOUNT_JSON }}' > service_account.json
- name: Set up Ruby
- name: Deploy with fastlane
uses: ruby/setup-ruby@v1
with:
ruby-version: '3.4'
ruby-version: '3.2' # Not needed with a .ruby-version file
bundler-cache: true
- name: Upload to Google Play
- name: Distribute app to Prod track 🚀
run: |
track=${{ github.event_name == 'push' && 'production' || inputs.track }}
bundle exec fastlane run upload_to_play_store \
track:"$track" \
aab:"${{ steps.find-aab.outputs.aab_path }}" \
json_key:"service_account.json" \
package_name:"com.zaneschepke.wireguardautotunnel" \
skip_upload_apk:true
(cd ${{ github.workspace }} && bundle install && bundle exec fastlane $track --verbose)
+1 -2
View File
@@ -1,4 +1,3 @@
source "https://rubygems.org"
gem "fastlane"
gem "multi_json"
gem "fastlane"
+17 -14
View File
@@ -49,8 +49,8 @@ and [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/)
## About
WG Tunnel is an alternative Android client for WireGuard and AmneziaWG, inspired by the official WireGuard Android app. It fills gaps in the official client by adding advanced features like auto-tunneling, AmneziaWG support, different app modes like **Lockdown** (a custom kill switch for leak prevention), and **Local Proxy** (expose a tunnel over a local SOCKS5/HTTP proxy server) for enhanced privacy, censorship resistance, and flexibility.
WG Tunnel is an alternative Android client for WireGuard and AmneziaWG, inspired by the official WireGuard Android app. It fills gaps in the official client by adding advanced features like auto-tunneling (on-demand VPN activation), while seamlessly supporting both protocols across app modes—including Kernel (for direct WireGuard kernel integration; AmneziaWG not supported), VPN (standard system-level tunneling), Lockdown (a custom kill switch for leak prevention), and Proxy (built-in HTTP/SOCKS5 forwarding)—for enhanced privacy, censorship resistance, and flexibility.
</div>
<div style="text-align: left;">
@@ -67,18 +67,21 @@ WG Tunnel is an alternative Android client for WireGuard and AmneziaWG, inspired
## Features
- **Auto-Tunneling:** Automatically activate tunnels based on your device's active network details.
- **Deferred Endpoint Bootstrapping:** Safely resolves endpoints and updates peers after the tunnel is up for better reliability and leak protection on startup.
- **Handshake Monitoring:** Real-time handshake monitoring for instant tunnel health feedback.
- **AmneziaWG Support:** Full support for AmneziaWG 2.0, providing robust censorship protection.
- **Split Tunneling:** Flexible support for routing specific apps or traffic through the VPN.
- **Local Proxy Mode:** Expose WireGuard tunnels over a local SOCKS5 or HTTP proxy to browsers or firewall apps (like AdGuard).
- **Lockdown Mode:** Advanced in-app kill switch that blocks all traffic while the tunnel is down.
- **Quick Controls:** Quick Settings tile and home screen shortcuts for easy toggling.
- **Remote Control Support:** Intent-based automation for controlling tunnels and auto-tunneling from automation apps (like Tasker).
- **Dynamic DNS Handling:** Automatically detect and update endpoints on server IP changes without requiring a restart.
- **IPv6 Endpoints:** Automatically upgrade to IPv6 endpoints or fall back to IPv4 based on network conditions without requiring a restart.
- **Android TV Support:** Full support for nearly all features on Android TV.
- **Tunnel Import Methods**: Easily add tunnels using .conf files, ZIP archives, manual entry, or QR code scanning.
- **Auto-Tunneling**: Automatically activate tunnels based on Wi-Fi SSID, Ethernet connections, or mobile data networks.
- **Split Tunneling**: Flexible support for routing specific apps or traffic through the VPN.
- **WireGuard Modes**: Full compatibility with WireGuard in both kernel and userspace implementations.
- **AmneziaWG Integration**: Userspace mode for AmneziaWG, providing robust censorship evasion.
- **Always-On VPN**: Ensures continuous protection with Android's Always-On VPN feature.
- **Quick Controls**: Quick Settings tile and home screen shortcuts for easy VPN toggling.
- **Automation Support**: Intent-based automation for controlling tunnels.
- **Auto-Restore**: Seamlessly restores auto-tunneling and active tunnels after device restarts or app updates.
- **Proxying Options**: Built-in HTTP and SOCKS5 proxy support within tunnels.
- **Lockdown Mode**: Custom kill switch for maximum leak prevention and security.
- **Dynamic DNS Handling**: Detects and updates DNS changes without tunnel restarts.
- **Monitoring Tools**: Advanced tunnel monitoring features for tunnel performance monitoring.
- **Android TV Support**: Android TV support for secure streaming and browsing.
- **Advanced DNS**: DNS over HTTPS support for tunnel endpoint resolutions.
## Building
+47 -53
View File
@@ -1,5 +1,6 @@
import com.android.build.api.dsl.ApplicationExtension
import com.android.build.api.variant.FilterConfiguration
import org.jetbrains.kotlin.gradle.dsl.JvmTarget
plugins {
alias(libs.plugins.android.application)
@@ -8,7 +9,6 @@ plugins {
alias(libs.plugins.compose.compiler)
alias(libs.plugins.grgit)
alias(libs.plugins.licensee)
alias(libs.plugins.aboutlibraries)
}
ksp {
@@ -18,14 +18,15 @@ ksp {
licensee {
allowedLicenses().forEach { allow(it) }
allowedLicenseUrls().forEach { allowUrl(it) }
allowDependency("com.github.T8RIN.QuickieExtended", "quickie-foss", "1.18.1") {
because("FOSS library, but JitPack doesn't publish license metadata")
allow("Apache-2.0")
}
// foss, but missing licenses
ignoreDependencies("com.github.T8RIN.QuickieExtended")
ignoreDependencies("com.github.topjohnwu.libsu")
}
allowDependency("com.github.topjohnwu.libsu", "core", "6.0.0") {
because("FOSS library, but JitPack doesn't publish license metadata")
allow("Apache-2.0")
kotlin {
compilerOptions {
jvmTarget = JvmTarget.JVM_17
freeCompilerArgs = listOf("-XXLanguage:+PropertyParamAnnotationDefaultTargetMode")
}
}
@@ -45,11 +46,10 @@ configure<ApplicationExtension> {
splits {
abi {
val noSplits = providers.gradleProperty("noSplits").isPresent
isEnable = !noSplits
isEnable = !project.hasProperty("noSplits")
reset()
include("armeabi-v7a", "arm64-v8a")
isUniversalApk = !noSplits
isUniversalApk = !project.hasProperty("noSplits")
}
}
@@ -57,17 +57,14 @@ configure<ApplicationExtension> {
applicationId = Constants.APP_ID
minSdk = Constants.MIN_SDK
targetSdk = Constants.TARGET_SDK
versionCode = Constants.VERSION_CODE
versionName = Constants.VERSION_NAME
experimentalProperties["android.experimental.disableGitVersion"] = true
versionCode = computeVersionCode()
versionName = computeVersionName()
sourceSets {
getByName("debug").assets.directories += "$projectDir/schemas"
}
val languagesProvider = project.languageListProvider()
val languagesArray = buildLanguagesArray(languagesProvider.get())
val languagesArray = buildLanguagesArray(languageList())
buildConfigField("String[]", "LANGUAGES", "new String[]{ $languagesArray }")
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
@@ -101,24 +98,21 @@ configure<ApplicationExtension> {
"proguard-rules.pro",
)
signingConfig = signingConfigs.getByName(Constants.RELEASE)
manifestPlaceholders["providerAuthority"] = "${Constants.APP_NAME}.provider"
buildConfigField("String", "FILE_PROVIDER_AUTHORITY", "\"${Constants.APP_NAME}.provider\"")
resValue("string", "provider", "\"${Constants.APP_NAME}.provider\"")
}
debug {
applicationIdSuffix = ".debug"
resValue("string", "app_name", "WG Tunnel Debug")
isDebuggable = true
manifestPlaceholders["providerAuthority"] = "${Constants.APP_NAME}.provider.debug"
buildConfigField("String", "FILE_PROVIDER_AUTHORITY", "\"${Constants.APP_NAME}.provider.debug\"")
resValue("string", "provider", "\"${Constants.APP_NAME}.provider.debug\"")
}
create(Constants.NIGHTLY) {
initWith(buildTypes.getByName(Constants.RELEASE))
applicationIdSuffix = ".nightly"
resValue("string", "app_name", "WG Tunnel Nightly")
manifestPlaceholders["providerAuthority"] = "${Constants.APP_NAME}.provider.nightly"
buildConfigField("String", "FILE_PROVIDER_AUTHORITY", "\"${Constants.APP_NAME}.provider.nightly\"")
resValue("string", "provider", "\"${Constants.APP_NAME}.provider.nightly\"")
}
}
@@ -140,6 +134,8 @@ configure<ApplicationExtension> {
compileOptions {
isCoreLibraryDesugaringEnabled = true
sourceCompatibility = JavaVersion.VERSION_17
targetCompatibility = JavaVersion.VERSION_17
}
buildFeatures {
@@ -152,42 +148,31 @@ configure<ApplicationExtension> {
androidComponents {
onVariants { variant ->
val isNightly = project.isNightlyBuild()
if (isNightly) {
variant.outputs.forEach { output ->
output.versionCode.set(
output.versionCode.get() + project.getVersionCodeIncrement()
)
val currentVersion = output.versionName.get()
val nextVersion = bumpToNextPatchVersion(currentVersion)
val gitHash = project.getGitCommitHash()
output.versionName.set("$nextVersion-nightly+git.$gitHash")
}
}
val abiNameMap = mapOf(
"armeabi-v7a" to "armv7",
"arm64-v8a" to "arm64",
"x86" to "x86",
"x86_64" to "x64",
)
val abiNameMap =
mapOf(
"armeabi-v7a" to "armv7",
"arm64-v8a" to "arm64",
"x86" to "x86",
"x86_64" to "x64",
)
variant.outputs.forEach { output ->
val abi = output.filters.find { it.filterType == FilterConfiguration.FilterType.ABI }?.identifier
val flavorName = variant.productFlavors.joinToString("-") { it.second }
val versionName = output.versionName.get()
val baseFileName = "${Constants.APP_NAME}-${flavorName}-v${versionName}"
val outputFileName = if (!abi.isNullOrEmpty()) {
val shortAbiName = abiNameMap.getOrDefault(abi, abi)
"${baseFileName}-${shortAbiName}.apk"
} else {
"${baseFileName}.apk"
}
val outputFileName =
if (!abi.isNullOrEmpty()) {
val shortAbiName = abiNameMap.getOrDefault(abi, abi)
"${baseFileName}-${shortAbiName}.apk"
} else {
"${baseFileName}.apk"
}
output.outputFileName.set(outputFileName)
}
@@ -240,8 +225,6 @@ dependencies {
// UI utilities
implementation(libs.bundles.ui.utilities)
implementation(libs.lottie.compose)
implementation(libs.sonner)
implementation(libs.aboutlibraries.compose)
// Misc utilities
implementation(libs.bundles.misc.utilities)
@@ -282,6 +265,17 @@ dependencies {
implementation(libs.koin.worker)
}
tasks.register<Copy>("copyLicenseeJsonToAssets") {
dependsOn("licensee")
val outputAssets = layout.projectDirectory.dir("src/main/assets")
from(layout.buildDirectory.file("reports/licensee/androidFdroidRelease/artifacts.json")) {
rename("artifacts.json", "licenses.json")
}
into(outputAssets)
}
tasks.named("preBuild") { dependsOn("copyLicenseeJsonToAssets") }
// https://gist.github.com/obfusk/61046e09cee352ae6dd109911534b12e#fix-proposed-by-linsui-disable-baseline-profiles
tasks.configureEach {
if (name.contains("ArtProfile")) {
@@ -1,513 +0,0 @@
{
"formatVersion": 1,
"database": {
"version": 31,
"identityHash": "1dee3799f1c6526c48723fd2fee58d11",
"entities": [
{
"tableName": "tunnel_config",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `name` TEXT NOT NULL, `tunnel_networks` TEXT NOT NULL DEFAULT '', `is_mobile_data_tunnel` INTEGER NOT NULL DEFAULT false, `is_primary_tunnel` INTEGER NOT NULL DEFAULT false, `quick_config` TEXT NOT NULL DEFAULT '', `dynamic_dns` INTEGER NOT NULL DEFAULT false, `is_ethernet_tunnel` INTEGER NOT NULL DEFAULT false, `prefer_ipv6` INTEGER NOT NULL DEFAULT false, `position` INTEGER NOT NULL DEFAULT 0, `auto_tunnel_apps` TEXT NOT NULL DEFAULT '[]', `is_metered` INTEGER NOT NULL DEFAULT false, `ipv4_fallback` INTEGER NOT NULL DEFAULT false, `ipv6_restore` INTEGER NOT NULL DEFAULT false)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "name",
"columnName": "name",
"affinity": "TEXT",
"notNull": true
},
{
"fieldPath": "tunnelNetworks",
"columnName": "tunnel_networks",
"affinity": "TEXT",
"notNull": true,
"defaultValue": "''"
},
{
"fieldPath": "isMobileDataTunnel",
"columnName": "is_mobile_data_tunnel",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "isPrimaryTunnel",
"columnName": "is_primary_tunnel",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "quickConfig",
"columnName": "quick_config",
"affinity": "TEXT",
"notNull": true,
"defaultValue": "''"
},
{
"fieldPath": "dynamicDnsEnabled",
"columnName": "dynamic_dns",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "isEthernetTunnel",
"columnName": "is_ethernet_tunnel",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "isIpv6Preferred",
"columnName": "prefer_ipv6",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "position",
"columnName": "position",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "autoTunnelApps",
"columnName": "auto_tunnel_apps",
"affinity": "TEXT",
"notNull": true,
"defaultValue": "'[]'"
},
{
"fieldPath": "isMetered",
"columnName": "is_metered",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "ipv4FallbackEnabled",
"columnName": "ipv4_fallback",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "ipv6RestoreEnabled",
"columnName": "ipv6_restore",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
},
"indices": [
{
"name": "index_tunnel_config_name",
"unique": true,
"columnNames": [
"name"
],
"orders": [],
"createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_tunnel_config_name` ON `${TABLE_NAME}` (`name`)"
}
]
},
{
"tableName": "proxy_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `socks5_proxy_enabled` INTEGER NOT NULL DEFAULT 0, `socks5_proxy_bind_address` TEXT, `http_proxy_enable` INTEGER NOT NULL DEFAULT 0, `http_proxy_bind_address` TEXT, `proxy_username` TEXT, `proxy_password` TEXT)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "socks5ProxyEnabled",
"columnName": "socks5_proxy_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "socks5ProxyBindAddress",
"columnName": "socks5_proxy_bind_address",
"affinity": "TEXT"
},
{
"fieldPath": "httpProxyEnabled",
"columnName": "http_proxy_enable",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "httpProxyBindAddress",
"columnName": "http_proxy_bind_address",
"affinity": "TEXT"
},
{
"fieldPath": "proxyUsername",
"columnName": "proxy_username",
"affinity": "TEXT"
},
{
"fieldPath": "proxyPassword",
"columnName": "proxy_password",
"affinity": "TEXT"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
},
{
"tableName": "general_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `is_shortcuts_enabled` INTEGER NOT NULL DEFAULT 0, `is_restore_on_boot_enabled` INTEGER NOT NULL DEFAULT 0, `is_multi_tunnel_enabled` INTEGER NOT NULL DEFAULT 0, `global_split_tunnel_enabled` INTEGER NOT NULL DEFAULT 0, `app_mode` INTEGER NOT NULL DEFAULT 0, `theme` TEXT NOT NULL DEFAULT 'AUTOMATIC', `locale` TEXT, `remote_key` TEXT, `is_remote_control_enabled` INTEGER NOT NULL DEFAULT 0, `is_pin_lock_enabled` INTEGER NOT NULL DEFAULT 0, `is_always_on_vpn_enabled` INTEGER NOT NULL DEFAULT 0, `already_donated` INTEGER NOT NULL DEFAULT 0, `screen_recording_security` INTEGER NOT NULL DEFAULT 1, `global_amnezia_enabled` INTEGER NOT NULL DEFAULT 0, `tunnel_scripting_enabled` INTEGER NOT NULL DEFAULT 0)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "isShortcutsEnabled",
"columnName": "is_shortcuts_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isRestoreOnBootEnabled",
"columnName": "is_restore_on_boot_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isMultiTunnelEnabled",
"columnName": "is_multi_tunnel_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isGlobalSplitTunnelEnabled",
"columnName": "global_split_tunnel_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "tunnelMode",
"columnName": "app_mode",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "theme",
"columnName": "theme",
"affinity": "TEXT",
"notNull": true,
"defaultValue": "'AUTOMATIC'"
},
{
"fieldPath": "locale",
"columnName": "locale",
"affinity": "TEXT"
},
{
"fieldPath": "remoteKey",
"columnName": "remote_key",
"affinity": "TEXT"
},
{
"fieldPath": "isRemoteControlEnabled",
"columnName": "is_remote_control_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isPinLockEnabled",
"columnName": "is_pin_lock_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isAlwaysOnVpnEnabled",
"columnName": "is_always_on_vpn_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "alreadyDonated",
"columnName": "already_donated",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "screenRecordingSecurityEnabled",
"columnName": "screen_recording_security",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "1"
},
{
"fieldPath": "isGlobalAmneziaEnabled",
"columnName": "global_amnezia_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "tunnelScriptingEnabled",
"columnName": "tunnel_scripting_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
},
{
"tableName": "auto_tunnel_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `is_tunnel_enabled` INTEGER NOT NULL DEFAULT 0, `is_tunnel_on_mobile_data_enabled` INTEGER NOT NULL DEFAULT 0, `trusted_network_ssids` TEXT NOT NULL DEFAULT '', `is_tunnel_on_ethernet_enabled` INTEGER NOT NULL DEFAULT 0, `is_tunnel_on_wifi_enabled` INTEGER NOT NULL DEFAULT 0, `is_wildcards_enabled` INTEGER NOT NULL DEFAULT 0, `is_stop_on_no_internet_enabled` INTEGER NOT NULL DEFAULT 0, `is_tunnel_on_unsecure_enabled` INTEGER NOT NULL DEFAULT 0, `wifi_detection_method` INTEGER NOT NULL DEFAULT 0, `start_on_boot` INTEGER NOT NULL DEFAULT 0, `disable_on_captive_portal` INTEGER NOT NULL DEFAULT 1)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "isAutoTunnelEnabled",
"columnName": "is_tunnel_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isTunnelOnMobileDataEnabled",
"columnName": "is_tunnel_on_mobile_data_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "trustedNetworkSSIDs",
"columnName": "trusted_network_ssids",
"affinity": "TEXT",
"notNull": true,
"defaultValue": "''"
},
{
"fieldPath": "isTunnelOnEthernetEnabled",
"columnName": "is_tunnel_on_ethernet_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isTunnelOnWifiEnabled",
"columnName": "is_tunnel_on_wifi_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isWildcardsEnabled",
"columnName": "is_wildcards_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isStopOnNoInternetEnabled",
"columnName": "is_stop_on_no_internet_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isTunnelOnUnsecureEnabled",
"columnName": "is_tunnel_on_unsecure_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "wifiDetectionMethod",
"columnName": "wifi_detection_method",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "startOnBoot",
"columnName": "start_on_boot",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "disableTunnelOnCaptivePortal",
"columnName": "disable_on_captive_portal",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "1"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
},
{
"tableName": "monitoring_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `is_local_logs_enabled` INTEGER NOT NULL DEFAULT 0, `tunnel_statistics_enabled` INTEGER NOT NULL DEFAULT 1, `tunnel_statistics_poll_interval` INTEGER NOT NULL DEFAULT 3)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "isLocalLogsEnabled",
"columnName": "is_local_logs_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "tunnelStatisticsEnabled",
"columnName": "tunnel_statistics_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "1"
},
{
"fieldPath": "tunnelStatisticsPollInterval",
"columnName": "tunnel_statistics_poll_interval",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "3"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
},
{
"tableName": "dns_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `dns_protocol` INTEGER NOT NULL DEFAULT 0, `dns_endpoint` TEXT, `global_tunnel_dns_enabled` INTEGER NOT NULL DEFAULT 0)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "dnsProtocol",
"columnName": "dns_protocol",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "dnsEndpoint",
"columnName": "dns_endpoint",
"affinity": "TEXT"
},
{
"fieldPath": "isGlobalTunnelDnsEnabled",
"columnName": "global_tunnel_dns_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
},
{
"tableName": "lockdown_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `bypass_lan` INTEGER NOT NULL DEFAULT 0, `metered` INTEGER NOT NULL DEFAULT 0, `dual_stack` INTEGER NOT NULL DEFAULT 0)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "bypassLan",
"columnName": "bypass_lan",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "metered",
"columnName": "metered",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "dualStack",
"columnName": "dual_stack",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
}
],
"setupQueries": [
"CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)",
"INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, '1dee3799f1c6526c48723fd2fee58d11')"
]
}
}
@@ -1,520 +0,0 @@
{
"formatVersion": 1,
"database": {
"version": 32,
"identityHash": "fd4803fc483f41704303be9246dcfb4d",
"entities": [
{
"tableName": "tunnel_config",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `name` TEXT NOT NULL, `tunnel_networks` TEXT NOT NULL DEFAULT '', `is_mobile_data_tunnel` INTEGER NOT NULL DEFAULT false, `is_primary_tunnel` INTEGER NOT NULL DEFAULT false, `quick_config` TEXT NOT NULL DEFAULT '', `dynamic_dns` INTEGER NOT NULL DEFAULT false, `is_ethernet_tunnel` INTEGER NOT NULL DEFAULT false, `prefer_ipv6` INTEGER NOT NULL DEFAULT false, `position` INTEGER NOT NULL DEFAULT 0, `auto_tunnel_apps` TEXT NOT NULL DEFAULT '[]', `is_metered` INTEGER NOT NULL DEFAULT false, `ipv4_fallback` INTEGER NOT NULL DEFAULT false, `ipv6_restore` INTEGER NOT NULL DEFAULT false)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "name",
"columnName": "name",
"affinity": "TEXT",
"notNull": true
},
{
"fieldPath": "tunnelNetworks",
"columnName": "tunnel_networks",
"affinity": "TEXT",
"notNull": true,
"defaultValue": "''"
},
{
"fieldPath": "isMobileDataTunnel",
"columnName": "is_mobile_data_tunnel",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "isPrimaryTunnel",
"columnName": "is_primary_tunnel",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "quickConfig",
"columnName": "quick_config",
"affinity": "TEXT",
"notNull": true,
"defaultValue": "''"
},
{
"fieldPath": "dynamicDnsEnabled",
"columnName": "dynamic_dns",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "isEthernetTunnel",
"columnName": "is_ethernet_tunnel",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "isIpv6Preferred",
"columnName": "prefer_ipv6",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "position",
"columnName": "position",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "autoTunnelApps",
"columnName": "auto_tunnel_apps",
"affinity": "TEXT",
"notNull": true,
"defaultValue": "'[]'"
},
{
"fieldPath": "isMetered",
"columnName": "is_metered",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "ipv4FallbackEnabled",
"columnName": "ipv4_fallback",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
},
{
"fieldPath": "ipv6RestoreEnabled",
"columnName": "ipv6_restore",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "false"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
},
"indices": [
{
"name": "index_tunnel_config_name",
"unique": true,
"columnNames": [
"name"
],
"orders": [],
"createSql": "CREATE UNIQUE INDEX IF NOT EXISTS `index_tunnel_config_name` ON `${TABLE_NAME}` (`name`)"
}
]
},
{
"tableName": "proxy_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `socks5_proxy_enabled` INTEGER NOT NULL DEFAULT 0, `socks5_proxy_bind_address` TEXT, `http_proxy_enable` INTEGER NOT NULL DEFAULT 0, `http_proxy_bind_address` TEXT, `proxy_username` TEXT, `proxy_password` TEXT)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "socks5ProxyEnabled",
"columnName": "socks5_proxy_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "socks5ProxyBindAddress",
"columnName": "socks5_proxy_bind_address",
"affinity": "TEXT"
},
{
"fieldPath": "httpProxyEnabled",
"columnName": "http_proxy_enable",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "httpProxyBindAddress",
"columnName": "http_proxy_bind_address",
"affinity": "TEXT"
},
{
"fieldPath": "proxyUsername",
"columnName": "proxy_username",
"affinity": "TEXT"
},
{
"fieldPath": "proxyPassword",
"columnName": "proxy_password",
"affinity": "TEXT"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
},
{
"tableName": "general_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `is_shortcuts_enabled` INTEGER NOT NULL DEFAULT 0, `is_restore_on_boot_enabled` INTEGER NOT NULL DEFAULT 0, `is_multi_tunnel_enabled` INTEGER NOT NULL DEFAULT 0, `global_split_tunnel_enabled` INTEGER NOT NULL DEFAULT 0, `app_mode` INTEGER NOT NULL DEFAULT 0, `theme` TEXT NOT NULL DEFAULT 'AUTOMATIC', `locale` TEXT, `remote_key` TEXT, `is_remote_control_enabled` INTEGER NOT NULL DEFAULT 0, `is_pin_lock_enabled` INTEGER NOT NULL DEFAULT 0, `is_always_on_vpn_enabled` INTEGER NOT NULL DEFAULT 0, `already_donated` INTEGER NOT NULL DEFAULT 0, `screen_recording_security` INTEGER NOT NULL DEFAULT 1, `global_amnezia_enabled` INTEGER NOT NULL DEFAULT 0, `tunnel_scripting_enabled` INTEGER NOT NULL DEFAULT 0, `seamless_roaming_enabled` INTEGER NOT NULL DEFAULT 0)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "isShortcutsEnabled",
"columnName": "is_shortcuts_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isRestoreOnBootEnabled",
"columnName": "is_restore_on_boot_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isMultiTunnelEnabled",
"columnName": "is_multi_tunnel_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isGlobalSplitTunnelEnabled",
"columnName": "global_split_tunnel_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "tunnelMode",
"columnName": "app_mode",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "theme",
"columnName": "theme",
"affinity": "TEXT",
"notNull": true,
"defaultValue": "'AUTOMATIC'"
},
{
"fieldPath": "locale",
"columnName": "locale",
"affinity": "TEXT"
},
{
"fieldPath": "remoteKey",
"columnName": "remote_key",
"affinity": "TEXT"
},
{
"fieldPath": "isRemoteControlEnabled",
"columnName": "is_remote_control_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isPinLockEnabled",
"columnName": "is_pin_lock_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isAlwaysOnVpnEnabled",
"columnName": "is_always_on_vpn_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "alreadyDonated",
"columnName": "already_donated",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "screenRecordingSecurityEnabled",
"columnName": "screen_recording_security",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "1"
},
{
"fieldPath": "isGlobalAmneziaEnabled",
"columnName": "global_amnezia_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "tunnelScriptingEnabled",
"columnName": "tunnel_scripting_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "seamlessRoamingEnabled",
"columnName": "seamless_roaming_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
},
{
"tableName": "auto_tunnel_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `is_tunnel_enabled` INTEGER NOT NULL DEFAULT 0, `is_tunnel_on_mobile_data_enabled` INTEGER NOT NULL DEFAULT 0, `trusted_network_ssids` TEXT NOT NULL DEFAULT '', `is_tunnel_on_ethernet_enabled` INTEGER NOT NULL DEFAULT 0, `is_tunnel_on_wifi_enabled` INTEGER NOT NULL DEFAULT 0, `is_wildcards_enabled` INTEGER NOT NULL DEFAULT 0, `is_stop_on_no_internet_enabled` INTEGER NOT NULL DEFAULT 0, `is_tunnel_on_unsecure_enabled` INTEGER NOT NULL DEFAULT 0, `wifi_detection_method` INTEGER NOT NULL DEFAULT 0, `start_on_boot` INTEGER NOT NULL DEFAULT 0, `disable_on_captive_portal` INTEGER NOT NULL DEFAULT 1)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "isAutoTunnelEnabled",
"columnName": "is_tunnel_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isTunnelOnMobileDataEnabled",
"columnName": "is_tunnel_on_mobile_data_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "trustedNetworkSSIDs",
"columnName": "trusted_network_ssids",
"affinity": "TEXT",
"notNull": true,
"defaultValue": "''"
},
{
"fieldPath": "isTunnelOnEthernetEnabled",
"columnName": "is_tunnel_on_ethernet_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isTunnelOnWifiEnabled",
"columnName": "is_tunnel_on_wifi_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isWildcardsEnabled",
"columnName": "is_wildcards_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isStopOnNoInternetEnabled",
"columnName": "is_stop_on_no_internet_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "isTunnelOnUnsecureEnabled",
"columnName": "is_tunnel_on_unsecure_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "wifiDetectionMethod",
"columnName": "wifi_detection_method",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "startOnBoot",
"columnName": "start_on_boot",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "disableTunnelOnCaptivePortal",
"columnName": "disable_on_captive_portal",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "1"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
},
{
"tableName": "monitoring_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `is_local_logs_enabled` INTEGER NOT NULL DEFAULT 0, `tunnel_statistics_enabled` INTEGER NOT NULL DEFAULT 1, `tunnel_statistics_poll_interval` INTEGER NOT NULL DEFAULT 3)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "isLocalLogsEnabled",
"columnName": "is_local_logs_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "tunnelStatisticsEnabled",
"columnName": "tunnel_statistics_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "1"
},
{
"fieldPath": "tunnelStatisticsPollInterval",
"columnName": "tunnel_statistics_poll_interval",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "3"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
},
{
"tableName": "dns_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `dns_protocol` INTEGER NOT NULL DEFAULT 0, `dns_endpoint` TEXT, `global_tunnel_dns_enabled` INTEGER NOT NULL DEFAULT 0)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "dnsProtocol",
"columnName": "dns_protocol",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "dnsEndpoint",
"columnName": "dns_endpoint",
"affinity": "TEXT"
},
{
"fieldPath": "isGlobalTunnelDnsEnabled",
"columnName": "global_tunnel_dns_enabled",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
},
{
"tableName": "lockdown_settings",
"createSql": "CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`id` INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, `bypass_lan` INTEGER NOT NULL DEFAULT 0, `metered` INTEGER NOT NULL DEFAULT 0, `dual_stack` INTEGER NOT NULL DEFAULT 0)",
"fields": [
{
"fieldPath": "id",
"columnName": "id",
"affinity": "INTEGER",
"notNull": true
},
{
"fieldPath": "bypassLan",
"columnName": "bypass_lan",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "metered",
"columnName": "metered",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
},
{
"fieldPath": "dualStack",
"columnName": "dual_stack",
"affinity": "INTEGER",
"notNull": true,
"defaultValue": "0"
}
],
"primaryKey": {
"autoGenerate": true,
"columnNames": [
"id"
]
}
}
],
"setupQueries": [
"CREATE TABLE IF NOT EXISTS room_master_table (id INTEGER PRIMARY KEY,identity_hash TEXT)",
"INSERT OR REPLACE INTO room_master_table (id,identity_hash) VALUES(42, 'fd4803fc483f41704303be9246dcfb4d')"
]
}
}
+7 -13
View File
@@ -53,6 +53,7 @@
android:fullBackupContent="@xml/backup_rules"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/Theme.App.Start"
tools:targetApi="tiramisu">
@@ -73,13 +74,6 @@
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
<category android:name="android.intent.category.BROWSABLE" />
<data android:scheme="wg" />
</intent-filter>
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<action android:name="android.intent.action.SHOW_APP_INFO" />
@@ -156,7 +150,7 @@
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="${providerAuthority}"
android:authorities="@string/provider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
@@ -175,9 +169,9 @@
tools:node="remove" />
</provider>
<service
android:name=".service.tile.TunnelControlTile"
android:name=".core.service.tile.TunnelControlTile"
android:exported="true"
android:icon="@drawable/qs_logo"
android:icon="@drawable/ic_notification"
android:label="@string/tunnel_control"
android:permission="android.permission.BIND_QUICK_SETTINGS_TILE">
<meta-data
@@ -192,9 +186,9 @@
</intent-filter>
</service>
<service
android:name=".service.tile.AutoTunnelControlTile"
android:name=".core.service.tile.AutoTunnelControlTile"
android:exported="true"
android:icon="@drawable/qs_logo"
android:icon="@drawable/ic_notification"
android:label="@string/auto_tunnel"
android:permission="android.permission.BIND_QUICK_SETTINGS_TILE">
<meta-data
@@ -209,7 +203,7 @@
</intent-filter>
</service>
<service
android:name=".service.autotunnel.AutoTunnelService"
android:name=".core.service.autotunnel.AutoTunnelService"
android:enabled="true"
android:exported="false"
android:foregroundServiceType="specialUse"
View File
@@ -1,14 +1,12 @@
package com.zaneschepke.wireguardautotunnel
import ProxySettingsScreen
import android.Manifest
import android.content.Intent
import android.graphics.Color
import android.net.Uri
import android.net.VpnService
import android.os.Build
import android.os.Bundle
import android.provider.Settings
import android.view.WindowManager
import androidx.activity.SystemBarStyle
import androidx.activity.compose.rememberLauncherForActivityResult
@@ -21,29 +19,17 @@ import androidx.compose.animation.fadeOut
import androidx.compose.animation.slideInHorizontally
import androidx.compose.animation.slideOutHorizontally
import androidx.compose.animation.togetherWith
import androidx.compose.foundation.BorderStroke
import androidx.compose.foundation.background
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.PaddingValues
import androidx.compose.foundation.layout.consumeWindowInsets
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.shape.RoundedCornerShape
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.outlined.CheckCircleOutline
import androidx.compose.material.icons.outlined.ErrorOutline
import androidx.compose.material.icons.outlined.FavoriteBorder
import androidx.compose.material.icons.outlined.Info
import androidx.compose.material.icons.outlined.WarningAmber
import androidx.compose.material3.ButtonDefaults
import androidx.compose.foundation.layout.wrapContentHeight
import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Scaffold
import androidx.compose.material3.Text
import androidx.compose.material3.TextButton
import androidx.compose.material3.surfaceColorAtElevation
import androidx.compose.runtime.CompositionLocalProvider
import androidx.compose.runtime.LaunchedEffect
@@ -52,19 +38,20 @@ import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberCoroutineScope
import androidx.compose.runtime.saveable.rememberSaveable
import androidx.compose.runtime.setValue
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Brush
import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.stringResource
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.text.LinkAnnotation
import androidx.compose.ui.text.SpanStyle
import androidx.compose.ui.text.TextLinkStyles
import androidx.compose.ui.text.buildAnnotatedString
import androidx.compose.ui.text.intl.Locale
import androidx.compose.ui.unit.IntOffset
import androidx.compose.ui.text.style.TextDecoration
import androidx.compose.ui.text.withLink
import androidx.compose.ui.unit.dp
import androidx.compose.ui.zIndex
import androidx.core.app.ActivityCompat
import androidx.core.splashscreen.SplashScreen.Companion.installSplashScreen
import androidx.lifecycle.lifecycleScope
import androidx.lifecycle.viewmodel.navigation3.rememberViewModelStoreNavEntryDecorator
@@ -72,10 +59,6 @@ import androidx.navigation3.runtime.entryProvider
import androidx.navigation3.runtime.rememberNavBackStack
import androidx.navigation3.runtime.rememberSaveableStateHolderNavEntryDecorator
import androidx.navigation3.ui.NavDisplay
import com.dokar.sonner.TextToastAction
import com.dokar.sonner.ToastType
import com.dokar.sonner.Toaster
import com.dokar.sonner.rememberToasterState
import com.zaneschepke.networkmonitor.NetworkMonitor
import com.zaneschepke.wireguardautotunnel.data.AppDatabase
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
@@ -86,9 +69,11 @@ import com.zaneschepke.wireguardautotunnel.domain.sideeffect.GlobalSideEffect
import com.zaneschepke.wireguardautotunnel.ui.LocalIsAndroidTV
import com.zaneschepke.wireguardautotunnel.ui.LocalNavController
import com.zaneschepke.wireguardautotunnel.ui.common.banner.AppAlertBanner
import com.zaneschepke.wireguardautotunnel.ui.common.dialog.InfoDialog
import com.zaneschepke.wireguardautotunnel.ui.common.dialog.LocalNetworkPermissionDialog
import com.zaneschepke.wireguardautotunnel.ui.common.dialog.VpnDeniedDialog
import com.zaneschepke.wireguardautotunnel.ui.common.snackbar.CustomSnackBar
import com.zaneschepke.wireguardautotunnel.ui.common.snackbar.SnackbarInfo
import com.zaneschepke.wireguardautotunnel.ui.common.snackbar.SnackbarType
import com.zaneschepke.wireguardautotunnel.ui.common.snackbar.rememberCustomSnackbarState
import com.zaneschepke.wireguardautotunnel.ui.navigation.Route
import com.zaneschepke.wireguardautotunnel.ui.navigation.SecureRoute
import com.zaneschepke.wireguardautotunnel.ui.navigation.Tab
@@ -125,29 +110,19 @@ import com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.settings.ipv6.IPv6
import com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.sort.SortScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.splittunnel.SplitTunnelScreen
import com.zaneschepke.wireguardautotunnel.ui.theme.AlertRed
import com.zaneschepke.wireguardautotunnel.ui.theme.Heart
import com.zaneschepke.wireguardautotunnel.ui.theme.OffWhite
import com.zaneschepke.wireguardautotunnel.ui.theme.SilverTree
import com.zaneschepke.wireguardautotunnel.ui.theme.Straw
import com.zaneschepke.wireguardautotunnel.ui.theme.WireguardAutoTunnelTheme
import com.zaneschepke.wireguardautotunnel.util.FileUtils
import com.zaneschepke.wireguardautotunnel.util.LocaleUtil
import com.zaneschepke.wireguardautotunnel.util.StringValue
import com.zaneschepke.wireguardautotunnel.util.extensions.installApk
import com.zaneschepke.wireguardautotunnel.util.extensions.isRunningOnTv
import com.zaneschepke.wireguardautotunnel.util.extensions.openWebUrl
import com.zaneschepke.wireguardautotunnel.util.extensions.restartApp
import com.zaneschepke.wireguardautotunnel.util.permission.LocalNetworkPermissionHelper
import com.zaneschepke.wireguardautotunnel.util.extensions.showToast
import com.zaneschepke.wireguardautotunnel.viewmodel.ConfigEditViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.SplitTunnelViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.TunnelViewModel
import de.raphaelebner.roomdatabasebackup.core.OnCompleteListener.Companion.EXIT_CODE_ERROR
import de.raphaelebner.roomdatabasebackup.core.OnCompleteListener.Companion.EXIT_CODE_ERROR_DECRYPTION_ERROR
import de.raphaelebner.roomdatabasebackup.core.OnCompleteListener.Companion.EXIT_CODE_ERROR_RESTORE_BACKUP_IS_ENCRYPTED
import de.raphaelebner.roomdatabasebackup.core.OnCompleteListener.Companion.EXIT_CODE_ERROR_WRONG_DECRYPTION_PASSWORD
import de.raphaelebner.roomdatabasebackup.core.RoomBackup
import kotlin.time.Duration.Companion.milliseconds
import kotlinx.coroutines.awaitCancellation
import kotlinx.coroutines.delay
import kotlinx.coroutines.flow.collectLatest
@@ -157,7 +132,6 @@ import org.koin.androidx.compose.koinViewModel
import org.koin.androidx.viewmodel.ext.android.viewModel
import org.koin.core.parameter.parametersOf
import org.orbitmvi.orbit.compose.collectAsState
import timber.log.Timber
import xyz.teamgravity.pin_lock_compose.PinManager
class MainActivity : AppCompatActivity() {
@@ -181,10 +155,9 @@ class MainActivity : AppCompatActivity() {
}
super.onCreate(savedInstanceState)
roomBackup = RoomBackup(this).database(appDatabase).enableLogDebug(true).maxFileCount(5)
handleIncomingIntent(intent)
handleConfigFileIntent(intent)
handleWgDeepLinkIntent(intent)
roomBackup = RoomBackup(this)
installSplashScreen().apply {
setKeepOnScreenCondition { !viewModel.container.stateFlow.value.isAppLoaded }
@@ -202,54 +175,12 @@ class MainActivity : AppCompatActivity() {
}
}
val toaster = rememberToasterState()
val snackbarState = rememberCustomSnackbarState()
var showVpnPermissionDialog by remember { mutableStateOf(false) }
var vpnPermissionDenied by remember { mutableStateOf(false) }
var requestingTunnelMode by remember {
mutableStateOf<Pair<TunnelMode?, TunnelConfig?>>(Pair(null, null))
}
var showLocalNetworkRationale by remember { mutableStateOf(false) }
var hasPromptedLocalNetwork by rememberSaveable { mutableStateOf(false) }
val localNetworkPermissionLauncher =
rememberLauncherForActivityResult(
contract = ActivityResultContracts.RequestPermission()
) { isGranted ->
if (!isGranted) {
val canAskAgain =
ActivityCompat.shouldShowRequestPermissionRationale(
this,
Manifest.permission.ACCESS_LOCAL_NETWORK,
)
if (!canAskAgain) {
val intent =
Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS).apply {
data = Uri.fromParts("package", packageName, null)
}
startActivity(intent)
} else {
toaster.show(
message =
context.getString(R.string.local_network_permission_denied),
type = ToastType.Warning,
duration = 6000.milliseconds,
)
}
}
}
LaunchedEffect(uiState.isAppLoaded) {
if (
uiState.isAppLoaded &&
!hasPromptedLocalNetwork &&
LocalNetworkPermissionHelper.shouldRequestPermission() &&
!LocalNetworkPermissionHelper.isPermissionGranted(context)
) {
hasPromptedLocalNetwork = true
showLocalNetworkRationale = true
}
}
val startingStack = buildList {
add(Route.Tunnels)
@@ -301,19 +232,23 @@ class MainActivity : AppCompatActivity() {
}
is GlobalSideEffect.Snackbar -> {
when (sideEffect.type) {
ToastType.Warning,
ToastType.Error -> toaster.dismissAll()
else -> Unit
scope.launch {
snackbarState.showSnackbar(
SnackbarInfo(
message =
buildAnnotatedString {
append(sideEffect.message.asString(context))
},
type = sideEffect.type ?: SnackbarType.INFO,
durationMs = sideEffect.durationMs ?: 4000L,
)
)
}
toaster.show(
message = sideEffect.message.asString(context),
type = sideEffect.type,
duration = (sideEffect.durationMs ?: 4000L).milliseconds,
)
}
is GlobalSideEffect.Toast ->
scope.launch { context.showToast(sideEffect.message.asString(context)) }
is GlobalSideEffect.LaunchUrl -> context.openWebUrl(sideEffect.url)
is GlobalSideEffect.InstallApk -> context.installApk(sideEffect.apk)
}
@@ -340,58 +275,49 @@ class MainActivity : AppCompatActivity() {
},
)
if (showLocalNetworkRationale) {
LocalNetworkPermissionDialog(
onDismiss = {
showLocalNetworkRationale = false
toaster.show(
message =
context.getString(R.string.local_network_permission_denied),
type = ToastType.Warning,
duration = 6000.milliseconds,
)
},
onAttest = {
showLocalNetworkRationale = false
localNetworkPermissionLauncher.launch(
Manifest.permission.ACCESS_LOCAL_NETWORK
)
},
)
}
uiState.pendingWgImportUrl?.let { url ->
val host = Uri.parse(url).host ?: url
InfoDialog(
onDismiss = { viewModel.dismissWgImport() },
onAttest = { viewModel.importFromUrl(url) },
title = stringResource(R.string.add_from_url),
body = { Text(stringResource(R.string.wg_url_confirm_message, host)) },
confirmText = stringResource(R.string.okay),
)
val annotatedMessage = buildAnnotatedString {
append(context.getString(R.string.donation_prompt_prefix))
append(" ")
withLink(
LinkAnnotation.Clickable(
tag = context.getString(R.string.support),
styles =
TextLinkStyles(
style =
SpanStyle(
textDecoration = TextDecoration.Underline,
color = MaterialTheme.colorScheme.primary,
),
focusedStyle =
SpanStyle(
textDecoration = TextDecoration.Underline,
color = MaterialTheme.colorScheme.primary,
background =
MaterialTheme.colorScheme.primary.copy(
alpha = 0.2f
),
),
),
) {
snackbarState.dismissCurrent()
navController.push(Route.Donate)
}
) {
append(context.getString(R.string.donation_prompt_link))
}
append(" ")
append(context.getString(R.string.donation_prompt_suffix))
}
LaunchedEffect(Unit) {
if (uiState.shouldShowDonationSnackbar && !uiState.alreadyDonated) {
viewModel.setShouldShowDonationSnackbar(false)
toaster.show(
message =
context.getString(R.string.donation_prompt_prefix) +
" " +
context.getString(R.string.donation_prompt_link) +
" " +
context.getString(R.string.donation_prompt_suffix),
type = ToastType.Normal,
duration = 30_000L.milliseconds,
action =
TextToastAction(
text = context.getString(R.string.donate_title),
onClick = { toastId ->
toaster.dismiss(toastId)
navController.push(Route.Donate)
},
),
snackbarState.showSnackbar(
SnackbarInfo(
message = annotatedMessage,
type = SnackbarType.THANK_YOU,
durationMs = 30_000L,
)
)
}
}
@@ -452,6 +378,25 @@ class MainActivity : AppCompatActivity() {
)
}
Scaffold(
snackbarHost = {
snackbarState.SnackbarHost(
modifier =
Modifier.align(Alignment.BottomCenter)
.padding(bottom = 80.dp)
) { info ->
CustomSnackBar(
message = info.message,
type = info.type,
onDismiss = { snackbarState.dismissCurrent() },
containerColor =
MaterialTheme.colorScheme.surfaceColorAtElevation(
2.dp
),
modifier =
Modifier.wrapContentHeight(align = Alignment.Top),
)
}
},
topBar = { DynamicTopAppBar(navState) },
bottomBar = {
if (navState.showBottomItems) {
@@ -603,70 +548,6 @@ class MainActivity : AppCompatActivity() {
)
}
}
Toaster(
state = toaster,
alignment = Alignment.BottomCenter,
offset = IntOffset(0, -220),
richColors = true,
background = {
Brush.linearGradient(
listOf(
MaterialTheme.colorScheme.surfaceColorAtElevation(2.dp),
MaterialTheme.colorScheme.surfaceColorAtElevation(2.dp),
)
)
},
elevation = 1.dp,
shadowAmbientColor =
MaterialTheme.colorScheme.onSurface.copy(alpha = 0.08f),
shadowSpotColor =
MaterialTheme.colorScheme.onSurface.copy(alpha = 0.12f),
border = {
BorderStroke(
0.dp,
androidx.compose.ui.graphics.Color.Transparent,
)
},
actionSlot = { toast ->
(toast.action as? TextToastAction)?.let { action ->
TextButton(
onClick = { action.onClick(toast) },
colors =
ButtonDefaults.textButtonColors(
contentColor = MaterialTheme.colorScheme.primary
),
contentPadding = PaddingValues(horizontal = 12.dp),
) {
Text(text = action.text, fontWeight = FontWeight.Medium)
}
}
},
iconSlot = { toast ->
val (icon, color) =
when (toast.type) {
ToastType.Success ->
Icons.Outlined.CheckCircleOutline to SilverTree
ToastType.Error ->
Icons.Outlined.ErrorOutline to AlertRed
ToastType.Warning ->
Icons.Outlined.WarningAmber to Straw
ToastType.Info ->
Icons.Outlined.Info to
MaterialTheme.colorScheme.onSurface
ToastType.Normal ->
Icons.Outlined.FavoriteBorder to Heart
}
Icon(
imageVector = icon,
contentDescription = null,
tint = color,
modifier = Modifier.padding(end = 12.dp),
)
},
contentColor = { MaterialTheme.colorScheme.onSurface },
shape = { RoundedCornerShape(16.dp) },
showCloseButton = true,
)
}
}
}
@@ -674,14 +555,55 @@ class MainActivity : AppCompatActivity() {
}
}
private fun handleWgDeepLinkIntent(intent: Intent) {
if (intent.action == Intent.ACTION_VIEW) {
val uri = intent.data ?: return
if (uri.scheme == "wg") {
val httpsUrl = uri.toString().replaceFirst("wg://", "https://")
viewModel.promptWgImport(httpsUrl)
fun performBackup() = lifecycleScope.launch {
roomBackup
.database(appDatabase)
.backupLocation(RoomBackup.BACKUP_FILE_LOCATION_CUSTOM_DIALOG)
.enableLogDebug(true)
.maxFileCount(5)
.apply {
onCompleteListener { success, _, _ ->
lifecycleScope.launch {
if (success) {
showToast(
getString(
R.string.backup_success,
getString(R.string.restarting_app),
)
)
restartApp()
} else {
showToast(R.string.backup_failed)
}
}
}
}
}
.backup()
}
fun performRestore() = lifecycleScope.launch {
roomBackup
.database(appDatabase)
.enableLogDebug(true)
.backupLocation(RoomBackup.BACKUP_FILE_LOCATION_CUSTOM_DIALOG)
.apply {
onCompleteListener { success, _, _ ->
lifecycleScope.launch {
if (success) {
showToast(
getString(
R.string.restore_success,
getString(R.string.restarting_app),
)
)
restartApp()
} else {
showToast(R.string.restore_failed)
}
}
}
}
.restore()
}
override fun onResume() {
@@ -689,105 +611,21 @@ class MainActivity : AppCompatActivity() {
networkMonitor.checkPermissionsAndUpdateState()
}
fun performBackup(encrypt: Boolean = false, password: String? = null) {
roomBackup
.backupLocation(RoomBackup.BACKUP_FILE_LOCATION_CUSTOM_DIALOG)
.apply {
if (encrypt && !password.isNullOrBlank()) {
backupIsEncrypted(true)
customEncryptPassword(password)
}
}
.onCompleteListener { success, _, _ ->
lifecycleScope.launch {
val sideEffect =
if (success) {
GlobalSideEffect.Snackbar(
StringValue.StringResource(R.string.backup_success),
ToastType.Success,
)
} else {
GlobalSideEffect.Snackbar(
StringValue.StringResource(R.string.backup_failed),
ToastType.Error,
)
}
viewModel.postSideEffect(sideEffect)
}
}
.backup()
}
fun performRestore(encrypt: Boolean = false, password: String? = null) {
roomBackup
.backupLocation(RoomBackup.BACKUP_FILE_LOCATION_CUSTOM_DIALOG)
.apply {
if (encrypt && !password.isNullOrBlank()) {
backupIsEncrypted(true)
customEncryptPassword(password)
}
}
.onCompleteListener { success, message, exitCode ->
lifecycleScope.launch {
if (success) {
viewModel.postSideEffect(
GlobalSideEffect.Snackbar(
StringValue.StringResource(R.string.restore_success),
ToastType.Success,
)
)
roomBackup.restartApp(Intent(this@MainActivity, MainActivity::class.java))
} else {
Timber.w("Restore failed, exitCode=$exitCode, message=$message")
val errorMessage =
when (exitCode) {
EXIT_CODE_ERROR_WRONG_DECRYPTION_PASSWORD ->
getString(R.string.restore_failed_wrong_password)
EXIT_CODE_ERROR,
EXIT_CODE_ERROR_DECRYPTION_ERROR,
EXIT_CODE_ERROR_RESTORE_BACKUP_IS_ENCRYPTED ->
getString(R.string.restore_failed_invalid_file)
else -> getString(R.string.restore_failed)
}
viewModel.postSideEffect(
GlobalSideEffect.Snackbar(
StringValue.DynamicString(errorMessage),
ToastType.Error,
)
)
}
}
}
.restore()
}
override fun onNewIntent(intent: Intent) {
super.onNewIntent(intent)
setIntent(intent)
handleConfigFileIntent(intent)
handleWgDeepLinkIntent(intent)
handleIncomingIntent(intent)
}
private fun handleConfigFileIntent(intent: Intent?) {
private fun handleIncomingIntent(intent: Intent?) {
intent ?: return
when (intent.action) {
Intent.ACTION_VIEW,
Intent.ACTION_EDIT,
Intent.ACTION_SEND -> {
val uri: Uri? = intent.data ?: return
val name = uri?.lastPathSegment?.lowercase() ?: return
if (
!name.endsWith(FileUtils.CONF_FILE_EXTENSION) &&
!name.endsWith(FileUtils.ZIP_FILE_EXTENSION)
) {
Timber.d("Ignoring non-config URI in handleIncomingIntent: $uri")
return
}
viewModel.importFromUri(uri)
val uri: Uri? = intent.data
uri?.let { viewModel.importFromUri(it) }
}
}
}
@@ -6,8 +6,11 @@ import com.zaneschepke.tunnel.backend.Backend
import com.zaneschepke.tunnel.di.tunnelModule
import com.zaneschepke.tunnel.service.VpnService
import com.zaneschepke.wireguardautotunnel.core.event.TunnelEventDispatcher
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService
import com.zaneschepke.wireguardautotunnel.core.orchestration.AppBoostrapCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.TunnelCoordinator
import com.zaneschepke.wireguardautotunnel.core.service.tile.AutoTunnelTileRefresher
import com.zaneschepke.wireguardautotunnel.core.service.tile.TunnelTileRefresher
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelProvider
import com.zaneschepke.wireguardautotunnel.di.Dispatcher
import com.zaneschepke.wireguardautotunnel.di.Scope
@@ -18,9 +21,6 @@ import com.zaneschepke.wireguardautotunnel.di.dispatchersModule
import com.zaneschepke.wireguardautotunnel.di.networkModule
import com.zaneschepke.wireguardautotunnel.di.tunnelBackendProviderModule
import com.zaneschepke.wireguardautotunnel.di.workerModule
import com.zaneschepke.wireguardautotunnel.notification.NotificationService
import com.zaneschepke.wireguardautotunnel.service.tile.AutoTunnelTileRefresher
import com.zaneschepke.wireguardautotunnel.service.tile.TunnelTileRefresher
import com.zaneschepke.wireguardautotunnel.util.ReleaseTree
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope
@@ -51,13 +51,6 @@ class WireGuardAutoTunnel : Application(), KoinComponent {
private val backend: Backend by inject()
private val alwaysOnCallback =
object : VpnService.AlwaysOnCallback {
override fun alwaysOnTriggered() {
applicationScope.launch { tunnelCoordinator.startDefault() }
}
}
@OptIn(KoinViewModelScopeApi::class)
override fun onCreate() {
super.onCreate()
@@ -78,10 +71,11 @@ class WireGuardAutoTunnel : Application(), KoinComponent {
lazyModules(networkModule)
}
instance = this
notificationService.createAllChannels()
syncTiles()
// Sync tiles
AutoTunnelTileRefresher.refresh(this)
TunnelTileRefresher.refresh(this)
if (BuildConfig.DEBUG) {
Timber.plant(Timber.DebugTree())
@@ -93,7 +87,13 @@ class WireGuardAutoTunnel : Application(), KoinComponent {
Timber.plant(ReleaseTree())
}
backend.setAlwaysOnCallback(alwaysOnCallback)
backend.setAlwaysOnCallback(
object : VpnService.AlwaysOnCallback {
override fun alwaysOnTriggered() {
applicationScope.launch { tunnelCoordinator.startDefault() }
}
}
)
val dispatcher = get<TunnelEventDispatcher>()
val coordinator = get<TunnelCoordinator>()
@@ -111,11 +111,6 @@ class WireGuardAutoTunnel : Application(), KoinComponent {
applicationScope.launch(ioDispatcher) { boostrapCoordinator.bootstrap() }
}
private fun syncTiles() {
AutoTunnelTileRefresher.refresh(this)
TunnelTileRefresher.refresh(this)
}
companion object {
lateinit var instance: WireGuardAutoTunnel
private set
@@ -3,11 +3,11 @@ package com.zaneschepke.wireguardautotunnel.core.broadcast
import android.content.BroadcastReceiver
import android.content.Context
import android.content.Intent
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService
import com.zaneschepke.wireguardautotunnel.core.orchestration.AutoTunnelCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.TunnelCoordinator
import com.zaneschepke.wireguardautotunnel.di.Scope
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.notification.NotificationService
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch
import org.koin.core.component.KoinComponent
@@ -48,51 +48,47 @@ class RemoteControlReceiver : BroadcastReceiver(), KoinComponent {
}
override fun onReceive(context: Context, intent: Intent) {
val pendingResult = goAsync()
val action = intent.action ?: return
val appAction = Action.fromAction(action) ?: return
applicationScope.launch {
try {
val action = intent.action ?: return@launch
val appAction = Action.fromAction(action) ?: return@launch
val settings = settingsRepository.getGeneralSettings()
val settings = settingsRepository.getGeneralSettings()
if (!settings.isRemoteControlEnabled) return@launch
if (!settings.isRemoteControlEnabled) return@launch
if (!validateKey(settings, intent)) return@launch
if (!validateKey(settings, intent)) return@launch
when (appAction) {
Action.START_TUNNEL -> {
val tunnel =
resolveTunnel(intent)
?: tunnelsRepository.getDefaultTunnel()
?: return@launch
when (appAction) {
Action.START_TUNNEL -> {
val tunnel =
resolveTunnel(intent)
?: tunnelsRepository.getDefaultTunnel()
?: return@launch
tunnelCoordinator.startTunnel(tunnel)
}
Action.STOP_TUNNEL -> {
val tunnelName = intent.getStringExtra(EXTRA_TUN_NAME)
if (tunnelName == null) {
tunnelCoordinator.stopActiveTunnels()
return@launch
}
val tunnel = tunnelsRepository.findByTunnelName(tunnelName) ?: return@launch
tunnelCoordinator.stopTunnel(tunnel.id)
}
Action.START_AUTO_TUNNEL -> {
autoTunnelCoordinator.enable()
}
Action.STOP_AUTO_TUNNEL -> {
autoTunnelCoordinator.disable()
}
tunnelCoordinator.startTunnel(tunnel)
}
Action.STOP_TUNNEL -> {
val tunnelName = intent.getStringExtra(EXTRA_TUN_NAME)
if (tunnelName == null) {
tunnelCoordinator.stopActiveTunnels()
return@launch
}
val tunnel = tunnelsRepository.findByTunnelName(tunnelName) ?: return@launch
tunnelCoordinator.stopTunnel(tunnel.id)
}
Action.START_AUTO_TUNNEL -> {
autoTunnelCoordinator.enable()
}
Action.STOP_AUTO_TUNNEL -> {
autoTunnelCoordinator.disable()
}
} finally {
pendingResult.finish()
}
}
}
@@ -27,31 +27,19 @@ class RestartReceiver : BroadcastReceiver(), KoinComponent {
override fun onReceive(context: Context, intent: Intent) {
Timber.d("RestartReceiver triggered with action: ${intent.action}")
val pendingResult = goAsync()
applicationScope.launch {
try {
when (intent.action) {
Intent.ACTION_BOOT_COMPLETED,
"android.intent.action.QUICKBOOT_POWERON",
"com.htc.intent.action.QUICKBOOT_POWERON" -> {
startupCoordinator.applyStartupPolicy()
}
Intent.ACTION_MY_PACKAGE_REPLACED -> {
Timber.i("Restoring state on package upgrade")
startupCoordinator.applyStartupPolicy()
logReader.deleteAndClearLogs()
appStateRepository.setShouldShowDonationSnackbar(true)
}
else -> {
Timber.w("Unhandled action in RestartReceiver: ${intent.action}")
}
when (intent.action) {
Intent.ACTION_BOOT_COMPLETED,
"android.intent.action.QUICKBOOT_POWERON",
"com.htc.intent.action.QUICKBOOT_POWERON" -> {
startupCoordinator.applyStartupPolicy()
}
Intent.ACTION_MY_PACKAGE_REPLACED -> {
Timber.i("Restoring state on package upgrade")
startupCoordinator.applyStartupPolicy()
logReader.deleteAndClearLogs()
appStateRepository.setShouldShowDonationSnackbar(true)
}
} finally {
pendingResult.finish()
}
}
}
@@ -1,41 +1,29 @@
package com.zaneschepke.wireguardautotunnel.core.event
import android.content.Context
import com.dokar.sonner.ToastType
import com.zaneschepke.tunnel.event.TunnelEvent
import com.zaneschepke.tunnel.model.BackendMode
import com.zaneschepke.tunnel.state.BackendStatus
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.domain.repository.GlobalEffectRepository
import com.zaneschepke.wireguardautotunnel.core.notification.TunnelNotificationLine
import com.zaneschepke.wireguardautotunnel.core.notification.TunnelNotificationService
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.domain.sideeffect.GlobalSideEffect
import com.zaneschepke.wireguardautotunnel.lifecyle.AppVisibilityObserver
import com.zaneschepke.wireguardautotunnel.notification.TunnelNotificationLine
import com.zaneschepke.wireguardautotunnel.notification.TunnelNotificationService
import com.zaneschepke.wireguardautotunnel.ui.state.DisplayTunnelState
import com.zaneschepke.wireguardautotunnel.util.StringValue
import kotlin.time.Duration.Companion.milliseconds
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.FlowPreview
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.combine
import kotlinx.coroutines.flow.debounce
import kotlinx.coroutines.flow.distinctUntilChanged
import kotlinx.coroutines.flow.launchIn
import kotlinx.coroutines.flow.map
import kotlinx.coroutines.flow.onEach
import kotlinx.coroutines.launch
class TunnelEventDispatcher(
private val notificationManager: TunnelNotificationService,
private val tunnelRepository: TunnelRepository,
private val context: Context,
private val appVisibilityObserver: AppVisibilityObserver,
private val globalEffectRepository: GlobalEffectRepository,
) {
@OptIn(FlowPreview::class)
fun bind(
scope: CoroutineScope,
providerEvents: Flow<TunnelEvent>,
@@ -44,174 +32,54 @@ class TunnelEventDispatcher(
tunnelDisplayStates: StateFlow<Map<Int, DisplayTunnelState>>,
) {
// Informational events from tunnel backend
// informational events
providerEvents
.distinctUntilChanged()
.onEach { event ->
when (event) {
is TunnelEvent.FallbackToIpv4 -> {
val name = getTunnelName(event.tunnelId)
showOrNotify(
scope = scope,
foregroundAction = {
globalEffectRepository.post(
GlobalSideEffect.Snackbar(
message =
StringValue.DynamicString(
context.getString(
R.string.notification_ipv4_fallback_message,
name,
)
),
type = ToastType.Info,
)
)
},
backgroundAction = { notificationManager.showIpv4Fallback(name) },
)
notificationManager.showIpv4Fallback(name)
}
is TunnelEvent.RecoveredToIpv6 -> {
val name = getTunnelName(event.tunnelId)
showOrNotify(
scope = scope,
foregroundAction = {
globalEffectRepository.post(
GlobalSideEffect.Snackbar(
message =
StringValue.DynamicString(
context.getString(
R.string.notification_ipv6_recovery_message,
name,
)
),
type = ToastType.Success,
)
)
},
backgroundAction = { notificationManager.showIpv6Recovery(name) },
)
notificationManager.showIpv6Recovery(name)
}
is TunnelEvent.DynamicDnsUpdate -> {
val name = getTunnelName(event.tunnelId)
showOrNotify(
scope = scope,
foregroundAction = {
globalEffectRepository.post(
GlobalSideEffect.Snackbar(
message =
StringValue.DynamicString(
context.getString(
R.string.notification_dynamic_dns_message,
name,
)
),
type = ToastType.Info,
)
)
},
backgroundAction = { notificationManager.showDynamicDnsUpdate(name) },
)
notificationManager.showDynamicDnsUpdate(name)
}
is TunnelEvent.NoRootShellAccess -> {
showOrNotify(
scope = scope,
foregroundAction = {
globalEffectRepository.post(
GlobalSideEffect.Snackbar(
message =
StringValue.DynamicString(
context.getString(R.string.error_root_denied)
),
type = ToastType.Error,
)
)
},
backgroundAction = { notificationManager.showRootShellAccess() },
)
notificationManager.showRootShellAccess()
}
}
}
.launchIn(scope)
// Errors from our tunnel coordinator
// errors from the coordinator
coordinatorErrors
.distinctUntilChanged()
.onEach { error ->
when (error) {
is TunnelErrorEvent.VpnPermissionDenied -> {
showOrNotify(
scope = scope,
foregroundAction = {
globalEffectRepository.post(
GlobalSideEffect.Snackbar(
message =
StringValue.DynamicString(
context.getString(R.string.vpn_permission_required)
),
type = ToastType.Error,
)
)
},
backgroundAction = { notificationManager.showVpnRequired() },
)
notificationManager.showVpnRequired()
}
is TunnelErrorEvent.InternalFailure -> {
showOrNotify(
scope = scope,
foregroundAction = {
globalEffectRepository.post(
GlobalSideEffect.Snackbar(
message = StringValue.DynamicString(error.message),
type = ToastType.Error,
)
)
},
backgroundAction = { notificationManager.showError(error.message) },
)
notificationManager.showError(error.message)
}
is TunnelErrorEvent.Socks5PortUnavailable -> {
val name = getTunnelName(error.tunnelId)
val message =
context.getString(R.string.error_socks5_port_unavailable, error.port)
showOrNotify(
scope = scope,
foregroundAction = {
globalEffectRepository.post(
GlobalSideEffect.Snackbar(
message = StringValue.DynamicString(message),
type = ToastType.Error,
)
)
},
backgroundAction = {
notificationManager.showSocks5PortUnavailable(error.port, name)
},
)
notificationManager.showSocks5PortUnavailable(error.port, name)
}
is TunnelErrorEvent.HttpPortUnavailable -> {
val name = getTunnelName(error.tunnelId)
val message =
context.getString(R.string.error_http_port_unavailable, error.port)
showOrNotify(
scope = scope,
foregroundAction = {
globalEffectRepository.post(
GlobalSideEffect.Snackbar(
message = StringValue.DynamicString(message),
type = ToastType.Error,
)
)
},
backgroundAction = {
notificationManager.showHttpPortUnavailable(error.port, name)
},
)
notificationManager.showHttpPortUnavailable(error.port, name)
}
}
}
@@ -245,7 +113,6 @@ class TunnelEventDispatcher(
.associateBy { it.id }
}
.distinctUntilChanged()
.debounce(500.milliseconds) // give the service notification time to display
.onEach { vpnLines -> notificationManager.updateVpnPersistentNotification(vpnLines) }
.launchIn(scope)
@@ -273,25 +140,12 @@ class TunnelEventDispatcher(
.associateBy { it.id }
}
.distinctUntilChanged()
.debounce(500.milliseconds) // give the service notification time to display
.onEach { proxyLines ->
notificationManager.updateProxyPersistentNotification(proxyLines)
}
.launchIn(scope)
}
private fun showOrNotify(
scope: CoroutineScope,
foregroundAction: suspend () -> Unit,
backgroundAction: () -> Unit,
) {
if (appVisibilityObserver.isForeground.value) {
scope.launch { foregroundAction() }
} else {
backgroundAction()
}
}
private suspend fun getTunnelName(tunnelId: Int): String {
return tunnelRepository.getById(tunnelId)?.name ?: context.getString(R.string.unknown)
}
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.notification
package com.zaneschepke.wireguardautotunnel.core.notification
import android.Manifest
import android.app.Notification
@@ -17,8 +17,8 @@ import androidx.core.app.NotificationManagerCompat.IMPORTANCE_HIGH
import com.zaneschepke.wireguardautotunnel.MainActivity
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.core.broadcast.NotificationActionReceiver
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService.Companion.EXTRA_ID
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.notification.NotificationService.Companion.EXTRA_ID
import com.zaneschepke.wireguardautotunnel.util.StringValue
class AndroidNotificationService(override val context: Context) : NotificationService {
@@ -1,116 +1,21 @@
package com.zaneschepke.wireguardautotunnel.notification
package com.zaneschepke.wireguardautotunnel.core.notification
import androidx.core.app.NotificationCompat
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.core.notification.AndroidNotificationService.NotificationChannels
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService.Companion.PROXY_GROUP_KEY
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService.Companion.PROXY_NOTIFICATION_ID
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService.Companion.TUNNEL_ERROR_NOTIFICATION_ID
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService.Companion.TUNNEL_MESSAGES_NOTIFICATION_ID
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService.Companion.VPN_GROUP_KEY
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService.Companion.VPN_NOTIFICATION_ID
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.notification.AndroidNotificationService.NotificationChannels
import com.zaneschepke.wireguardautotunnel.notification.NotificationService.Companion.PROXY_GROUP_KEY
import com.zaneschepke.wireguardautotunnel.notification.NotificationService.Companion.PROXY_NOTIFICATION_ID
import com.zaneschepke.wireguardautotunnel.notification.NotificationService.Companion.TUNNEL_ERROR_NOTIFICATION_ID
import com.zaneschepke.wireguardautotunnel.notification.NotificationService.Companion.TUNNEL_MESSAGES_NOTIFICATION_ID
import com.zaneschepke.wireguardautotunnel.notification.NotificationService.Companion.VPN_GROUP_KEY
import com.zaneschepke.wireguardautotunnel.notification.NotificationService.Companion.VPN_NOTIFICATION_ID
class AndroidTunnelNotificationService(private val notificationService: NotificationService) :
TunnelNotificationService {
private val context = notificationService.context
private fun createGroupNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine>,
channel: NotificationChannels.Tunnel,
groupKey: String,
): android.app.Notification {
val title =
if (tunnelNotificationLines.size == 1) {
val name = tunnelNotificationLines.values.first().name
when (channel) {
is NotificationChannels.Tunnel.VPN ->
"${context.getString(R.string.vpn)}$name"
is NotificationChannels.Tunnel.Proxy ->
"${context.getString(R.string.proxy)}$name"
}
} else {
when (channel) {
is NotificationChannels.Tunnel.VPN -> context.getString(R.string.vpn)
is NotificationChannels.Tunnel.Proxy -> context.getString(R.string.proxy)
}
}
val formattedLines =
tunnelNotificationLines.values.map { line ->
val status = line.displayState.asLocalizedString(context)
if (tunnelNotificationLines.size == 1) {
status
} else {
context.getString(R.string.notification_tunnel_status_format, line.name, status)
}
}
val description = formattedLines.joinToString("\n")
val actions =
if (tunnelNotificationLines.size == 1) {
val tunnelId = tunnelNotificationLines.keys.first()
listOf(
notificationService.createNotificationAction(
notificationAction = NotificationAction.TUNNEL_OFF,
extraId = tunnelId,
)
)
} else {
listOf(
notificationService.createNotificationAction(
notificationAction = NotificationAction.STOP_ALL,
extraId = null,
)
)
}
val style =
if (tunnelNotificationLines.size > 1) {
NotificationCompat.InboxStyle()
.setBigContentTitle(title)
.setSummaryText(
"${tunnelNotificationLines.size} ${context.getString(R.string.tunnels).lowercase()}"
)
.also { inbox -> formattedLines.forEach { inbox.addLine(it) } }
} else {
null
}
return notificationService.createNotification(
channel = channel,
title = title,
description = description,
actions = actions,
onGoing = true,
onlyAlertOnce = true,
groupKey = groupKey,
style = style,
)
}
override fun buildVpnPersistentNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine>
): android.app.Notification {
return createGroupNotification(
tunnelNotificationLines,
NotificationChannels.Tunnel.VPN,
VPN_GROUP_KEY,
)
}
override fun buildProxyPersistentNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine>
): android.app.Notification {
return createGroupNotification(
tunnelNotificationLines,
NotificationChannels.Tunnel.Proxy,
PROXY_GROUP_KEY,
)
}
private fun updateGroupNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine>,
notificationId: Int,
@@ -183,36 +88,26 @@ class AndroidTunnelNotificationService(private val notificationService: Notifica
notificationService.show(notificationId, notification)
}
override fun updateVpnPersistentNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine>
) {
if (tunnelNotificationLines.isEmpty()) {
notificationService.remove(VPN_NOTIFICATION_ID)
return
}
val notification =
createGroupNotification(
tunnelNotificationLines,
NotificationChannels.Tunnel.VPN,
VPN_GROUP_KEY,
)
notificationService.show(VPN_NOTIFICATION_ID, notification)
}
override fun updateProxyPersistentNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine>
) {
if (tunnelNotificationLines.isEmpty()) {
notificationService.remove(PROXY_NOTIFICATION_ID)
return
}
val notification =
createGroupNotification(
tunnelNotificationLines,
NotificationChannels.Tunnel.Proxy,
PROXY_GROUP_KEY,
)
notificationService.show(PROXY_NOTIFICATION_ID, notification)
updateGroupNotification(
tunnelNotificationLines = tunnelNotificationLines,
notificationId = PROXY_NOTIFICATION_ID,
channel = NotificationChannels.Tunnel.Proxy,
groupKey = PROXY_GROUP_KEY,
)
}
override fun updateVpnPersistentNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine>
) {
updateGroupNotification(
tunnelNotificationLines = tunnelNotificationLines,
notificationId = VPN_NOTIFICATION_ID,
channel = NotificationChannels.Tunnel.VPN,
groupKey = VPN_GROUP_KEY,
)
}
override fun showIpv4Fallback(tunnelName: String) {
@@ -1,10 +1,10 @@
package com.zaneschepke.wireguardautotunnel.notification
package com.zaneschepke.wireguardautotunnel.core.notification
import android.app.Notification
import android.content.Context
import androidx.core.app.NotificationCompat
import com.zaneschepke.wireguardautotunnel.core.notification.AndroidNotificationService.NotificationChannels
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.notification.AndroidNotificationService.NotificationChannels
import com.zaneschepke.wireguardautotunnel.util.StringValue
interface NotificationService {
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.notification
package com.zaneschepke.wireguardautotunnel.core.notification
import com.zaneschepke.wireguardautotunnel.ui.state.DisplayTunnelState
@@ -1,6 +1,4 @@
package com.zaneschepke.wireguardautotunnel.notification
import android.app.Notification
package com.zaneschepke.wireguardautotunnel.core.notification
interface TunnelNotificationService {
@@ -8,14 +6,6 @@ interface TunnelNotificationService {
fun updateVpnPersistentNotification(tunnelNotificationLines: Map<Int, TunnelNotificationLine>)
fun buildVpnPersistentNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine>
): Notification
fun buildProxyPersistentNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine>
): Notification
fun showIpv4Fallback(tunnelName: String)
fun showIpv6Recovery(tunnelName: String)
@@ -39,7 +39,7 @@ class AppBoostrapCoordinator(
listOf(
async { bootstrapDns() },
async { ensureGlobalConfig() },
async { restoreBackendConfiguration() },
async { restoreLockdown() },
)
try {
@@ -73,13 +73,9 @@ class AppBoostrapCoordinator(
tunnelRepository.ensureGlobalConfigExists()
}
private suspend fun restoreBackendConfiguration() {
private suspend fun restoreLockdown() {
val settings = settingsRepository.getGeneralSettings()
if (settings.seamlessRoamingEnabled) {
tunnelProvider.setSeamlessRoaming(true)
}
when (settings.tunnelMode) {
TunnelMode.LOCK_DOWN -> {
val lockdownSettings = lockdownRepository.getLockdownSettings()
@@ -1,8 +1,8 @@
package com.zaneschepke.wireguardautotunnel.core.orchestration
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.service.autotunnel.AutoTunnelStateHolder
import com.zaneschepke.wireguardautotunnel.domain.repository.AutoTunnelSettingsRepository
import com.zaneschepke.wireguardautotunnel.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.service.autotunnel.AutoTunnelStateHolder
class AutoTunnelCoordinator(
private val repository: AutoTunnelSettingsRepository,
@@ -2,6 +2,7 @@ package com.zaneschepke.wireguardautotunnel.core.orchestration
import com.zaneschepke.tunnel.model.BackendMode
import com.zaneschepke.wireguardautotunnel.core.event.TunnelErrorEvent
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelProvider
import com.zaneschepke.wireguardautotunnel.data.repository.RoomDnsSettingsRepository
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelActionSource
@@ -9,16 +10,13 @@ import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
import com.zaneschepke.wireguardautotunnel.domain.events.TunnelActionEvent
import com.zaneschepke.wireguardautotunnel.domain.model.DnsSettings
import com.zaneschepke.wireguardautotunnel.domain.model.GeneralSettings
import com.zaneschepke.wireguardautotunnel.domain.model.LockdownSettings
import com.zaneschepke.wireguardautotunnel.domain.model.MonitoringSettings
import com.zaneschepke.wireguardautotunnel.domain.model.ProxySettings
import com.zaneschepke.wireguardautotunnel.domain.model.TunnelConfig
import com.zaneschepke.wireguardautotunnel.domain.repository.GeneralSettingRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.LockdownSettingsRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.MonitoringSettingsRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.ProxySettingsRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.ui.state.DisplayTunnelState
import kotlin.time.Duration.Companion.milliseconds
import kotlinx.coroutines.CoroutineScope
@@ -46,13 +44,9 @@ class TunnelCoordinator(
dnsSettingsRepository: RoomDnsSettingsRepository,
monitoringSettingsRepository: MonitoringSettingsRepository,
proxyRepository: ProxySettingsRepository,
lockdownModeRepository: LockdownSettingsRepository,
scope: CoroutineScope,
) {
private val _userOverrideFlow = MutableSharedFlow<Unit>(extraBufferCapacity = 1)
val userOverrideFlow = _userOverrideFlow.asSharedFlow()
@OptIn(FlowPreview::class)
val tunnelDisplayStates: StateFlow<Map<Int, DisplayTunnelState>> =
tunnelProvider.backendStatus
@@ -69,7 +63,6 @@ class TunnelCoordinator(
val dns: DnsSettings,
val monitoring: MonitoringSettings,
val proxy: ProxySettings,
val lockdown: LockdownSettings,
)
private val runtimeSettingsSnapshot =
@@ -78,14 +71,12 @@ class TunnelCoordinator(
dnsSettingsRepository.flow,
monitoringSettingsRepository.flow,
proxyRepository.flow,
lockdownModeRepository.flow,
) { general, dns, monitoring, proxy, lockdown ->
) { general, dns, monitoring, proxy ->
RuntimeSettingsSnapshot(
general = general,
dns = dns,
monitoring = monitoring,
proxy = proxy,
lockdown = lockdown,
)
}
@@ -116,34 +107,15 @@ class TunnelCoordinator(
) = tunnelMutex.withLock {
// wait for app to be bootstrapped
bootstrapCoordinator.isReady.first { it }
if (source == TunnelActionSource.USER) {
_userOverrideFlow.tryEmit(Unit)
}
// enforce single tunnel, for now
if (backendStatus.value.activeTunnels.isNotEmpty()) {
stopActiveTunnelsInternal(source)
}
startTunnelInternal(config, source)
}
suspend fun stopTunnel(id: Int, source: TunnelActionSource = TunnelActionSource.USER) =
tunnelMutex.withLock {
if (source == TunnelActionSource.USER) {
_userOverrideFlow.tryEmit(Unit)
}
stopTunnelInternal(id, source)
}
suspend fun stopActiveTunnels(source: TunnelActionSource = TunnelActionSource.USER) =
tunnelMutex.withLock {
if (source == TunnelActionSource.USER) {
_userOverrideFlow.tryEmit(Unit)
}
stopActiveTunnelsInternal(source)
}
suspend fun stopActiveTunnels() = tunnelMutex.withLock { stopActiveTunnelsInternal() }
private suspend fun startTunnelInternal(
tunnelConfig: TunnelConfig,
@@ -155,7 +127,6 @@ class TunnelCoordinator(
val dnsSettings = snapshot.dns
val proxySettings = snapshot.proxy
val monitoringSettings = snapshot.monitoring
val lockdownSettings = snapshot.lockdown
val config = tunnelConfig.getConfig()
val policy =
@@ -191,13 +162,14 @@ class TunnelCoordinator(
}
TunnelMode.LOCK_DOWN -> {
BackendMode.Proxy.KillSwitchPrimary(
runConfig,
lockdownSettings.toKillSwitchConfig(),
)
BackendMode.Proxy.KillSwitchPrimary(runConfig)
}
}
// TODO for now, enforce single tunnel until multi-tunneling is implement
stopActiveTunnelsInternal()
tunnelProvider
.startTunnel(
tunnel =
@@ -221,10 +193,6 @@ class TunnelCoordinator(
suspend fun toggleTunnels(source: TunnelActionSource = TunnelActionSource.USER) =
tunnelMutex.withLock {
if (source == TunnelActionSource.USER) {
_userOverrideFlow.tryEmit(Unit)
}
val active = tunnelProvider.backendStatus.value.activeTunnels
if (active.isNotEmpty()) {
lastActiveTunnels = active.keys.toList()
@@ -233,7 +201,7 @@ class TunnelCoordinator(
_actions.emit(TunnelActionEvent.Stopped(tunnelId = id, source = source))
}
stopActiveTunnelsInternal(source)
stopActiveTunnelsInternal()
return@withLock
}
@@ -258,15 +226,7 @@ class TunnelCoordinator(
.onFailure { _errors.emit(TunnelErrorEvent.from(it, id)) }
}
private suspend fun stopActiveTunnelsInternal(
source: TunnelActionSource = TunnelActionSource.USER
) {
val active = tunnelProvider.backendStatus.value.activeTunnels
active.keys.forEach { id ->
_actions.emit(TunnelActionEvent.Stopped(tunnelId = id, source = source))
}
private suspend fun stopActiveTunnelsInternal() {
tunnelProvider.stopActiveTunnels()
}
}
@@ -5,7 +5,7 @@ import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
import com.zaneschepke.wireguardautotunnel.domain.repository.GeneralSettingRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.LockdownSettingsRepository
class TunnelBackendCoordinator(
class TunnelModeCoordinator(
private val tunnelProvider: TunnelProvider,
private val settingsRepository: GeneralSettingRepository,
private val lockdownRepository: LockdownSettingsRepository,
@@ -42,6 +42,7 @@ class TunnelBackendCoordinator(
when (newMode) {
TunnelMode.LOCK_DOWN -> {
val lockdownSettings = lockdownRepository.getLockdownSettings()
tunnelProvider.setLockDown(lockdownSettings).getOrThrow()
}
@@ -49,9 +50,4 @@ class TunnelBackendCoordinator(
TunnelMode.PROXY -> Unit
}
}
suspend fun changeSeamlessRoaming(enabled: Boolean) {
tunnelProvider.setSeamlessRoaming(enabled).getOrThrow()
settingsRepository.updateSeamlessRoaming(enabled)
}
}
@@ -1,9 +1,9 @@
package com.zaneschepke.wireguardautotunnel.service
package com.zaneschepke.wireguardautotunnel.core.service
import android.content.Context
import android.content.Intent
import android.net.VpnService
import com.zaneschepke.wireguardautotunnel.service.autotunnel.AutoTunnelService
import com.zaneschepke.wireguardautotunnel.core.service.autotunnel.AutoTunnelService
class ServiceManager(private val context: Context) {
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.service.autotunnel
package com.zaneschepke.wireguardautotunnel.core.service.autotunnel
import com.zaneschepke.wireguardautotunnel.domain.events.AutoTunnelEvent
import com.zaneschepke.wireguardautotunnel.domain.model.TunnelConfig
@@ -17,7 +17,6 @@ class AutoTunnelEngine {
}
}
Decision.None -> AutoTunnelEvent.DoNothing
is Decision.StopDueToNoInternet -> AutoTunnelEvent.StopAllDueToNoInternet
}
}
@@ -28,29 +27,13 @@ class AutoTunnelEngine {
val activeTunnelIds = backend.activeTunnels.keys.toSet()
val isOnCaptivePortalWifi =
network.activeNetwork is ActiveNetwork.Wifi &&
network.activeNetwork.requiresCaptivePortalLogin
if (isOnCaptivePortalWifi && settings.disableTunnelOnCaptivePortal) {
return if (activeTunnelIds.isNotEmpty()) {
Decision.Sync(start = emptySet(), stop = activeTunnelIds)
} else {
Decision.None
}
}
if (!network.hasUsableNetwork) {
return if (settings.isStopOnNoInternetEnabled) {
Decision.StopDueToNoInternet
} else {
// keep tunnel state neutral on no internet otherwise
Decision.None
}
}
val desiredTunnels = resolveDesiredTunnels(state).map { it.id }.toSet()
// stop condition overrides everything
if (!network.hasInternet() && settings.isStopOnNoInternetEnabled) {
return Decision.Sync(start = emptySet(), stop = activeTunnelIds)
}
val toStart = desiredTunnels - activeTunnelIds
val toStop = activeTunnelIds - desiredTunnels
@@ -113,7 +96,5 @@ class AutoTunnelEngine {
data class Sync(val start: Set<TunnelConfig>, val stop: Set<Int>) : Decision
data object None : Decision
data object StopDueToNoInternet : Decision
}
}
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.service.autotunnel
package com.zaneschepke.wireguardautotunnel.core.service.autotunnel
import android.content.Intent
import androidx.core.app.ServiceCompat
@@ -7,12 +7,16 @@ import androidx.lifecycle.lifecycleScope
import com.zaneschepke.networkmonitor.AndroidNetworkMonitor
import com.zaneschepke.networkmonitor.StableNetworkEngine
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.core.notification.AndroidNotificationService
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService
import com.zaneschepke.wireguardautotunnel.core.orchestration.TunnelCoordinator
import com.zaneschepke.wireguardautotunnel.core.service.tile.AutoTunnelTileRefresher
import com.zaneschepke.wireguardautotunnel.di.Dispatcher
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelActionSource
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
import com.zaneschepke.wireguardautotunnel.domain.events.AutoTunnelEvent
import com.zaneschepke.wireguardautotunnel.domain.events.TunnelActionEvent
import com.zaneschepke.wireguardautotunnel.domain.model.AutoTunnelSettings
import com.zaneschepke.wireguardautotunnel.domain.model.TunnelConfig
import com.zaneschepke.wireguardautotunnel.domain.repository.AutoTunnelSettingsRepository
@@ -20,22 +24,15 @@ import com.zaneschepke.wireguardautotunnel.domain.repository.GeneralSettingRepos
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.domain.state.AutoTunnelState
import com.zaneschepke.wireguardautotunnel.domain.state.toDomain
import com.zaneschepke.wireguardautotunnel.notification.AndroidNotificationService
import com.zaneschepke.wireguardautotunnel.notification.NotificationService
import com.zaneschepke.wireguardautotunnel.service.tile.AutoTunnelTileRefresher
import com.zaneschepke.wireguardautotunnel.util.Constants
import com.zaneschepke.wireguardautotunnel.util.extensions.to
import kotlin.time.Duration.Companion.milliseconds
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.FlowPreview
import kotlinx.coroutines.Job
import kotlinx.coroutines.delay
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.collectLatest
import kotlinx.coroutines.flow.combine
import kotlinx.coroutines.flow.debounce
import kotlinx.coroutines.flow.distinctUntilChanged
import kotlinx.coroutines.flow.firstOrNull
import kotlinx.coroutines.flow.distinctUntilChangedBy
import kotlinx.coroutines.flow.map
import kotlinx.coroutines.flow.mapNotNull
import kotlinx.coroutines.launch
@@ -66,7 +63,8 @@ class AutoTunnelService : LifecycleService() {
private var autoTunnelJob: Job? = null
private var permissionsJob: Job? = null
private var overridesJob: Job? = null
private var noInternetStopJob: Job? = null
@Volatile private var manualOverrideState = ManualOverrideState()
private data class PermissionWarningState(
val detectionMethod: AndroidNetworkMonitor.WifiDetectionMethod,
@@ -75,19 +73,19 @@ class AutoTunnelService : LifecycleService() {
val ssidReadRequired: Boolean,
)
@Volatile private var hasUserOverride = false
private var lastNetworkFingerprint: AutoTunnelState.NetworkFingerprint? = null
private data class ManualOverrideState(
val fingerprint: AutoTunnelState.NetworkFingerprint? = null,
val stoppedTunnelIds: Set<Int> = emptySet(),
val startedTunnelIds: Set<Int> = emptySet(),
)
@OptIn(FlowPreview::class)
private val autoTunnelStateFlow: Flow<AutoTunnelState> by lazy {
val networkFlow = networkEngine.stableState.mapNotNull { it?.state?.toDomain() }
val settingsFlow = combineSettings()
val backendFlow =
tunnelCoordinator.backendStatus
.distinctUntilChanged { old, new -> old.activeTunnels == new.activeTunnels }
.debounce(300L.milliseconds)
tunnelCoordinator.backendStatus.distinctUntilChangedBy { it.activeTunnels.keys.toSet() }
combine(networkFlow, settingsFlow, backendFlow) { network, settings, backend ->
AutoTunnelState(
@@ -123,7 +121,7 @@ class AutoTunnelService : LifecycleService() {
permissionsJob?.cancel()
permissionsJob = startLocationPermissionsNotificationJob()
overridesJob?.cancel()
overridesJob = startUserOverrideJob()
overridesJob = startOverridesJob()
}
fun stop() {
@@ -132,23 +130,48 @@ class AutoTunnelService : LifecycleService() {
}
override fun onDestroy() {
cancelNoInternetStopJob()
ServiceCompat.stopForeground(this, ServiceCompat.STOP_FOREGROUND_REMOVE)
stateHolder.setActive(false)
AutoTunnelTileRefresher.refresh(this)
super.onDestroy()
}
private fun startUserOverrideJob(): Job =
private fun startOverridesJob(): Job =
lifecycleScope.launch(ioDispatcher) {
tunnelCoordinator.userOverrideFlow.collect {
tunnelCoordinator.actions.collect { action ->
reconciliationMutex.withLock {
if (!hasUserOverride) {
Timber.d(
"User manually overrode Auto Tunnel on current network. Pausing auto decisions."
)
}
hasUserOverride = true
manualOverrideState =
when (action) {
is TunnelActionEvent.Started -> {
if (action.source != TunnelActionSource.USER) {
return@withLock
}
manualOverrideState.copy(
startedTunnelIds =
manualOverrideState.startedTunnelIds + action.tunnelId,
stoppedTunnelIds =
manualOverrideState.stoppedTunnelIds - action.tunnelId,
)
}
is TunnelActionEvent.Stopped -> {
if (action.source != TunnelActionSource.USER) {
return@withLock
}
manualOverrideState.copy(
stoppedTunnelIds =
manualOverrideState.stoppedTunnelIds + action.tunnelId,
startedTunnelIds =
manualOverrideState.startedTunnelIds - action.tunnelId,
)
}
}
Timber.d("Updated manual overrides: $manualOverrideState")
}
}
}
@@ -179,83 +202,50 @@ class AutoTunnelService : LifecycleService() {
)
}
// Instead of stopping tunnel right away on no internet, we kick off this job to add short delay
// and re-evaluation to prevent unwanted stops
// on flaky networks and network transitions
private fun scheduleNoInternetStop() {
noInternetStopJob?.cancel()
noInternetStopJob =
lifecycleScope.launch(ioDispatcher) {
delay(NO_INTERNET_GRACE_PERIOD_MS.milliseconds)
reconciliationMutex.withLock {
val currentNetworkState = networkEngine.stableState.value?.state?.toDomain()
val stillNoUsableNetwork = currentNetworkState?.hasUsableNetwork == false
val stopOnNoInternetEnabled =
autoTunnelRepository.flow.firstOrNull()?.isStopOnNoInternetEnabled == true
if (stillNoUsableNetwork && stopOnNoInternetEnabled) {
val currentActiveIds =
tunnelCoordinator.backendStatus.value.activeTunnels.keys
if (currentActiveIds.isNotEmpty()) {
Timber.w(
"No internet grace period expired and still no internet. Stopping tunnels: $currentActiveIds"
)
currentActiveIds.forEach { tunnelId ->
tunnelCoordinator.stopTunnel(
tunnelId,
TunnelActionSource.AUTO_TUNNEL,
)
}
}
} else {
Timber.d(
"No internet grace period expired, but internet is back or setting disabled. Doing nothing."
)
}
}
}
}
private fun cancelNoInternetStopJob() {
noInternetStopJob?.cancel()
noInternetStopJob = null
}
private fun startAutoTunnelStateJob(): Job =
lifecycleScope.launch(ioDispatcher) {
autoTunnelStateFlow.collectLatest { state ->
reconciliationMutex.withLock {
updateFingerprintIfNeeded(state)
val rawEvent = engine.evaluate(state)
val event = applyOverrides(rawEvent)
Timber.d("AutoTunnel reconciliation event: $event")
handleAutoTunnelEvent(event)
}
}
}
private fun updateFingerprintIfNeeded(state: AutoTunnelState) {
val currentFingerprint = state.networkFingerPrint
val fingerprint = state.networkFingerPrint
if (lastNetworkFingerprint != currentFingerprint) {
if (hasUserOverride) {
Timber.d("Network fingerprint changed, clearing user override")
}
hasUserOverride = false
lastNetworkFingerprint = currentFingerprint
if (manualOverrideState.fingerprint != fingerprint) {
Timber.d("Network changed, clearing overrides")
manualOverrideState = ManualOverrideState(fingerprint = fingerprint)
}
}
private fun applyOverrides(event: AutoTunnelEvent): AutoTunnelEvent {
return if (hasUserOverride) {
AutoTunnelEvent.DoNothing
} else {
event
if (event !is AutoTunnelEvent.Sync) {
return event
}
val filteredStart =
event.start.filterNot { it.id in manualOverrideState.stoppedTunnelIds }.toSet()
val filteredStop =
event.stop.filterNot { it in manualOverrideState.startedTunnelIds }.toSet()
if (filteredStart.isEmpty() && filteredStop.isEmpty()) {
return AutoTunnelEvent.DoNothing
}
return event.copy(start = filteredStart, stop = filteredStop)
}
private fun combineSettings():
@@ -344,7 +334,7 @@ class AutoTunnelService : LifecycleService() {
private suspend fun handleAutoTunnelEvent(event: AutoTunnelEvent) {
when (event) {
is AutoTunnelEvent.Sync -> {
cancelNoInternetStopJob()
event.stop.forEach { tunnelId ->
Timber.d("Stopping tunnel: $tunnelId")
tunnelCoordinator.stopTunnel(tunnelId, TunnelActionSource.AUTO_TUNNEL)
@@ -355,12 +345,8 @@ class AutoTunnelService : LifecycleService() {
tunnelCoordinator.startTunnel(config, TunnelActionSource.AUTO_TUNNEL)
}
}
AutoTunnelEvent.StopAllDueToNoInternet -> scheduleNoInternetStop()
AutoTunnelEvent.DoNothing -> Unit
}
}
companion object {
private const val NO_INTERNET_GRACE_PERIOD_MS = 10_000L
}
}
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.service.autotunnel
package com.zaneschepke.wireguardautotunnel.core.service.autotunnel
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.StateFlow
@@ -1,9 +1,9 @@
package com.zaneschepke.wireguardautotunnel.service.tile
package com.zaneschepke.wireguardautotunnel.core.service.tile
import android.service.quicksettings.Tile
import android.service.quicksettings.TileService
import com.zaneschepke.wireguardautotunnel.core.orchestration.AutoTunnelCoordinator
import com.zaneschepke.wireguardautotunnel.service.autotunnel.AutoTunnelStateHolder
import com.zaneschepke.wireguardautotunnel.core.service.autotunnel.AutoTunnelStateHolder
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.Job
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.service.tile
package com.zaneschepke.wireguardautotunnel.core.service.tile
import android.content.ComponentName
import android.content.Context
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.service.tile
package com.zaneschepke.wireguardautotunnel.core.service.tile
import android.content.Context
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.service.tile
package com.zaneschepke.wireguardautotunnel.core.service.tile
import android.os.Build
import android.service.quicksettings.Tile
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.service.tile
package com.zaneschepke.wireguardautotunnel.core.service.tile
import android.content.ComponentName
import android.content.Context
@@ -19,8 +19,9 @@ class ShortcutsActivity : ComponentActivity() {
super.onCreate(savedInstanceState)
finish()
applicationScope.launch { shortcutCoordinator.handle(intent) }
applicationScope.launch {
shortcutCoordinator.handle(intent)
finish()
}
}
}
@@ -1,116 +0,0 @@
package com.zaneschepke.wireguardautotunnel.core.tunnel
import android.app.Notification
import android.app.PendingIntent
import android.content.Context
import android.content.Intent
import com.zaneschepke.tunnel.ApplicationProvider
import com.zaneschepke.tunnel.model.BackendMode
import com.zaneschepke.tunnel.state.BackendStatus
import com.zaneschepke.wireguardautotunnel.MainActivity
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.notification.AndroidNotificationService
import com.zaneschepke.wireguardautotunnel.notification.NotificationService
import com.zaneschepke.wireguardautotunnel.notification.NotificationService.Companion.PROXY_GROUP_KEY
import com.zaneschepke.wireguardautotunnel.notification.NotificationService.Companion.VPN_GROUP_KEY
import com.zaneschepke.wireguardautotunnel.notification.TunnelNotificationLine
import com.zaneschepke.wireguardautotunnel.notification.TunnelNotificationService
import com.zaneschepke.wireguardautotunnel.service.tile.TunnelTileRefresher
import com.zaneschepke.wireguardautotunnel.ui.state.DisplayTunnelState
import kotlinx.coroutines.flow.first
class AndroidApplicationProvider(
private val notificationService: NotificationService,
private val tunnelNotificationService: TunnelNotificationService,
private val tunnelRepository: TunnelRepository,
) : ApplicationProvider {
private val context: Context = notificationService.context
override fun refreshTile(context: Context) {
TunnelTileRefresher.refresh(context)
}
override fun createVpnConfigurePendingIntent(context: Context): PendingIntent {
return PendingIntent.getActivity(
context,
0,
Intent(context, MainActivity::class.java).apply {
addFlags(Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TOP)
},
PendingIntent.FLAG_IMMUTABLE or PendingIntent.FLAG_UPDATE_CURRENT,
)
}
override val vpnInitNotification: Notification
get() =
notificationService.createNotification(
channel = AndroidNotificationService.NotificationChannels.Tunnel.VPN,
title = context.getString(R.string.initializing),
onGoing = true,
groupKey = VPN_GROUP_KEY,
)
override val proxyInitNotification: Notification
get() =
notificationService.createNotification(
channel = AndroidNotificationService.NotificationChannels.Tunnel.Proxy,
title = context.getString(R.string.initializing),
onGoing = true,
groupKey = PROXY_GROUP_KEY,
)
override val vpnNotificationId: Int
get() = NotificationService.VPN_NOTIFICATION_ID
override val proxyNotificationId: Int
get() = NotificationService.PROXY_NOTIFICATION_ID
override suspend fun buildVpnPersistentNotification(
currentStatus: BackendStatus
): Notification {
val lines = computeVpnNotificationLines(currentStatus)
return tunnelNotificationService.buildVpnPersistentNotification(lines)
}
override suspend fun buildProxyPersistentNotification(
currentStatus: BackendStatus
): Notification {
val lines = computeProxyNotificationLines(currentStatus)
return tunnelNotificationService.buildProxyPersistentNotification(lines)
}
private suspend fun computeVpnNotificationLines(
status: BackendStatus
): Map<Int, TunnelNotificationLine> {
val activeTunnels = status.activeTunnels
val allTunnels = tunnelRepository.userTunnelsFlow.first()
return activeTunnels
.mapNotNull { (id, activeTunnel) ->
val mode = activeTunnel.mode ?: return@mapNotNull null
if (mode !is BackendMode.Vpn && mode !is BackendMode.Proxy.KillSwitchPrimary)
return@mapNotNull null
val tunnel = allTunnels.find { it.id == id } ?: return@mapNotNull null
val displayState = DisplayTunnelState.from(activeTunnel)
TunnelNotificationLine(id, tunnel.name, displayState)
}
.associateBy { it.id }
}
private suspend fun computeProxyNotificationLines(
status: BackendStatus
): Map<Int, TunnelNotificationLine> {
val activeTunnels = status.activeTunnels
val allTunnels = tunnelRepository.userTunnelsFlow.first()
return activeTunnels
.mapNotNull { (id, activeTunnel) ->
val mode = activeTunnel.mode ?: return@mapNotNull null
if (mode !is BackendMode.Proxy.Standard) return@mapNotNull null
val tunnel = allTunnels.find { it.id == id } ?: return@mapNotNull null
val displayState = DisplayTunnelState.from(activeTunnel)
TunnelNotificationLine(id, tunnel.name, displayState)
}
.associateBy { it.id }
}
}
@@ -4,11 +4,14 @@ import com.zaneschepke.tunnel.Tunnel
import com.zaneschepke.tunnel.backend.Backend
import com.zaneschepke.tunnel.model.BackendMode
import com.zaneschepke.tunnel.state.BackendStatus
import com.zaneschepke.wireguardautotunnel.domain.events.BackendCoreException
import com.zaneschepke.wireguardautotunnel.domain.events.BackendMessage
import com.zaneschepke.wireguardautotunnel.domain.model.LockdownSettings
import kotlin.concurrent.atomics.ExperimentalAtomicApi
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.flow.MutableSharedFlow
import kotlinx.coroutines.flow.SharingStarted
import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.stateIn
@@ -50,7 +53,9 @@ class TunnelBackendProvider(
return backend.disableKillSwitch()
}
override suspend fun setSeamlessRoaming(enabled: Boolean): Result<Unit> {
return backend.setSeamlessRoaming(enabled)
}
@OptIn(ExperimentalCoroutinesApi::class)
private val localErrorEvents = MutableSharedFlow<Pair<String?, BackendCoreException>>()
@OptIn(ExperimentalCoroutinesApi::class)
private val localMessageEvents = MutableSharedFlow<Pair<String?, BackendMessage>>()
}
@@ -20,8 +20,6 @@ interface TunnelProvider {
suspend fun disableLockDown(): Result<Unit>
suspend fun setSeamlessRoaming(enabled: Boolean): Result<Unit>
val backendStatus: StateFlow<BackendStatus>
val events: Flow<TunnelEvent>
@@ -6,9 +6,9 @@ import androidx.work.ExistingPeriodicWorkPolicy
import androidx.work.PeriodicWorkRequestBuilder
import androidx.work.WorkManager
import androidx.work.WorkerParameters
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.service.autotunnel.AutoTunnelStateHolder
import com.zaneschepke.wireguardautotunnel.domain.repository.AutoTunnelSettingsRepository
import com.zaneschepke.wireguardautotunnel.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.service.autotunnel.AutoTunnelStateHolder
import java.util.concurrent.TimeUnit
import timber.log.Timber
@@ -34,7 +34,7 @@ import com.zaneschepke.wireguardautotunnel.data.entity.TunnelConfig
DnsSettings::class,
LockdownSettings::class,
],
version = 32,
version = 30,
autoMigrations =
[
AutoMigration(from = 1, to = 2),
@@ -63,8 +63,6 @@ import com.zaneschepke.wireguardautotunnel.data.entity.TunnelConfig
AutoMigration(from = 26, to = 27, spec = GlobalsMigration::class),
AutoMigration(from = 27, to = 28, spec = DonationMigration::class),
AutoMigration(from = 29, to = 30, spec = SingleConfigMigration::class),
AutoMigration(from = 30, to = 31),
AutoMigration(from = 31, to = 32),
],
exportSchema = true,
)
@@ -18,7 +18,4 @@ interface AutoTunnelSettingsDao {
@Query("UPDATE auto_tunnel_settings SET is_tunnel_enabled = :enabled")
suspend fun updateAutoTunnelEnabled(enabled: Boolean)
@Query("UPDATE auto_tunnel_settings SET disable_on_captive_portal = :enabled")
suspend fun updateDisableOnCaptivePortal(enabled: Boolean)
}
@@ -34,7 +34,4 @@ interface GeneralSettingsDao {
@Query("UPDATE general_settings SET screen_recording_security = :enabled")
suspend fun updateScreenRecordingSecurity(enabled: Boolean)
@Query("UPDATE general_settings SET seamless_roaming_enabled = :enabled")
suspend fun updateSeamlessRoaming(enabled: Boolean)
}
@@ -27,6 +27,4 @@ data class AutoTunnelSettings(
@ColumnInfo(name = "wifi_detection_method", defaultValue = "0")
val wifiDetectionMethod: WifiDetectionMethod = WifiDetectionMethod.fromValue(0),
@ColumnInfo(name = "start_on_boot", defaultValue = "0") val startOnBoot: Boolean = false,
@ColumnInfo(name = "disable_on_captive_portal", defaultValue = "1")
val disableTunnelOnCaptivePortal: Boolean = true,
)
@@ -34,6 +34,4 @@ data class GeneralSettings(
val isGlobalAmneziaEnabled: Boolean = false,
@ColumnInfo(name = "tunnel_scripting_enabled", defaultValue = "0")
val tunnelScriptingEnabled: Boolean = true,
@ColumnInfo(name = "seamless_roaming_enabled", defaultValue = "0")
val seamlessRoamingEnabled: Boolean = true,
)
@@ -16,7 +16,6 @@ fun Entity.toDomain(): Domain =
isTunnelOnUnsecureEnabled = isTunnelOnUnsecureEnabled,
wifiDetectionMethod = wifiDetectionMethod,
startOnBoot = startOnBoot,
disableTunnelOnCaptivePortal = disableTunnelOnCaptivePortal
)
fun Domain.toEntity(): Entity =
@@ -32,5 +31,4 @@ fun Domain.toEntity(): Entity =
isTunnelOnUnsecureEnabled = isTunnelOnUnsecureEnabled,
wifiDetectionMethod = wifiDetectionMethod,
startOnBoot = startOnBoot,
disableTunnelOnCaptivePortal = disableTunnelOnCaptivePortal
)
@@ -22,7 +22,6 @@ fun Entity.toDomain(): Domain =
screenRecordingSecurityEnabled = screenRecordingSecurityEnabled,
isGlobalAmneziaEnabled = isGlobalAmneziaEnabled,
tunnelScriptingEnabled = tunnelScriptingEnabled,
seamlessRoamingEnabled = seamlessRoamingEnabled,
)
fun Domain.toEntity(): Entity =
@@ -43,5 +42,4 @@ fun Domain.toEntity(): Entity =
screenRecordingSecurityEnabled = screenRecordingSecurityEnabled,
isGlobalAmneziaEnabled = isGlobalAmneziaEnabled,
tunnelScriptingEnabled = tunnelScriptingEnabled,
seamlessRoamingEnabled = seamlessRoamingEnabled,
)
@@ -1,25 +1,15 @@
package com.zaneschepke.wireguardautotunnel.data.network
import com.zaneschepke.wireguardautotunnel.BuildConfig
import io.ktor.client.HttpClient
import io.ktor.client.engine.okhttp.OkHttp
import io.ktor.client.plugins.DefaultRequest
import io.ktor.client.plugins.HttpTimeout
import io.ktor.client.plugins.contentnegotiation.ContentNegotiation
import io.ktor.http.HttpHeaders
import io.ktor.serialization.kotlinx.json.json
import kotlinx.serialization.json.Json
object KtorClient {
fun create(): HttpClient {
return HttpClient(OkHttp) {
install(DefaultRequest) {
headers {
append(HttpHeaders.UserAgent, "wgtunnel/${BuildConfig.VERSION_NAME} (Android)")
append(HttpHeaders.Accept, "*/*")
}
}
install(ContentNegotiation) {
json(
Json {
@@ -28,11 +18,10 @@ object KtorClient {
}
)
}
install(HttpTimeout) {
requestTimeoutMillis = 120_000L
connectTimeoutMillis = 30_000L
socketTimeoutMillis = 120_000L
requestTimeoutMillis = 15000
connectTimeoutMillis = 15000
socketTimeoutMillis = 15000
}
}
}
@@ -26,8 +26,4 @@ class RoomAutoTunnelSettingsRepository(private val autoTunnelSettingsDao: AutoTu
override suspend fun updateAutoTunnelEnabled(enabled: Boolean) {
autoTunnelSettingsDao.updateAutoTunnelEnabled(enabled)
}
override suspend fun updateDisableOnCaptivePortal(enabled: Boolean) {
autoTunnelSettingsDao.updateDisableOnCaptivePortal(enabled)
}
}
@@ -45,8 +45,4 @@ class RoomSettingsRepository(private val settingsDao: GeneralSettingsDao) :
override suspend fun updateScreenRecordingSecurity(enabled: Boolean) {
settingsDao.updateScreenRecordingSecurity(enabled)
}
override suspend fun updateSeamlessRoaming(enabled: Boolean) {
settingsDao.updateSeamlessRoaming(enabled)
}
}
@@ -6,19 +6,20 @@ import android.os.StrictMode
import com.zaneschepke.logcatter.LogReader
import com.zaneschepke.logcatter.LogcatReader
import com.zaneschepke.wireguardautotunnel.BuildConfig
import com.zaneschepke.wireguardautotunnel.core.notification.AndroidNotificationService
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService
import com.zaneschepke.wireguardautotunnel.core.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.core.service.autotunnel.AutoTunnelStateHolder
import com.zaneschepke.wireguardautotunnel.core.shortcut.DynamicShortcutManager
import com.zaneschepke.wireguardautotunnel.core.shortcut.ShortcutManager
import com.zaneschepke.wireguardautotunnel.domain.repository.GlobalEffectRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.SelectedTunnelsRepository
import com.zaneschepke.wireguardautotunnel.notification.AndroidNotificationService
import com.zaneschepke.wireguardautotunnel.notification.NotificationService
import com.zaneschepke.wireguardautotunnel.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.service.autotunnel.AutoTunnelStateHolder
import com.zaneschepke.wireguardautotunnel.util.FileUtils
import com.zaneschepke.wireguardautotunnel.util.network.NetworkUtils
import com.zaneschepke.wireguardautotunnel.viewmodel.AutoTunnelViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.ConfigEditViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.DnsViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.LicenseViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.LockdownViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.LoggerViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.MonitoringViewModel
@@ -79,6 +80,7 @@ val appModule = module {
viewModelOf(::AutoTunnelViewModel)
viewModel { (id: Int?) -> ConfigEditViewModel(get(), get(), get(), get(), get(), id) }
viewModelOf(::DnsViewModel)
viewModelOf(::LicenseViewModel)
viewModelOf(::LockdownViewModel)
viewModelOf(::LoggerViewModel)
viewModelOf(::MonitoringViewModel)
@@ -5,15 +5,15 @@ import com.zaneschepke.wireguardautotunnel.core.orchestration.AutoTunnelCoordina
import com.zaneschepke.wireguardautotunnel.core.orchestration.DnsSettingsCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.ShortcutCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.StartupCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.TunnelBackendCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.TunnelCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.TunnelModeCoordinator
import org.koin.core.module.dsl.singleOf
import org.koin.core.qualifier.named
import org.koin.dsl.module
val coordinatorModule = module {
singleOf(::ShortcutCoordinator)
singleOf(::TunnelBackendCoordinator)
singleOf(::TunnelModeCoordinator)
singleOf(::StartupCoordinator)
singleOf(::AutoTunnelCoordinator)
singleOf(::DnsSettingsCoordinator)
@@ -27,7 +27,6 @@ val coordinatorModule = module {
get(),
get(),
get(),
get(),
get(named(Scope.APPLICATION)),
)
}
@@ -1,19 +1,25 @@
package com.zaneschepke.wireguardautotunnel.di
import android.app.Notification
import android.content.Context
import com.zaneschepke.networkmonitor.AndroidNetworkMonitor
import com.zaneschepke.networkmonitor.NetworkMonitor
import com.zaneschepke.networkmonitor.StableNetworkEngine
import com.zaneschepke.tunnel.ApplicationProvider
import com.zaneschepke.tunnel.util.RootShell
import com.zaneschepke.tunnel.NotificationProvider
import com.zaneschepke.tunnel.backend.RootShell
import com.zaneschepke.tunnel.util.RootShellException
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.core.event.TunnelEventDispatcher
import com.zaneschepke.wireguardautotunnel.core.tunnel.AndroidApplicationProvider
import com.zaneschepke.wireguardautotunnel.core.notification.AndroidNotificationService.NotificationChannels
import com.zaneschepke.wireguardautotunnel.core.notification.AndroidTunnelNotificationService
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService.Companion.PROXY_GROUP_KEY
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService.Companion.VPN_GROUP_KEY
import com.zaneschepke.wireguardautotunnel.core.notification.TunnelNotificationService
import com.zaneschepke.wireguardautotunnel.core.service.tile.TunnelTileRefresher
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelBackendProvider
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelProvider
import com.zaneschepke.wireguardautotunnel.domain.repository.AutoTunnelSettingsRepository
import com.zaneschepke.wireguardautotunnel.lifecyle.AppVisibilityObserver
import com.zaneschepke.wireguardautotunnel.notification.AndroidTunnelNotificationService
import com.zaneschepke.wireguardautotunnel.notification.TunnelNotificationService
import com.zaneschepke.wireguardautotunnel.util.extensions.to
import kotlin.time.Duration.Companion.milliseconds
import kotlinx.coroutines.CoroutineScope
@@ -30,15 +36,40 @@ import timber.log.Timber
val tunnelBackendProviderModule = module {
single<TunnelNotificationService> { AndroidTunnelNotificationService(get()) }
single { AppVisibilityObserver() }
singleOf(::TunnelEventDispatcher)
single<ApplicationProvider> {
AndroidApplicationProvider(
notificationService = get(),
tunnelNotificationService = get(),
tunnelRepository = get(),
)
single<NotificationProvider> {
val notificationService = get<NotificationService>()
val context = androidContext()
object : NotificationProvider {
override val vpnInitNotification: Notification
get() =
notificationService.createNotification(
channel = NotificationChannels.Tunnel.VPN,
title = context.getString(R.string.initializing),
onGoing = true,
groupKey = VPN_GROUP_KEY,
)
override val proxyInitNotification: Notification
get() =
notificationService.createNotification(
channel = NotificationChannels.Tunnel.Proxy,
title = context.getString(R.string.initializing),
onGoing = true,
groupKey = PROXY_GROUP_KEY,
)
override val vpnNotificationId: Int
get() = NotificationService.VPN_NOTIFICATION_ID
override val proxyNotificationId: Int
get() = NotificationService.PROXY_NOTIFICATION_ID
override fun refreshTile(context: Context) {
TunnelTileRefresher.refresh(context)
}
}
}
single {
@@ -7,6 +7,4 @@ sealed interface AutoTunnelEvent {
data class Sync(val start: Set<TunnelConfig>, val stop: Set<Int>) : AutoTunnelEvent
data object DoNothing : AutoTunnelEvent
data object StopAllDueToNoInternet : AutoTunnelEvent
}
@@ -4,12 +4,7 @@ import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelActionSource
sealed interface TunnelActionEvent {
val source: TunnelActionSource
val tunnelId: Int
data class Started(val tunnelId: Int, val source: TunnelActionSource) : TunnelActionEvent
data class Started(override val tunnelId: Int, override val source: TunnelActionSource) :
TunnelActionEvent
data class Stopped(override val tunnelId: Int, override val source: TunnelActionSource) :
TunnelActionEvent
data class Stopped(val tunnelId: Int, val source: TunnelActionSource) : TunnelActionEvent
}
@@ -14,5 +14,4 @@ data class AutoTunnelSettings(
val isTunnelOnUnsecureEnabled: Boolean = false,
val wifiDetectionMethod: WifiDetectionMethod = WifiDetectionMethod.fromValue(0),
val startOnBoot: Boolean = false,
val disableTunnelOnCaptivePortal: Boolean = true,
)
@@ -20,6 +20,5 @@ data class GeneralSettings(
val alreadyDonated: Boolean = false,
val screenRecordingSecurityEnabled: Boolean = true,
val isGlobalAmneziaEnabled: Boolean = false,
val tunnelScriptingEnabled: Boolean = false,
val seamlessRoamingEnabled: Boolean = false,
val tunnelScriptingEnabled: Boolean = true,
)
@@ -11,6 +11,4 @@ interface AutoTunnelSettingsRepository {
suspend fun getAutoTunnelSettings(): AutoTunnelSettings
suspend fun updateAutoTunnelEnabled(enabled: Boolean)
suspend fun updateDisableOnCaptivePortal(enabled: Boolean)
}
@@ -23,6 +23,4 @@ interface GeneralSettingRepository {
suspend fun updateGlobalAmneziaEnabled(enabled: Boolean)
suspend fun updateScreenRecordingSecurity(enabled: Boolean)
suspend fun updateSeamlessRoaming(enabled: Boolean)
}
@@ -7,7 +7,7 @@ import kotlinx.coroutines.flow.asSharedFlow
class GlobalEffectRepository {
private val _globalEffectFlow =
MutableSharedFlow<GlobalSideEffect>(replay = 0, extraBufferCapacity = 0)
MutableSharedFlow<GlobalSideEffect>(replay = 0, extraBufferCapacity = 1)
val flow = _globalEffectFlow.asSharedFlow()
suspend fun post(effect: GlobalSideEffect) {
@@ -1,8 +1,8 @@
package com.zaneschepke.wireguardautotunnel.domain.sideeffect
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
import com.zaneschepke.wireguardautotunnel.domain.model.TunnelConfig
import com.zaneschepke.wireguardautotunnel.ui.common.snackbar.SnackbarType
import com.zaneschepke.wireguardautotunnel.util.StringValue
import java.io.File
@@ -10,12 +10,14 @@ sealed class GlobalSideEffect {
data class Snackbar(
val message: StringValue,
val type: ToastType,
val type: SnackbarType? = null,
val actionLabel: String? = null,
val onAction: (() -> Unit)? = null,
val durationMs: Long? = null,
) : GlobalSideEffect()
data class Toast(val message: StringValue) : GlobalSideEffect()
data object PopBackStack : GlobalSideEffect()
data class LaunchUrl(val url: String) : GlobalSideEffect()
@@ -11,20 +11,16 @@ sealed class ActiveNetwork {
data object Cellular : ActiveNetwork()
data class Wifi(
val ssid: String,
val isSecure: Boolean?,
val requiresCaptivePortalLogin: Boolean,
) : ActiveNetwork()
data class Wifi(val ssid: String, val isSecure: Boolean?) : ActiveNetwork()
}
data class NetworkState(
val activeNetwork: ActiveNetwork = ActiveNetwork.Disconnected,
val locationServicesEnabled: Boolean = false,
val locationPermissionGranted: Boolean = false,
// Has a network that can actually transfer data (not suspended)
val hasUsableNetwork: Boolean = false,
)
) {
fun hasInternet(): Boolean = activeNetwork !is ActiveNetwork.Disconnected
}
fun ConnectivityState.toDomain(): NetworkState {
val domainNetwork: ActiveNetwork =
@@ -37,11 +33,7 @@ fun ConnectivityState.toDomain(): NetworkState {
null -> null
else -> true
}
ActiveNetwork.Wifi(
ssid = network.ssid,
isSecure = isSecure,
requiresCaptivePortalLogin(),
)
ActiveNetwork.Wifi(ssid = network.ssid, isSecure = isSecure)
}
is MonitorActiveNetwork.Cellular -> ActiveNetwork.Cellular
is MonitorActiveNetwork.Ethernet -> ActiveNetwork.Ethernet
@@ -52,6 +44,5 @@ fun ConnectivityState.toDomain(): NetworkState {
activeNetwork = domainNetwork,
locationPermissionGranted = this.locationPermissionsGranted,
locationServicesEnabled = this.locationServicesEnabled,
hasUsableNetwork = hasUsableNetwork(),
)
}
@@ -1,26 +0,0 @@
package com.zaneschepke.wireguardautotunnel.lifecyle
import androidx.lifecycle.DefaultLifecycleObserver
import androidx.lifecycle.LifecycleOwner
import androidx.lifecycle.ProcessLifecycleOwner
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.asStateFlow
class AppVisibilityObserver : DefaultLifecycleObserver {
private val _isForeground = MutableStateFlow(false)
val isForeground: StateFlow<Boolean> = _isForeground.asStateFlow()
init {
ProcessLifecycleOwner.get().lifecycle.addObserver(this)
}
override fun onStart(owner: LifecycleOwner) {
_isForeground.value = true
}
override fun onStop(owner: LifecycleOwner) {
_isForeground.value = false
}
}
@@ -1,73 +0,0 @@
package com.zaneschepke.wireguardautotunnel.ui.common.dialog
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.Spacer
import androidx.compose.foundation.layout.height
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.ui.Modifier
import androidx.compose.ui.res.stringResource
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.unit.dp
import com.zaneschepke.wireguardautotunnel.R
@Composable
fun LocalNetworkPermissionDialog(onDismiss: () -> Unit, onAttest: () -> Unit) {
InfoDialog(
onAttest = onAttest,
onDismiss = onDismiss,
title = stringResource(R.string.local_network_permission_title),
body = {
Column {
Text(
text = stringResource(R.string.local_network_permission_intro),
style = MaterialTheme.typography.bodyMedium,
)
Spacer(modifier = Modifier.height(16.dp))
Text(
text = stringResource(R.string.local_network_permission_issues_intro),
style = MaterialTheme.typography.bodyMedium,
fontWeight = FontWeight.Bold,
)
Spacer(modifier = Modifier.height(8.dp))
Column(verticalArrangement = Arrangement.spacedBy(4.dp)) {
Text(
text = stringResource(R.string.local_network_permission_feature_tunnels),
style = MaterialTheme.typography.bodyMedium,
)
Text(
text = stringResource(R.string.local_network_permission_feature_autotunnel),
style = MaterialTheme.typography.bodyMedium,
)
Text(
text = stringResource(R.string.local_network_permission_feature_proxy),
style = MaterialTheme.typography.bodyMedium,
)
}
Spacer(modifier = Modifier.height(16.dp))
Text(
text = stringResource(R.string.local_network_permission_recommendation),
style = MaterialTheme.typography.bodyMedium,
fontWeight = FontWeight.Bold,
)
Spacer(modifier = Modifier.height(16.dp))
Text(
text = stringResource(R.string.local_network_permission_nearby_devices),
style = MaterialTheme.typography.bodySmall,
color = MaterialTheme.colorScheme.onSurfaceVariant,
)
}
},
confirmText = stringResource(R.string._continue),
)
}
@@ -0,0 +1,104 @@
package com.zaneschepke.wireguardautotunnel.ui.common.snackbar
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.IntrinsicSize
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.Spacer
import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.foundation.layout.height
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.layout.size
import androidx.compose.foundation.layout.width
import androidx.compose.foundation.layout.wrapContentHeight
import androidx.compose.foundation.shape.RoundedCornerShape
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.outlined.Favorite
import androidx.compose.material.icons.rounded.Close
import androidx.compose.material.icons.rounded.Info
import androidx.compose.material.icons.rounded.Warning
import androidx.compose.material3.Icon
import androidx.compose.material3.IconButton
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Snackbar
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.res.stringResource
import androidx.compose.ui.text.AnnotatedString
import androidx.compose.ui.text.style.TextOverflow
import androidx.compose.ui.unit.dp
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.ui.LocalIsAndroidTV
@Composable
fun CustomSnackBar(
message: AnnotatedString,
onDismiss: () -> Unit,
modifier: Modifier = Modifier,
type: SnackbarType = SnackbarType.INFO,
containerColor: Color = MaterialTheme.colorScheme.surface,
) {
val isTv = LocalIsAndroidTV.current
val icon =
when (type) {
SnackbarType.INFO -> Icons.Rounded.Info
SnackbarType.WARNING -> Icons.Rounded.Warning
SnackbarType.THANK_YOU -> Icons.Outlined.Favorite
}
val iconDescription =
when (type) {
SnackbarType.INFO -> stringResource(R.string.info)
SnackbarType.WARNING -> stringResource(R.string.warning)
SnackbarType.THANK_YOU -> stringResource(R.string.thank_you)
}
Snackbar(
containerColor = containerColor,
modifier =
modifier
.wrapContentHeight(align = Alignment.Top)
.padding(horizontal = if (isTv) 48.dp else 16.dp),
shape = RoundedCornerShape(16.dp),
) {
Row(
modifier =
Modifier.fillMaxWidth()
.height(IntrinsicSize.Min)
.width(IntrinsicSize.Min)
.padding(vertical = 16.dp),
verticalAlignment = Alignment.CenterVertically,
horizontalArrangement = Arrangement.SpaceBetween,
) {
Row(
modifier = Modifier.fillMaxWidth().weight(1f),
verticalAlignment = Alignment.CenterVertically,
horizontalArrangement = Arrangement.Start,
) {
Icon(
icon,
contentDescription = iconDescription,
tint = MaterialTheme.colorScheme.onSurface,
)
Spacer(modifier = Modifier.width(16.dp))
Text(
text = message,
color = MaterialTheme.colorScheme.onSurface,
maxLines = 8,
overflow = TextOverflow.Ellipsis,
)
}
Row {
IconButton(onClick = onDismiss, modifier = Modifier.size(24.dp)) {
Icon(
Icons.Rounded.Close,
contentDescription = stringResource(R.string.stop),
tint = MaterialTheme.colorScheme.onSurfaceVariant,
)
}
}
}
}
}
@@ -0,0 +1,76 @@
package com.zaneschepke.wireguardautotunnel.ui.common.snackbar
import androidx.compose.foundation.layout.Box
import androidx.compose.material3.MaterialTheme
import androidx.compose.runtime.Composable
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberCoroutineScope
import androidx.compose.runtime.setValue
import androidx.compose.ui.Modifier
import kotlinx.coroutines.channels.Channel
import kotlinx.coroutines.delay
import kotlinx.coroutines.launch
@Composable
fun rememberCustomSnackbarState(): CustomSnackbarState {
return remember { CustomSnackbarState() }
}
class CustomSnackbarState {
private val _snackbars = Channel<SnackbarInfo>(Channel.BUFFERED)
val snackbars: Channel<SnackbarInfo> = _snackbars
private var currentSnackbar by mutableStateOf<SnackbarInfo?>(null)
private var isShowing by mutableStateOf(false)
fun showSnackbar(info: SnackbarInfo) {
_snackbars.trySend(info)
}
fun dismissCurrent() {
currentSnackbar = null
isShowing = false
}
@Composable
fun SnackbarHost(
modifier: Modifier = Modifier,
snackbar: @Composable (SnackbarInfo) -> Unit = { info ->
CustomSnackBar(
message = info.message,
type = info.type,
onDismiss = { dismissCurrent() },
modifier = Modifier,
containerColor = MaterialTheme.colorScheme.surface.copy(.1f),
)
},
) {
val scope = rememberCoroutineScope()
LaunchedEffect(Unit) {
for (info in snackbars) {
currentSnackbar = info
isShowing = true
scope.launch {
delay(info.durationMs)
if (currentSnackbar?.id == info.id) {
dismissCurrent()
}
}
while (isShowing && currentSnackbar?.id == info.id) {
delay(100)
}
}
}
currentSnackbar?.let { info ->
if (isShowing) {
Box(modifier = modifier) { snackbar(info) }
}
}
}
}
@@ -0,0 +1,16 @@
package com.zaneschepke.wireguardautotunnel.ui.common.snackbar
import androidx.compose.ui.text.AnnotatedString
enum class SnackbarType {
INFO,
WARNING,
THANK_YOU,
}
data class SnackbarInfo(
val message: AnnotatedString,
val type: SnackbarType = SnackbarType.INFO,
val durationMs: Long = 4000L,
val id: String = System.currentTimeMillis().toString(),
)
@@ -1,18 +1,13 @@
package com.zaneschepke.wireguardautotunnel.ui.common.textbox
import androidx.compose.foundation.background
import androidx.compose.foundation.interaction.MutableInteractionSource
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.defaultMinSize
import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.foundation.layout.offset
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.text.KeyboardActions
import androidx.compose.foundation.text.KeyboardOptions
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.runtime.remember
import androidx.compose.ui.Modifier
import androidx.compose.ui.text.input.ImeAction
import androidx.compose.ui.text.input.KeyboardCapitalization
@@ -34,59 +29,48 @@ fun ConfigurationTextBox(
leading: (@Composable () -> Unit)? = null,
trailing: (@Composable (Modifier) -> Unit)? = null,
supportingText: (@Composable () -> Unit)? = null,
interactionSource: MutableInteractionSource = remember { MutableInteractionSource() },
interactionSource: MutableInteractionSource = MutableInteractionSource(),
visualTransformation: VisualTransformation = VisualTransformation.None,
enabled: Boolean = true,
readOnly: Boolean = false,
singleLine: Boolean = true,
) {
Box(modifier = modifier.padding(top = 6.dp)) {
CustomTextField(
isError = isError,
textStyle =
MaterialTheme.typography.bodySmall.copy(
color = MaterialTheme.colorScheme.onSurface
),
modifier = Modifier.fillMaxWidth().defaultMinSize(minHeight = 48.dp),
value = value,
visualTransformation = visualTransformation,
singleLine = singleLine,
interactionSource = interactionSource,
onValueChange = onValueChange,
label = null, // Disable built in label
containerColor = MaterialTheme.colorScheme.surface,
placeholder = {
Text(
text = hint,
style = MaterialTheme.typography.bodySmall,
color = MaterialTheme.colorScheme.outline,
overflow = TextOverflow.Ellipsis,
maxLines = 1,
)
},
keyboardOptions = keyboardOptions,
keyboardActions = keyboardActions,
trailing = trailing,
supportingText = supportingText,
leading = leading,
readOnly = readOnly,
enabled = enabled,
)
// custom static label notch
if (label.isNotEmpty()) {
CustomTextField(
isError = isError,
textStyle =
MaterialTheme.typography.bodySmall.copy(color = MaterialTheme.colorScheme.onSurface),
modifier = modifier.fillMaxWidth().defaultMinSize(minHeight = 48.dp),
value = value,
visualTransformation = visualTransformation,
singleLine = singleLine,
interactionSource = interactionSource,
onValueChange = { onValueChange(it) },
label = {
Text(
text = label,
label,
color = MaterialTheme.colorScheme.onSurface,
style = MaterialTheme.typography.labelSmall,
style = MaterialTheme.typography.labelMedium,
overflow = TextOverflow.Ellipsis,
maxLines = 1,
modifier =
Modifier.padding(start = 12.dp)
.offset(y = (-8).dp)
.background(MaterialTheme.colorScheme.surface)
.padding(horizontal = 4.dp),
)
}
}
},
containerColor = MaterialTheme.colorScheme.surface,
placeholder = {
Text(
hint,
style = MaterialTheme.typography.bodySmall,
color = MaterialTheme.colorScheme.outline,
overflow = TextOverflow.Ellipsis,
maxLines = 1,
)
},
keyboardOptions = keyboardOptions,
keyboardActions = keyboardActions,
trailing = trailing,
supportingText = supportingText,
leading = leading,
readOnly = readOnly,
enabled = enabled,
)
}
@@ -22,9 +22,7 @@ import androidx.compose.ui.focus.focusRequester
import androidx.compose.ui.focus.onFocusChanged
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.graphics.SolidColor
import androidx.compose.ui.text.TextRange
import androidx.compose.ui.text.TextStyle
import androidx.compose.ui.text.input.TextFieldValue
import androidx.compose.ui.text.input.VisualTransformation
import androidx.compose.ui.unit.dp
@@ -35,7 +33,7 @@ fun CustomTextField(
modifier: Modifier = Modifier,
textStyle: TextStyle =
MaterialTheme.typography.bodyLarge.copy(color = MaterialTheme.colorScheme.onSurface),
label: @Composable (() -> Unit)? = null,
label: @Composable () -> Unit,
containerColor: Color,
onValueChange: (value: String) -> Unit = {},
singleLine: Boolean = true,
@@ -49,19 +47,10 @@ fun CustomTextField(
readOnly: Boolean = false,
enabled: Boolean = true,
visualTransformation: VisualTransformation = VisualTransformation.None,
interactionSource: MutableInteractionSource = remember { MutableInteractionSource() },
interactionSource: MutableInteractionSource = MutableInteractionSource(),
) {
val space = " "
var isFocused by remember { mutableStateOf(false) }
var textFieldValueState by remember { mutableStateOf(TextFieldValue(text = value)) }
val textFieldValue =
remember(value, textFieldValueState) {
if (textFieldValueState.text == value) {
textFieldValueState
} else {
textFieldValueState.copy(text = value, selection = TextRange(value.length))
}
}
val cursorBrush =
if (isFocused) SolidColor(MaterialTheme.colorScheme.primary)
else SolidColor(Color.Transparent)
@@ -78,14 +67,9 @@ fun CustomTextField(
}
BasicTextField(
value = textFieldValue,
value = value,
textStyle = effectiveTextStyle,
onValueChange = { newTextFieldValue ->
textFieldValueState = newTextFieldValue
if (value != newTextFieldValue.text) {
onValueChange(newTextFieldValue.text)
}
},
onValueChange = { onValueChange(it) },
keyboardActions = keyboardActions,
keyboardOptions = keyboardOptions,
readOnly = readOnly,
@@ -106,9 +90,15 @@ fun CustomTextField(
visualTransformation = visualTransformation,
) {
OutlinedTextFieldDefaults.DecorationBox(
value = value,
innerTextField = it,
placeholder = placeholder,
value = space + value,
innerTextField = {
if (value.isEmpty()) {
if (placeholder != null) {
placeholder()
}
}
it.invoke()
},
contentPadding = OutlinedTextFieldDefaults.contentPadding(top = 14.dp, bottom = 14.dp),
leadingIcon = leading,
trailingIcon =
@@ -151,6 +141,7 @@ fun CustomTextField(
label = label,
visualTransformation = visualTransformation,
interactionSource = interactionSource,
placeholder = placeholder,
container = {
OutlinedTextFieldDefaults.Container(
enabled = enabled,
@@ -42,12 +42,7 @@ sealed class Route : NavKey {
@Keep @Serializable data object Display : Route()
@Keep
@Serializable
data object Tunnels : Route(), SecureRoute {
override val requiresProtection: Boolean
get() = true
}
@Keep @Serializable data object Tunnels : Route()
@Keep @Serializable data class TunnelSettings(val id: Int) : Route()
@@ -250,7 +250,7 @@ fun currentRouteAsNavbarState(
val global = route !is ConfigEdit
val tunnelName =
if (!global) globalState.tunnelNames[route.id]
else context.getString(R.string.tunnel_configuration)
else context.getString(R.string.configuration_globals)
NavbarState(
topLeading = { TvBackButton { navController.pop() } },
showBottomItems = true,
@@ -297,7 +297,7 @@ fun currentRouteAsNavbarState(
is SplitTunnelGlobal -> {
val tunnelName =
if (route is SplitTunnel) globalState.tunnelNames[route.id]
else context.getString(R.string.splt_tunneling)
else context.getString(R.string.global_split_tunneling)
NavbarState(
topLeading = { TvBackButton { navController.pop() } },
topTitle = tunnelName ?: "",
@@ -302,9 +302,7 @@ fun AutoTunnelScreen(
SurfaceRow(
leading = { Icon(Icons.Outlined.PublicOff, contentDescription = null) },
title = stringResource(R.string.stop_on_no_internet),
description = {
DescriptionText(stringResource(R.string.stop_on_no_internet_desc))
},
description = { DescriptionText(stringResource(R.string.stop_on_internet_loss)) },
trailing = {
ThemedSwitch(
checked = uiState.autoTunnelSettings.isStopOnNoInternetEnabled,
@@ -320,7 +318,7 @@ fun AutoTunnelScreen(
}
Column {
GroupLabel(
stringResource(R.string.automation),
stringResource(R.string.other),
modifier = Modifier.padding(horizontal = 16.dp),
)
SurfaceRow(
@@ -332,7 +330,6 @@ fun AutoTunnelScreen(
onClick = { viewModel.setStartAtBoot(it) },
)
},
description = { DescriptionText(stringResource(R.string.start_on_boot_desc)) },
onClick = { viewModel.setStartAtBoot(!uiState.autoTunnelSettings.startOnBoot) },
)
}
@@ -10,7 +10,6 @@ import androidx.compose.foundation.verticalScroll
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.outlined.Filter1
import androidx.compose.material.icons.outlined.Map
import androidx.compose.material.icons.outlined.PublicOff
import androidx.compose.material.icons.outlined.WifiFind
import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme
@@ -192,21 +191,6 @@ fun WifiSettingsScreen(viewModel: AutoTunnelViewModel = koinViewModel()) {
)
},
)
SurfaceRow(
leading = { Icon(Icons.Outlined.PublicOff, contentDescription = null) },
title = stringResource(R.string.stop_while_captive_portal),
onClick = {
viewModel.setDisabledOnCaptivePortal(
!uiState.autoTunnelSettings.disableTunnelOnCaptivePortal
)
},
trailing = {
ThemedSwitch(
checked = uiState.autoTunnelSettings.disableTunnelOnCaptivePortal,
onClick = { viewModel.setDisabledOnCaptivePortal(it) },
)
},
)
}
Column {
GroupLabel(stringResource(R.string.tunnels), Modifier.padding(horizontal = 16.dp))
@@ -9,7 +9,6 @@ import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.saveable.rememberSaveable
import androidx.compose.runtime.setValue
import androidx.compose.ui.res.stringResource
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.ui.LocalNavController
import com.zaneschepke.wireguardautotunnel.ui.navigation.Route
@@ -46,17 +45,11 @@ fun PinLockScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
textColor = MaterialTheme.colorScheme.onSurface,
onPinCorrect = { onPinCorrect() },
onPinIncorrect = {
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.incorrect_pin),
ToastType.Warning,
)
sharedViewModel.showToast(StringValue.StringResource(R.string.incorrect_pin))
},
onPinCreated = {
pinCreated = true
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.pin_created),
ToastType.Success,
)
sharedViewModel.showToast(StringValue.StringResource(R.string.pin_created))
sharedViewModel.setPinLockEnabled(true)
onPinCorrect()
},
@@ -9,7 +9,6 @@ import androidx.compose.foundation.verticalScroll
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.automirrored.outlined.ViewQuilt
import androidx.compose.material.icons.outlined.Android
import androidx.compose.material.icons.outlined.CellWifi
import androidx.compose.material.icons.outlined.Dns
import androidx.compose.material.icons.outlined.ExpandMore
import androidx.compose.material.icons.outlined.MonitorHeart
@@ -39,7 +38,6 @@ import androidx.compose.ui.res.stringResource
import androidx.compose.ui.res.vectorResource
import androidx.compose.ui.text.intl.Locale
import androidx.compose.ui.unit.dp
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.MainActivity
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
@@ -53,12 +51,12 @@ import com.zaneschepke.wireguardautotunnel.ui.common.label.GroupLabel
import com.zaneschepke.wireguardautotunnel.ui.common.text.DescriptionText
import com.zaneschepke.wireguardautotunnel.ui.navigation.Route
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.components.BackupBottomSheet
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.components.BackupEncryptionDialog
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.proxy.compoents.AppModeBottomSheet
import com.zaneschepke.wireguardautotunnel.util.StringValue
import com.zaneschepke.wireguardautotunnel.util.extensions.asString
import com.zaneschepke.wireguardautotunnel.util.extensions.asTitleString
import com.zaneschepke.wireguardautotunnel.util.extensions.capitalize
import com.zaneschepke.wireguardautotunnel.util.extensions.showToast
import com.zaneschepke.wireguardautotunnel.viewmodel.SettingsViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel
import org.koin.androidx.compose.koinViewModel
@@ -90,9 +88,6 @@ fun SettingsScreen(
}
var showBackupSheet by rememberSaveable { mutableStateOf(false) }
var showEncryptionDialog by rememberSaveable { mutableStateOf(false) }
var isRestoreAction by remember { mutableStateOf(false) }
var showAppModeSheet by rememberSaveable { mutableStateOf(false) }
val appMode = uiState.settings.tunnelMode
@@ -104,53 +99,19 @@ fun SettingsScreen(
}
fun performBackupRestore(action: () -> Unit) {
if (uiState.tunnelActive || globalUiState.isAutoTunnelActive)
return context.showToast(R.string.all_services_disabled)
showBackupSheet = false
if (uiState.tunnelActive || globalUiState.isAutoTunnelActive) {
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.all_services_disabled),
ToastType.Warning,
)
return
}
action()
}
if (showBackupSheet) {
if (showBackupSheet)
BackupBottomSheet(
onBackup = {
showBackupSheet = false
isRestoreAction = false
showEncryptionDialog = true
},
onRestore = {
showBackupSheet = false
isRestoreAction = true
showEncryptionDialog = true
},
onDismiss = { showBackupSheet = false },
)
}
if (showEncryptionDialog) {
BackupEncryptionDialog(
isRestore = isRestoreAction,
onConfirm = { encrypt, password ->
showEncryptionDialog = false
if (isRestoreAction) {
performBackupRestore {
(context as? MainActivity)?.performRestore(encrypt, password)
}
} else {
performBackupRestore {
(context as? MainActivity)?.performBackup(encrypt, password)
}
}
},
onDismiss = { showEncryptionDialog = false },
)
}
{ performBackupRestore { (context as? MainActivity)?.performBackup() } },
{ performBackupRestore { (context as? MainActivity)?.performRestore() } },
) {
showBackupSheet = false
}
if (showAppModeSheet)
AppModeBottomSheet(sharedViewModel::setAppMode, uiState.settings.tunnelMode) {
showAppModeSheet = false
@@ -207,8 +168,7 @@ fun SettingsScreen(
StringValue.StringResource(
R.string.mode_disabled_template,
appMode.asString(context),
),
ToastType.Info,
)
)
},
)
@@ -216,7 +176,6 @@ fun SettingsScreen(
leading = { Icon(Icons.Outlined.Public, contentDescription = null) },
title = stringResource(R.string.tunnel_globals),
onClick = { navController.push(Route.TunnelGlobals) },
description = { DescriptionText(stringResource(R.string.tunnel_globals_desc)) },
)
SurfaceRow(
leading = { Icon(Icons.Outlined.Terminal, contentDescription = null) },
@@ -235,23 +194,6 @@ fun SettingsScreen(
viewModel.setTunnelScriptedEnabled(!uiState.settings.tunnelScriptingEnabled)
},
)
SurfaceRow(
leading = { Icon(Icons.Outlined.CellWifi, contentDescription = null) },
title = stringResource(R.string.seamless_roaming),
trailing = { modifier ->
ThemedSwitch(
checked = uiState.settings.seamlessRoamingEnabled,
onClick = { viewModel.setSeamlessNetworkRoaming(enabled = it) },
modifier = modifier,
)
},
description = {
DescriptionText(stringResource(R.string.seamless_roaming_description))
},
onClick = {
viewModel.setSeamlessNetworkRoaming(!uiState.settings.seamlessRoamingEnabled)
},
)
SurfaceRow(
leading = { Icon(Icons.Outlined.MonitorHeart, null) },
title = stringResource(R.string.tunnel_monitoring),
@@ -290,7 +232,6 @@ fun SettingsScreen(
modifier = modifier,
)
},
description = { DescriptionText(stringResource(R.string.local_logging_desc)) },
onClick = { navController.push(Route.Logs) },
)
SurfaceRow(
@@ -1,146 +0,0 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.settings.components
import androidx.compose.foundation.layout.*
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.outlined.Visibility
import androidx.compose.material.icons.outlined.VisibilityOff
import androidx.compose.material3.Icon
import androidx.compose.material3.IconButton
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Text
import androidx.compose.runtime.*
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.res.stringResource
import androidx.compose.ui.text.input.PasswordVisualTransformation
import androidx.compose.ui.text.input.VisualTransformation
import androidx.compose.ui.unit.dp
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.ui.common.button.ThemedSwitch
import com.zaneschepke.wireguardautotunnel.ui.common.dialog.InfoDialog
import com.zaneschepke.wireguardautotunnel.ui.common.textbox.CustomTextField
@Composable
fun BackupEncryptionDialog(
isRestore: Boolean,
onConfirm: (encrypt: Boolean, password: String?) -> Unit,
onDismiss: () -> Unit,
) {
var encrypt by remember { mutableStateOf(false) }
var password by remember { mutableStateOf("") }
var confirmPassword by remember { mutableStateOf("") }
var showPasswordError by remember { mutableStateOf(false) }
var passwordVisible by remember { mutableStateOf(false) }
var confirmPasswordVisible by remember { mutableStateOf(false) }
InfoDialog(
title =
if (isRestore) {
stringResource(R.string.restore)
} else {
stringResource(R.string.backup)
},
confirmText =
if (isRestore) {
stringResource(R.string.restore)
} else {
stringResource(R.string.backup)
},
onAttest = {
if (!isRestore && encrypt && password != confirmPassword) {
showPasswordError = true
return@InfoDialog
}
if (encrypt && password.isBlank()) {
return@InfoDialog
}
val finalPassword = if (encrypt) password else null
onConfirm(encrypt, finalPassword)
},
onDismiss = onDismiss,
body = {
Column(
verticalArrangement = Arrangement.spacedBy(16.dp),
modifier = Modifier.fillMaxWidth(),
) {
Row(
modifier = Modifier.fillMaxWidth(),
horizontalArrangement = Arrangement.SpaceBetween,
verticalAlignment = Alignment.CenterVertically,
) {
Text(stringResource(R.string.encrypted))
ThemedSwitch(checked = encrypt, onClick = { encrypt = it })
}
if (encrypt) {
CustomTextField(
value = password,
onValueChange = {
password = it
showPasswordError = false
},
containerColor = MaterialTheme.colorScheme.surface,
label = { Text(stringResource(R.string.password)) },
visualTransformation =
if (passwordVisible) VisualTransformation.None
else PasswordVisualTransformation(),
trailing = {
IconButton(onClick = { passwordVisible = !passwordVisible }) {
Icon(
imageVector =
if (passwordVisible) Icons.Outlined.VisibilityOff
else Icons.Outlined.Visibility,
contentDescription =
if (passwordVisible) stringResource(R.string.hide_password)
else stringResource(R.string.show_password),
)
}
},
modifier = Modifier.fillMaxWidth(),
)
if (!isRestore) {
CustomTextField(
value = confirmPassword,
onValueChange = {
confirmPassword = it
showPasswordError = false
},
label = { Text(stringResource(R.string.confirm_password)) },
visualTransformation =
if (confirmPasswordVisible) VisualTransformation.None
else PasswordVisualTransformation(),
trailing = {
IconButton(
onClick = { confirmPasswordVisible = !confirmPasswordVisible }
) {
Icon(
imageVector =
if (confirmPasswordVisible) Icons.Outlined.VisibilityOff
else Icons.Outlined.Visibility,
contentDescription =
if (confirmPasswordVisible)
stringResource(R.string.hide_password)
else stringResource(R.string.show_password),
)
}
},
containerColor = MaterialTheme.colorScheme.surface,
modifier = Modifier.fillMaxWidth(),
isError = showPasswordError,
)
}
if (showPasswordError) {
Text(
text = stringResource(R.string.passwords_do_not_match),
color = MaterialTheme.colorScheme.error,
style = MaterialTheme.typography.bodySmall,
)
}
}
}
},
)
}
@@ -56,7 +56,7 @@ fun TunnelGlobalsScreen(
)
},
enabled = sharedUiState.tunnelMode != TunnelMode.PROXY,
title = stringResource(R.string.splt_tunneling),
title = stringResource(R.string.global_split_tunneling),
trailing = { modifier ->
SwitchWithDivider(
checked = uiState.settings.isGlobalSplitTunnelEnabled,
@@ -82,7 +82,7 @@ fun TunnelGlobalsScreen(
)
SurfaceRow(
leading = { Icon(Icons.Outlined.Description, contentDescription = null) },
title = stringResource(R.string.tunnel_configuration),
title = stringResource(R.string.configuration_globals),
onClick = {
uiState.globalTunnelConfig?.let {
navController.push(Route.ConfigGlobal(id = it.id))
@@ -11,7 +11,7 @@ import androidx.compose.foundation.verticalScroll
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.automirrored.outlined.Launch
import androidx.compose.material.icons.filled.AppShortcut
import androidx.compose.material.icons.filled.SettingsRemote
import androidx.compose.material.icons.filled.SmartToy
import androidx.compose.material.icons.outlined.AdminPanelSettings
import androidx.compose.material.icons.outlined.ContentCopy
import androidx.compose.material.icons.outlined.Key
@@ -133,22 +133,20 @@ fun AndroidIntegrationsScreen(viewModel: SettingsViewModel = koinViewModel()) {
onClick = { viewModel.setShortcutsEnabled(it) },
)
},
title = stringResource(R.string.app_shortcuts),
description = { DescriptionText(stringResource(R.string.app_shortcuts_desc)) },
title = stringResource(R.string.enabled_app_shortcuts),
onClick = {
viewModel.setShortcutsEnabled(!settingsState.settings.isShortcutsEnabled)
},
)
SurfaceRow(
leading = { Icon(Icons.Filled.SettingsRemote, contentDescription = null) },
leading = { Icon(Icons.Filled.SmartToy, contentDescription = null) },
trailing = {
ThemedSwitch(
checked = settingsState.isRemoteEnabled,
onClick = { viewModel.setRemoteEnabled(it) },
)
},
title = stringResource(R.string.remote_control),
description = { DescriptionText(stringResource(R.string.remote_control_desc)) },
title = stringResource(R.string.enable_remote_app_control),
onClick = { viewModel.setRemoteEnabled(!settingsState.isRemoteEnabled) },
)
AnimatedVisibility(settingsState.isRemoteEnabled) {
@@ -12,7 +12,6 @@ import androidx.compose.runtime.setValue
import androidx.compose.runtime.snapshotFlow
import androidx.compose.ui.Modifier
import androidx.compose.ui.res.stringResource
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.logs.components.LogList
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.logs.components.LogsBottomSheet
@@ -87,15 +86,13 @@ fun LogsScreen(
},
onCanceled = {
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.export_canceled),
ToastType.Warning,
StringValue.StringResource(R.string.export_canceled)
)
showLogsSheet = false
},
onUnsupported = {
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.export_unsupported),
ToastType.Warning,
StringValue.StringResource(R.string.export_unsupported)
)
showLogsSheet = false
},
@@ -1,12 +1,10 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.settings.logs.components
import androidx.compose.foundation.gestures.Orientation
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.lazy.LazyColumn
import androidx.compose.foundation.lazy.LazyListState
import androidx.compose.foundation.lazy.itemsIndexed
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable
import androidx.compose.ui.Modifier
import androidx.compose.ui.unit.dp
@@ -20,13 +18,7 @@ fun LogList(
) {
LazyColumn(
state = lazyColumnListState,
modifier =
modifier
.padding(horizontal = 12.dp)
.scrollbar(
state = lazyColumnListState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
modifier = modifier.padding(horizontal = 12.dp),
verticalArrangement = Arrangement.spacedBy(2.dp),
) {
itemsIndexed(items = logs, key = { index, _ -> index }) { _, log -> LogItem(log = log) }
@@ -2,8 +2,6 @@ package com.zaneschepke.wireguardautotunnel.ui.screens.settings.proxy.compoents
import androidx.compose.runtime.Composable
import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.stringResource
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
import com.zaneschepke.wireguardautotunnel.ui.common.sheet.CustomBottomSheet
import com.zaneschepke.wireguardautotunnel.ui.common.sheet.SheetOption
@@ -29,12 +27,6 @@ fun AppModeBottomSheet(
onDismiss()
onAppModeChange(it)
},
description =
when (it) {
TunnelMode.VPN -> stringResource(R.string.vpn_desc)
TunnelMode.PROXY -> stringResource(R.string.local_proxy_desc)
TunnelMode.LOCK_DOWN -> stringResource(R.string.lockdown_desc)
},
selected = tunnelMode == it,
)
}
@@ -19,7 +19,6 @@ import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.ui.LocalNavController
import com.zaneschepke.wireguardautotunnel.ui.common.button.SurfaceRow
import com.zaneschepke.wireguardautotunnel.ui.common.button.ThemedSwitch
import com.zaneschepke.wireguardautotunnel.ui.common.text.DescriptionText
import com.zaneschepke.wireguardautotunnel.ui.navigation.Route
import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel
import org.koin.compose.viewmodel.koinActivityViewModel
@@ -46,16 +45,13 @@ fun SecurityScreen(viewModel: SharedAppViewModel = koinActivityViewModel()) {
onClick = { viewModel.setScreenRecordingSecurity(it) },
)
},
description = {
DescriptionText(stringResource(R.string.screen_recording_protection_desc))
},
onClick = {
viewModel.setScreenRecordingSecurity(!uiState.isScreenRecordingProtectionEnabled)
},
)
SurfaceRow(
leading = { Icon(Icons.Outlined.Pin, contentDescription = null) },
title = stringResource(R.string.app_lock),
title = stringResource(R.string.enable_app_lock),
trailing = {
ThemedSwitch(
checked = uiState.pinLockEnabled,
@@ -68,7 +64,6 @@ fun SecurityScreen(viewModel: SharedAppViewModel = koinActivityViewModel()) {
},
)
},
description = { DescriptionText(stringResource(R.string.app_lock_desc)) },
onClick = {
if (!uiState.pinLockEnabled) {
navController.push(Route.Lock)
@@ -26,7 +26,6 @@ import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberCoroutineScope
import androidx.compose.runtime.saveable.rememberSaveable
import androidx.compose.runtime.setValue
import androidx.compose.ui.Alignment
@@ -38,10 +37,8 @@ import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.stringResource
import androidx.compose.ui.res.vectorResource
import androidx.compose.ui.unit.dp
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.BuildConfig
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.domain.sideeffect.GlobalSideEffect
import com.zaneschepke.wireguardautotunnel.ui.LocalIsAndroidTV
import com.zaneschepke.wireguardautotunnel.ui.LocalNavController
import com.zaneschepke.wireguardautotunnel.ui.common.button.SurfaceRow
@@ -52,13 +49,12 @@ import com.zaneschepke.wireguardautotunnel.ui.navigation.Route
import com.zaneschepke.wireguardautotunnel.ui.screens.support.components.PermissionDialog
import com.zaneschepke.wireguardautotunnel.ui.screens.support.components.UpdateDialog
import com.zaneschepke.wireguardautotunnel.util.Constants
import com.zaneschepke.wireguardautotunnel.util.StringValue
import com.zaneschepke.wireguardautotunnel.util.extensions.launchPlayStoreListing
import com.zaneschepke.wireguardautotunnel.util.extensions.launchPlayStoreReview
import com.zaneschepke.wireguardautotunnel.util.extensions.launchSupportEmail
import com.zaneschepke.wireguardautotunnel.util.extensions.openWebUrl
import com.zaneschepke.wireguardautotunnel.util.extensions.showToast
import com.zaneschepke.wireguardautotunnel.viewmodel.SupportViewModel
import kotlinx.coroutines.launch
import org.koin.androidx.compose.koinViewModel
import org.orbitmvi.orbit.compose.collectAsState
@@ -67,24 +63,11 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
val context = LocalContext.current
val navController = LocalNavController.current
val isTv = LocalIsAndroidTV.current
val scope = rememberCoroutineScope()
val supportState by viewModel.collectAsState()
val clipboardManager = rememberClipboardHelper()
val issuesUrl = stringResource(R.string.github_url)
val izzyUrl = stringResource(R.string.fdroid_url)
val telegramUrl = stringResource(R.string.telegram_url)
val matrixUrl = stringResource(R.string.matrix_url)
val docsUrl = stringResource(R.string.docs_url)
val websiteUrl = stringResource(R.string.website_url)
val translationUrl = stringResource(R.string.translation_url)
val privacyPolicyUrl = stringResource(R.string.privacy_policy_url)
val playStoreUrl = "https://play.google.com/store/apps/details?id=${context.packageName}"
val playReviewsUrl =
"https://play.google.com/store/apps/details?id=${context.packageName}&showAllReviews=true"
val version = remember {
"v${BuildConfig.VERSION_NAME +
if(BuildConfig.DEBUG) "-debug" else "" }"
@@ -112,19 +95,6 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
PermissionDialog(context = context, onDismiss = { showPermissionDialog = false })
}
fun openWebUrl(url: String) {
context.openWebUrl(url).onFailure {
scope.launch {
viewModel.postSideEffect(
GlobalSideEffect.Snackbar(
StringValue.StringResource(R.string.no_browser_detected),
ToastType.Error,
)
)
}
}
}
Column(
modifier = Modifier.fillMaxSize().verticalScroll(rememberScrollState()),
horizontalAlignment = Alignment.Start,
@@ -145,19 +115,19 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
)
SurfaceRow(
stringResource(R.string.docs_description),
onClick = { openWebUrl(docsUrl) },
onClick = { context.openWebUrl(context.getString(R.string.docs_url)) },
leading = { Icon(Icons.Outlined.Book, contentDescription = null) },
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
)
SurfaceRow(
stringResource(R.string.website),
onClick = { openWebUrl(websiteUrl) },
onClick = { context.openWebUrl(context.getString(R.string.website_url)) },
leading = { Icon(Icons.Outlined.Web, contentDescription = null) },
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
)
SurfaceRow(
stringResource(R.string.translation),
onClick = { openWebUrl(translationUrl) },
onClick = { context.openWebUrl(context.getString(R.string.translation_url)) },
description = { DescriptionText(stringResource(R.string.help_translate)) },
leading = { Icon(Icons.Outlined.Translate, contentDescription = null) },
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
@@ -171,16 +141,14 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
leading = { Icon(Icons.Outlined.Policy, contentDescription = null) },
title = stringResource(R.string.privacy_policy),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = { openWebUrl(privacyPolicyUrl) },
onClick = { context.openWebUrl(context.getString(R.string.privacy_policy_url)) },
)
if (BuildConfig.FLAVOR == Constants.GOOGLE_PLAY_FLAVOR) {
SurfaceRow(
leading = { Icon(Icons.Outlined.Reviews, contentDescription = null) },
title = stringResource(R.string.review),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = {
context.launchPlayStoreReview().onFailure { openWebUrl(playReviewsUrl) }
},
onClick = { context.launchPlayStoreReview() },
)
}
}
@@ -199,7 +167,7 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
},
title = stringResource(R.string.join_matrix),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = { openWebUrl(matrixUrl) },
onClick = { context.openWebUrl(context.getString(R.string.matrix_url)) },
)
SurfaceRow(
leading = {
@@ -211,7 +179,7 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
},
title = stringResource(R.string.join_telegram),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = { openWebUrl(telegramUrl) },
onClick = { context.openWebUrl(context.getString(R.string.telegram_url)) },
)
SurfaceRow(
leading = {
@@ -223,24 +191,13 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
},
title = stringResource(R.string.open_issue),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = { openWebUrl(issuesUrl) },
onClick = { context.openWebUrl(context.getString(R.string.github_url)) },
)
SurfaceRow(
leading = { Icon(Icons.Outlined.Mail, contentDescription = null) },
title = stringResource(R.string.email_description),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = {
context.launchSupportEmail().onFailure {
scope.launch {
viewModel.postSideEffect(
GlobalSideEffect.Snackbar(
StringValue.StringResource(R.string.no_email_detected),
ToastType.Error,
)
)
}
}
},
onClick = { context.launchSupportEmail() },
)
}
Column {
@@ -265,21 +222,12 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
leading = { Icon(Icons.Outlined.InstallMobile, contentDescription = null) },
title = stringResource(R.string.check_for_update),
onClick = {
if (BuildConfig.DEBUG) {
scope.launch {
viewModel.postSideEffect(
GlobalSideEffect.Snackbar(
StringValue.StringResource(R.string.update_check_unsupported),
ToastType.Warning,
)
)
}
return@SurfaceRow
}
if (BuildConfig.DEBUG)
return@SurfaceRow context.showToast(R.string.update_check_unsupported)
when (BuildConfig.FLAVOR) {
Constants.GOOGLE_PLAY_FLAVOR ->
context.launchPlayStoreListing().onFailure { openWebUrl(playStoreUrl) }
Constants.FDROID_FLAVOR -> openWebUrl(izzyUrl)
Constants.GOOGLE_PLAY_FLAVOR -> context.launchPlayStoreListing()
Constants.FDROID_FLAVOR ->
context.openWebUrl(context.getString(R.string.fdroid_url))
else -> viewModel.checkForStandaloneUpdate()
}
},
@@ -1,12 +1,10 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.support.donate.crypto
import androidx.compose.foundation.gestures.Orientation
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.rememberScrollState
import androidx.compose.foundation.verticalScroll
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
@@ -15,17 +13,10 @@ import com.zaneschepke.wireguardautotunnel.ui.screens.support.donate.crypto.comp
@Composable
fun AddressesScreen() {
val scrollState = rememberScrollState()
Column(
horizontalAlignment = Alignment.Start,
verticalArrangement = Arrangement.Top,
modifier =
Modifier.fillMaxSize()
.verticalScroll(scrollState)
.scrollbar(
state = scrollState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
modifier = Modifier.fillMaxSize().verticalScroll(rememberScrollState()),
) {
val clipboard = rememberClipboardHelper()
Address.allAddresses.forEach { AddressItem(it) { address -> clipboard.copy(address) } }
@@ -1,213 +0,0 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.support.license
import android.content.Context
import com.mikepenz.aboutlibraries.Libs
import com.mikepenz.aboutlibraries.entity.Developer
import com.mikepenz.aboutlibraries.entity.Library
import com.mikepenz.aboutlibraries.entity.License
import com.mikepenz.aboutlibraries.entity.Scm
import com.mikepenz.aboutlibraries.util.withContext
fun buildLibsWithAdditionalLibraries(context: Context): Libs {
val baseLibs = Libs.Builder().withContext(context).build()
val cleanedBaseLibs =
baseLibs.libraries.filterNot { library ->
library.uniqueId.contains("com.github.topjohnwu.libsu", ignoreCase = true) ||
library.uniqueId.contains("com.github.T8RIN.QuickieExtended", ignoreCase = true)
}
val nativeLibraries =
listOf(
Library(
uniqueId = "github.com.wgtunnel:amneziawg-go",
artifactVersion = "v0.0.0-20260618075902-e1b699b2104b",
name = "AmneziaWG Go (Fork)",
description = "WireGuard implementation with Amnezia obfuscation",
website = "https://wgtunnel.com",
developers =
listOf(
Developer(
name = "Zane Schepke (Fork Maintainer)",
organisationUrl = "https://wgtunnel.com",
),
Developer(
name = "Jason A. Donenfeld (Original WireGuard)",
organisationUrl = "https://www.wireguard.com/",
),
Developer(
name = "Amnezia VPN Team",
organisationUrl = "https://amnezia.org/",
),
),
organization = null,
scm = Scm(null, null, "https://github.com/wgtunnel/amneziawg-go"),
licenses =
setOf(
License(
name = "MIT License",
url = "https://opensource.org/licenses/MIT",
spdxId = "MIT",
hash = "mit-license-amneziawg-fork",
)
),
funding = emptySet(),
tag = "native",
),
Library(
uniqueId = "github.com.wgtunnel:wireproxy-awg",
artifactVersion = "v0.0.0-20260309043206-ff4200f20ff2",
name = "Wireproxy AWG (Fork)",
description = "WireGuard proxy with Amnezia support",
website = "https://wgtunnel.com",
developers =
listOf(
Developer(
name = "Zane Schepke (Fork Maintainer)",
organisationUrl = "https://wgtunnel.com",
),
Developer(name = "Artem Russkikh (Original)", organisationUrl = null),
),
organization = null,
scm = Scm(null, null, "https://github.com/wgtunnel/wireproxy-awg"),
licenses =
setOf(
License(
name = "MIT License",
url = "https://opensource.org/licenses/MIT",
spdxId = "MIT",
hash = "mit-license-wireproxy-fork",
)
),
funding = emptySet(),
tag = "native",
),
Library(
uniqueId = "github.com.wgtunnel:go-socks5",
artifactVersion = "v0.0.0-20260307052555-86f8d93b9534",
name = "go-socks5 (Fork)",
description = "SOCKS5 proxy server implementation",
website = "https://wgtunnel.com",
developers =
listOf(
Developer(
name = "Zane Schepke (Fork Maintainer)",
organisationUrl = "https://wgtunnel.com",
),
Developer(name = "Things-go Team (Original)", organisationUrl = null),
),
organization = null,
scm = Scm(null, null, "https://github.com/wgtunnel/go-socks5"),
licenses =
setOf(
License(
name = "MIT License",
url = "https://opensource.org/licenses/MIT",
spdxId = "MIT",
hash = "mit-license-go-socks5-fork",
)
),
funding = emptySet(),
tag = "native",
),
Library(
uniqueId = "github.com.miekg:dns",
artifactVersion = "v1.1.69",
name = "miekg/dns",
description = "DNS library for Go",
website = "https://github.com/miekg/dns",
developers = listOf(Developer(name = "Miek Gieben", organisationUrl = null)),
organization = null,
scm = Scm(null, null, "https://github.com/miekg/dns"),
licenses =
setOf(
License(
name = "BSD 3-Clause \"New\" or \"Revised\" License",
url = "https://opensource.org/licenses/BSD-3-Clause",
spdxId = "BSD-3-Clause",
hash = "bsd3-miekg-dns",
)
),
funding = emptySet(),
tag = "go",
),
Library(
uniqueId = "github.com.heiher:hev-socks5-tunnel",
artifactVersion = "2.15.0",
name = "hev-socks5-tunnel",
description = "High performance SOCKS5 tunnel",
website = "https://github.com/heiher/hev-socks5-tunnel",
developers = listOf(Developer(name = "heiher", organisationUrl = null)),
organization = null,
scm = Scm(null, null, "https://github.com/heiher/hev-socks5-tunnel"),
licenses =
setOf(
License(
name = "MIT License",
url = "https://opensource.org/licenses/MIT",
spdxId = "MIT",
hash = "mit-license-hev",
)
),
funding = emptySet(),
tag = "native",
),
)
val additionalLibraries =
listOf(
Library(
uniqueId = "com.github.T8RIN.QuickieExtended:quickie-foss",
artifactVersion = "1.18.1",
name = "QuickieFoss",
description = "Camera QR code scanner",
website = "https://github.com/T8RIN/QuickieExtended",
developers = listOf(Developer(name = "T8RIN", null)),
organization = null,
scm = Scm(null, null, "https://github.com/T8RIN/QuickieExtended"),
licenses =
setOf(
License(
name = "Apache License 2.0",
url = "https://www.apache.org/licenses/LICENSE-2.0",
spdxId = "Apache-2.0",
hash = "apache-2-quickie",
)
),
funding = emptySet(),
tag = "ui",
),
Library(
uniqueId = "com.github.topjohnwu.libsu:core",
artifactVersion = "6.0.0",
name = "libsu",
description = "Root shell library for Android",
website = "https://github.com/topjohnwu/libsu",
developers = listOf(Developer(name = "topjohnwu", null)),
organization = null,
scm = Scm(null, null, "https://github.com/topjohnwu/libsu"),
licenses =
setOf(
License(
name = "Apache License 2.0",
url = "https://www.apache.org/licenses/LICENSE-2.0",
spdxId = "Apache-2.0",
hash = "apache-2-libsu",
)
),
funding = emptySet(),
tag = "system",
),
)
return Libs(
libraries =
(cleanedBaseLibs + nativeLibraries + additionalLibraries).sortedBy {
it.name.lowercase()
},
licenses =
baseLibs.licenses +
nativeLibraries.flatMap { it.licenses } +
additionalLibraries.flatMap { it.licenses },
)
}
@@ -0,0 +1,15 @@
import kotlinx.serialization.Serializable
@Serializable
data class LicenseFileEntry(
val groupId: String,
val artifactId: String,
val version: String,
val name: String? = null,
val spdxLicenses: List<SpdxLicense> = emptyList(),
val scm: Scm? = null,
)
@Serializable data class SpdxLicense(val identifier: String, val name: String, val url: String)
@Serializable data class Scm(val url: String)
@@ -1,23 +1,30 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.support.license
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.size
import androidx.compose.material3.CircularWavyProgressIndicator
import androidx.compose.material3.ExperimentalMaterial3ExpressiveApi
import androidx.compose.runtime.Composable
import androidx.compose.runtime.remember
import androidx.compose.runtime.getValue
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.platform.LocalContext
import com.mikepenz.aboutlibraries.ui.compose.m3.LibrariesContainer
import com.mikepenz.aboutlibraries.ui.compose.variant.LibraryDetailMode
import androidx.compose.ui.unit.dp
import com.zaneschepke.wireguardautotunnel.ui.screens.support.license.components.LicenseList
import com.zaneschepke.wireguardautotunnel.viewmodel.LicenseViewModel
import org.koin.androidx.compose.koinViewModel
import org.orbitmvi.orbit.compose.collectAsState
@OptIn(ExperimentalMaterial3ExpressiveApi::class)
@Composable
fun LicenseScreen() {
val context = LocalContext.current
val libs = remember { buildLibsWithAdditionalLibraries(context) }
fun LicenseScreen(viewModel: LicenseViewModel = koinViewModel()) {
val licenseUiState by viewModel.collectAsState()
LibrariesContainer(
libraries = libs,
modifier = Modifier.fillMaxSize(),
detailMode = LibraryDetailMode.Sheet,
)
if (licenseUiState.isLoading) {
Box(Modifier.fillMaxSize(), contentAlignment = Alignment.Center) {
CircularWavyProgressIndicator(waveSpeed = 60.dp, modifier = Modifier.size(48.dp))
}
} else {
LicenseList(licenseUiState.licenses)
}
}
@@ -0,0 +1,57 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.support.license.components
import LicenseFileEntry
import androidx.compose.foundation.clickable
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.Row
import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.lazy.LazyColumn
import androidx.compose.foundation.lazy.items
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.automirrored.outlined.Launch
import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Text
import androidx.compose.runtime.Composable
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.unit.dp
import com.zaneschepke.wireguardautotunnel.util.extensions.openWebUrl
@Composable
fun LicenseList(licenses: List<LicenseFileEntry>) {
val context = LocalContext.current
LazyColumn(modifier = Modifier.fillMaxSize()) {
items(licenses) { entry ->
Row(
verticalAlignment = Alignment.CenterVertically,
horizontalArrangement = Arrangement.SpaceBetween,
modifier =
Modifier.clickable(enabled = entry.scm?.url != null) {
entry.scm?.url?.let { context.openWebUrl(it) }
}
.padding(horizontal = 16.dp, vertical = 8.dp),
) {
Column(modifier = Modifier.weight(1f)) {
Text(
text = "${entry.artifactId} (${entry.version})",
style = MaterialTheme.typography.titleSmall,
)
entry.spdxLicenses.forEach { license ->
Text(
text = license.name,
style = MaterialTheme.typography.bodySmall,
color = MaterialTheme.colorScheme.primary,
)
}
}
entry.scm?.url?.let { Icon(Icons.AutoMirrored.Outlined.Launch, null) }
}
}
}
}
@@ -13,7 +13,6 @@ import androidx.compose.ui.Modifier
import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.stringResource
import androidx.lifecycle.compose.collectAsStateWithLifecycle
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.ui.LocalNavController
import com.zaneschepke.wireguardautotunnel.ui.common.dialog.InfoDialog
@@ -51,15 +50,11 @@ fun TunnelsScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
rememberFileExportLauncherForResult(
onSuccess = { uri -> sharedViewModel.exportSelectedTunnels(uri) },
onCanceled = {
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.export_canceled),
ToastType.Warning,
)
sharedViewModel.showToast(StringValue.StringResource(R.string.export_canceled))
},
onUnsupported = {
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.export_unsupported),
ToastType.Warning,
StringValue.StringResource(R.string.export_unsupported)
)
},
)
@@ -78,8 +73,7 @@ fun TunnelsScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
selectedTunnelsExportLauncher.launch(fileName)
} else {
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.error_no_file_explorer),
ToastType.Error,
StringValue.StringResource(R.string.error_no_file_explorer)
)
}
}
@@ -93,8 +87,7 @@ fun TunnelsScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
rememberFileImportLauncherForResult(
onNoFileExplorer = {
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.error_no_file_explorer),
ToastType.Error,
StringValue.StringResource(R.string.error_no_file_explorer)
)
},
onData = { data -> sharedViewModel.importFromUri(data) },
@@ -108,15 +101,13 @@ fun TunnelsScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
}
QRResult.QRMissingPermission -> {
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.camera_permission_required),
ToastType.Warning,
StringValue.StringResource(R.string.camera_permission_required)
)
}
is QRResult.QRSuccess -> {
result.content.rawValue?.let { sharedViewModel.importFromQr(it) }
?: sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.config_error),
ToastType.Error,
StringValue.StringResource(R.string.config_error)
)
}
QRResult.QRUserCanceled -> Unit
@@ -128,8 +119,7 @@ fun TunnelsScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
->
if (!isGranted) {
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.camera_permission_required),
ToastType.Warning,
StringValue.StringResource(R.string.camera_permission_required)
)
return@rememberLauncherForActivityResult
}
@@ -42,8 +42,5 @@ fun PeerStatisticsSection(peer: ActivePeer) {
style = style,
color = color,
)
peer.endpoint?.let {
StatText(stringResource(R.string.endpoint_template, it), style = style, color = color)
}
}
}
@@ -42,6 +42,6 @@ private fun TransferMetric(icon: ImageVector, text: String, style: TextStyle, co
modifier = Modifier.size(12.dp),
)
Text(text = text, style = style, color = color)
Text(text = text.lowercase(), style = style, color = color)
}
}
@@ -1,7 +1,6 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.components
import androidx.compose.foundation.ExperimentalFoundationApi
import androidx.compose.foundation.gestures.Orientation
import androidx.compose.foundation.gestures.ScrollableDefaults
import androidx.compose.foundation.gestures.detectTapGestures
import androidx.compose.foundation.layout.size
@@ -13,10 +12,8 @@ import androidx.compose.foundation.rememberOverscrollEffect
import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.rounded.Circle
import androidx.compose.material3.Icon
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable
import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.getValue
import androidx.compose.runtime.remember
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
@@ -69,11 +66,7 @@ fun TunnelList(
viewModel.clearSelectedTunnels()
}
}
.overscroll(rememberOverscrollEffect())
.scrollbar(
state = lazyListState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
.overscroll(rememberOverscrollEffect()),
state = lazyListState,
userScrollEnabled = true,
reverseLayout = false,
@@ -94,7 +87,8 @@ fun TunnelList(
uiState.backendStatus.activeTunnels[tunnel.id] ?: ActiveTunnel()
}
val displayState = remember(activeTunnel) { DisplayTunnelState.from(activeTunnel) }
val displayState =
uiState.displayStates[tunnel.id] ?: DisplayTunnelState.from(activeTunnel)
val isRunning = uiState.backendStatus.activeTunnels.containsKey(tunnel.id)
@@ -113,7 +107,7 @@ fun TunnelList(
Icon(
Icons.Rounded.Circle,
contentDescription = stringResource(R.string.tunnel_monitoring),
tint = remember(displayState) { displayState.asColor() },
tint = displayState.asColor(),
modifier = Modifier.size(14.dp),
)
},
@@ -141,10 +141,6 @@ fun TunnelSettingsScreen(
onClick = { navController.push(Route.IPv6(tunnel.id)) },
)
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
val meteredDisabled = sharedUiState.tunnelMode == TunnelMode.PROXY
val meteredTunnelDesc =
if (meteredDisabled) stringResource(R.string.unavailable_in_mode)
else stringResource(R.string.metered_tunnel_desc)
SurfaceRow(
leading = {
Icon(
@@ -157,9 +153,15 @@ fun TunnelSettingsScreen(
},
title = stringResource(R.string.metered_tunnel),
enabled = sharedUiState.tunnelMode != TunnelMode.PROXY,
description = {
DescriptionText(meteredTunnelDesc, disabled = meteredDisabled)
},
description =
if (sharedUiState.tunnelMode == TunnelMode.PROXY) {
{
DescriptionText(
stringResource(R.string.unavailable_in_mode),
disabled = true,
)
}
} else null,
trailing = {
ThemedSwitch(
checked = tunnel.isMetered,
@@ -170,26 +172,26 @@ fun TunnelSettingsScreen(
onClick = { viewModel.onMetered(!tunnel.isMetered) },
)
}
}
Column {
GroupLabel(
stringResource(R.string.automation),
modifier = Modifier.padding(horizontal = 16.dp),
)
SurfaceRow(
leading = { Icon(Icons.Outlined.Dns, contentDescription = null) },
title = stringResource(R.string.ddns_auto_update),
description = {
DescriptionText(stringResource(R.string.ddns_auto_update_description))
},
trailing = {
ThemedSwitch(
checked = tunnel.dynamicDnsEnabled,
onClick = { viewModel.onDynamicDns(it) },
)
},
onClick = { viewModel.onDynamicDns(!tunnel.dynamicDnsEnabled) },
)
Column {
GroupLabel(
stringResource(R.string.automation),
modifier = Modifier.padding(horizontal = 16.dp),
)
SurfaceRow(
leading = { Icon(Icons.Outlined.Dns, contentDescription = null) },
title = stringResource(R.string.ddns_auto_update),
description = {
DescriptionText(stringResource(R.string.ddns_auto_update_description))
},
trailing = {
ThemedSwitch(
checked = tunnel.dynamicDnsEnabled,
onClick = { viewModel.onDynamicDns(it) },
)
},
onClick = { viewModel.onDynamicDns(!tunnel.dynamicDnsEnabled) },
)
}
}
}
}
@@ -1,6 +1,5 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.settings.config
import androidx.compose.foundation.gestures.Orientation
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.fillMaxSize
@@ -10,7 +9,6 @@ import androidx.compose.foundation.text.selection.SelectionContainer
import androidx.compose.foundation.verticalScroll
import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Text
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable
import androidx.compose.runtime.derivedStateOf
import androidx.compose.runtime.getValue
@@ -28,16 +26,14 @@ import androidx.compose.ui.text.font.FontFamily
import androidx.compose.ui.text.font.FontStyle
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.unit.dp
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.domain.sideeffect.GlobalSideEffect
import com.zaneschepke.wireguardautotunnel.ui.common.functions.rememberClipboardHelper
import com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.settings.config.components.QrCodeDialog
import com.zaneschepke.wireguardautotunnel.ui.sideeffect.LocalSideEffect
import com.zaneschepke.wireguardautotunnel.ui.theme.ConfigHeaderColor
import com.zaneschepke.wireguardautotunnel.ui.theme.ConfigKeyColor
import com.zaneschepke.wireguardautotunnel.util.StringValue
import com.zaneschepke.wireguardautotunnel.util.extensions.isTextTooLargeForQr
import com.zaneschepke.wireguardautotunnel.util.extensions.showToast
import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.TunnelViewModel
import org.koin.compose.viewmodel.koinActivityViewModel
@@ -61,8 +57,6 @@ fun ConfigScreen(
var showQrModal by rememberSaveable { mutableStateOf(false) }
val scrollState = rememberScrollState()
val rawConfig by
remember(liveConfig, uiState.activeConfig, uiState.tunnel?.quickConfig) {
derivedStateOf {
@@ -78,12 +72,7 @@ fun ConfigScreen(
when (sideEffect) {
is LocalSideEffect.Modal.QR -> {
if (tunnel.quickConfig.isTextTooLargeForQr()) {
sharedViewModel.postSideEffect(
GlobalSideEffect.Snackbar(
StringValue.StringResource(R.string.text_too_large_for_qr),
ToastType.Error,
)
)
context.showToast(R.string.text_too_large_for_qr)
} else {
showQrModal = true
}
@@ -101,13 +90,7 @@ fun ConfigScreen(
Column(
horizontalAlignment = Alignment.Start,
verticalArrangement = Arrangement.spacedBy(12.dp, Alignment.Top),
modifier =
Modifier.fillMaxSize()
.verticalScroll(scrollState)
.scrollbar(
state = scrollState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
modifier = Modifier.fillMaxSize().verticalScroll(rememberScrollState()),
) {
val displayText by
remember(rawConfig, showKeys) { derivedStateOf { maskSensitive(rawConfig, showKeys) } }
@@ -1,6 +1,5 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.settings.config.edit
import androidx.compose.foundation.gestures.Orientation
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.Spacer
@@ -13,7 +12,6 @@ import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.outlined.HdrAuto
import androidx.compose.material3.Icon
import androidx.compose.material3.Text
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
@@ -51,12 +49,11 @@ fun ConfigEditScreen(
val uiState by viewModel.collectAsState()
if (uiState.isLoading) return
val locale = Locale.current.platformLocale
var showSelectionDialog by rememberSaveable { mutableStateOf(false) }
val scrollState = rememberScrollState()
sharedViewModel.collectSideEffect { sideEffect ->
when (sideEffect) {
is LocalSideEffect.SaveChanges -> {
@@ -107,14 +104,7 @@ fun ConfigEditScreen(
Column(
horizontalAlignment = Alignment.Start,
verticalArrangement = Arrangement.spacedBy(16.dp, Alignment.Top),
modifier =
Modifier.fillMaxSize()
.imePadding()
.verticalScroll(scrollState)
.scrollbar(
state = scrollState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
modifier = Modifier.fillMaxSize().imePadding().verticalScroll(rememberScrollState()),
) {
if (uiState.isGlobalConfig) {
Column {
@@ -122,7 +112,7 @@ fun ConfigEditScreen(
leading = {
Icon(ImageVector.vectorResource(R.drawable.host), contentDescription = null)
},
title = stringResource(R.string.dns_servers),
title = stringResource(R.string.global_dns_servers),
trailing = { modifier ->
ThemedSwitch(
checked = uiState.globalSettings.dnsEnabled,
@@ -136,7 +126,7 @@ fun ConfigEditScreen(
)
SurfaceRow(
leading = { Icon(Icons.Outlined.HdrAuto, contentDescription = null) },
title = stringResource(R.string.amnezia_configuration),
title = stringResource(R.string.global_amnezia_configuration),
trailing = { modifier ->
ThemedSwitch(
checked = uiState.globalSettings.amneziaEnabled,
@@ -1,6 +1,5 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.sort
import androidx.compose.foundation.gestures.Orientation
import androidx.compose.foundation.gestures.ScrollableDefaults
import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Row
@@ -18,7 +17,6 @@ import androidx.compose.material.icons.filled.ArrowUpward
import androidx.compose.material.icons.filled.DragHandle
import androidx.compose.material3.Icon
import androidx.compose.material3.IconButton
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
@@ -109,11 +107,7 @@ fun SortScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel()) {
Modifier.pointerInput(Unit) {
if (tunnelsUiState.tunnels.isEmpty()) return@pointerInput
}
.overscroll(rememberOverscrollEffect())
.scrollbar(
state = lazyListState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
.overscroll(rememberOverscrollEffect()),
state = lazyListState,
userScrollEnabled = true,
reverseLayout = false,
@@ -26,33 +26,38 @@ sealed class DisplayTunnelState {
data object Connected : DisplayTunnelState()
data object HandshakeFailure : DisplayTunnelState()
data object Degraded : DisplayTunnelState()
@StringRes
fun labelRes(): Int =
when (this) {
fun labelRes(): Int {
return when (this) {
Disconnected -> R.string.tunnel_state_disconnected
Connecting,
EstablishingConnection -> R.string.tunnel_state_establishing_connection
Connecting -> R.string.tunnel_state_starting
ResolvingDns -> R.string.tunnel_state_resolving_dns
EstablishingConnection -> R.string.tunnel_state_establishing_connection
Ready -> R.string.ready
Connected -> R.string.tunnel_state_connected
HandshakeFailure -> R.string.tunnel_state_handshake_failure
Degraded -> R.string.tunnel_state_handshake_failure
}
}
fun asColor(): Color =
when (this) {
fun asLocalizedString(context: Context): String {
return context.getString(labelRes())
}
fun asColor(): Color {
return when (this) {
Disconnected -> CoolGray
Connecting,
ResolvingDns,
EstablishingConnection,
Ready -> Straw
Connected -> SilverTree
HandshakeFailure -> AlertRed
}
fun asLocalizedString(context: Context): String {
return context.getString(labelRes())
Connected -> SilverTree
Degraded -> AlertRed
}
}
companion object {
@@ -62,21 +67,49 @@ sealed class DisplayTunnelState {
val mode = activeTunnel.mode
val isVpnStyle = mode is BackendMode.Vpn || mode is BackendMode.Proxy.KillSwitchPrimary
// Static peers bootstrap never goes to complete, treat none the same
val bootstrapPhaseDone =
bootstrap is BootstrapState.Complete || bootstrap is BootstrapState.None
return when {
transport is Tunnel.State.Down -> Disconnected
bootstrap is BootstrapState.Failed -> HandshakeFailure
bootstrap is BootstrapState.Failed -> Degraded
// DNS resolution still in progress
bootstrap is BootstrapState.ResolvingDns ||
bootstrap is BootstrapState.UpdatingPeers -> ResolvingDns
transport is Tunnel.State.Up.Healthy -> Connected
transport is Tunnel.State.Up.HandshakeFailure -> HandshakeFailure
transport is Tunnel.State.Up.HandshakeFailure -> {
val age = System.currentTimeMillis() - activeTunnel.lastStateChangeMs
transport is Tunnel.State.Starting ->
if (isVpnStyle) EstablishingConnection else Ready
if (age > 15_000L && bootstrapPhaseDone) {
Degraded
} else if (isVpnStyle && bootstrapPhaseDone) {
EstablishingConnection
} else if (bootstrapPhaseDone) {
// For regular proxy mode, we go to ready once past bootstrap phase
Ready
} else {
Connecting
}
}
else -> if (isVpnStyle) EstablishingConnection else Ready
transport is Tunnel.State.Starting -> {
when {
bootstrapPhaseDone -> {
if (isVpnStyle) EstablishingConnection else Ready
}
else -> Connecting
}
}
// Final fallback after bootstrap phase is done
bootstrapPhaseDone -> if (isVpnStyle) EstablishingConnection else Ready
else -> Connecting
}
}
}
@@ -18,6 +18,5 @@ data class GlobalAppUiState(
val selectedTunnelCount: Int = 0,
val alreadyDonated: Boolean = false,
val isPinVerified: Boolean = false,
val pendingWgImportUrl: String? = null,
val isScreenRecordingProtectionEnabled: Boolean = false,
)
@@ -0,0 +1,8 @@
package com.zaneschepke.wireguardautotunnel.ui.state
import LicenseFileEntry
data class LicenseUiState(
val isLoading: Boolean = true,
val licenses: List<LicenseFileEntry> = emptyList(),
)

Some files were not shown because too many files have changed in this diff Show More