// 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 { About } from './About'; 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 { SharedGroupNames } from '../SharedGroupNames'; import { missingCaseError } from '../../util/missingCaseError'; import { UserText } from '../UserText'; export type PropsDataType = { areWeASubscriber: boolean; areWeAdmin: boolean; badges: ReadonlyArray; contact?: ConversationType; conversation?: ConversationType; hasStories?: HasStories; readonly i18n: LocalizerType; isAdmin: boolean; isMember: boolean; theme: ThemeType; }; type PropsActionType = { hideContactModal: () => void; removeMemberFromGroup: (conversationId: string, contactId: string) => void; showConversation: ShowConversationType; toggleAdmin: (conversationId: string, contactId: string) => void; toggleAboutContactModal: (conversationId: string) => unknown; 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', } export function ContactModal({ areWeAdmin, areWeASubscriber, badges, contact, conversation, hasStories, hideContactModal, i18n, isAdmin, isMember, removeMemberFromGroup, showConversation, theme, toggleAboutContactModal, toggleAddUserToAnotherGroupModal, toggleAdmin, 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.None ); useEffect(() => { if (contact?.id) { // Kick off the expensive hydration of the current sharedGroupNames updateConversationModelSharedGroups(contact.id); } }, [contact?.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 = ( 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, })} ); break; case SubModalState.MemberRemove: if (!contact || !conversation?.id) { log.warn( 'ContactModal: MemberRemove state - missing contact or conversationId' ); modalNode = undefined; break; } modalNode = ( { setSubModalState(SubModalState.None); }} onRemove={() => { removeMemberFromGroup(conversation?.id, contact.id); }} /> ); 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 (
{ 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} />
{contact.phoneNumber && (
{contact.phoneNumber}
)} {!contact.isMe && (
)}
{!contact.isMe && ( )} {!contact.isMe && isMember && conversation?.id && ( )} {!contact.isMe && areWeAdmin && isMember && conversation?.id && ( <> )}
{modalNode}
); } case ContactModalView.ShowingAvatar: return ( setView(ContactModalView.Default)} /> ); case ContactModalView.ShowingBadges: return ( setView(ContactModalView.Default)} title={contact.title} /> ); default: throw missingCaseError(view); } }