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

290 lines
9.3 KiB
TypeScript
Raw Normal View History

// Copyright 2020-2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
2021-09-21 22:37:10 +00:00
import React, { useEffect, useState } from 'react';
import type { ReactNode } from 'react';
import * as log from '../../logging/log';
2021-11-02 23:01:13 +00:00
import { missingCaseError } from '../../util/missingCaseError';
2021-01-26 01:01:19 +00:00
import { About } from './About';
import { Avatar } from '../Avatar';
2021-08-06 00:17:05 +00:00
import { AvatarLightbox } from '../AvatarLightbox';
import type { ConversationType } from '../../state/ducks/conversations';
2021-09-21 22:37:10 +00:00
import { Modal } from '../Modal';
2021-11-11 16:23:00 +00:00
import type { LocalizerType, ThemeType } from '../../types/Util';
2021-11-02 23:01:13 +00:00
import { BadgeDialog } from '../BadgeDialog';
import type { BadgeType } from '../../badges/types';
2021-08-06 00:17:05 +00:00
import { SharedGroupNames } from '../SharedGroupNames';
2021-09-21 22:37:10 +00:00
import { ConfirmationDialog } from '../ConfirmationDialog';
import { RemoveGroupMemberConfirmationDialog } from './RemoveGroupMemberConfirmationDialog';
2021-09-21 22:37:10 +00:00
export type PropsDataType = {
areWeASubscriber: boolean;
areWeAdmin: boolean;
2021-11-02 23:01:13 +00:00
badges: ReadonlyArray<BadgeType>;
contact?: ConversationType;
conversation?: ConversationType;
readonly i18n: LocalizerType;
isAdmin: boolean;
isMember: boolean;
2021-11-11 16:23:00 +00:00
theme: ThemeType;
};
2021-09-21 22:37:10 +00:00
type PropsActionType = {
hideContactModal: () => void;
openConversationInternal: (
options: Readonly<{
conversationId: string;
messageId?: string;
switchToAssociatedView?: boolean;
}>
) => void;
removeMemberFromGroup: (conversationId: string, contactId: string) => void;
toggleAdmin: (conversationId: string, contactId: string) => void;
toggleSafetyNumberModal: (conversationId: string) => unknown;
2021-09-21 22:37:10 +00:00
updateConversationModelSharedGroups: (conversationId: string) => void;
};
export type PropsType = PropsDataType & PropsActionType;
2021-11-02 23:01:13 +00:00
enum ContactModalView {
Default,
ShowingAvatar,
ShowingBadges,
}
enum SubModalState {
None = 'None',
ToggleAdmin = 'ToggleAdmin',
MemberRemove = 'MemberRemove',
}
export const ContactModal = ({
areWeASubscriber,
areWeAdmin,
2021-11-02 23:01:13 +00:00
badges,
contact,
conversation,
2021-09-21 22:37:10 +00:00
hideContactModal,
i18n,
isAdmin,
isMember,
2021-09-21 22:37:10 +00:00
openConversationInternal,
removeMemberFromGroup,
2021-11-11 16:23:00 +00:00
theme,
toggleAdmin,
toggleSafetyNumberModal,
2021-09-21 22:37:10 +00:00
updateConversationModelSharedGroups,
}: PropsType): JSX.Element => {
if (!contact) {
throw new Error('Contact modal opened without a matching contact');
}
2021-11-02 23:01:13 +00:00
const [view, setView] = useState(ContactModalView.Default);
const [subModalState, setSubModalState] = useState<SubModalState>(
SubModalState.None
);
2021-08-06 00:17:05 +00:00
useEffect(() => {
if (conversation?.id) {
2021-09-21 22:37:10 +00:00
// Kick off the expensive hydration of the current sharedGroupNames
updateConversationModelSharedGroups(conversation.id);
}
}, [conversation?.id, updateConversationModelSharedGroups]);
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
actions={[
{
action: () => toggleAdmin(conversation.id, contact.id),
text: isAdmin
? i18n('ContactModal--rm-admin')
: i18n('ContactModal--make-admin'),
},
]}
i18n={i18n}
onClose={() => setSubModalState(SubModalState.None)}
>
{isAdmin
? i18n('ContactModal--rm-admin-info', [contact.title])
: i18n('ContactModal--make-admin-info', [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;
default: {
const state: never = subModalState;
log.warn(`ContactModal: unexpected ${state}!`);
modalNode = undefined;
break;
}
}
2021-11-02 23:01:13 +00:00
switch (view) {
case ContactModalView.Default: {
const preferredBadge: undefined | BadgeType = badges[0];
2021-09-21 22:37:10 +00:00
2021-11-02 23:01:13 +00:00
return (
<Modal
moduleClassName="ContactModal__modal"
hasXButton
2021-08-06 00:17:05 +00:00
i18n={i18n}
2021-11-02 23:01:13 +00:00
onClose={hideContactModal}
>
<div className="ContactModal">
<Avatar
acceptedMessageRequest={contact.acceptedMessageRequest}
avatarPath={contact.avatarPath}
badge={preferredBadge}
color={contact.color}
conversationType="direct"
2021-09-21 22:37:10 +00:00
i18n={i18n}
2021-11-02 23:01:13 +00:00
isMe={contact.isMe}
name={contact.name}
profileName={contact.profileName}
sharedGroupNames={contact.sharedGroupNames}
size={96}
2021-11-11 16:23:00 +00:00
theme={theme}
2021-11-02 23:01:13 +00:00
title={contact.title}
unblurredAvatarPath={contact.unblurredAvatarPath}
2021-11-18 20:01:53 +00:00
onClick={() => setView(ContactModalView.ShowingAvatar)}
onClickBadge={() => setView(ContactModalView.ShowingBadges)}
2021-11-02 23:01:13 +00:00
/>
<div className="ContactModal__name">{contact.title}</div>
<div className="module-about__container">
<About text={contact.about} />
</div>
{contact.phoneNumber && (
<div className="ContactModal__info">{contact.phoneNumber}</div>
)}
{!contact.isMe && (
<div className="ContactModal__info">
<SharedGroupNames
i18n={i18n}
sharedGroupNames={contact.sharedGroupNames || []}
/>
</div>
2021-11-02 23:01:13 +00:00
)}
<div className="ContactModal__button-container">
<button
type="button"
2021-11-02 23:01:13 +00:00
className="ContactModal__button ContactModal__send-message"
onClick={() => {
hideContactModal();
openConversationInternal({ conversationId: contact.id });
}}
>
2021-09-21 22:37:10 +00:00
<div className="ContactModal__bubble-icon">
2021-11-02 23:01:13 +00:00
<div className="ContactModal__send-message__bubble-icon" />
</div>
2021-11-02 23:01:13 +00:00
<span>{i18n('ContactModal--message')}</span>
</button>
2021-11-02 23:01:13 +00:00
{!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('showSafetyNumber')}</span>
</button>
)}
{!contact.isMe && areWeAdmin && isMember && conversation?.id && (
2021-11-02 23:01:13 +00:00
<>
<button
type="button"
className="ContactModal__button ContactModal__make-admin"
onClick={() => setSubModalState(SubModalState.ToggleAdmin)}
2021-11-02 23:01:13 +00:00
>
<div className="ContactModal__bubble-icon">
<div className="ContactModal__make-admin__bubble-icon" />
</div>
{isAdmin ? (
<span>{i18n('ContactModal--rm-admin')}</span>
) : (
<span>{i18n('ContactModal--make-admin')}</span>
)}
</button>
<button
type="button"
className="ContactModal__button ContactModal__remove-from-group"
onClick={() => setSubModalState(SubModalState.MemberRemove)}
2021-11-02 23:01:13 +00:00
>
<div className="ContactModal__bubble-icon">
<div className="ContactModal__remove-from-group__bubble-icon" />
</div>
<span>{i18n('ContactModal--remove-from-group')}</span>
</button>
</>
)}
</div>
{modalNode}
2021-11-02 23:01:13 +00:00
</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}
2021-11-02 23:01:13 +00:00
badges={badges}
firstName={contact.firstName}
i18n={i18n}
onClose={() => setView(ContactModalView.Default)}
title={contact.title}
/>
);
default:
throw missingCaseError(view);
}
};