2024-02-06 02:13:13 +00:00
|
|
|
// Copyright 2024 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import React, { useCallback, useEffect } from 'react';
|
|
|
|
import type { ConversationType } from '../../state/ducks/conversations';
|
|
|
|
import type { LocalizerType } from '../../types/Util';
|
|
|
|
import { isInSystemContacts } from '../../util/isInSystemContacts';
|
2024-02-14 20:29:17 +00:00
|
|
|
import { shouldBlurAvatar } from '../../util/shouldBlurAvatar';
|
|
|
|
import { Avatar, AvatarBlur, AvatarSize } from '../Avatar';
|
2024-02-06 02:13:13 +00:00
|
|
|
import { Modal } from '../Modal';
|
|
|
|
import { UserText } from '../UserText';
|
|
|
|
import { SharedGroupNames } from '../SharedGroupNames';
|
|
|
|
import { About } from './About';
|
2024-05-15 21:48:02 +00:00
|
|
|
import { I18n } from '../I18n';
|
2024-06-20 18:39:40 +00:00
|
|
|
import { canHaveNicknameAndNote } from '../../util/nicknames';
|
2024-04-03 22:41:13 +00:00
|
|
|
import { Tooltip, TooltipPlacement } from '../Tooltip';
|
2024-03-26 19:48:33 +00:00
|
|
|
|
|
|
|
function muted(parts: Array<string | JSX.Element>) {
|
|
|
|
return (
|
|
|
|
<span className="AboutContactModal__TitleWithoutNickname">{parts}</span>
|
|
|
|
);
|
|
|
|
}
|
2024-02-06 02:13:13 +00:00
|
|
|
|
|
|
|
export type PropsType = Readonly<{
|
|
|
|
i18n: LocalizerType;
|
|
|
|
onClose: () => void;
|
2024-03-26 19:48:33 +00:00
|
|
|
onOpenNotePreviewModal: () => void;
|
2024-02-06 02:13:13 +00:00
|
|
|
conversation: ConversationType;
|
|
|
|
isSignalConnection: boolean;
|
|
|
|
toggleSignalConnectionsModal: () => void;
|
2024-02-14 20:29:17 +00:00
|
|
|
toggleSafetyNumberModal: (id: string) => void;
|
2024-02-06 02:13:13 +00:00
|
|
|
updateSharedGroups: (id: string) => void;
|
2024-02-14 20:29:17 +00:00
|
|
|
unblurAvatar: (conversationId: string) => void;
|
2024-02-06 02:13:13 +00:00
|
|
|
}>;
|
|
|
|
|
|
|
|
export function AboutContactModal({
|
|
|
|
i18n,
|
|
|
|
conversation,
|
|
|
|
isSignalConnection,
|
|
|
|
toggleSignalConnectionsModal,
|
2024-02-14 20:29:17 +00:00
|
|
|
toggleSafetyNumberModal,
|
2024-02-06 02:13:13 +00:00
|
|
|
updateSharedGroups,
|
2024-02-14 20:29:17 +00:00
|
|
|
unblurAvatar,
|
2024-02-06 02:13:13 +00:00
|
|
|
onClose,
|
2024-03-26 19:48:33 +00:00
|
|
|
onOpenNotePreviewModal,
|
2024-02-06 02:13:13 +00:00
|
|
|
}: PropsType): JSX.Element {
|
2024-02-16 22:16:13 +00:00
|
|
|
const { isMe } = conversation;
|
|
|
|
|
2024-02-06 02:13:13 +00:00
|
|
|
useEffect(() => {
|
|
|
|
// Kick off the expensive hydration of the current sharedGroupNames
|
|
|
|
updateSharedGroups(conversation.id);
|
|
|
|
}, [conversation.id, updateSharedGroups]);
|
|
|
|
|
2024-02-14 20:29:17 +00:00
|
|
|
const avatarBlur = shouldBlurAvatar(conversation)
|
|
|
|
? AvatarBlur.BlurPictureWithClickToView
|
|
|
|
: AvatarBlur.NoBlur;
|
|
|
|
|
|
|
|
const onAvatarClick = useCallback(() => {
|
|
|
|
if (avatarBlur === AvatarBlur.BlurPictureWithClickToView) {
|
|
|
|
unblurAvatar(conversation.id);
|
|
|
|
}
|
|
|
|
}, [avatarBlur, unblurAvatar, conversation.id]);
|
|
|
|
|
2024-02-06 02:13:13 +00:00
|
|
|
const onSignalConnectionClick = useCallback(
|
|
|
|
(ev: React.MouseEvent) => {
|
|
|
|
ev.preventDefault();
|
|
|
|
toggleSignalConnectionsModal();
|
|
|
|
},
|
|
|
|
[toggleSignalConnectionsModal]
|
|
|
|
);
|
|
|
|
|
2024-02-14 20:29:17 +00:00
|
|
|
const onVerifiedClick = useCallback(
|
|
|
|
(ev: React.MouseEvent) => {
|
|
|
|
ev.preventDefault();
|
|
|
|
toggleSafetyNumberModal(conversation.id);
|
|
|
|
},
|
|
|
|
[toggleSafetyNumberModal, conversation.id]
|
|
|
|
);
|
|
|
|
|
|
|
|
let statusRow: JSX.Element | undefined;
|
|
|
|
|
2024-02-16 22:16:13 +00:00
|
|
|
if (isMe) {
|
|
|
|
// No status for ourselves
|
|
|
|
} else if (conversation.isBlocked) {
|
2024-02-14 20:29:17 +00:00
|
|
|
statusRow = (
|
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<i className="AboutContactModal__row__icon AboutContactModal__row__icon--blocked" />
|
|
|
|
{i18n('icu:AboutContactModal__blocked', {
|
|
|
|
name: conversation.title,
|
|
|
|
})}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
} else if (!conversation.acceptedMessageRequest) {
|
|
|
|
statusRow = (
|
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<i className="AboutContactModal__row__icon AboutContactModal__row__icon--message-request" />
|
|
|
|
{i18n('icu:AboutContactModal__message-request')}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
} else if (!conversation.hasMessages && !conversation.profileSharing) {
|
|
|
|
statusRow = (
|
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<i className="AboutContactModal__row__icon AboutContactModal__row__icon--no-dms" />
|
|
|
|
{i18n('icu:AboutContactModal__no-dms', {
|
|
|
|
name: conversation.title,
|
|
|
|
})}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-02-06 02:13:13 +00:00
|
|
|
return (
|
|
|
|
<Modal
|
|
|
|
key="main"
|
|
|
|
modalName="AboutContactModal"
|
|
|
|
moduleClassName="AboutContactModal"
|
|
|
|
hasXButton
|
|
|
|
i18n={i18n}
|
|
|
|
onClose={onClose}
|
|
|
|
>
|
|
|
|
<div className="AboutContactModal__row AboutContactModal__row--centered">
|
|
|
|
<Avatar
|
|
|
|
acceptedMessageRequest={conversation.acceptedMessageRequest}
|
2024-07-11 19:44:09 +00:00
|
|
|
avatarUrl={conversation.avatarUrl}
|
2024-02-14 20:29:17 +00:00
|
|
|
blur={avatarBlur}
|
|
|
|
onClick={avatarBlur === AvatarBlur.NoBlur ? undefined : onAvatarClick}
|
2024-02-06 02:13:13 +00:00
|
|
|
badge={undefined}
|
|
|
|
color={conversation.color}
|
|
|
|
conversationType="direct"
|
|
|
|
i18n={i18n}
|
|
|
|
isMe={conversation.isMe}
|
|
|
|
profileName={conversation.profileName}
|
|
|
|
sharedGroupNames={[]}
|
|
|
|
size={AvatarSize.TWO_HUNDRED_SIXTEEN}
|
|
|
|
title={conversation.title}
|
2024-07-11 19:44:09 +00:00
|
|
|
unblurredAvatarUrl={conversation.unblurredAvatarUrl}
|
2024-02-06 02:13:13 +00:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<h3 className="AboutContactModal__title">
|
2024-02-16 22:16:13 +00:00
|
|
|
{isMe
|
|
|
|
? i18n('icu:AboutContactModal__title--myself')
|
|
|
|
: i18n('icu:AboutContactModal__title')}
|
2024-02-06 02:13:13 +00:00
|
|
|
</h3>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<i className="AboutContactModal__row__icon AboutContactModal__row__icon--profile" />
|
2024-03-26 19:48:33 +00:00
|
|
|
|
2024-03-28 22:28:50 +00:00
|
|
|
{canHaveNicknameAndNote(conversation) &&
|
2024-04-03 22:41:13 +00:00
|
|
|
(conversation.nicknameGivenName || conversation.nicknameFamilyName) &&
|
2024-03-26 19:48:33 +00:00
|
|
|
conversation.titleNoNickname ? (
|
|
|
|
<span>
|
2024-05-15 21:48:02 +00:00
|
|
|
<I18n
|
2024-03-26 19:48:33 +00:00
|
|
|
i18n={i18n}
|
|
|
|
id="icu:AboutContactModal__TitleAndTitleWithoutNickname"
|
|
|
|
components={{
|
|
|
|
nickname: <UserText text={conversation.title} />,
|
|
|
|
titleNoNickname: (
|
2024-04-03 22:41:13 +00:00
|
|
|
<Tooltip
|
|
|
|
className="AboutContactModal__TitleWithoutNickname__Tooltip"
|
|
|
|
direction={TooltipPlacement.Top}
|
|
|
|
content={
|
2024-05-15 21:48:02 +00:00
|
|
|
<I18n
|
2024-04-03 22:41:13 +00:00
|
|
|
i18n={i18n}
|
|
|
|
id="icu:AboutContactModal__TitleWithoutNickname__Tooltip"
|
|
|
|
components={{
|
|
|
|
title: (
|
|
|
|
<UserText text={conversation.titleNoNickname} />
|
|
|
|
),
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
delay={0}
|
|
|
|
>
|
|
|
|
<UserText text={conversation.titleNoNickname} />
|
|
|
|
</Tooltip>
|
2024-03-26 19:48:33 +00:00
|
|
|
),
|
|
|
|
muted,
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
) : (
|
|
|
|
<UserText text={conversation.title} />
|
|
|
|
)}
|
2024-02-06 02:13:13 +00:00
|
|
|
</div>
|
|
|
|
|
2024-02-16 22:16:13 +00:00
|
|
|
{!isMe && conversation.isVerified ? (
|
2024-02-14 20:29:17 +00:00
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<i className="AboutContactModal__row__icon AboutContactModal__row__icon--verified" />
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
className="AboutContactModal__verified"
|
|
|
|
onClick={onVerifiedClick}
|
|
|
|
>
|
|
|
|
{i18n('icu:AboutContactModal__verified')}
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
|
2024-02-16 22:16:13 +00:00
|
|
|
{!isMe && conversation.about ? (
|
2024-02-06 02:13:13 +00:00
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<i className="AboutContactModal__row__icon AboutContactModal__row__icon--about" />
|
|
|
|
<About
|
|
|
|
className="AboutContactModal__about"
|
|
|
|
text={conversation.about}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
|
2024-02-16 22:16:13 +00:00
|
|
|
{!isMe && isSignalConnection ? (
|
2024-02-06 02:13:13 +00:00
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<i className="AboutContactModal__row__icon AboutContactModal__row__icon--connections" />
|
|
|
|
<button
|
|
|
|
type="button"
|
2024-03-26 19:48:33 +00:00
|
|
|
className="AboutContactModal__button"
|
2024-02-06 02:13:13 +00:00
|
|
|
onClick={onSignalConnectionClick}
|
|
|
|
>
|
|
|
|
{i18n('icu:AboutContactModal__signal-connection')}
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
|
2024-02-16 22:16:13 +00:00
|
|
|
{!isMe && isInSystemContacts(conversation) ? (
|
2024-02-06 02:13:13 +00:00
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<i className="AboutContactModal__row__icon AboutContactModal__row__icon--person" />
|
|
|
|
{i18n('icu:AboutContactModal__system-contact', {
|
2024-03-19 00:46:47 +00:00
|
|
|
name:
|
|
|
|
conversation.systemGivenName ||
|
|
|
|
conversation.firstName ||
|
|
|
|
conversation.title,
|
2024-02-06 02:13:13 +00:00
|
|
|
})}
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
|
|
|
|
{conversation.phoneNumber ? (
|
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<i className="AboutContactModal__row__icon AboutContactModal__row__icon--phone" />
|
|
|
|
<UserText text={conversation.phoneNumber} />
|
|
|
|
</div>
|
|
|
|
) : null}
|
|
|
|
|
2024-02-16 22:16:13 +00:00
|
|
|
{!isMe && (
|
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<i className="AboutContactModal__row__icon AboutContactModal__row__icon--group" />
|
|
|
|
<div>
|
|
|
|
<SharedGroupNames
|
|
|
|
i18n={i18n}
|
|
|
|
sharedGroupNames={conversation.sharedGroupNames || []}
|
|
|
|
/>
|
|
|
|
</div>
|
2024-02-06 02:13:13 +00:00
|
|
|
</div>
|
2024-02-16 22:16:13 +00:00
|
|
|
)}
|
2024-02-14 20:29:17 +00:00
|
|
|
|
2024-06-20 18:39:40 +00:00
|
|
|
{conversation.note && (
|
2024-03-26 19:48:33 +00:00
|
|
|
<div className="AboutContactModal__row">
|
|
|
|
<i className="AboutContactModal__row__icon AboutContactModal__row__icon--note" />
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
className="AboutContactModal__button"
|
|
|
|
onClick={onOpenNotePreviewModal}
|
|
|
|
>
|
|
|
|
<div className="AboutContactModal__OneLineEllipsis">
|
|
|
|
<UserText text={conversation.note} />
|
|
|
|
</div>
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
|
2024-02-14 20:29:17 +00:00
|
|
|
{statusRow}
|
2024-02-06 02:13:13 +00:00
|
|
|
</Modal>
|
|
|
|
);
|
|
|
|
}
|