mirror of
https://github.com/wgtunnel/android.git
synced 2026-07-03 14:07:49 +02:00
Compare commits
2 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 79b5b039b0 | |||
| 29616f8325 |
@@ -69,15 +69,16 @@ and on while on different networks. This app was created to offer a free solutio
|
||||
|
||||
Want updates faster?
|
||||
|
||||
Check out my personal [fdroid repository](https://github.com/zaneschepke/fdroid) to get updates the moment they are released.
|
||||
Check out my personal [fdroid repository](https://github.com/zaneschepke/fdroid) to get updates the
|
||||
moment they are released.
|
||||
|
||||
## Docs
|
||||
|
||||
Information about features, behaviors, and answers to common questions can be found in the app [documentation](https://zaneschepke.com/wgtunnel-docs/overview.html).
|
||||
Information about features, behaviors, and answers to common questions can be found in the
|
||||
app [documentation](https://zaneschepke.com/wgtunnel-docs/overview.html).
|
||||
|
||||
The repository for these docs can be found [here](https://github.com/zaneschepke/wgtunnel-docs).
|
||||
|
||||
|
||||
## Translation
|
||||
|
||||
This app is using [Weblate](https://weblate.org) to assist with translations.
|
||||
|
||||
@@ -112,9 +112,6 @@ android {
|
||||
}
|
||||
create("general") {
|
||||
dimension = Constants.TYPE
|
||||
if (BuildHelper.isReleaseBuild(gradle) && BuildHelper.isGeneralFlavor(gradle)) {
|
||||
//any plugins general specific
|
||||
}
|
||||
}
|
||||
}
|
||||
compileOptions {
|
||||
@@ -211,4 +208,7 @@ dependencies {
|
||||
// shortcuts
|
||||
implementation(libs.androidx.core)
|
||||
implementation(libs.androidx.core.google.shortcuts)
|
||||
|
||||
// splash
|
||||
implementation(libs.androidx.core.splashscreen)
|
||||
}
|
||||
|
||||
@@ -60,31 +60,36 @@
|
||||
android:label="@string/app_name"
|
||||
android:roundIcon="@mipmap/ic_launcher_round"
|
||||
android:supportsRtl="true"
|
||||
android:theme="@style/Theme.WireguardAutoTunnel"
|
||||
android:theme="@style/Theme.AppSplashScreen"
|
||||
tools:targetApi="tiramisu">
|
||||
<activity
|
||||
android:name=".ui.MainActivity"
|
||||
android:name=".ui.SplashActivity"
|
||||
android:exported="true"
|
||||
android:theme="@style/Theme.WireguardAutoTunnel">
|
||||
android:theme="@style/Theme.AppSplashScreen">
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.MAIN" />
|
||||
|
||||
<category android:name="android.intent.category.LAUNCHER" />
|
||||
<category android:name="android.intent.category.LEANBACK_LAUNCHER" />
|
||||
|
||||
<action android:name="android.service.quicksettings.action.QS_TILE_PREFERENCES" />
|
||||
</intent-filter>
|
||||
<meta-data
|
||||
android:name="android.app.shortcuts"
|
||||
android:resource="@xml/shortcuts" />
|
||||
</activity>
|
||||
<activity
|
||||
android:name=".ui.CaptureActivityPortrait"
|
||||
android:screenOrientation="fullSensor"
|
||||
android:stateNotNeeded="true"
|
||||
android:theme="@style/zxing_CaptureTheme"
|
||||
android:windowSoftInputMode="stateAlwaysHidden"
|
||||
tools:ignore="DiscouragedApi" />
|
||||
android:name=".ui.MainActivity"
|
||||
android:exported="true"
|
||||
android:screenOrientation="portrait"
|
||||
android:theme="@style/Theme.WireguardAutoTunnel">
|
||||
<intent-filter>
|
||||
<action android:name="android.service.quicksettings.action.QS_TILE_PREFERENCES" />
|
||||
</intent-filter>
|
||||
</activity>
|
||||
<activity
|
||||
android:name="com.journeyapps.barcodescanner.CaptureActivity"
|
||||
android:screenOrientation="portrait"
|
||||
tools:replace="screenOrientation" />
|
||||
<activity
|
||||
android:name=".service.shortcut.ShortcutsActivity"
|
||||
android:enabled="true"
|
||||
|
||||
@@ -6,34 +6,15 @@ import android.content.pm.PackageManager
|
||||
import android.os.StrictMode
|
||||
import android.os.StrictMode.ThreadPolicy
|
||||
import android.service.quicksettings.TileService
|
||||
import com.zaneschepke.logcatter.LocalLogCollector
|
||||
import com.zaneschepke.wireguardautotunnel.module.ApplicationScope
|
||||
import com.zaneschepke.wireguardautotunnel.module.IoDispatcher
|
||||
import com.zaneschepke.wireguardautotunnel.service.tile.AutoTunnelControlTile
|
||||
import com.zaneschepke.wireguardautotunnel.service.tile.TunnelControlTile
|
||||
import com.zaneschepke.wireguardautotunnel.util.ReleaseTree
|
||||
import dagger.hilt.android.HiltAndroidApp
|
||||
import kotlinx.coroutines.CoroutineDispatcher
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.launch
|
||||
import timber.log.Timber
|
||||
import xyz.teamgravity.pin_lock_compose.PinManager
|
||||
import javax.inject.Inject
|
||||
|
||||
@HiltAndroidApp
|
||||
class WireGuardAutoTunnel : Application() {
|
||||
|
||||
@Inject
|
||||
lateinit var localLogCollector: LocalLogCollector
|
||||
|
||||
@Inject
|
||||
@ApplicationScope
|
||||
lateinit var applicationScope: CoroutineScope
|
||||
|
||||
@Inject
|
||||
@IoDispatcher
|
||||
lateinit var ioDispatcher: CoroutineDispatcher
|
||||
|
||||
override fun onCreate() {
|
||||
super.onCreate()
|
||||
instance = this
|
||||
@@ -48,11 +29,6 @@ class WireGuardAutoTunnel : Application() {
|
||||
.build(),
|
||||
)
|
||||
} else Timber.plant(ReleaseTree())
|
||||
applicationScope.launch(ioDispatcher) {
|
||||
//TODO disable pin lock for now
|
||||
//PinManager.initialize(this@WireGuardAutoTunnel)
|
||||
if (!isRunningOnAndroidTv()) localLogCollector.start()
|
||||
}
|
||||
}
|
||||
|
||||
companion object {
|
||||
|
||||
+1
@@ -28,6 +28,7 @@ class DataStoreManager(
|
||||
booleanPreferencesKey("TUNNEL_RUNNING_FROM_MANUAL_START")
|
||||
val ACTIVE_TUNNEL = intPreferencesKey("ACTIVE_TUNNEL")
|
||||
val CURRENT_SSID = stringPreferencesKey("CURRENT_SSID")
|
||||
val IS_PIN_LOCK_ENABLED = booleanPreferencesKey("PIN_LOCK_ENABLED")
|
||||
}
|
||||
|
||||
// preferences
|
||||
|
||||
@@ -1,14 +1,16 @@
|
||||
package com.zaneschepke.wireguardautotunnel.data.domain
|
||||
|
||||
data class GeneralState(
|
||||
val locationDisclosureShown: Boolean = LOCATION_DISCLOSURE_SHOWN_DEFAULT,
|
||||
val batteryOptimizationDisableShown: Boolean = BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT,
|
||||
val tunnelRunningFromManualStart: Boolean = TUNNELING_RUNNING_FROM_MANUAL_START_DEFAULT,
|
||||
val isLocationDisclosureShown: Boolean = LOCATION_DISCLOSURE_SHOWN_DEFAULT,
|
||||
val isBatteryOptimizationDisableShown: Boolean = BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT,
|
||||
val isTunnelRunningFromManualStart: Boolean = TUNNELING_RUNNING_FROM_MANUAL_START_DEFAULT,
|
||||
val isPinLockEnabled: Boolean = PIN_LOCK_ENABLED_DEFAULT,
|
||||
val activeTunnelId: Int? = null
|
||||
) {
|
||||
companion object {
|
||||
const val LOCATION_DISCLOSURE_SHOWN_DEFAULT = false
|
||||
const val BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT = false
|
||||
const val TUNNELING_RUNNING_FROM_MANUAL_START_DEFAULT = false
|
||||
const val PIN_LOCK_ENABLED_DEFAULT = false
|
||||
}
|
||||
}
|
||||
|
||||
+3
@@ -7,6 +7,9 @@ interface AppStateRepository {
|
||||
suspend fun isLocationDisclosureShown(): Boolean
|
||||
suspend fun setLocationDisclosureShown(shown: Boolean)
|
||||
|
||||
suspend fun isPinLockEnabled(): Boolean
|
||||
suspend fun setPinLockEnabled(enabled: Boolean)
|
||||
|
||||
suspend fun isBatteryOptimizationDisableShown(): Boolean
|
||||
suspend fun setBatteryOptimizationDisableShown(shown: Boolean)
|
||||
|
||||
|
||||
+14
-3
@@ -17,6 +17,15 @@ class DataStoreAppStateRepository(private val dataStoreManager: DataStoreManager
|
||||
dataStoreManager.saveToDataStore(DataStoreManager.LOCATION_DISCLOSURE_SHOWN, shown)
|
||||
}
|
||||
|
||||
override suspend fun isPinLockEnabled(): Boolean {
|
||||
return dataStoreManager.getFromStore(DataStoreManager.IS_PIN_LOCK_ENABLED)
|
||||
?: GeneralState.PIN_LOCK_ENABLED_DEFAULT
|
||||
}
|
||||
|
||||
override suspend fun setPinLockEnabled(enabled: Boolean) {
|
||||
dataStoreManager.saveToDataStore(DataStoreManager.IS_PIN_LOCK_ENABLED, enabled)
|
||||
}
|
||||
|
||||
override suspend fun isBatteryOptimizationDisableShown(): Boolean {
|
||||
return dataStoreManager.getFromStore(DataStoreManager.BATTERY_OPTIMIZE_DISABLE_SHOWN)
|
||||
?: GeneralState.BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT
|
||||
@@ -65,11 +74,13 @@ class DataStoreAppStateRepository(private val dataStoreManager: DataStoreManager
|
||||
prefs?.let { pref ->
|
||||
try {
|
||||
GeneralState(
|
||||
locationDisclosureShown = pref[DataStoreManager.LOCATION_DISCLOSURE_SHOWN]
|
||||
isLocationDisclosureShown = pref[DataStoreManager.LOCATION_DISCLOSURE_SHOWN]
|
||||
?: GeneralState.LOCATION_DISCLOSURE_SHOWN_DEFAULT,
|
||||
batteryOptimizationDisableShown = pref[DataStoreManager.BATTERY_OPTIMIZE_DISABLE_SHOWN]
|
||||
isBatteryOptimizationDisableShown = pref[DataStoreManager.BATTERY_OPTIMIZE_DISABLE_SHOWN]
|
||||
?: GeneralState.BATTERY_OPTIMIZATION_DISABLE_SHOWN_DEFAULT,
|
||||
tunnelRunningFromManualStart = pref[DataStoreManager.TUNNEL_RUNNING_FROM_MANUAL_START]
|
||||
isTunnelRunningFromManualStart = pref[DataStoreManager.TUNNEL_RUNNING_FROM_MANUAL_START]
|
||||
?: GeneralState.TUNNELING_RUNNING_FROM_MANUAL_START_DEFAULT,
|
||||
isPinLockEnabled = pref[DataStoreManager.IS_PIN_LOCK_ENABLED]
|
||||
?: GeneralState.TUNNELING_RUNNING_FROM_MANUAL_START_DEFAULT,
|
||||
)
|
||||
} catch (e: IllegalArgumentException) {
|
||||
|
||||
@@ -4,9 +4,11 @@ import android.content.BroadcastReceiver
|
||||
import android.content.Context
|
||||
import android.content.Intent
|
||||
import com.zaneschepke.wireguardautotunnel.data.repository.AppDataRepository
|
||||
import com.zaneschepke.wireguardautotunnel.module.ApplicationScope
|
||||
import com.zaneschepke.wireguardautotunnel.service.foreground.ServiceManager
|
||||
import com.zaneschepke.wireguardautotunnel.util.goAsync
|
||||
import dagger.hilt.android.AndroidEntryPoint
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.launch
|
||||
import timber.log.Timber
|
||||
import javax.inject.Inject
|
||||
|
||||
@@ -19,27 +21,36 @@ class BootReceiver : BroadcastReceiver() {
|
||||
@Inject
|
||||
lateinit var serviceManager: ServiceManager
|
||||
|
||||
override fun onReceive(context: Context?, intent: Intent?) = goAsync {
|
||||
if (Intent.ACTION_BOOT_COMPLETED != intent?.action) return@goAsync
|
||||
@Inject
|
||||
@ApplicationScope
|
||||
lateinit var applicationScope: CoroutineScope
|
||||
|
||||
override fun onReceive(context: Context?, intent: Intent?) {
|
||||
if (Intent.ACTION_BOOT_COMPLETED != intent?.action) return
|
||||
context?.run {
|
||||
val settings = appDataRepository.settings.getSettings()
|
||||
if (settings.isAutoTunnelEnabled) {
|
||||
Timber.i("Starting watcher service from boot")
|
||||
serviceManager.startWatcherServiceForeground(context)
|
||||
}
|
||||
if (appDataRepository.appState.isTunnelRunningFromManualStart()) {
|
||||
appDataRepository.appState.getActiveTunnelId()?.let {
|
||||
Timber.i("Starting tunnel that was active before reboot")
|
||||
serviceManager.startVpnServiceForeground(
|
||||
context,
|
||||
appDataRepository.tunnels.getById(it)?.id,
|
||||
)
|
||||
applicationScope.launch {
|
||||
val settings = appDataRepository.settings.getSettings()
|
||||
if(settings.isRestoreOnBootEnabled) {
|
||||
if (settings.isAutoTunnelEnabled) {
|
||||
Timber.i("Starting watcher service from boot")
|
||||
serviceManager.startWatcherServiceForeground(context)
|
||||
}
|
||||
if (appDataRepository.appState.isTunnelRunningFromManualStart()) {
|
||||
appDataRepository.appState.getActiveTunnelId()?.let {
|
||||
Timber.i("Starting tunnel that was active before reboot")
|
||||
serviceManager.startVpnServiceForeground(
|
||||
context,
|
||||
appDataRepository.tunnels.getById(it)?.id,
|
||||
)
|
||||
return@launch
|
||||
}
|
||||
}
|
||||
if (settings.isAlwaysOnVpnEnabled) {
|
||||
Timber.i("Starting vpn service from boot AOVPN")
|
||||
serviceManager.startVpnServiceForeground(context)
|
||||
}
|
||||
}
|
||||
}
|
||||
if (settings.isAlwaysOnVpnEnabled) {
|
||||
Timber.i("Starting vpn service from boot AOVPN")
|
||||
serviceManager.startVpnServiceForeground(context)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
+19
-11
@@ -4,12 +4,14 @@ import android.content.BroadcastReceiver
|
||||
import android.content.Context
|
||||
import android.content.Intent
|
||||
import com.zaneschepke.wireguardautotunnel.data.repository.SettingsRepository
|
||||
import com.zaneschepke.wireguardautotunnel.module.ApplicationScope
|
||||
import com.zaneschepke.wireguardautotunnel.service.foreground.ServiceManager
|
||||
import com.zaneschepke.wireguardautotunnel.util.Constants
|
||||
import com.zaneschepke.wireguardautotunnel.util.goAsync
|
||||
import dagger.hilt.android.AndroidEntryPoint
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.cancel
|
||||
import kotlinx.coroutines.delay
|
||||
import kotlinx.coroutines.launch
|
||||
import timber.log.Timber
|
||||
import javax.inject.Inject
|
||||
|
||||
@@ -21,16 +23,22 @@ class NotificationActionReceiver : BroadcastReceiver() {
|
||||
@Inject
|
||||
lateinit var serviceManager: ServiceManager
|
||||
|
||||
override fun onReceive(context: Context, intent: Intent?) = goAsync {
|
||||
try {
|
||||
//TODO fix for manual start changes when enabled
|
||||
serviceManager.stopVpnServiceForeground(context)
|
||||
delay(Constants.TOGGLE_TUNNEL_DELAY)
|
||||
serviceManager.startVpnServiceForeground(context)
|
||||
} catch (e: Exception) {
|
||||
Timber.e(e)
|
||||
} finally {
|
||||
cancel()
|
||||
@Inject
|
||||
@ApplicationScope
|
||||
lateinit var applicationScope: CoroutineScope
|
||||
|
||||
override fun onReceive(context: Context, intent: Intent?) {
|
||||
applicationScope.launch {
|
||||
try {
|
||||
//TODO fix for manual start changes when enabled
|
||||
serviceManager.stopVpnServiceForeground(context)
|
||||
delay(Constants.TOGGLE_TUNNEL_DELAY)
|
||||
serviceManager.startVpnServiceForeground(context)
|
||||
} catch (e: Exception) {
|
||||
Timber.e(e)
|
||||
} finally {
|
||||
cancel()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
+17
-23
@@ -11,7 +11,6 @@ import com.zaneschepke.wireguardautotunnel.service.tunnel.TunnelState
|
||||
import com.zaneschepke.wireguardautotunnel.service.tunnel.VpnService
|
||||
import dagger.hilt.android.AndroidEntryPoint
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.Job
|
||||
import kotlinx.coroutines.cancel
|
||||
import kotlinx.coroutines.launch
|
||||
import timber.log.Timber
|
||||
@@ -35,32 +34,27 @@ class TunnelControlTile : TileService() {
|
||||
|
||||
private var manualStartConfig: TunnelConfig? = null
|
||||
|
||||
private var job: Job? = null;
|
||||
|
||||
override fun onStartListening() {
|
||||
super.onStartListening()
|
||||
Timber.d("On start listening called")
|
||||
//TODO Fix this
|
||||
if (job == null || job?.isCancelled == true) job = applicationScope.launch {
|
||||
vpnService.vpnState.collect { it ->
|
||||
when (it.status) {
|
||||
TunnelState.UP -> {
|
||||
setActive()
|
||||
it.tunnelConfig?.name?.let { name -> setTileDescription(name) }
|
||||
}
|
||||
|
||||
TunnelState.DOWN -> {
|
||||
setInactive()
|
||||
val config = appDataRepository.getStartTunnelConfig()?.also { config ->
|
||||
manualStartConfig = config
|
||||
} ?: appDataRepository.getPrimaryOrFirstTunnel()
|
||||
config?.let {
|
||||
setTileDescription(it.name)
|
||||
} ?: setUnavailable()
|
||||
}
|
||||
|
||||
else -> setInactive()
|
||||
applicationScope.launch {
|
||||
when (vpnService.getState()) {
|
||||
TunnelState.UP -> {
|
||||
setActive()
|
||||
setTileDescription(vpnService.name)
|
||||
}
|
||||
|
||||
TunnelState.DOWN -> {
|
||||
setInactive()
|
||||
val config = appDataRepository.getStartTunnelConfig()?.also { config ->
|
||||
manualStartConfig = config
|
||||
} ?: appDataRepository.getPrimaryOrFirstTunnel()
|
||||
config?.let {
|
||||
setTileDescription(it.name)
|
||||
} ?: setUnavailable()
|
||||
}
|
||||
|
||||
else -> setInactive()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
+20
-8
@@ -45,24 +45,19 @@ constructor(
|
||||
|
||||
private var statsJob: Job? = null
|
||||
|
||||
private lateinit var backend: Backend;
|
||||
|
||||
private var backendIsWgUserspace = true
|
||||
|
||||
private var backendIsAmneziaUserspace = false
|
||||
|
||||
init {
|
||||
applicationScope.launch(ioDispatcher) {
|
||||
backend = userspaceBackend.get()
|
||||
appDataRepository.settings.getSettingsFlow().collect {
|
||||
if (it.isKernelEnabled && (backendIsWgUserspace || backendIsAmneziaUserspace)) {
|
||||
Timber.i("Setting kernel backend")
|
||||
backend = kernelBackend.get()
|
||||
backendIsWgUserspace = false
|
||||
backendIsAmneziaUserspace = false
|
||||
} else if (!it.isKernelEnabled && !it.isAmneziaEnabled && !backendIsWgUserspace) {
|
||||
Timber.i("Setting WireGuard userspace backend")
|
||||
backend = userspaceBackend.get()
|
||||
backendIsWgUserspace = true
|
||||
backendIsAmneziaUserspace = false
|
||||
} else if (it.isAmneziaEnabled && !backendIsAmneziaUserspace) {
|
||||
@@ -89,7 +84,7 @@ constructor(
|
||||
} else {
|
||||
Timber.i("Using Wg backend")
|
||||
val wgConfig = tunnelConfig?.let { TunnelConfig.configFromWgQuick(it.wgQuick) }
|
||||
val state = backend.setState(
|
||||
val state = backend().setState(
|
||||
this,
|
||||
tunnelState.toWgState(),
|
||||
wgConfig,
|
||||
@@ -102,6 +97,7 @@ constructor(
|
||||
return withContext(ioDispatcher) {
|
||||
try {
|
||||
//TODO we need better error handling here
|
||||
// need to bubble up these errors to the UI
|
||||
val config = tunnelConfig ?: appDataRepository.getPrimaryOrFirstTunnel()
|
||||
if (config != null) {
|
||||
emitTunnelConfig(config)
|
||||
@@ -114,6 +110,22 @@ constructor(
|
||||
}
|
||||
}
|
||||
|
||||
private fun backend(): Backend {
|
||||
return when {
|
||||
backendIsWgUserspace -> {
|
||||
userspaceBackend.get()
|
||||
}
|
||||
|
||||
!backendIsWgUserspace && !backendIsAmneziaUserspace -> {
|
||||
kernelBackend.get()
|
||||
}
|
||||
|
||||
else -> {
|
||||
userspaceBackend.get()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private fun emitTunnelState(state: TunnelState) {
|
||||
_vpnState.tryEmit(
|
||||
_vpnState.value.copy(
|
||||
@@ -162,7 +174,7 @@ constructor(
|
||||
return if (backendIsAmneziaUserspace) TunnelState.from(
|
||||
userspaceAmneziaBackend.get().getState(this),
|
||||
)
|
||||
else TunnelState.from(backend.getState(this))
|
||||
else TunnelState.from(backend().getState(this))
|
||||
}
|
||||
|
||||
override fun getName(): String {
|
||||
@@ -198,7 +210,7 @@ constructor(
|
||||
),
|
||||
)
|
||||
} else {
|
||||
emitBackendStatistics(WireGuardStatistics(backend.getStatistics(this@WireGuardTunnel)))
|
||||
emitBackendStatistics(WireGuardStatistics(backend().getStatistics(this@WireGuardTunnel)))
|
||||
}
|
||||
delay(Constants.VPN_STATISTIC_CHECK_INTERVAL)
|
||||
}
|
||||
|
||||
@@ -1,5 +0,0 @@
|
||||
package com.zaneschepke.wireguardautotunnel.ui
|
||||
|
||||
import com.journeyapps.barcodescanner.CaptureActivity
|
||||
|
||||
class CaptureActivityPortrait : CaptureActivity()
|
||||
@@ -43,7 +43,7 @@ import com.google.accompanist.permissions.isGranted
|
||||
import com.google.accompanist.permissions.rememberPermissionState
|
||||
import com.zaneschepke.wireguardautotunnel.R
|
||||
import com.zaneschepke.wireguardautotunnel.WireGuardAutoTunnel
|
||||
import com.zaneschepke.wireguardautotunnel.data.datastore.DataStoreManager
|
||||
import com.zaneschepke.wireguardautotunnel.data.repository.AppStateRepository
|
||||
import com.zaneschepke.wireguardautotunnel.data.repository.SettingsRepository
|
||||
import com.zaneschepke.wireguardautotunnel.service.foreground.ServiceManager
|
||||
import com.zaneschepke.wireguardautotunnel.ui.common.navigation.BottomNavBar
|
||||
@@ -61,14 +61,13 @@ import com.zaneschepke.wireguardautotunnel.util.StringValue
|
||||
import dagger.hilt.android.AndroidEntryPoint
|
||||
import kotlinx.coroutines.Dispatchers
|
||||
import kotlinx.coroutines.launch
|
||||
import xyz.teamgravity.pin_lock_compose.PinManager
|
||||
import javax.inject.Inject
|
||||
|
||||
@AndroidEntryPoint
|
||||
class MainActivity : AppCompatActivity() {
|
||||
|
||||
@Inject
|
||||
lateinit var dataStoreManager: DataStoreManager
|
||||
lateinit var appStateRepository: AppStateRepository
|
||||
|
||||
@Inject
|
||||
lateinit var settingsRepository: SettingsRepository
|
||||
@@ -82,17 +81,18 @@ class MainActivity : AppCompatActivity() {
|
||||
override fun onCreate(savedInstanceState: Bundle?) {
|
||||
super.onCreate(savedInstanceState)
|
||||
|
||||
val isPinLockEnabled = intent.extras?.getBoolean(SplashActivity.IS_PIN_LOCK_ENABLED_KEY)
|
||||
|
||||
enableEdgeToEdge(navigationBarStyle = SystemBarStyle.dark(Color.Transparent.toArgb()))
|
||||
|
||||
// load preferences into memory and init data
|
||||
lifecycleScope.launch {
|
||||
dataStoreManager.init()
|
||||
WireGuardAutoTunnel.requestTunnelTileServiceStateUpdate()
|
||||
val settings = settingsRepository.getSettings()
|
||||
if (settings.isAutoTunnelEnabled) {
|
||||
serviceManager.startWatcherService(application.applicationContext)
|
||||
}
|
||||
}
|
||||
|
||||
setContent {
|
||||
val appViewModel = hiltViewModel<AppViewModel>()
|
||||
val appUiState by appViewModel.appUiState.collectAsStateWithLifecycle()
|
||||
@@ -201,12 +201,8 @@ class MainActivity : AppCompatActivity() {
|
||||
) { padding ->
|
||||
NavHost(
|
||||
navController,
|
||||
startDestination =
|
||||
//TODO disable pin lock
|
||||
//(if (PinManager.pinExists()) Screen.Lock.route else Screen.Main.route),
|
||||
Screen.Main.route,
|
||||
modifier =
|
||||
Modifier
|
||||
startDestination = (if (isPinLockEnabled == true) Screen.Lock.route else Screen.Main.route),
|
||||
modifier = Modifier
|
||||
.padding(padding)
|
||||
.fillMaxSize(),
|
||||
) {
|
||||
|
||||
@@ -0,0 +1,68 @@
|
||||
package com.zaneschepke.wireguardautotunnel.ui
|
||||
|
||||
import android.annotation.SuppressLint
|
||||
import android.content.Intent
|
||||
import android.os.Build
|
||||
import android.os.Bundle
|
||||
import androidx.activity.ComponentActivity
|
||||
import androidx.core.splashscreen.SplashScreen.Companion.installSplashScreen
|
||||
import androidx.lifecycle.Lifecycle
|
||||
import androidx.lifecycle.lifecycleScope
|
||||
import androidx.lifecycle.repeatOnLifecycle
|
||||
import com.zaneschepke.logcatter.LocalLogCollector
|
||||
import com.zaneschepke.wireguardautotunnel.WireGuardAutoTunnel
|
||||
import com.zaneschepke.wireguardautotunnel.WireGuardAutoTunnel.Companion.isRunningOnAndroidTv
|
||||
import com.zaneschepke.wireguardautotunnel.data.repository.AppStateRepository
|
||||
import com.zaneschepke.wireguardautotunnel.module.ApplicationScope
|
||||
import dagger.hilt.android.AndroidEntryPoint
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.launch
|
||||
import xyz.teamgravity.pin_lock_compose.PinManager
|
||||
import javax.inject.Inject
|
||||
|
||||
@SuppressLint("CustomSplashScreen")
|
||||
@AndroidEntryPoint
|
||||
class SplashActivity : ComponentActivity() {
|
||||
|
||||
@Inject
|
||||
lateinit var appStateRepository: AppStateRepository
|
||||
|
||||
@Inject
|
||||
lateinit var localLogCollector: LocalLogCollector
|
||||
|
||||
@Inject
|
||||
@ApplicationScope
|
||||
lateinit var applicationScope: CoroutineScope
|
||||
|
||||
override fun onCreate(savedInstanceState: Bundle?) {
|
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.S) {
|
||||
val splashScreen = installSplashScreen()
|
||||
splashScreen.setKeepOnScreenCondition { true }
|
||||
}
|
||||
super.onCreate(savedInstanceState)
|
||||
|
||||
applicationScope.launch {
|
||||
if (!isRunningOnAndroidTv()) localLogCollector.start()
|
||||
}
|
||||
|
||||
lifecycleScope.launch {
|
||||
repeatOnLifecycle(Lifecycle.State.CREATED) {
|
||||
val pinLockEnabled = appStateRepository.isPinLockEnabled()
|
||||
if (pinLockEnabled) {
|
||||
PinManager.initialize(WireGuardAutoTunnel.instance)
|
||||
}
|
||||
|
||||
val intent = Intent(this@SplashActivity, MainActivity::class.java).apply {
|
||||
putExtra(IS_PIN_LOCK_ENABLED_KEY, pinLockEnabled)
|
||||
}
|
||||
startActivity(intent)
|
||||
finish()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
companion object {
|
||||
const val IS_PIN_LOCK_ENABLED_KEY = "is_pin_lock_enabled"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -101,7 +101,6 @@ import com.zaneschepke.wireguardautotunnel.data.domain.TunnelConfig
|
||||
import com.zaneschepke.wireguardautotunnel.service.tunnel.HandshakeStatus
|
||||
import com.zaneschepke.wireguardautotunnel.service.tunnel.TunnelState
|
||||
import com.zaneschepke.wireguardautotunnel.ui.AppViewModel
|
||||
import com.zaneschepke.wireguardautotunnel.ui.CaptureActivityPortrait
|
||||
import com.zaneschepke.wireguardautotunnel.ui.Screen
|
||||
import com.zaneschepke.wireguardautotunnel.ui.common.RowListItem
|
||||
import com.zaneschepke.wireguardautotunnel.ui.common.screen.LoadingScreen
|
||||
@@ -262,8 +261,6 @@ fun MainScreen(
|
||||
context.getString(R.string.scanning_qr),
|
||||
)
|
||||
scanOptions.setBeepEnabled(false)
|
||||
scanOptions.captureActivity =
|
||||
CaptureActivityPortrait::class.java
|
||||
scanLauncher.launch(scanOptions)
|
||||
}
|
||||
|
||||
|
||||
+23
-16
@@ -84,7 +84,6 @@ import com.zaneschepke.wireguardautotunnel.ui.common.text.SectionTitle
|
||||
import com.zaneschepke.wireguardautotunnel.util.getMessage
|
||||
import kotlinx.coroutines.launch
|
||||
import timber.log.Timber
|
||||
import xyz.teamgravity.pin_lock_compose.PinManager
|
||||
import java.io.File
|
||||
|
||||
@OptIn(
|
||||
@@ -652,21 +651,29 @@ fun SettingsScreen(
|
||||
onCheckChanged = { viewModel.onToggleShortcutsEnabled() },
|
||||
)
|
||||
}
|
||||
// TODO disable for now
|
||||
// ConfigurationToggle(
|
||||
// stringResource(R.string.enable_app_lock),
|
||||
// enabled = true,
|
||||
// checked = pinExists.value,
|
||||
// padding = screenPadding,
|
||||
// onCheckChanged = {
|
||||
// if (pinExists.value) {
|
||||
// PinManager.clearPin()
|
||||
// pinExists.value = PinManager.pinExists()
|
||||
// } else {
|
||||
// navController.navigate(Screen.Lock.route)
|
||||
// }
|
||||
// },
|
||||
// )
|
||||
ConfigurationToggle(
|
||||
stringResource(R.string.restart_at_boot),
|
||||
enabled = true,
|
||||
checked = uiState.settings.isRestoreOnBootEnabled,
|
||||
padding = screenPadding,
|
||||
onCheckChanged = {
|
||||
viewModel.onToggleRestartAtBoot()
|
||||
},
|
||||
)
|
||||
ConfigurationToggle(
|
||||
stringResource(R.string.enable_app_lock),
|
||||
enabled = true,
|
||||
checked = uiState.isPinLockEnabled,
|
||||
padding = screenPadding,
|
||||
onCheckChanged = {
|
||||
if (uiState.isPinLockEnabled) {
|
||||
viewModel.onPinLockDisabled()
|
||||
} else {
|
||||
viewModel.onPinLockEnabled()
|
||||
navController.navigate(Screen.Lock.route)
|
||||
}
|
||||
},
|
||||
)
|
||||
if (!WireGuardAutoTunnel.isRunningOnAndroidTv()) {
|
||||
Row(
|
||||
verticalAlignment = Alignment.CenterVertically,
|
||||
|
||||
+1
@@ -10,4 +10,5 @@ data class SettingsUiState(
|
||||
val vpnState: VpnState = VpnState(),
|
||||
val isLocationDisclosureShown: Boolean = true,
|
||||
val isBatteryOptimizeDisableShown: Boolean = false,
|
||||
val isPinLockEnabled: Boolean = false
|
||||
)
|
||||
|
||||
+22
-2
@@ -27,6 +27,7 @@ import kotlinx.coroutines.flow.update
|
||||
import kotlinx.coroutines.launch
|
||||
import kotlinx.coroutines.withContext
|
||||
import timber.log.Timber
|
||||
import xyz.teamgravity.pin_lock_compose.PinManager
|
||||
import java.io.File
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Provider
|
||||
@@ -57,8 +58,9 @@ constructor(
|
||||
settings,
|
||||
tunnels,
|
||||
tunnelState,
|
||||
generalState.locationDisclosureShown,
|
||||
generalState.batteryOptimizationDisableShown,
|
||||
generalState.isLocationDisclosureShown,
|
||||
generalState.isBatteryOptimizationDisableShown,
|
||||
generalState.isPinLockEnabled,
|
||||
)
|
||||
}
|
||||
.stateIn(
|
||||
@@ -234,4 +236,22 @@ constructor(
|
||||
kernelSupport
|
||||
}
|
||||
}
|
||||
|
||||
fun onPinLockDisabled() = viewModelScope.launch {
|
||||
PinManager.clearPin()
|
||||
appDataRepository.appState.setPinLockEnabled(false)
|
||||
}
|
||||
|
||||
fun onPinLockEnabled() = viewModelScope.launch {
|
||||
PinManager.initialize(WireGuardAutoTunnel.instance)
|
||||
appDataRepository.appState.setPinLockEnabled(true)
|
||||
}
|
||||
|
||||
fun onToggleRestartAtBoot() = viewModelScope.launch {
|
||||
saveSettings(
|
||||
uiState.value.settings.copy(
|
||||
isRestoreOnBootEnabled = !uiState.value.settings.isRestoreOnBootEnabled
|
||||
)
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,6 +1,5 @@
|
||||
package com.zaneschepke.wireguardautotunnel.util
|
||||
|
||||
import android.content.BroadcastReceiver
|
||||
import android.content.Context
|
||||
import android.content.pm.PackageInfo
|
||||
import com.zaneschepke.wireguardautotunnel.R
|
||||
@@ -10,7 +9,6 @@ import com.zaneschepke.wireguardautotunnel.service.tunnel.statistics.TunnelStati
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.DelicateCoroutinesApi
|
||||
import kotlinx.coroutines.ExperimentalCoroutinesApi
|
||||
import kotlinx.coroutines.GlobalScope
|
||||
import kotlinx.coroutines.ObsoleteCoroutinesApi
|
||||
import kotlinx.coroutines.channels.ClosedReceiveChannelException
|
||||
import kotlinx.coroutines.channels.ReceiveChannel
|
||||
@@ -19,7 +17,6 @@ import kotlinx.coroutines.channels.ticker
|
||||
import kotlinx.coroutines.coroutineScope
|
||||
import kotlinx.coroutines.flow.Flow
|
||||
import kotlinx.coroutines.flow.channelFlow
|
||||
import kotlinx.coroutines.launch
|
||||
import kotlinx.coroutines.selects.whileSelect
|
||||
import org.amnezia.awg.config.Config
|
||||
import timber.log.Timber
|
||||
@@ -27,25 +24,8 @@ import java.math.BigDecimal
|
||||
import java.text.DecimalFormat
|
||||
import java.time.Duration
|
||||
import java.util.concurrent.ConcurrentLinkedQueue
|
||||
import kotlin.coroutines.CoroutineContext
|
||||
import kotlin.coroutines.EmptyCoroutineContext
|
||||
import kotlin.coroutines.cancellation.CancellationException
|
||||
|
||||
fun BroadcastReceiver.goAsync(
|
||||
context: CoroutineContext = EmptyCoroutineContext,
|
||||
block: suspend CoroutineScope.() -> Unit
|
||||
) {
|
||||
val pendingResult = goAsync()
|
||||
@OptIn(DelicateCoroutinesApi::class) // Must run globally; there's no teardown callback.
|
||||
GlobalScope.launch(context) {
|
||||
try {
|
||||
block()
|
||||
} finally {
|
||||
pendingResult.finish()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fun BigDecimal.toThreeDecimalPlaceString(): String {
|
||||
val df = DecimalFormat("#.###")
|
||||
return df.format(this)
|
||||
|
||||
@@ -144,4 +144,11 @@
|
||||
<string name="general">Obecné</string>
|
||||
<string name="settings">Nastavení</string>
|
||||
<string name="support">Podpora</string>
|
||||
<string name="app_name">WG Tunnel</string>
|
||||
<string name="mtu">MTU</string>
|
||||
<string name="db_name">wg-tunnel-db</string>
|
||||
<string name="listen_port">Naslouchací port</string>
|
||||
<string name="auto">(automaticky)</string>
|
||||
<string name="kernel">Kernel</string>
|
||||
<string name="backend">Backend</string>
|
||||
</resources>
|
||||
@@ -135,7 +135,7 @@
|
||||
<string name="pin_created">Pin creado con éxito</string>
|
||||
<string name="enter_pin">Introduce tu pin</string>
|
||||
<string name="create_pin">Crear pin</string>
|
||||
<string name="enable_app_lock">Bloqueo de app activado</string>
|
||||
<string name="enable_app_lock">Activar el bloqueo de aplicaciones</string>
|
||||
<string name="restart_on_ping">Reiniciar al fallar ping (beta)</string>
|
||||
<string name="set_primary_tunnel">Establecer como túnel Principal</string>
|
||||
<string name="no_wifi_names_configured">No hay nombres Wi-Fi configurados para este túnel</string>
|
||||
@@ -152,4 +152,9 @@
|
||||
<string name="watcher_channel_id">Canal del obvervador</string>
|
||||
<string name="watcher_channel_name">Canal de notificación del obvervador</string>
|
||||
<string name="prominent_background_location_message">La monitorización SSID Wi-Fi necesita de permiso de ubicación en segundo plano incluso si la app está cerrada. Mira el enlace a la Política de Privacidad en la pantalla de ayuda para más detalles.</string>
|
||||
<string name="export_configs_failed">Error al exportar la configuración</string>
|
||||
<string name="use_amnezia">"Utilizar el entorno de usuario de Amnezia "</string>
|
||||
<string name="junk_packet_count">Recuento de paquetes basura</string>
|
||||
<string name="backend">Backend</string>
|
||||
<string name="junk_packet_minimum_size">Tamaño mínimo del paquete basura</string>
|
||||
</resources>
|
||||
@@ -0,0 +1,162 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<string name="watcher_notification_text_active">Monitoramento de mudanças do estado da rede: Ativo</string>
|
||||
<string name="watcher_notification_text_paused">Monitoramento de mudanças do estado da rede: pausado</string>
|
||||
<string name="tunnel_start_title">VPN conectada</string>
|
||||
<string name="tunnel_on_ethernet">Túnel na ethernet</string>
|
||||
<string name="save_changes">Salvar</string>
|
||||
<string name="public_key">Chave pública</string>
|
||||
<string name="addresses">Endereços</string>
|
||||
<string name="dns_servers">Servidores DNS</string>
|
||||
<string name="endpoint">Endpoint</string>
|
||||
<string name="name">Nome</string>
|
||||
<string name="restart">Reiniciar Túnel</string>
|
||||
<string name="create_import">Criar do zero</string>
|
||||
<string name="turn_off_auto">Esta ação precisa do auto-túnel desativado ou pausado</string>
|
||||
<string name="rotate_keys">Revezar chaves</string>
|
||||
<string name="private_key">Chave privada</string>
|
||||
<string name="base64_key">Chave base64</string>
|
||||
<string name="optional_no_recommend">(opcional, não recomendado)</string>
|
||||
<string name="preshared_key">Chave pré-compartilhada</string>
|
||||
<string name="seconds">segundos</string>
|
||||
<string name="export_configs">Exportar configurações</string>
|
||||
<string name="export_configs_failed">Falhou ao exportar configurações</string>
|
||||
<string name="go">ir</string>
|
||||
<string name="error_no_file_explorer">Nenhum explorador de arquivos instalado</string>
|
||||
<string name="error_invalid_code">Código QR inválido</string>
|
||||
<string name="auto_tunnel_title">Serviço de Auto-túnel</string>
|
||||
<string name="excluded">excluído</string>
|
||||
<string name="all">todos</string>
|
||||
<string name="enter_pin">Digite seu pin</string>
|
||||
<string name="use_tunnel_on_wifi_name">Usar túnel em wifi com nome</string>
|
||||
<string name="auto_tun_on">Continuar auto-túnel</string>
|
||||
<string name="auto_tun_off">Pausar auto-túnel</string>
|
||||
<string name="version">Versão</string>
|
||||
<string name="mode">Modo</string>
|
||||
<string name="use_amnezia">"Usar Amnezia em modo usuário "</string>
|
||||
<string name="junk_packet_count">Número de pacotes-lixo</string>
|
||||
<string name="junk_packet_minimum_size">Tamanho mínimo de pacote-lixo</string>
|
||||
<string name="junk_packet_maximum_size">Tamanho máximo de pacote-lixo</string>
|
||||
<string name="init_packet_junk_size">Tamanho de pacote-lixo inicial</string>
|
||||
<string name="response_packet_junk_size">Tamanho de resposta de pacote-lixo</string>
|
||||
<string name="app_name">WG Tunnel</string>
|
||||
<string name="no_tunnels">Nenhum túnel foi adicionado!</string>
|
||||
<string name="error_file_extension">O arquivo não é .conf ou .zip</string>
|
||||
<string name="prominent_background_location_message">Este recurso precisa de permissões de localização em segundo plano para habilitar o monitoramento do SSID da rede Wi-Fi mesmo quando o aplicativo está fechado. Para mais detalhes, por favor veja a Política de Privacidade na tela de Suporte.</string>
|
||||
<string name="turn_off_tunnel">Esta ação só é possível com o túnel inativo</string>
|
||||
<string name="enabled_app_shortcuts">Habilitar atalhos do aplicativo</string>
|
||||
<string name="notification_permission_required">Necessita permissões de notificação.</string>
|
||||
<string name="add_trusted_ssid">Adicionar nome de Wi-Fi confiável</string>
|
||||
<string name="enable_auto_tunnel">Iniciar auto-túnel</string>
|
||||
<string name="tunnels">Túneis</string>
|
||||
<string name="disable_auto_tunnel">Parar auto-túnel</string>
|
||||
<string name="tunnel_start_text">Conectado ao túnel</string>
|
||||
<string name="trusted_ssid_empty_description">Digite o SSID</string>
|
||||
<string name="no_thanks">Não, obrigado</string>
|
||||
<string name="privacy_policy">Ver a Política de Privacidade</string>
|
||||
<string name="okay">OK</string>
|
||||
<string name="tunnel_mobile_data">Túnel em dados móveis</string>
|
||||
<string name="one_tunnel_required">Pelo menos um túnel é necessário para usar este recurso</string>
|
||||
<string name="prominent_background_location_title">Revelar a localização em segundo plano</string>
|
||||
<string name="thank_you">Obrigado por usar o WG Tunnel!</string>
|
||||
<string name="trusted_ssid_value_description">Envie o SSID</string>
|
||||
<string name="open_file">Abrir Arquivo</string>
|
||||
<string name="add_from_qr">Adicionar a partir de código QR</string>
|
||||
<string name="add_tunnels_text">Adicionar a partir de arquivo ou zip</string>
|
||||
<string name="tunnel_all">Todos os aplicativos pelo túnel</string>
|
||||
<string name="icon">Ícone</string>
|
||||
<string name="turn_on">Ligar</string>
|
||||
<string name="qr_scan">Escanear o código QR</string>
|
||||
<string name="tunnel_name">Nome do Túnel</string>
|
||||
<string name="add_tunnel">Adicionar Túnel</string>
|
||||
<string name="config_changes_saved">Mudanças nas configurações salvas.</string>
|
||||
<string name="exclude">Excluir</string>
|
||||
<string name="include">Incluir</string>
|
||||
<string name="map">Mapa</string>
|
||||
<string name="vpn_connection_failed">Falha na conexão</string>
|
||||
<string name="mtu">MTU</string>
|
||||
<string name="always_on_vpn_support">Permitir VPN sempre ligada</string>
|
||||
<string name="allowed_ips">IPs Permitidos</string>
|
||||
<string name="attempt_connection">Tentando conexão..</string>
|
||||
<string name="peer">Par</string>
|
||||
<string name="location_services_not_detected">Serviço de localização não foi detectado</string>
|
||||
<string name="hint_search_packages">Procurar pacotes</string>
|
||||
<string name="vpn_starting">Iniciando VPN</string>
|
||||
<string name="other">Outro</string>
|
||||
<string name="scanning_qr">Escaneando código QR</string>
|
||||
<string name="none">Nenhum nome de Wi-Fi confiável</string>
|
||||
<string name="auto_tunneling">Auto-túnel</string>
|
||||
<string name="default_vpn_on">VPN Principal ligada</string>
|
||||
<string name="vpn_on">VPN ligada</string>
|
||||
<string name="vpn_off">VPN desligada</string>
|
||||
<string name="listen_port">Porta de escuta</string>
|
||||
<string name="default_vpn_off">VPN Principal desligada</string>
|
||||
<string name="turn_on_tunnel">Esta ação precisa um túnel ativo</string>
|
||||
<string name="done">Feito</string>
|
||||
<string name="add_peer">Adicionar par</string>
|
||||
<string name="interface_">Interface</string>
|
||||
<string name="copy_public_key">Copiar chave pública</string>
|
||||
<string name="comma_separated_list">Lista separada por vírgulas</string>
|
||||
<string name="optional">(opcional)</string>
|
||||
<string name="random">(aleatório)</string>
|
||||
<string name="persistent_keepalive">Manter a conexão persistente (keepalive)</string>
|
||||
<string name="cancel">Cancelar</string>
|
||||
<string name="error_authentication_failed">Autenticação falhou</string>
|
||||
<string name="error_authorization_failed">Autorização falhou</string>
|
||||
<string name="restart_on_ping">Reiniciar em falha de ping (beta)</string>
|
||||
<string name="background_location_required">Necessita a localização em segundo plano</string>
|
||||
<string name="location_services_required">Necessita dos serviços de localização</string>
|
||||
<string name="email_description">Me envie um email</string>
|
||||
<string name="error_ssid_exists">SSID já existe</string>
|
||||
<string name="delete_tunnel_message">Tem certeza que você quer apagar este túnel?</string>
|
||||
<string name="yes">Sim</string>
|
||||
<string name="precise_location_required">Necessita da localização precisa</string>
|
||||
<string name="exported_configs_message">Configurações exportadas para downloads</string>
|
||||
<string name="unknown_error">Ocorreu um erro desconhecido</string>
|
||||
<string name="email_subject">Suporte para o WG Tunnel</string>
|
||||
<string name="tunnel_on_wifi">Túnel em Wi-Fi não confiável</string>
|
||||
<string name="error_none">Nenhum erro</string>
|
||||
<string name="delete_tunnel">Apagar túnel</string>
|
||||
<string name="email_chooser">Enviar um email…</string>
|
||||
<string name="use_kernel">Usar o módulo do kernel</string>
|
||||
<string name="discord_description">Juntar-se à comunidade</string>
|
||||
<string name="docs_description">Ler a documentação</string>
|
||||
<string name="support_help_text">Se você enfrentar problemas, tiver ideias para melhorias ou apenas quiser participar, os seguintes recursos estão disponíveis:</string>
|
||||
<string name="error_root_denied">Shell Root negado</string>
|
||||
<string name="location_services_missing_message">O aplicativo não detectou o serviço de localização habilitado no seu dispositivo. Dependendo do dispositivo, isto pode causar que a função de Wi-Fi não confiável falhe em ler o nome do Wi-Fi. Quer continuar mesmo assim?</string>
|
||||
<string name="paused">pausado</string>
|
||||
<string name="included">incluso</string>
|
||||
<string name="resume">Continuar</string>
|
||||
<string name="pause">Pausar</string>
|
||||
<string name="active">ativo</string>
|
||||
<string name="always_on_disabled">VPN sempre ligada tentou iniciar um túnel, mas este recurso está desligado nas configurações.</string>
|
||||
<string name="open_issue">Abrir um problema</string>
|
||||
<string name="tunneling_apps">Aplicativos em túnel</string>
|
||||
<string name="no_email_detected">Nenhum aplicativo de email detectado</string>
|
||||
<string name="no_browser_detected">Nenhum navegador detectado</string>
|
||||
<string name="logs_saved">Registros salvos em downloads</string>
|
||||
<string name="incorrect_pin">O Pin está errado</string>
|
||||
<string name="auto">(automático)</string>
|
||||
<string name="disabled">desligado</string>
|
||||
<string name="read_logs">Ler os registros</string>
|
||||
<string name="config_parse_error">Falha na interpretação das configurações</string>
|
||||
<string name="pin_created">Pin criado com sucesso</string>
|
||||
<string name="auto_on">Continuar auto-túnel</string>
|
||||
<string name="create_pin">Criar um pin</string>
|
||||
<string name="enable_app_lock">Ligar trava de aplicativo</string>
|
||||
<string name="no_wifi_names_configured">Nenhum Wi-Fi configurado para este túnel</string>
|
||||
<string name="edit_tunnel">Editar túnel</string>
|
||||
<string name="mobile_data_tunnel">Selecionar como túnel em dados móveis</string>
|
||||
<string name="set_primary_tunnel">Selecionar como túnel principal</string>
|
||||
<string name="general">Geral</string>
|
||||
<string name="userspace">Modo usuário</string>
|
||||
<string name="support">Suporte</string>
|
||||
<string name="kernel">Kernel</string>
|
||||
<string name="auto_off">Pausar auto-túnel</string>
|
||||
<string name="backend">Backend</string>
|
||||
<string name="settings">Configurações</string>
|
||||
<string name="unsure_how">se não tiver certeza em como continuar</string>
|
||||
<string name="see_the">Veja o</string>
|
||||
<string name="getting_started_guide">guia de início rápido</string>
|
||||
<string name="error_file_format">Formato de arquivo de configuração inválido</string>
|
||||
</resources>
|
||||
@@ -59,7 +59,7 @@
|
||||
<string name="cancel">Отмена</string>
|
||||
<string name="docs_description">Посмотреть документацию</string>
|
||||
<string name="discord_description">Присоединиться к сообществу</string>
|
||||
<string name="email_chooser">Отправить письмо</string>
|
||||
<string name="email_chooser">Отправить письмо…</string>
|
||||
<string name="add_trusted_ssid">Добавить доверенное имя сети Wi-Fi</string>
|
||||
<string name="included">включено</string>
|
||||
<string name="vpn_starting">Идёт запуск VPN</string>
|
||||
|
||||
@@ -0,0 +1,169 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<string name="comma_separated_list">список розділений комами</string>
|
||||
<string name="no_tunnels">Тунелі ще не додані!</string>
|
||||
<string name="app_name">WG Tunnel</string>
|
||||
<string name="add_trusted_ssid">Додати ім\'я довіреної мережі Wi-Fi</string>
|
||||
<string name="tunnels">Тунелі</string>
|
||||
<string name="enable_auto_tunnel">Запустити авто-тунелі</string>
|
||||
<string name="disable_auto_tunnel">Зупинити авто-тунелі</string>
|
||||
<string name="okay">ОК</string>
|
||||
<string name="tunnel_on_ethernet">Тунелювати Ethernet</string>
|
||||
<string name="prominent_background_location_title">Фонова передача місцезнаходження</string>
|
||||
<string name="trusted_ssid_value_description">Підтвердити SSID</string>
|
||||
<string name="qr_scan">Сканувати QR</string>
|
||||
<string name="tunnel_name">Ім\'я тунелю</string>
|
||||
<string name="add_tunnel">Додати тунель</string>
|
||||
<string name="include">Включити</string>
|
||||
<string name="tunnel_all">Тунель для всіх додатків</string>
|
||||
<string name="config_changes_saved">Зміни налаштувань збережено.</string>
|
||||
<string name="icon">Іконка</string>
|
||||
<string name="no_thanks">Ні, дякую</string>
|
||||
<string name="default_vpn_off">Основний VPN вимк.</string>
|
||||
<string name="turn_on_tunnel">Дія потребує активного тунелю</string>
|
||||
<string name="rotate_keys">Оновити ключі</string>
|
||||
<string name="private_key">Закритий ключ</string>
|
||||
<string name="base64_key">Ключ в base64</string>
|
||||
<string name="random">(випадково)</string>
|
||||
<string name="optional">(необов\'язково)</string>
|
||||
<string name="optional_no_recommend">(необов\'язково, не рекомендується)</string>
|
||||
<string name="cancel">Скасувати</string>
|
||||
<string name="export_configs_failed">Помилка експорту конфігурації</string>
|
||||
<string name="location_services_required">Необхідно сервіси місцезнаходження</string>
|
||||
<string name="precise_location_required">Необхідно доступ до точного місцезнаходження</string>
|
||||
<string name="exported_configs_message">Експорт конфігурації в Завантаження</string>
|
||||
<string name="email_chooser">Надіслати E-Mail…</string>
|
||||
<string name="error_root_denied">Root доступ заборонено</string>
|
||||
<string name="error_invalid_code">Некоректний QR-код</string>
|
||||
<string name="error_none">Нема помилок</string>
|
||||
<string name="logs_saved">Логи збережено в Завантаженнях</string>
|
||||
<string name="config_parse_error">Помилка аналізу файлу конфігурації</string>
|
||||
<string name="incorrect_pin">Невірний PIN-код</string>
|
||||
<string name="use_tunnel_on_wifi_name">Використовувати тунель в мережі Wi-Fi</string>
|
||||
<string name="disabled">відключено</string>
|
||||
<string name="version">Версія</string>
|
||||
<string name="mode">Режим</string>
|
||||
<string name="transport_packet_magic_header">Заголовок транспортного пакету</string>
|
||||
<string name="getting_started_guide">інструкція щодо початку роботи</string>
|
||||
<string name="error_file_format">некоректний формат конфігурації тунелю</string>
|
||||
<string name="watcher_notification_text_active">Моніторинг стану мережі: активний</string>
|
||||
<string name="vpn_channel_name">Канал сповіщення VPN</string>
|
||||
<string name="vpn_channel_id">Канал VPN</string>
|
||||
<string name="error_file_extension">Файл не є .conf або .zip файлом</string>
|
||||
<string name="turn_off_tunnel">Дія потребує вимкнення тунелю</string>
|
||||
<string name="watcher_notification_text_paused">Моніторинг стану мережі: призупинено</string>
|
||||
<string name="tunnel_start_text">Підключення до тунелю</string>
|
||||
<string name="tunnel_start_title">VPN підключено</string>
|
||||
<string name="notification_permission_required">Потрібен дозвіл на відображення сповіщень.</string>
|
||||
<string name="tunnel_mobile_data">Тунелювати мобільні дані</string>
|
||||
<string name="one_tunnel_required">Для використання даної функції потрібно налаштувати мінімум один тунель</string>
|
||||
<string name="privacy_policy">Переглянути політику конфіденційності</string>
|
||||
<string name="thank_you">Спасибі за використання WG Tunnel!</string>
|
||||
<string name="prominent_background_location_message">Дана функція потребує фоновий доступ до служби місцезнаходження для моніторингу назви мереж Wi-Fi навіть коли додаток закрито. Для отримання додаткової інформації прочитайте політику приватності на екрані Підтримки.</string>
|
||||
<string name="trusted_ssid_empty_description">Введіть SSID</string>
|
||||
<string name="add_tunnels_text">Додати з файлу або архіву</string>
|
||||
<string name="open_file">Відкрити файл</string>
|
||||
<string name="exclude">Виключити</string>
|
||||
<string name="add_from_qr">Додати з QR коду</string>
|
||||
<string name="save_changes">Зберегти</string>
|
||||
<string name="turn_on">Увімкнути</string>
|
||||
<string name="map">Карта</string>
|
||||
<string name="public_key">Публічний ключ</string>
|
||||
<string name="addresses">Адреса</string>
|
||||
<string name="allowed_ips">Дозволені IP</string>
|
||||
<string name="dns_servers">DNS-сервери</string>
|
||||
<string name="mtu">MTU</string>
|
||||
<string name="peer">Пір</string>
|
||||
<string name="endpoint">Кінцева точка</string>
|
||||
<string name="hint_search_packages">Пошук програм</string>
|
||||
<string name="name">Ім\'я</string>
|
||||
<string name="vpn_connection_failed">Помилка з\'єднання</string>
|
||||
<string name="restart">Перезапустити тунель</string>
|
||||
<string name="always_on_vpn_support">Дозволили Always-ON VPN</string>
|
||||
<string name="location_services_not_detected">Сервіси місце знаходження не знайдено</string>
|
||||
<string name="db_name">wg-tunnel-db</string>
|
||||
<string name="scanning_qr">Сканування QR коду</string>
|
||||
<string name="attempt_connection">Спроба з\'єднання...</string>
|
||||
<string name="other">Інше</string>
|
||||
<string name="vpn_starting">Запуск VPN</string>
|
||||
<string name="auto_tunneling">Авто-тунелювання</string>
|
||||
<string name="none">Нема довірених мереж Wi-Fi</string>
|
||||
<string name="vpn_on">VPN увімк.</string>
|
||||
<string name="vpn_off">VPN вимк.</string>
|
||||
<string name="default_vpn_on">Основний VPN увімк.</string>
|
||||
<string name="create_import">Створити з нуля</string>
|
||||
<string name="turn_off_auto">Необхідно вимкнути або призупинити авто-тунелювання</string>
|
||||
<string name="add_peer">Додати peer</string>
|
||||
<string name="done">Готово</string>
|
||||
<string name="interface_">Інтерфейс</string>
|
||||
<string name="copy_public_key">Копіювати відкритий ключ</string>
|
||||
<string name="listen_port">Слухати порт</string>
|
||||
<string name="preshared_key">Pre-shared key</string>
|
||||
<string name="seconds">секунд</string>
|
||||
<string name="persistent_keepalive">Persistent keepalive</string>
|
||||
<string name="error_authorization_failed">Не вдалося авторизуватися</string>
|
||||
<string name="enabled_app_shortcuts">Дозволити ярлики</string>
|
||||
<string name="error_authentication_failed">Помилка автентифікації</string>
|
||||
<string name="export_configs">Експорт конфігурації</string>
|
||||
<string name="background_location_required">Необхідний фоновий доступ до місцезнаходження</string>
|
||||
<string name="unknown_error">Невідома помилка</string>
|
||||
<string name="tunnel_on_wifi">Тунелювати недовірені мережі Wi-Fi</string>
|
||||
<string name="email_subject">Підтримка WG-Tunnel</string>
|
||||
<string name="go">вперед</string>
|
||||
<string name="docs_description">Переглянути документацію</string>
|
||||
<string name="email_description">Відправити email автору</string>
|
||||
<string name="discord_description">Приєднатися до спільноти</string>
|
||||
<string name="support_help_text">Якщо у вас виникли проблеми, є ідеї щодо покращення, чи бажання долучитися, скористайтесь наступними ресурсами:</string>
|
||||
<string name="use_kernel">Використовувати модуль режиму ядра</string>
|
||||
<string name="error_ssid_exists">SSID вже існує</string>
|
||||
<string name="error_no_file_explorer">Не знайдено файловий менеджер</string>
|
||||
<string name="auto_tunnel_title">Сервіс авто-тунелювання</string>
|
||||
<string name="delete_tunnel">Видалити тунель</string>
|
||||
<string name="location_services_missing_message">Додаток не знайшов служб місце знаходження на вашому пристрої. На деяких пристроях це може привести до неможливості визначення назви мережі Wi-Fi і помилок функції недовірених Wi-Fi мереж. Все рівно хочете продовжити?</string>
|
||||
<string name="included">включено</string>
|
||||
<string name="delete_tunnel_message">Ви дійсно хочете видалити цей тунель?</string>
|
||||
<string name="yes">Так</string>
|
||||
<string name="active">активно</string>
|
||||
<string name="resume">Відновити</string>
|
||||
<string name="pause">Призупинити</string>
|
||||
<string name="paused">призупинено</string>
|
||||
<string name="tunneling_apps">Тунельовані додатки</string>
|
||||
<string name="excluded">виключено</string>
|
||||
<string name="always_on_disabled">Функція Always-on VPN спробувала запустити тунель, але функція вимкнена в налаштуваннях.</string>
|
||||
<string name="auto">(авто)</string>
|
||||
<string name="all">всі</string>
|
||||
<string name="no_email_detected">Програми для надсилання email не знайдено</string>
|
||||
<string name="open_issue">Повідомити про проблему</string>
|
||||
<string name="read_logs">Переглянути логи</string>
|
||||
<string name="no_browser_detected">Веб браузер не знайдено</string>
|
||||
<string name="pin_created">PIN-код створено успішно</string>
|
||||
<string name="enter_pin">Введіть PIN-код</string>
|
||||
<string name="create_pin">Створити PINhкод</string>
|
||||
<string name="enable_app_lock">Увімкнути блокування додатку</string>
|
||||
<string name="edit_tunnel">Редагувати тунель</string>
|
||||
<string name="auto_on">Відновити авто-тунель</string>
|
||||
<string name="restart_on_ping">Перезапуск при помилці ping (бета)</string>
|
||||
<string name="mobile_data_tunnel">Встановити як тунель для мобільних даних</string>
|
||||
<string name="set_primary_tunnel">Встановити як основний тунель</string>
|
||||
<string name="no_wifi_names_configured">Імена мереж Wi-Fi не налаштовано для цього тунелю</string>
|
||||
<string name="general">Загальне</string>
|
||||
<string name="auto_tun_off">Призупинити авто-тунель</string>
|
||||
<string name="auto_off">Призупинити авто-тунель</string>
|
||||
<string name="auto_tun_on">Відновити авто-тунель</string>
|
||||
<string name="userspace">Користувача</string>
|
||||
<string name="support">Підтримка</string>
|
||||
<string name="settings">Налаштування</string>
|
||||
<string name="use_amnezia">"Використовувати модуль Amnezia режиму користувача "</string>
|
||||
<string name="backend">Модуль</string>
|
||||
<string name="kernel">Модуль ядра</string>
|
||||
<string name="junk_packet_count">Кількість «сміттєвих» пакетів</string>
|
||||
<string name="junk_packet_minimum_size">Мінімальний розмір «сміттєвого» пакету</string>
|
||||
<string name="junk_packet_maximum_size">Максимальний розмір «сміттєвого» пакету</string>
|
||||
<string name="init_packet_junk_size">Початковий розмір «сміттєвого» пакету</string>
|
||||
<string name="response_packet_junk_size">Розмір відповіді «сміттєвого» пакету</string>
|
||||
<string name="init_packet_magic_header">Заголовок пакету ініціалізації</string>
|
||||
<string name="underload_packet_magic_header">Заголовок пакету під навантаженням</string>
|
||||
<string name="response_packet_magic_header">Заголовок пакету відповіді</string>
|
||||
<string name="unsure_how">, якщо не впевнені що робити далі</string>
|
||||
<string name="see_the">Дивіться</string>
|
||||
</resources>
|
||||
@@ -176,4 +176,5 @@
|
||||
<string name="amnezia" translatable="false">Amnezia</string>
|
||||
<string name="wireguard" translatable="false">WireGuard</string>
|
||||
<string name="error_file_format">Invalid tunnel config format</string>
|
||||
<string name="restart_at_boot">Restart on boot</string>
|
||||
</resources>
|
||||
@@ -4,4 +4,12 @@
|
||||
<style name="Theme.WireguardAutoTunnel" parent="@style/Theme.AppCompat.NoActionBar">
|
||||
<item name="android:windowBackground">@color/black_background</item>
|
||||
</style>
|
||||
</resources>
|
||||
|
||||
<style name="Theme.AppSplashScreen" parent="Theme.SplashScreen">
|
||||
<!--<item name="windowSplashScreenBackground">@color/white</item>-->
|
||||
<!-- icon has to be a circle -->
|
||||
<item name="windowSplashScreenAnimatedIcon">@mipmap/ic_launcher</item>
|
||||
<item name="windowSplashScreenAnimationDuration">500</item>
|
||||
<item name="postSplashScreenTheme">@style/Theme.WireguardAutoTunnel</item>
|
||||
</style>
|
||||
</resources>
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
object Constants {
|
||||
const val VERSION_NAME = "3.4.6"
|
||||
const val VERSION_NAME = "3.4.7"
|
||||
const val JVM_TARGET = "17"
|
||||
const val VERSION_CODE = 34600
|
||||
const val VERSION_CODE = 34700
|
||||
const val TARGET_SDK = 34
|
||||
const val MIN_SDK = 26
|
||||
const val APP_ID = "com.zaneschepke.wireguardautotunnel"
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
Verbesserungen:
|
||||
- Unterstützung für Tunnelung nur bei Verwendung von Mobilen Daten
|
||||
- Unterstützung für Auto-Tunneln nur bei Verwendung von Mobilen Daten
|
||||
- Verbesserungen der Support Oberfläche
|
||||
- Aktualisierung der Ressourcenlinks
|
||||
- Verschiedene andere Fehlerbehebungen
|
||||
|
||||
@@ -0,0 +1,5 @@
|
||||
Was ist neu?
|
||||
- Zusätzliche Sprachunterstützung
|
||||
- Fehler beim automatischen Tunneln von mobilen Daten behoben
|
||||
- AndroidTV-Schaltfläche für schwebende Aktionen behoben
|
||||
- Weitere Optimierungen und Erweiterungen
|
||||
@@ -0,0 +1,4 @@
|
||||
Was ist neu?
|
||||
- Behebt Auto-Tunneling-Fehler
|
||||
- Behebt Android-Backup-Fehler
|
||||
- Versionen erhöhen
|
||||
@@ -0,0 +1,6 @@
|
||||
What's new:
|
||||
- Fix crashing issues
|
||||
- Improve tile performance
|
||||
- Re-enable pin lock
|
||||
- Make restart on boot a setting
|
||||
- Various performance and bug fixes
|
||||
@@ -0,0 +1,4 @@
|
||||
Novedades:
|
||||
- Config editar corrección de errores de interfaz de usuario
|
||||
- Añadir GrapheneOS primer lanzamiento AOVPN notificación
|
||||
- Versiones Bump
|
||||
@@ -0,0 +1,5 @@
|
||||
Novedades:
|
||||
- Añadir pantalla de registros
|
||||
- Añadir bloqueo local de aplicaciones
|
||||
- Añadir reiniciar vpn en ping fallido
|
||||
- Varios errores corregidos
|
||||
@@ -0,0 +1,5 @@
|
||||
Novedades:
|
||||
- Selección automática de túnel por nombre de WiFi
|
||||
- Control automático de túneles a través de mosaicos y atajos.
|
||||
- Reinicio automático del túnel manual después de un reinicio del sistema.
|
||||
- Varias correcciones de errores y mejoras de rendimiento
|
||||
@@ -0,0 +1,5 @@
|
||||
Novedades:
|
||||
- Mejora de la fiabilidad del túnel automático
|
||||
- Mejora de la sincronización de azulejos
|
||||
- Añadidos activos AndroidTV
|
||||
- Añadida huella digital al apk
|
||||
@@ -1,13 +1,14 @@
|
||||
Características:
|
||||
Características
|
||||
|
||||
- Añadir túneles vía .conf file, zip, manualmente, o código QR
|
||||
- Conexión automática a la VPN basado en el SSID Wi-Fi, ethernet, o datos móviles
|
||||
- División de túnel por aplicacióno con búsqueda
|
||||
- Compatibilidad WireGuard para modos kernel y userspace
|
||||
- Compatibilidad con VPN Siempre-Activada
|
||||
- Exportar túmeles como zip
|
||||
- Añadido interruptor VPN en el Centro de Control
|
||||
- Accesos directos estáticos para túnel principal para intergración con apps de automatización
|
||||
- Intents de apps de automatización para todos los túneles
|
||||
- Inicio automático del servicio tras reinicio
|
||||
- Medidas para ahorro de batería
|
||||
- Añade túneles a través de un archivo .conf, zip, entrada manual o código QR
|
||||
- Conexión automática a VPN basada en Wi-Fi SSID, ethernet o datos móviles
|
||||
- Túneles divididos por aplicación con búsqueda
|
||||
- Soporte de WireGuard para los modos kernel y espacio de usuario
|
||||
- Soporte de Amnezia para el modo de espacio de usuario para protección DPI/censura
|
||||
- Compatible con VPN siempre activa
|
||||
- Exportación de túneles Amnezia y WireGuard a zip
|
||||
- Compatibilidad con Quick Tile para alternar entre VPN
|
||||
- Soporte de accesos directos estáticos para el túnel principal para la integración de automatización.
|
||||
- Soporte de automatización de intenciones para todos los túneles
|
||||
- Reinicio automático del servicio tras reiniciar
|
||||
- Medidas de conservación de la batería
|
||||
|
||||
@@ -0,0 +1,3 @@
|
||||
Melhorias:
|
||||
- Corrige o bug de permissões do Android 9
|
||||
- Outras otimizações
|
||||
@@ -0,0 +1,5 @@
|
||||
Melhorias:
|
||||
- Adicionada estatísticas do túnel na tela principal
|
||||
- Melhoria de navegação de configurações na tela do AndroidTV
|
||||
- Removida a vibração nas notificações
|
||||
- Outras correções de bugs
|
||||
@@ -0,0 +1,14 @@
|
||||
Recursos
|
||||
|
||||
- Adiciona túneis por arquivos .conf, zip, manualmente ou por código QR
|
||||
- Auto connecta à VPN baseado no nome (SSID) do Wi-Fi, ethernet ou dados móveis
|
||||
- Túnel dividido por aplicativo com busca
|
||||
- Suporte à WireGuard em modo kernel ou usuário
|
||||
- Suporte à Amnezia em modo usuário para proteção contra censura e DPI (Inspeção Profunda de Pacote)
|
||||
- Suporte à VPN sempre ligada
|
||||
- Exportação de túneis Amnezia e WireGuard em arquivos zip
|
||||
- Suporte à quick tile para ligar e desligar a VPN
|
||||
- Atalhos para o túnel principal para integração com automações
|
||||
- Intent automation para todos os túneis
|
||||
- Início automático depois de reiniciar o aparelho
|
||||
- Medidas para economia de bateria
|
||||
@@ -0,0 +1 @@
|
||||
Um cliente de VPN alternativo para WireGuard com recursos adicionais
|
||||
@@ -0,0 +1 @@
|
||||
WG Tunnel
|
||||
@@ -0,0 +1,5 @@
|
||||
Что нового:
|
||||
- Добавлены новые переводы
|
||||
- Исправлена работа авто-туннеля на мобильном интернете
|
||||
- Исправлена проблема с плавающей кнопкой на Android TV
|
||||
- Другие оптимизации и улучшения
|
||||
@@ -0,0 +1,4 @@
|
||||
Что нового:
|
||||
- Исправлены проблемы с работой авто-туннеля
|
||||
- Исправлена проблема с резервным копированием Android
|
||||
- Увеличен номер версии
|
||||
@@ -0,0 +1,14 @@
|
||||
Можливості:
|
||||
|
||||
- Додавання тунелів з .conf, .zip файлів, ручним введенням або за допомогою QR коду
|
||||
- Автоматичне підключення до VPN залежно від імені мережі Wi-Fi, підключення по Ethernet чи мобільних даних
|
||||
- Розділення тунелів між програмами з пошуком
|
||||
- Підтримка Wireguard в режимі ядра і простору користувача
|
||||
- Підтримка Amnezia в режимі простору користувача для обходу DPI/інтернет цензури
|
||||
- Підтримка Always-ON VPN
|
||||
- Експорт тунелів Amnezia і Wireguard в zip файл
|
||||
- Підтримка плиток для швидкого переключенняVPN
|
||||
- Підтримка статичних ярликів основного тунелю для автоматизації
|
||||
- Підтримка автоматизації через intents для всіх тунелів
|
||||
- Автоматичний перезапуск сервісу після перезавантаження пристрою
|
||||
- Заходи по збережено заряду акумулятора
|
||||
@@ -0,0 +1 @@
|
||||
Альтернативний VPN-клієнт для WireGuard з додатковими функціями
|
||||
@@ -0,0 +1 @@
|
||||
WG Tunnel
|
||||
@@ -22,12 +22,13 @@ pinLockCompose = "1.0.3"
|
||||
roomVersion = "2.6.1"
|
||||
timber = "5.0.1"
|
||||
tunnel = "1.0.20230706"
|
||||
androidGradlePlugin = "8.4.1"
|
||||
androidGradlePlugin = "8.5.0"
|
||||
kotlin = "1.9.24"
|
||||
ksp = "1.9.24-1.0.20"
|
||||
composeBom = "2024.05.00"
|
||||
compose = "1.6.7"
|
||||
composeBom = "2024.06.00"
|
||||
compose = "1.6.8"
|
||||
zxingAndroidEmbedded = "4.3.0"
|
||||
coreSplashscreen = "1.0.1"
|
||||
|
||||
#plugins
|
||||
gradlePlugins-kotlinxSerialization = "1.9.24"
|
||||
@@ -74,6 +75,7 @@ androidx-core-ktx = { module = "androidx.core:core-ktx", version.ref = "coreKtx"
|
||||
androidx-espresso-core = { module = "androidx.test.espresso:espresso-core", version.ref = "espressoCore" }
|
||||
androidx-hilt-navigation-compose = { module = "androidx.hilt:hilt-navigation-compose", version.ref = "hiltNavigationCompose" }
|
||||
androidx-junit = { module = "androidx.test.ext:junit", version.ref = "androidx-junit" }
|
||||
androidx-core-splashscreen = { module = "androidx.core:core-splashscreen", version.ref = "coreSplashscreen" }
|
||||
androidx-lifecycle-runtime-ktx = { module = "androidx.lifecycle:lifecycle-runtime-ktx", version.ref = "lifecycle-runtime-compose" }
|
||||
androidx-material3 = { module = "androidx.compose.material3:material3", version.ref = "material3" }
|
||||
androidx-navigation-compose = { module = "androidx.navigation:navigation-compose", version.ref = "navigationCompose" }
|
||||
|
||||
Reference in New Issue
Block a user