signal-desktop/ts/components/conversation/ContactSpoofingReviewDialog.tsx

354 lines
12 KiB
TypeScript
Raw Normal View History

2021-04-21 16:31:12 +00:00
// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import type { FunctionComponent, ReactChild, ReactNode } from 'react';
import React, { useState } from 'react';
2021-06-01 23:30:25 +00:00
import { concat, orderBy } from 'lodash';
2021-04-21 16:31:12 +00:00
2021-11-30 10:07:24 +00:00
import type { LocalizerType, ThemeType } from '../../types/Util';
import type { ConversationType } from '../../state/ducks/conversations';
2021-11-30 10:07:24 +00:00
import type { PreferredBadgeSelectorType } from '../../state/selectors/badges';
2021-04-21 16:31:12 +00:00
import {
MessageRequestActionsConfirmation,
MessageRequestState,
} from './MessageRequestActionsConfirmation';
2021-06-01 23:30:25 +00:00
import { ContactSpoofingType } from '../../util/contactSpoofing';
2021-04-21 16:31:12 +00:00
import { Modal } from '../Modal';
2021-06-01 23:30:25 +00:00
import { RemoveGroupMemberConfirmationDialog } from './RemoveGroupMemberConfirmationDialog';
2021-04-21 16:31:12 +00:00
import { ContactSpoofingReviewDialogPerson } from './ContactSpoofingReviewDialogPerson';
import { Button, ButtonVariant } from '../Button';
2021-06-01 23:30:25 +00:00
import { Intl } from '../Intl';
import { Emojify } from './Emojify';
2021-04-21 16:31:12 +00:00
import { assert } from '../../util/assert';
2021-06-01 23:30:25 +00:00
import { missingCaseError } from '../../util/missingCaseError';
import { isInSystemContacts } from '../../util/isInSystemContacts';
2021-04-21 16:31:12 +00:00
type PropsType = {
2021-11-30 10:07:24 +00:00
getPreferredBadge: PreferredBadgeSelectorType;
2021-04-21 16:31:12 +00:00
i18n: LocalizerType;
2021-06-01 23:30:25 +00:00
onBlock: (conversationId: string) => unknown;
onBlockAndReportSpam: (conversationId: string) => unknown;
2021-04-21 16:31:12 +00:00
onClose: () => void;
2021-06-01 23:30:25 +00:00
onDelete: (conversationId: string) => unknown;
2021-04-21 16:31:12 +00:00
onShowContactModal: (contactId: string) => unknown;
2021-06-01 23:30:25 +00:00
onUnblock: (conversationId: string) => unknown;
removeMember: (conversationId: string) => unknown;
2021-11-30 10:07:24 +00:00
theme: ThemeType;
2021-06-01 23:30:25 +00:00
} & (
| {
type: ContactSpoofingType.DirectConversationWithSameTitle;
possiblyUnsafeConversation: ConversationType;
safeConversation: ConversationType;
}
| {
type: ContactSpoofingType.MultipleGroupMembersWithSameTitle;
areWeAdmin: boolean;
collisionInfoByTitle: Record<
string,
Array<{
oldName?: string;
conversation: ConversationType;
}>
>;
}
);
2021-04-21 16:31:12 +00:00
2021-06-01 23:30:25 +00:00
enum ConfirmationStateType {
ConfirmingDelete,
ConfirmingBlock,
ConfirmingGroupRemoval,
}
2021-04-21 16:31:12 +00:00
2021-11-11 22:43:05 +00:00
export const ContactSpoofingReviewDialog: FunctionComponent<PropsType> =
props => {
const {
2021-11-30 10:07:24 +00:00
getPreferredBadge,
2021-11-11 22:43:05 +00:00
i18n,
onBlock,
onBlockAndReportSpam,
onClose,
onDelete,
onShowContactModal,
onUnblock,
removeMember,
2021-11-30 10:07:24 +00:00
theme,
2021-11-11 22:43:05 +00:00
} = props;
2021-04-21 16:31:12 +00:00
2021-11-11 22:43:05 +00:00
const [confirmationState, setConfirmationState] = useState<
| undefined
| {
type: ConfirmationStateType;
affectedConversation: ConversationType;
}
>();
2021-06-01 23:30:25 +00:00
2021-11-11 22:43:05 +00:00
if (confirmationState) {
const { affectedConversation, type } = confirmationState;
switch (type) {
case ConfirmationStateType.ConfirmingDelete:
case ConfirmationStateType.ConfirmingBlock:
return (
<MessageRequestActionsConfirmation
i18n={i18n}
onBlock={() => {
onBlock(affectedConversation.id);
}}
onBlockAndReportSpam={() => {
onBlockAndReportSpam(affectedConversation.id);
}}
onUnblock={() => {
onUnblock(affectedConversation.id);
}}
onDelete={() => {
onDelete(affectedConversation.id);
}}
title={affectedConversation.title}
conversationType="direct"
state={
type === ConfirmationStateType.ConfirmingDelete
? MessageRequestState.deleting
: MessageRequestState.blocking
2021-06-01 23:30:25 +00:00
}
2021-11-11 22:43:05 +00:00
onChangeState={messageRequestState => {
switch (messageRequestState) {
case MessageRequestState.blocking:
setConfirmationState({
type: ConfirmationStateType.ConfirmingBlock,
affectedConversation,
});
break;
case MessageRequestState.deleting:
setConfirmationState({
type: ConfirmationStateType.ConfirmingDelete,
affectedConversation,
});
break;
case MessageRequestState.unblocking:
assert(
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:
return (
<RemoveGroupMemberConfirmationDialog
conversation={affectedConversation}
i18n={i18n}
onClose={() => {
setConfirmationState(undefined);
}}
onRemove={() => {
removeMember(affectedConversation.id);
}}
/>
);
default:
throw missingCaseError(type);
}
2021-06-01 23:30:25 +00:00
}
2021-11-11 22:43:05 +00:00
let title: string;
let contents: ReactChild;
2021-06-01 23:30:25 +00:00
2021-11-11 22:43:05 +00:00
switch (props.type) {
case ContactSpoofingType.DirectConversationWithSameTitle: {
const { possiblyUnsafeConversation, safeConversation } = props;
assert(
possiblyUnsafeConversation.type === 'direct',
'<ContactSpoofingReviewDialog> expected a direct conversation for the "possibly unsafe" conversation'
);
assert(
safeConversation.type === 'direct',
'<ContactSpoofingReviewDialog> expected a direct conversation for the "safe" conversation'
);
2021-06-01 23:30:25 +00:00
2021-11-11 22:43:05 +00:00
title = i18n('ContactSpoofingReviewDialog__title');
contents = (
<>
<p>{i18n('ContactSpoofingReviewDialog__description')}</p>
<h2>
{i18n('ContactSpoofingReviewDialog__possibly-unsafe-title')}
</h2>
<ContactSpoofingReviewDialogPerson
conversation={possiblyUnsafeConversation}
2021-11-30 10:07:24 +00:00
getPreferredBadge={getPreferredBadge}
2021-11-11 22:43:05 +00:00
i18n={i18n}
2021-11-30 10:07:24 +00:00
theme={theme}
2021-11-11 22:43:05 +00:00
>
<div className="module-ContactSpoofingReviewDialog__buttons">
2021-06-01 23:30:25 +00:00
<Button
2021-11-11 22:43:05 +00:00
variant={ButtonVariant.SecondaryDestructive}
2021-06-01 23:30:25 +00:00
onClick={() => {
setConfirmationState({
2021-11-11 22:43:05 +00:00
type: ConfirmationStateType.ConfirmingDelete,
affectedConversation: possiblyUnsafeConversation,
2021-06-01 23:30:25 +00:00
});
}}
>
2021-11-11 22:43:05 +00:00
{i18n('MessageRequests--delete')}
2021-06-01 23:30:25 +00:00
</Button>
<Button
variant={ButtonVariant.SecondaryDestructive}
onClick={() => {
setConfirmationState({
type: ConfirmationStateType.ConfirmingBlock,
2021-11-11 22:43:05 +00:00
affectedConversation: possiblyUnsafeConversation,
2021-06-01 23:30:25 +00:00
});
}}
>
{i18n('MessageRequests--block')}
</Button>
2021-11-11 22:43:05 +00:00
</div>
</ContactSpoofingReviewDialogPerson>
<hr />
<h2>{i18n('ContactSpoofingReviewDialog__safe-title')}</h2>
<ContactSpoofingReviewDialogPerson
conversation={safeConversation}
2021-11-30 10:07:24 +00:00
getPreferredBadge={getPreferredBadge}
2021-11-11 22:43:05 +00:00
i18n={i18n}
onClick={() => {
onShowContactModal(safeConversation.id);
}}
2021-11-30 10:07:24 +00:00
theme={theme}
2021-11-11 22:43:05 +00:00
/>
</>
);
break;
}
case ContactSpoofingType.MultipleGroupMembersWithSameTitle: {
const { areWeAdmin, collisionInfoByTitle } = props;
2021-06-01 23:30:25 +00:00
2021-11-11 22:43:05 +00:00
const unsortedConversationInfos = concat(
// This empty array exists to appease Lodash's type definitions.
[],
...Object.values(collisionInfoByTitle)
);
const conversationInfos = orderBy(unsortedConversationInfos, [
// We normally use an `Intl.Collator` to sort by title. We do this instead, as
// we only really care about stability (not perfect ordering).
'title',
'id',
]);
2021-06-01 23:30:25 +00:00
2021-11-11 22:43:05 +00:00
title = i18n('ContactSpoofingReviewDialog__group__title');
contents = (
<>
<p>
{i18n('ContactSpoofingReviewDialog__group__description', [
conversationInfos.length.toString(),
])}
</p>
<h2>
{i18n('ContactSpoofingReviewDialog__group__members-header')}
</h2>
{conversationInfos.map((conversationInfo, index) => {
let button: ReactNode;
if (areWeAdmin) {
button = (
<Button
variant={ButtonVariant.SecondaryAffirmative}
onClick={() => {
setConfirmationState({
type: ConfirmationStateType.ConfirmingGroupRemoval,
affectedConversation: conversationInfo.conversation,
});
}}
>
{i18n('RemoveGroupMemberConfirmation__remove-button')}
</Button>
);
} else if (conversationInfo.conversation.isBlocked) {
button = (
<Button
variant={ButtonVariant.SecondaryAffirmative}
onClick={() => {
onUnblock(conversationInfo.conversation.id);
}}
>
{i18n('MessageRequests--unblock')}
</Button>
);
} else if (!isInSystemContacts(conversationInfo.conversation)) {
button = (
<Button
variant={ButtonVariant.SecondaryDestructive}
onClick={() => {
setConfirmationState({
type: ConfirmationStateType.ConfirmingBlock,
affectedConversation: conversationInfo.conversation,
});
}}
>
{i18n('MessageRequests--block')}
</Button>
);
}
const { oldName } = conversationInfo;
const newName =
conversationInfo.conversation.profileName ||
conversationInfo.conversation.title;
return (
<>
{index !== 0 && <hr />}
<ContactSpoofingReviewDialogPerson
key={conversationInfo.conversation.id}
conversation={conversationInfo.conversation}
2021-11-30 10:07:24 +00:00
getPreferredBadge={getPreferredBadge}
2021-11-11 22:43:05 +00:00
i18n={i18n}
2021-11-30 10:07:24 +00:00
theme={theme}
2021-11-11 22:43:05 +00:00
>
{Boolean(oldName) && oldName !== newName && (
<div className="module-ContactSpoofingReviewDialogPerson__info__property module-ContactSpoofingReviewDialogPerson__info__property--callout">
<Intl
i18n={i18n}
id="ContactSpoofingReviewDialog__group__name-change-info"
components={{
oldName: <Emojify text={oldName} />,
newName: <Emojify text={newName} />,
}}
/>
</div>
)}
{button && (
<div className="module-ContactSpoofingReviewDialog__buttons">
{button}
</div>
)}
</ContactSpoofingReviewDialogPerson>
</>
);
})}
</>
);
break;
}
default:
throw missingCaseError(props);
2021-06-01 23:30:25 +00:00
}
2021-04-21 16:31:12 +00:00
2021-11-11 22:43:05 +00:00
return (
<Modal
hasXButton
i18n={i18n}
moduleClassName="module-ContactSpoofingReviewDialog"
onClose={onClose}
title={title}
>
{contents}
</Modal>
);
};