233 lines
7.9 KiB
TypeScript
233 lines
7.9 KiB
TypeScript
// Copyright 2023 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import React, { useCallback, useState, useRef } from 'react';
|
|
import { noop } from 'lodash';
|
|
|
|
import type { AttachmentType } from '../types/Attachment';
|
|
import type { LocalizerType } from '../types/Util';
|
|
import type { MessagePropsType } from '../state/selectors/message';
|
|
import type { PreferredBadgeSelectorType } from '../state/selectors/badges';
|
|
import { Message, TextDirection } from './conversation/Message';
|
|
import { Modal } from './Modal';
|
|
import { WidthBreakpoint } from './_util';
|
|
import { shouldNeverBeCalled } from '../util/shouldNeverBeCalled';
|
|
import { useTheme } from '../hooks/useTheme';
|
|
import { isSameDay } from '../util/timestamp';
|
|
import { TimelineDateHeader } from './conversation/TimelineDateHeader';
|
|
import { drop } from '../util/drop';
|
|
|
|
export type PropsType = {
|
|
closeEditHistoryModal: () => unknown;
|
|
editHistoryMessages: Array<MessagePropsType>;
|
|
getPreferredBadge: PreferredBadgeSelectorType;
|
|
i18n: LocalizerType;
|
|
platform: string;
|
|
kickOffAttachmentDownload: (options: {
|
|
attachment: AttachmentType;
|
|
messageId: string;
|
|
}) => void;
|
|
showLightbox: (options: {
|
|
attachment: AttachmentType;
|
|
messageId: string;
|
|
}) => void;
|
|
};
|
|
|
|
const MESSAGE_DEFAULT_PROPS = {
|
|
canDeleteForEveryone: false,
|
|
checkForAccount: shouldNeverBeCalled,
|
|
clearSelectedMessage: shouldNeverBeCalled,
|
|
clearTargetedMessage: shouldNeverBeCalled,
|
|
containerWidthBreakpoint: WidthBreakpoint.Medium,
|
|
doubleCheckMissingQuoteReference: shouldNeverBeCalled,
|
|
interactionMode: 'mouse' as const,
|
|
isBlocked: false,
|
|
isMessageRequestAccepted: true,
|
|
markAttachmentAsCorrupted: shouldNeverBeCalled,
|
|
messageExpanded: shouldNeverBeCalled,
|
|
onReplyToMessage: shouldNeverBeCalled,
|
|
onToggleSelect: shouldNeverBeCalled,
|
|
openGiftBadge: shouldNeverBeCalled,
|
|
openLink: shouldNeverBeCalled,
|
|
previews: [],
|
|
retryMessageSend: shouldNeverBeCalled,
|
|
pushPanelForConversation: shouldNeverBeCalled,
|
|
renderAudioAttachment: () => <div />,
|
|
renderingContext: 'EditHistoryMessagesModal',
|
|
saveAttachment: shouldNeverBeCalled,
|
|
scrollToQuotedMessage: shouldNeverBeCalled,
|
|
shouldCollapseAbove: false,
|
|
shouldCollapseBelow: false,
|
|
shouldHideMetadata: false,
|
|
showContactModal: shouldNeverBeCalled,
|
|
showConversation: noop,
|
|
showEditHistoryModal: noop,
|
|
showExpiredIncomingTapToViewToast: shouldNeverBeCalled,
|
|
showExpiredOutgoingTapToViewToast: shouldNeverBeCalled,
|
|
showLightboxForViewOnceMedia: shouldNeverBeCalled,
|
|
startConversation: shouldNeverBeCalled,
|
|
textDirection: TextDirection.Default,
|
|
viewStory: shouldNeverBeCalled,
|
|
};
|
|
|
|
export function EditHistoryMessagesModal({
|
|
closeEditHistoryModal,
|
|
getPreferredBadge,
|
|
editHistoryMessages,
|
|
i18n,
|
|
platform,
|
|
kickOffAttachmentDownload,
|
|
showLightbox,
|
|
}: PropsType): JSX.Element {
|
|
const containerElementRef = useRef<HTMLDivElement | null>(null);
|
|
const theme = useTheme();
|
|
|
|
const closeAndShowLightbox = useCallback(
|
|
(options: { attachment: AttachmentType; messageId: string }) => {
|
|
closeEditHistoryModal();
|
|
showLightbox(options);
|
|
},
|
|
[closeEditHistoryModal, showLightbox]
|
|
);
|
|
|
|
// These states aren't in redux; they are meant to last only as long as this dialog.
|
|
const [revealedSpoilersById, setRevealedSpoilersById] = useState<
|
|
Record<string, Record<number, boolean> | undefined>
|
|
>({});
|
|
const [displayLimitById, setDisplayLimitById] = useState<
|
|
Record<string, number | undefined>
|
|
>({});
|
|
|
|
const [currentMessage, ...pastEdits] = editHistoryMessages;
|
|
const currentMessageId = `${currentMessage.id}.${currentMessage.timestamp}`;
|
|
|
|
let previousItem = currentMessage;
|
|
|
|
return (
|
|
<Modal
|
|
hasXButton
|
|
i18n={i18n}
|
|
modalName="EditHistoryMessagesModal"
|
|
moduleClassName="EditHistoryMessagesModal"
|
|
onClose={closeEditHistoryModal}
|
|
noTransform
|
|
>
|
|
<div ref={containerElementRef}>
|
|
<TimelineDateHeader i18n={i18n} timestamp={currentMessage.timestamp} />
|
|
<Message
|
|
{...MESSAGE_DEFAULT_PROPS}
|
|
{...currentMessage}
|
|
id={currentMessageId}
|
|
containerElementRef={containerElementRef}
|
|
displayLimit={displayLimitById[currentMessageId]}
|
|
getPreferredBadge={getPreferredBadge}
|
|
i18n={i18n}
|
|
isEditedMessage
|
|
isSpoilerExpanded={revealedSpoilersById[currentMessageId] || {}}
|
|
key={currentMessage.timestamp}
|
|
kickOffAttachmentDownload={({ attachment }) =>
|
|
kickOffAttachmentDownload({
|
|
attachment,
|
|
messageId: currentMessage.id,
|
|
})
|
|
}
|
|
messageExpanded={(messageId, displayLimit) => {
|
|
const update = {
|
|
...displayLimitById,
|
|
[messageId]: displayLimit,
|
|
};
|
|
setDisplayLimitById(update);
|
|
}}
|
|
onContextMenu={() => {
|
|
drop(
|
|
window.navigator.clipboard.writeText(
|
|
String(currentMessage.timestamp)
|
|
)
|
|
);
|
|
}}
|
|
platform={platform}
|
|
showLightbox={closeAndShowLightbox}
|
|
showSpoiler={(messageId, data) => {
|
|
const update = {
|
|
...revealedSpoilersById,
|
|
[messageId]: data,
|
|
};
|
|
setRevealedSpoilersById(update);
|
|
}}
|
|
theme={theme}
|
|
/>
|
|
|
|
<hr className="EditHistoryMessagesModal__divider" />
|
|
|
|
<h3 className="EditHistoryMessagesModal__title">
|
|
{i18n('icu:EditHistoryMessagesModal__title')}
|
|
</h3>
|
|
|
|
{pastEdits.map(messageAttributes => {
|
|
const syntheticId = `${messageAttributes.id}.${messageAttributes.timestamp}`;
|
|
|
|
const shouldShowDateHeader = Boolean(
|
|
!previousItem ||
|
|
// This comparison avoids strange header behavior for out-of-order messages.
|
|
(messageAttributes.timestamp > previousItem.timestamp &&
|
|
!isSameDay(previousItem.timestamp, messageAttributes.timestamp))
|
|
);
|
|
const dateHeaderElement = shouldShowDateHeader ? (
|
|
<TimelineDateHeader
|
|
i18n={i18n}
|
|
timestamp={messageAttributes.timestamp}
|
|
/>
|
|
) : null;
|
|
|
|
previousItem = messageAttributes;
|
|
|
|
return (
|
|
<React.Fragment key={messageAttributes.timestamp}>
|
|
{dateHeaderElement}
|
|
<Message
|
|
{...MESSAGE_DEFAULT_PROPS}
|
|
{...messageAttributes}
|
|
id={syntheticId}
|
|
containerElementRef={containerElementRef}
|
|
displayLimit={displayLimitById[syntheticId]}
|
|
getPreferredBadge={getPreferredBadge}
|
|
i18n={i18n}
|
|
isSpoilerExpanded={revealedSpoilersById[syntheticId] || {}}
|
|
kickOffAttachmentDownload={({ attachment }) =>
|
|
kickOffAttachmentDownload({
|
|
attachment,
|
|
messageId: messageAttributes.id,
|
|
})
|
|
}
|
|
messageExpanded={(messageId, displayLimit) => {
|
|
const update = {
|
|
...displayLimitById,
|
|
[messageId]: displayLimit,
|
|
};
|
|
setDisplayLimitById(update);
|
|
}}
|
|
onContextMenu={() => {
|
|
drop(
|
|
window.navigator.clipboard.writeText(
|
|
String(messageAttributes.timestamp)
|
|
)
|
|
);
|
|
}}
|
|
platform={platform}
|
|
showLightbox={closeAndShowLightbox}
|
|
showSpoiler={(messageId, data) => {
|
|
const update = {
|
|
...revealedSpoilersById,
|
|
[messageId]: data,
|
|
};
|
|
setRevealedSpoilersById(update);
|
|
}}
|
|
theme={theme}
|
|
/>
|
|
</React.Fragment>
|
|
);
|
|
})}
|
|
</div>
|
|
</Modal>
|
|
);
|
|
}
|