diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml
index 93f6ce8..a5f8831 100644
--- a/app/src/main/AndroidManifest.xml
+++ b/app/src/main/AndroidManifest.xml
@@ -85,7 +85,7 @@
@@ -97,4 +97,4 @@
-
\ No newline at end of file
+
diff --git a/app/src/main/java/de/jrpie/android/launcher/actions/LauncherAction.kt b/app/src/main/java/de/jrpie/android/launcher/actions/LauncherAction.kt
index 5d2be94..ee9502c 100644
--- a/app/src/main/java/de/jrpie/android/launcher/actions/LauncherAction.kt
+++ b/app/src/main/java/de/jrpie/android/launcher/actions/LauncherAction.kt
@@ -11,7 +11,9 @@ import android.view.KeyEvent
import android.widget.Toast
import androidx.appcompat.content.res.AppCompatResources
import de.jrpie.android.launcher.Application
+import de.jrpie.android.launcher.BuildConfig
import de.jrpie.android.launcher.R
+import de.jrpie.android.launcher.actions.lock.LauncherAccessibilityService
import de.jrpie.android.launcher.apps.AppFilter
import de.jrpie.android.launcher.apps.hidePrivateSpaceWhenLocked
import de.jrpie.android.launcher.apps.isPrivateSpaceSupported
@@ -132,6 +134,14 @@ enum class LauncherAction(
R.drawable.baseline_settings_applications_24,
::expandSettingsPanel
),
+ RECENT_APPS(
+ "recent_apps",
+ R.string.list_other_recent_apps,
+ R.drawable.baseline_apps_24,
+ LauncherAccessibilityService::openRecentApps,
+ false,
+ { _ -> BuildConfig.USE_ACCESSIBILITY_SERVICE }
+ ),
LOCK_SCREEN(
"lock_screen",
R.string.list_other_lock_screen,
@@ -142,7 +152,7 @@ enum class LauncherAction(
"toggle_torch",
R.string.list_other_torch,
R.drawable.baseline_flashlight_on_24,
- ::toggleTorch
+ ::toggleTorch,
),
NOP("nop", R.string.list_other_nop, R.drawable.baseline_not_interested_24, {});
diff --git a/app/src/main/java/de/jrpie/android/launcher/actions/lock/LauncherAccessibilityService.kt b/app/src/main/java/de/jrpie/android/launcher/actions/lock/LauncherAccessibilityService.kt
index a8ef6f2..7cb32d9 100644
--- a/app/src/main/java/de/jrpie/android/launcher/actions/lock/LauncherAccessibilityService.kt
+++ b/app/src/main/java/de/jrpie/android/launcher/actions/lock/LauncherAccessibilityService.kt
@@ -22,26 +22,44 @@ class LauncherAccessibilityService : AccessibilityService() {
companion object {
private const val TAG = "Launcher Accessibility"
+ private const val ACTION_REQUEST_ENABLE = "ACTION_REQUEST_ENABLE"
const val ACTION_LOCK_SCREEN = "ACTION_LOCK_SCREEN"
+ const val ACTION_RECENT_APPS = "ACTION_RECENT_APPS"
- fun lockScreen(context: Context) {
+ private fun invoke(context: Context, action: String, failureMessageRes: Int) {
try {
context.startService(
Intent(
context,
LauncherAccessibilityService::class.java
).apply {
- action = ACTION_LOCK_SCREEN
+ this.action = action
})
- } catch (e: Exception) {
+ } catch (_: Exception) {
Toast.makeText(
context,
- context.getString(R.string.alert_lock_screen_failed),
+ context.getString(failureMessageRes),
Toast.LENGTH_LONG
).show()
}
}
+ fun lockScreen(context: Context) {
+ if (!isEnabled(context)) {
+ showEnableDialog(context)
+ } else {
+ invoke(context, ACTION_LOCK_SCREEN, R.string.alert_lock_screen_failed)
+ }
+ }
+
+ fun openRecentApps(context: Context) {
+ if (!isEnabled(context)) {
+ showEnableDialog(context)
+ } else {
+ invoke(context, ACTION_RECENT_APPS, R.string.alert_recent_apps_failed)
+ }
+ }
+
fun isEnabled(context: Context): Boolean {
val enabledServices = Settings.Secure.getString(
context.contentResolver,
@@ -58,7 +76,7 @@ class LauncherAccessibilityService : AccessibilityService() {
setView(R.layout.dialog_consent_accessibility)
setTitle(R.string.dialog_consent_accessibility_title)
setPositiveButton(R.string.dialog_consent_accessibility_ok) { _, _ ->
- lockScreen(context)
+ invoke(context, ACTION_REQUEST_ENABLE, R.string.alert_enable_accessibility_failed)
}
setNegativeButton(R.string.dialog_cancel) { _, _ -> }
}.create().also { it.show() }.apply {
@@ -94,7 +112,9 @@ class LauncherAccessibilityService : AccessibilityService() {
}
when (action) {
+ ACTION_REQUEST_ENABLE -> {} // do nothing
ACTION_LOCK_SCREEN -> handleLockScreen()
+ ACTION_RECENT_APPS -> performGlobalAction(GLOBAL_ACTION_RECENTS)
}
}
return super.onStartCommand(intent, flags, startId)
diff --git a/app/src/main/java/de/jrpie/android/launcher/actions/lock/LockMethod.kt b/app/src/main/java/de/jrpie/android/launcher/actions/lock/LockMethod.kt
index 541510a..93b4cbf 100644
--- a/app/src/main/java/de/jrpie/android/launcher/actions/lock/LockMethod.kt
+++ b/app/src/main/java/de/jrpie/android/launcher/actions/lock/LockMethod.kt
@@ -6,6 +6,7 @@ import android.widget.Button
import androidx.appcompat.app.AlertDialog
import de.jrpie.android.launcher.BuildConfig
import de.jrpie.android.launcher.R
+import de.jrpie.android.launcher.actions.lock.LauncherAccessibilityService
import de.jrpie.android.launcher.preferences.LauncherPreferences
diff --git a/app/src/main/res/drawable/baseline_apps_24.xml b/app/src/main/res/drawable/baseline_apps_24.xml
new file mode 100644
index 0000000..c5a49a0
--- /dev/null
+++ b/app/src/main/res/drawable/baseline_apps_24.xml
@@ -0,0 +1,11 @@
+
+
+
+
+
diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml
index c426a5c..83b963b 100644
--- a/app/src/main/res/values/strings.xml
+++ b/app/src/main/res/values/strings.xml
@@ -252,6 +252,7 @@
Music: Previous
Music: Play / Pause
Expand notifications panel
+ Recent Apps
Do nothing
Lock Screen
Toggle Torch
@@ -307,6 +308,8 @@
No camera with torch detected.
Error: Can\'t access torch.
Error: Failed to lock screen. (If you just upgraded the app, try to disable and re-enable the accessibility service in phone settings)
+ Error: Failed to show recent apps. (If you just upgraded the app, try to disable and re-enable the accessibility service in phone settings)
+ Error: Failed to enable the accessibility service.
μLauncher\'s accessibility service is not enabled. Please enable it in settings
Private space locked
Private space unlocked
@@ -315,12 +318,17 @@
Lock private space
Unlock private space
Error: Locking the screen using accessibility is not supported on this device. Please use device admin instead.
- μLauncher - lock screen
+ μLauncher
- Setting μLauncher as an accessibility service allows it to lock the screen.
+ Setting μLauncher as an accessibility service allows it to lock the screen and open the recent apps menu.
Note that excessive permissions are required. You should never grant such permissions lightly to any app.
- μLauncher will use the accessibility service only for locking the screen. You can check the source code to make sure.
+ μLauncher will use the accessibility service only for performing the following actions when requested by the user:
+
+ * lock screen
+ * open recent apps
+
+ μLauncher will never use the accessibility service to collect data. You can check the source code to make sure.
Note that locking the screen can also be accomplished by granting μLauncher device administrator permissions. However that method doesn\'t work with fingerprint and face unlock.
@@ -365,7 +373,12 @@
I am aware that other options exist (using device administrator privileges or the power button).
I consent to μLauncher using the accessibility service to provide functionality unrelated to accessibility.
I consent to μLauncher not collecting any data.
- far-reaching privileges to μLauncher.
μLauncher will use these privileges only to lock the screen. μLauncher will never collect any data. In particular, μLauncher does not use the accessibility service to collect any data.]]>
+ far-reaching privileges to μLauncher.
μLauncher will use these privileges only to perform the following actions:
+
+ - Lock Screen
+ - Recent Apps
+
+ μLauncher will never collect any data. In particular, μLauncher does not use the accessibility service to collect any data.]]>
Activating the Accessibility Service
Activate Accessibility Service
Cancel