feat: support sending voice messages as replies

This commit is contained in:
Gianluca Iavicoli 2026-03-25 00:53:57 +01:00
parent aa5b1f5a07
commit bf7ab31517
No known key found for this signature in database
GPG key ID: 72FDD0FCF146C74C
2 changed files with 57 additions and 12 deletions

View file

@ -34,10 +34,12 @@ import io.element.android.libraries.audio.api.AudioFocus
import io.element.android.libraries.audio.api.AudioFocusRequester
import io.element.android.libraries.di.RoomScope
import io.element.android.libraries.di.annotations.SessionCoroutineScope
import io.element.android.libraries.matrix.api.core.EventId
import io.element.android.libraries.matrix.api.timeline.Timeline
import io.element.android.libraries.mediaupload.api.MediaSenderFactory
import io.element.android.libraries.permissions.api.PermissionsEvent
import io.element.android.libraries.permissions.api.PermissionsPresenter
import io.element.android.libraries.textcomposer.model.MessageComposerMode
import io.element.android.libraries.textcomposer.model.VoiceMessagePlayerEvent
import io.element.android.libraries.textcomposer.model.VoiceMessageRecorderEvent
import io.element.android.libraries.textcomposer.model.VoiceMessageState
@ -151,7 +153,7 @@ class DefaultVoiceMessageComposerPresenter(
}
}
fun sendVoiceMessage() {
fun sendVoiceMessage(inReplyToEventId: EventId?, composerEvent: Composer) {
val finishedState = recorderState as? VoiceRecorderState.Finished
if (finishedState == null) {
val exception = VoiceMessageException.FileException("No file to send")
@ -164,12 +166,13 @@ class DefaultVoiceMessageComposerPresenter(
}
isSending = true
player.pause()
analyticsService.captureComposerEvent()
analyticsService.capture(composerEvent)
sessionCoroutineScope.launch {
val result = sendMessage(
file = finishedState.file,
mimeType = finishedState.mimeType,
waveform = finishedState.waveform,
inReplyToEventId = inReplyToEventId,
)
if (result.isFailure) {
showSendFailureDialog = true
@ -183,8 +186,14 @@ class DefaultVoiceMessageComposerPresenter(
when (event) {
is VoiceMessageComposerEvent.RecorderEvent -> handleVoiceMessageRecorderEvent(event.recorderEvent)
is VoiceMessageComposerEvent.PlayerEvent -> handleVoiceMessagePlayerEvent(event.playerEvent)
is VoiceMessageComposerEvent.SendVoiceMessage -> localCoroutineScope.launch {
sendVoiceMessage()
is VoiceMessageComposerEvent.SendVoiceMessage -> {
// Capture mode eagerly before any coroutine dispatch, since CloseSpecialMode
// may reset it before the coroutine runs.
val inReplyToEventId = (messageComposerContext.composerMode as? MessageComposerMode.Reply)?.eventId
val composerEvent = buildComposerEvent()
localCoroutineScope.launch {
sendVoiceMessage(inReplyToEventId, composerEvent)
}
}
VoiceMessageComposerEvent.DeleteVoiceMessage -> {
player.pause()
@ -280,11 +289,13 @@ class DefaultVoiceMessageComposerPresenter(
file: File,
mimeType: String,
waveform: List<Float>,
inReplyToEventId: EventId? = null,
): Result<Unit> {
val result = mediaSender.sendVoiceMessage(
uri = file.toUri(),
mimeType = mimeType,
waveForm = waveform,
inReplyToEventId = inReplyToEventId,
)
if (result.isFailure) {
@ -297,14 +308,12 @@ class DefaultVoiceMessageComposerPresenter(
return result
}
private fun AnalyticsService.captureComposerEvent() =
capture(
Composer(
inThread = messageComposerContext.composerMode.inThread,
isEditing = messageComposerContext.composerMode.isEditing,
isReply = messageComposerContext.composerMode.isReply,
messageType = Composer.MessageType.VoiceMessage,
)
private fun buildComposerEvent() =
Composer(
inThread = messageComposerContext.composerMode.inThread,
isEditing = messageComposerContext.composerMode.isEditing,
isReply = messageComposerContext.composerMode.isReply,
messageType = Composer.MessageType.VoiceMessage,
)
}

View file

@ -24,6 +24,7 @@ import io.element.android.libraries.audio.api.AudioFocusRequester
import io.element.android.libraries.matrix.api.core.EventId
import io.element.android.libraries.matrix.api.media.AudioInfo
import io.element.android.libraries.matrix.api.timeline.Timeline
import io.element.android.libraries.matrix.test.AN_EVENT_ID
import io.element.android.libraries.matrix.test.media.FakeMediaUploadHandler
import io.element.android.libraries.matrix.test.room.FakeJoinedRoom
import io.element.android.libraries.matrix.test.timeline.FakeTimeline
@ -46,7 +47,9 @@ import io.element.android.libraries.voicerecorder.api.VoiceRecorder
import io.element.android.libraries.voicerecorder.test.FakeVoiceRecorder
import io.element.android.services.analytics.test.FakeAnalyticsService
import io.element.android.tests.testutils.WarmUpRule
import io.element.android.tests.testutils.lambda.any
import io.element.android.tests.testutils.lambda.lambdaRecorder
import io.element.android.tests.testutils.lambda.value
import io.element.android.tests.testutils.test
import kotlinx.collections.immutable.toImmutableList
import kotlinx.coroutines.ExperimentalCoroutinesApi
@ -59,6 +62,7 @@ import java.io.File
import kotlin.time.Duration
import kotlin.time.Duration.Companion.seconds
@Suppress("LargeClass")
class DefaultVoiceMessageComposerPresenterTest {
@get:Rule
val warmUpRule = WarmUpRule()
@ -406,6 +410,38 @@ class DefaultVoiceMessageComposerPresenterTest {
}
}
@Test
fun `present - send voice message passes reply event ID only when in reply mode`() = runTest {
val presenter = createDefaultVoiceMessageComposerPresenter()
presenter.test {
// Send without reply - should pass null
messageComposerContext.composerMode = MessageComposerMode.Normal
awaitItem().eventSink(VoiceMessageComposerEvent.RecorderEvent(VoiceMessageRecorderEvent.Start))
awaitItem().eventSink(VoiceMessageComposerEvent.RecorderEvent(VoiceMessageRecorderEvent.Stop))
awaitItem().eventSink(VoiceMessageComposerEvent.SendVoiceMessage)
skipItems(1) // Sending state
advanceUntilIdle()
sendVoiceMessageResult.assertions().isCalledOnce()
.with(any(), any(), any(), value(null))
// Send as reply - should pass event ID
messageComposerContext.composerMode = aReplyMode()
awaitItem().eventSink(VoiceMessageComposerEvent.RecorderEvent(VoiceMessageRecorderEvent.Start))
awaitItem().eventSink(VoiceMessageComposerEvent.RecorderEvent(VoiceMessageRecorderEvent.Stop))
awaitItem().eventSink(VoiceMessageComposerEvent.SendVoiceMessage)
val finalState = awaitItem() // Sending state
sendVoiceMessageResult.assertions().isCalledExactly(2)
.withSequence(
listOf(any(), any(), any(), value(null)),
listOf(any(), any(), any(), value(AN_EVENT_ID)),
)
testPauseAndDestroy(finalState)
}
}
@Test
fun `present - send while playing`() = runTest {
val presenter = createDefaultVoiceMessageComposerPresenter()