From bef82f9c7abbc0792446dcf31a601832cd4d8532 Mon Sep 17 00:00:00 2001 From: "sergei.bakhtiarov" Date: Thu, 19 Dec 2024 13:08:22 +0100 Subject: [PATCH] feat: send and receive in-call reactions [#WPB-14254] --- .../android/di/accountScoped/CallsModule.kt | 5 + .../android/di/accountScoped/MessageModule.kt | 6 + .../android/mapper/UICallParticipantMapper.kt | 4 +- .../ui/calling/SharedCallingViewModel.kt | 94 +++++- .../ui/calling/controlbuttons/CameraButton.kt | 4 - .../ui/calling/controlbuttons/HangUpButton.kt | 2 +- .../controlbuttons/HangUpOngoingButton.kt | 67 ++++ ...FlipButton.kt => InCallReactionsButton.kt} | 43 +-- .../controlbuttons/MicrophoneButton.kt | 4 - .../calling/controlbuttons/SpeakerButton.kt | 4 - .../controlbuttons/WireCallControlButton.kt | 14 +- .../ui/calling/model/InCallReaction.kt | 29 ++ .../ui/calling/model/UICallParticipant.kt | 1 + .../ui/calling/ongoing/OngoingCallScreen.kt | 285 +++++++++++------- .../calling/ongoing/OngoingCallViewModel.kt | 2 +- .../ongoing/fullscreen/FullScreenTile.kt | 9 +- .../incallreactions/InCallReactions.kt | 52 ++++ .../InCallReactionsModifier.kt | 137 +++++++++ .../incallreactions/InCallReactionsPanel.kt | 170 +++++++++++ .../incallreactions/InCallReactionsState.kt | 114 +++++++ .../participantsview/FlipCameraButton.kt | 66 ++++ .../participantsview/FloatingSelfUserTile.kt | 7 +- .../participantsview/ParticipantTile.kt | 90 +++++- .../participantsview/VerticalCallingPager.kt | 16 +- .../gridview/CallingGridView.kt | 21 +- .../horizentalview/CallingHorizontalView.kt | 16 +- .../com/wire/android/util/ExpiringMap.kt | 74 +++++ .../com/wire/android/util/extension/Flow.kt | 9 + app/src/main/res/drawable/ic_flip_camera.xml | 30 ++ .../main/res/drawable/ic_incall_reactions.xml | 32 ++ app/src/main/res/values/strings.xml | 3 + .../mapper/UICallParticipantMapperTest.kt | 30 +- .../ui/calling/OngoingCallViewModelTest.kt | 4 + .../ui/calling/SharedCallingViewModelTest.kt | 125 +++++++- .../com/wire/android/util/ExpiringMapTest.kt | 116 +++++++ .../wire/android/ui/theme/WireColorScheme.kt | 4 + .../wire/android/ui/theme/WireDimensions.kt | 10 +- .../wire/android/ui/theme/WireTypography.kt | 8 +- .../android/ui/theme/WireTypographyBase.kt | 6 + kalium | 2 +- 40 files changed, 1516 insertions(+), 199 deletions(-) create mode 100644 app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/HangUpOngoingButton.kt rename app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/{CameraFlipButton.kt => InCallReactionsButton.kt} (54%) create mode 100644 app/src/main/kotlin/com/wire/android/ui/calling/model/InCallReaction.kt create mode 100644 app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactions.kt create mode 100644 app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsModifier.kt create mode 100644 app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsPanel.kt create mode 100644 app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsState.kt create mode 100644 app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/FlipCameraButton.kt create mode 100644 app/src/main/kotlin/com/wire/android/util/ExpiringMap.kt create mode 100644 app/src/main/res/drawable/ic_flip_camera.xml create mode 100644 app/src/main/res/drawable/ic_incall_reactions.xml create mode 100644 app/src/test/kotlin/com/wire/android/util/ExpiringMapTest.kt diff --git a/app/src/main/kotlin/com/wire/android/di/accountScoped/CallsModule.kt b/app/src/main/kotlin/com/wire/android/di/accountScoped/CallsModule.kt index 803e627db87..b4252d8cce3 100644 --- a/app/src/main/kotlin/com/wire/android/di/accountScoped/CallsModule.kt +++ b/app/src/main/kotlin/com/wire/android/di/accountScoped/CallsModule.kt @@ -202,4 +202,9 @@ class CallsModule { @Provides fun provideObserveConferenceCallingEnabledUseCase(callsScope: CallsScope) = callsScope.observeConferenceCallingEnabled + + @ViewModelScoped + @Provides + fun provideObserveInCallReactionsUseCase(callsScope: CallsScope) = + callsScope.observeInCallReactions } diff --git a/app/src/main/kotlin/com/wire/android/di/accountScoped/MessageModule.kt b/app/src/main/kotlin/com/wire/android/di/accountScoped/MessageModule.kt index ad72e7b2fea..2c1f3dd1692 100644 --- a/app/src/main/kotlin/com/wire/android/di/accountScoped/MessageModule.kt +++ b/app/src/main/kotlin/com/wire/android/di/accountScoped/MessageModule.kt @@ -28,6 +28,7 @@ import com.wire.kalium.logic.feature.asset.ObserveAssetStatusesUseCase import com.wire.kalium.logic.feature.asset.ObservePaginatedAssetImageMessages import com.wire.kalium.logic.feature.asset.ScheduleNewAssetMessageUseCase import com.wire.kalium.logic.feature.asset.UpdateAssetMessageTransferStatusUseCase +import com.wire.kalium.logic.feature.incallreaction.SendInCallReactionUseCase import com.wire.kalium.logic.feature.message.DeleteMessageUseCase import com.wire.kalium.logic.feature.message.GetMessageByIdUseCase import com.wire.kalium.logic.feature.message.GetNotificationsUseCase @@ -216,4 +217,9 @@ class MessageModule { @Provides fun provideRemoveMessageDraftUseCase(messageScope: MessageScope): RemoveMessageDraftUseCase = messageScope.removeMessageDraftUseCase + + @ViewModelScoped + @Provides + fun provideSendInCallReactionUseCase(messageScope: MessageScope): SendInCallReactionUseCase = + messageScope.sendInCallReactionUseCase } diff --git a/app/src/main/kotlin/com/wire/android/mapper/UICallParticipantMapper.kt b/app/src/main/kotlin/com/wire/android/mapper/UICallParticipantMapper.kt index 3ce437b99df..78f0a45cc0a 100644 --- a/app/src/main/kotlin/com/wire/android/mapper/UICallParticipantMapper.kt +++ b/app/src/main/kotlin/com/wire/android/mapper/UICallParticipantMapper.kt @@ -21,14 +21,16 @@ package com.wire.android.mapper import com.wire.android.model.ImageAsset import com.wire.android.ui.calling.model.UICallParticipant import com.wire.kalium.logic.data.call.Participant +import com.wire.kalium.logic.data.conversation.ClientId import javax.inject.Inject class UICallParticipantMapper @Inject constructor( private val userTypeMapper: UserTypeMapper, ) { - fun toUICallParticipant(participant: Participant) = UICallParticipant( + fun toUICallParticipant(participant: Participant, currentClientId: ClientId) = UICallParticipant( id = participant.id, clientId = participant.clientId, + isSelfUser = participant.clientId == currentClientId.value, name = participant.name, isMuted = participant.isMuted, isSpeaking = participant.isSpeaking, diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/SharedCallingViewModel.kt b/app/src/main/kotlin/com/wire/android/ui/calling/SharedCallingViewModel.kt index 807f0600b43..536d175b451 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/SharedCallingViewModel.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/SharedCallingViewModel.kt @@ -20,6 +20,7 @@ package com.wire.android.ui.calling import android.view.View import androidx.compose.runtime.getValue +import androidx.compose.runtime.mutableStateMapOf import androidx.compose.runtime.mutableStateOf import androidx.compose.runtime.setValue import androidx.lifecycle.ViewModel @@ -29,26 +30,37 @@ import com.wire.android.mapper.UICallParticipantMapper import com.wire.android.mapper.UserTypeMapper import com.wire.android.media.CallRinger import com.wire.android.model.ImageAsset +import com.wire.android.ui.calling.model.InCallReaction +import com.wire.android.ui.calling.model.ReactionSender import com.wire.android.ui.calling.model.UICallParticipant +import com.wire.android.ui.calling.ongoing.incallreactions.InCallReactions +import com.wire.android.util.ExpiringMap import com.wire.android.util.dispatchers.DispatcherProvider +import com.wire.android.util.extension.withDelayAfterFirst import com.wire.kalium.logic.data.call.Call import com.wire.kalium.logic.data.call.ConversationTypeForCall import com.wire.kalium.logic.data.call.VideoState import com.wire.kalium.logic.data.conversation.Conversation import com.wire.kalium.logic.data.conversation.ConversationDetails import com.wire.kalium.logic.data.id.ConversationId +import com.wire.kalium.logic.data.id.QualifiedID +import com.wire.kalium.logic.data.user.UserId import com.wire.kalium.logic.feature.call.usecase.EndCallUseCase import com.wire.kalium.logic.feature.call.usecase.FlipToBackCameraUseCase import com.wire.kalium.logic.feature.call.usecase.FlipToFrontCameraUseCase import com.wire.kalium.logic.feature.call.usecase.MuteCallUseCase import com.wire.kalium.logic.feature.call.usecase.ObserveEstablishedCallWithSortedParticipantsUseCase +import com.wire.kalium.logic.feature.call.usecase.ObserveInCallReactionsUseCase import com.wire.kalium.logic.feature.call.usecase.ObserveSpeakerUseCase import com.wire.kalium.logic.feature.call.usecase.SetVideoPreviewUseCase import com.wire.kalium.logic.feature.call.usecase.TurnLoudSpeakerOffUseCase import com.wire.kalium.logic.feature.call.usecase.TurnLoudSpeakerOnUseCase import com.wire.kalium.logic.feature.call.usecase.UnMuteCallUseCase import com.wire.kalium.logic.feature.call.usecase.video.UpdateVideoStateUseCase +import com.wire.kalium.logic.feature.client.ObserveCurrentClientIdUseCase import com.wire.kalium.logic.feature.conversation.ObserveConversationDetailsUseCase +import com.wire.kalium.logic.feature.incallreaction.SendInCallReactionUseCase +import com.wire.kalium.logic.functional.onSuccess import com.wire.kalium.logic.util.PlatformView import dagger.assisted.Assisted import dagger.assisted.AssistedFactory @@ -57,13 +69,19 @@ import dagger.hilt.android.lifecycle.HiltViewModel import kotlinx.collections.immutable.persistentListOf import kotlinx.collections.immutable.toPersistentList import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.channels.BufferOverflow +import kotlinx.coroutines.channels.Channel import kotlinx.coroutines.flow.SharedFlow import kotlinx.coroutines.flow.SharingStarted +import kotlinx.coroutines.flow.collect import kotlinx.coroutines.flow.collectLatest +import kotlinx.coroutines.flow.combine import kotlinx.coroutines.flow.filterIsInstance +import kotlinx.coroutines.flow.filterNotNull import kotlinx.coroutines.flow.first import kotlinx.coroutines.flow.flowOn import kotlinx.coroutines.flow.map +import kotlinx.coroutines.flow.receiveAsFlow import kotlinx.coroutines.flow.shareIn import kotlinx.coroutines.launch @@ -83,6 +101,9 @@ class SharedCallingViewModel @AssistedInject constructor( private val flipToFrontCamera: FlipToFrontCameraUseCase, private val flipToBackCamera: FlipToBackCameraUseCase, private val observeSpeaker: ObserveSpeakerUseCase, + private val observeInCallReactionsUseCase: ObserveInCallReactionsUseCase, + private val sendInCallReactionUseCase: SendInCallReactionUseCase, + private val getCurrentClientId: ObserveCurrentClientIdUseCase, private val callRinger: CallRinger, private val uiCallParticipantMapper: UICallParticipantMapper, private val userTypeMapper: UserTypeMapper, @@ -93,6 +114,15 @@ class SharedCallingViewModel @AssistedInject constructor( var participantsState by mutableStateOf(persistentListOf()) + private val _inCallReactions = Channel( + capacity = 300, // Max reactions to keep in queue + onBufferOverflow = BufferOverflow.DROP_OLDEST, + ) + + val inCallReactions = _inCallReactions.receiveAsFlow().withDelayAfterFirst(InCallReactions.reactionsThrottleDelayMs) + + val recentReactions = recentInCallReactionMap() + init { viewModelScope.launch { val allCallsSharedFlow = observeEstablishedCallWithSortedParticipants(conversationId) @@ -110,6 +140,9 @@ class SharedCallingViewModel @AssistedInject constructor( launch { observeOnSpeaker(this) } + launch { + observeInCallReactions() + } } } @@ -172,18 +205,21 @@ class SharedCallingViewModel @AssistedInject constructor( } private suspend fun observeParticipants(sharedFlow: SharedFlow) { - sharedFlow.collectLatest { call -> - call?.let { - callState = callState.copy( - isMuted = it.isMuted, - callStatus = it.status, - isCameraOn = it.isCameraOn, - isCbrEnabled = it.isCbrEnabled && call.conversationType == Conversation.Type.ONE_ON_ONE, - callerName = it.callerName, - ) - participantsState = call.participants.map { uiCallParticipantMapper.toUICallParticipant(it) }.toPersistentList() - } - } + combine( + getCurrentClientId().filterNotNull(), + sharedFlow.filterNotNull(), + ) { clientId, call -> + callState = callState.copy( + isMuted = call.isMuted, + callStatus = call.status, + isCameraOn = call.isCameraOn, + isCbrEnabled = call.isCbrEnabled && call.conversationType == Conversation.Type.ONE_ON_ONE, + callerName = call.callerName, + ) + participantsState = call.participants.map { + uiCallParticipantMapper.toUICallParticipant(it, clientId) + }.toPersistentList() + }.collect() } fun hangUpCall(onCompleted: () -> Unit) { @@ -279,8 +315,42 @@ class SharedCallingViewModel @AssistedInject constructor( } } + private suspend fun observeInCallReactions() { + observeInCallReactionsUseCase(conversationId).collect { message -> + + val sender = participantsState.senderName(message.senderUserId)?.let { name -> + ReactionSender.Other(name) + } ?: ReactionSender.Unknown + + message.emojis.forEach { emoji -> + _inCallReactions.send(InCallReaction(emoji, sender)) + } + + if (message.emojis.isNotEmpty()) { + recentReactions.put(message.senderUserId, message.emojis.last()) + } + } + } + + fun onReactionClick(emoji: String) { + viewModelScope.launch { + sendInCallReactionUseCase(conversationId, emoji).onSuccess { + _inCallReactions.send(InCallReaction(emoji, ReactionSender.You)) + } + } + } + + private fun recentInCallReactionMap(): MutableMap = + ExpiringMap( + scope = viewModelScope, + expiration = InCallReactions.recentReactionShowDurationMs, + delegate = mutableStateMapOf() + ) + @AssistedFactory interface Factory { fun create(conversationId: ConversationId): SharedCallingViewModel } } + +private fun List.senderName(userId: QualifiedID) = firstOrNull { it.id.value == userId.value }?.name diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/CameraButton.kt b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/CameraButton.kt index cc920e44ffe..3acf75cd9a8 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/CameraButton.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/CameraButton.kt @@ -20,10 +20,8 @@ package com.wire.android.ui.calling.controlbuttons import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier -import androidx.compose.ui.unit.Dp import com.wire.android.R import com.wire.android.appLogger -import com.wire.android.ui.common.dimensions import com.wire.android.ui.theme.WireTheme import com.wire.android.util.permission.rememberCameraPermissionFlow import com.wire.android.util.ui.PreviewMultipleThemes @@ -34,7 +32,6 @@ fun CameraButton( onPermissionPermanentlyDenied: () -> Unit, modifier: Modifier = Modifier, isCameraOn: Boolean = false, - size: Dp = dimensions().defaultCallingControlsSize, ) { val cameraPermissionCheck = rememberCameraPermissionFlow( onPermissionGranted = { @@ -56,7 +53,6 @@ fun CameraButton( false -> R.string.content_description_calling_turn_camera_on }, onClick = cameraPermissionCheck::launch, - size = size, modifier = modifier, ) } diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/HangUpButton.kt b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/HangUpButton.kt index c64e3cb7379..41812569a70 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/HangUpButton.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/HangUpButton.kt @@ -38,7 +38,7 @@ import com.wire.android.util.ui.PreviewMultipleThemes fun HangUpButton( onHangUpButtonClicked: () -> Unit, modifier: Modifier = Modifier, - size: Dp = dimensions().bigCallingControlsSize, + size: Dp = dimensions().defaultCallingControlsHeight, iconSize: Dp = dimensions().bigCallingHangUpButtonIconSize, ) { WirePrimaryIconButton( diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/HangUpOngoingButton.kt b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/HangUpOngoingButton.kt new file mode 100644 index 00000000000..1a484e26b04 --- /dev/null +++ b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/HangUpOngoingButton.kt @@ -0,0 +1,67 @@ +/* + * Wire + * Copyright (C) 2024 Wire Swiss GmbH + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see http://www.gnu.org/licenses/. + */ + +package com.wire.android.ui.calling.controlbuttons + +import androidx.compose.foundation.layout.height +import androidx.compose.foundation.layout.width +import androidx.compose.foundation.shape.CircleShape +import androidx.compose.material3.MaterialTheme +import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier +import androidx.compose.ui.unit.Dp +import androidx.compose.ui.unit.DpSize +import com.wire.android.R +import com.wire.android.ui.common.button.WireButtonState +import com.wire.android.ui.common.button.WirePrimaryIconButton +import com.wire.android.ui.common.dimensions +import com.wire.android.ui.theme.WireTheme +import com.wire.android.ui.theme.wireDimensions +import com.wire.android.util.ui.PreviewMultipleThemes + +@Composable +fun HangUpOngoingButton( + onHangUpButtonClicked: () -> Unit, + modifier: Modifier = Modifier, + width: Dp = dimensions().defaultCallingControlsWidth, + height: Dp = dimensions().defaultCallingControlsHeight, + iconSize: Dp = dimensions().bigCallingHangUpButtonIconSize, +) { + WirePrimaryIconButton( + iconResource = R.drawable.ic_call_reject, + contentDescription = R.string.content_description_calling_hang_up_call, + state = WireButtonState.Error, + shape = CircleShape, + minSize = DpSize(width, height), + minClickableSize = DpSize(width, height), + iconSize = iconSize, + onButtonClicked = onHangUpButtonClicked, + modifier = modifier, + ) +} + +@PreviewMultipleThemes +@Composable +fun PreviewComposableHangUpOngoingButton() = WireTheme { + HangUpOngoingButton( + modifier = Modifier + .width(MaterialTheme.wireDimensions.bigCallingControlsSize) + .height(MaterialTheme.wireDimensions.bigCallingControlsSize), + onHangUpButtonClicked = { } + ) +} diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/CameraFlipButton.kt b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/InCallReactionsButton.kt similarity index 54% rename from app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/CameraFlipButton.kt rename to app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/InCallReactionsButton.kt index bc58e225943..0a8d3d3bb3e 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/CameraFlipButton.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/InCallReactionsButton.kt @@ -15,48 +15,49 @@ * You should have received a copy of the GNU General Public License * along with this program. If not, see http://www.gnu.org/licenses/. */ - package com.wire.android.ui.calling.controlbuttons import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier -import androidx.compose.ui.unit.Dp import com.wire.android.R -import com.wire.android.ui.common.dimensions import com.wire.android.ui.theme.WireTheme import com.wire.android.util.ui.PreviewMultipleThemes @Composable -fun CameraFlipButton( - onCameraFlipButtonClicked: () -> Unit, +fun InCallReactionsButton( + isSelected: Boolean, + onInCallReactionsClick: () -> Unit, modifier: Modifier = Modifier, - isOnFrontCamera: Boolean = false, - size: Dp = dimensions().defaultCallingControlsSize ) { WireCallControlButton( - isSelected = !isOnFrontCamera, - iconResId = when (isOnFrontCamera) { - true -> R.drawable.ic_camera_flipped - false -> R.drawable.ic_camera_flip + isSelected = isSelected, + iconResId = when (isSelected) { + true -> R.drawable.ic_incall_reactions + false -> R.drawable.ic_incall_reactions }, - contentDescription = when (isOnFrontCamera) { - true -> R.string.content_description_calling_flip_camera_on - false -> R.string.content_description_calling_flip_camera_off + contentDescription = when (isSelected) { + true -> R.string.content_description_calling_unmute_call + false -> R.string.content_description_calling_mute_call }, - onClick = onCameraFlipButtonClicked, - size = size, - modifier = modifier, + onClick = onInCallReactionsClick, + modifier = modifier ) } @PreviewMultipleThemes @Composable -fun PreviewCameraFlipButtonOn() = WireTheme { - CameraFlipButton(isOnFrontCamera = true, onCameraFlipButtonClicked = { }) +fun PreviewInCallReactionsButton() = WireTheme { + InCallReactionsButton( + isSelected = false, + onInCallReactionsClick = { } + ) } @PreviewMultipleThemes @Composable -fun PreviewCameraFlipButtonOff() = WireTheme { - CameraFlipButton(isOnFrontCamera = false, onCameraFlipButtonClicked = { }) +fun PreviewInCallReactionsButtonSelected() = WireTheme { + InCallReactionsButton( + isSelected = true, + onInCallReactionsClick = { } + ) } diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/MicrophoneButton.kt b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/MicrophoneButton.kt index 272f6da8b92..ea76130a913 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/MicrophoneButton.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/MicrophoneButton.kt @@ -20,9 +20,7 @@ package com.wire.android.ui.calling.controlbuttons import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier -import androidx.compose.ui.unit.Dp import com.wire.android.R -import com.wire.android.ui.common.dimensions import com.wire.android.ui.theme.WireTheme import com.wire.android.util.ui.PreviewMultipleThemes @@ -31,7 +29,6 @@ fun MicrophoneButton( isMuted: Boolean, onMicrophoneButtonClicked: () -> Unit, modifier: Modifier = Modifier, - size: Dp = dimensions().defaultCallingControlsSize ) { WireCallControlButton( isSelected = !isMuted, @@ -44,7 +41,6 @@ fun MicrophoneButton( false -> R.string.content_description_calling_mute_call }, onClick = onMicrophoneButtonClicked, - size = size, modifier = modifier ) } diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/SpeakerButton.kt b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/SpeakerButton.kt index 46bf66d8f50..d399f3e43c8 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/SpeakerButton.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/SpeakerButton.kt @@ -20,9 +20,7 @@ package com.wire.android.ui.calling.controlbuttons import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier -import androidx.compose.ui.unit.Dp import com.wire.android.R -import com.wire.android.ui.common.dimensions import com.wire.android.ui.theme.WireTheme import com.wire.android.util.ui.PreviewMultipleThemes @@ -31,7 +29,6 @@ fun SpeakerButton( isSpeakerOn: Boolean, onSpeakerButtonClicked: () -> Unit, modifier: Modifier = Modifier, - size: Dp = dimensions().defaultCallingControlsSize ) { WireCallControlButton( isSelected = isSpeakerOn, @@ -44,7 +41,6 @@ fun SpeakerButton( false -> R.string.content_description_calling_turn_speaker_on }, onClick = onSpeakerButtonClicked, - size = size, modifier = modifier, ) } diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/WireCallControlButton.kt b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/WireCallControlButton.kt index 3f438a43087..4208a19e0a5 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/WireCallControlButton.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/controlbuttons/WireCallControlButton.kt @@ -20,14 +20,13 @@ package com.wire.android.ui.calling.controlbuttons import androidx.annotation.DrawableRes import androidx.annotation.StringRes -import androidx.compose.foundation.layout.size import androidx.compose.foundation.shape.CircleShape import androidx.compose.runtime.Composable import androidx.compose.ui.Modifier import androidx.compose.ui.unit.Dp import androidx.compose.ui.unit.DpSize import com.wire.android.ui.common.button.WireButtonState -import com.wire.android.ui.common.button.WireSecondaryIconButton +import com.wire.android.ui.common.button.WirePrimaryIconButton import com.wire.android.ui.common.button.wireSecondaryButtonColors import com.wire.android.ui.common.colorsScheme import com.wire.android.ui.common.dimensions @@ -39,10 +38,11 @@ fun WireCallControlButton( @StringRes contentDescription: Int, onClick: () -> Unit, modifier: Modifier = Modifier, - size: Dp = dimensions().defaultCallingControlsSize, + width: Dp = dimensions().defaultCallingControlsWidth, + height: Dp = dimensions().defaultCallingControlsHeight, iconSize: Dp = dimensions().defaultCallingControlsIconSize ) { - WireSecondaryIconButton( + WirePrimaryIconButton( onButtonClicked = onClick, iconResource = iconResId, shape = CircleShape, @@ -60,9 +60,9 @@ fun WireCallControlButton( }, contentDescription = contentDescription, state = if (isSelected) WireButtonState.Selected else WireButtonState.Default, - minSize = DpSize(size, size), - minClickableSize = DpSize(size, size), + minSize = DpSize(width, height), + minClickableSize = DpSize(width, height), iconSize = iconSize, - modifier = modifier.size(size), + modifier = modifier, ) } diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/model/InCallReaction.kt b/app/src/main/kotlin/com/wire/android/ui/calling/model/InCallReaction.kt new file mode 100644 index 00000000000..0470beae5dc --- /dev/null +++ b/app/src/main/kotlin/com/wire/android/ui/calling/model/InCallReaction.kt @@ -0,0 +1,29 @@ +/* + * Wire + * Copyright (C) 2024 Wire Swiss GmbH + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see http://www.gnu.org/licenses/. + */ +package com.wire.android.ui.calling.model + +data class InCallReaction( + val emoji: String, + val sender: ReactionSender, +) + +sealed interface ReactionSender { + data object You : ReactionSender + data class Other(val name: String) : ReactionSender + data object Unknown : ReactionSender +} diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/model/UICallParticipant.kt b/app/src/main/kotlin/com/wire/android/ui/calling/model/UICallParticipant.kt index 105046d26b7..b7b3fcf2407 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/model/UICallParticipant.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/model/UICallParticipant.kt @@ -25,6 +25,7 @@ import com.wire.kalium.logic.data.id.QualifiedID data class UICallParticipant( val id: QualifiedID, val clientId: String, + val isSelfUser: Boolean, val name: String? = null, val isMuted: Boolean, val isSpeaking: Boolean = false, diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/OngoingCallScreen.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/OngoingCallScreen.kt index ec577f2c221..9a5b93df9f6 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/OngoingCallScreen.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/OngoingCallScreen.kt @@ -21,6 +21,10 @@ package com.wire.android.ui.calling.ongoing import android.content.pm.PackageManager import android.view.View import androidx.activity.compose.BackHandler +import androidx.compose.animation.AnimatedContent +import androidx.compose.animation.slideInVertically +import androidx.compose.animation.slideOutVertically +import androidx.compose.animation.togetherWith import androidx.compose.foundation.layout.Arrangement import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.BoxWithConstraints @@ -32,7 +36,6 @@ import androidx.compose.foundation.layout.fillMaxWidth import androidx.compose.foundation.layout.height import androidx.compose.foundation.layout.offset import androidx.compose.foundation.layout.padding -import androidx.compose.foundation.layout.size import androidx.compose.material3.ExperimentalMaterial3Api import androidx.compose.material3.MaterialTheme import androidx.compose.material3.SheetValue @@ -65,14 +68,20 @@ import com.wire.android.ui.calling.CallState import com.wire.android.ui.calling.ConversationName import com.wire.android.ui.calling.SharedCallingViewModel import com.wire.android.ui.calling.controlbuttons.CameraButton -import com.wire.android.ui.calling.controlbuttons.CameraFlipButton -import com.wire.android.ui.calling.controlbuttons.HangUpButton +import com.wire.android.ui.calling.controlbuttons.HangUpOngoingButton +import com.wire.android.ui.calling.controlbuttons.InCallReactionsButton import com.wire.android.ui.calling.controlbuttons.MicrophoneButton import com.wire.android.ui.calling.controlbuttons.SpeakerButton +import com.wire.android.ui.calling.model.InCallReaction import com.wire.android.ui.calling.model.UICallParticipant import com.wire.android.ui.calling.ongoing.fullscreen.DoubleTapToast import com.wire.android.ui.calling.ongoing.fullscreen.FullScreenTile import com.wire.android.ui.calling.ongoing.fullscreen.SelectedParticipant +import com.wire.android.ui.calling.ongoing.incallreactions.AnimatableReaction +import com.wire.android.ui.calling.ongoing.incallreactions.InCallReactionsPanel +import com.wire.android.ui.calling.ongoing.incallreactions.InCallReactionsState +import com.wire.android.ui.calling.ongoing.incallreactions.drawInCallReactions +import com.wire.android.ui.calling.ongoing.incallreactions.rememberInCallReactionsState import com.wire.android.ui.calling.ongoing.participantsview.FloatingSelfUserTile import com.wire.android.ui.calling.ongoing.participantsview.VerticalCallingPager import com.wire.android.ui.common.ConversationVerificationIcons @@ -85,11 +94,11 @@ import com.wire.android.ui.common.progress.WireCircularProgressIndicator import com.wire.android.ui.common.topappbar.NavigationIconType import com.wire.android.ui.common.topappbar.WireCenterAlignedTopAppBar import com.wire.android.ui.common.visbility.rememberVisibilityState +import com.wire.android.ui.emoji.EmojiPickerBottomSheet import com.wire.android.ui.home.conversations.PermissionPermanentlyDeniedDialogState import com.wire.android.ui.home.conversationslist.model.Membership import com.wire.android.ui.theme.WireTheme import com.wire.android.ui.theme.wireColorScheme -import com.wire.android.ui.theme.wireDimensions import com.wire.android.ui.theme.wireTypography import com.wire.android.util.ui.PreviewMultipleThemes import com.wire.kalium.logic.data.call.CallStatus @@ -100,6 +109,7 @@ import com.wire.kalium.logic.data.user.UserId import kotlinx.collections.immutable.PersistentList import kotlinx.collections.immutable.persistentListOf import kotlinx.collections.immutable.toPersistentList +import kotlinx.coroutines.flow.collectLatest import java.util.Locale @Suppress("ParameterWrapping") @@ -118,6 +128,8 @@ fun OngoingCallScreen( val permissionPermanentlyDeniedDialogState = rememberVisibilityState() + val inCallReactionsState = rememberInCallReactionsState() + val activity = LocalActivity.current val isPiPAvailableOnThisDevice = activity.packageManager.hasSystemFeature(PackageManager.FEATURE_PICTURE_IN_PICTURE) @@ -133,6 +145,13 @@ fun OngoingCallScreen( } } } + + LaunchedEffect(Unit) { + sharedCallingViewModel.inCallReactions.collectLatest { reaction -> + inCallReactionsState.runAnimation(reaction) + } + } + val hangUpCall = remember { { sharedCallingViewModel.hangUpCall { activity.finishAndRemoveTask() } @@ -167,6 +186,7 @@ fun OngoingCallScreen( val inPictureInPictureMode = activity.isInPictureInPictureMode OngoingCallContent( callState = sharedCallingViewModel.callState, + inCallReactionsState = inCallReactionsState, shouldShowDoubleTapToast = ongoingCallViewModel.shouldShowDoubleTapToast, toggleSpeaker = sharedCallingViewModel::toggleSpeaker, toggleMute = sharedCallingViewModel::toggleMute, @@ -181,9 +201,10 @@ fun OngoingCallScreen( selectedParticipantForFullScreen = ongoingCallViewModel.selectedParticipant, hideDoubleTapToast = ongoingCallViewModel::hideDoubleTapToast, onCameraPermissionPermanentlyDenied = onCameraPermissionPermanentlyDenied, + onReactionClick = sharedCallingViewModel::onReactionClick, participants = sharedCallingViewModel.participantsState, inPictureInPictureMode = inPictureInPictureMode, - currentUserId = ongoingCallViewModel.currentUserId, + recentReactions = sharedCallingViewModel.recentReactions, ) BackHandler { @@ -279,6 +300,7 @@ private fun HandleSendingVideoFeed( @Composable private fun OngoingCallContent( callState: CallState, + inCallReactionsState: InCallReactionsState, shouldShowDoubleTapToast: Boolean, toggleSpeaker: () -> Unit, toggleMute: () -> Unit, @@ -290,12 +312,13 @@ private fun OngoingCallContent( onCollapse: () -> Unit, hideDoubleTapToast: () -> Unit, onCameraPermissionPermanentlyDenied: () -> Unit, + onReactionClick: (String) -> Unit, requestVideoStreams: (participants: List) -> Unit, onSelectedParticipant: (selectedParticipant: SelectedParticipant) -> Unit, selectedParticipantForFullScreen: SelectedParticipant, participants: PersistentList, + recentReactions: Map, inPictureInPictureMode: Boolean, - currentUserId: UserId, ) { val sheetInitialValue = SheetValue.PartiallyExpanded val sheetState = rememberStandardBottomSheetState( @@ -308,6 +331,9 @@ private fun OngoingCallContent( var shouldOpenFullScreen by remember { mutableStateOf(false) } + var showInCallReactionsPanel by remember { mutableStateOf(false) } + var showEmojiPicker by remember { mutableStateOf(false) } + WireBottomSheetScaffold( sheetDragHandle = null, topBar = if (inPictureInPictureMode) { @@ -336,19 +362,22 @@ private fun OngoingCallContent( conversationId = callState.conversationId, isMuted = callState.isMuted ?: true, isCameraOn = callState.isCameraOn, - isOnFrontCamera = callState.isOnFrontCamera, isSpeakerOn = callState.isSpeakerOn, + isShowingCallReactions = showInCallReactionsPanel, toggleSpeaker = toggleSpeaker, toggleMute = toggleMute, onHangUpCall = hangUpCall, onToggleVideo = toggleVideo, - flipCamera = flipCamera, + onCallReactionsClick = { + showInCallReactionsPanel = !showInCallReactionsPanel + }, onCameraPermissionPermanentlyDenied = onCameraPermissionPermanentlyDenied ) } }, + sheetContainerColor = colorsScheme().background, ) { - BoxWithConstraints( + Column( modifier = Modifier .padding( top = it.calculateTopPadding(), @@ -356,96 +385,135 @@ private fun OngoingCallContent( ) ) { - if (participants.isEmpty()) { - Column( - modifier = Modifier.fillMaxSize(), - verticalArrangement = Arrangement.Center, - horizontalAlignment = Alignment.CenterHorizontally - ) { - WireCircularProgressIndicator( - progressColor = MaterialTheme.wireColorScheme.onSurface, - modifier = Modifier.align(Alignment.CenterHorizontally), - size = dimensions().spacing32x - ) - Text( - text = stringResource(id = R.string.calling_screen_connecting_until_call_established), - modifier = Modifier.align(Alignment.CenterHorizontally), - ) - } - } else { - Box( - modifier = Modifier.fillMaxSize() - ) { - - // if there is only one in the call, do not allow full screen - if (participants.size == 1) { - shouldOpenFullScreen = false - } - - // if we are on full screen, and that user left the call, then we leave the full screen - if (participants.find { user -> user.id == selectedParticipantForFullScreen.userId } == null) { - shouldOpenFullScreen = false - } - - if (shouldOpenFullScreen) { - hideDoubleTapToast() - FullScreenTile( - callState = callState, - selectedParticipant = selectedParticipantForFullScreen, - height = this@BoxWithConstraints.maxHeight - dimensions().spacing4x, - closeFullScreen = { - onSelectedParticipant(SelectedParticipant()) - shouldOpenFullScreen = !shouldOpenFullScreen - }, - onBackButtonClicked = { - onSelectedParticipant(SelectedParticipant()) - shouldOpenFullScreen = !shouldOpenFullScreen - }, - requestVideoStreams = requestVideoStreams, - setVideoPreview = setVideoPreview, - clearVideoPreview = clearVideoPreview, - participants = participants - ) - } else { - VerticalCallingPager( - participants = participants, - isSelfUserCameraOn = callState.isCameraOn, - isSelfUserMuted = callState.isMuted ?: true, - isInPictureInPictureMode = inPictureInPictureMode, - contentHeight = this@BoxWithConstraints.maxHeight, - onSelfVideoPreviewCreated = setVideoPreview, - onSelfClearVideoPreview = clearVideoPreview, - requestVideoStreams = requestVideoStreams, - currentUserId = currentUserId, - onDoubleTap = { selectedParticipant -> - onSelectedParticipant(selectedParticipant) - shouldOpenFullScreen = !shouldOpenFullScreen - }, + BoxWithConstraints( + modifier = Modifier + .fillMaxSize() + .weight(1f) + ) { + + if (participants.isEmpty()) { + Column( + modifier = Modifier.fillMaxSize(), + verticalArrangement = Arrangement.Center, + horizontalAlignment = Alignment.CenterHorizontally + ) { + WireCircularProgressIndicator( + progressColor = MaterialTheme.wireColorScheme.onSurface, + modifier = Modifier.align(Alignment.CenterHorizontally), + size = dimensions().spacing32x ) - DoubleTapToast( - modifier = Modifier.align(Alignment.TopCenter), - enabled = shouldShowDoubleTapToast, - text = stringResource(id = R.string.calling_ongoing_double_tap_for_full_screen), - onTap = hideDoubleTapToast + Text( + text = stringResource(id = R.string.calling_screen_connecting_until_call_established), + modifier = Modifier.align(Alignment.CenterHorizontally), ) } - if (BuildConfig.PICTURE_IN_PICTURE_ENABLED && participants.size > 1) { - val selfUser = - participants.first { participant -> - // API returns only id.value, without domain, till this get changed compare only id.value - participant.id.equalsIgnoringBlankDomain(currentUserId) - } - FloatingSelfUserTile( - modifier = Modifier.align(Alignment.TopEnd), - contentHeight = this@BoxWithConstraints.maxHeight, - contentWidth = this@BoxWithConstraints.maxWidth, - participant = selfUser, - onSelfUserVideoPreviewCreated = setVideoPreview, - onClearSelfUserVideoPreview = clearVideoPreview - ) + } else { + Box( + modifier = Modifier + .fillMaxSize() + .drawInCallReactions(state = inCallReactionsState) + ) { + + // if there is only one in the call, do not allow full screen + if (participants.size == 1) { + shouldOpenFullScreen = false + } + + // if we are on full screen, and that user left the call, then we leave the full screen + if (participants.find { user -> user.id == selectedParticipantForFullScreen.userId } == null) { + shouldOpenFullScreen = false + } + + if (shouldOpenFullScreen) { + hideDoubleTapToast() + FullScreenTile( + callState = callState, + selectedParticipant = selectedParticipantForFullScreen, + height = this@BoxWithConstraints.maxHeight - dimensions().spacing4x, + closeFullScreen = { + onSelectedParticipant(SelectedParticipant()) + shouldOpenFullScreen = !shouldOpenFullScreen + }, + onBackButtonClicked = { + onSelectedParticipant(SelectedParticipant()) + shouldOpenFullScreen = !shouldOpenFullScreen + }, + requestVideoStreams = requestVideoStreams, + setVideoPreview = setVideoPreview, + clearVideoPreview = clearVideoPreview, + participants = participants, + isOnFrontCamera = callState.isOnFrontCamera, + flipCamera = flipCamera, + ) + } else { + VerticalCallingPager( + participants = participants, + isSelfUserCameraOn = callState.isCameraOn, + isSelfUserMuted = callState.isMuted ?: true, + isInPictureInPictureMode = inPictureInPictureMode, + isOnFrontCamera = callState.isOnFrontCamera, + contentHeight = this@BoxWithConstraints.maxHeight, + onSelfVideoPreviewCreated = setVideoPreview, + onSelfClearVideoPreview = clearVideoPreview, + requestVideoStreams = requestVideoStreams, + recentReactions = recentReactions, + onDoubleTap = { selectedParticipant -> + onSelectedParticipant(selectedParticipant) + shouldOpenFullScreen = !shouldOpenFullScreen + }, + flipCamera = flipCamera, + ) + DoubleTapToast( + modifier = Modifier.align(Alignment.TopCenter), + enabled = shouldShowDoubleTapToast, + text = stringResource(id = R.string.calling_ongoing_double_tap_for_full_screen), + onTap = hideDoubleTapToast + ) + } + if (BuildConfig.PICTURE_IN_PICTURE_ENABLED && participants.size > 1) { + val selfUser = participants.first { it.isSelfUser } + FloatingSelfUserTile( + modifier = Modifier.align(Alignment.TopEnd), + contentHeight = this@BoxWithConstraints.maxHeight, + contentWidth = this@BoxWithConstraints.maxWidth, + participant = selfUser, + isOnFrontCamera = callState.isOnFrontCamera, + onSelfUserVideoPreviewCreated = setVideoPreview, + onClearSelfUserVideoPreview = clearVideoPreview, + flipCamera = flipCamera, + ) + } } } } + + AnimatedContent( + targetState = showInCallReactionsPanel, + transitionSpec = { + val enter = slideInVertically(initialOffsetY = { it }) + val exit = slideOutVertically(targetOffsetY = { it }) + enter.togetherWith(exit) + }, + label = "InCallReactions" + ) { show -> + if (show) { + InCallReactionsPanel( + onReactionClick = onReactionClick, + onMoreClick = { showEmojiPicker = true } + ) + } + } + + EmojiPickerBottomSheet( + isVisible = showEmojiPicker, + onEmojiSelected = { + showEmojiPicker = false + onReactionClick(it) + }, + onDismiss = { + showEmojiPicker = false + }, + ) } } } @@ -508,12 +576,12 @@ private fun CallingControls( isMuted: Boolean, isCameraOn: Boolean, isSpeakerOn: Boolean, - isOnFrontCamera: Boolean, + isShowingCallReactions: Boolean, toggleSpeaker: () -> Unit, toggleMute: () -> Unit, onHangUpCall: () -> Unit, onToggleVideo: () -> Unit, - flipCamera: () -> Unit, + onCallReactionsClick: () -> Unit, onCameraPermissionPermanentlyDenied: () -> Unit ) { Column( @@ -527,7 +595,10 @@ private fun CallingControls( .fillMaxWidth() .height(dimensions().spacing56x) ) { - MicrophoneButton(isMuted = isMuted, onMicrophoneButtonClicked = toggleMute) + MicrophoneButton( + isMuted = isMuted, + onMicrophoneButtonClicked = toggleMute + ) CameraButton( isCameraOn = isCameraOn, onPermissionPermanentlyDenied = onCameraPermissionPermanentlyDenied, @@ -539,15 +610,12 @@ private fun CallingControls( onSpeakerButtonClicked = toggleSpeaker ) - if (isCameraOn) { - CameraFlipButton( - isOnFrontCamera = isOnFrontCamera, - onCameraFlipButtonClicked = flipCamera - ) - } + InCallReactionsButton( + isSelected = isShowingCallReactions, + onInCallReactionsClick = onCallReactionsClick + ) - HangUpButton( - modifier = Modifier.size(MaterialTheme.wireDimensions.defaultCallingControlsSize), + HangUpOngoingButton( onHangUpButtonClicked = onHangUpCall ) } @@ -556,6 +624,7 @@ private fun CallingControls( } } +@Suppress("EmptyFunctionBlock") @Composable fun PreviewOngoingCallContent(participants: PersistentList) { OngoingCallContent( @@ -571,6 +640,10 @@ fun PreviewOngoingCallContent(participants: PersistentList) { mlsVerificationStatus = Conversation.VerificationStatus.NOT_VERIFIED, proteusVerificationStatus = Conversation.VerificationStatus.NOT_VERIFIED, ), + inCallReactionsState = object : InCallReactionsState { + override fun runAnimation(inCallReaction: InCallReaction) {} + override fun getReactions(): List = emptyList() + }, shouldShowDoubleTapToast = false, toggleSpeaker = {}, toggleMute = {}, @@ -582,12 +655,13 @@ fun PreviewOngoingCallContent(participants: PersistentList) { onCollapse = {}, hideDoubleTapToast = {}, onCameraPermissionPermanentlyDenied = {}, + onReactionClick = {}, requestVideoStreams = {}, participants = participants, inPictureInPictureMode = false, - currentUserId = UserId("userId", "domain"), onSelectedParticipant = {}, selectedParticipantForFullScreen = SelectedParticipant(), + recentReactions = emptyMap(), ) } @@ -621,6 +695,7 @@ fun buildPreviewParticipantsList(count: Int = 10) = buildList { UICallParticipant( id = QualifiedID("id_$index", ""), clientId = "client_id_$index", + isSelfUser = false, name = "Participant $index", isSpeaking = index % 3 == 1, isMuted = index % 3 == 2, diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/OngoingCallViewModel.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/OngoingCallViewModel.kt index 84c6e1a57d1..a70325a0cd7 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/OngoingCallViewModel.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/OngoingCallViewModel.kt @@ -47,7 +47,7 @@ import kotlinx.coroutines.flow.distinctUntilChanged import kotlinx.coroutines.flow.first import kotlinx.coroutines.launch -@Suppress("LongParameterList") +@Suppress("LongParameterList", "TooManyFunctions") @HiltViewModel(assistedFactory = OngoingCallViewModel.Factory::class) class OngoingCallViewModel @AssistedInject constructor( @Assisted diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/fullscreen/FullScreenTile.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/fullscreen/FullScreenTile.kt index cee64f83032..add7376f6dc 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/fullscreen/FullScreenTile.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/fullscreen/FullScreenTile.kt @@ -62,6 +62,8 @@ fun FullScreenTile( setVideoPreview: (View) -> Unit, requestVideoStreams: (participants: List) -> Unit, clearVideoPreview: () -> Unit, + isOnFrontCamera: Boolean, + flipCamera: () -> Unit, modifier: Modifier = Modifier, contentPadding: Dp = dimensions().spacing4x, ) { @@ -87,7 +89,6 @@ fun FullScreenTile( .height(height) .padding(contentPadding), participantTitleState = it, - isSelfUser = selectedParticipant.isSelfUser, isSelfUserCameraOn = if (selectedParticipant.isSelfUser) { callState.isCameraOn } else { @@ -101,7 +102,9 @@ fun FullScreenTile( shouldFillOthersVideoPreview = false, isZoomingEnabled = true, onSelfUserVideoPreviewCreated = setVideoPreview, - onClearSelfUserVideoPreview = clearVideoPreview + onClearSelfUserVideoPreview = clearVideoPreview, + isOnFrontCamera = isOnFrontCamera, + flipCamera = flipCamera, ) LaunchedEffect(Unit) { delay(200) @@ -147,5 +150,7 @@ fun PreviewFullScreenTile() = WireTheme { requestVideoStreams = {}, clearVideoPreview = {}, participants = participants, + isOnFrontCamera = false, + flipCamera = {}, ) } diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactions.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactions.kt new file mode 100644 index 00000000000..336a5d39397 --- /dev/null +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactions.kt @@ -0,0 +1,52 @@ +/* + * Wire + * Copyright (C) 2024 Wire Swiss GmbH + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see http://www.gnu.org/licenses/. + */ +package com.wire.android.ui.calling.ongoing.incallreactions + +@Suppress("MagicNumber") +object InCallReactions { + + /** + * Default in call reaction emojis + */ + val defaultReactions = listOf("👍", "🎉", "❤️", "😂", "😮", "👏", "🤔", "😢", "👎") + + /** + * Next reaction click is disabled until delay expires + */ + const val reactionDelayMs = 3000L + + /** + * Total duration for reaction animation + */ + const val animationDurationMs = 3000 + + /** + * Duration for reaction fade out animation + */ + const val fadeOutAnimationDuarationMs = 500 + + /** + * Delay between displaying reactions on screen + */ + const val reactionsThrottleDelayMs: Long = 200 + + /** + * Duration for showing recent reaction next to user image + */ + const val recentReactionShowDurationMs: Long = 6000 +} diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsModifier.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsModifier.kt new file mode 100644 index 00000000000..eaafe6c35fd --- /dev/null +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsModifier.kt @@ -0,0 +1,137 @@ +/* + * Wire + * Copyright (C) 2024 Wire Swiss GmbH + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see http://www.gnu.org/licenses/. + */ +package com.wire.android.ui.calling.ongoing.incallreactions + +import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.drawWithContent +import androidx.compose.ui.geometry.CornerRadius +import androidx.compose.ui.geometry.Offset +import androidx.compose.ui.geometry.Size +import androidx.compose.ui.geometry.center +import androidx.compose.ui.graphics.Color +import androidx.compose.ui.graphics.drawscope.clipRect +import androidx.compose.ui.graphics.drawscope.translate +import androidx.compose.ui.platform.LocalDensity +import androidx.compose.ui.res.stringResource +import androidx.compose.ui.text.TextStyle +import androidx.compose.ui.text.drawText +import androidx.compose.ui.text.rememberTextMeasurer +import androidx.compose.ui.unit.Dp +import androidx.compose.ui.unit.center +import androidx.compose.ui.unit.toSize +import com.wire.android.R +import com.wire.android.ui.calling.model.ReactionSender +import com.wire.android.ui.common.colorsScheme +import com.wire.android.ui.common.dimensions +import com.wire.android.ui.common.typography +import kotlin.math.max + +@Composable +fun Modifier.drawInCallReactions( + state: InCallReactionsState, + labelTextColor: Color = Color.White, + labelColor: Color = Color.Black, + emojiBackgroundColor: Color = colorsScheme().emojiBackgroundColor, + emojiBackgroundSize: Dp = dimensions().inCallReactionButtonSize, + emojiTextStyle: TextStyle = typography().inCallReactionEmoji, + labelTextStyle: TextStyle = typography().label01, +): Modifier { + + val textMeasurer = rememberTextMeasurer() + + val emojiBackgroundSizePx = with(LocalDensity.current) { emojiBackgroundSize.toPx() } + val labelTopMarginPx = with(LocalDensity.current) { dimensions().spacing12x.toPx() } + val labelTextPaddingPx = with(LocalDensity.current) { dimensions().spacing4x.toPx() } + val reactionSenderSelf = stringResource(R.string.reaction_sender_self) + + return this then Modifier.drawWithContent { + + drawContent() + + clipRect(left = 0f, top = 0f, right = size.width, bottom = size.height) { + + state.getReactions().forEach { reaction -> + + val senderText = when (reaction.inCallReaction.sender) { + is ReactionSender.You -> reactionSenderSelf + is ReactionSender.Other -> reaction.inCallReaction.sender.name + is ReactionSender.Unknown -> "" + } + + val emojiLayoutResult = textMeasurer.measure(reaction.inCallReaction.emoji, emojiTextStyle) + val labelLayoutResult = textMeasurer.measure(senderText, labelTextStyle) + + val emojiSize = emojiLayoutResult.size.toSize() + val labelSize = Size( + width = labelLayoutResult.size.width + labelTextPaddingPx * 2, + height = labelLayoutResult.size.height + labelTextPaddingPx * 2 + ) + + val offsetVertical = size.height - size.height * reaction.verticalOffset.value + val offsetHorizontal = (size.width - max(emojiSize.width, labelSize.width)) * reaction.horizontalOffset + + translate( + top = offsetVertical + emojiSize.height, + left = offsetHorizontal + (labelLayoutResult.size.width - emojiSize.width).coerceAtLeast(0f) / 2f + ) { + + // Draw emoji background + drawRoundRect( + color = emojiBackgroundColor.copy(alpha = reaction.alpha.value), + topLeft = Offset( + x = emojiSize.center.x - emojiBackgroundSizePx / 2, + y = emojiSize.center.y - emojiBackgroundSizePx / 2, + ), + size = Size(emojiBackgroundSizePx, emojiBackgroundSizePx), + cornerRadius = CornerRadius(20f), + ) + + // Draw emoji + drawText( + textLayoutResult = emojiLayoutResult, + color = Color.Black.copy(alpha = reaction.alpha.value), + ) + + if (reaction.inCallReaction.sender != ReactionSender.Unknown) { + // Draw label background + drawRoundRect( + color = labelColor.copy(alpha = reaction.alpha.value), + topLeft = Offset( + x = emojiSize.center.x - labelSize.center.x, + y = emojiSize.height + labelTopMarginPx, + ), + size = labelSize, + cornerRadius = CornerRadius(10f), + ) + + // Draw label text + drawText( + textLayoutResult = labelLayoutResult, + color = labelTextColor.copy(alpha = reaction.alpha.value), + topLeft = Offset( + x = emojiSize.center.x - labelLayoutResult.size.center.x, + y = emojiSize.height + labelTopMarginPx + labelTextPaddingPx + ) + ) + } + } + } + } + } +} diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsPanel.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsPanel.kt new file mode 100644 index 00000000000..5472e84cc7d --- /dev/null +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsPanel.kt @@ -0,0 +1,170 @@ +/* + * Wire + * Copyright (C) 2024 Wire Swiss GmbH + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see http://www.gnu.org/licenses/. + */ +package com.wire.android.ui.calling.ongoing.incallreactions + +import androidx.compose.animation.AnimatedVisibility +import androidx.compose.animation.fadeIn +import androidx.compose.animation.fadeOut +import androidx.compose.foundation.background +import androidx.compose.foundation.clickable +import androidx.compose.foundation.horizontalScroll +import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Box +import androidx.compose.foundation.layout.Row +import androidx.compose.foundation.layout.Spacer +import androidx.compose.foundation.layout.fillMaxSize +import androidx.compose.foundation.layout.fillMaxWidth +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.size +import androidx.compose.foundation.layout.width +import androidx.compose.foundation.rememberScrollState +import androidx.compose.foundation.shape.RoundedCornerShape +import androidx.compose.material3.Icon +import androidx.compose.material3.Text +import androidx.compose.runtime.Composable +import androidx.compose.runtime.mutableStateListOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.rememberCoroutineScope +import androidx.compose.ui.Alignment +import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.alpha +import androidx.compose.ui.draw.clip +import androidx.compose.ui.res.painterResource +import androidx.compose.ui.res.stringResource +import com.wire.android.R +import com.wire.android.ui.calling.ongoing.incallreactions.InCallReactions.reactionDelayMs +import com.wire.android.ui.calling.ongoing.incallreactions.InCallReactions.defaultReactions +import com.wire.android.ui.common.colorsScheme +import com.wire.android.ui.common.dimensions +import com.wire.android.ui.common.typography +import com.wire.android.ui.theme.WireTheme +import com.wire.android.util.ui.PreviewMultipleThemes +import kotlinx.coroutines.delay +import kotlinx.coroutines.launch + +@Composable +fun InCallReactionsPanel( + onReactionClick: (String) -> Unit, + onMoreClick: () -> Unit, + modifier: Modifier = Modifier, +) { + + val scope = rememberCoroutineScope() + + val disabledReactions = remember { mutableStateListOf() } + + Row( + modifier = modifier + .fillMaxWidth() + .padding(vertical = dimensions().spacing8x) + .horizontalScroll(rememberScrollState()), + verticalAlignment = Alignment.CenterVertically, + horizontalArrangement = Arrangement.spacedBy(dimensions().spacing16x) + ) { + + Spacer(modifier = Modifier.width(dimensions().spacing8x)) + + defaultReactions.forEach { reaction -> + EmojiButton( + isEnabled = disabledReactions.contains(reaction).not(), + emoji = reaction, + onClick = { + onReactionClick(reaction) + disabledReactions.add(reaction) + scope.launch { + delay(reactionDelayMs) + disabledReactions.remove(reaction) + } + } + ) + } + + Icon( + modifier = Modifier.clickable { onMoreClick() }, + painter = painterResource(id = R.drawable.ic_more_emojis), + contentDescription = stringResource(R.string.content_description_more_emojis) + ) + + Spacer(modifier = Modifier.width(dimensions().spacing8x)) + } +} + +@Composable +private fun EmojiButton( + emoji: String, + isEnabled: Boolean, + onClick: () -> Unit, +) { + Box( + modifier = Modifier + .size(dimensions().inCallReactionButtonSize) + .clip(RoundedCornerShape(dimensions().corner10x)) + .clickable(isEnabled) { onClick() }, + contentAlignment = Alignment.Center, + ) { + AnimatedVisibility( + visible = isEnabled, + enter = fadeIn(), + exit = fadeOut() + ) { + Box( + modifier = Modifier + .fillMaxSize() + .background( + color = colorsScheme().secondaryButtonEnabled, + shape = RoundedCornerShape(dimensions().corner10x) + ), + ) + } + Text( + modifier = Modifier + .alpha(if (isEnabled) 1f else 0.5f), + text = emoji, + fontSize = typography().inCallReactionEmoji.fontSize, + ) + } +} + +@PreviewMultipleThemes +@Composable +private fun PreviewEmojiButton() = WireTheme { + EmojiButton( + emoji = defaultReactions[0], + isEnabled = true, + onClick = {} + ) +} + +@PreviewMultipleThemes +@Composable +private fun PreviewEmojiButtonDisabled() = WireTheme { + EmojiButton( + emoji = defaultReactions[0], + isEnabled = false, + onClick = {} + ) +} + +@PreviewMultipleThemes +@Composable +private fun PreviewInCallReactionsPanel() = WireTheme { + InCallReactionsPanel( + onReactionClick = {}, + onMoreClick = {}, + ) +} diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsState.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsState.kt new file mode 100644 index 00000000000..c2e77c5624e --- /dev/null +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/incallreactions/InCallReactionsState.kt @@ -0,0 +1,114 @@ +/* + * Wire + * Copyright (C) 2024 Wire Swiss GmbH + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see http://www.gnu.org/licenses/. + */ +package com.wire.android.ui.calling.ongoing.incallreactions + +import androidx.compose.animation.core.Animatable +import androidx.compose.animation.core.AnimationVector1D +import androidx.compose.animation.core.LinearEasing +import androidx.compose.animation.core.tween +import androidx.compose.runtime.Composable +import androidx.compose.runtime.mutableStateListOf +import androidx.compose.runtime.remember +import androidx.compose.runtime.rememberCoroutineScope +import com.wire.android.ui.calling.model.InCallReaction +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.joinAll +import kotlinx.coroutines.launch +import okhttp3.internal.toImmutableList +import kotlin.random.Random + +/** + * Keeps state for currently animated reactions + */ +interface InCallReactionsState { + fun runAnimation(inCallReaction: InCallReaction) + fun getReactions(): List +} + +@Composable +internal fun rememberInCallReactionsState(): InCallReactionsState { + + val scope = rememberCoroutineScope() + + return remember { + InCallReactionsStateImpl( + scope = scope, + mutableReactions = mutableStateListOf(), + ) + } +} + +private class InCallReactionsStateImpl( + private val scope: CoroutineScope, + private val mutableReactions: MutableList, +) : InCallReactionsState { + + /** + * Used by modifier to draw each animated emoji with current animation state + */ + override fun getReactions(): List = mutableReactions.toImmutableList() + + /** + * Adds new emoji to the list of animated emojis. + * Runs animations + * Removes emoji from the list once animations are complete + */ + override fun runAnimation(inCallReaction: InCallReaction) { + scope.launch(Dispatchers.Main) { + val animatable = AnimatableReaction( + inCallReaction = inCallReaction, + horizontalOffset = Random.nextFloat(), + ) + + mutableReactions.add(animatable) + runAnimations(animatable) + mutableReactions.remove(animatable) + } + } + + /** + * Start transition and fade-out animations, wait for complete and return + */ + private suspend fun CoroutineScope.runAnimations(reaction: AnimatableReaction) { + listOf( + launch { + reaction.verticalOffset.animateTo( + targetValue = 1f, + animationSpec = tween(durationMillis = InCallReactions.animationDurationMs, easing = LinearEasing) + ) + }, + launch { + reaction.alpha.animateTo( + targetValue = 0.0f, + animationSpec = tween( + durationMillis = InCallReactions.fadeOutAnimationDuarationMs, + delayMillis = InCallReactions.animationDurationMs - InCallReactions.fadeOutAnimationDuarationMs + ) + ) + } + ).joinAll() + } +} + +data class AnimatableReaction( + val inCallReaction: InCallReaction, + val verticalOffset: Animatable = Animatable(0f), + val alpha: Animatable = Animatable(1f), + val horizontalOffset: Float, +) diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/FlipCameraButton.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/FlipCameraButton.kt new file mode 100644 index 00000000000..02a4d15cc30 --- /dev/null +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/FlipCameraButton.kt @@ -0,0 +1,66 @@ +/* + * Wire + * Copyright (C) 2024 Wire Swiss GmbH + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see http://www.gnu.org/licenses/. + */ +package com.wire.android.ui.calling.ongoing.participantsview + +import androidx.compose.foundation.background +import androidx.compose.foundation.clickable +import androidx.compose.foundation.layout.padding +import androidx.compose.foundation.layout.size +import androidx.compose.foundation.shape.CircleShape +import androidx.compose.material3.Icon +import androidx.compose.runtime.Composable +import androidx.compose.ui.Modifier +import androidx.compose.ui.draw.clip +import androidx.compose.ui.res.painterResource +import androidx.compose.ui.res.stringResource +import androidx.compose.ui.unit.dp +import com.wire.android.R +import com.wire.android.ui.common.colorsScheme +import com.wire.android.ui.common.dimensions +import com.wire.android.ui.theme.WireTheme +import com.wire.android.util.ui.PreviewMultipleThemes + +@Composable +internal fun FlipCameraButton( + isOnFrontCamera: Boolean, + modifier: Modifier = Modifier, + flipCamera: () -> Unit, +) { + Icon( + modifier = modifier + .padding(dimensions().spacing12x) + .size(32.dp) + .background(color = colorsScheme().surface, shape = CircleShape) + .clip(CircleShape) + .clickable { flipCamera() } + .padding(dimensions().spacing6x), + painter = painterResource(R.drawable.ic_flip_camera), + tint = colorsScheme().onSurface, + contentDescription = if (isOnFrontCamera) { + stringResource(R.string.content_description_calling_flip_camera_on) + } else { + stringResource(R.string.content_description_calling_flip_camera_on) + } + ) +} + +@PreviewMultipleThemes +@Composable +private fun PreviewFlipCameraButton() { + WireTheme { FlipCameraButton(isOnFrontCamera = true) { } } +} diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/FloatingSelfUserTile.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/FloatingSelfUserTile.kt index 32dea7ebfb5..7be5a2bb3e0 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/FloatingSelfUserTile.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/FloatingSelfUserTile.kt @@ -60,9 +60,11 @@ fun FloatingSelfUserTile( contentHeight: Dp, contentWidth: Dp, participant: UICallParticipant, + isOnFrontCamera: Boolean, onSelfUserVideoPreviewCreated: (view: View) -> Unit, + onClearSelfUserVideoPreview: () -> Unit, + flipCamera: () -> Unit, modifier: Modifier = Modifier, - onClearSelfUserVideoPreview: () -> Unit ) { var selfVideoTileHeight by remember { mutableStateOf(contentHeight / 4) @@ -163,13 +165,14 @@ fun FloatingSelfUserTile( ) { ParticipantTile( participantTitleState = participant, - isSelfUser = true, isOnPiPMode = isOnPiPMode, shouldFillSelfUserCameraPreview = true, isSelfUserMuted = participant.isMuted, isSelfUserCameraOn = participant.isCameraOn, onSelfUserVideoPreviewCreated = onSelfUserVideoPreviewCreated, onClearSelfUserVideoPreview = onClearSelfUserVideoPreview, + isOnFrontCamera = isOnFrontCamera, + flipCamera = flipCamera, ) } } diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/ParticipantTile.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/ParticipantTile.kt index 34dfecc6af2..6936a3d6bda 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/ParticipantTile.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/ParticipantTile.kt @@ -22,9 +22,14 @@ package com.wire.android.ui.calling.ongoing.participantsview import android.view.View import android.view.ViewGroup.LayoutParams.MATCH_PARENT import android.widget.FrameLayout +import androidx.compose.animation.AnimatedVisibility +import androidx.compose.animation.fadeIn +import androidx.compose.animation.fadeOut +import androidx.compose.foundation.background import androidx.compose.foundation.border import androidx.compose.foundation.gestures.detectTransformGestures import androidx.compose.foundation.layout.Arrangement +import androidx.compose.foundation.layout.Box import androidx.compose.foundation.layout.BoxWithConstraints import androidx.compose.foundation.layout.Row import androidx.compose.foundation.layout.fillMaxSize @@ -56,6 +61,7 @@ import androidx.compose.ui.platform.LocalContext import androidx.compose.ui.platform.LocalLifecycleOwner import androidx.compose.ui.res.stringResource import androidx.compose.ui.res.vectorResource +import androidx.compose.ui.text.style.TextAlign import androidx.compose.ui.text.style.TextOverflow import androidx.compose.ui.unit.Dp import androidx.compose.ui.unit.IntSize @@ -74,11 +80,13 @@ import com.wire.android.R import com.wire.android.model.NameBasedAvatar import com.wire.android.model.UserAvatarData import com.wire.android.ui.calling.model.UICallParticipant +import com.wire.android.ui.calling.ongoing.incallreactions.InCallReactions import com.wire.android.ui.common.avatar.UserProfileAvatar import com.wire.android.ui.common.avatar.UserProfileAvatarType import com.wire.android.ui.common.colorsScheme import com.wire.android.ui.common.darkColorsScheme import com.wire.android.ui.common.dimensions +import com.wire.android.ui.common.typography import com.wire.android.ui.home.conversationslist.model.Membership import com.wire.android.ui.theme.WireTheme import com.wire.android.ui.theme.wireTypography @@ -88,16 +96,18 @@ import com.wire.kalium.logic.data.id.QualifiedID @Composable fun ParticipantTile( participantTitleState: UICallParticipant, - isSelfUser: Boolean, isSelfUserMuted: Boolean, isSelfUserCameraOn: Boolean, onSelfUserVideoPreviewCreated: (view: View) -> Unit, + isOnFrontCamera: Boolean, + flipCamera: () -> Unit, modifier: Modifier = Modifier, isOnPiPMode: Boolean = false, shouldFillSelfUserCameraPreview: Boolean = false, shouldFillOthersVideoPreview: Boolean = true, isZoomingEnabled: Boolean = false, - onClearSelfUserVideoPreview: () -> Unit + recentReaction: String? = null, + onClearSelfUserVideoPreview: () -> Unit, ) { val alpha = if (participantTitleState.hasEstablishedAudio) ContentAlpha.high else ContentAlpha.medium @@ -107,8 +117,9 @@ fun ParticipantTile( color = darkColorsScheme().surfaceContainer, shape = RoundedCornerShape(if (participantTitleState.isSpeaking) dimensions().corner8x else dimensions().corner3x), ) { + ConstraintLayout { - val (avatar, bottomRow) = createRefs() + val (avatar, bottomRow, cameraButton) = createRefs() val maxAvatarSize = dimensions().onGoingCallUserAvatarSize val activeSpeakerBorderPadding = dimensions().spacing6x @@ -136,7 +147,7 @@ fun ParticipantTile( isOnPiPMode = isOnPiPMode ) - if (isSelfUser) { + if (participantTitleState.isSelfUser) { CameraPreview( isCameraOn = isSelfUserCameraOn, shouldFill = shouldFillSelfUserCameraPreview, @@ -157,7 +168,6 @@ fun ParticipantTile( if (!isOnPiPMode) { BottomRow( participantTitleState = participantTitleState, - isSelfUser = isSelfUser, isSelfUserMuted = isSelfUserMuted, modifier = Modifier .padding( @@ -173,6 +183,42 @@ fun ParticipantTile( } ) } + + AnimatedVisibility( + modifier = Modifier + .padding(dimensions().spacing12x), + visible = recentReaction != null, + enter = fadeIn(), + exit = fadeOut(), + ) { + Box( + modifier = Modifier + .size(dimensions().inCallReactionRecentReactionSize) + .background( + color = colorsScheme().emojiBackgroundColor, + shape = RoundedCornerShape(dimensions().corner6x) + ), + contentAlignment = Alignment.Center, + ) { + Text( + text = recentReaction ?: "", + textAlign = TextAlign.Center, + style = typography().inCallReactionRecentEmoji, + ) + } + } + + if (participantTitleState.isSelfUser && isSelfUserCameraOn) { + FlipCameraButton( + modifier = Modifier + .constrainAs(cameraButton) { + top.linkTo(parent.top) + end.linkTo(parent.end) + }, + isOnFrontCamera = isOnFrontCamera, + flipCamera = flipCamera, + ) + } } } } @@ -196,7 +242,6 @@ private val activeSpeakerBorderModifier @Composable private fun BottomRow( participantTitleState: UICallParticipant, - isSelfUser: Boolean, isSelfUserMuted: Boolean, modifier: Modifier = Modifier, ) { @@ -208,7 +253,7 @@ private fun BottomRow( modifier = Modifier .padding(end = dimensions().spacing8x) .layoutId("muteIcon"), - isMuted = if (isSelfUser) isSelfUserMuted else participantTitleState.isMuted, + isMuted = if (participantTitleState.isSelfUser) isSelfUserMuted else participantTitleState.isMuted, hasEstablishedAudio = participantTitleState.hasEstablishedAudio ) UsernameTile( @@ -464,12 +509,16 @@ private fun PreviewParticipantTile( isSpeaking: Boolean = false, hasEstablishedAudio: Boolean = true, shape: PreviewTileShape = PreviewTileShape.Wide, + recentReaction: String? = null, + isSelfUser: Boolean = false, + isSelfCameraOn: Boolean = false, ) { ParticipantTile( modifier = Modifier.size(width = shape.width, height = shape.height), participantTitleState = UICallParticipant( id = QualifiedID("", ""), clientId = "client-id", + isSelfUser = isSelfUser, name = if (longName) "long user name to be displayed in participant tile during a call" else "user name", isMuted = isMuted, isSpeaking = isSpeaking, @@ -482,9 +531,11 @@ private fun PreviewParticipantTile( ), onClearSelfUserVideoPreview = {}, onSelfUserVideoPreviewCreated = {}, - isSelfUser = false, isSelfUserMuted = false, - isSelfUserCameraOn = false + isSelfUserCameraOn = isSelfCameraOn, + recentReaction = recentReaction, + isOnFrontCamera = false, + flipCamera = { }, ) } @@ -569,3 +620,24 @@ fun PreviewParticipantTallLongNameTalking() = WireTheme { fun PreviewParticipantWideLongNameTalking() = WireTheme { PreviewParticipantTile(shape = PreviewTileShape.Wide, isSpeaking = true) } + +@PreviewMultipleThemes +@Composable +fun PreviewParticipantTalkingReaction() = WireTheme { + PreviewParticipantTile( + shape = PreviewTileShape.Regular, + isSpeaking = true, + recentReaction = InCallReactions.defaultReactions[2], + ) +} + +@PreviewMultipleThemes +@Composable +fun PreviewParticipantCameraButton() = WireTheme { + PreviewParticipantTile( + shape = PreviewTileShape.Regular, + recentReaction = InCallReactions.defaultReactions[2], + isSelfUser = true, + isSelfCameraOn = true, + ) +} diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/VerticalCallingPager.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/VerticalCallingPager.kt index 9aa4550af2c..95e7d793b2e 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/VerticalCallingPager.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/VerticalCallingPager.kt @@ -61,12 +61,14 @@ fun VerticalCallingPager( isSelfUserMuted: Boolean, isSelfUserCameraOn: Boolean, isInPictureInPictureMode: Boolean, + isOnFrontCamera: Boolean, contentHeight: Dp, - currentUserId: UserId, + recentReactions: Map, onSelfVideoPreviewCreated: (view: View) -> Unit, onSelfClearVideoPreview: () -> Unit, requestVideoStreams: (participants: List) -> Unit, onDoubleTap: (selectedParticipant: SelectedParticipant) -> Unit, + flipCamera: () -> Unit, modifier: Modifier = Modifier, ) { Column( @@ -106,7 +108,9 @@ fun VerticalCallingPager( onSelfVideoPreviewCreated = onSelfVideoPreviewCreated, onSelfClearVideoPreview = onSelfClearVideoPreview, onDoubleTap = onDoubleTap, - currentUserId = currentUserId, + recentReactions = recentReactions, + isOnFrontCamera = isOnFrontCamera, + flipCamera = flipCamera, ) } else { GroupCallGrid( @@ -118,8 +122,10 @@ fun VerticalCallingPager( onSelfVideoPreviewCreated = onSelfVideoPreviewCreated, onSelfClearVideoPreview = onSelfClearVideoPreview, onDoubleTap = onDoubleTap, - currentUserId = currentUserId, isInPictureInPictureMode = isInPictureInPictureMode, + recentReactions = recentReactions, + isOnFrontCamera = isOnFrontCamera, + flipCamera = flipCamera, ) } @@ -177,8 +183,10 @@ private fun PreviewVerticalCallingPager(participants: List) { onSelfClearVideoPreview = {}, requestVideoStreams = {}, onDoubleTap = { }, + flipCamera = { }, isInPictureInPictureMode = false, - currentUserId = participants[0].id, + recentReactions = emptyMap(), + isOnFrontCamera = false, ) } diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/gridview/CallingGridView.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/gridview/CallingGridView.kt index c1208dcf50e..5ad0e33a495 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/gridview/CallingGridView.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/gridview/CallingGridView.kt @@ -50,14 +50,16 @@ fun GroupCallGrid( isSelfUserMuted: Boolean, isSelfUserCameraOn: Boolean, contentHeight: Dp, - currentUserId: UserId, + isOnFrontCamera: Boolean, onSelfVideoPreviewCreated: (view: View) -> Unit, onSelfClearVideoPreview: () -> Unit, onDoubleTap: (selectedParticipant: SelectedParticipant) -> Unit, + flipCamera: () -> Unit, + isInPictureInPictureMode: Boolean, + recentReactions: Map, modifier: Modifier = Modifier, contentPadding: Dp = dimensions().spacing4x, spacedBy: Dp = dimensions().spacing2x, - isInPictureInPictureMode: Boolean, ) { // We need the number of tiles rows needed to calculate their height val numberOfTilesRows = remember(participants.size) { @@ -81,9 +83,6 @@ fun GroupCallGrid( key = { it.id.toString() + it.clientId + pageIndex }, contentType = { getContentType(it.isCameraOn, it.isSharingScreen) } ) { participant -> - // API returns only id.value, without domain, till this get changed compare only id.value - val isSelfUser = participant.id.equalsIgnoringBlankDomain(currentUserId) - ParticipantTile( modifier = Modifier .pointerInput(Unit) { @@ -93,7 +92,7 @@ fun GroupCallGrid( SelectedParticipant( userId = participant.id, clientId = participant.clientId, - isSelfUser = isSelfUser + isSelfUser = participant.isSelfUser, ) ) } @@ -103,11 +102,13 @@ fun GroupCallGrid( .animateItem(), participantTitleState = participant, isOnPiPMode = isInPictureInPictureMode, - isSelfUser = isSelfUser, isSelfUserMuted = isSelfUserMuted, isSelfUserCameraOn = isSelfUserCameraOn, onSelfUserVideoPreviewCreated = onSelfVideoPreviewCreated, - onClearSelfUserVideoPreview = onSelfClearVideoPreview + onClearSelfUserVideoPreview = onSelfClearVideoPreview, + recentReaction = recentReactions[participant.id], + isOnFrontCamera = isOnFrontCamera, + flipCamera = flipCamera, ) } } @@ -139,8 +140,10 @@ private fun PreviewGroupCallGrid(participants: List, modifier onSelfVideoPreviewCreated = {}, onSelfClearVideoPreview = {}, onDoubleTap = { }, - currentUserId = UserId("id", "domain"), isInPictureInPictureMode = false, + recentReactions = emptyMap(), + isOnFrontCamera = false, + flipCamera = {}, ) } } diff --git a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/horizentalview/CallingHorizontalView.kt b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/horizentalview/CallingHorizontalView.kt index 42dce497b4d..5ff9d31eb57 100644 --- a/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/horizentalview/CallingHorizontalView.kt +++ b/app/src/main/kotlin/com/wire/android/ui/calling/ongoing/participantsview/horizentalview/CallingHorizontalView.kt @@ -49,10 +49,12 @@ fun CallingHorizontalView( isSelfUserMuted: Boolean, isSelfUserCameraOn: Boolean, contentHeight: Dp, - currentUserId: UserId, onSelfVideoPreviewCreated: (view: View) -> Unit, onSelfClearVideoPreview: () -> Unit, + recentReactions: Map, onDoubleTap: (selectedParticipant: SelectedParticipant) -> Unit, + isOnFrontCamera: Boolean, + flipCamera: () -> Unit, modifier: Modifier = Modifier, contentPadding: Dp = dimensions().spacing4x, spacedBy: Dp = dimensions().spacing2x, @@ -69,8 +71,6 @@ fun CallingHorizontalView( verticalArrangement = Arrangement.spacedBy(spacedBy) ) { items(items = participants, key = { it.id.toString() + it.clientId }) { participant -> - // API returns only id.value, without domain, till this get changed compare only id.value - val isSelfUser = participant.id.equalsIgnoringBlankDomain(currentUserId) ParticipantTile( modifier = Modifier .pointerInput(Unit) { @@ -80,7 +80,7 @@ fun CallingHorizontalView( SelectedParticipant( userId = participant.id, clientId = participant.clientId, - isSelfUser = isSelfUser + isSelfUser = participant.isSelfUser, ) ) } @@ -90,11 +90,13 @@ fun CallingHorizontalView( .height(tileHeight) .animateItem(), participantTitleState = participant, - isSelfUser = isSelfUser, isSelfUserMuted = isSelfUserMuted, isSelfUserCameraOn = isSelfUserCameraOn, onSelfUserVideoPreviewCreated = onSelfVideoPreviewCreated, onClearSelfUserVideoPreview = onSelfClearVideoPreview, + recentReaction = recentReactions[participant.id], + isOnFrontCamera = isOnFrontCamera, + flipCamera = flipCamera, ) } } @@ -111,10 +113,12 @@ fun PreviewCallingHorizontalView( isSelfUserMuted = true, isSelfUserCameraOn = false, contentHeight = 800.dp, - currentUserId = UserId("id", "domain"), + recentReactions = emptyMap(), onSelfVideoPreviewCreated = {}, onSelfClearVideoPreview = {}, onDoubleTap = { }, + isOnFrontCamera = false, + flipCamera = { }, ) } } diff --git a/app/src/main/kotlin/com/wire/android/util/ExpiringMap.kt b/app/src/main/kotlin/com/wire/android/util/ExpiringMap.kt new file mode 100644 index 00000000000..4b60e3a4cff --- /dev/null +++ b/app/src/main/kotlin/com/wire/android/util/ExpiringMap.kt @@ -0,0 +1,74 @@ +/* + * Wire + * Copyright (C) 2024 Wire Swiss GmbH + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see http://www.gnu.org/licenses/. + */ +package com.wire.android.util + +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.Job +import kotlinx.coroutines.delay +import kotlinx.coroutines.launch +import java.util.concurrent.ConcurrentHashMap + +/** + * Map implementation that removes entries after a delay. Not thread-safe. + */ +class ExpiringMap( + private val scope: CoroutineScope, + private val expiration: Long, + private val delegate: MutableMap, + private val currentTime: () -> Long = { System.currentTimeMillis() }, +) : MutableMap by delegate { + + private val timestamps: MutableMap = ConcurrentHashMap() + private var cleanupJob: Job? = null + + override fun put(key: K, value: V): V? { + return delegate.put(key, value).also { + timestamps.put(key, currentTime() + expiration) + scheduleCleanup() + } + } + + override fun remove(key: K): V? { + return delegate.remove(key).also { + timestamps.remove(key) + scheduleCleanup() + } + } + + private fun scheduleCleanup() { + cleanupJob?.cancel() + timestamps.values.sorted().firstOrNull()?.let { nextExpiration -> + val delayToNext = nextExpiration - currentTime() + cleanupJob = scope.launch { + delay(delayToNext) + removeAllExpired() + } + } + } + + private fun removeAllExpired() { + val now = currentTime() + timestamps.entries.onEach { (key, expiration) -> + if (expiration <= now) { + delegate.remove(key) + } + } + timestamps.entries.removeAll { it.value <= now } + scheduleCleanup() + } +} diff --git a/app/src/main/kotlin/com/wire/android/util/extension/Flow.kt b/app/src/main/kotlin/com/wire/android/util/extension/Flow.kt index 7640755fae3..cabcc055463 100644 --- a/app/src/main/kotlin/com/wire/android/util/extension/Flow.kt +++ b/app/src/main/kotlin/com/wire/android/util/extension/Flow.kt @@ -19,7 +19,10 @@ package com.wire.android.util.extension import kotlinx.coroutines.delay +import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.flow.flow +import kotlinx.coroutines.flow.map +import kotlinx.coroutines.flow.withIndex fun intervalFlow(periodMs: Long, initialDelayMs: Long = 0L, stopWhen: () -> Boolean = { false }) = flow { @@ -29,3 +32,9 @@ fun intervalFlow(periodMs: Long, initialDelayMs: Long = 0L, stopWhen: () -> Bool delay(periodMs) } } + +fun Flow.withDelayAfterFirst(timeMillis: Long): Flow = withIndex() + .map { (index, value) -> + if (index > 0) delay(timeMillis) + value + } diff --git a/app/src/main/res/drawable/ic_flip_camera.xml b/app/src/main/res/drawable/ic_flip_camera.xml new file mode 100644 index 00000000000..1f0c24e6eb4 --- /dev/null +++ b/app/src/main/res/drawable/ic_flip_camera.xml @@ -0,0 +1,30 @@ + + + + + + + diff --git a/app/src/main/res/drawable/ic_incall_reactions.xml b/app/src/main/res/drawable/ic_incall_reactions.xml new file mode 100644 index 00000000000..d5b6fefbf03 --- /dev/null +++ b/app/src/main/res/drawable/ic_incall_reactions.xml @@ -0,0 +1,32 @@ + + + + + + + diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index f12f3cbc10a..0239968ec0e 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -162,6 +162,8 @@ Turn camera off Turn speaker on Turn speaker off + Show in call reactions panel + Hide in call reactions panel Show more options Reply to the message Cancel message reply @@ -1662,4 +1664,5 @@ In group conversations, the group admin can overwrite this setting. You\'ve created or joined a team with this email address on another device. Wire could not complete your team creation due to a slow internet connection. Wire could not complete your team creation due to an unknown error. + You diff --git a/app/src/test/kotlin/com/wire/android/mapper/UICallParticipantMapperTest.kt b/app/src/test/kotlin/com/wire/android/mapper/UICallParticipantMapperTest.kt index d3c23da2a63..a3781781a9f 100644 --- a/app/src/test/kotlin/com/wire/android/mapper/UICallParticipantMapperTest.kt +++ b/app/src/test/kotlin/com/wire/android/mapper/UICallParticipantMapperTest.kt @@ -19,6 +19,7 @@ package com.wire.android.mapper import com.wire.kalium.logic.data.call.Participant +import com.wire.kalium.logic.data.conversation.ClientId import com.wire.kalium.logic.data.id.QualifiedID import io.mockk.MockKAnnotations import kotlinx.coroutines.test.runTest @@ -42,12 +43,37 @@ class UICallParticipantMapperTest { accentId = -1 ) // When - val result = mapper.toUICallParticipant(item) + val result = mapper.toUICallParticipant(item, ClientId("clientId")) // Then assert( result.id == item.id && result.clientId == item.clientId && result.name == item.name && result.isMuted == item.isMuted && result.isSpeaking == item.isSpeaking && result.avatar?.userAssetId == item.avatarAssetId - && result.isCameraOn == item.isCameraOn + && result.isCameraOn == item.isCameraOn && result.isSelfUser == true + ) + } + + @Test + fun givenParticipant_whenMappingToUICallParticipant_thenCorrectValuesShouldBeReturnedForNonSelfUser() = runTest { + val (_, mapper) = Arrangement().arrange() + // Given + val item = Participant( + id = QualifiedID("value", "domain"), + clientId = "clientId", + name = "name", + isMuted = false, + isCameraOn = false, + isSpeaking = false, + isSharingScreen = false, + hasEstablishedAudio = true, + accentId = -1 + ) + // When + val result = mapper.toUICallParticipant(item, ClientId("otherClientId")) + // Then + assert( + result.id == item.id && result.clientId == item.clientId && result.name == item.name && result.isMuted == item.isMuted + && result.isSpeaking == item.isSpeaking && result.avatar?.userAssetId == item.avatarAssetId + && result.isCameraOn == item.isCameraOn && result.isSelfUser == false ) } diff --git a/app/src/test/kotlin/com/wire/android/ui/calling/OngoingCallViewModelTest.kt b/app/src/test/kotlin/com/wire/android/ui/calling/OngoingCallViewModelTest.kt index 92937700fdb..587abd03426 100644 --- a/app/src/test/kotlin/com/wire/android/ui/calling/OngoingCallViewModelTest.kt +++ b/app/src/test/kotlin/com/wire/android/ui/calling/OngoingCallViewModelTest.kt @@ -21,6 +21,7 @@ package com.wire.android.ui.calling import com.wire.android.config.CoroutineTestExtension import com.wire.android.config.NavigationTestExtension import com.wire.android.datastore.GlobalDataStore +import com.wire.android.ui.calling.OngoingCallViewModelTest.Arrangement import com.wire.android.ui.calling.model.UICallParticipant import com.wire.android.ui.calling.ongoing.OngoingCallViewModel import com.wire.android.ui.calling.ongoing.fullscreen.SelectedParticipant @@ -302,6 +303,7 @@ class OngoingCallViewModelTest { private val participant1 = UICallParticipant( id = QualifiedID("value1", "domain"), clientId = "client-id1", + isSelfUser = false, name = "name1", isMuted = false, isSpeaking = false, @@ -314,6 +316,7 @@ class OngoingCallViewModelTest { private val participant2 = UICallParticipant( id = QualifiedID("value2", "domain"), clientId = "client-id2", + isSelfUser = false, name = "name2", isMuted = false, isSpeaking = false, @@ -326,6 +329,7 @@ class OngoingCallViewModelTest { private val participant3 = UICallParticipant( id = QualifiedID("value3", "domain"), clientId = "client-id3", + isSelfUser = false, name = "name3", isMuted = false, isSpeaking = false, diff --git a/app/src/test/kotlin/com/wire/android/ui/calling/SharedCallingViewModelTest.kt b/app/src/test/kotlin/com/wire/android/ui/calling/SharedCallingViewModelTest.kt index 18e62a81877..59f69ff85b7 100644 --- a/app/src/test/kotlin/com/wire/android/ui/calling/SharedCallingViewModelTest.kt +++ b/app/src/test/kotlin/com/wire/android/ui/calling/SharedCallingViewModelTest.kt @@ -19,35 +19,50 @@ package com.wire.android.ui.calling import android.view.View +import app.cash.turbine.test +import com.wire.android.assertIs import com.wire.android.config.CoroutineTestExtension import com.wire.android.config.NavigationTestExtension import com.wire.android.config.TestDispatcherProvider +import com.wire.android.framework.TestUser import com.wire.android.mapper.UICallParticipantMapper import com.wire.android.mapper.UserTypeMapper import com.wire.android.media.CallRinger +import com.wire.android.ui.calling.model.ReactionSender +import com.wire.kalium.logic.NetworkFailure +import com.wire.kalium.logic.data.call.Call +import com.wire.kalium.logic.data.call.InCallReactionMessage import com.wire.kalium.logic.data.call.VideoState +import com.wire.kalium.logic.data.conversation.ClientId import com.wire.kalium.logic.data.id.ConversationId import com.wire.kalium.logic.feature.call.usecase.EndCallUseCase import com.wire.kalium.logic.feature.call.usecase.FlipToBackCameraUseCase import com.wire.kalium.logic.feature.call.usecase.FlipToFrontCameraUseCase import com.wire.kalium.logic.feature.call.usecase.MuteCallUseCase import com.wire.kalium.logic.feature.call.usecase.ObserveEstablishedCallWithSortedParticipantsUseCase +import com.wire.kalium.logic.feature.call.usecase.ObserveInCallReactionsUseCase import com.wire.kalium.logic.feature.call.usecase.ObserveSpeakerUseCase import com.wire.kalium.logic.feature.call.usecase.SetVideoPreviewUseCase import com.wire.kalium.logic.feature.call.usecase.TurnLoudSpeakerOffUseCase import com.wire.kalium.logic.feature.call.usecase.TurnLoudSpeakerOnUseCase import com.wire.kalium.logic.feature.call.usecase.UnMuteCallUseCase import com.wire.kalium.logic.feature.call.usecase.video.UpdateVideoStateUseCase +import com.wire.kalium.logic.feature.client.ObserveCurrentClientIdUseCase import com.wire.kalium.logic.feature.conversation.ObserveConversationDetailsUseCase +import com.wire.kalium.logic.feature.incallreaction.SendInCallReactionUseCase +import com.wire.kalium.logic.functional.Either import io.mockk.MockKAnnotations import io.mockk.coEvery import io.mockk.coVerify import io.mockk.every import io.mockk.impl.annotations.MockK import io.mockk.verify +import kotlinx.coroutines.flow.MutableSharedFlow import kotlinx.coroutines.flow.emptyFlow +import kotlinx.coroutines.flow.flowOf import kotlinx.coroutines.test.advanceUntilIdle import kotlinx.coroutines.test.runTest +import org.amshove.kluent.internal.assertEquals import org.amshove.kluent.shouldBeEqualTo import org.junit.jupiter.api.BeforeEach import org.junit.jupiter.api.Test @@ -93,6 +108,15 @@ class SharedCallingViewModelTest { @MockK private lateinit var observeSpeaker: ObserveSpeakerUseCase + @MockK + lateinit var observeInCallReactionsUseCase: ObserveInCallReactionsUseCase + + @MockK + lateinit var sendInCallReactionUseCase: SendInCallReactionUseCase + + @MockK + lateinit var getCurrentClientId: ObserveCurrentClientIdUseCase + @MockK private lateinit var callRinger: CallRinger @@ -111,12 +135,17 @@ class SharedCallingViewModelTest { private lateinit var sharedCallingViewModel: SharedCallingViewModel + val reactionsFlow = MutableSharedFlow() + val callFlow = MutableSharedFlow() + @BeforeEach fun setup() { MockKAnnotations.init(this) - coEvery { establishedCall.invoke(any()) } returns emptyFlow() + coEvery { establishedCall.invoke(any()) } returns callFlow coEvery { observeConversationDetails.invoke(any()) } returns emptyFlow() coEvery { observeSpeaker.invoke() } returns emptyFlow() + coEvery { observeInCallReactionsUseCase(any()) } returns reactionsFlow + coEvery { getCurrentClientId() } returns flowOf(ClientId("clientId")) sharedCallingViewModel = SharedCallingViewModel( conversationId = conversationId, @@ -135,6 +164,9 @@ class SharedCallingViewModelTest { callRinger = callRinger, uiCallParticipantMapper = uiCallParticipantMapper, userTypeMapper = userTypeMapper, + observeInCallReactionsUseCase = observeInCallReactionsUseCase, + sendInCallReactionUseCase = sendInCallReactionUseCase, + getCurrentClientId = getCurrentClientId, dispatchers = TestDispatcherProvider() ) } @@ -312,6 +344,97 @@ class SharedCallingViewModelTest { coVerify(exactly = 1) { setVideoPreview(any(), any()) } } + @Test + fun givenAnOngoingCall_WhenInCallReactionIsReceived_ThenNewEmojiIsEmitted() = runTest { + + sharedCallingViewModel.inCallReactions.test { + + // when + reactionsFlow.emit(InCallReactionMessage(conversationId, TestUser.USER_ID, setOf("👍", "🎉"))) + + val reaction1 = awaitItem() + val reaction2 = awaitItem() + + // then + assertEquals("👍", reaction1.emoji) + assertIs(reaction1.sender) + assertEquals("🎉", reaction2.emoji) + assertIs(reaction2.sender) + } + } + + @Test + fun givenAnOngoingCall_WhenInCallReactionIsReceived_ThenNewRecentReactionEmitted() = runTest { + + // when + reactionsFlow.emit(InCallReactionMessage(conversationId, TestUser.USER_ID, setOf("👍"))) + + val recentReaction = sharedCallingViewModel.recentReactions.getValue(TestUser.USER_ID) + + // then + assertEquals("👍", recentReaction) + } + + @Test + fun givenAnOngoingCall_WhenNewInCallReactionIsReceived_ThenRecentReactionUpdated() = runTest { + + // when + reactionsFlow.emit(InCallReactionMessage(conversationId, TestUser.USER_ID, setOf("👍", "🎉"))) + + val recentReaction = sharedCallingViewModel.recentReactions.getValue(TestUser.USER_ID) + + // then + assertEquals("🎉", recentReaction) + } + + @Test + fun givenAnOngoingCall_WhenInCallReactionIsSent_ThenReactionMessageIsSent() = runTest { + + // given + coEvery { sendInCallReactionUseCase(conversationId, any()) } returns Either.Right(Unit) + + // when + sharedCallingViewModel.onReactionClick("👍") + + // then + coVerify(exactly = 1) { + sendInCallReactionUseCase(OngoingCallViewModelTest.Companion.conversationId, "👍") + } + } + + @Test + fun givenAnOngoingCall_WhenInCallReactionIsSent_ThenNewEmojiIsEmitted() = runTest { + + // given + coEvery { sendInCallReactionUseCase(conversationId, any()) } returns Either.Right(Unit) + + sharedCallingViewModel.inCallReactions.test { + // when + sharedCallingViewModel.onReactionClick("👍") + + val reaction = awaitItem() + + // then + assertEquals("👍", reaction.emoji) + assertIs(reaction.sender) + } + } + + @Test + fun givenAnOngoingCall_WhenInCallReactionSentFails_ThenNoEmojiIsEmitted() = runTest { + // given + coEvery { sendInCallReactionUseCase(conversationId, any()) } returns + Either.Left(NetworkFailure.NoNetworkConnection(IllegalStateException())) + + sharedCallingViewModel.inCallReactions.test { + // when + sharedCallingViewModel.onReactionClick("👍") + + // then + expectNoEvents() + } + } + companion object { private val conversationId = ConversationId("some-dummy-value", "some.dummy.domain") } diff --git a/app/src/test/kotlin/com/wire/android/util/ExpiringMapTest.kt b/app/src/test/kotlin/com/wire/android/util/ExpiringMapTest.kt new file mode 100644 index 00000000000..0fa3f809b98 --- /dev/null +++ b/app/src/test/kotlin/com/wire/android/util/ExpiringMapTest.kt @@ -0,0 +1,116 @@ +/* + * Wire + * Copyright (C) 2024 Wire Swiss GmbH + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see http://www.gnu.org/licenses/. + */ +package com.wire.android.util + +import kotlinx.coroutines.test.TestScope +import kotlinx.coroutines.test.advanceTimeBy +import kotlinx.coroutines.test.currentTime +import kotlinx.coroutines.test.runTest +import org.amshove.kluent.internal.assertEquals +import org.junit.jupiter.api.Test + +class ExpiringMapTest { + + @Test + fun `check new item can be added map `() = runTest { + // given + val map = withTestExpiringMap() + + // when + map.put("testKey", "testValue") + + // then + assertEquals("testValue", map["testKey"]) + } + + @Test + fun `check item can be removed from map before expiration`() = runTest { + // given + val map = withTestExpiringMap() + + // when + map.put("testKey", "testValue") + map.remove("testKey") + + // then + assertEquals(null, map["testKey"]) + } + + @Test + fun `check item can not be obtained before expiration`() = runTest { + // given + val map = withTestExpiringMap() + + // when + map.put("testKey", "testValue") + advanceTimeBy(300) + + // then + assertEquals("testValue", map["testKey"]) + } + + @Test + fun `check item can not be obtained after expiration`() = runTest { + // given + val map = withTestExpiringMap() + + // when + map.put("testKey", "testValue") + advanceTimeBy(301) + + // then + assertEquals(null, map["testKey"]) + } + + @Test + fun `check adding item with existing key resets expiration`() = runTest { + // given + val map = withTestExpiringMap() + + // when + map.put("testKey", "testValue") + advanceTimeBy(300) + map.put("testKey", "testValue2") + advanceTimeBy(300) + + // then + assertEquals("testValue2", map["testKey"]) + } + + @Test + fun `check adding item with non-existing key keeps expiration for other keys`() = runTest { + // given + val map = withTestExpiringMap() + + // when + map.put("testKey", "testValue") + advanceTimeBy(200) + map.put("testKey2", "testValue2") + advanceTimeBy(200) + + // then + assertEquals(null, map["testKey"]) + } + + private fun TestScope.withTestExpiringMap(): MutableMap = ExpiringMap( + scope = this.backgroundScope, + expiration = 300, + delegate = mutableMapOf(), + currentTime = { currentTime } + ) +} diff --git a/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireColorScheme.kt b/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireColorScheme.kt index e6599228ef0..bfcd31393dc 100644 --- a/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireColorScheme.kt +++ b/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireColorScheme.kt @@ -81,6 +81,8 @@ class WireColorScheme( // accents val groupAvatarColors: List, val wireAccentColors: WireAccentColors, + + val emojiBackgroundColor: Color, ) { fun toColorScheme(): ColorScheme = ColorScheme( primary = primary, onPrimary = onPrimary, @@ -182,6 +184,7 @@ private val LightWireColorScheme = WireColorScheme( Accent.Unknown -> WireColorPalette.LightBlue500 } }, + emojiBackgroundColor = Color.White, ) // Dark WireColorScheme @@ -257,6 +260,7 @@ private val DarkWireColorScheme = WireColorScheme( Accent.Unknown -> WireColorPalette.DarkBlue500 } }, + emojiBackgroundColor = Color.White, ) @PackagePrivate diff --git a/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireDimensions.kt b/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireDimensions.kt index 324e9477336..d88a5b59776 100644 --- a/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireDimensions.kt +++ b/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireDimensions.kt @@ -107,6 +107,8 @@ data class WireDimensions( val badgeSmallMinSize: DpSize, val badgeSmallMinClickableSize: DpSize, val onMoreOptionsButtonCornerRadius: Dp, + val inCallReactionButtonSize: Dp, + val inCallReactionRecentReactionSize: Dp, // Dialog val dialogButtonsSpacing: Dp, val dialogTextsSpacing: Dp, @@ -183,6 +185,8 @@ data class WireDimensions( val groupButtonHeight: Dp, // Calling val defaultCallingControlsSize: Dp, + val defaultCallingControlsHeight: Dp, + val defaultCallingControlsWidth: Dp, val defaultCallingControlsIconSize: Dp, val bigCallingControlsSize: Dp, val bigCallingHangUpButtonIconSize: Dp, @@ -341,11 +345,13 @@ private val DefaultPhonePortraitWireDimensions: WireDimensions = WireDimensions( systemMessageIconLargeSize = 18.dp, groupButtonHeight = 82.dp, defaultCallingControlsSize = 56.dp, + defaultCallingControlsHeight = 40.dp, + defaultCallingControlsWidth = 56.dp, defaultCallingControlsIconSize = 20.dp, bigCallingControlsSize = 72.dp, bigCallingHangUpButtonIconSize = 32.dp, bigCallingAcceptButtonIconSize = 24.dp, - defaultSheetPeekHeight = 100.dp, + defaultSheetPeekHeight = 72.dp, defaultOutgoingCallSheetPeekHeight = 281.dp, onGoingCallUserAvatarSize = 72.dp, onGoingCallTileUsernameMaxWidth = 120.dp, @@ -360,6 +366,8 @@ private val DefaultPhonePortraitWireDimensions: WireDimensions = WireDimensions( importedMediaAssetSize = 120.dp, typingIndicatorHeight = 24.dp, legalHoldBannerMinHeight = 26.dp, + inCallReactionButtonSize = 48.dp, + inCallReactionRecentReactionSize = 32.dp, ) private val DefaultPhoneLandscapeWireDimensions: WireDimensions = DefaultPhonePortraitWireDimensions diff --git a/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireTypography.kt b/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireTypography.kt index 79da10ada0b..e1f290c34d5 100644 --- a/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireTypography.kt +++ b/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireTypography.kt @@ -37,7 +37,9 @@ data class WireTypography( val label01: TextStyle, val label02: TextStyle, val label03: TextStyle, val label04: TextStyle, val label05: TextStyle, val badge01: TextStyle, val subline01: TextStyle, - val code01: TextStyle + val code01: TextStyle, + val inCallReactionEmoji: TextStyle, + val inCallReactionRecentEmoji: TextStyle, ) { fun toTypography() = Typography( titleLarge = title01, titleMedium = title02, titleSmall = title03, @@ -69,7 +71,9 @@ private val DefaultWireTypography = WireTypography( label05 = WireTypographyBase.Label05, badge01 = WireTypographyBase.Badge01, subline01 = WireTypographyBase.SubLine01, - code01 = WireTypographyBase.Code01 + code01 = WireTypographyBase.Code01, + inCallReactionEmoji = WireTypographyBase.InCallEmoji, + inCallReactionRecentEmoji = WireTypographyBase.InCallEmojiRecent, ) @PackagePrivate diff --git a/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireTypographyBase.kt b/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireTypographyBase.kt index 0a12f33559e..e8ef201ffd2 100644 --- a/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireTypographyBase.kt +++ b/core/ui-common/src/main/kotlin/com/wire/android/ui/theme/WireTypographyBase.kt @@ -148,4 +148,10 @@ object WireTypographyBase { lineHeight = 28.13.sp, textAlign = TextAlign.Center ) + val InCallEmoji = TextStyle( + fontSize = 32.sp, + ) + val InCallEmojiRecent = TextStyle( + fontSize = 20.sp, + ) } diff --git a/kalium b/kalium index 91b8319e99d..d168aa2c17c 160000 --- a/kalium +++ b/kalium @@ -1 +1 @@ -Subproject commit 91b8319e99d83e486751967c8adf4f027d57a82e +Subproject commit d168aa2c17c460263910294b80f9ca402baaf4cb