| 
									
										
										
										
											2023-01-03 11:55:46 -08:00
										 |  |  | // Copyright 2018 Signal Messenger, LLC
 | 
					
						
							| 
									
										
										
										
											2020-10-30 15:34:04 -05:00
										 |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { ReactChild, ReactNode } from 'react'; | 
					
						
							| 
									
										
										
										
											2023-08-18 17:48:39 -04:00
										 |  |  | import React, { useRef } from 'react'; | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | import classNames from 'classnames'; | 
					
						
							| 
									
										
										
										
											2021-06-17 10:15:10 -07:00
										 |  |  | import { noop } from 'lodash'; | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  | import { Avatar, AvatarSize } from '../Avatar'; | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | import { ContactName } from './ContactName'; | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  | import { ContextMenu } from '../ContextMenu'; | 
					
						
							| 
									
										
										
										
											2022-01-26 17:05:26 -06:00
										 |  |  | import { Time } from '../Time'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { | 
					
						
							| 
									
										
										
										
											2021-03-24 17:06:12 -05:00
										 |  |  |   Props as MessagePropsType, | 
					
						
							|  |  |  |   PropsData as MessagePropsDataType, | 
					
						
							|  |  |  | } from './Message'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import { Message } from './Message'; | 
					
						
							| 
									
										
										
										
											2021-11-15 16:53:42 -06:00
										 |  |  | import type { LocalizerType, ThemeType } from '../../types/Util'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { ConversationType } from '../../state/ducks/conversations'; | 
					
						
							| 
									
										
										
										
											2021-11-16 09:53:41 -06:00
										 |  |  | import type { PreferredBadgeSelectorType } from '../../state/selectors/badges'; | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  | import { groupBy } from '../../util/mapUtil'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { ContactNameColorType } from '../../types/Colors'; | 
					
						
							| 
									
										
										
										
											2024-09-12 16:48:27 -07:00
										 |  |  | import { | 
					
						
							|  |  |  |   SendStatus, | 
					
						
							|  |  |  |   type VisibleSendStatus, | 
					
						
							|  |  |  | } from '../../messages/MessageSendState'; | 
					
						
							| 
									
										
										
										
											2021-10-12 18:59:08 -05:00
										 |  |  | import { WidthBreakpoint } from '../_util'; | 
					
						
							| 
									
										
										
										
											2021-09-17 14:27:53 -04:00
										 |  |  | import * as log from '../../logging/log'; | 
					
						
							| 
									
										
										
										
											2022-01-26 17:05:26 -06:00
										 |  |  | import { formatDateTimeLong } from '../../util/timestamp'; | 
					
						
							| 
									
										
										
										
											2022-11-16 12:18:02 -08:00
										 |  |  | import { DurationInSeconds } from '../../util/durations'; | 
					
						
							| 
									
										
										
										
											2022-09-09 12:35:00 -06:00
										 |  |  | import { format as formatRelativeTime } from '../../util/expirationTimer'; | 
					
						
							| 
									
										
										
										
											2023-04-10 14:30:33 -07:00
										 |  |  | import { missingCaseError } from '../../util/missingCaseError'; | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  | import { PanelRow } from './conversation-details/PanelRow'; | 
					
						
							|  |  |  | import { PanelSection } from './conversation-details/PanelSection'; | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   ConversationDetailsIcon, | 
					
						
							|  |  |  |   IconType, | 
					
						
							|  |  |  | } from './conversation-details/ConversationDetailsIcon'; | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-07 17:21:10 -05:00
										 |  |  | export type Contact = Pick< | 
					
						
							|  |  |  |   ConversationType, | 
					
						
							|  |  |  |   | 'acceptedMessageRequest' | 
					
						
							| 
									
										
										
										
											2024-07-11 12:44:09 -07:00
										 |  |  |   | 'avatarUrl' | 
					
						
							| 
									
										
										
										
											2021-11-16 09:53:41 -06:00
										 |  |  |   | 'badges' | 
					
						
							| 
									
										
										
										
											2021-05-07 17:21:10 -05:00
										 |  |  |   | 'color' | 
					
						
							| 
									
										
										
										
											2021-06-15 17:44:14 -07:00
										 |  |  |   | 'id' | 
					
						
							| 
									
										
										
										
											2021-05-07 17:21:10 -05:00
										 |  |  |   | 'isMe' | 
					
						
							|  |  |  |   | 'phoneNumber' | 
					
						
							|  |  |  |   | 'profileName' | 
					
						
							|  |  |  |   | 'sharedGroupNames' | 
					
						
							|  |  |  |   | 'title' | 
					
						
							| 
									
										
										
										
											2024-07-11 12:44:09 -07:00
										 |  |  |   | 'unblurredAvatarUrl' | 
					
						
							| 
									
										
										
										
											2021-05-07 17:21:10 -05:00
										 |  |  | > & { | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  |   status?: SendStatus; | 
					
						
							| 
									
										
										
										
											2021-10-12 19:40:42 -04:00
										 |  |  |   statusTimestamp?: number; | 
					
						
							| 
									
										
										
										
											2020-07-23 18:35:32 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |   isOutgoingKeyError: boolean; | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |   isUnidentifiedDelivery: boolean; | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   errors?: Array<Error>; | 
					
						
							| 
									
										
										
										
											2021-01-14 12:07:05 -06:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-30 14:32:56 -07:00
										 |  |  | export type PropsData = { | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05: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 12:50:18 -04:00
										 |  |  |   contactNameColor?: ContactNameColorType; | 
					
						
							| 
									
										
										
										
											2021-03-24 17:06:12 -05:00
										 |  |  |   errors: Array<Error>; | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |   message: Omit< | 
					
						
							|  |  |  |     MessagePropsDataType, | 
					
						
							|  |  |  |     'renderingContext' | 'menu' | 'contextMenu' | 'showMenu' | 
					
						
							|  |  |  |   >; | 
					
						
							| 
									
										
										
										
											2021-03-24 17:06:12 -05:00
										 |  |  |   receivedAt: number; | 
					
						
							|  |  |  |   sentAt: number; | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-14 13:49:58 -08:00
										 |  |  |   i18n: LocalizerType; | 
					
						
							| 
									
										
										
										
											2023-04-03 13:16:27 -07:00
										 |  |  |   platform: string; | 
					
						
							| 
									
										
										
										
											2021-11-15 16:53:42 -06:00
										 |  |  |   theme: ThemeType; | 
					
						
							| 
									
										
										
										
											2021-11-16 09:53:41 -06:00
										 |  |  |   getPreferredBadge: PreferredBadgeSelectorType; | 
					
						
							| 
									
										
										
										
											2022-12-20 19:25:10 -08:00
										 |  |  | } & Pick<MessagePropsType, 'getPreferredBadge' | 'interactionMode'>; | 
					
						
							| 
									
										
										
										
											2021-08-30 14:32:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 19:25:10 -08:00
										 |  |  | export type PropsSmartActions = Pick<MessagePropsType, 'renderAudioAttachment'>; | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-30 14:32:56 -07:00
										 |  |  | export type PropsReduxActions = Pick< | 
					
						
							|  |  |  |   MessagePropsType, | 
					
						
							| 
									
										
										
										
											2022-12-09 21:02:22 -05:00
										 |  |  |   | 'checkForAccount' | 
					
						
							| 
									
										
										
										
											2023-03-20 15:23:53 -07:00
										 |  |  |   | 'clearTargetedMessage' | 
					
						
							| 
									
										
										
										
											2021-08-30 14:32:56 -07:00
										 |  |  |   | 'doubleCheckMissingQuoteReference' | 
					
						
							| 
									
										
										
										
											2022-12-19 17:04:47 -08:00
										 |  |  |   | 'kickOffAttachmentDownload' | 
					
						
							|  |  |  |   | 'markAttachmentAsCorrupted' | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   | 'messageExpanded' | 
					
						
							| 
									
										
										
										
											2022-12-19 17:04:47 -08:00
										 |  |  |   | 'openGiftBadge' | 
					
						
							| 
									
										
										
										
											2022-12-14 20:10:09 -05:00
										 |  |  |   | 'pushPanelForConversation' | 
					
						
							| 
									
										
										
										
											2023-04-20 12:31:59 -04:00
										 |  |  |   | 'retryMessageSend' | 
					
						
							| 
									
										
										
										
											2022-12-14 13:12:04 -05:00
										 |  |  |   | 'saveAttachment' | 
					
						
							| 
									
										
										
										
											2024-10-24 07:44:12 +10:00
										 |  |  |   | 'saveAttachments' | 
					
						
							| 
									
										
										
										
											2022-12-09 01:08:55 -05:00
										 |  |  |   | 'showContactModal' | 
					
						
							| 
									
										
										
										
											2022-12-14 11:05:32 -08:00
										 |  |  |   | 'showConversation' | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |   | 'showEditHistoryModal' | 
					
						
							| 
									
										
										
										
											2024-10-24 07:44:12 +10:00
										 |  |  |   | 'showAttachmentDownloadStillInProgressToast' | 
					
						
							| 
									
										
										
										
											2022-12-19 17:04:47 -08:00
										 |  |  |   | 'showExpiredIncomingTapToViewToast' | 
					
						
							|  |  |  |   | 'showExpiredOutgoingTapToViewToast' | 
					
						
							| 
									
										
										
										
											2022-12-09 21:02:22 -05:00
										 |  |  |   | 'showLightbox' | 
					
						
							|  |  |  |   | 'showLightboxForViewOnceMedia' | 
					
						
							| 
									
										
										
										
											2023-04-10 09:31:45 -07:00
										 |  |  |   | 'showSpoiler' | 
					
						
							| 
									
										
										
										
											2022-12-20 19:25:10 -08:00
										 |  |  |   | 'startConversation' | 
					
						
							| 
									
										
										
										
											2022-07-06 15:06:20 -04:00
										 |  |  |   | 'viewStory' | 
					
						
							| 
									
										
										
										
											2022-12-09 00:53:19 -05:00
										 |  |  | > & { | 
					
						
							|  |  |  |   toggleSafetyNumberModal: (contactId: string) => void; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2021-08-30 14:32:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-20 19:25:10 -08:00
										 |  |  | export type Props = PropsData & PropsSmartActions & PropsReduxActions; | 
					
						
							| 
									
										
										
										
											2021-08-30 14:32:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  | const contactSortCollator = new Intl.Collator(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-14 12:51:27 -07:00
										 |  |  | const _keyForError = (error: Error): string => { | 
					
						
							|  |  |  |   return `${error.name}-${error.message}`; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  | export function MessageDetail({ | 
					
						
							|  |  |  |   contacts, | 
					
						
							|  |  |  |   errors, | 
					
						
							|  |  |  |   message, | 
					
						
							|  |  |  |   receivedAt, | 
					
						
							|  |  |  |   sentAt, | 
					
						
							|  |  |  |   checkForAccount, | 
					
						
							|  |  |  |   clearTargetedMessage, | 
					
						
							|  |  |  |   contactNameColor, | 
					
						
							|  |  |  |   doubleCheckMissingQuoteReference, | 
					
						
							|  |  |  |   getPreferredBadge, | 
					
						
							|  |  |  |   i18n, | 
					
						
							|  |  |  |   interactionMode, | 
					
						
							|  |  |  |   kickOffAttachmentDownload, | 
					
						
							|  |  |  |   markAttachmentAsCorrupted, | 
					
						
							|  |  |  |   messageExpanded, | 
					
						
							|  |  |  |   openGiftBadge, | 
					
						
							|  |  |  |   platform, | 
					
						
							|  |  |  |   pushPanelForConversation, | 
					
						
							| 
									
										
										
										
											2023-04-20 12:31:59 -04:00
										 |  |  |   retryMessageSend, | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |   renderAudioAttachment, | 
					
						
							|  |  |  |   saveAttachment, | 
					
						
							| 
									
										
										
										
											2024-10-24 07:44:12 +10:00
										 |  |  |   saveAttachments, | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |   showContactModal, | 
					
						
							|  |  |  |   showConversation, | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |   showEditHistoryModal, | 
					
						
							| 
									
										
										
										
											2024-10-24 07:44:12 +10:00
										 |  |  |   showAttachmentDownloadStillInProgressToast, | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |   showExpiredIncomingTapToViewToast, | 
					
						
							|  |  |  |   showExpiredOutgoingTapToViewToast, | 
					
						
							|  |  |  |   showLightbox, | 
					
						
							|  |  |  |   showLightboxForViewOnceMedia, | 
					
						
							|  |  |  |   showSpoiler, | 
					
						
							|  |  |  |   startConversation, | 
					
						
							|  |  |  |   theme, | 
					
						
							|  |  |  |   toggleSafetyNumberModal, | 
					
						
							|  |  |  |   viewStory, | 
					
						
							|  |  |  | }: Props): JSX.Element { | 
					
						
							| 
									
										
										
										
											2023-09-12 12:52:16 -07:00
										 |  |  |   const messageDetailRef = useRef<HTMLDivElement>(null); | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   function renderAvatar(contact: Contact): JSX.Element { | 
					
						
							| 
									
										
										
										
											2020-07-23 18:35:32 -07:00
										 |  |  |     const { | 
					
						
							| 
									
										
										
										
											2021-05-04 18:19:36 -05:00
										 |  |  |       acceptedMessageRequest, | 
					
						
							| 
									
										
										
										
											2024-07-11 12:44:09 -07:00
										 |  |  |       avatarUrl, | 
					
						
							| 
									
										
										
										
											2021-11-16 09:53:41 -06:00
										 |  |  |       badges, | 
					
						
							| 
									
										
										
										
											2020-07-23 18:35:32 -07:00
										 |  |  |       color, | 
					
						
							| 
									
										
										
										
											2021-05-07 17:21:10 -05:00
										 |  |  |       isMe, | 
					
						
							| 
									
										
										
										
											2021-05-04 18:19:36 -05:00
										 |  |  |       phoneNumber, | 
					
						
							| 
									
										
										
										
											2020-07-23 18:35:32 -07:00
										 |  |  |       profileName, | 
					
						
							| 
									
										
										
										
											2021-05-04 18:19:36 -05:00
										 |  |  |       sharedGroupNames, | 
					
						
							| 
									
										
										
										
											2020-07-23 18:35:32 -07:00
										 |  |  |       title, | 
					
						
							| 
									
										
										
										
											2024-07-11 12:44:09 -07:00
										 |  |  |       unblurredAvatarUrl, | 
					
						
							| 
									
										
										
										
											2020-07-23 18:35:32 -07:00
										 |  |  |     } = contact; | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return ( | 
					
						
							| 
									
										
										
										
											2018-09-26 17:23:17 -07:00
										 |  |  |       <Avatar | 
					
						
							| 
									
										
										
										
											2021-05-04 18:19:36 -05:00
										 |  |  |         acceptedMessageRequest={acceptedMessageRequest} | 
					
						
							| 
									
										
										
										
											2024-07-11 12:44:09 -07:00
										 |  |  |         avatarUrl={avatarUrl} | 
					
						
							| 
									
										
										
										
											2021-11-16 09:53:41 -06:00
										 |  |  |         badge={getPreferredBadge(badges)} | 
					
						
							| 
									
										
										
										
											2018-09-26 17:23:17 -07:00
										 |  |  |         color={color} | 
					
						
							|  |  |  |         conversationType="direct" | 
					
						
							|  |  |  |         i18n={i18n} | 
					
						
							| 
									
										
										
										
											2021-05-07 17:21:10 -05:00
										 |  |  |         isMe={isMe} | 
					
						
							| 
									
										
										
										
											2018-09-26 17:23:17 -07:00
										 |  |  |         phoneNumber={phoneNumber} | 
					
						
							|  |  |  |         profileName={profileName} | 
					
						
							| 
									
										
										
										
											2021-11-16 09:53:41 -06:00
										 |  |  |         theme={theme} | 
					
						
							| 
									
										
										
										
											2020-07-23 18:35:32 -07:00
										 |  |  |         title={title} | 
					
						
							| 
									
										
										
										
											2021-05-04 18:19:36 -05:00
										 |  |  |         sharedGroupNames={sharedGroupNames} | 
					
						
							| 
									
										
										
										
											2022-12-09 13:37:45 -07:00
										 |  |  |         size={AvatarSize.THIRTY_TWO} | 
					
						
							| 
									
										
										
										
											2024-07-11 12:44:09 -07:00
										 |  |  |         unblurredAvatarUrl={unblurredAvatarUrl} | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |       /> | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |   function renderContact(contact: Contact): JSX.Element { | 
					
						
							|  |  |  |     const contactErrors = contact.errors || []; | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const errorComponent = contact.isOutgoingKeyError ? ( | 
					
						
							|  |  |  |       <div className="module-message-detail__contact__error-buttons"> | 
					
						
							|  |  |  |         <button | 
					
						
							| 
									
										
										
										
											2020-09-14 12:51:27 -07:00
										 |  |  |           type="button" | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |           className="module-message-detail__contact__show-safety-number" | 
					
						
							| 
									
										
										
										
											2022-12-09 00:53:19 -05:00
										 |  |  |           onClick={() => toggleSafetyNumberModal(contact.id)} | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |         > | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |           {i18n('icu:showSafetyNumber')} | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |         </button> | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     ) : null; | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |     const unidentifiedDeliveryComponent = contact.isUnidentifiedDelivery ? ( | 
					
						
							|  |  |  |       <div className="module-message-detail__contact__unidentified-delivery-icon" /> | 
					
						
							|  |  |  |     ) : null; | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return ( | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  |       <div key={contact.id} className="module-message-detail__contact"> | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |         {renderAvatar(contact)} | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |         <div className="module-message-detail__contact__text"> | 
					
						
							|  |  |  |           <div className="module-message-detail__contact__name"> | 
					
						
							| 
									
										
										
										
											2021-09-16 11:15:43 -05:00
										 |  |  |             <ContactName title={contact.title} /> | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |           </div> | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |           {contactErrors.map(contactError => ( | 
					
						
							| 
									
										
										
										
											2020-09-14 12:51:27 -07:00
										 |  |  |             <div | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |               key={_keyForError(contactError)} | 
					
						
							| 
									
										
										
										
											2020-09-14 12:51:27 -07:00
										 |  |  |               className="module-message-detail__contact__error" | 
					
						
							|  |  |  |             > | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |               {contactError.message} | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |             </div> | 
					
						
							|  |  |  |           ))} | 
					
						
							|  |  |  |         </div> | 
					
						
							|  |  |  |         {errorComponent} | 
					
						
							| 
									
										
										
										
											2018-10-17 18:01:21 -07:00
										 |  |  |         {unidentifiedDeliveryComponent} | 
					
						
							| 
									
										
										
										
											2021-10-12 19:40:42 -04:00
										 |  |  |         {contact.statusTimestamp && ( | 
					
						
							| 
									
										
										
										
											2022-01-26 17:05:26 -06:00
										 |  |  |           <Time | 
					
						
							|  |  |  |             className="module-message-detail__status-timestamp" | 
					
						
							| 
									
										
										
										
											2021-10-12 19:40:42 -04:00
										 |  |  |             timestamp={contact.statusTimestamp} | 
					
						
							| 
									
										
										
										
											2022-01-26 17:05:26 -06:00
										 |  |  |           > | 
					
						
							|  |  |  |             {formatDateTimeLong(i18n, contact.statusTimestamp)} | 
					
						
							|  |  |  |           </Time> | 
					
						
							| 
									
										
										
										
											2021-10-12 19:40:42 -04:00
										 |  |  |         )} | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |       </div> | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |   function renderContactGroupHeaderText( | 
					
						
							| 
									
										
										
										
											2024-09-12 16:48:27 -07:00
										 |  |  |     sendStatus: undefined | VisibleSendStatus | 
					
						
							| 
									
										
										
										
											2023-03-29 10:15:54 -07:00
										 |  |  |   ): string { | 
					
						
							|  |  |  |     if (sendStatus === undefined) { | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |       return i18n('icu:from'); | 
					
						
							| 
									
										
										
										
											2023-03-29 10:15:54 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (sendStatus) { | 
					
						
							|  |  |  |       case SendStatus.Failed: | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |         return i18n('icu:MessageDetailsHeader--Failed'); | 
					
						
							| 
									
										
										
										
											2023-03-29 10:15:54 -07:00
										 |  |  |       case SendStatus.Pending: | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |         return i18n('icu:MessageDetailsHeader--Pending'); | 
					
						
							| 
									
										
										
										
											2023-03-29 10:15:54 -07:00
										 |  |  |       case SendStatus.Sent: | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |         return i18n('icu:MessageDetailsHeader--Sent'); | 
					
						
							| 
									
										
										
										
											2023-03-29 10:15:54 -07:00
										 |  |  |       case SendStatus.Delivered: | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |         return i18n('icu:MessageDetailsHeader--Delivered'); | 
					
						
							| 
									
										
										
										
											2023-03-29 10:15:54 -07:00
										 |  |  |       case SendStatus.Read: | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |         return i18n('icu:MessageDetailsHeader--Read'); | 
					
						
							| 
									
										
										
										
											2023-03-29 10:15:54 -07:00
										 |  |  |       case SendStatus.Viewed: | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |         return i18n('icu:MessageDetailsHeader--Viewed'); | 
					
						
							| 
									
										
										
										
											2023-03-29 10:15:54 -07:00
										 |  |  |       default: | 
					
						
							|  |  |  |         throw missingCaseError(sendStatus); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |   function renderContactGroup( | 
					
						
							| 
									
										
										
										
											2024-09-12 16:48:27 -07:00
										 |  |  |     sendStatus: undefined | VisibleSendStatus, | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |     statusContacts: undefined | ReadonlyArray<Contact> | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  |   ): ReactNode { | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |     if (!statusContacts || !statusContacts.length) { | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |       return null; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |     const sortedContacts = [...statusContacts].sort((a, b) => | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  |       contactSortCollator.compare(a.title, b.title) | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |     const headerText = renderContactGroupHeaderText(sendStatus); | 
					
						
							| 
									
										
										
										
											2023-03-29 10:15:54 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  |     return ( | 
					
						
							| 
									
										
										
										
											2023-03-29 10:15:54 -07:00
										 |  |  |       <div key={headerText} className="module-message-detail__contact-group"> | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  |         <div | 
					
						
							|  |  |  |           className={classNames( | 
					
						
							|  |  |  |             'module-message-detail__contact-group__header', | 
					
						
							|  |  |  |             sendStatus && | 
					
						
							|  |  |  |               `module-message-detail__contact-group__header--${sendStatus}` | 
					
						
							|  |  |  |           )} | 
					
						
							|  |  |  |         > | 
					
						
							| 
									
										
										
										
											2023-03-29 10:15:54 -07:00
										 |  |  |           {headerText} | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  |         </div> | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |         {sortedContacts.map(contact => renderContact(contact))} | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  |       </div> | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |   function renderContacts(): ReactChild { | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05: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 14:29:13 -07:00
										 |  |  |     return ( | 
					
						
							|  |  |  |       <div className="module-message-detail__contact-container"> | 
					
						
							| 
									
										
										
										
											2024-09-12 16:48:27 -07:00
										 |  |  |         {( | 
					
						
							|  |  |  |           [ | 
					
						
							|  |  |  |             undefined, | 
					
						
							|  |  |  |             SendStatus.Failed, | 
					
						
							|  |  |  |             SendStatus.Viewed, | 
					
						
							|  |  |  |             SendStatus.Read, | 
					
						
							|  |  |  |             SendStatus.Delivered, | 
					
						
							|  |  |  |             SendStatus.Sent, | 
					
						
							|  |  |  |             SendStatus.Pending, | 
					
						
							|  |  |  |           ] as Array<VisibleSendStatus | undefined> | 
					
						
							|  |  |  |         ).map(sendStatus => | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |           renderContactGroup(sendStatus, contactsBySendStatus.get(sendStatus)) | 
					
						
							| 
									
										
										
										
											2021-07-20 14:56:50 -05:00
										 |  |  |         )} | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |       </div> | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |   const timeRemaining = message.expirationTimestamp | 
					
						
							|  |  |  |     ? DurationInSeconds.fromMillis(message.expirationTimestamp - Date.now()) | 
					
						
							|  |  |  |     : undefined; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							| 
									
										
										
										
											2023-09-12 12:52:16 -07:00
										 |  |  |     <div className="module-message-detail" ref={messageDetailRef}> | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |       <PanelSection> | 
					
						
							| 
									
										
										
										
											2023-09-12 12:52:16 -07:00
										 |  |  |         <div className="module-message-detail__message-container"> | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |           <Message | 
					
						
							|  |  |  |             {...message} | 
					
						
							|  |  |  |             renderingContext="conversation/MessageDetail" | 
					
						
							|  |  |  |             checkForAccount={checkForAccount} | 
					
						
							|  |  |  |             clearTargetedMessage={clearTargetedMessage} | 
					
						
							|  |  |  |             contactNameColor={contactNameColor} | 
					
						
							| 
									
										
										
										
											2023-09-12 12:52:16 -07:00
										 |  |  |             containerElementRef={messageDetailRef} | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |             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} | 
					
						
							|  |  |  |             retryMessageSend={retryMessageSend} | 
					
						
							|  |  |  |             renderAudioAttachment={renderAudioAttachment} | 
					
						
							|  |  |  |             saveAttachment={saveAttachment} | 
					
						
							| 
									
										
										
										
											2024-10-24 07:44:12 +10:00
										 |  |  |             saveAttachments={saveAttachments} | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |             shouldCollapseAbove={false} | 
					
						
							|  |  |  |             shouldCollapseBelow={false} | 
					
						
							|  |  |  |             shouldHideMetadata={false} | 
					
						
							|  |  |  |             showConversation={showConversation} | 
					
						
							|  |  |  |             showSpoiler={showSpoiler} | 
					
						
							|  |  |  |             scrollToQuotedMessage={() => { | 
					
						
							|  |  |  |               log.warn('MessageDetail: scrollToQuotedMessage called!'); | 
					
						
							|  |  |  |             }} | 
					
						
							|  |  |  |             showContactModal={showContactModal} | 
					
						
							| 
									
										
										
										
											2024-10-24 07:44:12 +10:00
										 |  |  |             showAttachmentDownloadStillInProgressToast={ | 
					
						
							|  |  |  |               showAttachmentDownloadStillInProgressToast | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |             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)}> | 
					
						
							|  |  |  |                 <td className="module-message-detail__label"> | 
					
						
							|  |  |  |                   {i18n('icu:error')} | 
					
						
							|  |  |  |                 </td> | 
					
						
							|  |  |  |                 <td> | 
					
						
							|  |  |  |                   {' '} | 
					
						
							|  |  |  |                   <span className="error-message">{error.message}</span>{' '} | 
					
						
							|  |  |  |                 </td> | 
					
						
							|  |  |  |               </tr> | 
					
						
							|  |  |  |             ))} | 
					
						
							|  |  |  |             <tr> | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |               <td className="module-message-detail__label"> | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |                 {i18n('icu:sent')} | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |               </td> | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |               <td> | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |                 <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)} | 
					
						
							|  |  |  |                     </Time>{' '} | 
					
						
							|  |  |  |                     <span className="module-message-detail__unix-timestamp"> | 
					
						
							|  |  |  |                       ({sentAt}) | 
					
						
							|  |  |  |                     </span> | 
					
						
							|  |  |  |                   </> | 
					
						
							|  |  |  |                 </ContextMenu> | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |               </td> | 
					
						
							|  |  |  |             </tr> | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |             {receivedAt && message.direction === 'incoming' ? ( | 
					
						
							|  |  |  |               <tr> | 
					
						
							|  |  |  |                 <td className="module-message-detail__label"> | 
					
						
							|  |  |  |                   {i18n('icu:received')} | 
					
						
							|  |  |  |                 </td> | 
					
						
							|  |  |  |                 <td> | 
					
						
							|  |  |  |                   <Time timestamp={receivedAt}> | 
					
						
							|  |  |  |                     {formatDateTimeLong(i18n, receivedAt)} | 
					
						
							| 
									
										
										
										
											2022-01-26 17:05:26 -06:00
										 |  |  |                   </Time>{' '} | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |                   <span className="module-message-detail__unix-timestamp"> | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |                     ({receivedAt}) | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  |                   </span> | 
					
						
							| 
									
										
										
										
											2023-06-15 11:34:20 -07:00
										 |  |  |                 </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> | 
					
						
							|  |  |  |       </PanelSection> | 
					
						
							|  |  |  |       {message.isEditedMessage && ( | 
					
						
							|  |  |  |         <PanelSection> | 
					
						
							|  |  |  |           <PanelRow | 
					
						
							|  |  |  |             icon={ | 
					
						
							|  |  |  |               <ConversationDetailsIcon | 
					
						
							|  |  |  |                 ariaLabel={i18n('icu:MessageDetail__view-edits')} | 
					
						
							|  |  |  |                 icon={IconType.edit} | 
					
						
							|  |  |  |               /> | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             label={i18n('icu:MessageDetail__view-edits')} | 
					
						
							|  |  |  |             onClick={() => { | 
					
						
							|  |  |  |               showEditHistoryModal?.(message.id); | 
					
						
							|  |  |  |             }} | 
					
						
							|  |  |  |           /> | 
					
						
							|  |  |  |         </PanelSection> | 
					
						
							|  |  |  |       )} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       <PanelSection>{renderContacts()}</PanelSection> | 
					
						
							| 
									
										
										
										
											2023-04-12 16:17:56 -07:00
										 |  |  |     </div> | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2018-07-09 14:29:13 -07:00
										 |  |  | } |