// Copyright 2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { ReactChild, ReactNode } from 'react'; import React, { useState } from 'react'; import type { LocalizerType, ThemeType } from '../../types/Util'; import type { ConversationType } from '../../state/ducks/conversations'; import type { PreferredBadgeSelectorType } from '../../state/selectors/badges'; import { MessageRequestActionsConfirmation, MessageRequestState, } from './MessageRequestActionsConfirmation'; import { ContactSpoofingType } from '../../util/contactSpoofing'; import { Modal } from '../Modal'; import { RemoveGroupMemberConfirmationDialog } from './RemoveGroupMemberConfirmationDialog'; import { ContactSpoofingReviewDialogPerson } from './ContactSpoofingReviewDialogPerson'; import { Button, ButtonVariant } from '../Button'; import { assertDev } from '../../util/assert'; import { missingCaseError } from '../../util/missingCaseError'; import { isInSystemContacts } from '../../util/isInSystemContacts'; export type ReviewPropsType = Readonly< | { type: ContactSpoofingType.DirectConversationWithSameTitle; possiblyUnsafe: { conversation: ConversationType; isSignalConnection: boolean; }; safe: { conversation: ConversationType; isSignalConnection: boolean; }; } | { type: ContactSpoofingType.MultipleGroupMembersWithSameTitle; group: ConversationType; collisionInfoByTitle: Record< string, Array<{ oldName?: string; isSignalConnection: boolean; conversation: ConversationType; }> >; } >; export type PropsType = { conversationId: string; acceptConversation: (conversationId: string) => unknown; blockAndReportSpam: (conversationId: string) => unknown; blockConversation: (conversationId: string) => unknown; deleteConversation: (conversationId: string) => unknown; toggleSignalConnectionsModal: () => void; getPreferredBadge: PreferredBadgeSelectorType; i18n: LocalizerType; onClose: () => void; showContactModal: (contactId: string, conversationId?: string) => unknown; removeMember: ( conversationId: string, memberConversationId: string ) => unknown; theme: ThemeType; } & ReviewPropsType; enum ConfirmationStateType { ConfirmingDelete, ConfirmingBlock, ConfirmingGroupRemoval, } export function ContactSpoofingReviewDialog(props: PropsType): JSX.Element { const { acceptConversation, blockAndReportSpam, blockConversation, conversationId, deleteConversation, toggleSignalConnectionsModal, getPreferredBadge, i18n, onClose, showContactModal, removeMember, theme, } = props; const [confirmationState, setConfirmationState] = useState< | undefined | { type: ConfirmationStateType.ConfirmingGroupRemoval; affectedConversation: ConversationType; group: ConversationType; } | { type: | ConfirmationStateType.ConfirmingDelete | ConfirmationStateType.ConfirmingBlock; affectedConversation: ConversationType; } >(); if (confirmationState) { const { type, affectedConversation } = confirmationState; switch (type) { case ConfirmationStateType.ConfirmingDelete: case ConfirmationStateType.ConfirmingBlock: return ( { switch (messageRequestState) { case MessageRequestState.blocking: setConfirmationState({ type: ConfirmationStateType.ConfirmingBlock, affectedConversation, }); break; case MessageRequestState.deleting: setConfirmationState({ type: ConfirmationStateType.ConfirmingDelete, affectedConversation, }); break; case MessageRequestState.unblocking: assertDev( false, 'Got unexpected MessageRequestState.unblocking state. Clearing confiration state' ); setConfirmationState(undefined); break; case MessageRequestState.default: setConfirmationState(undefined); break; default: throw missingCaseError(messageRequestState); } }} /> ); case ConfirmationStateType.ConfirmingGroupRemoval: { const { group } = confirmationState; return ( { setConfirmationState(undefined); }} onRemove={() => { removeMember(conversationId, affectedConversation.id); }} /> ); } default: throw missingCaseError(type); } } let title: string; let contents: ReactChild; switch (props.type) { case ContactSpoofingType.DirectConversationWithSameTitle: { const { possiblyUnsafe, safe } = props; assertDev( possiblyUnsafe.conversation.type === 'direct', ' expected a direct conversation for the "possibly unsafe" conversation' ); assertDev( safe.conversation.type === 'direct', ' expected a direct conversation for the "safe" conversation' ); title = i18n('icu:ContactSpoofingReviewDialog__title'); contents = ( <>

{i18n('icu:ContactSpoofingReviewDialog__description')}

{i18n('icu:ContactSpoofingReviewDialog__possibly-unsafe-title')}


{i18n('icu:ContactSpoofingReviewDialog__safe-title')}

{ showContactModal(safe.conversation.id); }} theme={theme} isSignalConnection={safe.isSignalConnection} oldName={undefined} /> ); break; } case ContactSpoofingType.MultipleGroupMembersWithSameTitle: { const { group, collisionInfoByTitle } = props; const sharedTitles = Object.keys(collisionInfoByTitle); const numSharedTitles = sharedTitles.length; const totalConversations = Object.values(collisionInfoByTitle).reduce( (sum, conversationInfos) => sum + conversationInfos.length, 0 ); title = i18n('icu:ContactSpoofingReviewDialog__group__title'); contents = ( <>

{numSharedTitles > 1 ? i18n( 'icu:ContactSpoofingReviewDialog__group__multiple-conflicts__description', { count: numSharedTitles, } ) : i18n('icu:ContactSpoofingReviewDialog__group__description', { count: totalConversations, })}

{Object.values(collisionInfoByTitle) .map((conversationInfos, titleIdx) => conversationInfos.map((conversationInfo, conversationIdx) => { let button: ReactNode; if (group.areWeAdmin) { button = ( ); } else if (conversationInfo.conversation.isBlocked) { button = ( ); } else if (!isInSystemContacts(conversationInfo.conversation)) { button = ( ); } const { oldName, isSignalConnection } = conversationInfo; return ( <> {button && (
{button}
)}
{titleIdx < sharedTitles.length - 1 || conversationIdx < conversationInfos.length - 1 ? (
) : null} ); }) ) .flat()} ); break; } default: throw missingCaseError(props); } return ( {contents} ); }