Compare commits

..

87 Commits

Author SHA1 Message Date
zaneschepke 186f063e23 chore(deps): bump deps
closes #1327
closes #1326
closes #1328
closes #1329
2026-07-03 04:56:54 -04:00
zaneschepke 2c78e9fcbd refactor: app icons and assets, change nightly launcher color
Remove old lincensee task
2026-07-03 04:43:10 -04:00
zaneschepke 8bdeff515e fix: aboutlibraries r8 crash 2026-07-03 02:56:19 -04:00
zaneschepke fa89715ede fix: disable on captive portal toggle bug
closes #1325
2026-07-03 01:35:50 -04:00
zaneschepke 4a94905893 feat: seamless roaming
Fix for broadcast receivers to goAsync to fix issues on slower devices
Add "seamless roaming" to bind wireguard UDP sockets to the active network to help resolve handshake issues on network transitions
2026-07-03 01:30:20 -04:00
zaneschepke cf184f2042 refactor: remove old license viewmodel 2026-07-01 03:15:58 -04:00
zaneschepke e0ddb8730d refactor: switch license view to aboutlibraries, add native dep licenses 2026-07-01 02:37:00 -04:00
zaneschepke c8c041b872 fix: improve ktor client to prevent download failures from github
Add user agent headers with default request and increased timeout to prevent download denial from GitHub
2026-06-30 00:35:38 -04:00
zaneschepke 7c8adb380b chore: release 5.0.7 2026-06-29 12:56:55 -04:00
zaneschepke 614f97fd14 feat: add disable tunnel on captive portal to auto tunnel
Refactor of network monitoring and auto tunnel logic to adapt to the lest strict network monitoring of active network to support system DNS.

Add disable tunnel on captive portal feature to allow auto disable of vpn while captive portal is not completed.
2026-06-29 12:08:34 -04:00
zaneschepke fbd470f5d2 fix: make network monitor less strict for network capabilities
Network monitor was too strict with capability checks, impacting our DNS resolver which needs to bind to underlying network. Capabilities have been separated out into a separate state property so we always pass the active network to connectivity state for system dns

Improve system dns by supporting DnsResolver on modern devices

#1270
2026-06-28 13:41:08 -04:00
zaneschepke 5f89b2ed31 refactor: improve mobile network detection, cleanup network monitor
#1270
2026-06-28 04:52:41 -04:00
zaneschepke 9503a3284b fix: kill switch should restore properly on tunnel up if it was killed by system or another app
closes #1313
2026-06-28 02:52:24 -04:00
zaneschepke 68c1a19bd3 fix: remove ipv6 address from lockdown causing routing issues ipv4 only tunnels 2026-06-28 02:15:45 -04:00
zaneschepke f3bb6667c3 fix: private dns to use network bind, bootstrap custom with system dns
closes #1312
closes #1311

