// Copyright 2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React, { memo, useCallback } from 'react'; import { useSelector } from 'react-redux'; import { mapValues } from 'lodash'; import type { StateType } from '../reducer'; import { ContactSpoofingReviewDialog } from '../../components/conversation/ContactSpoofingReviewDialog'; import { useConversationsActions } from '../ducks/conversations'; import { getConversationSelector, getConversationByServiceIdSelector, getSafeConversationWithSameTitle, } from '../selectors/conversations'; import { getOwn } from '../../util/getOwn'; import { assertDev } from '../../util/assert'; import { ContactSpoofingType } from '../../util/contactSpoofing'; import { getGroupMemberships } from '../../util/getGroupMemberships'; import { isSignalConnection } from '../../util/getSignalConnections'; import { getCollisionsFromMemberships, invertIdsByTitle, } from '../../util/groupMemberNameCollisions'; import { useGlobalModalActions } from '../ducks/globalModals'; import { getPreferredBadgeSelector } from '../selectors/badges'; import { getIntl, getTheme } from '../selectors/user'; export type PropsType = Readonly<{ conversationId: string; onClose: () => void; }>; export const SmartContactSpoofingReviewDialog = memo( function SmartContactSpoofingReviewDialog(props: PropsType) { const { conversationId } = props; const getConversation = useSelector(getConversationSelector); const { acceptConversation, reportSpam, blockAndReportSpam, blockConversation, deleteConversation, removeMember, updateSharedGroups, } = useConversationsActions(); const { showContactModal, toggleSignalConnectionsModal } = useGlobalModalActions(); const getPreferredBadge = useSelector(getPreferredBadgeSelector); const i18n = useSelector(getIntl); const theme = useSelector(getTheme); const getConversationByServiceId = useSelector( getConversationByServiceIdSelector ); const conversation = getConversation(conversationId); // Just binding the options argument const safeConversationSelector = useCallback( (state: StateType) => { return getSafeConversationWithSameTitle(state, { possiblyUnsafeConversation: conversation, }); }, [conversation] ); const safeConvo = useSelector(safeConversationSelector); const sharedProps = { ...props, acceptConversation, reportSpam, blockAndReportSpam, blockConversation, deleteConversation, getPreferredBadge, i18n, removeMember, updateSharedGroups, showContactModal, toggleSignalConnectionsModal, theme, }; if (conversation.type === 'group') { const { memberships } = getGroupMemberships( conversation, getConversationByServiceId ); const groupNameCollisions = getCollisionsFromMemberships(memberships); const previouslyAcknowledgedTitlesById = invertIdsByTitle( conversation.acknowledgedGroupNameCollisions ); const collisionInfoByTitle = mapValues(groupNameCollisions, collisions => collisions.map(collision => ({ conversation: collision, isSignalConnection: isSignalConnection(collision), oldName: getOwn(previouslyAcknowledgedTitlesById, collision.id), })) ); return ( ); } const possiblyUnsafeConvo = conversation; assertDev( possiblyUnsafeConvo.type === 'direct', 'DirectConversationWithSameTitle: expects possibly unsafe direct ' + 'conversation' ); if (!safeConvo) { return null; } const possiblyUnsafe = { conversation: possiblyUnsafeConvo, isSignalConnection: isSignalConnection(possiblyUnsafeConvo), }; const safe = { conversation: safeConvo, isSignalConnection: isSignalConnection(safeConvo), }; return ( ); } );