From 36a74f4912f0461920220fd2cdd5781eacf0398f Mon Sep 17 00:00:00 2001 From: Christian Rowlands Date: Mon, 13 Jan 2025 11:58:10 -0500 Subject: [PATCH 1/2] #8964 Fix app crash on incoming call when running Android 14+ --- changelog.d/8964.bugfix | 1 + .../vector/app/features/call/VectorCallActivity.kt | 13 ++++++++++--- 2 files changed, 11 insertions(+), 3 deletions(-) create mode 100644 changelog.d/8964.bugfix diff --git a/changelog.d/8964.bugfix b/changelog.d/8964.bugfix new file mode 100644 index 00000000000..2b9144a8f5d --- /dev/null +++ b/changelog.d/8964.bugfix @@ -0,0 +1 @@ +Fix incoming call crash on Android 14+. ([#8964](https://github.com/element-hq/element-android/issues/8964)) diff --git a/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt b/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt index dcbb5e5d942..99fdfb2a489 100644 --- a/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt +++ b/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt @@ -246,9 +246,16 @@ class VectorCallActivity : == PackageManager.PERMISSION_GRANTED) { // Only start the service if the app is in the foreground if (isAppInForeground()) { - Timber.tag(loggerTag.value).v("Starting microphone foreground service") - val intent = Intent(this, MicrophoneAccessService::class.java) - ContextCompat.startForegroundService(this, intent) + // Starting in Android 14, you can't create a microphone foreground service while your app is in + // the background. If we call startForegroundService the app will crash. + // https://github.com/element-hq/element-android/issues/8964 + if (Build.VERSION.SDK_INT < Build.VERSION_CODES.UPSIDE_DOWN_CAKE) { + Timber.tag(loggerTag.value).v("Starting microphone foreground service") + val intent = Intent(this, MicrophoneAccessService::class.java) + ContextCompat.startForegroundService(this, intent) + } else { + Timber.tag(loggerTag.value).v("App is in running Android 14+; cannot start microphone service") + } } else { Timber.tag(loggerTag.value).v("App is not in foreground; cannot start microphone service") } From af906ce67b8bbc930ea314c57abd0422a158e2fc Mon Sep 17 00:00:00 2001 From: Christian Rowlands Date: Mon, 20 Jan 2025 10:54:08 -0500 Subject: [PATCH 2/2] #8964 Start the MicrophoneAccessService during onPause if the call has been answered --- .../app/features/call/VectorCallActivity.kt | 26 ++++++++++++------- 1 file changed, 17 insertions(+), 9 deletions(-) diff --git a/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt b/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt index 99fdfb2a489..e8143b58c58 100644 --- a/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt +++ b/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt @@ -246,15 +246,20 @@ class VectorCallActivity : == PackageManager.PERMISSION_GRANTED) { // Only start the service if the app is in the foreground if (isAppInForeground()) { - // Starting in Android 14, you can't create a microphone foreground service while your app is in - // the background. If we call startForegroundService the app will crash. - // https://github.com/element-hq/element-android/issues/8964 - if (Build.VERSION.SDK_INT < Build.VERSION_CODES.UPSIDE_DOWN_CAKE) { - Timber.tag(loggerTag.value).v("Starting microphone foreground service") - val intent = Intent(this, MicrophoneAccessService::class.java) - ContextCompat.startForegroundService(this, intent) - } else { - Timber.tag(loggerTag.value).v("App is in running Android 14+; cannot start microphone service") + withState(callViewModel) { + // Starting in Android 14, you can't create a microphone foreground service while your app is in + // the background. If we call startForegroundService while the call state is ringing (i.e. the + // user has not interacted with the device at all) the app will crash. Make sure the call has + // already been answered before starting the MicrophoneAccessService + // https://github.com/element-hq/element-android/issues/8964 + val callState = it.callState.invoke() + if (callState !is CallState.LocalRinging && callState !is CallState.Ended) { + Timber.tag(loggerTag.value).v("Starting microphone foreground service") + val intent = Intent(this, MicrophoneAccessService::class.java) + ContextCompat.startForegroundService(this, intent) + } else { + Timber.tag(loggerTag.value).v("Call is in ringing or ended state; cannot start microphone service. callState: $callState") + } } } else { Timber.tag(loggerTag.value).v("App is not in foreground; cannot start microphone service") @@ -276,6 +281,9 @@ class VectorCallActivity : override fun onPause() { super.onPause() + + // Start the microphone service to keep access to the microphone when the call is in the background + // https://github.com/element-hq/element-android/issues/8881 startMicrophoneService() }