495 lines
16 KiB
TypeScript
495 lines
16 KiB
TypeScript
// Copyright 2020 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import React, { useEffect, useState } from 'react';
|
|
import type { ReactNode } from 'react';
|
|
|
|
import type {
|
|
ConversationType,
|
|
ShowConversationType,
|
|
} from '../../state/ducks/conversations';
|
|
import type { BadgeType } from '../../badges/types';
|
|
import type { HasStories } from '../../types/Stories';
|
|
import type { LocalizerType, ThemeType } from '../../types/Util';
|
|
import type { ViewUserStoriesActionCreatorType } from '../../state/ducks/stories';
|
|
import { StoryViewModeType } from '../../types/Stories';
|
|
import * as log from '../../logging/log';
|
|
import { Avatar, AvatarSize } from '../Avatar';
|
|
import { AvatarLightbox } from '../AvatarLightbox';
|
|
import { BadgeDialog } from '../BadgeDialog';
|
|
import { ConfirmationDialog } from '../ConfirmationDialog';
|
|
import { Modal } from '../Modal';
|
|
import { RemoveGroupMemberConfirmationDialog } from './RemoveGroupMemberConfirmationDialog';
|
|
import { missingCaseError } from '../../util/missingCaseError';
|
|
import { UserText } from '../UserText';
|
|
import { Button, ButtonIconType, ButtonVariant } from '../Button';
|
|
import { isInSystemContacts } from '../../util/isInSystemContacts';
|
|
import { InContactsIcon } from '../InContactsIcon';
|
|
import { canHaveNicknameAndNote } from '../../util/nicknames';
|
|
import { Tooltip, TooltipPlacement } from '../Tooltip';
|
|
import { offsetDistanceModifier } from '../../util/popperUtil';
|
|
import { getThemeByThemeType } from '../../util/theme';
|
|
|
|
export type PropsDataType = {
|
|
areWeASubscriber: boolean;
|
|
areWeAdmin: boolean;
|
|
badges: ReadonlyArray<BadgeType>;
|
|
contact?: ConversationType;
|
|
conversation?: ConversationType;
|
|
hasStories?: HasStories;
|
|
readonly i18n: LocalizerType;
|
|
isAdmin: boolean;
|
|
isMember: boolean;
|
|
theme: ThemeType;
|
|
hasActiveCall: boolean;
|
|
isInFullScreenCall: boolean;
|
|
};
|
|
|
|
type PropsActionType = {
|
|
blockConversation: (id: string) => void;
|
|
hideContactModal: () => void;
|
|
onOpenEditNicknameAndNoteModal: () => void;
|
|
onOutgoingAudioCallInConversation: (conversationId: string) => unknown;
|
|
onOutgoingVideoCallInConversation: (conversationId: string) => unknown;
|
|
removeMemberFromGroup: (conversationId: string, contactId: string) => void;
|
|
showConversation: ShowConversationType;
|
|
toggleAdmin: (conversationId: string, contactId: string) => void;
|
|
toggleAboutContactModal: (conversationId: string) => unknown;
|
|
togglePip: () => void;
|
|
toggleSafetyNumberModal: (conversationId: string) => unknown;
|
|
toggleAddUserToAnotherGroupModal: (conversationId: string) => void;
|
|
updateConversationModelSharedGroups: (conversationId: string) => void;
|
|
viewUserStories: ViewUserStoriesActionCreatorType;
|
|
};
|
|
|
|
export type PropsType = PropsDataType & PropsActionType;
|
|
|
|
enum ContactModalView {
|
|
Default,
|
|
ShowingAvatar,
|
|
ShowingBadges,
|
|
}
|
|
|
|
enum SubModalState {
|
|
None = 'None',
|
|
ToggleAdmin = 'ToggleAdmin',
|
|
MemberRemove = 'MemberRemove',
|
|
ConfirmingBlock = 'ConfirmingBlock',
|
|
}
|
|
|
|
export function ContactModal({
|
|
areWeAdmin,
|
|
areWeASubscriber,
|
|
badges,
|
|
blockConversation,
|
|
contact,
|
|
conversation,
|
|
hasActiveCall,
|
|
hasStories,
|
|
hideContactModal,
|
|
isInFullScreenCall,
|
|
i18n,
|
|
isAdmin,
|
|
isMember,
|
|
onOpenEditNicknameAndNoteModal,
|
|
onOutgoingAudioCallInConversation,
|
|
onOutgoingVideoCallInConversation,
|
|
removeMemberFromGroup,
|
|
showConversation,
|
|
theme,
|
|
toggleAboutContactModal,
|
|
toggleAddUserToAnotherGroupModal,
|
|
toggleAdmin,
|
|
togglePip,
|
|
toggleSafetyNumberModal,
|
|
updateConversationModelSharedGroups,
|
|
viewUserStories,
|
|
}: PropsType): JSX.Element {
|
|
if (!contact) {
|
|
throw new Error('Contact modal opened without a matching contact');
|
|
}
|
|
|
|
const [view, setView] = useState(ContactModalView.Default);
|
|
const [subModalState, setSubModalState] = useState<SubModalState>(
|
|
SubModalState.None
|
|
);
|
|
const modalTheme = getThemeByThemeType(theme);
|
|
|
|
useEffect(() => {
|
|
if (contact?.id) {
|
|
// Kick off the expensive hydration of the current sharedGroupNames
|
|
updateConversationModelSharedGroups(contact.id);
|
|
}
|
|
}, [contact?.id, updateConversationModelSharedGroups]);
|
|
|
|
const renderQuickActions = React.useCallback(
|
|
(conversationId: string) => {
|
|
const videoCallButton = (
|
|
<Button
|
|
icon={ButtonIconType.video}
|
|
variant={ButtonVariant.Details}
|
|
disabled={hasActiveCall}
|
|
onClick={() => {
|
|
hideContactModal();
|
|
onOutgoingVideoCallInConversation(conversationId);
|
|
}}
|
|
>
|
|
{i18n('icu:video')}
|
|
</Button>
|
|
);
|
|
const audioCallButton = (
|
|
<Button
|
|
icon={ButtonIconType.audio}
|
|
variant={ButtonVariant.Details}
|
|
disabled={hasActiveCall}
|
|
onClick={() => {
|
|
hideContactModal();
|
|
onOutgoingAudioCallInConversation(conversationId);
|
|
}}
|
|
>
|
|
{i18n('icu:audio')}
|
|
</Button>
|
|
);
|
|
|
|
return (
|
|
<div className="ContactModal__quick-actions">
|
|
<Button
|
|
icon={ButtonIconType.message}
|
|
variant={ButtonVariant.Details}
|
|
onClick={() => {
|
|
hideContactModal();
|
|
showConversation({
|
|
conversationId,
|
|
switchToAssociatedView: true,
|
|
});
|
|
if (isInFullScreenCall) {
|
|
togglePip();
|
|
}
|
|
}}
|
|
>
|
|
{i18n('icu:ConversationDetails__HeaderButton--Message')}
|
|
</Button>
|
|
{hasActiveCall ? (
|
|
<Tooltip
|
|
className="ContactModal__tooltip"
|
|
wrapperClassName="ContactModal__tooltip-wrapper"
|
|
content={i18n('icu:ContactModal--already-in-call')}
|
|
direction={TooltipPlacement.Top}
|
|
popperModifiers={[offsetDistanceModifier(5)]}
|
|
>
|
|
{videoCallButton}
|
|
</Tooltip>
|
|
) : (
|
|
videoCallButton
|
|
)}
|
|
{hasActiveCall ? (
|
|
<Tooltip
|
|
className="ContactModal__tooltip"
|
|
wrapperClassName="ContactModal__tooltip-wrapper"
|
|
content={i18n('icu:ContactModal--already-in-call')}
|
|
direction={TooltipPlacement.Top}
|
|
popperModifiers={[offsetDistanceModifier(5)]}
|
|
>
|
|
{audioCallButton}
|
|
</Tooltip>
|
|
) : (
|
|
audioCallButton
|
|
)}
|
|
</div>
|
|
);
|
|
},
|
|
[
|
|
hasActiveCall,
|
|
hideContactModal,
|
|
i18n,
|
|
isInFullScreenCall,
|
|
onOutgoingAudioCallInConversation,
|
|
onOutgoingVideoCallInConversation,
|
|
showConversation,
|
|
togglePip,
|
|
]
|
|
);
|
|
|
|
let modalNode: ReactNode;
|
|
switch (subModalState) {
|
|
case SubModalState.None:
|
|
modalNode = undefined;
|
|
break;
|
|
case SubModalState.ToggleAdmin:
|
|
if (!conversation?.id) {
|
|
log.warn('ContactModal: ToggleAdmin state - missing conversationId');
|
|
modalNode = undefined;
|
|
break;
|
|
}
|
|
|
|
modalNode = (
|
|
<ConfirmationDialog
|
|
dialogName="ContactModal.toggleAdmin"
|
|
actions={[
|
|
{
|
|
action: () => toggleAdmin(conversation.id, contact.id),
|
|
text: isAdmin
|
|
? i18n('icu:ContactModal--rm-admin')
|
|
: i18n('icu:ContactModal--make-admin'),
|
|
},
|
|
]}
|
|
i18n={i18n}
|
|
onClose={() => setSubModalState(SubModalState.None)}
|
|
>
|
|
{isAdmin
|
|
? i18n('icu:ContactModal--rm-admin-info', {
|
|
contact: contact.title,
|
|
})
|
|
: i18n('icu:ContactModal--make-admin-info', {
|
|
contact: contact.title,
|
|
})}
|
|
</ConfirmationDialog>
|
|
);
|
|
break;
|
|
case SubModalState.MemberRemove:
|
|
if (!contact || !conversation?.id) {
|
|
log.warn(
|
|
'ContactModal: MemberRemove state - missing contact or conversationId'
|
|
);
|
|
modalNode = undefined;
|
|
break;
|
|
}
|
|
|
|
modalNode = (
|
|
<RemoveGroupMemberConfirmationDialog
|
|
conversation={contact}
|
|
group={conversation}
|
|
i18n={i18n}
|
|
onClose={() => {
|
|
setSubModalState(SubModalState.None);
|
|
}}
|
|
onRemove={() => {
|
|
removeMemberFromGroup(conversation?.id, contact.id);
|
|
}}
|
|
/>
|
|
);
|
|
break;
|
|
case SubModalState.ConfirmingBlock:
|
|
modalNode = (
|
|
<ConfirmationDialog
|
|
dialogName="ContactModal.confirmBlock"
|
|
actions={[
|
|
{
|
|
text: i18n('icu:MessageRequests--block'),
|
|
action: () => blockConversation(contact.id),
|
|
style: 'affirmative',
|
|
},
|
|
]}
|
|
i18n={i18n}
|
|
onClose={() => setSubModalState(SubModalState.None)}
|
|
title={i18n('icu:MessageRequests--block-direct-confirm-title', {
|
|
title: contact.title,
|
|
})}
|
|
>
|
|
{i18n('icu:MessageRequests--block-direct-confirm-body')}
|
|
</ConfirmationDialog>
|
|
);
|
|
break;
|
|
default: {
|
|
const state: never = subModalState;
|
|
log.warn(`ContactModal: unexpected ${state}!`);
|
|
modalNode = undefined;
|
|
break;
|
|
}
|
|
}
|
|
|
|
switch (view) {
|
|
case ContactModalView.Default: {
|
|
const preferredBadge: undefined | BadgeType = badges[0];
|
|
return (
|
|
<Modal
|
|
modalName="ContactModal"
|
|
moduleClassName="ContactModal__modal"
|
|
hasXButton
|
|
i18n={i18n}
|
|
onClose={hideContactModal}
|
|
padded={false}
|
|
theme={modalTheme}
|
|
>
|
|
<div className="ContactModal">
|
|
<Avatar
|
|
acceptedMessageRequest={contact.acceptedMessageRequest}
|
|
avatarPath={contact.avatarPath}
|
|
badge={preferredBadge}
|
|
color={contact.color}
|
|
conversationType="direct"
|
|
i18n={i18n}
|
|
isMe={contact.isMe}
|
|
onClick={() => {
|
|
if (conversation && hasStories) {
|
|
viewUserStories({
|
|
conversationId: contact.id,
|
|
storyViewMode: StoryViewModeType.User,
|
|
});
|
|
hideContactModal();
|
|
} else {
|
|
setView(ContactModalView.ShowingAvatar);
|
|
}
|
|
}}
|
|
onClickBadge={() => setView(ContactModalView.ShowingBadges)}
|
|
profileName={contact.profileName}
|
|
sharedGroupNames={contact.sharedGroupNames}
|
|
size={AvatarSize.EIGHTY}
|
|
storyRing={hasStories}
|
|
theme={theme}
|
|
title={contact.title}
|
|
unblurredAvatarPath={contact.unblurredAvatarPath}
|
|
/>
|
|
<button
|
|
type="button"
|
|
className="ContactModal__name"
|
|
onClick={ev => {
|
|
ev.preventDefault();
|
|
toggleAboutContactModal(contact.id);
|
|
}}
|
|
>
|
|
<div className="ContactModal__name__text">
|
|
<UserText text={contact.title} />
|
|
{isInSystemContacts(contact) && (
|
|
<span>
|
|
{' '}
|
|
<InContactsIcon
|
|
className="ContactModal__name__contact-icon"
|
|
i18n={i18n}
|
|
/>
|
|
</span>
|
|
)}
|
|
</div>
|
|
<i className="ContactModal__name__chevron" />
|
|
</button>
|
|
{!contact.isMe && renderQuickActions(contact.id)}
|
|
<div className="ContactModal__divider" />
|
|
<div className="ContactModal__button-container">
|
|
{canHaveNicknameAndNote(contact) && (
|
|
<button
|
|
type="button"
|
|
className="ContactModal__button ContactModal__block"
|
|
onClick={onOpenEditNicknameAndNoteModal}
|
|
>
|
|
<div className="ContactModal__bubble-icon">
|
|
<div className="ContactModal__nickname__bubble-icon" />
|
|
</div>
|
|
<span>{i18n('icu:ContactModal--nickname')}</span>
|
|
</button>
|
|
)}
|
|
|
|
{!contact.isMe &&
|
|
(contact.isBlocked ? (
|
|
<div className="ContactModal__button ContactModal__block">
|
|
<div className="ContactModal__bubble-icon">
|
|
<div className="ContactModal__block__bubble-icon" />
|
|
</div>
|
|
<span>
|
|
{i18n('icu:AboutContactModal__blocked', {
|
|
name: contact.title,
|
|
})}
|
|
</span>
|
|
</div>
|
|
) : (
|
|
<button
|
|
type="button"
|
|
className="ContactModal__button ContactModal__block"
|
|
onClick={() =>
|
|
setSubModalState(SubModalState.ConfirmingBlock)
|
|
}
|
|
>
|
|
<div className="ContactModal__bubble-icon">
|
|
<div className="ContactModal__block__bubble-icon" />
|
|
</div>
|
|
<span>{i18n('icu:MessageRequests--block')}</span>
|
|
</button>
|
|
))}
|
|
{!contact.isMe && (
|
|
<button
|
|
type="button"
|
|
className="ContactModal__button ContactModal__safety-number"
|
|
onClick={() => {
|
|
hideContactModal();
|
|
toggleSafetyNumberModal(contact.id);
|
|
}}
|
|
>
|
|
<div className="ContactModal__bubble-icon">
|
|
<div className="ContactModal__safety-number__bubble-icon" />
|
|
</div>
|
|
<span>{i18n('icu:showSafetyNumber')}</span>
|
|
</button>
|
|
)}
|
|
{!contact.isMe && isMember && conversation?.id && (
|
|
<button
|
|
type="button"
|
|
className="ContactModal__button"
|
|
onClick={() => {
|
|
hideContactModal();
|
|
toggleAddUserToAnotherGroupModal(contact.id);
|
|
}}
|
|
>
|
|
<div className="ContactModal__bubble-icon">
|
|
<div className="ContactModal__add-to-another-group__bubble-icon" />
|
|
</div>
|
|
{i18n('icu:ContactModal--add-to-group')}
|
|
</button>
|
|
)}
|
|
{!contact.isMe && areWeAdmin && isMember && conversation?.id && (
|
|
<>
|
|
<button
|
|
type="button"
|
|
className="ContactModal__button ContactModal__make-admin"
|
|
onClick={() => setSubModalState(SubModalState.ToggleAdmin)}
|
|
>
|
|
<div className="ContactModal__bubble-icon">
|
|
<div className="ContactModal__make-admin__bubble-icon" />
|
|
</div>
|
|
{isAdmin ? (
|
|
<span>{i18n('icu:ContactModal--rm-admin')}</span>
|
|
) : (
|
|
<span>{i18n('icu:ContactModal--make-admin')}</span>
|
|
)}
|
|
</button>
|
|
<button
|
|
type="button"
|
|
className="ContactModal__button ContactModal__remove-from-group"
|
|
onClick={() => setSubModalState(SubModalState.MemberRemove)}
|
|
>
|
|
<div className="ContactModal__bubble-icon">
|
|
<div className="ContactModal__remove-from-group__bubble-icon" />
|
|
</div>
|
|
<span>{i18n('icu:ContactModal--remove-from-group')}</span>
|
|
</button>
|
|
</>
|
|
)}
|
|
</div>
|
|
{modalNode}
|
|
</div>
|
|
</Modal>
|
|
);
|
|
}
|
|
case ContactModalView.ShowingAvatar:
|
|
return (
|
|
<AvatarLightbox
|
|
avatarColor={contact.color}
|
|
avatarPath={contact.avatarPath}
|
|
conversationTitle={contact.title}
|
|
i18n={i18n}
|
|
onClose={() => setView(ContactModalView.Default)}
|
|
/>
|
|
);
|
|
case ContactModalView.ShowingBadges:
|
|
return (
|
|
<BadgeDialog
|
|
areWeASubscriber={areWeASubscriber}
|
|
badges={badges}
|
|
firstName={contact.firstName}
|
|
i18n={i18n}
|
|
onClose={() => setView(ContactModalView.Default)}
|
|
title={contact.title}
|
|
/>
|
|
);
|
|
default:
|
|
throw missingCaseError(view);
|
|
}
|
|
}
|