signal-desktop/ts/components/conversation/conversation-details/ConversationDetails.tsx

590 lines
18 KiB
TypeScript
Raw Normal View History

// Copyright 2021-2022 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import type { ReactNode } from 'react';
import React, { useState } from 'react';
import { Button, ButtonIconType, ButtonVariant } from '../../Button';
import { Tooltip } from '../../Tooltip';
import type { ConversationType } from '../../../state/ducks/conversations';
2021-11-17 21:11:21 +00:00
import type { PreferredBadgeSelectorType } from '../../../state/selectors/badges';
import type { SmartChooseGroupMembersModalPropsType } from '../../../state/smart/ChooseGroupMembersModal';
import type { SmartConfirmAdditionsModalPropsType } from '../../../state/smart/ConfirmAdditionsModal';
2021-03-11 21:29:31 +00:00
import { assert } from '../../../util/assert';
2021-08-05 12:35:33 +00:00
import { getMutedUntilText } from '../../../util/getMutedUntilText';
2021-11-02 23:01:13 +00:00
import type { LocalizerType, ThemeType } from '../../../types/Util';
import type { MediaItemType } from '../../../types/MediaItem';
2021-11-02 23:01:13 +00:00
import type { BadgeType } from '../../../badges/types';
2021-09-22 00:58:03 +00:00
import { CapabilityError } from '../../../types/errors';
2021-03-11 21:29:31 +00:00
import { missingCaseError } from '../../../util/missingCaseError';
import { DisappearingTimerSelect } from '../../DisappearingTimerSelect';
2021-06-01 20:45:43 +00:00
import { PanelRow } from './PanelRow';
import { PanelSection } from './PanelSection';
2021-03-11 21:29:31 +00:00
import { AddGroupMembersModal } from './AddGroupMembersModal';
import { ConversationDetailsActions } from './ConversationDetailsActions';
import { ConversationDetailsHeader } from './ConversationDetailsHeader';
import { ConversationDetailsIcon, IconType } from './ConversationDetailsIcon';
import { ConversationDetailsMediaList } from './ConversationDetailsMediaList';
import type { GroupV2Membership } from './ConversationDetailsMembershipList';
import { ConversationDetailsMembershipList } from './ConversationDetailsMembershipList';
import type {
GroupV2PendingMembership,
GroupV2RequestingMembership,
} from './PendingInvites';
2021-03-11 21:29:31 +00:00
import { EditConversationAttributesModal } from './EditConversationAttributesModal';
import { RequestState } from './util';
2021-05-28 16:15:17 +00:00
import { getCustomColorStyle } from '../../../util/getCustomColorStyle';
2021-07-20 20:18:35 +00:00
import { ConfirmationDialog } from '../../ConfirmationDialog';
import { ConversationNotificationsModal } from './ConversationNotificationsModal';
import type {
2021-08-06 00:17:05 +00:00
AvatarDataType,
DeleteAvatarFromDiskActionType,
ReplaceAvatarActionType,
SaveAvatarToDiskActionType,
} from '../../../types/Avatar';
import { isMuted } from '../../../util/isMuted';
2021-03-11 21:29:31 +00:00
enum ModalState {
NothingOpen,
EditingGroupDescription,
EditingGroupTitle,
2021-03-11 21:29:31 +00:00
AddingGroupMembers,
MuteNotifications,
UnmuteNotifications,
2021-03-11 21:29:31 +00:00
}
export type StateProps = {
2021-03-11 21:29:31 +00:00
addMembers: (conversationIds: ReadonlyArray<string>) => Promise<void>;
areWeASubscriber: boolean;
2021-11-02 23:01:13 +00:00
badges?: ReadonlyArray<BadgeType>;
canEditGroupInfo: boolean;
conversation?: ConversationType;
hasGroupLink: boolean;
2021-11-17 21:11:21 +00:00
getPreferredBadge: PreferredBadgeSelectorType;
hasActiveCall: boolean;
i18n: LocalizerType;
isAdmin: boolean;
isGroup: boolean;
loadRecentMediaItems: (limit: number) => void;
memberships: Array<GroupV2Membership>;
pendingApprovalMemberships: ReadonlyArray<GroupV2RequestingMembership>;
pendingMemberships: ReadonlyArray<GroupV2PendingMembership>;
setDisappearingMessages: (seconds: number) => void;
showAllMedia: () => void;
showChatColorEditor: () => void;
showGroupLinkManagement: () => void;
showGroupV2Permissions: () => void;
showPendingInvites: () => void;
showLightboxForMedia: (
selectedMediaItem: MediaItemType,
media: Array<MediaItemType>
) => void;
2021-08-05 12:35:33 +00:00
showConversationNotificationsSettings: () => void;
updateGroupAttributes: (
_: Readonly<{
2021-09-24 00:49:05 +00:00
avatar?: undefined | Uint8Array;
2021-06-02 00:24:28 +00:00
description?: string;
title?: string;
}>
) => Promise<void>;
onBlock: () => void;
onLeave: () => void;
onUnblock: () => void;
2021-11-02 23:01:13 +00:00
theme: ThemeType;
2021-08-06 00:17:05 +00:00
userAvatarData: Array<AvatarDataType>;
setMuteExpiration: (muteExpiresAt: undefined | number) => unknown;
onOutgoingAudioCallInConversation: () => unknown;
onOutgoingVideoCallInConversation: () => unknown;
renderChooseGroupMembersModal: (
props: SmartChooseGroupMembersModalPropsType
) => JSX.Element;
renderConfirmAdditionsModal: (
props: SmartConfirmAdditionsModalPropsType
) => JSX.Element;
2021-08-06 00:17:05 +00:00
};
type ActionProps = {
deleteAvatarFromDisk: DeleteAvatarFromDiskActionType;
replaceAvatar: ReplaceAvatarActionType;
saveAvatarToDisk: SaveAvatarToDiskActionType;
showContactModal: (contactId: string, conversationId?: string) => void;
toggleSafetyNumberModal: (conversationId: string) => unknown;
2021-11-01 18:43:02 +00:00
searchInConversation: (id: string) => unknown;
};
2021-08-06 00:17:05 +00:00
export type Props = StateProps & ActionProps;
export const ConversationDetails: React.ComponentType<Props> = ({
2021-03-11 21:29:31 +00:00
addMembers,
areWeASubscriber,
2021-11-02 23:01:13 +00:00
badges,
canEditGroupInfo,
conversation,
deleteAvatarFromDisk,
hasGroupLink,
2021-11-17 21:11:21 +00:00
getPreferredBadge,
hasActiveCall,
i18n,
isAdmin,
isGroup,
loadRecentMediaItems,
memberships,
onBlock,
onLeave,
onOutgoingAudioCallInConversation,
onOutgoingVideoCallInConversation,
onUnblock,
pendingApprovalMemberships,
pendingMemberships,
renderChooseGroupMembersModal,
renderConfirmAdditionsModal,
replaceAvatar,
saveAvatarToDisk,
searchInConversation,
setDisappearingMessages,
setMuteExpiration,
showAllMedia,
showChatColorEditor,
showContactModal,
showConversationNotificationsSettings,
showGroupLinkManagement,
showGroupV2Permissions,
showLightboxForMedia,
showPendingInvites,
2021-11-02 23:01:13 +00:00
theme,
toggleSafetyNumberModal,
updateGroupAttributes,
2021-08-06 00:17:05 +00:00
userAvatarData,
}) => {
2021-03-11 21:29:31 +00:00
const [modalState, setModalState] = useState<ModalState>(
ModalState.NothingOpen
);
2021-11-11 22:43:05 +00:00
const [editGroupAttributesRequestState, setEditGroupAttributesRequestState] =
useState<RequestState>(RequestState.Inactive);
const [addGroupMembersRequestState, setAddGroupMembersRequestState] =
useState<RequestState>(RequestState.Inactive);
const [membersMissingCapability, setMembersMissingCapability] =
useState(false);
if (conversation === undefined) {
throw new Error('ConversationDetails rendered without a conversation');
}
const invitesCount =
pendingMemberships.length + pendingApprovalMemberships.length;
const otherMemberships = memberships.filter(({ member }) => !member.isMe);
const isJustMe = otherMemberships.length === 0;
const isAnyoneElseAnAdmin = otherMemberships.some(
membership => membership.isAdmin
);
const cannotLeaveBecauseYouAreLastAdmin =
isAdmin && !isJustMe && !isAnyoneElseAnAdmin;
2021-03-11 21:29:31 +00:00
let modalNode: ReactNode;
switch (modalState) {
case ModalState.NothingOpen:
modalNode = undefined;
break;
case ModalState.EditingGroupDescription:
case ModalState.EditingGroupTitle:
2021-03-11 21:29:31 +00:00
modalNode = (
<EditConversationAttributesModal
2021-08-06 00:17:05 +00:00
avatarColor={conversation.color}
2021-03-11 21:29:31 +00:00
avatarPath={conversation.avatarPath}
2021-08-06 00:17:05 +00:00
conversationId={conversation.id}
2021-06-02 00:24:28 +00:00
groupDescription={conversation.groupDescription}
2021-03-11 21:29:31 +00:00
i18n={i18n}
initiallyFocusDescription={
modalState === ModalState.EditingGroupDescription
}
2021-03-11 21:29:31 +00:00
makeRequest={async (
options: Readonly<{
2021-09-24 00:49:05 +00:00
avatar?: undefined | Uint8Array;
2021-06-02 00:24:28 +00:00
description?: string;
2021-03-11 21:29:31 +00:00
title?: string;
}>
) => {
setEditGroupAttributesRequestState(RequestState.Active);
try {
await updateGroupAttributes(options);
setModalState(ModalState.NothingOpen);
setEditGroupAttributesRequestState(RequestState.Inactive);
} catch (err) {
setEditGroupAttributesRequestState(
RequestState.InactiveWithError
);
}
}}
onClose={() => {
setModalState(ModalState.NothingOpen);
setEditGroupAttributesRequestState(RequestState.Inactive);
}}
requestState={editGroupAttributesRequestState}
title={conversation.title}
2021-08-06 00:17:05 +00:00
deleteAvatarFromDisk={deleteAvatarFromDisk}
replaceAvatar={replaceAvatar}
saveAvatarToDisk={saveAvatarToDisk}
userAvatarData={userAvatarData}
2021-03-11 21:29:31 +00:00
/>
);
break;
case ModalState.AddingGroupMembers:
modalNode = (
<AddGroupMembersModal
renderChooseGroupMembersModal={renderChooseGroupMembersModal}
renderConfirmAdditionsModal={renderConfirmAdditionsModal}
2021-03-11 21:29:31 +00:00
clearRequestError={() => {
setAddGroupMembersRequestState(oldRequestState => {
assert(
oldRequestState !== RequestState.Active,
'Should not be clearing an active request state'
);
return RequestState.Inactive;
});
}}
conversationIdsAlreadyInGroup={
new Set(memberships.map(membership => membership.member.id))
2021-03-11 21:29:31 +00:00
}
groupTitle={conversation.title}
i18n={i18n}
makeRequest={async conversationIds => {
setAddGroupMembersRequestState(RequestState.Active);
try {
await addMembers(conversationIds);
setModalState(ModalState.NothingOpen);
setAddGroupMembersRequestState(RequestState.Inactive);
} catch (err) {
2021-09-22 00:58:03 +00:00
if (err instanceof CapabilityError) {
2021-07-20 20:18:35 +00:00
setMembersMissingCapability(true);
setAddGroupMembersRequestState(RequestState.InactiveWithError);
} else {
setAddGroupMembersRequestState(RequestState.InactiveWithError);
}
2021-03-11 21:29:31 +00:00
}
}}
onClose={() => {
setModalState(ModalState.NothingOpen);
setEditGroupAttributesRequestState(RequestState.Inactive);
}}
requestState={addGroupMembersRequestState}
/>
);
break;
case ModalState.MuteNotifications:
modalNode = (
<ConversationNotificationsModal
i18n={i18n}
muteExpiresAt={conversation.muteExpiresAt}
onClose={() => {
setModalState(ModalState.NothingOpen);
}}
setMuteExpiration={setMuteExpiration}
/>
);
break;
case ModalState.UnmuteNotifications:
modalNode = (
<ConfirmationDialog
actions={[
{
action: () => setMuteExpiration(0),
style: 'affirmative',
text: i18n('unmute'),
},
]}
hasXButton
i18n={i18n}
title={i18n('ConversationDetails__unmute--title')}
onClose={() => {
setModalState(ModalState.NothingOpen);
}}
>
{getMutedUntilText(Number(conversation.muteExpiresAt), i18n)}
</ConfirmationDialog>
);
break;
2021-03-11 21:29:31 +00:00
default:
throw missingCaseError(modalState);
}
const isConversationMuted = isMuted(conversation.muteExpiresAt);
return (
<div className="conversation-details-panel">
2021-07-20 20:18:35 +00:00
{membersMissingCapability && (
<ConfirmationDialog
cancelText={i18n('Confirmation--confirm')}
i18n={i18n}
onClose={() => setMembersMissingCapability(false)}
>
{i18n('GroupV2--add--missing-capability')}
</ConfirmationDialog>
)}
<ConversationDetailsHeader
areWeASubscriber={areWeASubscriber}
2021-11-02 23:01:13 +00:00
badges={badges}
canEdit={canEditGroupInfo}
conversation={conversation}
i18n={i18n}
isMe={conversation.isMe}
isGroup={isGroup}
memberships={memberships}
startEditing={(isGroupTitle: boolean) => {
setModalState(
isGroupTitle
? ModalState.EditingGroupTitle
: ModalState.EditingGroupDescription
);
}}
2021-11-02 23:01:13 +00:00
theme={theme}
/>
<div className="ConversationDetails__header-buttons">
{!conversation.isMe && (
<>
<ConversationDetailsCallButton
disabled={hasActiveCall}
i18n={i18n}
onClick={onOutgoingVideoCallInConversation}
type="video"
/>
{!isGroup && (
<ConversationDetailsCallButton
disabled={hasActiveCall}
i18n={i18n}
onClick={onOutgoingAudioCallInConversation}
type="audio"
/>
)}
</>
)}
<Button
icon={
isConversationMuted ? ButtonIconType.muted : ButtonIconType.unmuted
}
onClick={() => {
if (isConversationMuted) {
setModalState(ModalState.UnmuteNotifications);
} else {
setModalState(ModalState.MuteNotifications);
}
}}
variant={ButtonVariant.Details}
>
{isConversationMuted ? i18n('unmute') : i18n('mute')}
</Button>
<Button
icon={ButtonIconType.search}
onClick={() => {
2021-11-01 18:43:02 +00:00
searchInConversation(conversation.id);
}}
variant={ButtonVariant.Details}
>
{i18n('search')}
</Button>
</div>
2021-05-28 16:15:17 +00:00
<PanelSection>
{!isGroup || canEditGroupInfo ? (
<PanelRow
icon={
<ConversationDetailsIcon
ariaLabel={i18n(
'ConversationDetails--disappearing-messages-label'
)}
icon={IconType.timer}
/>
}
info={i18n(
isGroup
? 'ConversationDetails--disappearing-messages-info--group'
: 'ConversationDetails--disappearing-messages-info--direct'
)}
label={i18n('ConversationDetails--disappearing-messages-label')}
right={
<DisappearingTimerSelect
i18n={i18n}
value={conversation.expireTimer || 0}
onChange={setDisappearingMessages}
2021-06-01 20:45:43 +00:00
/>
}
/>
2021-05-28 16:15:17 +00:00
) : null}
<PanelRow
icon={
<ConversationDetailsIcon
ariaLabel={i18n('showChatColorEditor')}
icon={IconType.color}
2021-05-28 16:15:17 +00:00
/>
}
label={i18n('showChatColorEditor')}
onClick={showChatColorEditor}
2021-05-28 16:15:17 +00:00
right={
<div
className={`ConversationDetails__chat-color ConversationDetails__chat-color--${conversation.conversationColor}`}
2021-05-28 16:15:17 +00:00
style={{
...getCustomColorStyle(conversation.customColor),
}}
/>
}
/>
{isGroup && (
<PanelRow
icon={
<ConversationDetailsIcon
ariaLabel={i18n('ConversationDetails--notifications')}
icon={IconType.notifications}
/>
}
label={i18n('ConversationDetails--notifications')}
onClick={showConversationNotificationsSettings}
right={
conversation.muteExpiresAt
? getMutedUntilText(conversation.muteExpiresAt, i18n)
: undefined
}
/>
)}
{!isGroup && !conversation.isMe && (
<>
<PanelRow
onClick={() => toggleSafetyNumberModal(conversation.id)}
icon={
<ConversationDetailsIcon
ariaLabel={i18n('verifyNewNumber')}
icon={IconType.verify}
/>
}
label={
<div className="ConversationDetails__safety-number">
{i18n('verifyNewNumber')}
</div>
}
/>
</>
)}
</PanelSection>
{isGroup && (
<ConversationDetailsMembershipList
canAddNewMembers={canEditGroupInfo}
conversationId={conversation.id}
2021-11-17 21:11:21 +00:00
getPreferredBadge={getPreferredBadge}
i18n={i18n}
memberships={memberships}
showContactModal={showContactModal}
startAddingNewMembers={() => {
setModalState(ModalState.AddingGroupMembers);
}}
2021-11-02 23:01:13 +00:00
theme={theme}
/>
)}
{isGroup && (
<PanelSection>
{isAdmin || hasGroupLink ? (
<PanelRow
icon={
<ConversationDetailsIcon
ariaLabel={i18n('ConversationDetails--group-link')}
icon={IconType.link}
/>
}
label={i18n('ConversationDetails--group-link')}
onClick={showGroupLinkManagement}
right={hasGroupLink ? i18n('on') : i18n('off')}
/>
) : null}
<PanelRow
icon={
<ConversationDetailsIcon
ariaLabel={i18n('ConversationDetails--requests-and-invites')}
icon={IconType.invites}
/>
}
label={i18n('ConversationDetails--requests-and-invites')}
onClick={showPendingInvites}
right={invitesCount}
/>
{isAdmin ? (
<PanelRow
icon={
<ConversationDetailsIcon
ariaLabel={i18n('permissions')}
icon={IconType.lock}
/>
}
label={i18n('permissions')}
onClick={showGroupV2Permissions}
/>
) : null}
</PanelSection>
)}
<ConversationDetailsMediaList
conversation={conversation}
i18n={i18n}
loadRecentMediaItems={loadRecentMediaItems}
showAllMedia={showAllMedia}
showLightboxForMedia={showLightboxForMedia}
/>
{!conversation.isMe && (
<ConversationDetailsActions
cannotLeaveBecauseYouAreLastAdmin={cannotLeaveBecauseYouAreLastAdmin}
conversationTitle={conversation.title}
i18n={i18n}
isBlocked={Boolean(conversation.isBlocked)}
isGroup={isGroup}
left={Boolean(conversation.left)}
onBlock={onBlock}
onLeave={onLeave}
onUnblock={onUnblock}
/>
)}
2021-03-11 21:29:31 +00:00
{modalNode}
</div>
);
};
function ConversationDetailsCallButton({
disabled,
i18n,
onClick,
type,
}: Readonly<{
disabled: boolean;
i18n: LocalizerType;
onClick: () => unknown;
type: 'audio' | 'video';
}>) {
const button = (
<Button
disabled={disabled}
icon={ButtonIconType[type]}
onClick={onClick}
variant={ButtonVariant.Details}
>
{i18n(type)}
</Button>
);
if (disabled) {
return (
<Tooltip content={i18n('calling__in-another-call-tooltip')}>
{button}
</Tooltip>
);
}
return button;
}