// Copyright 2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { ReactNode } from 'react'; import React, { useEffect, useState, useCallback } from 'react'; import classNames from 'classnames'; import { Button, ButtonIconType, ButtonVariant } from '../../Button'; import { Tooltip } from '../../Tooltip'; import type { ConversationType, PushPanelForConversationActionType, ShowConversationType, } from '../../../state/ducks/conversations'; import type { PreferredBadgeSelectorType } from '../../../state/selectors/badges'; import type { SmartChooseGroupMembersModalPropsType } from '../../../state/smart/ChooseGroupMembersModal'; import type { SmartConfirmAdditionsModalPropsType } from '../../../state/smart/ConfirmAdditionsModal'; import { assertDev } from '../../../util/assert'; import { getMutedUntilText } from '../../../util/getMutedUntilText'; import type { LocalizerType, ThemeType } from '../../../types/Util'; import type { BadgeType } from '../../../badges/types'; import { missingCaseError } from '../../../util/missingCaseError'; import { DurationInSeconds } from '../../../util/durations'; import { DisappearingTimerSelect } from '../../DisappearingTimerSelect'; import { PanelRow } from './PanelRow'; import { PanelSection } from './PanelSection'; import { AddGroupMembersModal } from './AddGroupMembersModal'; import { ConversationDetailsActions } from './ConversationDetailsActions'; import { ConversationDetailsHeader } from './ConversationDetailsHeader'; import { ConversationDetailsIcon, IconType } from './ConversationDetailsIcon'; import type { Props as ConversationDetailsMediaListPropsType } from './ConversationDetailsMediaList'; import { ConversationDetailsMediaList } from './ConversationDetailsMediaList'; import type { GroupV2Membership } from './ConversationDetailsMembershipList'; import { ConversationDetailsMembershipList } from './ConversationDetailsMembershipList'; import type { GroupV2PendingMembership, GroupV2RequestingMembership, } from './PendingInvites'; import { EditConversationAttributesModal } from './EditConversationAttributesModal'; import { RequestState } from './util'; import { getCustomColorStyle } from '../../../util/getCustomColorStyle'; import { ConfirmationDialog } from '../../ConfirmationDialog'; import { ConversationNotificationsModal } from './ConversationNotificationsModal'; import type { AvatarDataType, DeleteAvatarFromDiskActionType, ReplaceAvatarActionType, SaveAvatarToDiskActionType, } from '../../../types/Avatar'; import { isConversationMuted } from '../../../util/isConversationMuted'; import { ConversationDetailsGroups } from './ConversationDetailsGroups'; import { PanelType } from '../../../types/Panels'; import type { CallStatus } from '../../../types/CallDisposition'; import { CallType, type CallHistoryGroup, CallDirection, DirectCallStatus, GroupCallStatus, } from '../../../types/CallDisposition'; import { formatDate, formatTime } from '../../../util/timestamp'; import { NavTab } from '../../../state/ducks/nav'; import { ContextMenu } from '../../ContextMenu'; import { canHaveNicknameAndNote } from '../../../util/nicknames'; function describeCallHistory( i18n: LocalizerType, type: CallType, direction: CallDirection, status: CallStatus ): string { if (type === CallType.Adhoc) { return i18n('icu:CallHistory__Description--Adhoc'); } if (status === DirectCallStatus.Missed || status === GroupCallStatus.Missed) { if (direction === CallDirection.Incoming) { return i18n('icu:CallHistory__Description--Missed', { type }); } return i18n('icu:CallHistory__Description--Unanswered', { type }); } if ( status === DirectCallStatus.Declined || status === GroupCallStatus.Declined ) { return i18n('icu:CallHistory__Description--Declined', { type }); } return i18n('icu:CallHistory__Description--Default', { type, direction }); } enum ModalState { AddingGroupMembers, ConfirmDeleteNicknameAndNote, EditingGroupDescription, EditingGroupTitle, MuteNotifications, NothingOpen, UnmuteNotifications, } export type StateProps = { areWeASubscriber: boolean; badges?: ReadonlyArray; callHistoryGroup?: CallHistoryGroup | null; canEditGroupInfo: boolean; canAddNewMembers: boolean; conversation?: ConversationType; hasGroupLink: boolean; getPreferredBadge: PreferredBadgeSelectorType; hasActiveCall: boolean; i18n: LocalizerType; isAdmin: boolean; isGroup: boolean; groupsInCommon: ReadonlyArray; maxGroupSize: number; maxRecommendedGroupSize: number; memberships: ReadonlyArray; pendingApprovalMemberships: ReadonlyArray; pendingMemberships: ReadonlyArray; selectedNavTab: NavTab; theme: ThemeType; userAvatarData: ReadonlyArray; renderChooseGroupMembersModal: ( props: SmartChooseGroupMembersModalPropsType ) => JSX.Element; renderConfirmAdditionsModal: ( props: SmartConfirmAdditionsModalPropsType ) => JSX.Element; }; type ActionProps = { acceptConversation: (id: string) => void; addMembersToGroup: ( conversationId: string, conversationIds: ReadonlyArray, opts: { onSuccess?: () => unknown; onFailure?: () => unknown; } ) => unknown; blockConversation: (id: string) => void; deleteAvatarFromDisk: DeleteAvatarFromDiskActionType; getProfilesForConversation: (id: string) => unknown; leaveGroup: (conversationId: string) => void; loadRecentMediaItems: (id: string, limit: number) => void; onDeleteNicknameAndNote: () => void; onOpenEditNicknameAndNoteModal: () => void; onOutgoingAudioCallInConversation: (conversationId: string) => unknown; onOutgoingVideoCallInConversation: (conversationId: string) => unknown; pushPanelForConversation: PushPanelForConversationActionType; replaceAvatar: ReplaceAvatarActionType; saveAvatarToDisk: SaveAvatarToDiskActionType; searchInConversation: (id: string) => unknown; setDisappearingMessages: (id: string, seconds: DurationInSeconds) => void; setMuteExpiration: (id: string, muteExpiresAt: undefined | number) => unknown; showContactModal: (contactId: string, conversationId?: string) => void; showConversation: ShowConversationType; toggleAboutContactModal: (contactId: string) => void; toggleAddUserToAnotherGroupModal: (contactId?: string) => void; toggleSafetyNumberModal: (conversationId: string) => unknown; updateGroupAttributes: ( conversationId: string, _: Readonly<{ avatar?: undefined | Uint8Array; description?: string; title?: string; }>, opts: { onSuccess?: () => unknown; onFailure?: () => unknown; } ) => unknown; } & Pick; export type Props = StateProps & ActionProps; export function getCannotLeaveBecauseYouAreLastAdmin( memberships: ReadonlyArray, isAdmin: boolean ): boolean { const otherMemberships = memberships.filter(({ member }) => !member.isMe); const isJustMe = otherMemberships.length === 0; const isAnyoneElseAnAdmin = otherMemberships.some( membership => membership.isAdmin ); const cannotLeaveBecauseYouAreLastAdmin = isAdmin && !isJustMe && !isAnyoneElseAnAdmin; return cannotLeaveBecauseYouAreLastAdmin; } export function ConversationDetails({ acceptConversation, addMembersToGroup, areWeASubscriber, badges, blockConversation, callHistoryGroup, canEditGroupInfo, canAddNewMembers, conversation, deleteAvatarFromDisk, hasGroupLink, getPreferredBadge, getProfilesForConversation, groupsInCommon, hasActiveCall, i18n, isAdmin, isGroup, leaveGroup, loadRecentMediaItems, memberships, maxGroupSize, maxRecommendedGroupSize, onDeleteNicknameAndNote, onOpenEditNicknameAndNoteModal, onOutgoingAudioCallInConversation, onOutgoingVideoCallInConversation, pendingApprovalMemberships, pendingMemberships, pushPanelForConversation, renderChooseGroupMembersModal, renderConfirmAdditionsModal, replaceAvatar, saveAvatarToDisk, searchInConversation, selectedNavTab, setDisappearingMessages, setMuteExpiration, showContactModal, showConversation, showLightboxWithMedia, theme, toggleAboutContactModal, toggleSafetyNumberModal, toggleAddUserToAnotherGroupModal, updateGroupAttributes, userAvatarData, }: Props): JSX.Element { const [modalState, setModalState] = useState( ModalState.NothingOpen ); const [editGroupAttributesRequestState, setEditGroupAttributesRequestState] = useState(RequestState.Inactive); const [addGroupMembersRequestState, setAddGroupMembersRequestState] = useState(RequestState.Inactive); if (conversation === undefined) { throw new Error('ConversationDetails rendered without a conversation'); } useEffect(() => { getProfilesForConversation(conversation.id); }, [conversation.id, getProfilesForConversation]); const invitesCount = pendingMemberships.length + pendingApprovalMemberships.length; const cannotLeaveBecauseYouAreLastAdmin = getCannotLeaveBecauseYouAreLastAdmin(memberships, isAdmin); const onCloseModal = useCallback(() => { setModalState(ModalState.NothingOpen); setEditGroupAttributesRequestState(RequestState.Inactive); }, []); let modalNode: ReactNode; switch (modalState) { case ModalState.NothingOpen: modalNode = undefined; break; case ModalState.EditingGroupDescription: case ModalState.EditingGroupTitle: modalNode = ( ) => { setEditGroupAttributesRequestState(RequestState.Active); updateGroupAttributes(conversation.id, options, { onSuccess: () => { setModalState(ModalState.NothingOpen); setEditGroupAttributesRequestState(RequestState.Inactive); }, onFailure: () => { setEditGroupAttributesRequestState( RequestState.InactiveWithError ); }, }); }} onClose={onCloseModal} requestState={editGroupAttributesRequestState} title={conversation.title} deleteAvatarFromDisk={deleteAvatarFromDisk} replaceAvatar={replaceAvatar} saveAvatarToDisk={saveAvatarToDisk} userAvatarData={userAvatarData} /> ); break; case ModalState.AddingGroupMembers: modalNode = ( { setAddGroupMembersRequestState(oldRequestState => { assertDev( oldRequestState !== RequestState.Active, 'Should not be clearing an active request state' ); return RequestState.Inactive; }); }} conversationIdsAlreadyInGroup={ new Set(memberships.map(membership => membership.member.id)) } groupTitle={conversation.title} i18n={i18n} makeRequest={async conversationIds => { setAddGroupMembersRequestState(RequestState.Active); addMembersToGroup(conversation.id, conversationIds, { onSuccess: () => { setModalState(ModalState.NothingOpen); setAddGroupMembersRequestState(RequestState.Inactive); }, onFailure: () => { setAddGroupMembersRequestState(RequestState.InactiveWithError); }, }); }} maxGroupSize={maxGroupSize} maxRecommendedGroupSize={maxRecommendedGroupSize} onClose={onCloseModal} requestState={addGroupMembersRequestState} /> ); break; case ModalState.ConfirmDeleteNicknameAndNote: modalNode = ( {i18n( 'icu:ConversationDetails__ConfirmDeleteNicknameAndNote__Description' )} ); break; case ModalState.MuteNotifications: modalNode = ( ); break; case ModalState.UnmuteNotifications: modalNode = ( setMuteExpiration(conversation.id, 0), style: 'affirmative', text: i18n('icu:unmute'), }, ]} hasXButton i18n={i18n} title={i18n('icu:ConversationDetails__unmute--title')} onClose={onCloseModal} > {getMutedUntilText(Number(conversation.muteExpiresAt), i18n)} ); break; default: throw missingCaseError(modalState); } const isMuted = isConversationMuted(conversation); return (
{ setModalState( isGroupTitle ? ModalState.EditingGroupTitle : ModalState.EditingGroupDescription ); }} theme={theme} toggleAboutContactModal={toggleAboutContactModal} />
{selectedNavTab === NavTab.Calls && ( )} {!conversation.isMe && ( <> onOutgoingVideoCallInConversation(conversation.id)} type="video" /> {!isGroup && ( onOutgoingAudioCallInConversation(conversation.id) } type="audio" /> )} )} {selectedNavTab !== NavTab.Calls && ( )}
{callHistoryGroup && (
    {callHistoryGroup.children.map(child => { return (
  1. {describeCallHistory( i18n, callHistoryGroup.type, callHistoryGroup.direction, callHistoryGroup.status )} {formatTime(i18n, child.timestamp, Date.now(), false)}
  2. ); })}
)} {!isGroup || canEditGroupInfo ? ( } info={ isGroup ? i18n( 'icu:ConversationDetails--disappearing-messages-info--group' ) : i18n( 'icu:ConversationDetails--disappearing-messages-info--direct' ) } label={i18n('icu:ConversationDetails--disappearing-messages-label')} right={ setDisappearingMessages(conversation.id, value) } /> } /> ) : null} {canHaveNicknameAndNote(conversation) && ( } label={i18n('icu:ConversationDetails--nickname-label')} onClick={onOpenEditNicknameAndNoteModal} actions={ (conversation.nicknameGivenName || conversation.nicknameFamilyName || conversation.note) && ( { setModalState(ModalState.ConfirmDeleteNicknameAndNote); }, }, ]} > {({ openMenu }) => { return ( ); }} ) } /> )} {selectedNavTab === NavTab.Chats && ( } label={i18n('icu:showChatColorEditor')} onClick={() => { pushPanelForConversation({ type: PanelType.ChatColorEditor, }); }} right={
} /> )} {isGroup && ( } label={i18n('icu:ConversationDetails--notifications')} onClick={() => pushPanelForConversation({ type: PanelType.NotificationSettings, }) } right={ conversation.muteExpiresAt ? getMutedUntilText(conversation.muteExpiresAt, i18n) : undefined } /> )} {!isGroup && !conversation.isMe && ( toggleSafetyNumberModal(conversation.id)} icon={ } label={
{i18n('icu:ConversationDetails__viewSafetyNumber')}
} /> )} {isGroup && ( { setModalState(ModalState.AddingGroupMembers); }} theme={theme} /> )} {isGroup && ( {isAdmin || hasGroupLink ? ( } label={i18n('icu:ConversationDetails--group-link')} onClick={() => pushPanelForConversation({ type: PanelType.GroupLinkManagement, }) } right={hasGroupLink ? i18n('icu:on') : i18n('icu:off')} /> ) : null} } label={i18n('icu:ConversationDetails--requests-and-invites')} onClick={() => pushPanelForConversation({ type: PanelType.GroupInvites, }) } right={invitesCount} /> {isAdmin ? ( } label={i18n('icu:permissions')} onClick={() => pushPanelForConversation({ type: PanelType.GroupPermissions, }) } /> ) : null} )} pushPanelForConversation({ type: PanelType.AllMedia, }) } showLightboxWithMedia={showLightboxWithMedia} /> {!isGroup && !conversation.isMe && ( )} {!conversation.isMe && ( leaveGroup(conversation.id)} /> )} {modalNode}
); } function ConversationDetailsCallButton({ disabled, i18n, onClick, type, }: Readonly<{ disabled: boolean; i18n: LocalizerType; onClick: () => unknown; type: 'audio' | 'video'; }>) { const button = ( ); if (disabled) { return ( {button} ); } return button; }