#1303
#1270
2026-06-27 20:06:06 -04:00
zaneschepke 244a990c37 fix: ddns job logic, respect user dns setting for DDNS with default fallback if cache suspected
#1312
#1303
2026-06-27 03:57:18 -04:00
zaneschepke cbf07600b4 fix: ddns checking logic, force well known DoH to bypass system dns cache
#1303
2026-06-26 12:47:38 -04:00
zaneschepke ec8d90d13d chore: bump ktor and leakcanary
closes #1309
closes #1308
2026-06-26 03:45:38 -04:00
zaneschepke 85acca8604 fix: local network permission dialog theme and wording 2026-06-26 03:36:59 -04:00
zaneschepke 0a9773d202 chore: release 5.0.6 2026-06-25 13:10:15 -04:00
zaneschepke 3cb4480a65 fix: android 17 local devices/network permission requirement
closes #1299
2026-06-25 04:41:18 -04:00
zaneschepke a7f3255a76 refactor: remove legacy round icons 2026-06-25 02:32:03 -04:00
zaneschepke 7d7b99f448 fix: quick tile logo for samsung OneUI
#1301
2026-06-25 01:12:40 -04:00
zaneschepke 74e9e462bb fix: app shortcuts crash
closes #1302
2026-06-24 02:11:02 -04:00
zaneschepke 619e3c1cde chore: release 5.0.5 2026-06-23 12:51:01 -04:00
zaneschepke 77f8a8215b fix: improve mobile network detection for dual sim setups 2026-06-23 11:18:31 -04:00
zaneschepke 8772036dd7 build: fix localization string mismatch 2026-06-23 10:57:11 -04:00
zaneschepke 63625ccbd7 refactor: service manager to use new user start function 2026-06-23 10:46:08 -04:00
zaneschepke 9ac7ae77b3 fix: improve always on vpn reliability
#1289
2026-06-23 10:38:28 -04:00
zaneschepke e062fbb34d fix: vpnservice not cleaned up properly in certain scenarios 2026-06-23 09:40:44 -04:00
alexandervlpl 16d5586433 feat: config import via wg:// deep links (#1213)
Co-authored-by: zaneschepke <dev@zaneschepke.com>
2026-06-22 11:40:00 -04:00
zaneschepke 48a3ad64f4 chore: release 5.0.4 2026-06-20 13:13:19 -04:00
zaneschepke e5796d641d fix: auto tunnel rapid toggle bug
Improve notification efficiency
#1288
2026-06-20 12:33:25 -04:00
zaneschepke daf5eebdd2 chore: release 5.0.3 2026-06-20 01:51:46 -04:00
zaneschepke 4c725491f4 fix: import from clipboard crash on invalid data
closes #1287
2026-06-20 01:21:15 -04:00
zaneschepke 7529c11172 refactor: make bypass socket jni glue more robust against races 2026-06-19 15:16:53 -04:00
zaneschepke 83f530df42 Merge branch 'master' of github.com:wgtunnel/wgtunnel 2026-06-19 14:40:31 -04:00
zaneschepke 8083ab9526 fix: add small delay to help jni propagation of socket protector on slow devices 2026-06-19 14:40:15 -04:00
dependabot[bot] 7d1312da0f chore(deps): bump actions/checkout from 6 to 7 (#1285)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-06-19 10:52:45 -04:00
zaneschepke d4dbc43c70 refactor: improve bypass socket jni for potential races 2026-06-19 02:47:13 -04:00
zaneschepke 294f2624c7 refactor: clean up proxy jni 2026-06-19 02:21:23 -04:00
zaneschepke 0603cb2fdd fix: switch to foregrounded companion service to prevent Android Auto VPN detection
#1203
2026-06-19 00:49:47 -04:00
zaneschepke 48ddbcbb0e fix: auto tunnel not respecting tunnel tile toggle overrides
closes #1284
2026-06-18 23:51:07 -04:00
zaneschepke e6c3e3f5b3 fix: notification sync and tunnel name in title
closes #1273
closes #1275
2026-06-18 23:39:21 -04:00
dependabot[bot] 0d75699b40 chore(deps): bump gradle/actions from 3 to 6 (#1279)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2026-06-18 19:02:41 -04:00
zaneschepke 5c98aab9e0 chore: bump deps
closes #1281
closes #1280
closes #1272
2026-06-18 19:01:19 -04:00
zaneschepke a1e3489ba2 refactor: bring tunnel up after successful dns resolution
Switch from starting tunnel with dummy ip and updating peers to only bring the tunnel up once peers are resolved. We still get the benefit of protection from bringing the vpn interface up, while preventing heavy peer updates post resolve.

Fix for dns resolution when no underlying dns servers are detected, use custom dns resolution with well known servers.

#1270
2026-06-18 15:02:25 -04:00
zaneschepke bcd19b5494 ci: format, nightly version bump fix 2026-06-17 06:31:23 -04:00
zaneschepke 160a6ca84d fix: peer stats units should not be lowercase
closes #1278
2026-06-17 06:20:30 -04:00
zaneschepke aaf7ebd326 ci: fix nightly naming 2026-06-17 06:07:00 -04:00
zaneschepke b8c75a45e4 ci: fix nightly git hash detection 2026-06-17 06:04:21 -04:00
zaneschepke ac17a09e19 build: fix configuration cache issues 2026-06-17 05:42:28 -04:00
zaneschepke c51a7ee393 chore: bump koin 2026-06-17 03:55:37 -04:00
zaneschepke c534516e33 ci: add gradle validation check to pr workflow 2026-06-17 03:50:31 -04:00
zaneschepke 9c999cc62c ci: remove daemon properties
Add gradle checksum
2026-06-17 03:44:00 -04:00
zaneschepke cc3c865211 build: migrate to jvm 21 2026-06-17 03:31:28 -04:00
zaneschepke 8648a67fdc ci: fix toolchain issue 2026-06-17 02:54:25 -04:00
zaneschepke 9ee1fa69ed fix: tunnel sockets protection race
This race was especially impacting GrapheneOS devices

#1274
2026-06-17 02:32:55 -04:00
zaneschepke 379ffdcbbf feat: encrypted backup
Fixes bug where sometimes restore of backup can fail
Adds support for encrypted backups with better error messages
2026-06-16 23:09:54 -04:00
zaneschepke 6e3c1324b2 fix: snackbar and notification coordination
Refactored custom snackbar to use sonner
Added foreground/background detection so we show the proper notification or snackbar based on foregrounded state
2026-06-16 15:27:09 -04:00
zaneschepke 660bea0104 fix: allow duplicate error and tunnel event notifications 2026-06-16 01:16:20 -04:00
zaneschepke 2b8610fa8a fix: bug where split tunnel settings overwrote tunnel name comment 2026-06-16 00:41:24 -04:00
zaneschepke 944034ac74 fix: add fallback for networks without configured dns
#1270
2026-06-14 19:51:01 -04:00
zaneschepke 9f394aeffb refactor: improve vpn integration for older android version, add revoke for faster cleanup 2026-06-14 19:03:17 -04:00
zaneschepke 6c3c6891eb chore: release v5.0.2 2026-06-14 01:59:14 -04:00
zaneschepke af1848f12d fix: legacy mode triggering location pings more often than necessary due to network security check
#1062
2026-06-13 12:01:20 -04:00
zaneschepke 96cffdfa7d fix: optimize tunnel status callback 2026-06-13 00:51:41 -04:00
zaneschepke afebd975ea fix: remove automatic active tunnel to top, add scrollbars 2026-06-12 02:33:28 -04:00
zaneschepke 588a2a18bd fix: use testnet ip during bootstrap phase 2026-06-11 18:45:28 -04:00
zaneschepke 221b38a119 chore: change proxy mode wording for consistency
#1268
2026-06-10 12:19:11 -04:00
zaneschepke 0008d8b9bb chore: update docs and fastlane metadata for latest features
closes #1268
2026-06-10 12:12:16 -04:00
zaneschepke 9f85638b9a chore: fix whitespace in changelog 2026-06-08 20:47:24 -04:00
zaneschepke fe54c9cd0e chore: release v5.0.1 2026-06-08 20:19:26 -04:00
zaneschepke 554499f9de refactor: add back tunnel endpoint ip to main screen
closes #1265
2026-06-08 20:03:47 -04:00
zaneschepke 12c9b52653 fix: split tunneling bug bypass bug, service shutdown handling
#1266
#1261
2026-06-08 17:21:40 -04:00
zaneschepke 03712a6c1d fix: custom text box input race 2026-06-08 02:29:21 -04:00
zaneschepke 5f03a97fcc fix: proxy settings ui state reset when config active 2026-06-08 02:04:31 -04:00
zaneschepke 6788b05fa0 fix: globals ui state bug while tunnel active 2026-06-08 01:57:38 -04:00
zaneschepke 9494853dee fix: use underlying network for system bootstrap
#1263
2026-06-07 23:22:17 -04:00
zaneschepke 01695c3286 fix: short description too long 2026-06-07 04:56:39 -04:00
zaneschepke bb6e45ed92 fix: localized short descriptions too long 2026-06-07 04:55:52 -04:00
zaneschepke 63e257f419 ci: fix release notes length 2026-06-07 04:32:10 -04:00
zaneschepke e145cd95e1 ci: remove invalid gws locale 2026-06-07 04:18:45 -04:00
zaneschepke 7e790acbfe ci: remove invalid nod locale 2026-06-07 04:05:55 -04:00
zaneschepke 334aaa1c2b ci: fix aab finding 2026-06-07 03:49:23 -04:00
zaneschepke 6201671dd0 fix: google play release pipeline 2026-06-07 03:34:44 -04:00
zaneschepke 517d90c3bf ci: fix fastlane pipeline 2026-06-07 03:23:18 -04:00
330 changed files with 5239 additions and 2858 deletions
+3 -3
View File
@@ -70,16 +70,16 @@ jobs:
outputs: outputs:
UPLOAD_DIR_ANDROID: ${{ env.UPLOAD_DIR_ANDROID }} UPLOAD_DIR_ANDROID: ${{ env.UPLOAD_DIR_ANDROID }}
steps: steps:
- uses: actions/checkout@v6 - uses: actions/checkout@v7
with: with:
fetch-depth: 0 fetch-depth: 0
submodules: recursive submodules: recursive
- name: Set up JDK 17 - name: Set up JDK 21
uses: actions/setup-java@v5 uses: actions/setup-java@v5
with: with:
distribution: 'temurin' distribution: 'temurin'
java-version: '17' java-version: '21'
cache: gradle cache: gradle
- name: Grant execute permission for gradlew - name: Grant execute permission for gradlew
+6 -3
View File
@@ -72,15 +72,15 @@ jobs:
outputs: outputs:
UPLOAD_DIR_ANDROID: ${{ env.UPLOAD_DIR_ANDROID }} UPLOAD_DIR_ANDROID: ${{ env.UPLOAD_DIR_ANDROID }}
steps: steps:
- uses: actions/checkout@v6 - uses: actions/checkout@v7
with: with:
fetch-depth: 0 fetch-depth: 0
submodules: recursive submodules: recursive
- name: Set up JDK 17 - name: Set up JDK 21
uses: actions/setup-java@v5 uses: actions/setup-java@v5
with: with:
distribution: 'temurin' distribution: 'temurin'
java-version: '17' java-version: '21'
cache: gradle cache: gradle
- name: Grant execute permission for gradlew - name: Grant execute permission for gradlew
run: chmod +x gradlew run: chmod +x gradlew
@@ -112,6 +112,9 @@ jobs:
./gradlew :app:assemble${flavor^}Debug --stacktrace ./gradlew :app:assemble${flavor^}Debug --stacktrace
;; ;;
esac esac
env:
GITHUB_SHA: ${{ github.sha }}
GITHUB_RUN_NUMBER: ${{ github.run_number }}
- name: Get release apk path - name: Get release apk path
id: 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 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 }} has_new_commits: ${{ steps.check.outputs.new_commits }}
steps: steps:
- name: Checkout Repository - name: Checkout Repository
uses: actions/checkout@v6 uses: actions/checkout@v7
- name: Check for new commits - name: Check for new commits
id: check id: check
env: env:
@@ -43,7 +43,7 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v6 - uses: actions/checkout@v7
with: with:
submodules: recursive submodules: recursive
+11 -6
View File
@@ -1,25 +1,30 @@
name: on-pr name: on-pr
permissions: permissions:
contents: read contents: read
on: on:
workflow_dispatch: workflow_dispatch:
pull_request: pull_request:
jobs: jobs:
format_check: format_check:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v6 - uses: actions/checkout@v7
- name: Set up JDK 17
- name: Verify Gradle Wrapper
uses: gradle/actions/wrapper-validation@v6
- name: Set up JDK 21
uses: actions/setup-java@v5 uses: actions/setup-java@v5
with: with:
distribution: 'temurin' distribution: 'temurin'
java-version: '17' java-version: '21'
cache: gradle cache: gradle
- name: Grant execute permission for gradlew - name: Grant execute permission for gradlew
run: chmod +x gradlew run: chmod +x gradlew
- name: Run ktfmt - name: Run ktfmt
run: ./gradlew ktfmtCheck run: ./gradlew ktfmtCheck
+39 -35
View File
@@ -78,7 +78,7 @@ jobs:
name: publish-github name: publish-github
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v6 - uses: actions/checkout@v7
with: with:
ref: ${{ github.event_name == 'push' && github.ref || 'master' }} ref: ${{ github.event_name == 'push' && github.ref || 'master' }}
- name: Install system dependencies - name: Install system dependencies
@@ -187,57 +187,61 @@ jobs:
repository: wgtunnel/fdroid repository: wgtunnel/fdroid
event-type: fdroid-update 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: publish-play:
if: ${{ github.event_name == 'push' || inputs.track != 'none' }} if: ${{ github.event_name == 'push' || inputs.track != 'none' }}
name: Publish to Google Play name: Publish to Google Play
runs-on: ubuntu-latest 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: steps:
- uses: actions/checkout@v6 - uses: actions/checkout@v7
- name: Set up JDK 17
uses: actions/setup-java@v5
with: with:
distribution: 'temurin' ref: ${{ github.event_name == 'push' && github.ref || 'master' }}
java-version: '17'
cache: gradle
- name: Grant execute permission for gradlew - name: Download AAB artifact
run: chmod +x gradlew uses: actions/download-artifact@v8
# 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: with:
fileName: ${{ env.KEY_STORE_FILE }} name: google-play-aab
fileDir: ${{ env.KEY_STORE_LOCATION }} path: ${{ github.workspace }}/aab
encodedString: ${{ secrets.KEYSTORE }}
# create keystore path for gradle to read - name: Find exact AAB file path
- name: Create keystore path env var id: find-aab
run: | run: |
store_path=${{ env.KEY_STORE_LOCATION }}${{ env.KEY_STORE_FILE }} AAB_PATH=$(find "${{ github.workspace }}/aab" -name "*.aab" -type f | head -1)
echo "KEY_STORE_PATH=$store_path" >> $GITHUB_ENV 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
- name: Create service_account.json - name: Create service_account.json
id: createServiceAccount
run: echo '${{ secrets.SERVICE_ACCOUNT_JSON }}' > service_account.json run: echo '${{ secrets.SERVICE_ACCOUNT_JSON }}' > service_account.json
- name: Deploy with fastlane - name: Set up Ruby
uses: ruby/setup-ruby@v1 uses: ruby/setup-ruby@v1
with: with:
ruby-version: '3.2' # Not needed with a .ruby-version file ruby-version: '3.4'
bundler-cache: true bundler-cache: true
- name: Distribute app to Prod track 🚀 - name: Upload to Google Play
run: | run: |
track=${{ github.event_name == 'push' && 'production' || inputs.track }} track=${{ github.event_name == 'push' && 'production' || inputs.track }}
(cd ${{ github.workspace }} && bundle install && bundle exec fastlane $track --verbose) 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
+2 -1
View File
@@ -1,3 +1,4 @@
source "https://rubygems.org" source "https://rubygems.org"
gem "fastlane" gem "fastlane"
gem "multi_json"
+14 -17
View File
@@ -49,8 +49,8 @@ and [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/)
## About ## 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 (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. 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.
</div> </div>
<div style="text-align: left;"> <div style="text-align: left;">
@@ -67,21 +67,18 @@ WG Tunnel is an alternative Android client for WireGuard and AmneziaWG, inspired
## Features ## Features
- **Tunnel Import Methods**: Easily add tunnels using .conf files, ZIP archives, manual entry, or QR code scanning. - **Auto-Tunneling:** Automatically activate tunnels based on your device's active network details.
- **Auto-Tunneling**: Automatically activate tunnels based on Wi-Fi SSID, Ethernet connections, or mobile data networks. - **Deferred Endpoint Bootstrapping:** Safely resolves endpoints and updates peers after the tunnel is up for better reliability and leak protection on startup.
- **Split Tunneling**: Flexible support for routing specific apps or traffic through the VPN. - **Handshake Monitoring:** Real-time handshake monitoring for instant tunnel health feedback.
- **WireGuard Modes**: Full compatibility with WireGuard in both kernel and userspace implementations. - **AmneziaWG Support:** Full support for AmneziaWG 2.0, providing robust censorship protection.
- **AmneziaWG Integration**: Userspace mode for AmneziaWG, providing robust censorship evasion. - **Split Tunneling:** Flexible support for routing specific apps or traffic through the VPN.
- **Always-On VPN**: Ensures continuous protection with Android's Always-On VPN feature. - **Local Proxy Mode:** Expose WireGuard tunnels over a local SOCKS5 or HTTP proxy to browsers or firewall apps (like AdGuard).
- **Quick Controls**: Quick Settings tile and home screen shortcuts for easy VPN toggling. - **Lockdown Mode:** Advanced in-app kill switch that blocks all traffic while the tunnel is down.
- **Automation Support**: Intent-based automation for controlling tunnels. - **Quick Controls:** Quick Settings tile and home screen shortcuts for easy toggling.
- **Auto-Restore**: Seamlessly restores auto-tunneling and active tunnels after device restarts or app updates. - **Remote Control Support:** Intent-based automation for controlling tunnels and auto-tunneling from automation apps (like Tasker).
- **Proxying Options**: Built-in HTTP and SOCKS5 proxy support within tunnels. - **Dynamic DNS Handling:** Automatically detect and update endpoints on server IP changes without requiring a restart.
- **Lockdown Mode**: Custom kill switch for maximum leak prevention and security. - **IPv6 Endpoints:** Automatically upgrade to IPv6 endpoints or fall back to IPv4 based on network conditions without requiring a restart.
- **Dynamic DNS Handling**: Detects and updates DNS changes without tunnel restarts. - **Android TV Support:** Full support for nearly all features on Android TV.
- **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 ## Building
+53 -47
View File
@@ -1,6 +1,5 @@
import com.android.build.api.dsl.ApplicationExtension import com.android.build.api.dsl.ApplicationExtension
import com.android.build.api.variant.FilterConfiguration import com.android.build.api.variant.FilterConfiguration
import org.jetbrains.kotlin.gradle.dsl.JvmTarget
plugins { plugins {
alias(libs.plugins.android.application) alias(libs.plugins.android.application)
@@ -9,6 +8,7 @@ plugins {
alias(libs.plugins.compose.compiler) alias(libs.plugins.compose.compiler)
alias(libs.plugins.grgit) alias(libs.plugins.grgit)
alias(libs.plugins.licensee) alias(libs.plugins.licensee)
alias(libs.plugins.aboutlibraries)
} }
ksp { ksp {
@@ -18,15 +18,14 @@ ksp {
licensee { licensee {
allowedLicenses().forEach { allow(it) } allowedLicenses().forEach { allow(it) }
allowedLicenseUrls().forEach { allowUrl(it) } allowedLicenseUrls().forEach { allowUrl(it) }
// foss, but missing licenses allowDependency("com.github.T8RIN.QuickieExtended", "quickie-foss", "1.18.1") {
ignoreDependencies("com.github.T8RIN.QuickieExtended") because("FOSS library, but JitPack doesn't publish license metadata")
ignoreDependencies("com.github.topjohnwu.libsu") allow("Apache-2.0")
} }
kotlin { allowDependency("com.github.topjohnwu.libsu", "core", "6.0.0") {
compilerOptions { because("FOSS library, but JitPack doesn't publish license metadata")
jvmTarget = JvmTarget.JVM_17 allow("Apache-2.0")
freeCompilerArgs = listOf("-XXLanguage:+PropertyParamAnnotationDefaultTargetMode")
} }
} }
@@ -46,10 +45,11 @@ configure<ApplicationExtension> {
splits { splits {
abi { abi {
isEnable = !project.hasProperty("noSplits") val noSplits = providers.gradleProperty("noSplits").isPresent
isEnable = !noSplits
reset() reset()
include("armeabi-v7a", "arm64-v8a") include("armeabi-v7a", "arm64-v8a")
isUniversalApk = !project.hasProperty("noSplits") isUniversalApk = !noSplits
} }
} }
@@ -57,14 +57,17 @@ configure<ApplicationExtension> {
applicationId = Constants.APP_ID applicationId = Constants.APP_ID
minSdk = Constants.MIN_SDK minSdk = Constants.MIN_SDK
targetSdk = Constants.TARGET_SDK targetSdk = Constants.TARGET_SDK
versionCode = computeVersionCode() versionCode = Constants.VERSION_CODE
versionName = computeVersionName() versionName = Constants.VERSION_NAME
experimentalProperties["android.experimental.disableGitVersion"] = true
sourceSets { sourceSets {
getByName("debug").assets.directories += "$projectDir/schemas" getByName("debug").assets.directories += "$projectDir/schemas"
} }
val languagesArray = buildLanguagesArray(languageList()) val languagesProvider = project.languageListProvider()
val languagesArray = buildLanguagesArray(languagesProvider.get())
buildConfigField("String[]", "LANGUAGES", "new String[]{ $languagesArray }") buildConfigField("String[]", "LANGUAGES", "new String[]{ $languagesArray }")
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
@@ -98,21 +101,24 @@ configure<ApplicationExtension> {
"proguard-rules.pro", "proguard-rules.pro",
) )
signingConfig = signingConfigs.getByName(Constants.RELEASE) signingConfig = signingConfigs.getByName(Constants.RELEASE)
resValue("string", "provider", "\"${Constants.APP_NAME}.provider\"") manifestPlaceholders["providerAuthority"] = "${Constants.APP_NAME}.provider"
buildConfigField("String", "FILE_PROVIDER_AUTHORITY", "\"${Constants.APP_NAME}.provider\"")
} }
debug { debug {
applicationIdSuffix = ".debug" applicationIdSuffix = ".debug"
resValue("string", "app_name", "WG Tunnel Debug") resValue("string", "app_name", "WG Tunnel Debug")
isDebuggable = true isDebuggable = true
resValue("string", "provider", "\"${Constants.APP_NAME}.provider.debug\"") manifestPlaceholders["providerAuthority"] = "${Constants.APP_NAME}.provider.debug"
buildConfigField("String", "FILE_PROVIDER_AUTHORITY", "\"${Constants.APP_NAME}.provider.debug\"")
} }
create(Constants.NIGHTLY) { create(Constants.NIGHTLY) {
initWith(buildTypes.getByName(Constants.RELEASE)) initWith(buildTypes.getByName(Constants.RELEASE))
applicationIdSuffix = ".nightly" applicationIdSuffix = ".nightly"
resValue("string", "app_name", "WG Tunnel Nightly") resValue("string", "app_name", "WG Tunnel Nightly")
resValue("string", "provider", "\"${Constants.APP_NAME}.provider.nightly\"") manifestPlaceholders["providerAuthority"] = "${Constants.APP_NAME}.provider.nightly"
buildConfigField("String", "FILE_PROVIDER_AUTHORITY", "\"${Constants.APP_NAME}.provider.nightly\"")
} }
} }
@@ -134,8 +140,6 @@ configure<ApplicationExtension> {
compileOptions { compileOptions {
isCoreLibraryDesugaringEnabled = true isCoreLibraryDesugaringEnabled = true
sourceCompatibility = JavaVersion.VERSION_17
targetCompatibility = JavaVersion.VERSION_17
} }
buildFeatures { buildFeatures {
@@ -148,31 +152,42 @@ configure<ApplicationExtension> {
androidComponents { androidComponents {
onVariants { variant -> onVariants { variant ->
val isNightly = project.isNightlyBuild()
val abiNameMap = if (isNightly) {
mapOf( variant.outputs.forEach { output ->
"armeabi-v7a" to "armv7",
"arm64-v8a" to "arm64", output.versionCode.set(
"x86" to "x86", output.versionCode.get() + project.getVersionCodeIncrement()
"x86_64" to "x64", )
)
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",
)
variant.outputs.forEach { output -> variant.outputs.forEach { output ->
val abi = output.filters.find { it.filterType == FilterConfiguration.FilterType.ABI }?.identifier val abi = output.filters.find { it.filterType == FilterConfiguration.FilterType.ABI }?.identifier
val flavorName = variant.productFlavors.joinToString("-") { it.second } val flavorName = variant.productFlavors.joinToString("-") { it.second }
val versionName = output.versionName.get() val versionName = output.versionName.get()
val baseFileName = "${Constants.APP_NAME}-${flavorName}-v${versionName}" val baseFileName = "${Constants.APP_NAME}-${flavorName}-v${versionName}"
val outputFileName = val outputFileName = if (!abi.isNullOrEmpty()) {
if (!abi.isNullOrEmpty()) { val shortAbiName = abiNameMap.getOrDefault(abi, abi)
val shortAbiName = abiNameMap.getOrDefault(abi, abi) "${baseFileName}-${shortAbiName}.apk"
"${baseFileName}-${shortAbiName}.apk" } else {
} else { "${baseFileName}.apk"
"${baseFileName}.apk" }
}
output.outputFileName.set(outputFileName) output.outputFileName.set(outputFileName)
} }
@@ -225,6 +240,8 @@ dependencies {
// UI utilities // UI utilities
implementation(libs.bundles.ui.utilities) implementation(libs.bundles.ui.utilities)
implementation(libs.lottie.compose) implementation(libs.lottie.compose)
implementation(libs.sonner)
implementation(libs.aboutlibraries.compose)
// Misc utilities // Misc utilities
implementation(libs.bundles.misc.utilities) implementation(libs.bundles.misc.utilities)
@@ -265,17 +282,6 @@ dependencies {
implementation(libs.koin.worker) 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 // https://gist.github.com/obfusk/61046e09cee352ae6dd109911534b12e#fix-proposed-by-linsui-disable-baseline-profiles
tasks.configureEach { tasks.configureEach {
if (name.contains("ArtProfile")) { if (name.contains("ArtProfile")) {
@@ -0,0 +1,513 @@
{
"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')"
]
}
}
@@ -0,0 +1,520 @@
{
"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')"
]
}
}
+13 -7
View File
@@ -53,7 +53,6 @@
android:fullBackupContent="@xml/backup_rules" android:fullBackupContent="@xml/backup_rules"
android:icon="@mipmap/ic_launcher" android:icon="@mipmap/ic_launcher"
android:label="@string/app_name" android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true" android:supportsRtl="true"
android:theme="@style/Theme.App.Start" android:theme="@style/Theme.App.Start"
tools:targetApi="tiramisu"> tools:targetApi="tiramisu">
@@ -74,6 +73,13 @@
<category android:name="android.intent.category.DEFAULT" /> <category android:name="android.intent.category.DEFAULT" />
</intent-filter> </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> <intent-filter>
<action android:name="android.intent.action.MAIN" /> <action android:name="android.intent.action.MAIN" />
<action android:name="android.intent.action.SHOW_APP_INFO" /> <action android:name="android.intent.action.SHOW_APP_INFO" />
@@ -150,7 +156,7 @@
<provider <provider
android:name="androidx.core.content.FileProvider" android:name="androidx.core.content.FileProvider"
android:authorities="@string/provider" android:authorities="${providerAuthority}"
android:exported="false" android:exported="false"
android:grantUriPermissions="true"> android:grantUriPermissions="true">
<meta-data <meta-data
@@ -169,9 +175,9 @@
tools:node="remove" /> tools:node="remove" />
</provider> </provider>
<service <service
android:name=".core.service.tile.TunnelControlTile" android:name=".service.tile.TunnelControlTile"
android:exported="true" android:exported="true"
android:icon="@drawable/ic_notification" android:icon="@drawable/qs_logo"
android:label="@string/tunnel_control" android:label="@string/tunnel_control"
android:permission="android.permission.BIND_QUICK_SETTINGS_TILE"> android:permission="android.permission.BIND_QUICK_SETTINGS_TILE">
<meta-data <meta-data
@@ -186,9 +192,9 @@
</intent-filter> </intent-filter>
</service> </service>
<service <service
android:name=".core.service.tile.AutoTunnelControlTile" android:name=".service.tile.AutoTunnelControlTile"
android:exported="true" android:exported="true"
android:icon="@drawable/ic_notification" android:icon="@drawable/qs_logo"
android:label="@string/auto_tunnel" android:label="@string/auto_tunnel"
android:permission="android.permission.BIND_QUICK_SETTINGS_TILE"> android:permission="android.permission.BIND_QUICK_SETTINGS_TILE">
<meta-data <meta-data
@@ -203,7 +209,7 @@
</intent-filter> </intent-filter>
</service> </service>
<service <service
android:name=".core.service.autotunnel.AutoTunnelService" android:name=".service.autotunnel.AutoTunnelService"
android:enabled="true" android:enabled="true"
android:exported="false" android:exported="false"
android:foregroundServiceType="specialUse" android:foregroundServiceType="specialUse"
View File
@@ -1,12 +1,14 @@
package com.zaneschepke.wireguardautotunnel package com.zaneschepke.wireguardautotunnel
import ProxySettingsScreen import ProxySettingsScreen
import android.Manifest
import android.content.Intent import android.content.Intent
import android.graphics.Color import android.graphics.Color
import android.net.Uri import android.net.Uri
import android.net.VpnService import android.net.VpnService
import android.os.Build import android.os.Build
import android.os.Bundle import android.os.Bundle
import android.provider.Settings
import android.view.WindowManager import android.view.WindowManager
import androidx.activity.SystemBarStyle import androidx.activity.SystemBarStyle
import androidx.activity.compose.rememberLauncherForActivityResult import androidx.activity.compose.rememberLauncherForActivityResult
@@ -19,17 +21,29 @@ import androidx.compose.animation.fadeOut
import androidx.compose.animation.slideInHorizontally import androidx.compose.animation.slideInHorizontally
import androidx.compose.animation.slideOutHorizontally import androidx.compose.animation.slideOutHorizontally
import androidx.compose.animation.togetherWith import androidx.compose.animation.togetherWith
import androidx.compose.foundation.BorderStroke
import androidx.compose.foundation.background import androidx.compose.foundation.background
import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.PaddingValues
import androidx.compose.foundation.layout.consumeWindowInsets import androidx.compose.foundation.layout.consumeWindowInsets
import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.fillMaxWidth
import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.layout.wrapContentHeight 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.material3.ExperimentalMaterial3Api import androidx.compose.material3.ExperimentalMaterial3Api
import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Scaffold import androidx.compose.material3.Scaffold
import androidx.compose.material3.Text
import androidx.compose.material3.TextButton
import androidx.compose.material3.surfaceColorAtElevation import androidx.compose.material3.surfaceColorAtElevation
import androidx.compose.runtime.CompositionLocalProvider import androidx.compose.runtime.CompositionLocalProvider
import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.LaunchedEffect
@@ -38,20 +52,19 @@ import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberCoroutineScope import androidx.compose.runtime.rememberCoroutineScope
import androidx.compose.runtime.saveable.rememberSaveable
import androidx.compose.runtime.setValue import androidx.compose.runtime.setValue
import androidx.compose.ui.Alignment import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Brush
import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.stringResource
import androidx.compose.ui.text.LinkAnnotation import androidx.compose.ui.text.font.FontWeight
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.text.intl.Locale
import androidx.compose.ui.text.style.TextDecoration import androidx.compose.ui.unit.IntOffset
import androidx.compose.ui.text.withLink
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
import androidx.compose.ui.zIndex import androidx.compose.ui.zIndex
import androidx.core.app.ActivityCompat
import androidx.core.splashscreen.SplashScreen.Companion.installSplashScreen import androidx.core.splashscreen.SplashScreen.Companion.installSplashScreen
import androidx.lifecycle.lifecycleScope import androidx.lifecycle.lifecycleScope
import androidx.lifecycle.viewmodel.navigation3.rememberViewModelStoreNavEntryDecorator import androidx.lifecycle.viewmodel.navigation3.rememberViewModelStoreNavEntryDecorator
@@ -59,6 +72,10 @@ import androidx.navigation3.runtime.entryProvider
import androidx.navigation3.runtime.rememberNavBackStack import androidx.navigation3.runtime.rememberNavBackStack
import androidx.navigation3.runtime.rememberSaveableStateHolderNavEntryDecorator import androidx.navigation3.runtime.rememberSaveableStateHolderNavEntryDecorator
import androidx.navigation3.ui.NavDisplay 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.networkmonitor.NetworkMonitor
import com.zaneschepke.wireguardautotunnel.data.AppDatabase import com.zaneschepke.wireguardautotunnel.data.AppDatabase
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
@@ -69,11 +86,9 @@ import com.zaneschepke.wireguardautotunnel.domain.sideeffect.GlobalSideEffect
import com.zaneschepke.wireguardautotunnel.ui.LocalIsAndroidTV import com.zaneschepke.wireguardautotunnel.ui.LocalIsAndroidTV
import com.zaneschepke.wireguardautotunnel.ui.LocalNavController import com.zaneschepke.wireguardautotunnel.ui.LocalNavController
import com.zaneschepke.wireguardautotunnel.ui.common.banner.AppAlertBanner 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.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.Route
import com.zaneschepke.wireguardautotunnel.ui.navigation.SecureRoute import com.zaneschepke.wireguardautotunnel.ui.navigation.SecureRoute
import com.zaneschepke.wireguardautotunnel.ui.navigation.Tab import com.zaneschepke.wireguardautotunnel.ui.navigation.Tab
@@ -110,19 +125,29 @@ 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.sort.SortScreen
import com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.splittunnel.SplitTunnelScreen import com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.splittunnel.SplitTunnelScreen
import com.zaneschepke.wireguardautotunnel.ui.theme.AlertRed 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.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.ui.theme.WireguardAutoTunnelTheme
import com.zaneschepke.wireguardautotunnel.util.FileUtils
import com.zaneschepke.wireguardautotunnel.util.LocaleUtil 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.installApk
import com.zaneschepke.wireguardautotunnel.util.extensions.isRunningOnTv import com.zaneschepke.wireguardautotunnel.util.extensions.isRunningOnTv
import com.zaneschepke.wireguardautotunnel.util.extensions.openWebUrl import com.zaneschepke.wireguardautotunnel.util.extensions.openWebUrl
import com.zaneschepke.wireguardautotunnel.util.extensions.restartApp import com.zaneschepke.wireguardautotunnel.util.extensions.restartApp
import com.zaneschepke.wireguardautotunnel.util.extensions.showToast import com.zaneschepke.wireguardautotunnel.util.permission.LocalNetworkPermissionHelper
import com.zaneschepke.wireguardautotunnel.viewmodel.ConfigEditViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.ConfigEditViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.SplitTunnelViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.SplitTunnelViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.TunnelViewModel 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 de.raphaelebner.roomdatabasebackup.core.RoomBackup
import kotlin.time.Duration.Companion.milliseconds
import kotlinx.coroutines.awaitCancellation import kotlinx.coroutines.awaitCancellation
import kotlinx.coroutines.delay import kotlinx.coroutines.delay
import kotlinx.coroutines.flow.collectLatest import kotlinx.coroutines.flow.collectLatest
@@ -132,6 +157,7 @@ import org.koin.androidx.compose.koinViewModel
import org.koin.androidx.viewmodel.ext.android.viewModel import org.koin.androidx.viewmodel.ext.android.viewModel
import org.koin.core.parameter.parametersOf import org.koin.core.parameter.parametersOf
import org.orbitmvi.orbit.compose.collectAsState import org.orbitmvi.orbit.compose.collectAsState
import timber.log.Timber
import xyz.teamgravity.pin_lock_compose.PinManager import xyz.teamgravity.pin_lock_compose.PinManager
class MainActivity : AppCompatActivity() { class MainActivity : AppCompatActivity() {
@@ -155,9 +181,10 @@ class MainActivity : AppCompatActivity() {
} }
super.onCreate(savedInstanceState) super.onCreate(savedInstanceState)
handleIncomingIntent(intent) roomBackup = RoomBackup(this).database(appDatabase).enableLogDebug(true).maxFileCount(5)
roomBackup = RoomBackup(this) handleConfigFileIntent(intent)
handleWgDeepLinkIntent(intent)
installSplashScreen().apply { installSplashScreen().apply {
setKeepOnScreenCondition { !viewModel.container.stateFlow.value.isAppLoaded } setKeepOnScreenCondition { !viewModel.container.stateFlow.value.isAppLoaded }
@@ -175,12 +202,54 @@ class MainActivity : AppCompatActivity() {
} }
} }
val snackbarState = rememberCustomSnackbarState() val toaster = rememberToasterState()
var showVpnPermissionDialog by remember { mutableStateOf(false) } var showVpnPermissionDialog by remember { mutableStateOf(false) }
var vpnPermissionDenied by remember { mutableStateOf(false) } var vpnPermissionDenied by remember { mutableStateOf(false) }
var requestingTunnelMode by remember { var requestingTunnelMode by remember {
mutableStateOf<Pair<TunnelMode?, TunnelConfig?>>(Pair(null, null)) 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 { val startingStack = buildList {
add(Route.Tunnels) add(Route.Tunnels)
@@ -232,22 +301,18 @@ class MainActivity : AppCompatActivity() {
} }
is GlobalSideEffect.Snackbar -> { is GlobalSideEffect.Snackbar -> {
scope.launch { when (sideEffect.type) {
snackbarState.showSnackbar( ToastType.Warning,
SnackbarInfo( ToastType.Error -> toaster.dismissAll()
message = else -> Unit
buildAnnotatedString {
append(sideEffect.message.asString(context))
},
type = sideEffect.type ?: SnackbarType.INFO,
durationMs = sideEffect.durationMs ?: 4000L,
)
)
} }
}
is GlobalSideEffect.Toast -> toaster.show(
scope.launch { context.showToast(sideEffect.message.asString(context)) } message = sideEffect.message.asString(context),
type = sideEffect.type,
duration = (sideEffect.durationMs ?: 4000L).milliseconds,
)
}
is GlobalSideEffect.LaunchUrl -> context.openWebUrl(sideEffect.url) is GlobalSideEffect.LaunchUrl -> context.openWebUrl(sideEffect.url)
is GlobalSideEffect.InstallApk -> context.installApk(sideEffect.apk) is GlobalSideEffect.InstallApk -> context.installApk(sideEffect.apk)
@@ -275,49 +340,58 @@ class MainActivity : AppCompatActivity() {
}, },
) )
val annotatedMessage = buildAnnotatedString { if (showLocalNetworkRationale) {
append(context.getString(R.string.donation_prompt_prefix)) LocalNetworkPermissionDialog(
append(" ") onDismiss = {
withLink( showLocalNetworkRationale = false
LinkAnnotation.Clickable( toaster.show(
tag = context.getString(R.string.support), message =
styles = context.getString(R.string.local_network_permission_denied),
TextLinkStyles( type = ToastType.Warning,
style = duration = 6000.milliseconds,
SpanStyle( )
textDecoration = TextDecoration.Underline, },
color = MaterialTheme.colorScheme.primary, onAttest = {
), showLocalNetworkRationale = false
focusedStyle =
SpanStyle( localNetworkPermissionLauncher.launch(
textDecoration = TextDecoration.Underline, Manifest.permission.ACCESS_LOCAL_NETWORK
color = MaterialTheme.colorScheme.primary, )
background = },
MaterialTheme.colorScheme.primary.copy( )
alpha = 0.2f }
),
), uiState.pendingWgImportUrl?.let { url ->
), val host = Uri.parse(url).host ?: url
) { InfoDialog(
snackbarState.dismissCurrent() onDismiss = { viewModel.dismissWgImport() },
navController.push(Route.Donate) onAttest = { viewModel.importFromUrl(url) },
} title = stringResource(R.string.add_from_url),
) { body = { Text(stringResource(R.string.wg_url_confirm_message, host)) },
append(context.getString(R.string.donation_prompt_link)) confirmText = stringResource(R.string.okay),
} )
append(" ")
append(context.getString(R.string.donation_prompt_suffix))
} }
LaunchedEffect(Unit) { LaunchedEffect(Unit) {
if (uiState.shouldShowDonationSnackbar && !uiState.alreadyDonated) { if (uiState.shouldShowDonationSnackbar && !uiState.alreadyDonated) {
viewModel.setShouldShowDonationSnackbar(false) viewModel.setShouldShowDonationSnackbar(false)
snackbarState.showSnackbar( toaster.show(
SnackbarInfo( message =
message = annotatedMessage, context.getString(R.string.donation_prompt_prefix) +
type = SnackbarType.THANK_YOU, " " +
durationMs = 30_000L, 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)
},
),
) )
} }
} }
@@ -378,25 +452,6 @@ class MainActivity : AppCompatActivity() {
) )
} }
Scaffold( 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) }, topBar = { DynamicTopAppBar(navState) },
bottomBar = { bottomBar = {
if (navState.showBottomItems) { if (navState.showBottomItems) {
@@ -548,6 +603,70 @@ 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,
)
} }
} }
} }
@@ -555,55 +674,14 @@ class MainActivity : AppCompatActivity() {
} }
} }
fun performBackup() = lifecycleScope.launch { private fun handleWgDeepLinkIntent(intent: Intent) {
roomBackup if (intent.action == Intent.ACTION_VIEW) {
.database(appDatabase) val uri = intent.data ?: return
.backupLocation(RoomBackup.BACKUP_FILE_LOCATION_CUSTOM_DIALOG) if (uri.scheme == "wg") {
.enableLogDebug(true) val httpsUrl = uri.toString().replaceFirst("wg://", "https://")
.maxFileCount(5) viewModel.promptWgImport(httpsUrl)
.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() { override fun onResume() {
@@ -611,21 +689,105 @@ class MainActivity : AppCompatActivity() {
networkMonitor.checkPermissionsAndUpdateState() 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) { override fun onNewIntent(intent: Intent) {
super.onNewIntent(intent) super.onNewIntent(intent)
setIntent(intent) setIntent(intent)
handleIncomingIntent(intent) handleConfigFileIntent(intent)
handleWgDeepLinkIntent(intent)
} }
private fun handleIncomingIntent(intent: Intent?) { private fun handleConfigFileIntent(intent: Intent?) {
intent ?: return intent ?: return
when (intent.action) { when (intent.action) {
Intent.ACTION_VIEW, Intent.ACTION_VIEW,
Intent.ACTION_EDIT, Intent.ACTION_EDIT,
Intent.ACTION_SEND -> { Intent.ACTION_SEND -> {
val uri: Uri? = intent.data val uri: Uri? = intent.data ?: return
uri?.let { viewModel.importFromUri(it) } 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)
} }
} }
} }
@@ -6,11 +6,8 @@ import com.zaneschepke.tunnel.backend.Backend
import com.zaneschepke.tunnel.di.tunnelModule import com.zaneschepke.tunnel.di.tunnelModule
import com.zaneschepke.tunnel.service.VpnService import com.zaneschepke.tunnel.service.VpnService
import com.zaneschepke.wireguardautotunnel.core.event.TunnelEventDispatcher 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.AppBoostrapCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.TunnelCoordinator 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.core.tunnel.TunnelProvider
import com.zaneschepke.wireguardautotunnel.di.Dispatcher import com.zaneschepke.wireguardautotunnel.di.Dispatcher
import com.zaneschepke.wireguardautotunnel.di.Scope import com.zaneschepke.wireguardautotunnel.di.Scope
@@ -21,6 +18,9 @@ import com.zaneschepke.wireguardautotunnel.di.dispatchersModule
import com.zaneschepke.wireguardautotunnel.di.networkModule import com.zaneschepke.wireguardautotunnel.di.networkModule
import com.zaneschepke.wireguardautotunnel.di.tunnelBackendProviderModule import com.zaneschepke.wireguardautotunnel.di.tunnelBackendProviderModule
import com.zaneschepke.wireguardautotunnel.di.workerModule 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 com.zaneschepke.wireguardautotunnel.util.ReleaseTree
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
@@ -51,6 +51,13 @@ class WireGuardAutoTunnel : Application(), KoinComponent {
private val backend: Backend by inject() private val backend: Backend by inject()
private val alwaysOnCallback =
object : VpnService.AlwaysOnCallback {
override fun alwaysOnTriggered() {
applicationScope.launch { tunnelCoordinator.startDefault() }
}
}
@OptIn(KoinViewModelScopeApi::class) @OptIn(KoinViewModelScopeApi::class)
override fun onCreate() { override fun onCreate() {
super.onCreate() super.onCreate()
@@ -71,11 +78,10 @@ class WireGuardAutoTunnel : Application(), KoinComponent {
lazyModules(networkModule) lazyModules(networkModule)
} }
instance = this instance = this
notificationService.createAllChannels() notificationService.createAllChannels()
// Sync tiles syncTiles()
AutoTunnelTileRefresher.refresh(this)
TunnelTileRefresher.refresh(this)
if (BuildConfig.DEBUG) { if (BuildConfig.DEBUG) {
Timber.plant(Timber.DebugTree()) Timber.plant(Timber.DebugTree())
@@ -87,13 +93,7 @@ class WireGuardAutoTunnel : Application(), KoinComponent {
Timber.plant(ReleaseTree()) Timber.plant(ReleaseTree())
} }
backend.setAlwaysOnCallback( backend.setAlwaysOnCallback(alwaysOnCallback)
object : VpnService.AlwaysOnCallback {
override fun alwaysOnTriggered() {
applicationScope.launch { tunnelCoordinator.startDefault() }
}
}
)
val dispatcher = get<TunnelEventDispatcher>() val dispatcher = get<TunnelEventDispatcher>()
val coordinator = get<TunnelCoordinator>() val coordinator = get<TunnelCoordinator>()
@@ -111,6 +111,11 @@ class WireGuardAutoTunnel : Application(), KoinComponent {
applicationScope.launch(ioDispatcher) { boostrapCoordinator.bootstrap() } applicationScope.launch(ioDispatcher) { boostrapCoordinator.bootstrap() }
} }
private fun syncTiles() {
AutoTunnelTileRefresher.refresh(this)
TunnelTileRefresher.refresh(this)
}
companion object { companion object {
lateinit var instance: WireGuardAutoTunnel lateinit var instance: WireGuardAutoTunnel
private set private set
@@ -3,11 +3,11 @@ package com.zaneschepke.wireguardautotunnel.core.broadcast
import android.content.BroadcastReceiver import android.content.BroadcastReceiver
import android.content.Context import android.content.Context
import android.content.Intent import android.content.Intent
import com.zaneschepke.wireguardautotunnel.core.notification.NotificationService
import com.zaneschepke.wireguardautotunnel.core.orchestration.AutoTunnelCoordinator import com.zaneschepke.wireguardautotunnel.core.orchestration.AutoTunnelCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.TunnelCoordinator import com.zaneschepke.wireguardautotunnel.core.orchestration.TunnelCoordinator
import com.zaneschepke.wireguardautotunnel.di.Scope import com.zaneschepke.wireguardautotunnel.di.Scope
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.notification.NotificationService
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import org.koin.core.component.KoinComponent import org.koin.core.component.KoinComponent
@@ -48,47 +48,51 @@ class RemoteControlReceiver : BroadcastReceiver(), KoinComponent {
} }
override fun onReceive(context: Context, intent: Intent) { override fun onReceive(context: Context, intent: Intent) {
val pendingResult = goAsync()
val action = intent.action ?: return
val appAction = Action.fromAction(action) ?: return
applicationScope.launch { applicationScope.launch {
val settings = settingsRepository.getGeneralSettings() try {
val action = intent.action ?: return@launch
val appAction = Action.fromAction(action) ?: return@launch
if (!settings.isRemoteControlEnabled) return@launch val settings = settingsRepository.getGeneralSettings()
if (!validateKey(settings, intent)) return@launch if (!settings.isRemoteControlEnabled) return@launch
when (appAction) { if (!validateKey(settings, intent)) return@launch
Action.START_TUNNEL -> {
val tunnel =
resolveTunnel(intent)
?: tunnelsRepository.getDefaultTunnel()
?: return@launch
tunnelCoordinator.startTunnel(tunnel) when (appAction) {
} Action.START_TUNNEL -> {
val tunnel =
resolveTunnel(intent)
?: tunnelsRepository.getDefaultTunnel()
?: return@launch
Action.STOP_TUNNEL -> { tunnelCoordinator.startTunnel(tunnel)
val tunnelName = intent.getStringExtra(EXTRA_TUN_NAME)
if (tunnelName == null) {
tunnelCoordinator.stopActiveTunnels()
return@launch
} }
val tunnel = tunnelsRepository.findByTunnelName(tunnelName) ?: return@launch Action.STOP_TUNNEL -> {
val tunnelName = intent.getStringExtra(EXTRA_TUN_NAME)
tunnelCoordinator.stopTunnel(tunnel.id) if (tunnelName == null) {
} tunnelCoordinator.stopActiveTunnels()
return@launch
}
Action.START_AUTO_TUNNEL -> { val tunnel = tunnelsRepository.findByTunnelName(tunnelName) ?: return@launch
autoTunnelCoordinator.enable()
}
Action.STOP_AUTO_TUNNEL -> { tunnelCoordinator.stopTunnel(tunnel.id)
autoTunnelCoordinator.disable() }
Action.START_AUTO_TUNNEL -> {
autoTunnelCoordinator.enable()
}
Action.STOP_AUTO_TUNNEL -> {
autoTunnelCoordinator.disable()
}
} }
} finally {
pendingResult.finish()
} }
} }
} }
@@ -27,19 +27,31 @@ class RestartReceiver : BroadcastReceiver(), KoinComponent {
override fun onReceive(context: Context, intent: Intent) { override fun onReceive(context: Context, intent: Intent) {
Timber.d("RestartReceiver triggered with action: ${intent.action}") Timber.d("RestartReceiver triggered with action: ${intent.action}")
val pendingResult = goAsync()
applicationScope.launch { applicationScope.launch {
when (intent.action) { try {
Intent.ACTION_BOOT_COMPLETED, when (intent.action) {
"android.intent.action.QUICKBOOT_POWERON", Intent.ACTION_BOOT_COMPLETED,
"com.htc.intent.action.QUICKBOOT_POWERON" -> { "android.intent.action.QUICKBOOT_POWERON",
startupCoordinator.applyStartupPolicy() "com.htc.intent.action.QUICKBOOT_POWERON" -> {
} startupCoordinator.applyStartupPolicy()
Intent.ACTION_MY_PACKAGE_REPLACED -> { }
Timber.i("Restoring state on package upgrade")
startupCoordinator.applyStartupPolicy() Intent.ACTION_MY_PACKAGE_REPLACED -> {
logReader.deleteAndClearLogs() Timber.i("Restoring state on package upgrade")
appStateRepository.setShouldShowDonationSnackbar(true) startupCoordinator.applyStartupPolicy()
logReader.deleteAndClearLogs()
appStateRepository.setShouldShowDonationSnackbar(true)
}
else -> {
Timber.w("Unhandled action in RestartReceiver: ${intent.action}")
}
} }
} finally {
pendingResult.finish()
} }
} }
} }
@@ -1,17 +1,22 @@
package com.zaneschepke.wireguardautotunnel.core.event package com.zaneschepke.wireguardautotunnel.core.event
import android.content.Context import android.content.Context
import com.dokar.sonner.ToastType
import com.zaneschepke.tunnel.event.TunnelEvent import com.zaneschepke.tunnel.event.TunnelEvent
import com.zaneschepke.tunnel.model.BackendMode import com.zaneschepke.tunnel.model.BackendMode
import com.zaneschepke.tunnel.state.BackendStatus import com.zaneschepke.tunnel.state.BackendStatus
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.core.notification.TunnelNotificationLine import com.zaneschepke.wireguardautotunnel.domain.repository.GlobalEffectRepository
import com.zaneschepke.wireguardautotunnel.core.notification.TunnelNotificationService
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository 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.ui.state.DisplayTunnelState
import com.zaneschepke.wireguardautotunnel.util.StringValue
import kotlin.time.Duration.Companion.milliseconds
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.FlowPreview import kotlinx.coroutines.FlowPreview
import kotlinx.coroutines.delay
import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.StateFlow import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.combine import kotlinx.coroutines.flow.combine
@@ -20,13 +25,14 @@ import kotlinx.coroutines.flow.distinctUntilChanged
import kotlinx.coroutines.flow.launchIn import kotlinx.coroutines.flow.launchIn
import kotlinx.coroutines.flow.map import kotlinx.coroutines.flow.map
import kotlinx.coroutines.flow.onEach import kotlinx.coroutines.flow.onEach
import kotlinx.coroutines.flow.onStart import kotlinx.coroutines.launch
import kotlin.time.Duration.Companion.milliseconds
class TunnelEventDispatcher( class TunnelEventDispatcher(
private val notificationManager: TunnelNotificationService, private val notificationManager: TunnelNotificationService,
private val tunnelRepository: TunnelRepository, private val tunnelRepository: TunnelRepository,
private val context: Context, private val context: Context,
private val appVisibilityObserver: AppVisibilityObserver,
private val globalEffectRepository: GlobalEffectRepository,
) { ) {
@OptIn(FlowPreview::class) @OptIn(FlowPreview::class)
@@ -38,54 +44,174 @@ class TunnelEventDispatcher(
tunnelDisplayStates: StateFlow<Map<Int, DisplayTunnelState>>, tunnelDisplayStates: StateFlow<Map<Int, DisplayTunnelState>>,
) { ) {
// informational events // Informational events from tunnel backend
providerEvents providerEvents
.distinctUntilChanged()
.onEach { event -> .onEach { event ->
when (event) { when (event) {
is TunnelEvent.FallbackToIpv4 -> { is TunnelEvent.FallbackToIpv4 -> {
val name = getTunnelName(event.tunnelId) val name = getTunnelName(event.tunnelId)
notificationManager.showIpv4Fallback(name) 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) },
)
} }
is TunnelEvent.RecoveredToIpv6 -> { is TunnelEvent.RecoveredToIpv6 -> {
val name = getTunnelName(event.tunnelId) val name = getTunnelName(event.tunnelId)
notificationManager.showIpv6Recovery(name) 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) },
)
} }
is TunnelEvent.DynamicDnsUpdate -> { is TunnelEvent.DynamicDnsUpdate -> {
val name = getTunnelName(event.tunnelId) val name = getTunnelName(event.tunnelId)
notificationManager.showDynamicDnsUpdate(name) 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) },
)
} }
is TunnelEvent.NoRootShellAccess -> { is TunnelEvent.NoRootShellAccess -> {
notificationManager.showRootShellAccess() showOrNotify(
scope = scope,
foregroundAction = {
globalEffectRepository.post(
GlobalSideEffect.Snackbar(
message =
StringValue.DynamicString(
context.getString(R.string.error_root_denied)
),
type = ToastType.Error,
)
)
},
backgroundAction = { notificationManager.showRootShellAccess() },
)
} }
} }
} }
.launchIn(scope) .launchIn(scope)
// errors from the coordinator // Errors from our tunnel coordinator
coordinatorErrors coordinatorErrors
.distinctUntilChanged()
.onEach { error -> .onEach { error ->
when (error) { when (error) {
is TunnelErrorEvent.VpnPermissionDenied -> { is TunnelErrorEvent.VpnPermissionDenied -> {
notificationManager.showVpnRequired() showOrNotify(
scope = scope,
foregroundAction = {
globalEffectRepository.post(
GlobalSideEffect.Snackbar(
message =
StringValue.DynamicString(
context.getString(R.string.vpn_permission_required)
),
type = ToastType.Error,
)
)
},
backgroundAction = { notificationManager.showVpnRequired() },
)
} }
is TunnelErrorEvent.InternalFailure -> { is TunnelErrorEvent.InternalFailure -> {
notificationManager.showError(error.message) showOrNotify(
scope = scope,
foregroundAction = {
globalEffectRepository.post(
GlobalSideEffect.Snackbar(
message = StringValue.DynamicString(error.message),
type = ToastType.Error,
)
)
},
backgroundAction = { notificationManager.showError(error.message) },
)
} }
is TunnelErrorEvent.Socks5PortUnavailable -> { is TunnelErrorEvent.Socks5PortUnavailable -> {
val name = getTunnelName(error.tunnelId) val name = getTunnelName(error.tunnelId)
notificationManager.showSocks5PortUnavailable(error.port, name) 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)
},
)
} }
is TunnelErrorEvent.HttpPortUnavailable -> { is TunnelErrorEvent.HttpPortUnavailable -> {
val name = getTunnelName(error.tunnelId) val name = getTunnelName(error.tunnelId)
notificationManager.showHttpPortUnavailable(error.port, name) 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)
},
)
} }
} }
} }
@@ -108,8 +234,7 @@ class TunnelEventDispatcher(
val tunnel = allTunnels.find { it.id == id } ?: return@mapNotNull null val tunnel = allTunnels.find { it.id == id } ?: return@mapNotNull null
val displayState = val displayState =
displayStates[id] displayStates[id] ?: DisplayTunnelState.from(activeTunnel)
?: DisplayTunnelState.from(activeTunnel, System.currentTimeMillis())
TunnelNotificationLine( TunnelNotificationLine(
id = id, id = id,
@@ -137,8 +262,7 @@ class TunnelEventDispatcher(
val tunnel = allTunnels.find { it.id == id } ?: return@mapNotNull null val tunnel = allTunnels.find { it.id == id } ?: return@mapNotNull null
val displayState = val displayState =
displayStates[id] displayStates[id] ?: DisplayTunnelState.from(activeTunnel)
?: DisplayTunnelState.from(activeTunnel, System.currentTimeMillis())
TunnelNotificationLine( TunnelNotificationLine(
id = id, id = id,
@@ -156,6 +280,18 @@ class TunnelEventDispatcher(
.launchIn(scope) .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 { private suspend fun getTunnelName(tunnelId: Int): String {
return tunnelRepository.getById(tunnelId)?.name ?: context.getString(R.string.unknown) return tunnelRepository.getById(tunnelId)?.name ?: context.getString(R.string.unknown)
} }
@@ -39,7 +39,7 @@ class AppBoostrapCoordinator(
listOf( listOf(
async { bootstrapDns() }, async { bootstrapDns() },
async { ensureGlobalConfig() }, async { ensureGlobalConfig() },
async { restoreLockdown() }, async { restoreBackendConfiguration() },
) )
try { try {
@@ -73,9 +73,13 @@ class AppBoostrapCoordinator(
tunnelRepository.ensureGlobalConfigExists() tunnelRepository.ensureGlobalConfigExists()
} }
private suspend fun restoreLockdown() { private suspend fun restoreBackendConfiguration() {
val settings = settingsRepository.getGeneralSettings() val settings = settingsRepository.getGeneralSettings()
if (settings.seamlessRoamingEnabled) {
tunnelProvider.setSeamlessRoaming(true)
}
when (settings.tunnelMode) { when (settings.tunnelMode) {
TunnelMode.LOCK_DOWN -> { TunnelMode.LOCK_DOWN -> {
val lockdownSettings = lockdownRepository.getLockdownSettings() val lockdownSettings = lockdownRepository.getLockdownSettings()
@@ -1,8 +1,8 @@
package com.zaneschepke.wireguardautotunnel.core.orchestration 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.domain.repository.AutoTunnelSettingsRepository
import com.zaneschepke.wireguardautotunnel.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.service.autotunnel.AutoTunnelStateHolder
class AutoTunnelCoordinator( class AutoTunnelCoordinator(
private val repository: AutoTunnelSettingsRepository, private val repository: AutoTunnelSettingsRepository,
@@ -5,7 +5,7 @@ import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
import com.zaneschepke.wireguardautotunnel.domain.repository.GeneralSettingRepository import com.zaneschepke.wireguardautotunnel.domain.repository.GeneralSettingRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.LockdownSettingsRepository import com.zaneschepke.wireguardautotunnel.domain.repository.LockdownSettingsRepository
class TunnelModeCoordinator( class TunnelBackendCoordinator(
private val tunnelProvider: TunnelProvider, private val tunnelProvider: TunnelProvider,
private val settingsRepository: GeneralSettingRepository, private val settingsRepository: GeneralSettingRepository,
private val lockdownRepository: LockdownSettingsRepository, private val lockdownRepository: LockdownSettingsRepository,
@@ -42,7 +42,6 @@ class TunnelModeCoordinator(
when (newMode) { when (newMode) {
TunnelMode.LOCK_DOWN -> { TunnelMode.LOCK_DOWN -> {
val lockdownSettings = lockdownRepository.getLockdownSettings() val lockdownSettings = lockdownRepository.getLockdownSettings()
tunnelProvider.setLockDown(lockdownSettings).getOrThrow() tunnelProvider.setLockDown(lockdownSettings).getOrThrow()
} }
@@ -50,4 +49,9 @@ class TunnelModeCoordinator(
TunnelMode.PROXY -> Unit TunnelMode.PROXY -> Unit
} }
} }
suspend fun changeSeamlessRoaming(enabled: Boolean) {
tunnelProvider.setSeamlessRoaming(enabled).getOrThrow()
settingsRepository.updateSeamlessRoaming(enabled)
}
} }
@@ -2,7 +2,6 @@ package com.zaneschepke.wireguardautotunnel.core.orchestration
import com.zaneschepke.tunnel.model.BackendMode import com.zaneschepke.tunnel.model.BackendMode
import com.zaneschepke.wireguardautotunnel.core.event.TunnelErrorEvent 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.core.tunnel.TunnelProvider
import com.zaneschepke.wireguardautotunnel.data.repository.RoomDnsSettingsRepository import com.zaneschepke.wireguardautotunnel.data.repository.RoomDnsSettingsRepository
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelActionSource import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelActionSource
@@ -10,13 +9,16 @@ import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
import com.zaneschepke.wireguardautotunnel.domain.events.TunnelActionEvent import com.zaneschepke.wireguardautotunnel.domain.events.TunnelActionEvent
import com.zaneschepke.wireguardautotunnel.domain.model.DnsSettings import com.zaneschepke.wireguardautotunnel.domain.model.DnsSettings
import com.zaneschepke.wireguardautotunnel.domain.model.GeneralSettings 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.MonitoringSettings
import com.zaneschepke.wireguardautotunnel.domain.model.ProxySettings import com.zaneschepke.wireguardautotunnel.domain.model.ProxySettings
import com.zaneschepke.wireguardautotunnel.domain.model.TunnelConfig import com.zaneschepke.wireguardautotunnel.domain.model.TunnelConfig
import com.zaneschepke.wireguardautotunnel.domain.repository.GeneralSettingRepository 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.MonitoringSettingsRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.ProxySettingsRepository import com.zaneschepke.wireguardautotunnel.domain.repository.ProxySettingsRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.ui.state.DisplayTunnelState import com.zaneschepke.wireguardautotunnel.ui.state.DisplayTunnelState
import kotlin.time.Duration.Companion.milliseconds import kotlin.time.Duration.Companion.milliseconds
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
@@ -44,6 +46,7 @@ class TunnelCoordinator(
dnsSettingsRepository: RoomDnsSettingsRepository, dnsSettingsRepository: RoomDnsSettingsRepository,
monitoringSettingsRepository: MonitoringSettingsRepository, monitoringSettingsRepository: MonitoringSettingsRepository,
proxyRepository: ProxySettingsRepository, proxyRepository: ProxySettingsRepository,
lockdownModeRepository: LockdownSettingsRepository,
scope: CoroutineScope, scope: CoroutineScope,
) { ) {
@@ -55,7 +58,7 @@ class TunnelCoordinator(
tunnelProvider.backendStatus tunnelProvider.backendStatus
.map { status -> .map { status ->
status.activeTunnels.mapValues { (_, activeTunnel) -> status.activeTunnels.mapValues { (_, activeTunnel) ->
DisplayTunnelState.from(activeTunnel, System.currentTimeMillis()) DisplayTunnelState.from(activeTunnel)
} }
} }
.debounce(400L.milliseconds) .debounce(400L.milliseconds)
@@ -66,6 +69,7 @@ class TunnelCoordinator(
val dns: DnsSettings, val dns: DnsSettings,
val monitoring: MonitoringSettings, val monitoring: MonitoringSettings,
val proxy: ProxySettings, val proxy: ProxySettings,
val lockdown: LockdownSettings,
) )
private val runtimeSettingsSnapshot = private val runtimeSettingsSnapshot =
@@ -74,12 +78,14 @@ class TunnelCoordinator(
dnsSettingsRepository.flow, dnsSettingsRepository.flow,
monitoringSettingsRepository.flow, monitoringSettingsRepository.flow,
proxyRepository.flow, proxyRepository.flow,
) { general, dns, monitoring, proxy -> lockdownModeRepository.flow,
) { general, dns, monitoring, proxy, lockdown ->
RuntimeSettingsSnapshot( RuntimeSettingsSnapshot(
general = general, general = general,
dns = dns, dns = dns,
monitoring = monitoring, monitoring = monitoring,
proxy = proxy, proxy = proxy,
lockdown = lockdown,
) )
} }
@@ -117,7 +123,7 @@ class TunnelCoordinator(
// enforce single tunnel, for now // enforce single tunnel, for now
if (backendStatus.value.activeTunnels.isNotEmpty()) { if (backendStatus.value.activeTunnels.isNotEmpty()) {
stopActiveTunnelsInternal() stopActiveTunnelsInternal(source)
} }
startTunnelInternal(config, source) startTunnelInternal(config, source)
@@ -131,7 +137,13 @@ class TunnelCoordinator(
stopTunnelInternal(id, source) stopTunnelInternal(id, source)
} }
suspend fun stopActiveTunnels() = tunnelMutex.withLock { stopActiveTunnelsInternal() } suspend fun stopActiveTunnels(source: TunnelActionSource = TunnelActionSource.USER) =
tunnelMutex.withLock {
if (source == TunnelActionSource.USER) {
_userOverrideFlow.tryEmit(Unit)
}
stopActiveTunnelsInternal(source)
}
private suspend fun startTunnelInternal( private suspend fun startTunnelInternal(
tunnelConfig: TunnelConfig, tunnelConfig: TunnelConfig,
@@ -143,6 +155,7 @@ class TunnelCoordinator(
val dnsSettings = snapshot.dns val dnsSettings = snapshot.dns
val proxySettings = snapshot.proxy val proxySettings = snapshot.proxy
val monitoringSettings = snapshot.monitoring val monitoringSettings = snapshot.monitoring
val lockdownSettings = snapshot.lockdown
val config = tunnelConfig.getConfig() val config = tunnelConfig.getConfig()
val policy = val policy =
@@ -178,8 +191,10 @@ class TunnelCoordinator(
} }
TunnelMode.LOCK_DOWN -> { TunnelMode.LOCK_DOWN -> {
BackendMode.Proxy.KillSwitchPrimary(
BackendMode.Proxy.KillSwitchPrimary(runConfig) runConfig,
lockdownSettings.toKillSwitchConfig(),
)
} }
} }
@@ -206,6 +221,10 @@ class TunnelCoordinator(
suspend fun toggleTunnels(source: TunnelActionSource = TunnelActionSource.USER) = suspend fun toggleTunnels(source: TunnelActionSource = TunnelActionSource.USER) =
tunnelMutex.withLock { tunnelMutex.withLock {
if (source == TunnelActionSource.USER) {
_userOverrideFlow.tryEmit(Unit)
}
val active = tunnelProvider.backendStatus.value.activeTunnels val active = tunnelProvider.backendStatus.value.activeTunnels
if (active.isNotEmpty()) { if (active.isNotEmpty()) {
lastActiveTunnels = active.keys.toList() lastActiveTunnels = active.keys.toList()
@@ -214,7 +233,7 @@ class TunnelCoordinator(
_actions.emit(TunnelActionEvent.Stopped(tunnelId = id, source = source)) _actions.emit(TunnelActionEvent.Stopped(tunnelId = id, source = source))
} }
stopActiveTunnelsInternal() stopActiveTunnelsInternal(source)
return@withLock return@withLock
} }
@@ -239,7 +258,15 @@ class TunnelCoordinator(
.onFailure { _errors.emit(TunnelErrorEvent.from(it, id)) } .onFailure { _errors.emit(TunnelErrorEvent.from(it, id)) }
} }
private suspend fun stopActiveTunnelsInternal() { 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))
}
tunnelProvider.stopActiveTunnels() tunnelProvider.stopActiveTunnels()
} }
} }
@@ -19,9 +19,8 @@ class ShortcutsActivity : ComponentActivity() {
super.onCreate(savedInstanceState) super.onCreate(savedInstanceState)
applicationScope.launch { finish()
shortcutCoordinator.handle(intent)
finish() applicationScope.launch { shortcutCoordinator.handle(intent) }
}
} }
} }
@@ -0,0 +1,116 @@
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,14 +4,11 @@ import com.zaneschepke.tunnel.Tunnel
import com.zaneschepke.tunnel.backend.Backend import com.zaneschepke.tunnel.backend.Backend
import com.zaneschepke.tunnel.model.BackendMode import com.zaneschepke.tunnel.model.BackendMode
import com.zaneschepke.tunnel.state.BackendStatus 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 com.zaneschepke.wireguardautotunnel.domain.model.LockdownSettings
import kotlin.concurrent.atomics.ExperimentalAtomicApi import kotlin.concurrent.atomics.ExperimentalAtomicApi
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.ExperimentalCoroutinesApi import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.flow.MutableSharedFlow
import kotlinx.coroutines.flow.SharingStarted import kotlinx.coroutines.flow.SharingStarted
import kotlinx.coroutines.flow.StateFlow import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.stateIn import kotlinx.coroutines.flow.stateIn
@@ -53,9 +50,7 @@ class TunnelBackendProvider(
return backend.disableKillSwitch() return backend.disableKillSwitch()
} }
@OptIn(ExperimentalCoroutinesApi::class) override suspend fun setSeamlessRoaming(enabled: Boolean): Result<Unit> {
private val localErrorEvents = MutableSharedFlow<Pair<String?, BackendCoreException>>() return backend.setSeamlessRoaming(enabled)
}
@OptIn(ExperimentalCoroutinesApi::class)
private val localMessageEvents = MutableSharedFlow<Pair<String?, BackendMessage>>()
} }
@@ -20,6 +20,8 @@ interface TunnelProvider {
suspend fun disableLockDown(): Result<Unit> suspend fun disableLockDown(): Result<Unit>
suspend fun setSeamlessRoaming(enabled: Boolean): Result<Unit>
val backendStatus: StateFlow<BackendStatus> val backendStatus: StateFlow<BackendStatus>
val events: Flow<TunnelEvent> val events: Flow<TunnelEvent>
@@ -6,9 +6,9 @@ import androidx.work.ExistingPeriodicWorkPolicy
import androidx.work.PeriodicWorkRequestBuilder import androidx.work.PeriodicWorkRequestBuilder
import androidx.work.WorkManager import androidx.work.WorkManager
import androidx.work.WorkerParameters 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.domain.repository.AutoTunnelSettingsRepository
import com.zaneschepke.wireguardautotunnel.service.ServiceManager
import com.zaneschepke.wireguardautotunnel.service.autotunnel.AutoTunnelStateHolder
import java.util.concurrent.TimeUnit import java.util.concurrent.TimeUnit
import timber.log.Timber import timber.log.Timber
@@ -34,7 +34,7 @@ import com.zaneschepke.wireguardautotunnel.data.entity.TunnelConfig
DnsSettings::class, DnsSettings::class,
LockdownSettings::class, LockdownSettings::class,
], ],
version = 30, version = 32,
autoMigrations = autoMigrations =
[ [
AutoMigration(from = 1, to = 2), AutoMigration(from = 1, to = 2),
@@ -63,6 +63,8 @@ import com.zaneschepke.wireguardautotunnel.data.entity.TunnelConfig
AutoMigration(from = 26, to = 27, spec = GlobalsMigration::class), AutoMigration(from = 26, to = 27, spec = GlobalsMigration::class),
AutoMigration(from = 27, to = 28, spec = DonationMigration::class), AutoMigration(from = 27, to = 28, spec = DonationMigration::class),
AutoMigration(from = 29, to = 30, spec = SingleConfigMigration::class), AutoMigration(from = 29, to = 30, spec = SingleConfigMigration::class),
AutoMigration(from = 30, to = 31),
AutoMigration(from = 31, to = 32),
], ],
exportSchema = true, exportSchema = true,
) )
@@ -18,4 +18,7 @@ interface AutoTunnelSettingsDao {
@Query("UPDATE auto_tunnel_settings SET is_tunnel_enabled = :enabled") @Query("UPDATE auto_tunnel_settings SET is_tunnel_enabled = :enabled")
suspend fun updateAutoTunnelEnabled(enabled: Boolean) suspend fun updateAutoTunnelEnabled(enabled: Boolean)
@Query("UPDATE auto_tunnel_settings SET disable_on_captive_portal = :enabled")
suspend fun updateDisableOnCaptivePortal(enabled: Boolean)
} }
@@ -34,4 +34,7 @@ interface GeneralSettingsDao {
@Query("UPDATE general_settings SET screen_recording_security = :enabled") @Query("UPDATE general_settings SET screen_recording_security = :enabled")
suspend fun updateScreenRecordingSecurity(enabled: Boolean) suspend fun updateScreenRecordingSecurity(enabled: Boolean)
@Query("UPDATE general_settings SET seamless_roaming_enabled = :enabled")
suspend fun updateSeamlessRoaming(enabled: Boolean)
} }
@@ -27,4 +27,6 @@ data class AutoTunnelSettings(
@ColumnInfo(name = "wifi_detection_method", defaultValue = "0") @ColumnInfo(name = "wifi_detection_method", defaultValue = "0")
val wifiDetectionMethod: WifiDetectionMethod = WifiDetectionMethod.fromValue(0), val wifiDetectionMethod: WifiDetectionMethod = WifiDetectionMethod.fromValue(0),
@ColumnInfo(name = "start_on_boot", defaultValue = "0") val startOnBoot: Boolean = false, @ColumnInfo(name = "start_on_boot", defaultValue = "0") val startOnBoot: Boolean = false,
@ColumnInfo(name = "disable_on_captive_portal", defaultValue = "1")
val disableTunnelOnCaptivePortal: Boolean = true,
) )
@@ -34,4 +34,6 @@ data class GeneralSettings(
val isGlobalAmneziaEnabled: Boolean = false, val isGlobalAmneziaEnabled: Boolean = false,
@ColumnInfo(name = "tunnel_scripting_enabled", defaultValue = "0") @ColumnInfo(name = "tunnel_scripting_enabled", defaultValue = "0")
val tunnelScriptingEnabled: Boolean = true, val tunnelScriptingEnabled: Boolean = true,
@ColumnInfo(name = "seamless_roaming_enabled", defaultValue = "0")
val seamlessRoamingEnabled: Boolean = true,
) )
@@ -16,6 +16,7 @@ fun Entity.toDomain(): Domain =
isTunnelOnUnsecureEnabled = isTunnelOnUnsecureEnabled, isTunnelOnUnsecureEnabled = isTunnelOnUnsecureEnabled,
wifiDetectionMethod = wifiDetectionMethod, wifiDetectionMethod = wifiDetectionMethod,
startOnBoot = startOnBoot, startOnBoot = startOnBoot,
disableTunnelOnCaptivePortal = disableTunnelOnCaptivePortal
) )
fun Domain.toEntity(): Entity = fun Domain.toEntity(): Entity =
@@ -31,4 +32,5 @@ fun Domain.toEntity(): Entity =
isTunnelOnUnsecureEnabled = isTunnelOnUnsecureEnabled, isTunnelOnUnsecureEnabled = isTunnelOnUnsecureEnabled,
wifiDetectionMethod = wifiDetectionMethod, wifiDetectionMethod = wifiDetectionMethod,
startOnBoot = startOnBoot, startOnBoot = startOnBoot,
disableTunnelOnCaptivePortal = disableTunnelOnCaptivePortal
) )
@@ -22,6 +22,7 @@ fun Entity.toDomain(): Domain =
screenRecordingSecurityEnabled = screenRecordingSecurityEnabled, screenRecordingSecurityEnabled = screenRecordingSecurityEnabled,
isGlobalAmneziaEnabled = isGlobalAmneziaEnabled, isGlobalAmneziaEnabled = isGlobalAmneziaEnabled,
tunnelScriptingEnabled = tunnelScriptingEnabled, tunnelScriptingEnabled = tunnelScriptingEnabled,
seamlessRoamingEnabled = seamlessRoamingEnabled,
) )
fun Domain.toEntity(): Entity = fun Domain.toEntity(): Entity =
@@ -42,4 +43,5 @@ fun Domain.toEntity(): Entity =
screenRecordingSecurityEnabled = screenRecordingSecurityEnabled, screenRecordingSecurityEnabled = screenRecordingSecurityEnabled,
isGlobalAmneziaEnabled = isGlobalAmneziaEnabled, isGlobalAmneziaEnabled = isGlobalAmneziaEnabled,
tunnelScriptingEnabled = tunnelScriptingEnabled, tunnelScriptingEnabled = tunnelScriptingEnabled,
seamlessRoamingEnabled = seamlessRoamingEnabled,
) )
@@ -1,15 +1,25 @@
package com.zaneschepke.wireguardautotunnel.data.network package com.zaneschepke.wireguardautotunnel.data.network
import com.zaneschepke.wireguardautotunnel.BuildConfig
import io.ktor.client.HttpClient import io.ktor.client.HttpClient
import io.ktor.client.engine.okhttp.OkHttp import io.ktor.client.engine.okhttp.OkHttp
import io.ktor.client.plugins.DefaultRequest
import io.ktor.client.plugins.HttpTimeout import io.ktor.client.plugins.HttpTimeout
import io.ktor.client.plugins.contentnegotiation.ContentNegotiation import io.ktor.client.plugins.contentnegotiation.ContentNegotiation
import io.ktor.http.HttpHeaders
import io.ktor.serialization.kotlinx.json.json import io.ktor.serialization.kotlinx.json.json
import kotlinx.serialization.json.Json import kotlinx.serialization.json.Json
object KtorClient { object KtorClient {
fun create(): HttpClient { fun create(): HttpClient {
return HttpClient(OkHttp) { return HttpClient(OkHttp) {
install(DefaultRequest) {
headers {
append(HttpHeaders.UserAgent, "wgtunnel/${BuildConfig.VERSION_NAME} (Android)")
append(HttpHeaders.Accept, "*/*")
}
}
install(ContentNegotiation) { install(ContentNegotiation) {
json( json(
Json { Json {
@@ -18,10 +28,11 @@ object KtorClient {
} }
) )
} }
install(HttpTimeout) { install(HttpTimeout) {
requestTimeoutMillis = 15000 requestTimeoutMillis = 120_000L
connectTimeoutMillis = 15000 connectTimeoutMillis = 30_000L
socketTimeoutMillis = 15000 socketTimeoutMillis = 120_000L
} }
} }
} }
@@ -26,4 +26,8 @@ class RoomAutoTunnelSettingsRepository(private val autoTunnelSettingsDao: AutoTu
override suspend fun updateAutoTunnelEnabled(enabled: Boolean) { override suspend fun updateAutoTunnelEnabled(enabled: Boolean) {
autoTunnelSettingsDao.updateAutoTunnelEnabled(enabled) autoTunnelSettingsDao.updateAutoTunnelEnabled(enabled)
} }
override suspend fun updateDisableOnCaptivePortal(enabled: Boolean) {
autoTunnelSettingsDao.updateDisableOnCaptivePortal(enabled)
}
} }
@@ -45,4 +45,8 @@ class RoomSettingsRepository(private val settingsDao: GeneralSettingsDao) :
override suspend fun updateScreenRecordingSecurity(enabled: Boolean) { override suspend fun updateScreenRecordingSecurity(enabled: Boolean) {
settingsDao.updateScreenRecordingSecurity(enabled) settingsDao.updateScreenRecordingSecurity(enabled)
} }
override suspend fun updateSeamlessRoaming(enabled: Boolean) {
settingsDao.updateSeamlessRoaming(enabled)
}
} }
@@ -6,20 +6,19 @@ import android.os.StrictMode
import com.zaneschepke.logcatter.LogReader import com.zaneschepke.logcatter.LogReader
import com.zaneschepke.logcatter.LogcatReader import com.zaneschepke.logcatter.LogcatReader
import com.zaneschepke.wireguardautotunnel.BuildConfig 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.DynamicShortcutManager
import com.zaneschepke.wireguardautotunnel.core.shortcut.ShortcutManager import com.zaneschepke.wireguardautotunnel.core.shortcut.ShortcutManager
import com.zaneschepke.wireguardautotunnel.domain.repository.GlobalEffectRepository import com.zaneschepke.wireguardautotunnel.domain.repository.GlobalEffectRepository
import com.zaneschepke.wireguardautotunnel.domain.repository.SelectedTunnelsRepository 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.FileUtils
import com.zaneschepke.wireguardautotunnel.util.network.NetworkUtils import com.zaneschepke.wireguardautotunnel.util.network.NetworkUtils
import com.zaneschepke.wireguardautotunnel.viewmodel.AutoTunnelViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.AutoTunnelViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.ConfigEditViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.ConfigEditViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.DnsViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.DnsViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.LicenseViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.LockdownViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.LockdownViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.LoggerViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.LoggerViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.MonitoringViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.MonitoringViewModel
@@ -80,7 +79,6 @@ val appModule = module {
viewModelOf(::AutoTunnelViewModel) viewModelOf(::AutoTunnelViewModel)
viewModel { (id: Int?) -> ConfigEditViewModel(get(), get(), get(), get(), get(), id) } viewModel { (id: Int?) -> ConfigEditViewModel(get(), get(), get(), get(), get(), id) }
viewModelOf(::DnsViewModel) viewModelOf(::DnsViewModel)
viewModelOf(::LicenseViewModel)
viewModelOf(::LockdownViewModel) viewModelOf(::LockdownViewModel)
viewModelOf(::LoggerViewModel) viewModelOf(::LoggerViewModel)
viewModelOf(::MonitoringViewModel) 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.DnsSettingsCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.ShortcutCoordinator import com.zaneschepke.wireguardautotunnel.core.orchestration.ShortcutCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.StartupCoordinator 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.TunnelCoordinator
import com.zaneschepke.wireguardautotunnel.core.orchestration.TunnelModeCoordinator
import org.koin.core.module.dsl.singleOf import org.koin.core.module.dsl.singleOf
import org.koin.core.qualifier.named import org.koin.core.qualifier.named
import org.koin.dsl.module import org.koin.dsl.module
val coordinatorModule = module { val coordinatorModule = module {
singleOf(::ShortcutCoordinator) singleOf(::ShortcutCoordinator)
singleOf(::TunnelModeCoordinator) singleOf(::TunnelBackendCoordinator)
singleOf(::StartupCoordinator) singleOf(::StartupCoordinator)
singleOf(::AutoTunnelCoordinator) singleOf(::AutoTunnelCoordinator)
singleOf(::DnsSettingsCoordinator) singleOf(::DnsSettingsCoordinator)
@@ -27,6 +27,7 @@ val coordinatorModule = module {
get(), get(),
get(), get(),
get(), get(),
get(),
get(named(Scope.APPLICATION)), get(named(Scope.APPLICATION)),
) )
} }
@@ -1,25 +1,19 @@
package com.zaneschepke.wireguardautotunnel.di package com.zaneschepke.wireguardautotunnel.di
import android.app.Notification
import android.content.Context
import com.zaneschepke.networkmonitor.AndroidNetworkMonitor import com.zaneschepke.networkmonitor.AndroidNetworkMonitor
import com.zaneschepke.networkmonitor.NetworkMonitor import com.zaneschepke.networkmonitor.NetworkMonitor
import com.zaneschepke.networkmonitor.StableNetworkEngine import com.zaneschepke.networkmonitor.StableNetworkEngine
import com.zaneschepke.tunnel.NotificationProvider import com.zaneschepke.tunnel.ApplicationProvider
import com.zaneschepke.tunnel.backend.RootShell import com.zaneschepke.tunnel.util.RootShell
import com.zaneschepke.tunnel.util.RootShellException import com.zaneschepke.tunnel.util.RootShellException
import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.core.event.TunnelEventDispatcher import com.zaneschepke.wireguardautotunnel.core.event.TunnelEventDispatcher
import com.zaneschepke.wireguardautotunnel.core.notification.AndroidNotificationService.NotificationChannels import com.zaneschepke.wireguardautotunnel.core.tunnel.AndroidApplicationProvider
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.TunnelBackendProvider
import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelProvider import com.zaneschepke.wireguardautotunnel.core.tunnel.TunnelProvider
import com.zaneschepke.wireguardautotunnel.domain.repository.AutoTunnelSettingsRepository 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 com.zaneschepke.wireguardautotunnel.util.extensions.to
import kotlin.time.Duration.Companion.milliseconds import kotlin.time.Duration.Companion.milliseconds
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
@@ -36,40 +30,15 @@ import timber.log.Timber
val tunnelBackendProviderModule = module { val tunnelBackendProviderModule = module {
single<TunnelNotificationService> { AndroidTunnelNotificationService(get()) } single<TunnelNotificationService> { AndroidTunnelNotificationService(get()) }
single { AppVisibilityObserver() }
singleOf(::TunnelEventDispatcher) singleOf(::TunnelEventDispatcher)
single<NotificationProvider> { single<ApplicationProvider> {
val notificationService = get<NotificationService>() AndroidApplicationProvider(
val context = androidContext() notificationService = get(),
object : NotificationProvider { tunnelNotificationService = get(),
override val vpnInitNotification: Notification tunnelRepository = get(),
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 { single {
@@ -14,4 +14,5 @@ data class AutoTunnelSettings(
val isTunnelOnUnsecureEnabled: Boolean = false, val isTunnelOnUnsecureEnabled: Boolean = false,
val wifiDetectionMethod: WifiDetectionMethod = WifiDetectionMethod.fromValue(0), val wifiDetectionMethod: WifiDetectionMethod = WifiDetectionMethod.fromValue(0),
val startOnBoot: Boolean = false, val startOnBoot: Boolean = false,
val disableTunnelOnCaptivePortal: Boolean = true,
) )
@@ -20,5 +20,6 @@ data class GeneralSettings(
val alreadyDonated: Boolean = false, val alreadyDonated: Boolean = false,
val screenRecordingSecurityEnabled: Boolean = true, val screenRecordingSecurityEnabled: Boolean = true,
val isGlobalAmneziaEnabled: Boolean = false, val isGlobalAmneziaEnabled: Boolean = false,
val tunnelScriptingEnabled: Boolean = true, val tunnelScriptingEnabled: Boolean = false,
val seamlessRoamingEnabled: Boolean = false,
) )
@@ -11,4 +11,6 @@ interface AutoTunnelSettingsRepository {
suspend fun getAutoTunnelSettings(): AutoTunnelSettings suspend fun getAutoTunnelSettings(): AutoTunnelSettings
suspend fun updateAutoTunnelEnabled(enabled: Boolean) suspend fun updateAutoTunnelEnabled(enabled: Boolean)
suspend fun updateDisableOnCaptivePortal(enabled: Boolean)
} }
@@ -23,4 +23,6 @@ interface GeneralSettingRepository {
suspend fun updateGlobalAmneziaEnabled(enabled: Boolean) suspend fun updateGlobalAmneziaEnabled(enabled: Boolean)
suspend fun updateScreenRecordingSecurity(enabled: Boolean) suspend fun updateScreenRecordingSecurity(enabled: Boolean)
suspend fun updateSeamlessRoaming(enabled: Boolean)
} }
@@ -7,7 +7,7 @@ import kotlinx.coroutines.flow.asSharedFlow
class GlobalEffectRepository { class GlobalEffectRepository {
private val _globalEffectFlow = private val _globalEffectFlow =
MutableSharedFlow<GlobalSideEffect>(replay = 0, extraBufferCapacity = 1) MutableSharedFlow<GlobalSideEffect>(replay = 0, extraBufferCapacity = 0)
val flow = _globalEffectFlow.asSharedFlow() val flow = _globalEffectFlow.asSharedFlow()
suspend fun post(effect: GlobalSideEffect) { suspend fun post(effect: GlobalSideEffect) {
@@ -1,8 +1,8 @@
package com.zaneschepke.wireguardautotunnel.domain.sideeffect package com.zaneschepke.wireguardautotunnel.domain.sideeffect
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
import com.zaneschepke.wireguardautotunnel.domain.model.TunnelConfig import com.zaneschepke.wireguardautotunnel.domain.model.TunnelConfig
import com.zaneschepke.wireguardautotunnel.ui.common.snackbar.SnackbarType
import com.zaneschepke.wireguardautotunnel.util.StringValue import com.zaneschepke.wireguardautotunnel.util.StringValue
import java.io.File import java.io.File
@@ -10,14 +10,12 @@ sealed class GlobalSideEffect {
data class Snackbar( data class Snackbar(
val message: StringValue, val message: StringValue,
val type: SnackbarType? = null, val type: ToastType,
val actionLabel: String? = null, val actionLabel: String? = null,
val onAction: (() -> Unit)? = null, val onAction: (() -> Unit)? = null,
val durationMs: Long? = null, val durationMs: Long? = null,
) : GlobalSideEffect() ) : GlobalSideEffect()
data class Toast(val message: StringValue) : GlobalSideEffect()
data object PopBackStack : GlobalSideEffect() data object PopBackStack : GlobalSideEffect()
data class LaunchUrl(val url: String) : GlobalSideEffect() data class LaunchUrl(val url: String) : GlobalSideEffect()
@@ -11,16 +11,20 @@ sealed class ActiveNetwork {
data object Cellular : ActiveNetwork() data object Cellular : ActiveNetwork()
data class Wifi(val ssid: String, val isSecure: Boolean?) : ActiveNetwork() data class Wifi(
val ssid: String,
val isSecure: Boolean?,
val requiresCaptivePortalLogin: Boolean,
) : ActiveNetwork()
} }
data class NetworkState( data class NetworkState(
val activeNetwork: ActiveNetwork = ActiveNetwork.Disconnected, val activeNetwork: ActiveNetwork = ActiveNetwork.Disconnected,
val locationServicesEnabled: Boolean = false, val locationServicesEnabled: Boolean = false,
val locationPermissionGranted: Boolean = false, val locationPermissionGranted: Boolean = false,
) { // Has a network that can actually transfer data (not suspended)
fun hasInternet(): Boolean = activeNetwork !is ActiveNetwork.Disconnected val hasUsableNetwork: Boolean = false,
} )
fun ConnectivityState.toDomain(): NetworkState { fun ConnectivityState.toDomain(): NetworkState {
val domainNetwork: ActiveNetwork = val domainNetwork: ActiveNetwork =
@@ -33,7 +37,11 @@ fun ConnectivityState.toDomain(): NetworkState {
null -> null null -> null
else -> true else -> true
} }
ActiveNetwork.Wifi(ssid = network.ssid, isSecure = isSecure) ActiveNetwork.Wifi(
ssid = network.ssid,
isSecure = isSecure,
requiresCaptivePortalLogin(),
)
} }
is MonitorActiveNetwork.Cellular -> ActiveNetwork.Cellular is MonitorActiveNetwork.Cellular -> ActiveNetwork.Cellular
is MonitorActiveNetwork.Ethernet -> ActiveNetwork.Ethernet is MonitorActiveNetwork.Ethernet -> ActiveNetwork.Ethernet
@@ -44,5 +52,6 @@ fun ConnectivityState.toDomain(): NetworkState {
activeNetwork = domainNetwork, activeNetwork = domainNetwork,
locationPermissionGranted = this.locationPermissionsGranted, locationPermissionGranted = this.locationPermissionsGranted,
locationServicesEnabled = this.locationServicesEnabled, locationServicesEnabled = this.locationServicesEnabled,
hasUsableNetwork = hasUsableNetwork(),
) )
} }
@@ -0,0 +1,26 @@
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,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.core.notification package com.zaneschepke.wireguardautotunnel.notification
import android.Manifest import android.Manifest
import android.app.Notification import android.app.Notification
@@ -17,8 +17,8 @@ import androidx.core.app.NotificationManagerCompat.IMPORTANCE_HIGH
import com.zaneschepke.wireguardautotunnel.MainActivity import com.zaneschepke.wireguardautotunnel.MainActivity
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.core.broadcast.NotificationActionReceiver 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.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.notification.NotificationService.Companion.EXTRA_ID
import com.zaneschepke.wireguardautotunnel.util.StringValue import com.zaneschepke.wireguardautotunnel.util.StringValue
class AndroidNotificationService(override val context: Context) : NotificationService { class AndroidNotificationService(override val context: Context) : NotificationService {
@@ -1,21 +1,116 @@
package com.zaneschepke.wireguardautotunnel.core.notification package com.zaneschepke.wireguardautotunnel.notification
import androidx.core.app.NotificationCompat import androidx.core.app.NotificationCompat
import com.zaneschepke.wireguardautotunnel.R 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.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) : class AndroidTunnelNotificationService(private val notificationService: NotificationService) :
TunnelNotificationService { TunnelNotificationService {
private val context = notificationService.context 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( private fun updateGroupNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine>, tunnelNotificationLines: Map<Int, TunnelNotificationLine>,
notificationId: Int, notificationId: Int,
@@ -88,26 +183,36 @@ class AndroidTunnelNotificationService(private val notificationService: Notifica
notificationService.show(notificationId, notification) notificationService.show(notificationId, notification)
} }
override fun updateProxyPersistentNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine>
) {
updateGroupNotification(
tunnelNotificationLines = tunnelNotificationLines,
notificationId = PROXY_NOTIFICATION_ID,
channel = NotificationChannels.Tunnel.Proxy,
groupKey = PROXY_GROUP_KEY,
)
}
override fun updateVpnPersistentNotification( override fun updateVpnPersistentNotification(
tunnelNotificationLines: Map<Int, TunnelNotificationLine> tunnelNotificationLines: Map<Int, TunnelNotificationLine>
) { ) {
updateGroupNotification( if (tunnelNotificationLines.isEmpty()) {
tunnelNotificationLines = tunnelNotificationLines, notificationService.remove(VPN_NOTIFICATION_ID)
notificationId = VPN_NOTIFICATION_ID, return
channel = NotificationChannels.Tunnel.VPN, }
groupKey = VPN_GROUP_KEY, 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)
} }
override fun showIpv4Fallback(tunnelName: String) { override fun showIpv4Fallback(tunnelName: String) {
@@ -1,10 +1,10 @@
package com.zaneschepke.wireguardautotunnel.core.notification package com.zaneschepke.wireguardautotunnel.notification
import android.app.Notification import android.app.Notification
import android.content.Context import android.content.Context
import androidx.core.app.NotificationCompat import androidx.core.app.NotificationCompat
import com.zaneschepke.wireguardautotunnel.core.notification.AndroidNotificationService.NotificationChannels
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.notification.AndroidNotificationService.NotificationChannels
import com.zaneschepke.wireguardautotunnel.util.StringValue import com.zaneschepke.wireguardautotunnel.util.StringValue
interface NotificationService { interface NotificationService {
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.core.notification package com.zaneschepke.wireguardautotunnel.notification
import com.zaneschepke.wireguardautotunnel.ui.state.DisplayTunnelState import com.zaneschepke.wireguardautotunnel.ui.state.DisplayTunnelState
@@ -1,4 +1,6 @@
package com.zaneschepke.wireguardautotunnel.core.notification package com.zaneschepke.wireguardautotunnel.notification
import android.app.Notification
interface TunnelNotificationService { interface TunnelNotificationService {
@@ -6,6 +8,14 @@ interface TunnelNotificationService {
fun updateVpnPersistentNotification(tunnelNotificationLines: Map<Int, TunnelNotificationLine>) 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 showIpv4Fallback(tunnelName: String)
fun showIpv6Recovery(tunnelName: String) fun showIpv6Recovery(tunnelName: String)
@@ -1,9 +1,9 @@
package com.zaneschepke.wireguardautotunnel.core.service package com.zaneschepke.wireguardautotunnel.service
import android.content.Context import android.content.Context
import android.content.Intent import android.content.Intent
import android.net.VpnService import android.net.VpnService
import com.zaneschepke.wireguardautotunnel.core.service.autotunnel.AutoTunnelService import com.zaneschepke.wireguardautotunnel.service.autotunnel.AutoTunnelService
class ServiceManager(private val context: Context) { class ServiceManager(private val context: Context) {
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.core.service.autotunnel package com.zaneschepke.wireguardautotunnel.service.autotunnel
import com.zaneschepke.wireguardautotunnel.domain.events.AutoTunnelEvent import com.zaneschepke.wireguardautotunnel.domain.events.AutoTunnelEvent
import com.zaneschepke.wireguardautotunnel.domain.model.TunnelConfig import com.zaneschepke.wireguardautotunnel.domain.model.TunnelConfig
@@ -28,7 +28,19 @@ class AutoTunnelEngine {
val activeTunnelIds = backend.activeTunnels.keys.toSet() val activeTunnelIds = backend.activeTunnels.keys.toSet()
if (!network.hasInternet()) { 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) { return if (settings.isStopOnNoInternetEnabled) {
Decision.StopDueToNoInternet Decision.StopDueToNoInternet
} else { } else {
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.core.service.autotunnel package com.zaneschepke.wireguardautotunnel.service.autotunnel
import android.content.Intent import android.content.Intent
import androidx.core.app.ServiceCompat import androidx.core.app.ServiceCompat
@@ -7,10 +7,7 @@ import androidx.lifecycle.lifecycleScope
import com.zaneschepke.networkmonitor.AndroidNetworkMonitor import com.zaneschepke.networkmonitor.AndroidNetworkMonitor
import com.zaneschepke.networkmonitor.StableNetworkEngine import com.zaneschepke.networkmonitor.StableNetworkEngine
import com.zaneschepke.wireguardautotunnel.R 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.orchestration.TunnelCoordinator
import com.zaneschepke.wireguardautotunnel.core.service.tile.AutoTunnelTileRefresher
import com.zaneschepke.wireguardautotunnel.di.Dispatcher import com.zaneschepke.wireguardautotunnel.di.Dispatcher
import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction import com.zaneschepke.wireguardautotunnel.domain.enums.NotificationAction
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelActionSource import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelActionSource
@@ -23,17 +20,21 @@ import com.zaneschepke.wireguardautotunnel.domain.repository.GeneralSettingRepos
import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository import com.zaneschepke.wireguardautotunnel.domain.repository.TunnelRepository
import com.zaneschepke.wireguardautotunnel.domain.state.AutoTunnelState import com.zaneschepke.wireguardautotunnel.domain.state.AutoTunnelState
import com.zaneschepke.wireguardautotunnel.domain.state.toDomain 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.Constants
import com.zaneschepke.wireguardautotunnel.util.extensions.to import com.zaneschepke.wireguardautotunnel.util.extensions.to
import kotlin.time.Duration.Companion.milliseconds import kotlin.time.Duration.Companion.milliseconds
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.FlowPreview
import kotlinx.coroutines.Job import kotlinx.coroutines.Job
import kotlinx.coroutines.delay import kotlinx.coroutines.delay
import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.collectLatest import kotlinx.coroutines.flow.collectLatest
import kotlinx.coroutines.flow.combine import kotlinx.coroutines.flow.combine
import kotlinx.coroutines.flow.debounce
import kotlinx.coroutines.flow.distinctUntilChanged import kotlinx.coroutines.flow.distinctUntilChanged
import kotlinx.coroutines.flow.distinctUntilChangedBy
import kotlinx.coroutines.flow.firstOrNull import kotlinx.coroutines.flow.firstOrNull
import kotlinx.coroutines.flow.map import kotlinx.coroutines.flow.map
import kotlinx.coroutines.flow.mapNotNull import kotlinx.coroutines.flow.mapNotNull
@@ -77,13 +78,16 @@ class AutoTunnelService : LifecycleService() {
@Volatile private var hasUserOverride = false @Volatile private var hasUserOverride = false
private var lastNetworkFingerprint: AutoTunnelState.NetworkFingerprint? = null private var lastNetworkFingerprint: AutoTunnelState.NetworkFingerprint? = null
@OptIn(FlowPreview::class)
private val autoTunnelStateFlow: Flow<AutoTunnelState> by lazy { private val autoTunnelStateFlow: Flow<AutoTunnelState> by lazy {
val networkFlow = networkEngine.stableState.mapNotNull { it?.state?.toDomain() } val networkFlow = networkEngine.stableState.mapNotNull { it?.state?.toDomain() }
val settingsFlow = combineSettings() val settingsFlow = combineSettings()
val backendFlow = val backendFlow =
tunnelCoordinator.backendStatus.distinctUntilChangedBy { it.activeTunnels.keys.toSet() } tunnelCoordinator.backendStatus
.distinctUntilChanged { old, new -> old.activeTunnels == new.activeTunnels }
.debounce(300L.milliseconds)
combine(networkFlow, settingsFlow, backendFlow) { network, settings, backend -> combine(networkFlow, settingsFlow, backendFlow) { network, settings, backend ->
AutoTunnelState( AutoTunnelState(
@@ -188,11 +192,11 @@ class AutoTunnelService : LifecycleService() {
reconciliationMutex.withLock { reconciliationMutex.withLock {
val currentNetworkState = networkEngine.stableState.value?.state?.toDomain() val currentNetworkState = networkEngine.stableState.value?.state?.toDomain()
val stillNoInternet = currentNetworkState?.hasInternet() == false val stillNoUsableNetwork = currentNetworkState?.hasUsableNetwork == false
val stopOnNoInternetEnabled = val stopOnNoInternetEnabled =
autoTunnelRepository.flow.firstOrNull()?.isStopOnNoInternetEnabled == true autoTunnelRepository.flow.firstOrNull()?.isStopOnNoInternetEnabled == true
if (stillNoInternet && stopOnNoInternetEnabled) { if (stillNoUsableNetwork && stopOnNoInternetEnabled) {
val currentActiveIds = val currentActiveIds =
tunnelCoordinator.backendStatus.value.activeTunnels.keys tunnelCoordinator.backendStatus.value.activeTunnels.keys
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.core.service.autotunnel package com.zaneschepke.wireguardautotunnel.service.autotunnel
import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.StateFlow import kotlinx.coroutines.flow.StateFlow
@@ -1,9 +1,9 @@
package com.zaneschepke.wireguardautotunnel.core.service.tile package com.zaneschepke.wireguardautotunnel.service.tile
import android.service.quicksettings.Tile import android.service.quicksettings.Tile
import android.service.quicksettings.TileService import android.service.quicksettings.TileService
import com.zaneschepke.wireguardautotunnel.core.orchestration.AutoTunnelCoordinator import com.zaneschepke.wireguardautotunnel.core.orchestration.AutoTunnelCoordinator
import com.zaneschepke.wireguardautotunnel.core.service.autotunnel.AutoTunnelStateHolder import com.zaneschepke.wireguardautotunnel.service.autotunnel.AutoTunnelStateHolder
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.Job import kotlinx.coroutines.Job
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.core.service.tile package com.zaneschepke.wireguardautotunnel.service.tile
import android.content.ComponentName import android.content.ComponentName
import android.content.Context import android.content.Context
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.core.service.tile package com.zaneschepke.wireguardautotunnel.service.tile
import android.content.Context import android.content.Context
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.core.service.tile package com.zaneschepke.wireguardautotunnel.service.tile
import android.os.Build import android.os.Build
import android.service.quicksettings.Tile import android.service.quicksettings.Tile
@@ -1,4 +1,4 @@
package com.zaneschepke.wireguardautotunnel.core.service.tile package com.zaneschepke.wireguardautotunnel.service.tile
import android.content.ComponentName import android.content.ComponentName
import android.content.Context import android.content.Context
@@ -0,0 +1,73 @@
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),
)
}
@@ -1,104 +0,0 @@
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,
)
}
}
}
}
}
@@ -1,76 +0,0 @@
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) }
}
}
}
}
@@ -1,16 +0,0 @@
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,13 +1,18 @@
package com.zaneschepke.wireguardautotunnel.ui.common.textbox package com.zaneschepke.wireguardautotunnel.ui.common.textbox
import androidx.compose.foundation.background
import androidx.compose.foundation.interaction.MutableInteractionSource import androidx.compose.foundation.interaction.MutableInteractionSource
import androidx.compose.foundation.layout.Box
import androidx.compose.foundation.layout.defaultMinSize import androidx.compose.foundation.layout.defaultMinSize
import androidx.compose.foundation.layout.fillMaxWidth 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.KeyboardActions
import androidx.compose.foundation.text.KeyboardOptions import androidx.compose.foundation.text.KeyboardOptions
import androidx.compose.material3.MaterialTheme import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Text import androidx.compose.material3.Text
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.compose.runtime.remember
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.text.input.ImeAction import androidx.compose.ui.text.input.ImeAction
import androidx.compose.ui.text.input.KeyboardCapitalization import androidx.compose.ui.text.input.KeyboardCapitalization
@@ -29,48 +34,59 @@ fun ConfigurationTextBox(
leading: (@Composable () -> Unit)? = null, leading: (@Composable () -> Unit)? = null,
trailing: (@Composable (Modifier) -> Unit)? = null, trailing: (@Composable (Modifier) -> Unit)? = null,
supportingText: (@Composable () -> Unit)? = null, supportingText: (@Composable () -> Unit)? = null,
interactionSource: MutableInteractionSource = MutableInteractionSource(), interactionSource: MutableInteractionSource = remember { MutableInteractionSource() },
visualTransformation: VisualTransformation = VisualTransformation.None, visualTransformation: VisualTransformation = VisualTransformation.None,
enabled: Boolean = true, enabled: Boolean = true,
readOnly: Boolean = false, readOnly: Boolean = false,
singleLine: Boolean = true, 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,
)
CustomTextField( // custom static label notch
isError = isError, if (label.isNotEmpty()) {
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, text = label,
color = MaterialTheme.colorScheme.onSurface, color = MaterialTheme.colorScheme.onSurface,
style = MaterialTheme.typography.labelMedium, style = MaterialTheme.typography.labelSmall,
overflow = TextOverflow.Ellipsis, overflow = TextOverflow.Ellipsis,
maxLines = 1, 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,7 +22,9 @@ import androidx.compose.ui.focus.focusRequester
import androidx.compose.ui.focus.onFocusChanged import androidx.compose.ui.focus.onFocusChanged
import androidx.compose.ui.graphics.Color import androidx.compose.ui.graphics.Color
import androidx.compose.ui.graphics.SolidColor import androidx.compose.ui.graphics.SolidColor
import androidx.compose.ui.text.TextRange
import androidx.compose.ui.text.TextStyle import androidx.compose.ui.text.TextStyle
import androidx.compose.ui.text.input.TextFieldValue
import androidx.compose.ui.text.input.VisualTransformation import androidx.compose.ui.text.input.VisualTransformation
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
@@ -33,7 +35,7 @@ fun CustomTextField(
modifier: Modifier = Modifier, modifier: Modifier = Modifier,
textStyle: TextStyle = textStyle: TextStyle =
MaterialTheme.typography.bodyLarge.copy(color = MaterialTheme.colorScheme.onSurface), MaterialTheme.typography.bodyLarge.copy(color = MaterialTheme.colorScheme.onSurface),
label: @Composable () -> Unit, label: @Composable (() -> Unit)? = null,
containerColor: Color, containerColor: Color,
onValueChange: (value: String) -> Unit = {}, onValueChange: (value: String) -> Unit = {},
singleLine: Boolean = true, singleLine: Boolean = true,
@@ -47,10 +49,19 @@ fun CustomTextField(
readOnly: Boolean = false, readOnly: Boolean = false,
enabled: Boolean = true, enabled: Boolean = true,
visualTransformation: VisualTransformation = VisualTransformation.None, visualTransformation: VisualTransformation = VisualTransformation.None,
interactionSource: MutableInteractionSource = MutableInteractionSource(), interactionSource: MutableInteractionSource = remember { MutableInteractionSource() },
) { ) {
val space = " "
var isFocused by remember { mutableStateOf(false) } 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 = val cursorBrush =
if (isFocused) SolidColor(MaterialTheme.colorScheme.primary) if (isFocused) SolidColor(MaterialTheme.colorScheme.primary)
else SolidColor(Color.Transparent) else SolidColor(Color.Transparent)
@@ -67,9 +78,14 @@ fun CustomTextField(
} }
BasicTextField( BasicTextField(
value = value, value = textFieldValue,
textStyle = effectiveTextStyle, textStyle = effectiveTextStyle,
onValueChange = { onValueChange(it) }, onValueChange = { newTextFieldValue ->
textFieldValueState = newTextFieldValue
if (value != newTextFieldValue.text) {
onValueChange(newTextFieldValue.text)
}
},
keyboardActions = keyboardActions, keyboardActions = keyboardActions,
keyboardOptions = keyboardOptions, keyboardOptions = keyboardOptions,
readOnly = readOnly, readOnly = readOnly,
@@ -90,15 +106,9 @@ fun CustomTextField(
visualTransformation = visualTransformation, visualTransformation = visualTransformation,
) { ) {
OutlinedTextFieldDefaults.DecorationBox( OutlinedTextFieldDefaults.DecorationBox(
value = space + value, value = value,
innerTextField = { innerTextField = it,
if (value.isEmpty()) { placeholder = placeholder,
if (placeholder != null) {
placeholder()
}
}
it.invoke()
},
contentPadding = OutlinedTextFieldDefaults.contentPadding(top = 14.dp, bottom = 14.dp), contentPadding = OutlinedTextFieldDefaults.contentPadding(top = 14.dp, bottom = 14.dp),
leadingIcon = leading, leadingIcon = leading,
trailingIcon = trailingIcon =
@@ -141,7 +151,6 @@ fun CustomTextField(
label = label, label = label,
visualTransformation = visualTransformation, visualTransformation = visualTransformation,
interactionSource = interactionSource, interactionSource = interactionSource,
placeholder = placeholder,
container = { container = {
OutlinedTextFieldDefaults.Container( OutlinedTextFieldDefaults.Container(
enabled = enabled, enabled = enabled,
@@ -42,7 +42,12 @@ sealed class Route : NavKey {
@Keep @Serializable data object Display : Route() @Keep @Serializable data object Display : Route()
@Keep @Serializable data object Tunnels : Route() @Keep
@Serializable
data object Tunnels : Route(), SecureRoute {
override val requiresProtection: Boolean
get() = true
}
@Keep @Serializable data class TunnelSettings(val id: Int) : Route() @Keep @Serializable data class TunnelSettings(val id: Int) : Route()
@@ -10,6 +10,7 @@ import androidx.compose.foundation.verticalScroll
import androidx.compose.material.icons.Icons import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.outlined.Filter1 import androidx.compose.material.icons.outlined.Filter1
import androidx.compose.material.icons.outlined.Map import androidx.compose.material.icons.outlined.Map
import androidx.compose.material.icons.outlined.PublicOff
import androidx.compose.material.icons.outlined.WifiFind import androidx.compose.material.icons.outlined.WifiFind
import androidx.compose.material3.Icon import androidx.compose.material3.Icon
import androidx.compose.material3.MaterialTheme import androidx.compose.material3.MaterialTheme
@@ -191,6 +192,21 @@ 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 { Column {
GroupLabel(stringResource(R.string.tunnels), Modifier.padding(horizontal = 16.dp)) GroupLabel(stringResource(R.string.tunnels), Modifier.padding(horizontal = 16.dp))
@@ -9,6 +9,7 @@ import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.saveable.rememberSaveable import androidx.compose.runtime.saveable.rememberSaveable
import androidx.compose.runtime.setValue import androidx.compose.runtime.setValue
import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.stringResource
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.ui.LocalNavController import com.zaneschepke.wireguardautotunnel.ui.LocalNavController
import com.zaneschepke.wireguardautotunnel.ui.navigation.Route import com.zaneschepke.wireguardautotunnel.ui.navigation.Route
@@ -45,11 +46,17 @@ fun PinLockScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
textColor = MaterialTheme.colorScheme.onSurface, textColor = MaterialTheme.colorScheme.onSurface,
onPinCorrect = { onPinCorrect() }, onPinCorrect = { onPinCorrect() },
onPinIncorrect = { onPinIncorrect = {
sharedViewModel.showToast(StringValue.StringResource(R.string.incorrect_pin)) sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.incorrect_pin),
ToastType.Warning,
)
}, },
onPinCreated = { onPinCreated = {
pinCreated = true pinCreated = true
sharedViewModel.showToast(StringValue.StringResource(R.string.pin_created)) sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.pin_created),
ToastType.Success,
)
sharedViewModel.setPinLockEnabled(true) sharedViewModel.setPinLockEnabled(true)
onPinCorrect() onPinCorrect()
}, },
@@ -9,6 +9,7 @@ import androidx.compose.foundation.verticalScroll
import androidx.compose.material.icons.Icons import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.automirrored.outlined.ViewQuilt import androidx.compose.material.icons.automirrored.outlined.ViewQuilt
import androidx.compose.material.icons.outlined.Android 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.Dns
import androidx.compose.material.icons.outlined.ExpandMore import androidx.compose.material.icons.outlined.ExpandMore
import androidx.compose.material.icons.outlined.MonitorHeart import androidx.compose.material.icons.outlined.MonitorHeart
@@ -38,6 +39,7 @@ import androidx.compose.ui.res.stringResource
import androidx.compose.ui.res.vectorResource import androidx.compose.ui.res.vectorResource
import androidx.compose.ui.text.intl.Locale import androidx.compose.ui.text.intl.Locale
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.MainActivity import com.zaneschepke.wireguardautotunnel.MainActivity
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode import com.zaneschepke.wireguardautotunnel.domain.enums.TunnelMode
@@ -51,12 +53,12 @@ import com.zaneschepke.wireguardautotunnel.ui.common.label.GroupLabel
import com.zaneschepke.wireguardautotunnel.ui.common.text.DescriptionText import com.zaneschepke.wireguardautotunnel.ui.common.text.DescriptionText
import com.zaneschepke.wireguardautotunnel.ui.navigation.Route import com.zaneschepke.wireguardautotunnel.ui.navigation.Route
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.components.BackupBottomSheet 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.ui.screens.settings.proxy.compoents.AppModeBottomSheet
import com.zaneschepke.wireguardautotunnel.util.StringValue import com.zaneschepke.wireguardautotunnel.util.StringValue
import com.zaneschepke.wireguardautotunnel.util.extensions.asString import com.zaneschepke.wireguardautotunnel.util.extensions.asString
import com.zaneschepke.wireguardautotunnel.util.extensions.asTitleString import com.zaneschepke.wireguardautotunnel.util.extensions.asTitleString
import com.zaneschepke.wireguardautotunnel.util.extensions.capitalize 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.SettingsViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel
import org.koin.androidx.compose.koinViewModel import org.koin.androidx.compose.koinViewModel
@@ -88,6 +90,9 @@ fun SettingsScreen(
} }
var showBackupSheet by rememberSaveable { mutableStateOf(false) } var showBackupSheet by rememberSaveable { mutableStateOf(false) }
var showEncryptionDialog by rememberSaveable { mutableStateOf(false) }
var isRestoreAction by remember { mutableStateOf(false) }
var showAppModeSheet by rememberSaveable { mutableStateOf(false) } var showAppModeSheet by rememberSaveable { mutableStateOf(false) }
val appMode = uiState.settings.tunnelMode val appMode = uiState.settings.tunnelMode
@@ -99,19 +104,53 @@ fun SettingsScreen(
} }
fun performBackupRestore(action: () -> Unit) { fun performBackupRestore(action: () -> Unit) {
if (uiState.tunnelActive || globalUiState.isAutoTunnelActive)
return context.showToast(R.string.all_services_disabled)
showBackupSheet = false showBackupSheet = false
if (uiState.tunnelActive || globalUiState.isAutoTunnelActive) {
sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.all_services_disabled),
ToastType.Warning,
)
return
}
action() action()
} }
if (showBackupSheet) if (showBackupSheet) {
BackupBottomSheet( BackupBottomSheet(
{ performBackupRestore { (context as? MainActivity)?.performBackup() } }, onBackup = {
{ performBackupRestore { (context as? MainActivity)?.performRestore() } }, showBackupSheet = false
) { isRestoreAction = false
showBackupSheet = 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 },
)
}
if (showAppModeSheet) if (showAppModeSheet)
AppModeBottomSheet(sharedViewModel::setAppMode, uiState.settings.tunnelMode) { AppModeBottomSheet(sharedViewModel::setAppMode, uiState.settings.tunnelMode) {
showAppModeSheet = false showAppModeSheet = false
@@ -168,7 +207,8 @@ fun SettingsScreen(
StringValue.StringResource( StringValue.StringResource(
R.string.mode_disabled_template, R.string.mode_disabled_template,
appMode.asString(context), appMode.asString(context),
) ),
ToastType.Info,
) )
}, },
) )
@@ -195,6 +235,23 @@ fun SettingsScreen(
viewModel.setTunnelScriptedEnabled(!uiState.settings.tunnelScriptingEnabled) 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( SurfaceRow(
leading = { Icon(Icons.Outlined.MonitorHeart, null) }, leading = { Icon(Icons.Outlined.MonitorHeart, null) },
title = stringResource(R.string.tunnel_monitoring), title = stringResource(R.string.tunnel_monitoring),
@@ -0,0 +1,146 @@
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,
)
}
}
}
},
)
}
@@ -12,6 +12,7 @@ import androidx.compose.runtime.setValue
import androidx.compose.runtime.snapshotFlow import androidx.compose.runtime.snapshotFlow
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.stringResource
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.logs.components.LogList import com.zaneschepke.wireguardautotunnel.ui.screens.settings.logs.components.LogList
import com.zaneschepke.wireguardautotunnel.ui.screens.settings.logs.components.LogsBottomSheet import com.zaneschepke.wireguardautotunnel.ui.screens.settings.logs.components.LogsBottomSheet
@@ -86,13 +87,15 @@ fun LogsScreen(
}, },
onCanceled = { onCanceled = {
sharedViewModel.showSnackMessage( sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.export_canceled) StringValue.StringResource(R.string.export_canceled),
ToastType.Warning,
) )
showLogsSheet = false showLogsSheet = false
}, },
onUnsupported = { onUnsupported = {
sharedViewModel.showSnackMessage( sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.export_unsupported) StringValue.StringResource(R.string.export_unsupported),
ToastType.Warning,
) )
showLogsSheet = false showLogsSheet = false
}, },
@@ -1,10 +1,12 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.settings.logs.components 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.Arrangement
import androidx.compose.foundation.layout.padding import androidx.compose.foundation.layout.padding
import androidx.compose.foundation.lazy.LazyColumn import androidx.compose.foundation.lazy.LazyColumn
import androidx.compose.foundation.lazy.LazyListState import androidx.compose.foundation.lazy.LazyListState
import androidx.compose.foundation.lazy.itemsIndexed import androidx.compose.foundation.lazy.itemsIndexed
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
@@ -18,7 +20,13 @@ fun LogList(
) { ) {
LazyColumn( LazyColumn(
state = lazyColumnListState, state = lazyColumnListState,
modifier = modifier.padding(horizontal = 12.dp), modifier =
modifier
.padding(horizontal = 12.dp)
.scrollbar(
state = lazyColumnListState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
verticalArrangement = Arrangement.spacedBy(2.dp), verticalArrangement = Arrangement.spacedBy(2.dp),
) { ) {
itemsIndexed(items = logs, key = { index, _ -> index }) { _, log -> LogItem(log = log) } itemsIndexed(items = logs, key = { index, _ -> index }) { _, log -> LogItem(log = log) }
@@ -26,6 +26,7 @@ import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.getValue import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.remember import androidx.compose.runtime.remember
import androidx.compose.runtime.rememberCoroutineScope
import androidx.compose.runtime.saveable.rememberSaveable import androidx.compose.runtime.saveable.rememberSaveable
import androidx.compose.runtime.setValue import androidx.compose.runtime.setValue
import androidx.compose.ui.Alignment import androidx.compose.ui.Alignment
@@ -37,8 +38,10 @@ import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.stringResource
import androidx.compose.ui.res.vectorResource import androidx.compose.ui.res.vectorResource
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.BuildConfig import com.zaneschepke.wireguardautotunnel.BuildConfig
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.domain.sideeffect.GlobalSideEffect
import com.zaneschepke.wireguardautotunnel.ui.LocalIsAndroidTV import com.zaneschepke.wireguardautotunnel.ui.LocalIsAndroidTV
import com.zaneschepke.wireguardautotunnel.ui.LocalNavController import com.zaneschepke.wireguardautotunnel.ui.LocalNavController
import com.zaneschepke.wireguardautotunnel.ui.common.button.SurfaceRow import com.zaneschepke.wireguardautotunnel.ui.common.button.SurfaceRow
@@ -49,12 +52,13 @@ import com.zaneschepke.wireguardautotunnel.ui.navigation.Route
import com.zaneschepke.wireguardautotunnel.ui.screens.support.components.PermissionDialog import com.zaneschepke.wireguardautotunnel.ui.screens.support.components.PermissionDialog
import com.zaneschepke.wireguardautotunnel.ui.screens.support.components.UpdateDialog import com.zaneschepke.wireguardautotunnel.ui.screens.support.components.UpdateDialog
import com.zaneschepke.wireguardautotunnel.util.Constants 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.launchPlayStoreListing
import com.zaneschepke.wireguardautotunnel.util.extensions.launchPlayStoreReview import com.zaneschepke.wireguardautotunnel.util.extensions.launchPlayStoreReview
import com.zaneschepke.wireguardautotunnel.util.extensions.launchSupportEmail import com.zaneschepke.wireguardautotunnel.util.extensions.launchSupportEmail
import com.zaneschepke.wireguardautotunnel.util.extensions.openWebUrl import com.zaneschepke.wireguardautotunnel.util.extensions.openWebUrl
import com.zaneschepke.wireguardautotunnel.util.extensions.showToast
import com.zaneschepke.wireguardautotunnel.viewmodel.SupportViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.SupportViewModel
import kotlinx.coroutines.launch
import org.koin.androidx.compose.koinViewModel import org.koin.androidx.compose.koinViewModel
import org.orbitmvi.orbit.compose.collectAsState import org.orbitmvi.orbit.compose.collectAsState
@@ -63,11 +67,24 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
val context = LocalContext.current val context = LocalContext.current
val navController = LocalNavController.current val navController = LocalNavController.current
val isTv = LocalIsAndroidTV.current val isTv = LocalIsAndroidTV.current
val scope = rememberCoroutineScope()
val supportState by viewModel.collectAsState() val supportState by viewModel.collectAsState()
val clipboardManager = rememberClipboardHelper() 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 { val version = remember {
"v${BuildConfig.VERSION_NAME + "v${BuildConfig.VERSION_NAME +
if(BuildConfig.DEBUG) "-debug" else "" }" if(BuildConfig.DEBUG) "-debug" else "" }"
@@ -95,6 +112,19 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
PermissionDialog(context = context, onDismiss = { showPermissionDialog = false }) 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( Column(
modifier = Modifier.fillMaxSize().verticalScroll(rememberScrollState()), modifier = Modifier.fillMaxSize().verticalScroll(rememberScrollState()),
horizontalAlignment = Alignment.Start, horizontalAlignment = Alignment.Start,
@@ -115,19 +145,19 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
) )
SurfaceRow( SurfaceRow(
stringResource(R.string.docs_description), stringResource(R.string.docs_description),
onClick = { context.openWebUrl(context.getString(R.string.docs_url)) }, onClick = { openWebUrl(docsUrl) },
leading = { Icon(Icons.Outlined.Book, contentDescription = null) }, leading = { Icon(Icons.Outlined.Book, contentDescription = null) },
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) }, trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
) )
SurfaceRow( SurfaceRow(
stringResource(R.string.website), stringResource(R.string.website),
onClick = { context.openWebUrl(context.getString(R.string.website_url)) }, onClick = { openWebUrl(websiteUrl) },
leading = { Icon(Icons.Outlined.Web, contentDescription = null) }, leading = { Icon(Icons.Outlined.Web, contentDescription = null) },
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) }, trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
) )
SurfaceRow( SurfaceRow(
stringResource(R.string.translation), stringResource(R.string.translation),
onClick = { context.openWebUrl(context.getString(R.string.translation_url)) }, onClick = { openWebUrl(translationUrl) },
description = { DescriptionText(stringResource(R.string.help_translate)) }, description = { DescriptionText(stringResource(R.string.help_translate)) },
leading = { Icon(Icons.Outlined.Translate, contentDescription = null) }, leading = { Icon(Icons.Outlined.Translate, contentDescription = null) },
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) }, trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
@@ -141,14 +171,16 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
leading = { Icon(Icons.Outlined.Policy, contentDescription = null) }, leading = { Icon(Icons.Outlined.Policy, contentDescription = null) },
title = stringResource(R.string.privacy_policy), title = stringResource(R.string.privacy_policy),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) }, trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = { context.openWebUrl(context.getString(R.string.privacy_policy_url)) }, onClick = { openWebUrl(privacyPolicyUrl) },
) )
if (BuildConfig.FLAVOR == Constants.GOOGLE_PLAY_FLAVOR) { if (BuildConfig.FLAVOR == Constants.GOOGLE_PLAY_FLAVOR) {
SurfaceRow( SurfaceRow(
leading = { Icon(Icons.Outlined.Reviews, contentDescription = null) }, leading = { Icon(Icons.Outlined.Reviews, contentDescription = null) },
title = stringResource(R.string.review), title = stringResource(R.string.review),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) }, trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = { context.launchPlayStoreReview() }, onClick = {
context.launchPlayStoreReview().onFailure { openWebUrl(playReviewsUrl) }
},
) )
} }
} }
@@ -167,7 +199,7 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
}, },
title = stringResource(R.string.join_matrix), title = stringResource(R.string.join_matrix),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) }, trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = { context.openWebUrl(context.getString(R.string.matrix_url)) }, onClick = { openWebUrl(matrixUrl) },
) )
SurfaceRow( SurfaceRow(
leading = { leading = {
@@ -179,7 +211,7 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
}, },
title = stringResource(R.string.join_telegram), title = stringResource(R.string.join_telegram),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) }, trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = { context.openWebUrl(context.getString(R.string.telegram_url)) }, onClick = { openWebUrl(telegramUrl) },
) )
SurfaceRow( SurfaceRow(
leading = { leading = {
@@ -191,13 +223,24 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
}, },
title = stringResource(R.string.open_issue), title = stringResource(R.string.open_issue),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) }, trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = { context.openWebUrl(context.getString(R.string.github_url)) }, onClick = { openWebUrl(issuesUrl) },
) )
SurfaceRow( SurfaceRow(
leading = { Icon(Icons.Outlined.Mail, contentDescription = null) }, leading = { Icon(Icons.Outlined.Mail, contentDescription = null) },
title = stringResource(R.string.email_description), title = stringResource(R.string.email_description),
trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) }, trailing = { Icon(Icons.AutoMirrored.Outlined.Launch, null) },
onClick = { context.launchSupportEmail() }, onClick = {
context.launchSupportEmail().onFailure {
scope.launch {
viewModel.postSideEffect(
GlobalSideEffect.Snackbar(
StringValue.StringResource(R.string.no_email_detected),
ToastType.Error,
)
)
}
}
},
) )
} }
Column { Column {
@@ -222,12 +265,21 @@ fun SupportScreen(viewModel: SupportViewModel = koinViewModel()) {
leading = { Icon(Icons.Outlined.InstallMobile, contentDescription = null) }, leading = { Icon(Icons.Outlined.InstallMobile, contentDescription = null) },
title = stringResource(R.string.check_for_update), title = stringResource(R.string.check_for_update),
onClick = { onClick = {
if (BuildConfig.DEBUG) if (BuildConfig.DEBUG) {
return@SurfaceRow context.showToast(R.string.update_check_unsupported) scope.launch {
viewModel.postSideEffect(
GlobalSideEffect.Snackbar(
StringValue.StringResource(R.string.update_check_unsupported),
ToastType.Warning,
)
)
}
return@SurfaceRow
}
when (BuildConfig.FLAVOR) { when (BuildConfig.FLAVOR) {
Constants.GOOGLE_PLAY_FLAVOR -> context.launchPlayStoreListing() Constants.GOOGLE_PLAY_FLAVOR ->
Constants.FDROID_FLAVOR -> context.launchPlayStoreListing().onFailure { openWebUrl(playStoreUrl) }
context.openWebUrl(context.getString(R.string.fdroid_url)) Constants.FDROID_FLAVOR -> openWebUrl(izzyUrl)
else -> viewModel.checkForStandaloneUpdate() else -> viewModel.checkForStandaloneUpdate()
} }
}, },
@@ -1,10 +1,12 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.support.donate.crypto 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.Arrangement
import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.fillMaxSize
import androidx.compose.foundation.rememberScrollState import androidx.compose.foundation.rememberScrollState
import androidx.compose.foundation.verticalScroll import androidx.compose.foundation.verticalScroll
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.compose.ui.Alignment import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
@@ -13,10 +15,17 @@ import com.zaneschepke.wireguardautotunnel.ui.screens.support.donate.crypto.comp
@Composable @Composable
fun AddressesScreen() { fun AddressesScreen() {
val scrollState = rememberScrollState()
Column( Column(
horizontalAlignment = Alignment.Start, horizontalAlignment = Alignment.Start,
verticalArrangement = Arrangement.Top, verticalArrangement = Arrangement.Top,
modifier = Modifier.fillMaxSize().verticalScroll(rememberScrollState()), modifier =
Modifier.fillMaxSize()
.verticalScroll(scrollState)
.scrollbar(
state = scrollState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
) { ) {
val clipboard = rememberClipboardHelper() val clipboard = rememberClipboardHelper()
Address.allAddresses.forEach { AddressItem(it) { address -> clipboard.copy(address) } } Address.allAddresses.forEach { AddressItem(it) { address -> clipboard.copy(address) } }
@@ -0,0 +1,213 @@
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 },
)
}
@@ -1,15 +0,0 @@
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,30 +1,23 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.support.license 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.fillMaxSize
import androidx.compose.foundation.layout.size
import androidx.compose.material3.CircularWavyProgressIndicator
import androidx.compose.material3.ExperimentalMaterial3ExpressiveApi import androidx.compose.material3.ExperimentalMaterial3ExpressiveApi
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.compose.runtime.getValue import androidx.compose.runtime.remember
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.unit.dp import androidx.compose.ui.platform.LocalContext
import com.zaneschepke.wireguardautotunnel.ui.screens.support.license.components.LicenseList import com.mikepenz.aboutlibraries.ui.compose.m3.LibrariesContainer
import com.zaneschepke.wireguardautotunnel.viewmodel.LicenseViewModel import com.mikepenz.aboutlibraries.ui.compose.variant.LibraryDetailMode
import org.koin.androidx.compose.koinViewModel
import org.orbitmvi.orbit.compose.collectAsState
@OptIn(ExperimentalMaterial3ExpressiveApi::class) @OptIn(ExperimentalMaterial3ExpressiveApi::class)
@Composable @Composable
fun LicenseScreen(viewModel: LicenseViewModel = koinViewModel()) { fun LicenseScreen() {
val licenseUiState by viewModel.collectAsState() val context = LocalContext.current
val libs = remember { buildLibsWithAdditionalLibraries(context) }
if (licenseUiState.isLoading) { LibrariesContainer(
Box(Modifier.fillMaxSize(), contentAlignment = Alignment.Center) { libraries = libs,
CircularWavyProgressIndicator(waveSpeed = 60.dp, modifier = Modifier.size(48.dp)) modifier = Modifier.fillMaxSize(),
} detailMode = LibraryDetailMode.Sheet,
} else { )
LicenseList(licenseUiState.licenses)
}
} }
@@ -1,57 +0,0 @@
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,6 +13,7 @@ import androidx.compose.ui.Modifier
import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.platform.LocalContext
import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.stringResource
import androidx.lifecycle.compose.collectAsStateWithLifecycle import androidx.lifecycle.compose.collectAsStateWithLifecycle
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.ui.LocalNavController import com.zaneschepke.wireguardautotunnel.ui.LocalNavController
import com.zaneschepke.wireguardautotunnel.ui.common.dialog.InfoDialog import com.zaneschepke.wireguardautotunnel.ui.common.dialog.InfoDialog
@@ -50,11 +51,15 @@ fun TunnelsScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
rememberFileExportLauncherForResult( rememberFileExportLauncherForResult(
onSuccess = { uri -> sharedViewModel.exportSelectedTunnels(uri) }, onSuccess = { uri -> sharedViewModel.exportSelectedTunnels(uri) },
onCanceled = { onCanceled = {
sharedViewModel.showToast(StringValue.StringResource(R.string.export_canceled)) sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.export_canceled),
ToastType.Warning,
)
}, },
onUnsupported = { onUnsupported = {
sharedViewModel.showSnackMessage( sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.export_unsupported) StringValue.StringResource(R.string.export_unsupported),
ToastType.Warning,
) )
}, },
) )
@@ -73,7 +78,8 @@ fun TunnelsScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
selectedTunnelsExportLauncher.launch(fileName) selectedTunnelsExportLauncher.launch(fileName)
} else { } else {
sharedViewModel.showSnackMessage( sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.error_no_file_explorer) StringValue.StringResource(R.string.error_no_file_explorer),
ToastType.Error,
) )
} }
} }
@@ -87,7 +93,8 @@ fun TunnelsScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
rememberFileImportLauncherForResult( rememberFileImportLauncherForResult(
onNoFileExplorer = { onNoFileExplorer = {
sharedViewModel.showSnackMessage( sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.error_no_file_explorer) StringValue.StringResource(R.string.error_no_file_explorer),
ToastType.Error,
) )
}, },
onData = { data -> sharedViewModel.importFromUri(data) }, onData = { data -> sharedViewModel.importFromUri(data) },
@@ -101,13 +108,15 @@ fun TunnelsScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
} }
QRResult.QRMissingPermission -> { QRResult.QRMissingPermission -> {
sharedViewModel.showSnackMessage( sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.camera_permission_required) StringValue.StringResource(R.string.camera_permission_required),
ToastType.Warning,
) )
} }
is QRResult.QRSuccess -> { is QRResult.QRSuccess -> {
result.content.rawValue?.let { sharedViewModel.importFromQr(it) } result.content.rawValue?.let { sharedViewModel.importFromQr(it) }
?: sharedViewModel.showSnackMessage( ?: sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.config_error) StringValue.StringResource(R.string.config_error),
ToastType.Error,
) )
} }
QRResult.QRUserCanceled -> Unit QRResult.QRUserCanceled -> Unit
@@ -119,7 +128,8 @@ fun TunnelsScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel())
-> ->
if (!isGranted) { if (!isGranted) {
sharedViewModel.showSnackMessage( sharedViewModel.showSnackMessage(
StringValue.StringResource(R.string.camera_permission_required) StringValue.StringResource(R.string.camera_permission_required),
ToastType.Warning,
) )
return@rememberLauncherForActivityResult return@rememberLauncherForActivityResult
} }
@@ -42,5 +42,8 @@ fun PeerStatisticsSection(peer: ActivePeer) {
style = style, style = style,
color = color, 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), modifier = Modifier.size(12.dp),
) )
Text(text = text.lowercase(), style = style, color = color) Text(text = text, style = style, color = color)
} }
} }
@@ -1,6 +1,7 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.components package com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.components
import androidx.compose.foundation.ExperimentalFoundationApi import androidx.compose.foundation.ExperimentalFoundationApi
import androidx.compose.foundation.gestures.Orientation
import androidx.compose.foundation.gestures.ScrollableDefaults import androidx.compose.foundation.gestures.ScrollableDefaults
import androidx.compose.foundation.gestures.detectTapGestures import androidx.compose.foundation.gestures.detectTapGestures
import androidx.compose.foundation.layout.size import androidx.compose.foundation.layout.size
@@ -12,10 +13,10 @@ import androidx.compose.foundation.rememberOverscrollEffect
import androidx.compose.material.icons.Icons import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.rounded.Circle import androidx.compose.material.icons.rounded.Circle
import androidx.compose.material3.Icon import androidx.compose.material3.Icon
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.compose.runtime.LaunchedEffect import androidx.compose.runtime.LaunchedEffect
import androidx.compose.runtime.getValue import androidx.compose.runtime.getValue
import androidx.compose.runtime.produceState
import androidx.compose.runtime.remember import androidx.compose.runtime.remember
import androidx.compose.ui.Alignment import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
@@ -36,8 +37,6 @@ import com.zaneschepke.wireguardautotunnel.ui.state.DisplayTunnelState
import com.zaneschepke.wireguardautotunnel.ui.state.TunnelsUiState import com.zaneschepke.wireguardautotunnel.ui.state.TunnelsUiState
import com.zaneschepke.wireguardautotunnel.util.extensions.openWebUrl import com.zaneschepke.wireguardautotunnel.util.extensions.openWebUrl
import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel
import kotlin.time.Duration.Companion.milliseconds
import kotlinx.coroutines.delay
@OptIn(ExperimentalFoundationApi::class) @OptIn(ExperimentalFoundationApi::class)
@Composable @Composable
@@ -50,14 +49,6 @@ fun TunnelList(
val context = LocalContext.current val context = LocalContext.current
val isTv = LocalIsAndroidTV.current val isTv = LocalIsAndroidTV.current
val now by
produceState(System.currentTimeMillis()) {
while (true) {
delay(1_000L.milliseconds)
value = System.currentTimeMillis()
}
}
val focusRequester = remember { FocusRequester() } val focusRequester = remember { FocusRequester() }
LaunchedEffect(Unit) { LaunchedEffect(Unit) {
@@ -78,7 +69,11 @@ fun TunnelList(
viewModel.clearSelectedTunnels() viewModel.clearSelectedTunnels()
} }
} }
.overscroll(rememberOverscrollEffect()), .overscroll(rememberOverscrollEffect())
.scrollbar(
state = lazyListState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
state = lazyListState, state = lazyListState,
userScrollEnabled = true, userScrollEnabled = true,
reverseLayout = false, reverseLayout = false,
@@ -99,10 +94,7 @@ fun TunnelList(
uiState.backendStatus.activeTunnels[tunnel.id] ?: ActiveTunnel() uiState.backendStatus.activeTunnels[tunnel.id] ?: ActiveTunnel()
} }
val displayState = val displayState = remember(activeTunnel) { DisplayTunnelState.from(activeTunnel) }
remember(activeTunnel, now, uiState.displayStates[tunnel.id]) {
uiState.displayStates[tunnel.id] ?: DisplayTunnelState.from(activeTunnel, now)
}
val isRunning = uiState.backendStatus.activeTunnels.containsKey(tunnel.id) val isRunning = uiState.backendStatus.activeTunnels.containsKey(tunnel.id)
@@ -1,5 +1,6 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.settings.config 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.Arrangement
import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.fillMaxSize import androidx.compose.foundation.layout.fillMaxSize
@@ -9,6 +10,7 @@ import androidx.compose.foundation.text.selection.SelectionContainer
import androidx.compose.foundation.verticalScroll import androidx.compose.foundation.verticalScroll
import androidx.compose.material3.MaterialTheme import androidx.compose.material3.MaterialTheme
import androidx.compose.material3.Text import androidx.compose.material3.Text
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.compose.runtime.derivedStateOf import androidx.compose.runtime.derivedStateOf
import androidx.compose.runtime.getValue import androidx.compose.runtime.getValue
@@ -26,14 +28,16 @@ import androidx.compose.ui.text.font.FontFamily
import androidx.compose.ui.text.font.FontStyle import androidx.compose.ui.text.font.FontStyle
import androidx.compose.ui.text.font.FontWeight import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.unit.dp import androidx.compose.ui.unit.dp
import com.dokar.sonner.ToastType
import com.zaneschepke.wireguardautotunnel.R 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.common.functions.rememberClipboardHelper
import com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.settings.config.components.QrCodeDialog import com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.settings.config.components.QrCodeDialog
import com.zaneschepke.wireguardautotunnel.ui.sideeffect.LocalSideEffect import com.zaneschepke.wireguardautotunnel.ui.sideeffect.LocalSideEffect
import com.zaneschepke.wireguardautotunnel.ui.theme.ConfigHeaderColor import com.zaneschepke.wireguardautotunnel.ui.theme.ConfigHeaderColor
import com.zaneschepke.wireguardautotunnel.ui.theme.ConfigKeyColor 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.isTextTooLargeForQr
import com.zaneschepke.wireguardautotunnel.util.extensions.showToast
import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.SharedAppViewModel
import com.zaneschepke.wireguardautotunnel.viewmodel.TunnelViewModel import com.zaneschepke.wireguardautotunnel.viewmodel.TunnelViewModel
import org.koin.compose.viewmodel.koinActivityViewModel import org.koin.compose.viewmodel.koinActivityViewModel
@@ -57,6 +61,8 @@ fun ConfigScreen(
var showQrModal by rememberSaveable { mutableStateOf(false) } var showQrModal by rememberSaveable { mutableStateOf(false) }
val scrollState = rememberScrollState()
val rawConfig by val rawConfig by
remember(liveConfig, uiState.activeConfig, uiState.tunnel?.quickConfig) { remember(liveConfig, uiState.activeConfig, uiState.tunnel?.quickConfig) {
derivedStateOf { derivedStateOf {
@@ -72,7 +78,12 @@ fun ConfigScreen(
when (sideEffect) { when (sideEffect) {
is LocalSideEffect.Modal.QR -> { is LocalSideEffect.Modal.QR -> {
if (tunnel.quickConfig.isTextTooLargeForQr()) { if (tunnel.quickConfig.isTextTooLargeForQr()) {
context.showToast(R.string.text_too_large_for_qr) sharedViewModel.postSideEffect(
GlobalSideEffect.Snackbar(
StringValue.StringResource(R.string.text_too_large_for_qr),
ToastType.Error,
)
)
} else { } else {
showQrModal = true showQrModal = true
} }
@@ -90,7 +101,13 @@ fun ConfigScreen(
Column( Column(
horizontalAlignment = Alignment.Start, horizontalAlignment = Alignment.Start,
verticalArrangement = Arrangement.spacedBy(12.dp, Alignment.Top), verticalArrangement = Arrangement.spacedBy(12.dp, Alignment.Top),
modifier = Modifier.fillMaxSize().verticalScroll(rememberScrollState()), modifier =
Modifier.fillMaxSize()
.verticalScroll(scrollState)
.scrollbar(
state = scrollState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
) { ) {
val displayText by val displayText by
remember(rawConfig, showKeys) { derivedStateOf { maskSensitive(rawConfig, showKeys) } } remember(rawConfig, showKeys) { derivedStateOf { maskSensitive(rawConfig, showKeys) } }
@@ -1,5 +1,6 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.settings.config.edit 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.Arrangement
import androidx.compose.foundation.layout.Column import androidx.compose.foundation.layout.Column
import androidx.compose.foundation.layout.Spacer import androidx.compose.foundation.layout.Spacer
@@ -12,6 +13,7 @@ import androidx.compose.material.icons.Icons
import androidx.compose.material.icons.outlined.HdrAuto import androidx.compose.material.icons.outlined.HdrAuto
import androidx.compose.material3.Icon import androidx.compose.material3.Icon
import androidx.compose.material3.Text import androidx.compose.material3.Text
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.compose.runtime.getValue import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.mutableStateOf
@@ -49,11 +51,12 @@ fun ConfigEditScreen(
val uiState by viewModel.collectAsState() val uiState by viewModel.collectAsState()
if (uiState.isLoading) return if (uiState.isLoading) return
val locale = Locale.current.platformLocale val locale = Locale.current.platformLocale
var showSelectionDialog by rememberSaveable { mutableStateOf(false) } var showSelectionDialog by rememberSaveable { mutableStateOf(false) }
val scrollState = rememberScrollState()
sharedViewModel.collectSideEffect { sideEffect -> sharedViewModel.collectSideEffect { sideEffect ->
when (sideEffect) { when (sideEffect) {
is LocalSideEffect.SaveChanges -> { is LocalSideEffect.SaveChanges -> {
@@ -104,7 +107,14 @@ fun ConfigEditScreen(
Column( Column(
horizontalAlignment = Alignment.Start, horizontalAlignment = Alignment.Start,
verticalArrangement = Arrangement.spacedBy(16.dp, Alignment.Top), verticalArrangement = Arrangement.spacedBy(16.dp, Alignment.Top),
modifier = Modifier.fillMaxSize().imePadding().verticalScroll(rememberScrollState()), modifier =
Modifier.fillMaxSize()
.imePadding()
.verticalScroll(scrollState)
.scrollbar(
state = scrollState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
) { ) {
if (uiState.isGlobalConfig) { if (uiState.isGlobalConfig) {
Column { Column {
@@ -1,5 +1,6 @@
package com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.sort package com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.sort
import androidx.compose.foundation.gestures.Orientation
import androidx.compose.foundation.gestures.ScrollableDefaults import androidx.compose.foundation.gestures.ScrollableDefaults
import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Arrangement
import androidx.compose.foundation.layout.Row import androidx.compose.foundation.layout.Row
@@ -17,6 +18,7 @@ import androidx.compose.material.icons.filled.ArrowUpward
import androidx.compose.material.icons.filled.DragHandle import androidx.compose.material.icons.filled.DragHandle
import androidx.compose.material3.Icon import androidx.compose.material3.Icon
import androidx.compose.material3.IconButton import androidx.compose.material3.IconButton
import androidx.compose.material3.scrollbar
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.compose.runtime.getValue import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.mutableStateOf
@@ -107,7 +109,11 @@ fun SortScreen(sharedViewModel: SharedAppViewModel = koinActivityViewModel()) {
Modifier.pointerInput(Unit) { Modifier.pointerInput(Unit) {
if (tunnelsUiState.tunnels.isEmpty()) return@pointerInput if (tunnelsUiState.tunnels.isEmpty()) return@pointerInput
} }
.overscroll(rememberOverscrollEffect()), .overscroll(rememberOverscrollEffect())
.scrollbar(
state = lazyListState.scrollIndicatorState,
orientation = Orientation.Vertical,
),
state = lazyListState, state = lazyListState,
userScrollEnabled = true, userScrollEnabled = true,
reverseLayout = false, reverseLayout = false,
@@ -26,97 +26,57 @@ sealed class DisplayTunnelState {
data object Connected : DisplayTunnelState() data object Connected : DisplayTunnelState()
data object Degraded : DisplayTunnelState() data object HandshakeFailure : DisplayTunnelState()
@StringRes @StringRes
fun labelRes(): Int { fun labelRes(): Int =
return when (this) { when (this) {
Disconnected -> R.string.tunnel_state_disconnected Disconnected -> R.string.tunnel_state_disconnected
Connecting -> R.string.tunnel_state_starting Connecting,
ResolvingDns -> R.string.tunnel_state_resolving_dns
EstablishingConnection -> R.string.tunnel_state_establishing_connection EstablishingConnection -> R.string.tunnel_state_establishing_connection
ResolvingDns -> R.string.tunnel_state_resolving_dns
Ready -> R.string.ready Ready -> R.string.ready
Connected -> R.string.tunnel_state_connected Connected -> R.string.tunnel_state_connected
Degraded -> R.string.tunnel_state_handshake_failure HandshakeFailure -> R.string.tunnel_state_handshake_failure
}
fun asColor(): Color =
when (this) {
Disconnected -> CoolGray
Connecting,
ResolvingDns,
EstablishingConnection,
Ready -> Straw
Connected -> SilverTree
HandshakeFailure -> AlertRed
} }
}
fun asLocalizedString(context: Context): String { fun asLocalizedString(context: Context): String {
return context.getString(labelRes()) return context.getString(labelRes())
} }
fun asColor(): Color {
return when (this) {
Disconnected -> CoolGray
Connecting,
ResolvingDns,
EstablishingConnection,
Ready -> Straw
Connected -> SilverTree
Degraded -> AlertRed
}
}
companion object { companion object {
private const val HANDSHAKE_FAILURE_DEGRADED_THRESHOLD_MS = 6_000L fun from(activeTunnel: ActiveTunnel): DisplayTunnelState {
// During this window we avoid showing Degraded even if we see HandshakeFailure
private const val POST_RESOLUTION_GRACE_PERIOD_MS = 3_500L
fun from(activeTunnel: ActiveTunnel, now: Long): DisplayTunnelState {
val transport = activeTunnel.transportState val transport = activeTunnel.transportState
val bootstrap = activeTunnel.bootstrapState val bootstrap = activeTunnel.bootstrapState
val mode = activeTunnel.mode val mode = activeTunnel.mode
val isVpnStyle = mode is BackendMode.Vpn || mode is BackendMode.Proxy.KillSwitchPrimary val isVpnStyle = mode is BackendMode.Vpn || mode is BackendMode.Proxy.KillSwitchPrimary
val bootstrapPhaseDone =
bootstrap is BootstrapState.Complete || bootstrap is BootstrapState.None
// Check if we recently completed peer resolution
val recentlyResolvedPeers =
activeTunnel.lastPeerUpdateMs > 0 &&
(now - activeTunnel.lastPeerUpdateMs) < POST_RESOLUTION_GRACE_PERIOD_MS
return when { return when {
transport is Tunnel.State.Down -> Disconnected transport is Tunnel.State.Down -> Disconnected
bootstrap is BootstrapState.Failed -> HandshakeFailure
bootstrap is BootstrapState.Failed -> Degraded
bootstrap is BootstrapState.ResolvingDns || bootstrap is BootstrapState.ResolvingDns ||
bootstrap is BootstrapState.UpdatingPeers -> ResolvingDns bootstrap is BootstrapState.UpdatingPeers -> ResolvingDns
transport is Tunnel.State.Up.Healthy -> Connected transport is Tunnel.State.Up.Healthy -> Connected
transport is Tunnel.State.Up.HandshakeFailure -> { transport is Tunnel.State.Up.HandshakeFailure -> HandshakeFailure
val age = now - activeTunnel.lastStateChangeMs
if (recentlyResolvedPeers && bootstrapPhaseDone) { transport is Tunnel.State.Starting ->
if (isVpnStyle) EstablishingConnection else Ready if (isVpnStyle) EstablishingConnection else Ready
} else if (
age > HANDSHAKE_FAILURE_DEGRADED_THRESHOLD_MS && bootstrapPhaseDone
) {
Degraded
} else if (isVpnStyle && bootstrapPhaseDone) {
EstablishingConnection
} else if (bootstrapPhaseDone) {
Ready
} else {
Connecting
}
}
transport is Tunnel.State.Starting -> { else -> if (isVpnStyle) EstablishingConnection else Ready
when {
bootstrapPhaseDone -> if (isVpnStyle) EstablishingConnection else Ready
else -> Connecting
}
}
bootstrapPhaseDone -> if (isVpnStyle) EstablishingConnection else Ready
else -> Connecting
} }
} }
} }
@@ -18,5 +18,6 @@ data class GlobalAppUiState(
val selectedTunnelCount: Int = 0, val selectedTunnelCount: Int = 0,
val alreadyDonated: Boolean = false, val alreadyDonated: Boolean = false,
val isPinVerified: Boolean = false, val isPinVerified: Boolean = false,
val pendingWgImportUrl: String? = null,
val isScreenRecordingProtectionEnabled: Boolean = false, val isScreenRecordingProtectionEnabled: Boolean = false,
) )
@@ -1,8 +0,0 @@
package com.zaneschepke.wireguardautotunnel.ui.state
import LicenseFileEntry
data class LicenseUiState(
val isLoading: Boolean = true,
val licenses: List<LicenseFileEntry> = emptyList(),
)
@@ -16,14 +16,14 @@ val ElectricTeal = Color(0xFF4DD0E1)
// Status colors // Status colors
val SilverTree = Color(0xFF6DB58B) val SilverTree = Color(0xFF6DB58B)
val AlertRed = Color(0xFFCF6679) val AlertRed = Color(0xFFCF6679)
val Straw = Color(0xFFD4C483) val Straw = Color(0xFFD4C483)
val Disabled = CoolGray.copy(alpha = 0.4f) val Disabled = CoolGray.copy(alpha = 0.4f)
// Config colors // Other colors
val ConfigHeaderColor = Color(0xFFBB86FC) val ConfigHeaderColor = Color(0xFFBB86FC)
val ConfigKeyColor = Color(0xFF03DAC5) val ConfigKeyColor = Color(0xFF03DAC5)
val Heart = Color(0xFFDB61A2)
sealed class ThemeColors( sealed class ThemeColors(
val background: Color, val background: Color,
@@ -6,15 +6,11 @@ object Constants {
const val BASE_LOG_FILE_NAME = "wg_tunnel_logs" const val BASE_LOG_FILE_NAME = "wg_tunnel_logs"
const val VPN_SETTINGS_PACKAGE = "android.net.vpn.SETTINGS" const val VPN_SETTINGS_PACKAGE = "android.net.vpn.SETTINGS"
const val SYSTEM_EXEMPT_SERVICE_TYPE_ID = 1 shl 10
const val SPECIAL_USE_SERVICE_TYPE_ID = 1 shl 30 const val SPECIAL_USE_SERVICE_TYPE_ID = 1 shl 30
const val QR_CODE_NAME_PROPERTY = "# Name ="
const val FDROID_FLAVOR = "fdroid" const val FDROID_FLAVOR = "fdroid"
const val GOOGLE_PLAY_FLAVOR = "google" const val GOOGLE_PLAY_FLAVOR = "google"
const val STANDALONE_FLAVOR = "standalone" const val STANDALONE_FLAVOR = "standalone"
const val RELEASE = "release"
const val BASE_RELEASE_URL = "https://github.com/wgtunnel/wgtunnel/releases/tag/" const val BASE_RELEASE_URL = "https://github.com/wgtunnel/wgtunnel/releases/tag/"
} }
@@ -81,7 +81,7 @@ object DnsValidator {
return Result.Valid return Result.Valid
} }
private fun validateUdp(value: String): DnsValidator.Result { private fun validateUdp(value: String): Result {
val parts = value.split(":") val parts = value.split(":")
val host = parts.getOrNull(0)?.trim() val host = parts.getOrNull(0)?.trim()
@@ -93,14 +93,14 @@ object DnsValidator {
// basic IP/hostname sanity check // basic IP/hostname sanity check
if (!isValidHostOrIp(host)) { if (!isValidHostOrIp(host)) {
return DnsValidator.Result.Invalid(DnsError.InvalidIpOrHost) return Result.Invalid(DnsError.InvalidIpOrHost)
} }
if (port !in 1..65535) { if (port !in 1..65535) {
return DnsValidator.Result.Invalid(DnsError.InvalidPort) return Result.Invalid(DnsError.InvalidPort)
} }
return DnsValidator.Result.Valid return Result.Valid
} }
private fun isValidHostOrIp(value: String): Boolean { private fun isValidHostOrIp(value: String): Boolean {
@@ -1,6 +1,5 @@
package com.zaneschepke.wireguardautotunnel.util package com.zaneschepke.wireguardautotunnel.util
import LicenseFileEntry
import android.content.ContentValues import android.content.ContentValues
import android.content.Context import android.content.Context
import android.database.Cursor import android.database.Cursor
@@ -23,7 +22,6 @@ import java.util.zip.ZipInputStream
import java.util.zip.ZipOutputStream import java.util.zip.ZipOutputStream
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.withContext import kotlinx.coroutines.withContext
import kotlinx.serialization.json.Json
import timber.log.Timber import timber.log.Timber
class FileUtils(private val context: Context, private val ioDispatcher: CoroutineDispatcher) { class FileUtils(private val context: Context, private val ioDispatcher: CoroutineDispatcher) {
@@ -293,15 +291,6 @@ class FileUtils(private val context: Context, private val ioDispatcher: Coroutin
return@withContext itemUri return@withContext itemUri
} }
suspend fun readLibraryLicensesFromAssets(): List<LicenseFileEntry> =
withContext(ioDispatcher) {
val json = Json { ignoreUnknownKeys = true }
val jsonResult =
context.assets.open("licenses.json").bufferedReader().use { it.readText() }
json.decodeFromString(jsonResult)
}
companion object { companion object {
const val CONF_FILE_EXTENSION = ".conf" const val CONF_FILE_EXTENSION = ".conf"
const val ZIP_FILE_EXTENSION = ".zip" const val ZIP_FILE_EXTENSION = ".zip"
@@ -1,22 +1,9 @@
package com.zaneschepke.wireguardautotunnel.util package com.zaneschepke.wireguardautotunnel.util
import com.vdurmont.semver4j.Semver import com.vdurmont.semver4j.Semver
import java.math.BigDecimal
import kotlin.math.pow
import timber.log.Timber import timber.log.Timber
object NumberUtils { object NumberUtils {
private const val BYTES_IN_KB = 1024.0
private val BYTES_IN_MB = BYTES_IN_KB.pow(2.0)
private val keyValidationRegex = """^[A-Za-z0-9+/]{42}[AEIMQUYcgkosw480]=${'$'}""".toRegex()
fun bytesToMB(bytes: Long): BigDecimal {
return bytes.toBigDecimal().divide(BYTES_IN_MB.toBigDecimal())
}
fun isValidKey(key: String): Boolean {
return key.matches(keyValidationRegex)
}
fun generateRandomTunnelName(): String { fun generateRandomTunnelName(): String {
return "tunnel${randomFive()}" return "tunnel${randomFive()}"
@@ -1,27 +1,18 @@
package com.zaneschepke.wireguardautotunnel.util.extensions package com.zaneschepke.wireguardautotunnel.util.extensions
import android.Manifest
import android.app.Activity import android.app.Activity
import android.content.ComponentName
import android.content.Context import android.content.Context
import android.content.Context.POWER_SERVICE
import android.content.Intent import android.content.Intent
import android.content.pm.ApplicationInfo import android.content.pm.ApplicationInfo
import android.content.pm.PackageInfo
import android.content.pm.PackageManager import android.content.pm.PackageManager
import android.net.Uri import android.net.Uri
import android.os.Build import android.os.Build
import android.os.PowerManager
import android.provider.Settings import android.provider.Settings
import android.service.quicksettings.TileService
import android.widget.Toast
import androidx.core.content.FileProvider import androidx.core.content.FileProvider
import androidx.core.net.toUri import androidx.core.net.toUri
import com.zaneschepke.wireguardautotunnel.BuildConfig
import com.zaneschepke.wireguardautotunnel.MainActivity import com.zaneschepke.wireguardautotunnel.MainActivity
import com.zaneschepke.wireguardautotunnel.R import com.zaneschepke.wireguardautotunnel.R
import com.zaneschepke.wireguardautotunnel.core.service.tile.AutoTunnelControlTile
import com.zaneschepke.wireguardautotunnel.core.service.tile.TunnelControlTile
import com.zaneschepke.wireguardautotunnel.ui.screens.tunnels.splittunnel.state.TunnelApp
import com.zaneschepke.wireguardautotunnel.util.Constants import com.zaneschepke.wireguardautotunnel.util.Constants
import com.zaneschepke.wireguardautotunnel.util.FileUtils import com.zaneschepke.wireguardautotunnel.util.FileUtils
import java.io.File import java.io.File
@@ -30,22 +21,11 @@ import java.util.Locale
import kotlin.system.exitProcess import kotlin.system.exitProcess
import timber.log.Timber import timber.log.Timber
fun Context.openWebUrl(url: String): Result<Unit> { fun Context.openWebUrl(url: String): Result<Unit> = runCatching {
return kotlin val webpage: Uri = url.toUri()
.runCatching { val intent =
val webpage: Uri = url.toUri() Intent(Intent.ACTION_VIEW, webpage).apply { addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) }
val intent = startActivity(intent)
Intent(Intent.ACTION_VIEW, webpage).apply {
addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
}
startActivity(intent)
}
.onFailure { showToast(R.string.no_browser_detected) }
}
fun Context.isBatteryOptimizationsDisabled(): Boolean {
val pm = getSystemService(POWER_SERVICE) as PowerManager
return pm.isIgnoringBatteryOptimizations(packageName)
} }
fun Context.launchNotificationSettings() { fun Context.launchNotificationSettings() {
@@ -94,30 +74,7 @@ fun Context.hasSAFSupport(mimeType: String): Boolean {
} }
} }
fun Context.launchShareFile(file: File) { fun Context.launchSupportEmail(): Result<Unit> = runCatching {
FileProvider.getUriForFile(this, getString(R.string.provider), file)
val shareIntent =
Intent().apply {
action = Intent.ACTION_SEND
type = FileUtils.ALL_FILE_TYPES
putExtra(Intent.EXTRA_STREAM, file)
addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
}
val chooserIntent =
Intent.createChooser(shareIntent, "").apply { addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) }
this.startActivity(chooserIntent)
}
fun Context.showToast(resId: Int) {
Toast.makeText(this, this.getString(resId), Toast.LENGTH_LONG).show()
}
fun Context.showToast(message: String) {
Toast.makeText(this, message, Toast.LENGTH_LONG).show()
}
fun Context.launchSupportEmail() {
val intent = val intent =
Intent(Intent.ACTION_SENDTO).apply { Intent(Intent.ACTION_SENDTO).apply {
data = "mailto:".toUri() data = "mailto:".toUri()
@@ -132,7 +89,7 @@ fun Context.launchSupportEmail() {
} }
) )
} else { } else {
showToast(R.string.no_email_detected) throw IllegalStateException("No email client found")
} }
} }
@@ -143,7 +100,7 @@ fun Context.isRunningOnTv(): Boolean {
fun Context.launchVpnSettings(): Result<Unit> { fun Context.launchVpnSettings(): Result<Unit> {
return kotlin.runCatching { return kotlin.runCatching {
val intent = val intent =
Intent(Constants.VPN_SETTINGS_PACKAGE).apply { setFlags(Intent.FLAG_ACTIVITY_NEW_TASK) } Intent(Constants.VPN_SETTINGS_PACKAGE).apply { flags = Intent.FLAG_ACTIVITY_NEW_TASK }
startActivity(intent) startActivity(intent)
} }
} }
@@ -162,14 +119,6 @@ fun Context.launchLocationServicesSettings(): Result<Unit> {
} }
} }
fun Context.launchSettings(): Result<Unit> {
return kotlin.runCatching {
val intent =
Intent(Settings.ACTION_SETTINGS).apply { flags = Intent.FLAG_ACTIVITY_NEW_TASK }
startActivity(intent)
}
}
fun Context.launchAppSettings() { fun Context.launchAppSettings() {
kotlin kotlin
.runCatching { .runCatching {
@@ -186,48 +135,6 @@ fun Context.launchAppSettings() {
} }
} }
fun Context.requestTunnelTileServiceStateUpdate() =
runCatching {
TileService.requestListeningState(
this,
ComponentName(this, TunnelControlTile::class.java),
)
}
.onFailure { Timber.w(it) }
fun Context.requestAutoTunnelTileServiceUpdate() =
runCatching {
TileService.requestListeningState(
this,
ComponentName(this, AutoTunnelControlTile::class.java),
)
}
.onFailure { Timber.w(it) }
fun Context.getAllInternetCapablePackages(): List<PackageInfo> {
val permissions = arrayOf(Manifest.permission.INTERNET)
return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
packageManager.getPackagesHoldingPermissions(
permissions,
PackageManager.PackageInfoFlags.of(0L),
)
} else {
packageManager.getPackagesHoldingPermissions(permissions, 0)
}
}
fun Context.getSplitTunnelApps(): List<TunnelApp> {
val packages = getAllInternetCapablePackages()
return packages
.filter { it.applicationInfo != null }
.map { pkg ->
TunnelApp(
packageManager.getApplicationLabel(pkg.applicationInfo!!).toString(),
pkg.packageName,
)
}
}
fun Context.canInstallPackages(): Boolean { fun Context.canInstallPackages(): Boolean {
return packageManager.canRequestPackageInstalls() return packageManager.canRequestPackageInstalls()
} }
@@ -242,7 +149,7 @@ fun Context.requestInstallPackagesPermission() {
} }
fun Context.installApk(apkFile: File) { fun Context.installApk(apkFile: File) {
val apkUri = FileProvider.getUriForFile(this, getString(R.string.provider), apkFile) val apkUri = FileProvider.getUriForFile(this, BuildConfig.FILE_PROVIDER_AUTHORITY, apkFile)
val intent = val intent =
Intent(Intent.ACTION_VIEW).apply { Intent(Intent.ACTION_VIEW).apply {
setDataAndType(apkUri, "application/vnd.android.package-archive") setDataAndType(apkUri, "application/vnd.android.package-archive")
@@ -252,9 +159,9 @@ fun Context.installApk(apkFile: File) {
startActivity(intent) startActivity(intent)
} }
fun Context.launchPlayStoreListing() { fun Context.launchPlayStoreListing(): Result<Unit> = runCatching {
val intent = val intent =
Intent(Intent.ACTION_VIEW, Uri.parse("market://details?id=$packageName")).apply { Intent(Intent.ACTION_VIEW, "market://details?id=$packageName".toUri()).apply {
addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
setPackage("com.android.vending") setPackage("com.android.vending")
} }
@@ -262,12 +169,12 @@ fun Context.launchPlayStoreListing() {
if (intent.resolveActivity(packageManager) != null) { if (intent.resolveActivity(packageManager) != null) {
startActivity(intent) startActivity(intent)
} else { } else {
openWebUrl("https://play.google.com/store/apps/details?id=$packageName") throw IllegalStateException("Play Store not found")
} }
} }
fun Context.launchPlayStoreReview() { fun Context.launchPlayStoreReview(): Result<Unit> = runCatching {
val uri = Uri.parse("market://details?id=$packageName&showAllReviews=true") val uri = "market://details?id=$packageName&showAllReviews=true".toUri()
val intent = val intent =
Intent(Intent.ACTION_VIEW, uri).apply { Intent(Intent.ACTION_VIEW, uri).apply {
addFlags(Intent.FLAG_ACTIVITY_NEW_TASK) addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
@@ -276,7 +183,7 @@ fun Context.launchPlayStoreReview() {
if (intent.resolveActivity(packageManager) != null) { if (intent.resolveActivity(packageManager) != null) {
startActivity(intent) startActivity(intent)
} else { } else {
openWebUrl("https://play.google.com/store/apps/details?id=$packageName&showAllReviews=true") throw IllegalStateException("Play Store not found")
} }
} }
@@ -1,8 +0,0 @@
package com.zaneschepke.wireguardautotunnel.util.extensions
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.distinctUntilChanged
fun <K, V> Flow<Map<K, V>>.distinctByKeys(): Flow<Map<K, V>> {
return distinctUntilChanged { old, new -> old.keys == new.keys }
}
@@ -18,10 +18,6 @@ fun <T, R : Comparable<R>> List<T>.isSortedBy(selector: (T) -> R): Boolean {
return zipWithNext().all { (a, b) -> selector(a) <= selector(b) } return zipWithNext().all { (a, b) -> selector(a) <= selector(b) }
} }
fun Int.toMillis(): Long {
return this * 1_000L
}
fun Double.round(decimals: Int): Double { fun Double.round(decimals: Int): Double {
val factor = 10.0.pow(decimals) val factor = 10.0.pow(decimals)
return (this * factor).roundToInt() / factor return (this * factor).roundToInt() / factor
@@ -0,0 +1,13 @@
package com.zaneschepke.wireguardautotunnel.util.extensions
import io.ktor.client.statement.HttpResponse
import io.ktor.client.statement.bodyAsText
suspend fun HttpResponse.isHtmlResponse(): Boolean {
val contentType = headers["Content-Type"] ?: ""
if (contentType.contains("text/html", ignoreCase = true)) return true
val bodyStart = bodyAsText().trimStart()
return bodyStart.startsWith("<!DOCTYPE", ignoreCase = true) ||
bodyStart.startsWith("<html", ignoreCase = true)
}

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