From 6e37dc8bd1e3ec9db1a031b63d127029d833cac8 Mon Sep 17 00:00:00 2001 From: Oussama Hassine Date: Fri, 13 Dec 2024 17:47:02 +0100 Subject: [PATCH 1/2] feat: update Countly events for personal to team migration (WPB-11318) - RC (#3741) --- .../userprofile/self/SelfUserProfileScreen.kt | 9 ++++- .../self/SelfUserProfileViewModel.kt | 10 ++++- .../teammigration/TeamMigrationScreen.kt | 18 ++++++++- .../teammigration/TeamMigrationViewModel.kt | 24 ++++++----- .../self/SelfUserProfileViewModelTest.kt | 20 +++++++++- .../TeamMigrationViewModelTest.kt | 21 ---------- .../feature/analytics/model/AnalyticsEvent.kt | 40 ++++++++++++++----- kalium | 2 +- 8 files changed, 97 insertions(+), 47 deletions(-) diff --git a/app/src/main/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileScreen.kt b/app/src/main/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileScreen.kt index 628b9107c6b..8814568aff8 100644 --- a/app/src/main/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileScreen.kt +++ b/app/src/main/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileScreen.kt @@ -125,7 +125,12 @@ fun SelfUserProfileScreen( SelfUserProfileContent( state = viewModelSelf.userProfileState, - onCloseClick = navigator::navigateBack, + onCloseClick = { + if (viewModelSelf.userProfileState.isAbleToMigrateToTeamAccount) { + viewModelSelf.sendPersonalToTeamMigrationDismissed() + } + navigator.navigateBack() + }, logout = { viewModelSelf.logout(it, NavigationSwitchAccountActions(navigator::navigate)) }, onChangeUserProfilePicture = { navigator.navigate( @@ -154,7 +159,7 @@ fun SelfUserProfileScreen( navigator.navigate(NavigationCommand(SelfQRCodeScreenDestination(viewModelSelf.userProfileState.userName))) }, onCreateAccount = { - viewModelSelf.sendPersonalToTeamMigrationEvent() + viewModelSelf.sendPersonalToTeamMigrationEventCTAClicked() navigator.navigate(NavigationCommand(TeamMigrationScreenDestination)) }, onAccountDetailsClick = { navigator.navigate(NavigationCommand(MyAccountScreenDestination)) }, diff --git a/app/src/main/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileViewModel.kt b/app/src/main/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileViewModel.kt index d2cce6e052a..f90238bafeb 100644 --- a/app/src/main/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileViewModel.kt +++ b/app/src/main/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileViewModel.kt @@ -317,7 +317,7 @@ class SelfUserProfileViewModel @Inject constructor( anonymousAnalyticsManager.sendEvent(AnalyticsEvent.QrCode.Click(!userProfileState.teamName.isNullOrBlank())) } - fun sendPersonalToTeamMigrationEvent() { + fun sendPersonalToTeamMigrationEventCTAClicked() { anonymousAnalyticsManager.sendEvent( AnalyticsEvent.PersonalTeamMigration.ClickedPersonalTeamMigrationCta( createTeamButtonClicked = true @@ -325,6 +325,14 @@ class SelfUserProfileViewModel @Inject constructor( ) } + fun sendPersonalToTeamMigrationDismissed() { + anonymousAnalyticsManager.sendEvent( + AnalyticsEvent.PersonalTeamMigration.ClickedPersonalTeamMigrationCta( + dismissCreateTeamButtonClicked = true + ) + ) + } + sealed class ErrorCodes { data object DownloadUserInfoError : ErrorCodes() } diff --git a/app/src/main/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationScreen.kt b/app/src/main/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationScreen.kt index 1fb026631a7..acea62d7afc 100644 --- a/app/src/main/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationScreen.kt +++ b/app/src/main/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationScreen.kt @@ -113,7 +113,6 @@ fun TeamMigrationScreen( if (navController.currentDestination?.route == NavGraphs.personalToTeamMigration.destinations.last().route) { navigator.navigateBack() } else { - teamMigrationViewModel.sendPersonalToTeamMigrationDismissed() teamMigrationViewModel.showMigrationLeaveDialog() } } @@ -150,6 +149,23 @@ fun TeamMigrationScreen( teamMigrationViewModel.sendPersonalTeamCreationFlowCanceledEvent( modalLeaveClicked = true ) + when (teamMigrationViewModel.teamMigrationState.currentStep) { + TEAM_MIGRATION_TEAM_PLAN_STEP -> { + teamMigrationViewModel.sendPersonalTeamCreationFlowStoppedEvent(isOnDisclaimerStep = true) + } + + TEAM_MIGRATION_TEAM_NAME_STEP -> { + teamMigrationViewModel.sendPersonalTeamCreationFlowStoppedEvent(isOnTeamNameStep = true) + } + + TEAM_MIGRATION_CONFIRMATION_STEP -> { + teamMigrationViewModel.sendPersonalTeamCreationFlowStoppedEvent(isOnConfirmationStep = true) + } + + else -> { + // Nothing to send + } + } navigator.navigateBack() } } diff --git a/app/src/main/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationViewModel.kt b/app/src/main/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationViewModel.kt index 1c527b15d76..bf2b90fde04 100644 --- a/app/src/main/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationViewModel.kt +++ b/app/src/main/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationViewModel.kt @@ -57,14 +57,6 @@ class TeamMigrationViewModel @Inject constructor( teamMigrationState = teamMigrationState.copy(shouldShowMigrationLeaveDialog = false) } - fun sendPersonalToTeamMigrationDismissed() { - anonymousAnalyticsManager.sendEvent( - AnalyticsEvent.PersonalTeamMigration.ClickedPersonalTeamMigrationCta( - dismissCreateTeamButtonClicked = true - ) - ) - } - fun sendPersonalTeamCreationFlowStartedEvent(step: Int) { anonymousAnalyticsManager.sendEvent( AnalyticsEvent.PersonalTeamMigration.PersonalTeamCreationFlowStarted(step) @@ -81,20 +73,32 @@ class TeamMigrationViewModel @Inject constructor( ) { anonymousAnalyticsManager.sendEvent( AnalyticsEvent.PersonalTeamMigration.PersonalTeamCreationFlowCanceled( - teamName = teamMigrationState.teamNameTextState.text.toString(), modalLeaveClicked = modalLeaveClicked, modalContinueClicked = modalContinueClicked ) ) } + fun sendPersonalTeamCreationFlowStoppedEvent( + isOnDisclaimerStep: Boolean? = null, + isOnTeamNameStep: Boolean? = null, + isOnConfirmationStep: Boolean? = null + ) { + anonymousAnalyticsManager.sendEvent( + AnalyticsEvent.PersonalTeamMigration.PersonalTeamCreationFlowStopped( + isOnTeamNameStep = isOnTeamNameStep, + isOnConfirmationStep = isOnConfirmationStep, + isOnDisclaimerStep = isOnDisclaimerStep + ) + ) + } + fun sendPersonalTeamCreationFlowCompletedEvent( modalOpenTeamManagementButtonClicked: Boolean? = null, backToWireButtonClicked: Boolean? = null ) { anonymousAnalyticsManager.sendEvent( AnalyticsEvent.PersonalTeamMigration.PersonalTeamCreationFlowCompleted( - teamName = teamMigrationState.teamNameTextState.text.toString(), modalOpenTeamManagementButtonClicked = modalOpenTeamManagementButtonClicked, backToWireButtonClicked = backToWireButtonClicked ) diff --git a/app/src/test/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileViewModelTest.kt b/app/src/test/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileViewModelTest.kt index 74d035e56c5..79d58e2c378 100644 --- a/app/src/test/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileViewModelTest.kt +++ b/app/src/test/kotlin/com/wire/android/ui/userprofile/self/SelfUserProfileViewModelTest.kt @@ -72,7 +72,7 @@ class SelfUserProfileViewModelTest { .withLegalHoldStatus(LegalHoldStateForSelfUser.Disabled) .arrange() - viewModel.sendPersonalToTeamMigrationEvent() + viewModel.sendPersonalToTeamMigrationEventCTAClicked() verify(exactly = 1) { arrangement.anonymousAnalyticsManager.sendEvent( @@ -82,4 +82,22 @@ class SelfUserProfileViewModelTest { ) } } + + @Test + fun `given close modal event, when sendPersonalToTeamMigrationDismissed is called, then send the event`() = + runTest { + val (arrangement, viewModel) = SelfUserProfileViewModelArrangement() + .withLegalHoldStatus(LegalHoldStateForSelfUser.Disabled) + .arrange() + + viewModel.sendPersonalToTeamMigrationDismissed() + + verify(exactly = 1) { + arrangement.anonymousAnalyticsManager.sendEvent( + AnalyticsEvent.PersonalTeamMigration.ClickedPersonalTeamMigrationCta( + dismissCreateTeamButtonClicked = true + ) + ) + } + } } diff --git a/app/src/test/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationViewModelTest.kt b/app/src/test/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationViewModelTest.kt index fcaf3011bb8..8fb30394cc2 100644 --- a/app/src/test/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationViewModelTest.kt +++ b/app/src/test/kotlin/com/wire/android/ui/userprofile/teammigration/TeamMigrationViewModelTest.kt @@ -65,23 +65,6 @@ class TeamMigrationViewModelTest { assertEquals(false, viewModel.teamMigrationState.shouldShowMigrationLeaveDialog) } - @Test - fun `given close modal event, when sendPersonalToTeamMigrationDismissed is called, then send the event`() = - runTest { - val (arrangement, viewModel) = Arrangement() - .arrange() - - viewModel.sendPersonalToTeamMigrationDismissed() - - verify(exactly = 1) { - arrangement.anonymousAnalyticsManager.sendEvent( - AnalyticsEvent.PersonalTeamMigration.ClickedPersonalTeamMigrationCta( - dismissCreateTeamButtonClicked = true - ) - ) - } - } - @Test fun `given the step of migration flow, when sendPersonalTeamCreationFlowStartedEvent is called, then send the event`() = runTest { @@ -109,7 +92,6 @@ class TeamMigrationViewModelTest { verify(exactly = 1) { arrangement.anonymousAnalyticsManager.sendEvent( AnalyticsEvent.PersonalTeamMigration.PersonalTeamCreationFlowCanceled( - teamName = viewModel.teamMigrationState.teamNameTextState.text.toString(), modalLeaveClicked = true ) ) @@ -127,7 +109,6 @@ class TeamMigrationViewModelTest { verify(exactly = 1) { arrangement.anonymousAnalyticsManager.sendEvent( AnalyticsEvent.PersonalTeamMigration.PersonalTeamCreationFlowCanceled( - teamName = viewModel.teamMigrationState.teamNameTextState.text.toString(), modalContinueClicked = true ) ) @@ -147,7 +128,6 @@ class TeamMigrationViewModelTest { verify(exactly = 1) { arrangement.anonymousAnalyticsManager.sendEvent( AnalyticsEvent.PersonalTeamMigration.PersonalTeamCreationFlowCompleted( - teamName = viewModel.teamMigrationState.teamNameTextState.text.toString(), modalOpenTeamManagementButtonClicked = true ) ) @@ -165,7 +145,6 @@ class TeamMigrationViewModelTest { verify(exactly = 1) { arrangement.anonymousAnalyticsManager.sendEvent( AnalyticsEvent.PersonalTeamMigration.PersonalTeamCreationFlowCompleted( - teamName = viewModel.teamMigrationState.teamNameTextState.text.toString(), backToWireButtonClicked = true ) ) diff --git a/core/analytics/src/main/kotlin/com/wire/android/feature/analytics/model/AnalyticsEvent.kt b/core/analytics/src/main/kotlin/com/wire/android/feature/analytics/model/AnalyticsEvent.kt index b08eb4585b5..b5fe97b7946 100644 --- a/core/analytics/src/main/kotlin/com/wire/android/feature/analytics/model/AnalyticsEvent.kt +++ b/core/analytics/src/main/kotlin/com/wire/android/feature/analytics/model/AnalyticsEvent.kt @@ -29,17 +29,20 @@ import com.wire.android.feature.analytics.model.AnalyticsEventConstants.CLICKED_ import com.wire.android.feature.analytics.model.AnalyticsEventConstants.CLICKED_PERSONAL_MIGRATION_CTA_EVENT import com.wire.android.feature.analytics.model.AnalyticsEventConstants.CONTRIBUTED_LOCATION import com.wire.android.feature.analytics.model.AnalyticsEventConstants.MESSAGE_ACTION_KEY -import com.wire.android.feature.analytics.model.AnalyticsEventConstants.QR_CODE_SEGMENTATION_USER_TYPE_PERSONAL -import com.wire.android.feature.analytics.model.AnalyticsEventConstants.QR_CODE_SEGMENTATION_USER_TYPE_TEAM import com.wire.android.feature.analytics.model.AnalyticsEventConstants.MIGRATION_DOT_ACTIVE import com.wire.android.feature.analytics.model.AnalyticsEventConstants.MODAL_BACK_TO_WIRE_CLICKED +import com.wire.android.feature.analytics.model.AnalyticsEventConstants.MODAL_CONFIRMATION import com.wire.android.feature.analytics.model.AnalyticsEventConstants.MODAL_CONTINUE_CLICKED +import com.wire.android.feature.analytics.model.AnalyticsEventConstants.MODAL_DISCLAIMERS import com.wire.android.feature.analytics.model.AnalyticsEventConstants.MODAL_LEAVE_CLICKED import com.wire.android.feature.analytics.model.AnalyticsEventConstants.MODAL_OPEN_TEAM_MANAGEMENT_CLICKED import com.wire.android.feature.analytics.model.AnalyticsEventConstants.MODAL_TEAM_NAME import com.wire.android.feature.analytics.model.AnalyticsEventConstants.PERSONAL_TEAM_CREATION_FLOW_CANCELLED import com.wire.android.feature.analytics.model.AnalyticsEventConstants.PERSONAL_TEAM_CREATION_FLOW_COMPLETED import com.wire.android.feature.analytics.model.AnalyticsEventConstants.PERSONAL_TEAM_CREATION_FLOW_STARTED_EVENT +import com.wire.android.feature.analytics.model.AnalyticsEventConstants.PERSONAL_TEAM_CREATION_FLOW_STOPPED_EVENT +import com.wire.android.feature.analytics.model.AnalyticsEventConstants.QR_CODE_SEGMENTATION_USER_TYPE_PERSONAL +import com.wire.android.feature.analytics.model.AnalyticsEventConstants.QR_CODE_SEGMENTATION_USER_TYPE_TEAM import com.wire.android.feature.analytics.model.AnalyticsEventConstants.STEP_MODAL_CREATE_TEAM import com.wire.android.feature.analytics.model.AnalyticsEventConstants.USER_PROFILE_OPENED @@ -278,8 +281,29 @@ interface AnalyticsEvent { } } + data class PersonalTeamCreationFlowStopped( + val isOnTeamNameStep: Boolean?, + val isOnConfirmationStep: Boolean?, + val isOnDisclaimerStep: Boolean?, + ) : AnalyticsEvent { + override val key: String = PERSONAL_TEAM_CREATION_FLOW_STOPPED_EVENT + + override fun toSegmentation(): Map { + val segmentations = mutableMapOf() + isOnTeamNameStep?.let { + segmentations.put(MODAL_TEAM_NAME, it) + } + isOnConfirmationStep?.let { + segmentations.put(MODAL_CONFIRMATION, it) + } + isOnDisclaimerStep?.let { + segmentations.put(MODAL_DISCLAIMERS, it) + } + return segmentations + } + } + data class PersonalTeamCreationFlowCanceled( - val teamName: String?, val modalLeaveClicked: Boolean? = null, val modalContinueClicked: Boolean? = null ) : AnalyticsEvent { @@ -293,15 +317,11 @@ interface AnalyticsEvent { modalContinueClicked?.let { segmentations.put(MODAL_CONTINUE_CLICKED, it) } - teamName?.let { - segmentations.put(MODAL_TEAM_NAME, it) - } return segmentations } } data class PersonalTeamCreationFlowCompleted( - val teamName: String? = null, val modalOpenTeamManagementButtonClicked: Boolean? = null, val backToWireButtonClicked: Boolean? = null ) : AnalyticsEvent { @@ -309,9 +329,6 @@ interface AnalyticsEvent { override fun toSegmentation(): Map { val segmentations = mutableMapOf() - teamName?.let { - segmentations.put(MODAL_TEAM_NAME, it) - } modalOpenTeamManagementButtonClicked?.let { segmentations.put(MODAL_OPEN_TEAM_MANAGEMENT_CLICKED, it) } @@ -393,12 +410,15 @@ object AnalyticsEventConstants { */ const val CLICKED_PERSONAL_MIGRATION_CTA_EVENT = "ui.clicked-personal-migration-cta" const val PERSONAL_TEAM_CREATION_FLOW_STARTED_EVENT = "user.personal-team-creation-flow-started" + const val PERSONAL_TEAM_CREATION_FLOW_STOPPED_EVENT = "personal-team-creation-flow-stopped" const val PERSONAL_TEAM_CREATION_FLOW_CANCELLED = "user.personal-team-creation-flow-cancelled" const val PERSONAL_TEAM_CREATION_FLOW_COMPLETED = "user.personal-team-creation-flow-completed" const val MIGRATION_DOT_ACTIVE = "migration_dot_active" const val CLICKED_CREATE_TEAM = "clicked_create_team" const val CLICKED_DISMISS_CTA = "clicked_dismiss_cta" const val STEP_MODAL_CREATE_TEAM = "step_modalcreateteam" + const val MODAL_DISCLAIMERS = "modal_disclaimers" + const val MODAL_CONFIRMATION = "modal_confirmation" const val MODAL_TEAM_NAME = "modal_team-name" const val MODAL_CONTINUE_CLICKED = "modal_continue-clicked" const val MODAL_LEAVE_CLICKED = "modal_leave-clicked" diff --git a/kalium b/kalium index c9f091b3923..6518915d0ac 160000 --- a/kalium +++ b/kalium @@ -1 +1 @@ -Subproject commit c9f091b392378afa02a1ca0df92d41052b0439a1 +Subproject commit 6518915d0ac5ab78f9e34d7ab676c686e0c14d2f From 90282eadfd35a6917d6bd246efb35421b74c2aea Mon Sep 17 00:00:00 2001 From: Yamil Medina Date: Fri, 13 Dec 2024 18:37:48 -0300 Subject: [PATCH 2/2] fix: add signature used by mls client (WPB-15040) (#3740) --- .../wire/android/ui/settings/devices/DeviceDetailsScreen.kt | 6 ++++++ .../android/ui/settings/devices/DeviceDetailsViewModel.kt | 4 ++++ .../android/ui/settings/devices/model/DeviceDetailsState.kt | 3 ++- kalium | 2 +- 4 files changed, 13 insertions(+), 2 deletions(-) diff --git a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt index 422d9647472..7a8894fe512 100644 --- a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt +++ b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsScreen.kt @@ -204,6 +204,12 @@ fun DeviceDetailsContent( ) { state.device.mlsClientIdentity?.let { identity -> item { + FolderHeader( + name = stringResource(id = R.string.label_mls_signature, state.mlsCipherSuiteSignature.orEmpty()).uppercase(), + modifier = Modifier + .background(MaterialTheme.wireColorScheme.background) + .fillMaxWidth() + ) DeviceMLSSignatureItem(identity.thumbprint, screenState::copyMessage) HorizontalDivider(color = MaterialTheme.wireColorScheme.background) } diff --git a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt index f6d06b76b68..826154cc6b6 100644 --- a/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt +++ b/app/src/main/kotlin/com/wire/android/ui/settings/devices/DeviceDetailsViewModel.kt @@ -37,6 +37,7 @@ import com.wire.kalium.logic.CoreFailure import com.wire.kalium.logic.data.client.ClientType import com.wire.kalium.logic.data.client.DeleteClientParam import com.wire.kalium.logic.data.conversation.ClientId +import com.wire.kalium.logic.data.mlspublickeys.MLSPublicKeyType import com.wire.kalium.logic.data.user.UserId import com.wire.kalium.logic.feature.client.ClientFingerprintUseCase import com.wire.kalium.logic.feature.client.DeleteClientResult @@ -198,6 +199,9 @@ class DeviceDetailsViewModel @Inject constructor( isCurrentDevice = result.isCurrentClient, removeDeviceDialogState = RemoveDeviceDialogState.Hidden, canBeRemoved = !result.isCurrentClient && isSelfClient && result.client.type != ClientType.LegalHold, + mlsCipherSuiteSignature = MLSPublicKeyType.from( + result.client.mlsPublicKeys?.keys?.firstOrNull().orEmpty() + ).value.toString() ) } } diff --git a/app/src/main/kotlin/com/wire/android/ui/settings/devices/model/DeviceDetailsState.kt b/app/src/main/kotlin/com/wire/android/ui/settings/devices/model/DeviceDetailsState.kt index b68745619e9..96216d33068 100644 --- a/app/src/main/kotlin/com/wire/android/ui/settings/devices/model/DeviceDetailsState.kt +++ b/app/src/main/kotlin/com/wire/android/ui/settings/devices/model/DeviceDetailsState.kt @@ -37,5 +37,6 @@ data class DeviceDetailsState( val isE2EICertificateEnrollSuccess: Boolean = false, val isE2EICertificateEnrollError: Boolean = false, val isE2EIEnabled: Boolean = false, - val startGettingE2EICertificate: Boolean = false + val startGettingE2EICertificate: Boolean = false, + val mlsCipherSuiteSignature: String? = null, ) diff --git a/kalium b/kalium index 6518915d0ac..4b88fac85f1 160000 --- a/kalium +++ b/kalium @@ -1 +1 @@ -Subproject commit 6518915d0ac5ab78f9e34d7ab676c686e0c14d2f +Subproject commit 4b88fac85f12168b6c5dde9f2fd7853e257dad37