2023-01-03 19:55:46 +00:00
|
|
|
// Copyright 2018 Signal Messenger, LLC
|
2020-10-30 20:34:04 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ReactChild, ReactNode } from 'react';
|
2023-04-12 23:17:56 +00:00
|
|
|
import React, { useEffect, useRef } from 'react';
|
2018-07-09 21:29:13 +00:00
|
|
|
import classNames from 'classnames';
|
2021-06-17 17:15:10 +00:00
|
|
|
import { noop } from 'lodash';
|
2018-07-09 21:29:13 +00:00
|
|
|
|
2021-07-20 19:56:50 +00:00
|
|
|
import { Avatar, AvatarSize } from '../Avatar';
|
2018-07-09 21:29:13 +00:00
|
|
|
import { ContactName } from './ContactName';
|
2022-07-25 18:55:44 +00:00
|
|
|
import { ContextMenu } from '../ContextMenu';
|
2022-01-26 23:05:26 +00:00
|
|
|
import { Time } from '../Time';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type {
|
2021-03-24 22:06:12 +00:00
|
|
|
Props as MessagePropsType,
|
|
|
|
PropsData as MessagePropsDataType,
|
|
|
|
} from './Message';
|
2021-10-26 19:15:33 +00:00
|
|
|
import { Message } from './Message';
|
2021-11-15 22:53:42 +00:00
|
|
|
import type { LocalizerType, ThemeType } from '../../types/Util';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ConversationType } from '../../state/ducks/conversations';
|
2021-11-16 15:53:41 +00:00
|
|
|
import type { PreferredBadgeSelectorType } from '../../state/selectors/badges';
|
2021-07-20 19:56:50 +00:00
|
|
|
import { groupBy } from '../../util/mapUtil';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ContactNameColorType } from '../../types/Colors';
|
2021-07-19 22:44:49 +00:00
|
|
|
import { SendStatus } from '../../messages/MessageSendState';
|
2021-10-12 23:59:08 +00:00
|
|
|
import { WidthBreakpoint } from '../_util';
|
2021-09-17 18:27:53 +00:00
|
|
|
import * as log from '../../logging/log';
|
2022-01-26 23:05:26 +00:00
|
|
|
import { formatDateTimeLong } from '../../util/timestamp';
|
2022-11-16 20:18:02 +00:00
|
|
|
import { DurationInSeconds } from '../../util/durations';
|
2022-09-09 18:35:00 +00:00
|
|
|
import { format as formatRelativeTime } from '../../util/expirationTimer';
|
2023-04-10 21:30:33 +00:00
|
|
|
import { missingCaseError } from '../../util/missingCaseError';
|
2018-07-09 21:29:13 +00:00
|
|
|
|
2021-05-07 22:21:10 +00:00
|
|
|
export type Contact = Pick<
|
|
|
|
ConversationType,
|
|
|
|
| 'acceptedMessageRequest'
|
|
|
|
| 'avatarPath'
|
2021-11-16 15:53:41 +00:00
|
|
|
| 'badges'
|
2021-05-07 22:21:10 +00:00
|
|
|
| 'color'
|
2021-06-16 00:44:14 +00:00
|
|
|
| 'id'
|
2021-05-07 22:21:10 +00:00
|
|
|
| 'isMe'
|
|
|
|
| 'phoneNumber'
|
|
|
|
| 'profileName'
|
|
|
|
| 'sharedGroupNames'
|
|
|
|
| 'title'
|
|
|
|
| 'unblurredAvatarPath'
|
|
|
|
> & {
|
2021-07-20 19:56:50 +00:00
|
|
|
status?: SendStatus;
|
2021-10-12 23:40:42 +00:00
|
|
|
statusTimestamp?: number;
|
2020-07-24 01:35:32 +00:00
|
|
|
|
2018-07-09 21:29:13 +00:00
|
|
|
isOutgoingKeyError: boolean;
|
2018-10-18 01:01:21 +00:00
|
|
|
isUnidentifiedDelivery: boolean;
|
2018-07-09 21:29:13 +00:00
|
|
|
|
|
|
|
errors?: Array<Error>;
|
2021-01-14 18:07:05 +00:00
|
|
|
};
|
2018-07-09 21:29:13 +00:00
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
export type PropsData = {
|
2021-07-20 19:56:50 +00:00
|
|
|
// An undefined status means they were the sender and it's an incoming message. If
|
|
|
|
// `undefined` is a status, there should be no other items in the array; if there are
|
|
|
|
// any defined statuses, `undefined` shouldn't be present.
|
|
|
|
contacts: ReadonlyArray<Contact>;
|
|
|
|
|
2021-06-07 16:50:18 +00:00
|
|
|
contactNameColor?: ContactNameColorType;
|
2021-03-24 22:06:12 +00:00
|
|
|
errors: Array<Error>;
|
2022-11-04 13:22:07 +00:00
|
|
|
message: Omit<
|
|
|
|
MessagePropsDataType,
|
|
|
|
'renderingContext' | 'menu' | 'contextMenu' | 'showMenu'
|
|
|
|
>;
|
2021-03-24 22:06:12 +00:00
|
|
|
receivedAt: number;
|
|
|
|
sentAt: number;
|
2018-07-09 21:29:13 +00:00
|
|
|
|
2019-01-14 21:49:58 +00:00
|
|
|
i18n: LocalizerType;
|
2023-04-03 20:16:27 +00:00
|
|
|
platform: string;
|
2021-11-15 22:53:42 +00:00
|
|
|
theme: ThemeType;
|
2021-11-16 15:53:41 +00:00
|
|
|
getPreferredBadge: PreferredBadgeSelectorType;
|
2022-12-21 03:25:10 +00:00
|
|
|
} & Pick<MessagePropsType, 'getPreferredBadge' | 'interactionMode'>;
|
2021-08-30 21:32:56 +00:00
|
|
|
|
2022-12-21 03:25:10 +00:00
|
|
|
export type PropsSmartActions = Pick<MessagePropsType, 'renderAudioAttachment'>;
|
2018-07-09 21:29:13 +00:00
|
|
|
|
2021-08-30 21:32:56 +00:00
|
|
|
export type PropsReduxActions = Pick<
|
|
|
|
MessagePropsType,
|
2022-12-10 02:02:22 +00:00
|
|
|
| 'checkForAccount'
|
2023-03-20 22:23:53 +00:00
|
|
|
| 'clearTargetedMessage'
|
2021-08-30 21:32:56 +00:00
|
|
|
| 'doubleCheckMissingQuoteReference'
|
2022-12-20 01:04:47 +00:00
|
|
|
| 'kickOffAttachmentDownload'
|
|
|
|
| 'markAttachmentAsCorrupted'
|
2023-04-10 16:31:45 +00:00
|
|
|
| 'messageExpanded'
|
2022-12-20 01:04:47 +00:00
|
|
|
| 'openGiftBadge'
|
2022-12-15 01:10:09 +00:00
|
|
|
| 'pushPanelForConversation'
|
2022-12-14 18:12:04 +00:00
|
|
|
| 'saveAttachment'
|
2022-12-09 06:08:55 +00:00
|
|
|
| 'showContactModal'
|
2022-12-14 19:05:32 +00:00
|
|
|
| 'showConversation'
|
2022-12-20 01:04:47 +00:00
|
|
|
| 'showExpiredIncomingTapToViewToast'
|
|
|
|
| 'showExpiredOutgoingTapToViewToast'
|
2022-12-10 02:02:22 +00:00
|
|
|
| 'showLightbox'
|
|
|
|
| 'showLightboxForViewOnceMedia'
|
2023-04-10 16:31:45 +00:00
|
|
|
| 'showSpoiler'
|
2022-12-21 03:25:10 +00:00
|
|
|
| 'startConversation'
|
2022-07-06 19:06:20 +00:00
|
|
|
| 'viewStory'
|
2022-12-09 05:53:19 +00:00
|
|
|
> & {
|
|
|
|
toggleSafetyNumberModal: (contactId: string) => void;
|
|
|
|
};
|
2021-08-30 21:32:56 +00:00
|
|
|
|
2022-12-21 03:25:10 +00:00
|
|
|
export type Props = PropsData & PropsSmartActions & PropsReduxActions;
|
2021-08-30 21:32:56 +00:00
|
|
|
|
2021-07-20 19:56:50 +00:00
|
|
|
const contactSortCollator = new Intl.Collator();
|
|
|
|
|
2020-09-14 19:51:27 +00:00
|
|
|
const _keyForError = (error: Error): string => {
|
|
|
|
return `${error.name}-${error.message}`;
|
|
|
|
};
|
|
|
|
|
2023-04-12 23:17:56 +00:00
|
|
|
export function MessageDetail({
|
|
|
|
contacts,
|
|
|
|
errors,
|
|
|
|
message,
|
|
|
|
receivedAt,
|
|
|
|
sentAt,
|
|
|
|
checkForAccount,
|
|
|
|
clearTargetedMessage,
|
|
|
|
contactNameColor,
|
|
|
|
doubleCheckMissingQuoteReference,
|
|
|
|
getPreferredBadge,
|
|
|
|
i18n,
|
|
|
|
interactionMode,
|
|
|
|
kickOffAttachmentDownload,
|
|
|
|
markAttachmentAsCorrupted,
|
|
|
|
messageExpanded,
|
|
|
|
openGiftBadge,
|
|
|
|
platform,
|
|
|
|
pushPanelForConversation,
|
|
|
|
renderAudioAttachment,
|
|
|
|
saveAttachment,
|
|
|
|
showContactModal,
|
|
|
|
showConversation,
|
|
|
|
showExpiredIncomingTapToViewToast,
|
|
|
|
showExpiredOutgoingTapToViewToast,
|
|
|
|
showLightbox,
|
|
|
|
showLightboxForViewOnceMedia,
|
|
|
|
showSpoiler,
|
|
|
|
startConversation,
|
|
|
|
theme,
|
|
|
|
toggleSafetyNumberModal,
|
|
|
|
viewStory,
|
|
|
|
}: Props): JSX.Element {
|
|
|
|
const focusRef = useRef<HTMLDivElement>(null);
|
|
|
|
const messageContainerRef = useRef<HTMLDivElement>(null);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
const timer = setTimeout(() => {
|
|
|
|
// When this component is created, it's initially not part of the DOM, and then it's
|
|
|
|
// added off-screen and animated in. This ensures that the focus takes.
|
|
|
|
focusRef.current?.focus();
|
2019-11-07 21:36:16 +00:00
|
|
|
});
|
2023-04-12 23:17:56 +00:00
|
|
|
return () => {
|
|
|
|
clearTimeout(timer);
|
|
|
|
};
|
|
|
|
}, []);
|
2019-11-07 21:36:16 +00:00
|
|
|
|
2023-04-12 23:17:56 +00:00
|
|
|
function renderAvatar(contact: Contact): JSX.Element {
|
2020-07-24 01:35:32 +00:00
|
|
|
const {
|
2021-05-04 23:19:36 +00:00
|
|
|
acceptedMessageRequest,
|
2020-07-24 01:35:32 +00:00
|
|
|
avatarPath,
|
2021-11-16 15:53:41 +00:00
|
|
|
badges,
|
2020-07-24 01:35:32 +00:00
|
|
|
color,
|
2021-05-07 22:21:10 +00:00
|
|
|
isMe,
|
2021-05-04 23:19:36 +00:00
|
|
|
phoneNumber,
|
2020-07-24 01:35:32 +00:00
|
|
|
profileName,
|
2021-05-04 23:19:36 +00:00
|
|
|
sharedGroupNames,
|
2020-07-24 01:35:32 +00:00
|
|
|
title,
|
2021-05-04 23:19:36 +00:00
|
|
|
unblurredAvatarPath,
|
2020-07-24 01:35:32 +00:00
|
|
|
} = contact;
|
2018-07-09 21:29:13 +00:00
|
|
|
|
|
|
|
return (
|
2018-09-27 00:23:17 +00:00
|
|
|
<Avatar
|
2021-05-04 23:19:36 +00:00
|
|
|
acceptedMessageRequest={acceptedMessageRequest}
|
2018-09-27 00:23:17 +00:00
|
|
|
avatarPath={avatarPath}
|
2021-11-16 15:53:41 +00:00
|
|
|
badge={getPreferredBadge(badges)}
|
2018-09-27 00:23:17 +00:00
|
|
|
color={color}
|
|
|
|
conversationType="direct"
|
|
|
|
i18n={i18n}
|
2021-05-07 22:21:10 +00:00
|
|
|
isMe={isMe}
|
2018-09-27 00:23:17 +00:00
|
|
|
phoneNumber={phoneNumber}
|
|
|
|
profileName={profileName}
|
2021-11-16 15:53:41 +00:00
|
|
|
theme={theme}
|
2020-07-24 01:35:32 +00:00
|
|
|
title={title}
|
2021-05-04 23:19:36 +00:00
|
|
|
sharedGroupNames={sharedGroupNames}
|
2022-12-09 20:37:45 +00:00
|
|
|
size={AvatarSize.THIRTY_TWO}
|
2021-05-04 23:19:36 +00:00
|
|
|
unblurredAvatarPath={unblurredAvatarPath}
|
2018-07-09 21:29:13 +00:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-04-12 23:17:56 +00:00
|
|
|
function renderContact(contact: Contact): JSX.Element {
|
|
|
|
const contactErrors = contact.errors || [];
|
2018-07-09 21:29:13 +00:00
|
|
|
|
|
|
|
const errorComponent = contact.isOutgoingKeyError ? (
|
|
|
|
<div className="module-message-detail__contact__error-buttons">
|
|
|
|
<button
|
2020-09-14 19:51:27 +00:00
|
|
|
type="button"
|
2018-07-09 21:29:13 +00:00
|
|
|
className="module-message-detail__contact__show-safety-number"
|
2022-12-09 05:53:19 +00:00
|
|
|
onClick={() => toggleSafetyNumberModal(contact.id)}
|
2018-07-09 21:29:13 +00:00
|
|
|
>
|
2023-03-30 00:03:25 +00:00
|
|
|
{i18n('icu:showSafetyNumber')}
|
2018-07-09 21:29:13 +00:00
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
) : null;
|
2018-10-18 01:01:21 +00:00
|
|
|
const unidentifiedDeliveryComponent = contact.isUnidentifiedDelivery ? (
|
|
|
|
<div className="module-message-detail__contact__unidentified-delivery-icon" />
|
|
|
|
) : null;
|
2018-07-09 21:29:13 +00:00
|
|
|
|
|
|
|
return (
|
2021-07-20 19:56:50 +00:00
|
|
|
<div key={contact.id} className="module-message-detail__contact">
|
2023-04-12 23:17:56 +00:00
|
|
|
{renderAvatar(contact)}
|
2018-07-09 21:29:13 +00:00
|
|
|
<div className="module-message-detail__contact__text">
|
|
|
|
<div className="module-message-detail__contact__name">
|
2021-09-16 16:15:43 +00:00
|
|
|
<ContactName title={contact.title} />
|
2018-07-09 21:29:13 +00:00
|
|
|
</div>
|
2023-04-12 23:17:56 +00:00
|
|
|
{contactErrors.map(contactError => (
|
2020-09-14 19:51:27 +00:00
|
|
|
<div
|
2023-04-12 23:17:56 +00:00
|
|
|
key={_keyForError(contactError)}
|
2020-09-14 19:51:27 +00:00
|
|
|
className="module-message-detail__contact__error"
|
|
|
|
>
|
2023-04-12 23:17:56 +00:00
|
|
|
{contactError.message}
|
2018-07-09 21:29:13 +00:00
|
|
|
</div>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
{errorComponent}
|
2018-10-18 01:01:21 +00:00
|
|
|
{unidentifiedDeliveryComponent}
|
2021-10-12 23:40:42 +00:00
|
|
|
{contact.statusTimestamp && (
|
2022-01-26 23:05:26 +00:00
|
|
|
<Time
|
|
|
|
className="module-message-detail__status-timestamp"
|
2021-10-12 23:40:42 +00:00
|
|
|
timestamp={contact.statusTimestamp}
|
2022-01-26 23:05:26 +00:00
|
|
|
>
|
|
|
|
{formatDateTimeLong(i18n, contact.statusTimestamp)}
|
|
|
|
</Time>
|
2021-10-12 23:40:42 +00:00
|
|
|
)}
|
2018-07-09 21:29:13 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-04-12 23:17:56 +00:00
|
|
|
function renderContactGroupHeaderText(
|
2023-03-29 17:15:54 +00:00
|
|
|
sendStatus: undefined | SendStatus
|
|
|
|
): string {
|
|
|
|
if (sendStatus === undefined) {
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:from');
|
2023-03-29 17:15:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (sendStatus) {
|
|
|
|
case SendStatus.Failed:
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:MessageDetailsHeader--Failed');
|
2023-03-29 17:15:54 +00:00
|
|
|
case SendStatus.Pending:
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:MessageDetailsHeader--Pending');
|
2023-03-29 17:15:54 +00:00
|
|
|
case SendStatus.Sent:
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:MessageDetailsHeader--Sent');
|
2023-03-29 17:15:54 +00:00
|
|
|
case SendStatus.Delivered:
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:MessageDetailsHeader--Delivered');
|
2023-03-29 17:15:54 +00:00
|
|
|
case SendStatus.Read:
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:MessageDetailsHeader--Read');
|
2023-03-29 17:15:54 +00:00
|
|
|
case SendStatus.Viewed:
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:MessageDetailsHeader--Viewed');
|
2023-03-29 17:15:54 +00:00
|
|
|
default:
|
|
|
|
throw missingCaseError(sendStatus);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-12 23:17:56 +00:00
|
|
|
function renderContactGroup(
|
2021-07-20 19:56:50 +00:00
|
|
|
sendStatus: undefined | SendStatus,
|
2023-04-12 23:17:56 +00:00
|
|
|
statusContacts: undefined | ReadonlyArray<Contact>
|
2021-07-20 19:56:50 +00:00
|
|
|
): ReactNode {
|
2023-04-12 23:17:56 +00:00
|
|
|
if (!statusContacts || !statusContacts.length) {
|
2018-07-09 21:29:13 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2023-04-12 23:17:56 +00:00
|
|
|
const sortedContacts = [...statusContacts].sort((a, b) =>
|
2021-07-20 19:56:50 +00:00
|
|
|
contactSortCollator.compare(a.title, b.title)
|
|
|
|
);
|
|
|
|
|
2023-04-12 23:17:56 +00:00
|
|
|
const headerText = renderContactGroupHeaderText(sendStatus);
|
2023-03-29 17:15:54 +00:00
|
|
|
|
2021-07-20 19:56:50 +00:00
|
|
|
return (
|
2023-03-29 17:15:54 +00:00
|
|
|
<div key={headerText} className="module-message-detail__contact-group">
|
2021-07-20 19:56:50 +00:00
|
|
|
<div
|
|
|
|
className={classNames(
|
|
|
|
'module-message-detail__contact-group__header',
|
|
|
|
sendStatus &&
|
|
|
|
`module-message-detail__contact-group__header--${sendStatus}`
|
|
|
|
)}
|
|
|
|
>
|
2023-03-29 17:15:54 +00:00
|
|
|
{headerText}
|
2021-07-20 19:56:50 +00:00
|
|
|
</div>
|
2023-04-12 23:17:56 +00:00
|
|
|
{sortedContacts.map(contact => renderContact(contact))}
|
2021-07-20 19:56:50 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-04-12 23:17:56 +00:00
|
|
|
function renderContacts(): ReactChild {
|
2021-07-20 19:56:50 +00:00
|
|
|
// This assumes that the list either contains one sender (a status of `undefined`) or
|
|
|
|
// 1+ contacts with `SendStatus`es, but it doesn't check that assumption.
|
|
|
|
const contactsBySendStatus = groupBy(contacts, contact => contact.status);
|
|
|
|
|
2018-07-09 21:29:13 +00:00
|
|
|
return (
|
|
|
|
<div className="module-message-detail__contact-container">
|
2021-07-20 19:56:50 +00:00
|
|
|
{[
|
|
|
|
undefined,
|
|
|
|
SendStatus.Failed,
|
|
|
|
SendStatus.Viewed,
|
|
|
|
SendStatus.Read,
|
|
|
|
SendStatus.Delivered,
|
|
|
|
SendStatus.Sent,
|
|
|
|
SendStatus.Pending,
|
|
|
|
].map(sendStatus =>
|
2023-04-12 23:17:56 +00:00
|
|
|
renderContactGroup(sendStatus, contactsBySendStatus.get(sendStatus))
|
2021-07-20 19:56:50 +00:00
|
|
|
)}
|
2018-07-09 21:29:13 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-04-12 23:17:56 +00:00
|
|
|
const timeRemaining = message.expirationTimestamp
|
|
|
|
? DurationInSeconds.fromMillis(message.expirationTimestamp - Date.now())
|
|
|
|
: undefined;
|
|
|
|
|
|
|
|
return (
|
|
|
|
// eslint-disable-next-line jsx-a11y/no-noninteractive-tabindex
|
|
|
|
<div className="module-message-detail" tabIndex={0} ref={focusRef}>
|
|
|
|
<div
|
|
|
|
className="module-message-detail__message-container"
|
|
|
|
ref={messageContainerRef}
|
|
|
|
>
|
|
|
|
<Message
|
|
|
|
{...message}
|
|
|
|
renderingContext="conversation/MessageDetail"
|
|
|
|
checkForAccount={checkForAccount}
|
|
|
|
clearTargetedMessage={clearTargetedMessage}
|
|
|
|
contactNameColor={contactNameColor}
|
|
|
|
containerElementRef={messageContainerRef}
|
|
|
|
containerWidthBreakpoint={WidthBreakpoint.Wide}
|
|
|
|
renderMenu={undefined}
|
|
|
|
disableScroll
|
|
|
|
displayLimit={Number.MAX_SAFE_INTEGER}
|
|
|
|
showLightboxForViewOnceMedia={showLightboxForViewOnceMedia}
|
|
|
|
doubleCheckMissingQuoteReference={doubleCheckMissingQuoteReference}
|
|
|
|
getPreferredBadge={getPreferredBadge}
|
|
|
|
i18n={i18n}
|
|
|
|
interactionMode={interactionMode}
|
|
|
|
kickOffAttachmentDownload={kickOffAttachmentDownload}
|
|
|
|
markAttachmentAsCorrupted={markAttachmentAsCorrupted}
|
|
|
|
messageExpanded={messageExpanded}
|
|
|
|
openGiftBadge={openGiftBadge}
|
|
|
|
platform={platform}
|
|
|
|
pushPanelForConversation={pushPanelForConversation}
|
|
|
|
renderAudioAttachment={renderAudioAttachment}
|
|
|
|
saveAttachment={saveAttachment}
|
|
|
|
shouldCollapseAbove={false}
|
|
|
|
shouldCollapseBelow={false}
|
|
|
|
shouldHideMetadata={false}
|
|
|
|
showConversation={showConversation}
|
|
|
|
showSpoiler={showSpoiler}
|
|
|
|
scrollToQuotedMessage={() => {
|
|
|
|
log.warn('MessageDetail: scrollToQuotedMessage called!');
|
|
|
|
}}
|
|
|
|
showContactModal={showContactModal}
|
|
|
|
showExpiredIncomingTapToViewToast={showExpiredIncomingTapToViewToast}
|
|
|
|
showExpiredOutgoingTapToViewToast={showExpiredOutgoingTapToViewToast}
|
|
|
|
showLightbox={showLightbox}
|
|
|
|
startConversation={startConversation}
|
|
|
|
theme={theme}
|
|
|
|
viewStory={viewStory}
|
|
|
|
onToggleSelect={noop}
|
|
|
|
onReplyToMessage={noop}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<table className="module-message-detail__info">
|
|
|
|
<tbody>
|
|
|
|
{(errors || []).map(error => (
|
|
|
|
<tr key={_keyForError(error)}>
|
2023-03-30 00:03:25 +00:00
|
|
|
<td className="module-message-detail__label">
|
2023-04-12 23:17:56 +00:00
|
|
|
{i18n('icu:error')}
|
2023-03-30 00:03:25 +00:00
|
|
|
</td>
|
2018-07-09 21:29:13 +00:00
|
|
|
<td>
|
2023-04-12 23:17:56 +00:00
|
|
|
{' '}
|
|
|
|
<span className="error-message">{error.message}</span>{' '}
|
2018-07-09 21:29:13 +00:00
|
|
|
</td>
|
|
|
|
</tr>
|
2023-04-12 23:17:56 +00:00
|
|
|
))}
|
|
|
|
<tr>
|
|
|
|
<td className="module-message-detail__label">{i18n('icu:sent')}</td>
|
|
|
|
<td>
|
|
|
|
<ContextMenu
|
|
|
|
i18n={i18n}
|
|
|
|
menuOptions={[
|
|
|
|
{
|
|
|
|
icon: 'StoryDetailsModal__copy-icon',
|
|
|
|
label: i18n('icu:StoryDetailsModal__copy-timestamp'),
|
|
|
|
onClick: () => {
|
|
|
|
void window.navigator.clipboard.writeText(String(sentAt));
|
|
|
|
},
|
|
|
|
},
|
|
|
|
]}
|
|
|
|
>
|
|
|
|
<>
|
|
|
|
<Time timestamp={sentAt}>
|
|
|
|
{formatDateTimeLong(i18n, sentAt)}
|
2022-01-26 23:05:26 +00:00
|
|
|
</Time>{' '}
|
2018-07-09 21:29:13 +00:00
|
|
|
<span className="module-message-detail__unix-timestamp">
|
2023-04-12 23:17:56 +00:00
|
|
|
({sentAt})
|
2018-07-09 21:29:13 +00:00
|
|
|
</span>
|
2023-04-12 23:17:56 +00:00
|
|
|
</>
|
|
|
|
</ContextMenu>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
{receivedAt && message.direction === 'incoming' ? (
|
|
|
|
<tr>
|
|
|
|
<td className="module-message-detail__label">
|
|
|
|
{i18n('icu:received')}
|
|
|
|
</td>
|
|
|
|
<td>
|
|
|
|
<Time timestamp={receivedAt}>
|
|
|
|
{formatDateTimeLong(i18n, receivedAt)}
|
|
|
|
</Time>{' '}
|
|
|
|
<span className="module-message-detail__unix-timestamp">
|
|
|
|
({receivedAt})
|
|
|
|
</span>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
) : null}
|
|
|
|
{timeRemaining && timeRemaining > 0 && (
|
|
|
|
<tr>
|
|
|
|
<td className="module-message-detail__label">
|
|
|
|
{i18n('icu:MessageDetail--disappears-in')}
|
|
|
|
</td>
|
|
|
|
<td>
|
|
|
|
{formatRelativeTime(i18n, timeRemaining, {
|
|
|
|
largest: 2,
|
|
|
|
})}
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
)}
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
{renderContacts()}
|
|
|
|
</div>
|
|
|
|
);
|
2018-07-09 21:29:13 +00:00
|
|
|
}
|