signal-desktop/ts/components/conversation/Message.tsx

2974 lines
84 KiB
TypeScript
Raw Normal View History

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
/* eslint-disable react/jsx-pascal-case */
2023-03-20 22:23:53 +00:00
import type {
DetailedHTMLProps,
HTMLAttributes,
ReactNode,
RefObject,
} from 'react';
import React from 'react';
2022-11-04 13:22:07 +00:00
import { createPortal } from 'react-dom';
import classNames from 'classnames';
2022-05-11 20:59:58 +00:00
import getDirection from 'direction';
import { drop, groupBy, noop, orderBy, take, unescape } from 'lodash';
2020-01-17 22:23:19 +00:00
import { Manager, Popper, Reference } from 'react-popper';
import type { PreventOverflowModifier } from '@popperjs/core/lib/modifiers/preventOverflow';
import type { ReadonlyDeep } from 'type-fest';
import type {
ConversationType,
ConversationTypeType,
InteractionModeType,
PushPanelForConversationActionType,
2022-12-14 18:12:04 +00:00
SaveAttachmentActionCreatorType,
ShowConversationType,
} from '../../state/ducks/conversations';
2022-07-06 19:06:20 +00:00
import type { ViewStoryActionCreatorType } from '../../state/ducks/stories';
2022-10-03 23:43:44 +00:00
import type { ReadStatus } from '../../messages/MessageReadStatus';
import { Avatar, AvatarSize } from '../Avatar';
import { AvatarSpacer } from '../AvatarSpacer';
import { Spinner } from '../Spinner';
2022-11-04 13:22:07 +00:00
import { MessageBodyReadMore } from './MessageBodyReadMore';
import { MessageMetadata } from './MessageMetadata';
import { MessageTextMetadataSpacer } from './MessageTextMetadataSpacer';
2019-01-14 21:49:58 +00:00
import { ImageGrid } from './ImageGrid';
2021-04-27 22:11:59 +00:00
import { GIF } from './GIF';
import { CurveType, Image } from './Image';
2019-01-14 21:49:58 +00:00
import { ContactName } from './ContactName';
2024-05-23 21:06:41 +00:00
import type { QuotedAttachmentForUIType } from './Quote';
import { Quote } from './Quote';
2019-01-14 21:49:58 +00:00
import { EmbeddedContact } from './EmbeddedContact';
import type { OwnProps as ReactionViewerProps } from './ReactionViewer';
import { ReactionViewer } from './ReactionViewer';
2020-01-17 22:23:19 +00:00
import { Emoji } from '../emoji/Emoji';
2020-09-28 23:46:31 +00:00
import { LinkPreviewDate } from './LinkPreviewDate';
import type { LinkPreviewType } from '../../types/message/LinkPreviews';
import { shouldUseFullSizeLinkPreviewImage } from '../../linkPreviews/shouldUseFullSizeLinkPreviewImage';
2022-11-04 13:22:07 +00:00
import type { WidthBreakpoint } from '../_util';
2022-05-11 20:59:58 +00:00
import { OutgoingGiftBadgeModal } from '../OutgoingGiftBadgeModal';
import * as log from '../../logging/log';
2022-07-06 19:06:20 +00:00
import { StoryViewModeType } from '../../types/Stories';
import type { AttachmentType } from '../../types/Attachment';
import {
2019-01-14 21:49:58 +00:00
canDisplayImage,
getExtensionForDisplay,
getGridDimensions,
2019-01-16 03:03:56 +00:00
getImageDimensions,
hasImage,
2022-03-29 01:10:08 +00:00
isDownloaded,
hasVideoScreenshot,
2019-01-14 21:49:58 +00:00
isAudio,
isImage,
2019-01-16 03:03:56 +00:00
isImageAttachment,
isVideo,
2021-04-27 22:11:59 +00:00
isGIF,
2022-10-03 23:43:44 +00:00
isPlayed,
2020-09-14 19:51:27 +00:00
} from '../../types/Attachment';
import type { EmbeddedContactType } from '../../types/EmbeddedContact';
2019-01-14 21:49:58 +00:00
import { getIncrement } from '../../util/timer';
import { clearTimeoutIfNecessary } from '../../util/clearTimeoutIfNecessary';
2018-10-04 01:12:42 +00:00
import { isFileDangerous } from '../../util/isFileDangerous';
import { missingCaseError } from '../../util/missingCaseError';
import type { HydratedBodyRangesType } from '../../types/BodyRange';
import type { LocalizerType, ThemeType } from '../../types/Util';
2022-05-11 20:59:58 +00:00
2021-11-17 21:11:46 +00:00
import type { PreferredBadgeSelectorType } from '../../state/selectors/badges';
import type {
2021-05-28 16:15:17 +00:00
ContactNameColorType,
ConversationColorType,
CustomColorType,
} from '../../types/Colors';
import { createRefMerger } from '../../util/refMerger';
import { emojiToData, getEmojiCount, hasNonEmojiText } from '../emoji/lib';
2021-05-28 16:15:17 +00:00
import { getCustomColorStyle } from '../../util/getCustomColorStyle';
import type { ServiceIdString } from '../../types/ServiceId';
2022-05-11 20:59:58 +00:00
import { DAY, HOUR, MINUTE, SECOND } from '../../util/durations';
import { BadgeImageTheme } from '../../badges/BadgeImageTheme';
import { getBadgeImageFileLocalPath } from '../../badges/getBadgeImageFileLocalPath';
import { handleOutsideClick } from '../../util/handleOutsideClick';
import { isPaymentNotificationEvent } from '../../types/Payment';
2022-11-30 21:47:54 +00:00
import type { AnyPaymentEvent } from '../../types/Payment';
import { getPaymentEventDescription } from '../../messages/helpers';
import { PanelType } from '../../types/Panels';
import { openLinkInWebBrowser } from '../../util/openLinkInWebBrowser';
import { RenderLocation } from './MessageTextRenderer';
2023-04-20 17:03:43 +00:00
import { UserText } from '../UserText';
import { getColorForCallLink } from '../../util/getColorForCallLink';
import { getKeyFromCallLink } from '../../util/callLinks';
import { InAnotherCallTooltip } from './InAnotherCallTooltip';
2022-12-19 22:33:55 +00:00
const GUESS_METADATA_WIDTH_TIMESTAMP_SIZE = 16;
const GUESS_METADATA_WIDTH_EXPIRE_TIMER_SIZE = 18;
2024-06-13 23:26:26 +00:00
const GUESS_METADATA_WIDTH_SMS_SIZE = 18;
const GUESS_METADATA_WIDTH_EDITED_SIZE = 40;
const GUESS_METADATA_WIDTH_OUTGOING_SIZE: Record<MessageStatusType, number> = {
delivered: 24,
error: 24,
paused: 18,
'partial-sent': 24,
read: 24,
sending: 18,
sent: 24,
viewed: 24,
};
const EXPIRATION_CHECK_MINIMUM = 2000;
const EXPIRED_DELAY = 600;
const GROUP_AVATAR_SIZE = AvatarSize.TWENTY_EIGHT;
const STICKER_SIZE = 200;
2021-04-27 22:11:59 +00:00
const GIF_SIZE = 300;
// Note: this needs to match the animation time
2023-03-20 22:23:53 +00:00
const TARGETED_TIMEOUT = 1200;
const SENT_STATUSES = new Set<MessageStatusType>([
'delivered',
'read',
'sent',
'viewed',
]);
2022-05-11 20:59:58 +00:00
const GIFT_BADGE_UPDATE_INTERVAL = 30 * SECOND;
enum MetadataPlacement {
NotRendered,
RenderedByMessageAudioComponent,
InlineWithText,
Bottom,
}
2019-01-16 03:03:56 +00:00
export enum TextDirection {
LeftToRight = 'LeftToRight',
RightToLeft = 'RightToLeft',
Default = 'Default',
None = 'None',
}
2023-04-20 17:03:43 +00:00
const TextDirectionToDirAttribute = {
[TextDirection.LeftToRight]: 'ltr',
[TextDirection.RightToLeft]: 'rtl',
[TextDirection.Default]: 'auto',
[TextDirection.None]: 'auto',
};
2020-08-27 18:10:35 +00:00
export const MessageStatuses = [
2020-08-27 16:57:12 +00:00
'delivered',
'error',
'paused',
2020-08-27 16:57:12 +00:00
'partial-sent',
'read',
'sending',
'sent',
'viewed',
2020-08-27 16:57:12 +00:00
] as const;
2024-07-24 00:31:40 +00:00
export type MessageStatusType = (typeof MessageStatuses)[number];
2020-08-27 18:10:35 +00:00
export const Directions = ['incoming', 'outgoing'] as const;
2024-07-24 00:31:40 +00:00
export type DirectionType = (typeof Directions)[number];
2020-08-27 18:10:35 +00:00
export type AudioAttachmentProps = {
renderingContext: string;
i18n: LocalizerType;
buttonRef: React.RefObject<HTMLButtonElement>;
theme: ThemeType | undefined;
attachment: AttachmentType;
collapseMetadata: boolean;
withContentAbove: boolean;
withContentBelow: boolean;
direction: DirectionType;
expirationLength?: number;
expirationTimestamp?: number;
id: string;
conversationId: string;
played: boolean;
pushPanelForConversation: PushPanelForConversationActionType;
status?: MessageStatusType;
textPending?: boolean;
timestamp: number;
kickOffAttachmentDownload(): void;
onCorrupted(): void;
};
2022-05-11 20:59:58 +00:00
export enum GiftBadgeStates {
Unopened = 'Unopened',
2022-05-16 19:54:38 +00:00
Opened = 'Opened',
2022-05-11 20:59:58 +00:00
Redeemed = 'Redeemed',
Failed = 'Failed',
2022-05-11 20:59:58 +00:00
}
export type GiftBadgeType =
| {
state:
| GiftBadgeStates.Unopened
| GiftBadgeStates.Opened
| GiftBadgeStates.Redeemed;
expiration: number;
id: string | undefined;
level: number;
}
| {
state: GiftBadgeStates.Failed;
};
2022-05-11 20:59:58 +00:00
export type PropsData = {
id: string;
renderingContext: string;
2021-05-28 16:15:17 +00:00
contactNameColor?: ContactNameColorType;
conversationColor: ConversationColorType;
2022-05-11 20:59:58 +00:00
conversationTitle: string;
2021-05-28 16:15:17 +00:00
customColor?: CustomColorType;
2019-11-07 21:36:16 +00:00
conversationId: string;
displayLimit?: number;
activeCallConversationId?: string;
text?: string;
textDirection: TextDirection;
textAttachment?: AttachmentType;
2023-03-27 23:48:57 +00:00
isEditedMessage?: boolean;
isSticker?: boolean;
2023-03-20 22:23:53 +00:00
isTargeted?: boolean;
isTargetedCounter?: number;
isSelected: boolean;
isSelectMode: boolean;
2024-06-13 23:26:26 +00:00
isSMS: boolean;
isSpoilerExpanded?: Record<number, boolean>;
2020-08-27 18:10:35 +00:00
direction: DirectionType;
timestamp: number;
receivedAtMS?: number;
2020-08-27 18:10:35 +00:00
status?: MessageStatusType;
contact?: ReadonlyDeep<EmbeddedContactType>;
2021-04-27 19:55:21 +00:00
author: Pick<
ConversationType,
| 'acceptedMessageRequest'
2024-07-11 19:44:09 +00:00
| 'avatarUrl'
| 'badges'
2021-04-27 19:55:21 +00:00
| 'color'
| 'id'
| 'isMe'
2021-04-27 19:55:21 +00:00
| 'phoneNumber'
| 'profileName'
| 'sharedGroupNames'
2021-04-27 19:55:21 +00:00
| 'title'
2024-07-11 19:44:09 +00:00
| 'unblurredAvatarUrl'
2021-04-27 19:55:21 +00:00
>;
conversationType: ConversationTypeType;
attachments?: ReadonlyArray<AttachmentType>;
2022-05-11 20:59:58 +00:00
giftBadge?: GiftBadgeType;
2022-11-30 21:47:54 +00:00
payment?: AnyPaymentEvent;
quote?: {
2021-05-28 16:15:17 +00:00
conversationColor: ConversationColorType;
2022-11-30 21:47:54 +00:00
conversationTitle: string;
2021-05-28 16:15:17 +00:00
customColor?: CustomColorType;
text: string;
2024-05-23 21:06:41 +00:00
rawAttachment?: QuotedAttachmentForUIType;
2022-11-30 21:47:54 +00:00
payment?: AnyPaymentEvent;
isFromMe: boolean;
sentAt: number;
authorId: string;
2020-07-24 01:35:32 +00:00
authorPhoneNumber?: string;
authorProfileName?: string;
2020-07-24 01:35:32 +00:00
authorTitle: string;
authorName?: string;
2022-11-10 04:59:36 +00:00
bodyRanges?: HydratedBodyRangesType;
referencedMessageNotFound: boolean;
isViewOnce: boolean;
2022-05-11 20:59:58 +00:00
isGiftBadge: boolean;
};
2022-03-16 17:30:14 +00:00
storyReplyContext?: {
authorTitle: string;
conversationColor: ConversationColorType;
customColor?: CustomColorType;
emoji?: string;
2022-03-16 17:30:14 +00:00
isFromMe: boolean;
2024-05-23 21:06:41 +00:00
rawAttachment?: QuotedAttachmentForUIType;
2022-07-06 19:06:20 +00:00
storyId?: string;
text: string;
2022-03-16 17:30:14 +00:00
};
previews: ReadonlyArray<LinkPreviewType>;
2019-06-26 19:33:13 +00:00
isTapToView?: boolean;
isTapToViewExpired?: boolean;
isTapToViewError?: boolean;
readStatus?: ReadStatus;
expirationLength?: number;
expirationTimestamp?: number;
2020-01-17 22:23:19 +00:00
reactions?: ReactionViewerProps['reactions'];
2020-04-29 21:24:12 +00:00
deletedForEveryone?: boolean;
attachmentDroppedDueToSize?: boolean;
2020-04-29 21:24:12 +00:00
canDeleteForEveryone: boolean;
isBlocked: boolean;
isMessageRequestAccepted: boolean;
2022-11-10 04:59:36 +00:00
bodyRanges?: HydratedBodyRangesType;
2022-11-04 13:22:07 +00:00
2022-12-19 22:33:55 +00:00
renderMenu?: () => JSX.Element | undefined;
2022-11-04 13:22:07 +00:00
onKeyDown?: (event: React.KeyboardEvent<HTMLDivElement>) => void;
2022-12-19 22:33:55 +00:00
item?: never;
};
export type PropsHousekeeping = {
containerElementRef: RefObject<HTMLElement>;
containerWidthBreakpoint: WidthBreakpoint;
disableScroll?: boolean;
2021-11-17 21:11:46 +00:00
getPreferredBadge: PreferredBadgeSelectorType;
i18n: LocalizerType;
interactionMode: InteractionModeType;
2023-04-03 20:16:27 +00:00
platform: string;
renderAudioAttachment: (props: AudioAttachmentProps) => JSX.Element;
shouldCollapseAbove: boolean;
shouldCollapseBelow: boolean;
shouldHideMetadata: boolean;
2022-11-04 13:22:07 +00:00
onContextMenu?: (event: React.MouseEvent<HTMLDivElement>) => void;
theme: ThemeType;
};
export type PropsActions = {
2023-03-20 22:23:53 +00:00
clearTargetedMessage: () => unknown;
doubleCheckMissingQuoteReference: (messageId: string) => unknown;
messageExpanded: (id: string, displayLimit: number) => unknown;
checkForAccount: (phoneNumber: string) => unknown;
2023-08-16 20:54:39 +00:00
startConversation: (e164: string, serviceId: ServiceIdString) => void;
showConversation: ShowConversationType;
2022-05-11 20:59:58 +00:00
openGiftBadge: (messageId: string) => void;
pushPanelForConversation: PushPanelForConversationActionType;
retryMessageSend: (messageId: string) => unknown;
showContactModal: (contactId: string, conversationId?: string) => void;
showSpoiler: (messageId: string, data: Record<number, boolean>) => void;
2020-01-08 17:44:54 +00:00
2021-01-29 22:58:28 +00:00
kickOffAttachmentDownload: (options: {
attachment: AttachmentType;
messageId: string;
}) => void;
markAttachmentAsCorrupted: (options: {
attachment: AttachmentType;
messageId: string;
}) => void;
2022-12-14 18:12:04 +00:00
saveAttachment: SaveAttachmentActionCreatorType;
2022-12-10 02:02:22 +00:00
showLightbox: (options: {
2020-01-08 17:44:54 +00:00
attachment: AttachmentType;
messageId: string;
}) => void;
2022-12-10 02:02:22 +00:00
showLightboxForViewOnceMedia: (messageId: string) => unknown;
scrollToQuotedMessage: (options: {
authorId: string;
conversationId: string;
sentAt: number;
}) => void;
2023-03-20 22:23:53 +00:00
targetMessage?: (messageId: string, conversationId: string) => unknown;
2023-03-27 23:48:57 +00:00
showEditHistoryModal?: (id: string) => unknown;
showExpiredIncomingTapToViewToast: () => unknown;
showExpiredOutgoingTapToViewToast: () => unknown;
2022-07-06 19:06:20 +00:00
viewStory: ViewStoryActionCreatorType;
2023-03-20 22:23:53 +00:00
onToggleSelect: (selected: boolean, shift: boolean) => void;
onReplyToMessage: () => void;
};
2022-11-04 13:22:07 +00:00
export type Props = PropsData & PropsHousekeeping & PropsActions;
type State = {
metadataWidth: number;
expiring: boolean;
expired: boolean;
imageBroken: boolean;
2023-03-20 22:23:53 +00:00
isTargeted?: boolean;
prevTargetedCounter?: number;
2020-01-17 22:23:19 +00:00
reactionViewerRoot: HTMLDivElement | null;
reactionViewerOutsideClickDestructor?: () => void;
2020-01-23 23:57:37 +00:00
2022-05-11 20:59:58 +00:00
giftBadgeCounter: number | null;
showOutgoingGiftBadgeModal: boolean;
hasDeleteForEveryoneTimerExpired: boolean;
};
2021-08-11 16:23:21 +00:00
export class Message extends React.PureComponent<Props, State> {
2020-01-17 22:23:19 +00:00
public focusRef: React.RefObject<HTMLDivElement> = React.createRef();
2020-09-14 19:51:27 +00:00
public audioButtonRef: React.RefObject<HTMLButtonElement> = React.createRef();
2021-11-11 22:43:05 +00:00
public reactionsContainerRef: React.RefObject<HTMLDivElement> =
React.createRef();
2020-09-14 19:51:27 +00:00
private hasSelectedTextRef: React.MutableRefObject<boolean> = {
current: false,
};
private metadataRef: React.RefObject<HTMLDivElement> = React.createRef();
2020-03-23 21:09:12 +00:00
public reactionsContainerRefMerger = createRefMerger();
2019-11-07 21:36:16 +00:00
2020-09-14 19:51:27 +00:00
public expirationCheckInterval: NodeJS.Timeout | undefined;
2022-05-11 20:59:58 +00:00
public giftBadgeInterval: NodeJS.Timeout | undefined;
2020-09-14 19:51:27 +00:00
public expiredTimeout: NodeJS.Timeout | undefined;
2023-03-20 22:23:53 +00:00
public targetedTimeout: NodeJS.Timeout | undefined;
public deleteForEveryoneTimeout: NodeJS.Timeout | undefined;
public constructor(props: Props) {
super(props);
this.state = {
metadataWidth: this.guessMetadataWidth(),
expiring: false,
expired: false,
imageBroken: false,
2023-03-20 22:23:53 +00:00
isTargeted: props.isTargeted,
prevTargetedCounter: props.isTargetedCounter,
2020-01-17 22:23:19 +00:00
reactionViewerRoot: null,
2020-01-23 23:57:37 +00:00
2022-05-11 20:59:58 +00:00
giftBadgeCounter: null,
showOutgoingGiftBadgeModal: false,
hasDeleteForEveryoneTimerExpired:
this.getTimeRemainingForDeleteForEveryone() <= 0,
};
}
public static getDerivedStateFromProps(props: Props, state: State): State {
2023-03-20 22:23:53 +00:00
if (!props.isTargeted) {
return {
...state,
2023-03-20 22:23:53 +00:00
isTargeted: false,
prevTargetedCounter: 0,
};
}
if (
2023-03-20 22:23:53 +00:00
props.isTargeted &&
props.isTargetedCounter !== state.prevTargetedCounter
) {
return {
...state,
2023-03-20 22:23:53 +00:00
isTargeted: props.isTargeted,
prevTargetedCounter: props.isTargetedCounter,
};
}
return state;
}
2021-01-27 21:15:43 +00:00
private hasReactions(): boolean {
const { reactions } = this.props;
return Boolean(reactions && reactions.length);
}
2022-11-04 13:22:07 +00:00
public handleFocus = (): void => {
2023-03-20 22:23:53 +00:00
const { interactionMode, isTargeted } = this.props;
2023-03-20 22:23:53 +00:00
if (interactionMode === 'keyboard' && !isTargeted) {
this.setTargeted();
}
2021-06-09 22:30:05 +00:00
};
2020-09-14 19:51:27 +00:00
public handleImageError = (): void => {
2019-11-07 21:36:16 +00:00
const { id } = this.props;
log.info(
2019-11-07 21:36:16 +00:00
`Message ${id}: Image failed to load; failing over to placeholder`
);
this.setState({
imageBroken: true,
});
};
2023-03-20 22:23:53 +00:00
public setTargeted = (): void => {
const { id, conversationId, targetMessage } = this.props;
2023-03-20 22:23:53 +00:00
if (targetMessage) {
targetMessage(id, conversationId);
}
2019-11-07 21:36:16 +00:00
};
2020-09-14 19:51:27 +00:00
public setFocus = (): void => {
const container = this.focusRef.current;
if (container && !container.contains(document.activeElement)) {
container.focus();
2019-11-07 21:36:16 +00:00
}
};
public override componentDidMount(): void {
2022-01-20 00:40:29 +00:00
const { conversationId } = this.props;
window.ConversationController?.onConvoMessageMount(conversationId);
2022-01-20 00:40:29 +00:00
2023-03-20 22:23:53 +00:00
this.startTargetedTimer();
this.startDeleteForEveryoneTimerIfApplicable();
2022-05-11 20:59:58 +00:00
this.startGiftBadgeInterval();
2023-03-20 22:23:53 +00:00
const { isTargeted } = this.props;
if (isTargeted) {
2019-11-07 21:36:16 +00:00
this.setFocus();
}
const { expirationLength } = this.props;
if (expirationLength) {
const increment = getIncrement(expirationLength);
const checkFrequency = Math.max(EXPIRATION_CHECK_MINIMUM, increment);
this.checkExpired();
this.expirationCheckInterval = setInterval(() => {
this.checkExpired();
}, checkFrequency);
}
const { contact, checkForAccount } = this.props;
2023-08-16 20:54:39 +00:00
if (contact && contact.firstNumber && !contact.serviceId) {
checkForAccount(contact.firstNumber);
}
document.addEventListener('selectionchange', this.handleSelectionChange);
}
public override componentWillUnmount(): void {
2023-03-20 22:23:53 +00:00
clearTimeoutIfNecessary(this.targetedTimeout);
clearTimeoutIfNecessary(this.expirationCheckInterval);
clearTimeoutIfNecessary(this.expiredTimeout);
clearTimeoutIfNecessary(this.deleteForEveryoneTimeout);
2022-05-11 20:59:58 +00:00
clearTimeoutIfNecessary(this.giftBadgeInterval);
2020-01-17 22:23:19 +00:00
this.toggleReactionViewer(true);
document.removeEventListener('selectionchange', this.handleSelectionChange);
}
public override componentDidUpdate(prevProps: Readonly<Props>): void {
2023-03-20 22:23:53 +00:00
const { isTargeted, status, timestamp } = this.props;
2020-09-14 19:51:27 +00:00
2023-03-20 22:23:53 +00:00
this.startTargetedTimer();
this.startDeleteForEveryoneTimerIfApplicable();
2023-03-20 22:23:53 +00:00
if (!prevProps.isTargeted && isTargeted) {
2019-11-07 21:36:16 +00:00
this.setFocus();
}
2019-11-07 21:36:16 +00:00
this.checkExpired();
2021-07-30 18:37:03 +00:00
if (
prevProps.status === 'sending' &&
(status === 'sent' ||
status === 'delivered' ||
status === 'read' ||
status === 'viewed')
) {
const delta = Date.now() - timestamp;
window.SignalCI?.handleEvent('message:send-complete', {
2021-08-11 19:29:07 +00:00
timestamp,
delta,
});
log.info(
2021-07-30 18:37:03 +00:00
`Message.tsx: Rendered 'send complete' for message ${timestamp}; took ${delta}ms`
);
}
}
private getMetadataPlacement(
{
attachments,
attachmentDroppedDueToSize,
deletedForEveryone,
direction,
expirationLength,
expirationTimestamp,
2022-05-11 20:59:58 +00:00
giftBadge,
i18n,
shouldHideMetadata,
status,
text,
}: Readonly<Props> = this.props
): MetadataPlacement {
if (
!expirationLength &&
!expirationTimestamp &&
(!status || SENT_STATUSES.has(status)) &&
shouldHideMetadata
) {
return MetadataPlacement.NotRendered;
}
2022-05-11 20:59:58 +00:00
if (giftBadge) {
const description =
direction === 'incoming'
? i18n('icu:message--donation--unopened--incoming')
: i18n('icu:message--donation--unopened--outgoing');
2022-05-11 20:59:58 +00:00
const isDescriptionRTL = getDirection(description) === 'rtl';
if (giftBadge.state === GiftBadgeStates.Unopened && !isDescriptionRTL) {
return MetadataPlacement.InlineWithText;
}
return MetadataPlacement.Bottom;
}
if (!text && !deletedForEveryone && !attachmentDroppedDueToSize) {
return isAudio(attachments)
? MetadataPlacement.RenderedByMessageAudioComponent
: MetadataPlacement.Bottom;
}
if (!text && attachmentDroppedDueToSize) {
return MetadataPlacement.InlineWithText;
}
if (this.canRenderStickerLikeEmoji()) {
return MetadataPlacement.Bottom;
}
2024-07-17 02:24:56 +00:00
if (this.shouldShowJoinButton()) {
return MetadataPlacement.Bottom;
}
return MetadataPlacement.InlineWithText;
}
/**
* A lot of the time, we add an invisible inline spacer for messages. This spacer is the
* same size as the message metadata. Unfortunately, we don't know how wide it is until
* we render it.
*
* This will probably guess wrong, but it's valuable to get close to the real value
* because it can reduce layout jumpiness.
*/
private guessMetadataWidth(): number {
2024-06-13 23:26:26 +00:00
const { direction, expirationLength, isSMS, status, isEditedMessage } =
this.props;
let result = GUESS_METADATA_WIDTH_TIMESTAMP_SIZE;
if (isEditedMessage) {
result += GUESS_METADATA_WIDTH_EDITED_SIZE;
}
const hasExpireTimer = Boolean(expirationLength);
if (hasExpireTimer) {
result += GUESS_METADATA_WIDTH_EXPIRE_TIMER_SIZE;
}
2024-06-13 23:26:26 +00:00
if (isSMS) {
result += GUESS_METADATA_WIDTH_SMS_SIZE;
}
if (direction === 'outgoing' && status) {
result += GUESS_METADATA_WIDTH_OUTGOING_SIZE[status];
}
return result;
}
2023-03-20 22:23:53 +00:00
public startTargetedTimer(): void {
const { clearTargetedMessage, interactionMode } = this.props;
const { isTargeted } = this.state;
2023-03-20 22:23:53 +00:00
if (interactionMode === 'keyboard' || !isTargeted) {
return;
}
2023-03-20 22:23:53 +00:00
if (!this.targetedTimeout) {
this.targetedTimeout = setTimeout(() => {
this.targetedTimeout = undefined;
this.setState({ isTargeted: false });
clearTargetedMessage();
}, TARGETED_TIMEOUT);
}
}
2022-05-11 20:59:58 +00:00
public startGiftBadgeInterval(): void {
const { giftBadge } = this.props;
if (!giftBadge) {
return;
}
this.giftBadgeInterval = setInterval(() => {
this.updateGiftBadgeCounter();
}, GIFT_BADGE_UPDATE_INTERVAL);
}
public updateGiftBadgeCounter(): void {
this.setState((state: State) => ({
giftBadgeCounter: (state.giftBadgeCounter || 0) + 1,
}));
}
private getTimeRemainingForDeleteForEveryone(): number {
const { timestamp } = this.props;
2023-10-13 21:54:36 +00:00
return Math.max(timestamp - Date.now() + DAY, 0);
}
private startDeleteForEveryoneTimerIfApplicable(): void {
const { canDeleteForEveryone } = this.props;
const { hasDeleteForEveryoneTimerExpired } = this.state;
if (
!canDeleteForEveryone ||
hasDeleteForEveryoneTimerExpired ||
this.deleteForEveryoneTimeout
) {
return;
}
this.deleteForEveryoneTimeout = setTimeout(() => {
this.setState({ hasDeleteForEveryoneTimerExpired: true });
delete this.deleteForEveryoneTimeout;
}, this.getTimeRemainingForDeleteForEveryone());
}
2020-09-14 19:51:27 +00:00
public checkExpired(): void {
const now = Date.now();
2021-06-16 22:20:17 +00:00
const { expirationTimestamp, expirationLength } = this.props;
if (!expirationTimestamp || !expirationLength) {
return;
}
if (this.expiredTimeout) {
return;
}
2021-06-16 22:20:17 +00:00
if (now >= expirationTimestamp) {
this.setState({
expiring: true,
});
const setExpired = () => {
this.setState({
expired: true,
});
};
this.expiredTimeout = setTimeout(setExpired, EXPIRED_DELAY);
}
}
private areLinksEnabled(): boolean {
const { isMessageRequestAccepted, isBlocked } = this.props;
return isMessageRequestAccepted && !isBlocked;
}
private shouldRenderAuthor(): boolean {
const { author, conversationType, direction, shouldCollapseAbove } =
this.props;
return Boolean(
direction === 'incoming' &&
conversationType === 'group' &&
author.title &&
!shouldCollapseAbove
);
}
private canRenderStickerLikeEmoji(): boolean {
const {
attachments,
bodyRanges,
previews,
quote,
storyReplyContext,
text,
} = this.props;
return Boolean(
text &&
!hasNonEmojiText(text) &&
getEmojiCount(text) < 6 &&
!quote &&
!storyReplyContext &&
(!attachments || !attachments.length) &&
(!bodyRanges || !bodyRanges.length) &&
(!previews || !previews.length)
);
}
private updateMetadataWidth = (newMetadataWidth: number): void => {
this.setState(({ metadataWidth }) => ({
// We don't want text to jump around if the metadata shrinks, but we want to make
// sure we have enough room.
metadataWidth: Math.max(metadataWidth, newMetadataWidth),
}));
};
private handleSelectionChange = () => {
const selection = document.getSelection();
if (selection != null && !selection.isCollapsed) {
this.hasSelectedTextRef.current = true;
}
};
private renderMetadata(): ReactNode {
let isInline: boolean;
const metadataPlacement = this.getMetadataPlacement();
switch (metadataPlacement) {
case MetadataPlacement.NotRendered:
case MetadataPlacement.RenderedByMessageAudioComponent:
return null;
case MetadataPlacement.InlineWithText:
isInline = true;
break;
case MetadataPlacement.Bottom:
isInline = false;
break;
default:
log.error(missingCaseError(metadataPlacement));
isInline = false;
break;
}
const {
attachmentDroppedDueToSize,
deletedForEveryone,
direction,
expirationLength,
expirationTimestamp,
i18n,
id,
2023-03-27 23:48:57 +00:00
isEditedMessage,
2024-06-13 23:26:26 +00:00
isSMS,
isSticker,
2019-06-26 19:33:13 +00:00
isTapToViewExpired,
retryMessageSend,
pushPanelForConversation,
2023-03-27 23:48:57 +00:00
showEditHistoryModal,
status,
text,
textAttachment,
timestamp,
} = this.props;
const isStickerLike = isSticker || this.canRenderStickerLikeEmoji();
2021-10-06 17:37:53 +00:00
return (
<MessageMetadata
deletedForEveryone={deletedForEveryone}
direction={direction}
expirationLength={expirationLength}
expirationTimestamp={expirationTimestamp}
hasText={Boolean(text || attachmentDroppedDueToSize)}
i18n={i18n}
id={id}
2023-03-27 23:48:57 +00:00
isEditedMessage={isEditedMessage}
2024-06-13 23:26:26 +00:00
isSMS={isSMS}
isInline={isInline}
isOutlineOnlyBubble={
deletedForEveryone || (attachmentDroppedDueToSize && !text)
}
isShowingImage={this.isShowingImage()}
2021-10-06 17:37:53 +00:00
isSticker={isStickerLike}
isTapToViewExpired={isTapToViewExpired}
onWidthMeasured={isInline ? this.updateMetadataWidth : undefined}
pushPanelForConversation={pushPanelForConversation}
ref={this.metadataRef}
retryMessageSend={retryMessageSend}
2023-03-27 23:48:57 +00:00
showEditHistoryModal={showEditHistoryModal}
status={status}
textPending={textAttachment?.pending}
timestamp={timestamp}
/>
);
}
private renderAuthor(): ReactNode {
const {
2021-04-27 19:55:21 +00:00
author,
2021-05-28 16:15:17 +00:00
contactNameColor,
2022-08-25 16:10:56 +00:00
i18n,
isSticker,
2019-06-26 19:33:13 +00:00
isTapToView,
isTapToViewExpired,
} = this.props;
if (!this.shouldRenderAuthor()) {
return null;
}
2019-06-26 19:33:13 +00:00
const withTapToViewExpired = isTapToView && isTapToViewExpired;
const stickerSuffix = isSticker ? '_with_sticker' : '';
const tapToViewSuffix = withTapToViewExpired
? '--with-tap-to-view-expired'
: '';
const moduleName = `module-message__author${stickerSuffix}${tapToViewSuffix}`;
return (
<div className={moduleName}>
<ContactName
2021-05-28 16:15:17 +00:00
contactNameColor={contactNameColor}
2023-03-30 00:03:25 +00:00
title={author.isMe ? i18n('icu:you') : author.title}
module={moduleName}
/>
</div>
);
}
2020-09-14 19:51:27 +00:00
public renderAttachment(): JSX.Element | null {
const {
attachments,
attachmentDroppedDueToSize,
conversationId,
direction,
expirationLength,
expirationTimestamp,
i18n,
id,
isSticker,
2021-01-29 22:58:28 +00:00
kickOffAttachmentDownload,
markAttachmentAsCorrupted,
pushPanelForConversation,
quote,
readStatus,
renderAudioAttachment,
renderingContext,
shouldCollapseAbove,
shouldCollapseBelow,
showLightbox,
status,
text,
textAttachment,
theme,
timestamp,
} = this.props;
2019-11-07 21:36:16 +00:00
const { imageBroken } = this.state;
const collapseMetadata =
this.getMetadataPlacement() === MetadataPlacement.NotRendered;
if (!attachments || !attachments[0]) {
return null;
}
const firstAttachment = attachments[0];
// For attachments which aren't full-frame
const withContentBelow = Boolean(text || attachmentDroppedDueToSize);
const withContentAbove = Boolean(quote) || this.shouldRenderAuthor();
const displayImage = canDisplayImage(attachments);
2021-04-27 22:11:59 +00:00
if (displayImage && !imageBroken) {
const prefix = isSticker ? 'sticker' : 'attachment';
2021-04-27 22:11:59 +00:00
const containerClassName = classNames(
`module-message__${prefix}-container`,
withContentAbove
? `module-message__${prefix}-container--with-content-above`
: null,
withContentBelow
? 'module-message__attachment-container--with-content-below'
: null,
isSticker && !collapseMetadata
? 'module-message__sticker-container--with-content-below'
: null
);
2021-04-27 22:11:59 +00:00
if (isGIF(attachments)) {
return (
<div className={containerClassName}>
<GIF
attachment={firstAttachment}
size={GIF_SIZE}
theme={theme}
i18n={i18n}
tabIndex={0}
onError={this.handleImageError}
2021-07-14 23:39:52 +00:00
showVisualAttachment={() => {
2022-12-10 02:02:22 +00:00
showLightbox({
2021-07-14 23:39:52 +00:00
attachment: firstAttachment,
messageId: id,
});
}}
2021-04-27 22:11:59 +00:00
kickOffAttachmentDownload={() => {
kickOffAttachmentDownload({
attachment: firstAttachment,
messageId: id,
});
}}
/>
</div>
);
}
if (
isImage(attachments) ||
(isVideo(attachments) &&
(!isDownloaded(attachments[0]) ||
!attachments?.[0].pending ||
hasVideoScreenshot(attachments)))
) {
2021-04-27 22:11:59 +00:00
const bottomOverlay = !isSticker && !collapseMetadata;
// We only want users to tab into this if there's more than one
const tabIndex = attachments.length > 1 ? 0 : -1;
return (
<div className={containerClassName}>
<ImageGrid
attachments={attachments}
direction={direction}
withContentAbove={isSticker || withContentAbove}
withContentBelow={isSticker || withContentBelow}
2021-04-27 22:11:59 +00:00
isSticker={isSticker}
stickerSize={STICKER_SIZE}
bottomOverlay={bottomOverlay}
i18n={i18n}
onError={this.handleImageError}
theme={theme}
shouldCollapseAbove={shouldCollapseAbove}
shouldCollapseBelow={shouldCollapseBelow}
2021-04-27 22:11:59 +00:00
tabIndex={tabIndex}
onClick={attachment => {
2022-03-29 01:10:08 +00:00
if (!isDownloaded(attachment)) {
2021-04-27 22:11:59 +00:00
kickOffAttachmentDownload({ attachment, messageId: id });
} else {
2022-12-10 02:02:22 +00:00
showLightbox({ attachment, messageId: id });
2021-04-27 22:11:59 +00:00
}
}}
/>
</div>
);
}
2020-09-14 19:51:27 +00:00
}
if (isAudio(attachments)) {
2022-10-03 23:43:44 +00:00
const played = isPlayed(direction, status, readStatus);
return renderAudioAttachment({
i18n,
buttonRef: this.audioButtonRef,
renderingContext,
theme,
attachment: firstAttachment,
collapseMetadata,
withContentAbove,
withContentBelow,
direction,
expirationLength,
expirationTimestamp,
id,
conversationId,
played,
pushPanelForConversation,
status,
textPending: textAttachment?.pending,
timestamp,
kickOffAttachmentDownload() {
kickOffAttachmentDownload({
attachment: firstAttachment,
messageId: id,
});
},
onCorrupted() {
markAttachmentAsCorrupted({
attachment: firstAttachment,
messageId: id,
});
},
});
2020-09-14 19:51:27 +00:00
}
const { pending, fileName, fileSize, contentType } = firstAttachment;
const extension = getExtensionForDisplay({ contentType, fileName });
const isDangerous = isFileDangerous(fileName || '');
2020-09-14 19:51:27 +00:00
return (
<button
type="button"
className={classNames(
'module-message__generic-attachment',
withContentBelow
? 'module-message__generic-attachment--with-content-below'
: null,
withContentAbove
? 'module-message__generic-attachment--with-content-above'
: null
)}
// There's only ever one of these, so we don't want users to tab into it
tabIndex={-1}
onClick={event => {
event.stopPropagation();
event.preventDefault();
if (!isDownloaded(firstAttachment)) {
kickOffAttachmentDownload({
attachment: firstAttachment,
messageId: id,
});
} else {
this.openGenericAttachment();
}
}}
2020-09-14 19:51:27 +00:00
>
{pending ? (
<div className="module-message__generic-attachment__spinner-container">
<Spinner svgSize="small" size="24px" direction={direction} />
</div>
) : (
<div className="module-message__generic-attachment__icon-container">
<div className="module-message__generic-attachment__icon">
{extension ? (
<div className="module-message__generic-attachment__icon__extension">
{extension}
2018-10-04 01:12:42 +00:00
</div>
) : null}
</div>
2020-09-14 19:51:27 +00:00
{isDangerous ? (
<div className="module-message__generic-attachment__icon-dangerous-container">
<div className="module-message__generic-attachment__icon-dangerous" />
</div>
) : null}
</div>
2020-09-14 19:51:27 +00:00
)}
<div className="module-message__generic-attachment__text">
<div
className={classNames(
'module-message__generic-attachment__file-name',
`module-message__generic-attachment__file-name--${direction}`
)}
>
{fileName}
</div>
<div
className={classNames(
'module-message__generic-attachment__file-size',
`module-message__generic-attachment__file-size--${direction}`
)}
>
{fileSize}
</div>
</div>
</button>
);
}
2020-09-14 19:51:27 +00:00
public renderPreview(): JSX.Element | null {
2019-01-16 03:03:56 +00:00
const {
attachments,
conversationType,
direction,
i18n,
2022-05-11 20:59:58 +00:00
id,
kickOffAttachmentDownload,
2019-01-16 03:03:56 +00:00
previews,
quote,
shouldCollapseAbove,
theme,
2019-01-16 03:03:56 +00:00
} = this.props;
// Attachments take precedence over Link Previews
if (attachments && attachments.length) {
return null;
}
if (!previews || previews.length < 1) {
return null;
}
const first = previews[0];
if (!first) {
return null;
}
const withContentAbove =
Boolean(quote) ||
(!shouldCollapseAbove &&
conversationType === 'group' &&
direction === 'incoming');
2019-01-16 03:03:56 +00:00
const previewHasImage = isImageAttachment(first.image);
const isFullSizeImage = shouldUseFullSizeLinkPreviewImage(first);
2019-01-16 03:03:56 +00:00
2020-09-28 23:46:31 +00:00
const linkPreviewDate = first.date || null;
const title =
first.title ||
(first.isCallLink
? i18n('icu:calling__call-link-default-title')
: undefined);
const description =
first.description ||
(first.isCallLink
? i18n('icu:message--call-link-description')
: undefined);
const isClickable = this.areLinksEnabled();
2019-11-07 21:36:16 +00:00
const className = classNames(
'module-message__link-preview',
`module-message__link-preview--${direction}`,
{
'module-message__link-preview--with-content-above': withContentAbove,
'module-message__link-preview--nonclickable': !isClickable,
}
);
const onPreviewImageClick = isClickable
? () => {
if (first.image && !isDownloaded(first.image)) {
kickOffAttachmentDownload({
attachment: first.image,
messageId: id,
});
return;
}
openLinkInWebBrowser(first.url);
}
: noop;
const contents = (
<>
2019-01-16 03:03:56 +00:00
{first.image && previewHasImage && isFullSizeImage ? (
<ImageGrid
attachments={[first.image]}
withContentAbove={withContentAbove}
direction={direction}
shouldCollapseAbove={shouldCollapseAbove}
2020-09-14 19:51:27 +00:00
withContentBelow
2019-11-07 21:36:16 +00:00
onError={this.handleImageError}
2019-01-16 03:03:56 +00:00
i18n={i18n}
theme={theme}
onClick={onPreviewImageClick}
2019-01-16 03:03:56 +00:00
/>
) : null}
<div dir="auto" className="module-message__link-preview__content">
2022-06-17 00:48:57 +00:00
{first.image &&
first.domain &&
previewHasImage &&
!isFullSizeImage ? (
2019-01-16 03:03:56 +00:00
<div className="module-message__link-preview__icon_container">
<Image
2020-09-14 19:51:27 +00:00
noBorder
noBackground
curveBottomLeft={
withContentAbove ? CurveType.Tiny : CurveType.Small
}
curveBottomRight={CurveType.Tiny}
curveTopRight={CurveType.Tiny}
curveTopLeft={CurveType.Tiny}
2023-03-30 00:03:25 +00:00
alt={i18n('icu:previewThumbnail', {
2023-03-27 23:37:39 +00:00
domain: first.domain,
})}
2019-01-16 03:03:56 +00:00
height={72}
width={72}
url={first.image.url}
attachment={first.image}
blurHash={first.image.blurHash}
2019-11-07 21:36:16 +00:00
onError={this.handleImageError}
2019-01-16 03:03:56 +00:00
i18n={i18n}
onClick={onPreviewImageClick}
2019-01-16 03:03:56 +00:00
/>
</div>
) : null}
2024-02-22 21:19:50 +00:00
{first.isCallLink && (
2024-07-17 02:24:56 +00:00
<div className="module-message__link-preview__call-link-icon">
<Avatar
acceptedMessageRequest
badge={undefined}
color={getColorForCallLink(getKeyFromCallLink(first.url))}
conversationType="callLink"
i18n={i18n}
isMe={false}
sharedGroupNames={[]}
size={64}
title={title ?? i18n('icu:calling__call-link-default-title')}
2024-07-17 02:24:56 +00:00
/>
</div>
2024-02-22 21:19:50 +00:00
)}
2019-01-16 03:03:56 +00:00
<div
className={classNames(
'module-message__link-preview__text',
previewHasImage && !isFullSizeImage
? 'module-message__link-preview__text--with-icon'
: null
)}
>
<div className="module-message__link-preview__title">{title}</div>
{description && (
<div className="module-message__link-preview__description">
{unescape(description)}
</div>
)}
<div className="module-message__link-preview__footer">
<div className="module-message__link-preview__location">
{first.domain}
</div>
2020-09-28 23:46:31 +00:00
<LinkPreviewDate
date={linkPreviewDate}
className="module-message__link-preview__date"
/>
2019-01-16 03:03:56 +00:00
</div>
</div>
</div>
</>
);
return isClickable ? (
<div
role="link"
tabIndex={0}
className={className}
onKeyDown={(event: React.KeyboardEvent) => {
if (event.key === 'Enter' || event.key === 'Space') {
event.stopPropagation();
event.preventDefault();
openLinkInWebBrowser(first.url);
}
}}
onClick={(event: React.MouseEvent) => {
event.stopPropagation();
event.preventDefault();
openLinkInWebBrowser(first.url);
}}
>
{contents}
</div>
) : (
<div className={className}>{contents}</div>
2019-01-16 03:03:56 +00:00
);
}
public renderAttachmentTooBig(): JSX.Element | null {
const {
attachments,
attachmentDroppedDueToSize,
direction,
i18n,
quote,
shouldCollapseAbove,
shouldCollapseBelow,
text,
} = this.props;
const { metadataWidth } = this.state;
if (!attachmentDroppedDueToSize) {
return null;
}
const labelText = attachments?.length
? i18n('icu:message--attachmentTooBig--multiple')
: i18n('icu:message--attachmentTooBig--one');
const isContentAbove = quote || attachments?.length;
const isContentBelow = Boolean(text);
const willCollapseAbove = shouldCollapseAbove && !isContentAbove;
const willCollapseBelow = shouldCollapseBelow && !isContentBelow;
const maybeSpacer = text
? undefined
: this.getMetadataPlacement() === MetadataPlacement.InlineWithText && (
<MessageTextMetadataSpacer metadataWidth={metadataWidth} />
);
return (
<div
className={classNames(
'module-message__attachment-too-big',
isContentAbove
? 'module-message__attachment-too-big--content-above'
: null,
isContentBelow
? 'module-message__attachment-too-big--content-below'
: null,
willCollapseAbove
? `module-message__attachment-too-big--collapse-above--${direction}`
: null,
willCollapseBelow
? `module-message__attachment-too-big--collapse-below--${direction}`
: null
)}
>
{labelText}
{maybeSpacer}
</div>
);
}
2022-05-11 20:59:58 +00:00
public renderGiftBadge(): JSX.Element | null {
const { conversationTitle, direction, getPreferredBadge, giftBadge, i18n } =
this.props;
const { showOutgoingGiftBadgeModal } = this.state;
if (!giftBadge) {
return null;
}
if (
giftBadge.state === GiftBadgeStates.Unopened ||
giftBadge.state === GiftBadgeStates.Failed
) {
const description =
direction === 'incoming'
? i18n('icu:message--donation--unopened--incoming')
: i18n('icu:message--donation--unopened--outgoing');
2022-05-11 20:59:58 +00:00
const { metadataWidth } = this.state;
return (
<div className="module-message__unopened-gift-badge__container">
<div
className={classNames(
'module-message__unopened-gift-badge',
`module-message__unopened-gift-badge--${direction}`
)}
2023-01-23 20:42:40 +00:00
aria-label={i18n('icu:message--donation--unopened--label', {
sender: conversationTitle,
})}
2022-05-11 20:59:58 +00:00
>
<div
className="module-message__unopened-gift-badge__ribbon-horizontal"
aria-hidden
/>
<div
className="module-message__unopened-gift-badge__ribbon-vertical"
aria-hidden
/>
<img
className="module-message__unopened-gift-badge__bow"
src="images/gift-bow.svg"
alt=""
aria-hidden
/>
</div>
<div
className={classNames(
'module-message__unopened-gift-badge__text',
`module-message__unopened-gift-badge__text--${direction}`
)}
>
<div
className={classNames(
'module-message__text',
`module-message__text--${direction}`
)}
>
{description}
{this.getMetadataPlacement() ===
MetadataPlacement.InlineWithText && (
<MessageTextMetadataSpacer metadataWidth={metadataWidth} />
)}
</div>
{this.renderMetadata()}
</div>
</div>
);
}
2022-05-16 19:54:38 +00:00
if (
giftBadge.state === GiftBadgeStates.Redeemed ||
giftBadge.state === GiftBadgeStates.Opened
) {
const badgeId = giftBadge.id || `BOOST-${giftBadge.level}`;
2022-05-11 20:59:58 +00:00
const badgeSize = 64;
const badge = getPreferredBadge([{ id: badgeId }]);
const badgeImagePath = getBadgeImageFileLocalPath(
badge,
badgeSize,
BadgeImageTheme.Transparent
);
let remaining: string;
const duration = giftBadge.expiration - Date.now();
const remainingDays = Math.floor(duration / DAY);
const remainingHours = Math.floor(duration / HOUR);
const remainingMinutes = Math.floor(duration / MINUTE);
if (remainingDays > 1) {
2023-01-23 20:42:40 +00:00
remaining = i18n('icu:message--donation--remaining--days', {
2022-05-11 20:59:58 +00:00
days: remainingDays,
});
} else if (remainingHours > 1) {
2023-01-23 20:42:40 +00:00
remaining = i18n('icu:message--donation--remaining--hours', {
2022-05-11 20:59:58 +00:00
hours: remainingHours,
});
2023-01-23 20:42:40 +00:00
} else if (remainingMinutes > 0) {
remaining = i18n('icu:message--donation--remaining--minutes', {
2022-05-11 20:59:58 +00:00
minutes: remainingMinutes,
});
} else {
2023-01-23 20:42:40 +00:00
remaining = i18n('icu:message--donation--expired');
2022-05-11 20:59:58 +00:00
}
const wasSent = direction === 'outgoing';
const buttonContents = wasSent ? (
2023-01-23 20:42:40 +00:00
i18n('icu:message--donation--view')
2022-05-11 20:59:58 +00:00
) : (
<>
<span
className={classNames(
'module-message__redeemed-gift-badge__icon-check',
`module-message__redeemed-gift-badge__icon-check--${direction}`
)}
/>{' '}
2023-01-23 20:42:40 +00:00
{i18n('icu:message--donation--redeemed')}
2022-05-11 20:59:58 +00:00
</>
);
const badgeElement = badge ? (
<img
className="module-message__redeemed-gift-badge__badge"
src={badgeImagePath}
alt={badge.name}
/>
) : (
<div
className={classNames(
'module-message__redeemed-gift-badge__badge',
`module-message__redeemed-gift-badge__badge--missing-${direction}`
)}
2023-01-23 20:42:40 +00:00
aria-label={i18n('icu:donation--missing')}
2022-05-11 20:59:58 +00:00
/>
);
return (
<div className="module-message__redeemed-gift-badge__container">
<div className="module-message__redeemed-gift-badge">
{badgeElement}
<div className="module-message__redeemed-gift-badge__text">
<div className="module-message__redeemed-gift-badge__title">
2023-01-23 20:42:40 +00:00
{i18n('icu:message--donation')}
2022-05-11 20:59:58 +00:00
</div>
<div
className={classNames(
'module-message__redeemed-gift-badge__remaining',
`module-message__redeemed-gift-badge__remaining--${direction}`
)}
>
{remaining}
</div>
</div>
</div>
<button
className={classNames(
'module-message__redeemed-gift-badge__button',
`module-message__redeemed-gift-badge__button--${direction}`
)}
disabled={!wasSent}
onClick={
wasSent
? () => this.setState({ showOutgoingGiftBadgeModal: true })
: undefined
}
type="button"
>
<div className="module-message__redeemed-gift-badge__button__text">
{buttonContents}
</div>
</button>
{this.renderMetadata()}
{showOutgoingGiftBadgeModal ? (
<OutgoingGiftBadgeModal
i18n={i18n}
recipientTitle={conversationTitle}
badgeId={badgeId}
getPreferredBadge={getPreferredBadge}
hideOutgoingGiftBadgeModal={() =>
this.setState({ showOutgoingGiftBadgeModal: false })
}
/>
) : null}
</div>
);
}
throw missingCaseError(giftBadge.state);
}
2022-11-30 21:47:54 +00:00
public renderPayment(): JSX.Element | null {
const {
payment,
direction,
author,
conversationTitle,
conversationColor,
i18n,
} = this.props;
if (payment == null || !isPaymentNotificationEvent(payment)) {
2022-11-30 21:47:54 +00:00
return null;
}
return (
<div
className={`module-payment-notification__container ${
direction === 'outgoing'
? `module-payment-notification--outgoing module-payment-notification--outgoing-${conversationColor}`
: ''
}`}
>
<p className="module-payment-notification__label">
{getPaymentEventDescription(
payment,
author.title,
conversationTitle,
author.isMe,
i18n
)}
</p>
<p className="module-payment-notification__check_device_box">
{i18n('icu:payment-event-notification-check-primary-device')}
</p>
{payment.note != null && (
<p className="module-payment-notification__note">
2023-04-20 17:03:43 +00:00
<UserText text={payment.note} />
2022-11-30 21:47:54 +00:00
</p>
)}
</div>
);
}
2020-09-14 19:51:27 +00:00
public renderQuote(): JSX.Element | null {
const {
2021-05-28 16:15:17 +00:00
conversationColor,
conversationId,
2022-11-30 21:47:54 +00:00
conversationTitle,
2021-05-28 16:15:17 +00:00
customColor,
direction,
disableScroll,
doubleCheckMissingQuoteReference,
i18n,
id,
quote,
scrollToQuotedMessage,
} = this.props;
if (!quote) {
return null;
}
2022-05-11 20:59:58 +00:00
const { isGiftBadge, isViewOnce, referencedMessageNotFound } = quote;
const clickHandler = disableScroll
? undefined
: () => {
scrollToQuotedMessage({
authorId: quote.authorId,
conversationId,
sentAt: quote.sentAt,
});
};
const isIncoming = direction === 'incoming';
return (
<Quote
i18n={i18n}
onClick={clickHandler}
text={quote.text}
2021-04-02 21:35:28 +00:00
rawAttachment={quote.rawAttachment}
2022-11-30 21:47:54 +00:00
payment={quote.payment}
isIncoming={isIncoming}
2020-07-24 01:35:32 +00:00
authorTitle={quote.authorTitle}
2020-09-16 22:42:48 +00:00
bodyRanges={quote.bodyRanges}
2021-05-28 16:15:17 +00:00
conversationColor={conversationColor}
2022-11-30 21:47:54 +00:00
conversationTitle={conversationTitle}
2021-05-28 16:15:17 +00:00
customColor={customColor}
isViewOnce={isViewOnce}
2022-05-11 20:59:58 +00:00
isGiftBadge={isGiftBadge}
referencedMessageNotFound={referencedMessageNotFound}
isFromMe={quote.isFromMe}
doubleCheckMissingQuoteReference={() =>
doubleCheckMissingQuoteReference(id)
}
/>
);
}
2022-03-16 17:30:14 +00:00
public renderStoryReplyContext(): JSX.Element | null {
const {
2022-11-30 21:47:54 +00:00
conversationTitle,
2022-03-16 17:30:14 +00:00
conversationColor,
customColor,
direction,
i18n,
storyReplyContext,
2022-07-06 19:06:20 +00:00
viewStory,
2022-03-16 17:30:14 +00:00
} = this.props;
if (!storyReplyContext) {
return null;
}
const isIncoming = direction === 'incoming';
return (
<>
{storyReplyContext.emoji && (
<div className="module-message__quote-story-reaction-header">
{isIncoming
? i18n('icu:Quote__story-reaction--you')
: i18n('icu:Quote__story-reaction', {
name: storyReplyContext.authorTitle,
})}
</div>
)}
<Quote
authorTitle={storyReplyContext.authorTitle}
conversationColor={conversationColor}
2022-11-30 21:47:54 +00:00
conversationTitle={conversationTitle}
customColor={customColor}
i18n={i18n}
isFromMe={storyReplyContext.isFromMe}
2022-05-11 20:59:58 +00:00
isGiftBadge={false}
isIncoming={isIncoming}
isStoryReply
isViewOnce={false}
moduleClassName="StoryReplyQuote"
onClick={() => {
2022-08-22 17:44:23 +00:00
if (!storyReplyContext.storyId) {
return;
}
2022-07-25 18:55:44 +00:00
viewStory({
storyId: storyReplyContext.storyId,
storyViewMode: StoryViewModeType.Single,
});
}}
rawAttachment={storyReplyContext.rawAttachment}
reactionEmoji={storyReplyContext.emoji}
2022-07-06 19:06:20 +00:00
referencedMessageNotFound={!storyReplyContext.storyId}
text={storyReplyContext.text}
/>
</>
2022-03-16 17:30:14 +00:00
);
}
2020-09-14 19:51:27 +00:00
public renderEmbeddedContact(): JSX.Element | null {
const {
contact,
conversationType,
direction,
i18n,
pushPanelForConversation,
text,
} = this.props;
if (!contact) {
return null;
}
const withCaption = Boolean(text);
const withContentAbove =
conversationType === 'group' && direction === 'incoming';
const withContentBelow =
withCaption ||
this.getMetadataPlacement() !== MetadataPlacement.NotRendered;
const otherContent =
2023-08-16 20:54:39 +00:00
(contact && contact.firstNumber && contact.serviceId) || withCaption;
const tabIndex = otherContent ? 0 : -1;
return (
<EmbeddedContact
contact={contact}
isIncoming={direction === 'incoming'}
i18n={i18n}
onClick={() => {
const signalAccount =
2023-08-16 20:54:39 +00:00
contact.firstNumber && contact.serviceId
? {
phoneNumber: contact.firstNumber,
2023-08-16 20:54:39 +00:00
serviceId: contact.serviceId,
}
: undefined;
pushPanelForConversation({
type: PanelType.ContactDetails,
args: {
contact,
signalAccount,
},
});
}}
withContentAbove={withContentAbove}
withContentBelow={withContentBelow}
tabIndex={tabIndex}
/>
);
}
2020-09-14 19:51:27 +00:00
public renderSendMessageButton(): JSX.Element | null {
const { contact, direction, shouldCollapseBelow, startConversation, i18n } =
this.props;
const noBottomLeftCurve = direction === 'incoming' && shouldCollapseBelow;
const noBottomRightCurve = direction === 'outgoing' && shouldCollapseBelow;
if (!contact) {
return null;
}
2023-08-16 20:54:39 +00:00
const { firstNumber, serviceId } = contact;
if (!firstNumber || !serviceId) {
return null;
}
return (
2019-11-07 21:36:16 +00:00
<button
2020-09-14 19:51:27 +00:00
type="button"
2022-06-28 00:37:05 +00:00
onClick={e => {
e.preventDefault();
e.stopPropagation();
2023-08-16 20:54:39 +00:00
startConversation(firstNumber, serviceId);
2022-06-28 00:37:05 +00:00
}}
className={classNames(
'module-message__send-message-button',
noBottomLeftCurve &&
'module-message__send-message-button--no-bottom-left-curve',
noBottomRightCurve &&
'module-message__send-message-button--no-bottom-right-curve'
)}
>
2023-03-30 00:03:25 +00:00
{i18n('icu:sendMessageToContact')}
2019-11-07 21:36:16 +00:00
</button>
);
}
private renderAvatar(): ReactNode {
const {
author,
conversationId,
conversationType,
direction,
getPreferredBadge,
i18n,
shouldCollapseBelow,
showContactModal,
theme,
} = this.props;
if (conversationType !== 'group' || direction !== 'incoming') {
return null;
}
return (
2021-01-27 21:15:43 +00:00
<div
className={classNames('module-message__author-avatar-container', {
2021-11-11 22:43:05 +00:00
'module-message__author-avatar-container--with-reactions':
this.hasReactions(),
2021-01-27 21:15:43 +00:00
})}
>
{shouldCollapseBelow ? (
<AvatarSpacer size={GROUP_AVATAR_SIZE} />
) : (
<Avatar
acceptedMessageRequest={author.acceptedMessageRequest}
2024-07-11 19:44:09 +00:00
avatarUrl={author.avatarUrl}
badge={getPreferredBadge(author.badges)}
color={author.color}
conversationType="direct"
i18n={i18n}
isMe={author.isMe}
onClick={event => {
event.stopPropagation();
event.preventDefault();
showContactModal(author.id, conversationId);
}}
phoneNumber={author.phoneNumber}
profileName={author.profileName}
sharedGroupNames={author.sharedGroupNames}
size={GROUP_AVATAR_SIZE}
theme={theme}
title={author.title}
2024-07-11 19:44:09 +00:00
unblurredAvatarUrl={author.unblurredAvatarUrl}
/>
)}
2021-01-27 21:15:43 +00:00
</div>
);
}
private getContents(): string | undefined {
const { deletedForEveryone, direction, i18n, status, text } = this.props;
if (deletedForEveryone) {
return i18n('icu:message--deletedForEveryone');
}
if (direction === 'incoming' && status === 'error') {
return i18n('icu:incomingError');
}
return text;
}
2020-09-14 19:51:27 +00:00
public renderText(): JSX.Element | null {
2020-04-29 21:24:12 +00:00
const {
2020-09-16 22:42:48 +00:00
bodyRanges,
2020-04-29 21:24:12 +00:00
deletedForEveryone,
direction,
displayLimit,
2020-04-29 21:24:12 +00:00
i18n,
id,
isSpoilerExpanded,
kickOffAttachmentDownload,
messageExpanded,
payment,
showConversation,
showSpoiler,
2020-04-29 21:24:12 +00:00
status,
textAttachment,
2020-04-29 21:24:12 +00:00
} = this.props;
const { metadataWidth } = this.state;
const contents = this.getContents();
if (!contents) {
return null;
}
// Payment notifications are rendered in renderPayment, but they may have additional
// text in message.body for backwards-compatibility that we don't want to render
if (payment && isPaymentNotificationEvent(payment)) {
return null;
}
return (
<div // eslint-disable-line jsx-a11y/click-events-have-key-events,jsx-a11y/no-static-element-interactions
className={classNames(
'module-message__text',
`module-message__text--${direction}`,
status === 'error' && direction === 'incoming'
? 'module-message__text--error'
: null,
deletedForEveryone
? 'module-message__text--delete-for-everyone'
: null
)}
onClick={e => {
// Prevent metadata from being selected on triple clicks.
const clickCount = e.detail;
const range = window.getSelection()?.getRangeAt(0);
if (
clickCount === 3 &&
this.metadataRef.current &&
range?.intersectsNode(this.metadataRef.current)
) {
range.setEndBefore(this.metadataRef.current);
}
}}
onDoubleClick={(event: React.MouseEvent) => {
// Prevent double-click interefering with interactions _inside_
// the bubble.
event.stopPropagation();
}}
>
2021-10-20 20:46:42 +00:00
<MessageBodyReadMore
2020-09-16 22:42:48 +00:00
bodyRanges={bodyRanges}
direction={direction}
disableLinks={!this.areLinksEnabled()}
displayLimit={displayLimit}
i18n={i18n}
id={id}
isSpoilerExpanded={isSpoilerExpanded || {}}
kickOffBodyDownload={() => {
if (!textAttachment) {
return;
}
2024-09-23 19:24:41 +00:00
if (isDownloaded(textAttachment)) {
return;
}
kickOffAttachmentDownload({
attachment: textAttachment,
messageId: id,
});
}}
messageExpanded={messageExpanded}
showConversation={showConversation}
renderLocation={RenderLocation.Timeline}
onExpandSpoiler={data => showSpoiler(id, data)}
2020-09-16 22:42:48 +00:00
text={contents || ''}
textAttachment={textAttachment}
/>
2023-04-20 17:03:43 +00:00
{this.getMetadataPlacement() === MetadataPlacement.InlineWithText && (
<MessageTextMetadataSpacer metadataWidth={metadataWidth} />
)}
</div>
);
}
2024-07-17 02:24:56 +00:00
private shouldShowJoinButton(): boolean {
const { previews } = this.props;
2024-02-22 21:19:50 +00:00
if (previews?.length !== 1) {
2024-07-17 02:24:56 +00:00
return false;
2024-02-22 21:19:50 +00:00
}
const onlyPreview = previews[0];
2024-07-17 02:24:56 +00:00
return Boolean(onlyPreview.isCallLink);
}
private renderAction(): JSX.Element | null {
const { direction, activeCallConversationId, i18n, previews } = this.props;
2024-07-17 02:24:56 +00:00
if (this.shouldShowJoinButton()) {
const firstPreview = previews[0];
const inAnotherCall = Boolean(
activeCallConversationId &&
(!firstPreview.callLinkRoomId ||
activeCallConversationId !== firstPreview.callLinkRoomId)
);
2024-07-17 02:24:56 +00:00
const joinButton = (
2024-02-22 21:19:50 +00:00
<button
type="button"
className={classNames('module-message__action', {
'module-message__action--incoming': direction === 'incoming',
'module-message__action--outgoing': direction === 'outgoing',
'module-message__action--incoming--in-another-call':
direction === 'incoming' && inAnotherCall,
'module-message__action--outgoing--in-another-call':
direction === 'outgoing' && inAnotherCall,
2024-02-22 21:19:50 +00:00
})}
2024-07-17 02:24:56 +00:00
onClick={() => openLinkInWebBrowser(firstPreview?.url)}
2024-02-22 21:19:50 +00:00
>
{i18n('icu:calling__join')}
</button>
);
return inAnotherCall ? (
<InAnotherCallTooltip i18n={i18n}>{joinButton}</InAnotherCallTooltip>
) : (
joinButton
);
2024-02-22 21:19:50 +00:00
}
return null;
}
private renderError(): ReactNode {
const { status, direction } = this.props;
if (
status !== 'paused' &&
status !== 'error' &&
status !== 'partial-sent'
) {
return null;
}
return (
<div className="module-message__error-container">
<div
className={classNames(
'module-message__error',
`module-message__error--${direction}`,
`module-message__error--${status}`
)}
/>
</div>
);
}
2019-01-14 21:49:58 +00:00
public getWidth(): number | undefined {
2022-05-11 20:59:58 +00:00
const { attachments, giftBadge, isSticker, previews } = this.props;
if (giftBadge) {
return 240;
}
2019-01-16 03:03:56 +00:00
if (attachments && attachments.length) {
2021-04-27 22:11:59 +00:00
if (isGIF(attachments)) {
2021-06-24 21:00:11 +00:00
// Message container border
return GIF_SIZE + 2;
2021-04-27 22:11:59 +00:00
}
if (isSticker) {
// Padding is 8px, on both sides
return STICKER_SIZE + 8 * 2;
}
2019-01-16 03:03:56 +00:00
const dimensions = getGridDimensions(attachments);
if (dimensions) {
return dimensions.width;
2019-01-16 03:03:56 +00:00
}
}
const firstLinkPreview = (previews || [])[0];
if (
firstLinkPreview &&
firstLinkPreview.image &&
shouldUseFullSizeLinkPreviewImage(firstLinkPreview)
) {
const dimensions = getImageDimensions(firstLinkPreview.image);
if (dimensions) {
return dimensions.width;
2019-01-16 03:03:56 +00:00
}
}
2024-07-17 02:24:56 +00:00
if (firstLinkPreview && firstLinkPreview.isCallLink) {
return 300;
}
2020-09-14 19:51:27 +00:00
return undefined;
2019-01-16 03:03:56 +00:00
}
public isShowingImage(): boolean {
2019-06-26 19:33:13 +00:00
const { isTapToView, attachments, previews } = this.props;
2019-01-16 03:03:56 +00:00
const { imageBroken } = this.state;
2019-06-26 19:33:13 +00:00
if (imageBroken || isTapToView) {
2019-01-16 03:03:56 +00:00
return false;
}
if (attachments && attachments.length) {
const displayImage = canDisplayImage(attachments);
return displayImage && (isImage(attachments) || isVideo(attachments));
2019-01-16 03:03:56 +00:00
}
if (previews && previews.length) {
const first = previews[0];
const { image } = first;
return isImageAttachment(image);
}
return false;
}
2020-09-14 19:51:27 +00:00
public isAttachmentPending(): boolean {
2019-06-26 19:33:13 +00:00
const { attachments } = this.props;
if (!attachments || attachments.length < 1) {
return false;
}
const first = attachments[0];
return Boolean(first.pending);
}
2020-09-14 19:51:27 +00:00
public renderTapToViewIcon(): JSX.Element {
2019-06-26 19:33:13 +00:00
const { direction, isTapToViewExpired } = this.props;
const isDownloadPending = this.isAttachmentPending();
return !isTapToViewExpired && isDownloadPending ? (
<div className="module-message__tap-to-view__spinner-container">
<Spinner svgSize="small" size="20px" direction={direction} />
</div>
) : (
<div
className={classNames(
'module-message__tap-to-view__icon',
`module-message__tap-to-view__icon--${direction}`,
isTapToViewExpired
? 'module-message__tap-to-view__icon--expired'
: null
)}
/>
);
}
2020-09-14 19:51:27 +00:00
public renderTapToViewText(): string | undefined {
2019-06-26 19:33:13 +00:00
const {
2019-10-03 19:03:46 +00:00
attachments,
2019-06-26 19:33:13 +00:00
direction,
i18n,
isTapToViewExpired,
isTapToViewError,
} = this.props;
const isDownloadPending = this.isAttachmentPending();
if (isDownloadPending) {
return;
}
if (isTapToViewError) {
2023-03-30 00:03:25 +00:00
return i18n('icu:incomingError');
}
if (direction === 'outgoing') {
2023-03-30 00:03:25 +00:00
return i18n('icu:Message--tap-to-view--outgoing');
}
if (isTapToViewExpired) {
2023-03-30 00:03:25 +00:00
return i18n('icu:Message--tap-to-view-expired');
}
if (isVideo(attachments)) {
2023-03-30 00:03:25 +00:00
return i18n('icu:Message--tap-to-view--incoming-video');
}
2023-03-30 00:03:25 +00:00
return i18n('icu:Message--tap-to-view--incoming');
2019-06-26 19:33:13 +00:00
}
2020-09-14 19:51:27 +00:00
public renderTapToView(): JSX.Element {
2019-06-26 19:33:13 +00:00
const {
conversationType,
direction,
isTapToViewExpired,
isTapToViewError,
} = this.props;
const collapseMetadata =
this.getMetadataPlacement() === MetadataPlacement.NotRendered;
2019-06-26 19:33:13 +00:00
const withContentBelow = !collapseMetadata;
const withContentAbove =
!collapseMetadata &&
conversationType === 'group' &&
direction === 'incoming';
return (
<div
className={classNames(
'module-message__tap-to-view',
withContentBelow
? 'module-message__tap-to-view--with-content-below'
: null,
withContentAbove
? 'module-message__tap-to-view--with-content-above'
: null
)}
>
{isTapToViewError ? null : this.renderTapToViewIcon()}
<div
className={classNames(
'module-message__tap-to-view__text',
`module-message__tap-to-view__text--${direction}`,
isTapToViewExpired
? `module-message__tap-to-view__text--${direction}-expired`
: null,
isTapToViewError
? `module-message__tap-to-view__text--${direction}-error`
: null
)}
>
{this.renderTapToViewText()}
</div>
</div>
);
}
private popperPreventOverflowModifier(): Partial<PreventOverflowModifier> {
const { containerElementRef } = this.props;
return {
name: 'preventOverflow',
options: {
altAxis: true,
boundary: containerElementRef.current || undefined,
padding: {
bottom: 16,
left: 8,
right: 8,
top: 16,
},
},
};
}
2020-09-14 19:51:27 +00:00
public toggleReactionViewer = (onlyRemove = false): void => {
this.setState(oldState => {
const { reactionViewerRoot } = oldState;
2020-01-17 22:23:19 +00:00
if (reactionViewerRoot) {
document.body.removeChild(reactionViewerRoot);
oldState.reactionViewerOutsideClickDestructor?.();
return {
reactionViewerRoot: null,
reactionViewerOutsideClickDestructor: undefined,
};
2020-01-17 22:23:19 +00:00
}
if (!onlyRemove) {
const root = document.createElement('div');
document.body.appendChild(root);
const reactionViewerOutsideClickDestructor = handleOutsideClick(
() => {
this.toggleReactionViewer(true);
return true;
},
2022-09-27 20:24:21 +00:00
{
containerElements: [root, this.reactionsContainerRef],
name: 'Message.reactionViewer',
}
2020-01-23 23:57:37 +00:00
);
2020-01-17 22:23:19 +00:00
return {
reactionViewerRoot: root,
reactionViewerOutsideClickDestructor,
2020-01-17 22:23:19 +00:00
};
}
return null;
2020-01-17 22:23:19 +00:00
});
};
2020-09-14 19:51:27 +00:00
public renderReactions(outgoing: boolean): JSX.Element | null {
2021-11-17 21:11:46 +00:00
const { getPreferredBadge, reactions = [], i18n, theme } = this.props;
2020-01-17 22:23:19 +00:00
2021-01-27 21:15:43 +00:00
if (!this.hasReactions()) {
2020-01-17 22:23:19 +00:00
return null;
}
2020-10-02 20:05:09 +00:00
const reactionsWithEmojiData = reactions.map(reaction => ({
...reaction,
...emojiToData(reaction.emoji),
}));
2020-01-17 22:23:19 +00:00
// Group by emoji and order each group by timestamp descending
2020-10-02 20:05:09 +00:00
const groupedAndSortedReactions = Object.values(
groupBy(reactionsWithEmojiData, 'short_name')
).map(groupedReactions =>
orderBy(
groupedReactions,
[reaction => reaction.from.isMe, 'timestamp'],
['desc', 'desc']
)
2020-01-17 22:23:19 +00:00
);
// Order groups by length and subsequently by most recent reaction
const ordered = orderBy(
2020-10-02 20:05:09 +00:00
groupedAndSortedReactions,
2020-01-17 22:23:19 +00:00
['length', ([{ timestamp }]) => timestamp],
['desc', 'desc']
);
// Take the first three groups for rendering
const toRender = take(ordered, 3).map(res => {
const isMe = res.some(re => Boolean(re.from.isMe));
const count = res.length;
const { emoji } = res[0];
let label: string;
if (isMe) {
label = i18n('icu:Message__reaction-emoji-label--you', { emoji });
} else if (count === 1) {
label = i18n('icu:Message__reaction-emoji-label--single', {
title: res[0].from.title,
emoji,
});
} else {
label = i18n('icu:Message__reaction-emoji-label--many', {
count,
emoji,
});
}
return {
count,
emoji,
isMe,
label,
};
});
const someNotRendered = ordered.length > 3;
// We only drop two here because the third emoji would be replaced by the
// more button
const maybeNotRendered = drop(ordered, 2);
const maybeNotRenderedTotal = maybeNotRendered.reduce(
(sum, res) => sum + res.length,
0
);
const notRenderedIsMe =
someNotRendered &&
maybeNotRendered.some(res => res.some(re => Boolean(re.from.isMe)));
2021-01-27 21:15:43 +00:00
const { reactionViewerRoot } = this.state;
2020-01-17 22:23:19 +00:00
const popperPlacement = outgoing ? 'bottom-end' : 'bottom-start';
return (
<Manager>
<Reference>
{({ ref: popperRef }) => (
<div
2020-03-23 21:09:12 +00:00
ref={this.reactionsContainerRefMerger(
this.reactionsContainerRef,
popperRef
)}
className={classNames(
'module-message__reactions',
outgoing
? 'module-message__reactions--outgoing'
: 'module-message__reactions--incoming'
)}
onDoubleClick={ev => {
ev.stopPropagation();
}}
2020-01-17 22:23:19 +00:00
>
{toRender.map((re, i) => {
const isLast = i === toRender.length - 1;
const isMore = isLast && someNotRendered;
const isMoreWithMe = isMore && notRenderedIsMe;
return (
<button
aria-label={re.label}
2020-09-14 19:51:27 +00:00
type="button"
// eslint-disable-next-line react/no-array-index-key
key={`${re.emoji}-${i}`}
className={classNames(
'module-message__reactions__reaction',
re.count > 1
? 'module-message__reactions__reaction--with-count'
: null,
outgoing
? 'module-message__reactions__reaction--outgoing'
: 'module-message__reactions__reaction--incoming',
isMoreWithMe || (re.isMe && !isMoreWithMe)
? 'module-message__reactions__reaction--is-me'
: null
)}
onClick={e => {
e.stopPropagation();
e.preventDefault();
this.toggleReactionViewer(false);
}}
onKeyDown={e => {
// Prevent enter key from opening stickers/attachments
if (e.key === 'Enter') {
2020-01-17 22:23:19 +00:00
e.stopPropagation();
}
}}
>
{isMore ? (
<span
className={classNames(
'module-message__reactions__reaction__count',
'module-message__reactions__reaction__count--no-emoji',
isMoreWithMe
? 'module-message__reactions__reaction__count--is-me'
: null
)}
>
+{maybeNotRenderedTotal}
</span>
) : (
2020-09-14 19:51:27 +00:00
<>
<Emoji size={16} emoji={re.emoji} />
{re.count > 1 ? (
<span
className={classNames(
'module-message__reactions__reaction__count',
re.isMe
? 'module-message__reactions__reaction__count--is-me'
: null
)}
>
{re.count}
</span>
) : null}
2020-09-14 19:51:27 +00:00
</>
)}
</button>
);
})}
</div>
2020-01-17 22:23:19 +00:00
)}
</Reference>
{reactionViewerRoot &&
createPortal(
<Popper
placement={popperPlacement}
strategy="fixed"
modifiers={[this.popperPreventOverflowModifier()]}
>
{({ ref, style }) => (
<ReactionViewer
ref={ref}
style={{
...style,
zIndex: 2,
}}
getPreferredBadge={getPreferredBadge}
reactions={reactions}
i18n={i18n}
onClose={this.toggleReactionViewer}
theme={theme}
/>
)}
</Popper>,
2020-01-17 22:23:19 +00:00
reactionViewerRoot
)}
</Manager>
);
}
2020-09-14 19:51:27 +00:00
public renderContents(): JSX.Element | null {
const { deletedForEveryone, giftBadge, isTapToView } = this.props;
2020-04-29 21:24:12 +00:00
if (deletedForEveryone) {
return (
<>
{this.renderText()}
{this.renderMetadata()}
</>
);
2020-04-29 21:24:12 +00:00
}
2019-06-26 19:33:13 +00:00
2022-05-11 20:59:58 +00:00
if (giftBadge) {
return this.renderGiftBadge();
}
2019-06-26 19:33:13 +00:00
if (isTapToView) {
return (
<>
{this.renderTapToView()}
{this.renderMetadata()}
</>
);
}
return (
<>
{this.renderQuote()}
2022-03-16 17:30:14 +00:00
{this.renderStoryReplyContext()}
2019-06-26 19:33:13 +00:00
{this.renderAttachment()}
{this.renderPreview()}
{this.renderAttachmentTooBig()}
2022-11-30 21:47:54 +00:00
{this.renderPayment()}
2019-06-26 19:33:13 +00:00
{this.renderEmbeddedContact()}
{this.renderText()}
2024-02-22 21:19:50 +00:00
{this.renderAction()}
2019-06-26 19:33:13 +00:00
{this.renderMetadata()}
{this.renderSendMessageButton()}
</>
);
}
2019-11-07 21:36:16 +00:00
public handleOpen = (
event: React.KeyboardEvent<HTMLDivElement> | React.MouseEvent
2020-09-14 19:51:27 +00:00
): void => {
const {
attachments,
contact,
2022-12-10 02:02:22 +00:00
showLightboxForViewOnceMedia,
direction,
2022-05-11 20:59:58 +00:00
giftBadge,
id,
2019-11-07 21:36:16 +00:00
isTapToView,
isTapToViewExpired,
2021-01-29 22:58:28 +00:00
kickOffAttachmentDownload,
2022-06-28 00:37:05 +00:00
startConversation,
2022-05-11 20:59:58 +00:00
openGiftBadge,
pushPanelForConversation,
2022-12-10 02:02:22 +00:00
showLightbox,
showExpiredIncomingTapToViewToast,
showExpiredOutgoingTapToViewToast,
2019-11-07 21:36:16 +00:00
} = this.props;
const { imageBroken } = this.state;
const isAttachmentPending = this.isAttachmentPending();
2022-05-11 20:59:58 +00:00
if (giftBadge && giftBadge.state === GiftBadgeStates.Unopened) {
openGiftBadge(id);
return;
}
2019-11-07 21:36:16 +00:00
if (isTapToView) {
if (isAttachmentPending) {
log.info(
'<Message> handleOpen: tap-to-view attachment is pending; not showing the lightbox'
);
return;
}
event.preventDefault();
event.stopPropagation();
if (isTapToViewExpired) {
const action =
direction === 'outgoing'
? showExpiredOutgoingTapToViewToast
: showExpiredIncomingTapToViewToast;
action();
2019-11-07 21:36:16 +00:00
return;
2019-11-07 21:36:16 +00:00
}
if (attachments && !isDownloaded(attachments[0])) {
kickOffAttachmentDownload({
attachment: attachments[0],
messageId: id,
});
return;
}
showLightboxForViewOnceMedia(id);
2019-11-07 21:36:16 +00:00
return;
}
2021-01-29 22:58:28 +00:00
if (
!imageBroken &&
attachments &&
attachments.length > 0 &&
!isAttachmentPending &&
2022-03-29 01:10:08 +00:00
!isDownloaded(attachments[0])
2021-01-29 22:58:28 +00:00
) {
event.preventDefault();
event.stopPropagation();
const attachment = attachments[0];
kickOffAttachmentDownload({ attachment, messageId: id });
return;
}
2019-11-07 21:36:16 +00:00
if (
!imageBroken &&
attachments &&
attachments.length > 0 &&
!isAttachmentPending &&
canDisplayImage(attachments) &&
((isImage(attachments) && hasImage(attachments)) ||
(isVideo(attachments) && hasVideoScreenshot(attachments)))
2019-11-07 21:36:16 +00:00
) {
event.preventDefault();
event.stopPropagation();
const attachment = attachments[0];
2022-12-10 02:02:22 +00:00
showLightbox({ attachment, messageId: id });
2019-11-07 21:36:16 +00:00
return;
}
if (
attachments &&
attachments.length === 1 &&
!isAttachmentPending &&
!isAudio(attachments)
) {
event.preventDefault();
event.stopPropagation();
this.openGenericAttachment();
return;
}
if (
!isAttachmentPending &&
isAudio(attachments) &&
this.audioButtonRef &&
this.audioButtonRef.current
2019-11-07 21:36:16 +00:00
) {
event.preventDefault();
event.stopPropagation();
this.audioButtonRef.current.click();
2022-06-28 00:37:05 +00:00
return;
2019-11-07 21:36:16 +00:00
}
2023-08-16 20:54:39 +00:00
if (contact && contact.firstNumber && contact.serviceId) {
startConversation(contact.firstNumber, contact.serviceId);
event.preventDefault();
event.stopPropagation();
2022-06-28 00:37:05 +00:00
return;
}
if (contact) {
const signalAccount =
2023-08-16 20:54:39 +00:00
contact.firstNumber && contact.serviceId
? {
phoneNumber: contact.firstNumber,
2023-08-16 20:54:39 +00:00
serviceId: contact.serviceId,
}
: undefined;
pushPanelForConversation({
type: PanelType.ContactDetails,
args: {
contact,
signalAccount,
},
});
event.preventDefault();
event.stopPropagation();
}
2019-11-07 21:36:16 +00:00
};
2020-09-14 19:51:27 +00:00
public openGenericAttachment = (event?: React.MouseEvent): void => {
2022-12-14 18:12:04 +00:00
const {
id,
attachments,
saveAttachment,
timestamp,
kickOffAttachmentDownload,
} = this.props;
2019-11-07 21:36:16 +00:00
if (event) {
event.preventDefault();
event.stopPropagation();
}
if (!attachments || attachments.length !== 1) {
return;
}
const attachment = attachments[0];
2022-03-29 01:10:08 +00:00
if (!isDownloaded(attachment)) {
kickOffAttachmentDownload({
attachment,
messageId: id,
});
return;
}
2022-12-10 02:02:22 +00:00
saveAttachment(attachment, timestamp);
2019-11-07 21:36:16 +00:00
};
2020-09-14 19:51:27 +00:00
public handleClick = (event: React.MouseEvent): void => {
2019-11-07 21:36:16 +00:00
// We don't want clicks on body text to result in the 'default action' for the message
const { text } = this.props;
if (text && text.length > 0) {
return;
}
this.handleOpen(event);
};
2020-09-14 19:51:27 +00:00
public renderContainer(): JSX.Element {
2019-11-07 21:36:16 +00:00
const {
2021-04-27 22:11:59 +00:00
attachments,
attachmentDroppedDueToSize,
2021-05-28 16:15:17 +00:00
conversationColor,
customColor,
2020-04-29 21:24:12 +00:00
deletedForEveryone,
2019-11-07 21:36:16 +00:00
direction,
id,
isSticker,
2019-06-26 19:33:13 +00:00
isTapToView,
isTapToViewExpired,
isTapToViewError,
2022-11-04 13:22:07 +00:00
onContextMenu,
onKeyDown,
text,
2023-04-20 17:03:43 +00:00
textDirection,
} = this.props;
2023-03-20 22:23:53 +00:00
const { isTargeted } = this.state;
2019-06-26 19:33:13 +00:00
const isAttachmentPending = this.isAttachmentPending();
2019-11-07 21:36:16 +00:00
const width = this.getWidth();
const isEmojiOnly = this.canRenderStickerLikeEmoji();
2021-10-06 17:37:53 +00:00
const isStickerLike = isSticker || isEmojiOnly;
// If it's a mostly-normal gray incoming text box, we don't want to darken it as much
const lighterSelect =
2023-03-20 22:23:53 +00:00
isTargeted &&
direction === 'incoming' &&
!isStickerLike &&
(text || (!isVideo(attachments) && !isImage(attachments)));
2019-11-07 21:36:16 +00:00
const containerClassnames = classNames(
'module-message__container',
2021-04-27 22:11:59 +00:00
isGIF(attachments) ? 'module-message__container--gif' : null,
2023-03-20 22:23:53 +00:00
isTargeted ? 'module-message__container--targeted' : null,
lighterSelect ? 'module-message__container--targeted-lighter' : null,
2021-10-06 17:37:53 +00:00
!isStickerLike ? `module-message__container--${direction}` : null,
isEmojiOnly ? 'module-message__container--emoji' : null,
2019-11-07 21:36:16 +00:00
isTapToView ? 'module-message__container--with-tap-to-view' : null,
isTapToView && isTapToViewExpired
? 'module-message__container--with-tap-to-view-expired'
: null,
2021-10-06 17:37:53 +00:00
!isStickerLike && direction === 'outgoing'
2021-05-28 16:15:17 +00:00
? `module-message__container--outgoing-${conversationColor}`
2019-11-07 21:36:16 +00:00
: null,
isTapToView && isAttachmentPending && !isTapToViewExpired
? 'module-message__container--with-tap-to-view-pending'
: null,
isTapToView && isAttachmentPending && !isTapToViewExpired
2021-05-28 16:15:17 +00:00
? `module-message__container--${direction}-${conversationColor}-tap-to-view-pending`
2019-11-07 21:36:16 +00:00
: null,
isTapToViewError
? 'module-message__container--with-tap-to-view-error'
: null,
2021-01-27 21:15:43 +00:00
this.hasReactions() ? 'module-message__container--with-reactions' : null,
2020-04-29 21:24:12 +00:00
deletedForEveryone
? 'module-message__container--deleted-for-everyone'
2019-11-07 21:36:16 +00:00
: null
);
const containerStyles = {
2024-07-17 02:24:56 +00:00
width,
2019-11-07 21:36:16 +00:00
};
if (
!isStickerLike &&
!deletedForEveryone &&
!(attachmentDroppedDueToSize && !text) &&
direction === 'outgoing'
) {
2021-05-28 16:15:17 +00:00
Object.assign(containerStyles, getCustomColorStyle(customColor));
}
2019-11-07 21:36:16 +00:00
return (
2021-01-27 21:15:43 +00:00
<div className="module-message__container-outer">
2021-06-09 22:30:05 +00:00
<div
className={containerClassnames}
id={`message-accessibility-contents:${id}`}
2021-06-09 22:30:05 +00:00
style={containerStyles}
2022-11-04 13:22:07 +00:00
onContextMenu={onContextMenu}
role="row"
2022-11-04 13:22:07 +00:00
onKeyDown={onKeyDown}
onClick={this.handleClick}
onDoubleClick={ev => {
// Prevent double click from triggering the replyToMessage action
ev.stopPropagation();
}}
tabIndex={-1}
2021-06-09 22:30:05 +00:00
>
2021-01-27 21:15:43 +00:00
{this.renderAuthor()}
2023-04-20 17:03:43 +00:00
<div dir={TextDirectionToDirAttribute[textDirection]}>
{this.renderContents()}
</div>
2021-01-27 21:15:43 +00:00
</div>
{this.renderReactions(direction === 'outgoing')}
</div>
2019-11-07 21:36:16 +00:00
);
}
2023-03-20 22:23:53 +00:00
renderAltAccessibilityTree(): JSX.Element {
const { id, i18n, author } = this.props;
return (
<span className="module-message__alt-accessibility-tree">
<span id={`message-accessibility-label:${id}`}>
{author.isMe
? i18n('icu:messageAccessibilityLabel--outgoing')
2023-03-20 22:23:53 +00:00
: i18n('icu:messageAccessibilityLabel--incoming', {
author: author.title,
})}
</span>
<span id={`message-accessibility-description:${id}`}>
{this.renderText()}
</span>
</span>
);
}
public override render(): JSX.Element | null {
const {
2023-03-20 22:23:53 +00:00
id,
attachments,
direction,
i18n,
isSticker,
2023-03-20 22:23:53 +00:00
isSelected,
isSelectMode,
2023-01-13 00:24:59 +00:00
onKeyDown,
2023-04-03 20:16:27 +00:00
platform,
2023-01-13 00:24:59 +00:00
renderMenu,
shouldCollapseAbove,
shouldCollapseBelow,
2023-01-13 00:24:59 +00:00
timestamp,
2023-03-20 22:23:53 +00:00
onToggleSelect,
onReplyToMessage,
} = this.props;
2023-04-03 20:16:27 +00:00
const isMacOS = platform === 'darwin';
2023-03-20 22:23:53 +00:00
const { expired, expiring, isTargeted, imageBroken } = this.state;
if (expired) {
return null;
}
if (isSticker && (imageBroken || !attachments || !attachments.length)) {
return null;
}
2023-03-20 22:23:53 +00:00
let wrapperProps: DetailedHTMLProps<
HTMLAttributes<HTMLDivElement>,
HTMLDivElement
>;
if (isSelectMode) {
wrapperProps = {
role: 'checkbox',
'aria-checked': isSelected,
'aria-labelledby': `message-accessibility-label:${id}`,
'aria-describedby': `message-accessibility-description:${id}`,
tabIndex: 0,
onClick: event => {
event.preventDefault();
onToggleSelect(!isSelected, event.shiftKey);
},
onKeyDown: event => {
if (event.code === 'Space') {
event.preventDefault();
onToggleSelect(!isSelected, event.shiftKey);
}
},
};
} else {
wrapperProps = {
onMouseDown: () => {
this.hasSelectedTextRef.current = false;
},
// We use `onClickCapture` here and preven default/stop propagation to
// prevent other click handlers from firing.
onClickCapture: event => {
2023-04-03 20:16:27 +00:00
if (isMacOS ? event.metaKey : event.ctrlKey) {
if (this.hasSelectedTextRef.current) {
return;
}
const target = event.target as HTMLElement;
const link = target.closest('a[href], [role=link]');
if (event.currentTarget.contains(link)) {
return;
}
event.preventDefault();
event.stopPropagation();
onToggleSelect(true, false);
}
},
2023-03-20 22:23:53 +00:00
onDoubleClick: event => {
event.stopPropagation();
event.preventDefault();
if (!isSelectMode) {
onReplyToMessage();
}
},
};
}
return (
<div
aria-labelledby={`message-accessibility-contents:${id}`}
aria-roledescription={i18n('icu:Message__role-description')}
className={classNames(
2023-03-20 22:23:53 +00:00
'module-message__wrapper',
isSelectMode && 'module-message__wrapper--select-mode',
isSelected && 'module-message__wrapper--selected'
)}
role="article"
2023-03-20 22:23:53 +00:00
{...wrapperProps}
>
2023-03-20 22:23:53 +00:00
{isSelectMode && (
<>
<span
role="presentation"
className="module-message__select-checkbox"
/>
{this.renderAltAccessibilityTree()}
</>
)}
<div
className={classNames(
'module-message',
`module-message--${direction}`,
shouldCollapseAbove && 'module-message--collapsed-above',
shouldCollapseBelow && 'module-message--collapsed-below',
isTargeted ? 'module-message--targeted' : null,
expiring ? 'module-message--expired' : null
)}
data-testid={timestamp}
tabIndex={0}
// We need to have a role because screenreaders need to be able to focus here to
// read the message, but we can't be a button; that would break inner buttons.
role="row"
onKeyDown={onKeyDown}
onFocus={this.handleFocus}
ref={this.focusRef}
// @ts-expect-error -- React/TS doesn't know about inert
// eslint-disable-next-line react/no-unknown-property
inert={isSelectMode ? '' : undefined}
>
{this.renderError()}
{this.renderAvatar()}
{this.renderContainer()}
{renderMenu?.()}
</div>
</div>
);
}
}