diff --git a/app/build.gradle.kts b/app/build.gradle.kts index b47383b..6ce5264 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -17,7 +17,7 @@ android { val versionMajor = 2 val versionMinor = 3 - val versionPatch = 5 + val versionPatch = 6 val versionBuild = 0 defaultConfig { @@ -89,7 +89,7 @@ dependencies { implementation("com.jakewharton.timber:timber:5.0.1") // compose navigation - implementation("androidx.navigation:navigation-compose:2.7.0") + implementation("androidx.navigation:navigation-compose:2.7.1") implementation("androidx.hilt:hilt-navigation-compose:1.0.0") // hilt diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index 951bb88..716f987 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -64,6 +64,20 @@ android:foregroundServiceType="remoteMessaging" android:exported="false"> + + + + + + + + + @Inject + lateinit var configRepo : Repository + + @Inject + lateinit var vpnService : VpnService + + private val scope = CoroutineScope(Dispatchers.Main); + + private lateinit var job : Job + + override fun onStartListening() { + if (!this::job.isInitialized) { + job = scope.launch { + updateTileState() + } + } + Timber.d("On start listening") + super.onStartListening() + } + + override fun onTileAdded() { + super.onTileAdded() + qsTile.contentDescription = "Toggle VPN" + scope.launch { + updateTileState(); + } + } + + override fun onTileRemoved() { + super.onTileRemoved() + cancelJob() + } + + override fun onClick() { + unlockAndRun { + scope.launch { + try { + if(vpnService.getState() == Tunnel.State.UP) { + stopTunnel(); + return@launch + } + val settings = settingsRepo.getAll() + if (!settings.isNullOrEmpty()) { + val setting = settings.first() + if (setting.defaultTunnel != null) { + startTunnel(setting.defaultTunnel!!) + } else { + val config = configRepo.getAll()?.first(); + if(config != null) { + startTunnel(config.toString()); + } + } + } + } finally { + cancel() + } + } + super.onClick() + } + } + + private fun stopTunnel() { + ServiceTracker.actionOnService( + Action.STOP, this@TunnelControlTile, + WireGuardConnectivityWatcherService::class.java) + ServiceTracker.actionOnService( + Action.STOP, this@TunnelControlTile, + WireGuardTunnelService::class.java) + } + + private fun startTunnel(tunnelConfig : String) { + ServiceTracker.actionOnService( + Action.START, this.applicationContext, + WireGuardTunnelService::class.java, + mapOf(this.applicationContext.resources. + getString(R.string.tunnel_extras_key) to + tunnelConfig)) + } + + private suspend fun updateTileState() { + vpnService.state.collect { + when(it) { + Tunnel.State.UP -> { + setTileOn() + } + Tunnel.State.DOWN -> { + setTileOff() + } + else -> { + qsTile.state = Tile.STATE_UNAVAILABLE + } + } + qsTile.updateTile() + } + } + + private fun setTileOff() { + qsTile.state = Tile.STATE_INACTIVE; + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) { + qsTile.subtitle = "Off" + } + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { + qsTile.stateDescription = "VPN Off"; + } + } + + private fun setTileOn() { + qsTile.state = Tile.STATE_ACTIVE; + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) { + qsTile.subtitle = "On" + } + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { + qsTile.stateDescription = "VPN On"; + } + } + private fun cancelJob() { + if(this::job.isInitialized) { + job.cancel(); + } + } +} \ No newline at end of file diff --git a/app/src/main/java/com/zaneschepke/wireguardautotunnel/ui/MainActivity.kt b/app/src/main/java/com/zaneschepke/wireguardautotunnel/ui/MainActivity.kt index c25dc78..4e5758a 100644 --- a/app/src/main/java/com/zaneschepke/wireguardautotunnel/ui/MainActivity.kt +++ b/app/src/main/java/com/zaneschepke/wireguardautotunnel/ui/MainActivity.kt @@ -152,7 +152,7 @@ class MainActivity : AppCompatActivity() { } } }) { - MainScreen(padding = padding, snackbarHostState = snackbarHostState, navController = navController, focusRequester = focusRequester) + MainScreen(padding = padding, snackbarHostState = snackbarHostState, navController = navController) } composable(Routes.Settings.name, enterTransition = { when (initialState.destination.route) { diff --git a/app/src/main/java/com/zaneschepke/wireguardautotunnel/ui/screens/main/MainScreen.kt b/app/src/main/java/com/zaneschepke/wireguardautotunnel/ui/screens/main/MainScreen.kt index 2cb3e0a..162c497 100644 --- a/app/src/main/java/com/zaneschepke/wireguardautotunnel/ui/screens/main/MainScreen.kt +++ b/app/src/main/java/com/zaneschepke/wireguardautotunnel/ui/screens/main/MainScreen.kt @@ -18,6 +18,7 @@ import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.padding import androidx.compose.foundation.lazy.LazyColumn import androidx.compose.foundation.lazy.items +import androidx.compose.foundation.lazy.itemsIndexed import androidx.compose.foundation.shape.RoundedCornerShape import androidx.compose.material.icons.Icons import androidx.compose.material.icons.filled.FileOpen @@ -85,7 +86,6 @@ import kotlinx.coroutines.launch @Composable fun MainScreen( viewModel: MainViewModel = hiltViewModel(), padding: PaddingValues, - focusRequester: FocusRequester, snackbarHostState: SnackbarHostState, navController: NavController ) { @@ -149,7 +149,7 @@ fun MainScreen( }) }, floatingActionButtonPosition = FabPosition.End, - floatingActionButton = { + floatingActionButton = { AnimatedVisibility( visible = isVisible.value, enter = slideInVertically(initialOffsetY = { it * 2 }), @@ -241,9 +241,12 @@ fun MainScreen( .padding(padding) ) { - LazyColumn(modifier = Modifier.fillMaxSize() - .nestedScroll(nestedScrollConnection),) { - items(tunnels.toList()) { tunnel -> + LazyColumn( + modifier = Modifier.fillMaxSize() + .nestedScroll(nestedScrollConnection), + ) { + itemsIndexed(tunnels.toList()) { index, tunnel -> + val focusRequester = FocusRequester(); RowListItem(leadingIcon = Icons.Rounded.Circle, leadingIconColor = if (tunnelName == tunnel.name) when (handshakeStatus) { HandshakeStatus.HEALTHY -> mint @@ -263,15 +266,15 @@ fun MainScreen( selectedTunnel = tunnel; }, onClick = { - if(!WireGuardAutoTunnel.isRunningOnAndroidTv(context)){ + if (!WireGuardAutoTunnel.isRunningOnAndroidTv(context)) { navController.navigate("${Routes.Detail.name}/${tunnel.id}") } else { focusRequester.requestFocus() } - }, + }, rowButton = { if (tunnel.id == selectedTunnel?.id) { - Row() { + Row { IconButton(onClick = { navController.navigate("${Routes.Config.name}/${selectedTunnel?.id}") }) { @@ -287,30 +290,43 @@ fun MainScreen( } } } else { - if(WireGuardAutoTunnel.isRunningOnAndroidTv(context)){ - Row() { - IconButton(modifier = Modifier.focusRequester(focusRequester),onClick = { - navController.navigate("${Routes.Detail.name}/${tunnel.id}") - }) { + if (WireGuardAutoTunnel.isRunningOnAndroidTv(context)) { + Row { + IconButton( + modifier = Modifier.focusRequester(focusRequester), + onClick = { + navController.navigate("${Routes.Detail.name}/${tunnel.id}") + }) { Icon(Icons.Rounded.Info, "Info") } IconButton(onClick = { if (state == Tunnel.State.UP && tunnel.name == tunnelName) scope.launch { - viewModel.showSnackBarMessage(context.resources.getString(R.string.turn_off_tunnel)) + viewModel.showSnackBarMessage( + context.resources.getString( + R.string.turn_off_tunnel + ) + ) } else { - navController.navigate("${Routes.Config.name}/${tunnel.id}") - } + navController.navigate("${Routes.Config.name}/${tunnel.id}") + } }) { - Icon(Icons.Rounded.Edit, stringResource(id = R.string.edit)) + Icon( + Icons.Rounded.Edit, + stringResource(id = R.string.edit) + ) } IconButton(onClick = { if (state == Tunnel.State.UP && tunnel.name == tunnelName) scope.launch { - viewModel.showSnackBarMessage(context.resources.getString(R.string.turn_off_tunnel)) + viewModel.showSnackBarMessage( + context.resources.getString( + R.string.turn_off_tunnel + ) + ) } else { - viewModel.onDelete(tunnel) - } + viewModel.onDelete(tunnel) + } }) { Icon( Icons.Rounded.Delete, diff --git a/app/src/main/res/drawable/shield.xml b/app/src/main/res/drawable/shield.xml new file mode 100644 index 0000000..e49a1f2 --- /dev/null +++ b/app/src/main/res/drawable/shield.xml @@ -0,0 +1,5 @@ + + + diff --git a/build.gradle.kts b/build.gradle.kts index 1acbd18..f9d7e05 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -13,7 +13,7 @@ buildscript { } plugins { - id("com.android.application") version "8.2.0-alpha15" apply false + id("com.android.application") version "8.2.0-beta01" apply false id("org.jetbrains.kotlin.android") version "1.8.22" apply false id("com.google.dagger.hilt.android") version "2.44" apply false kotlin("plugin.serialization") version "1.8.22" apply false