// Copyright 2018-2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { KeyboardEvent } from 'react'; import React from 'react'; import type { AttachmentType } from '../../types/Attachment'; import { canBeDownloaded } from '../../types/Attachment'; import type { SizeClassType } from '../emoji/lib'; import { getSizeClass } from '../emoji/lib'; import { AtMentionify } from './AtMentionify'; import { Emojify } from './Emojify'; import { AddNewLines } from './AddNewLines'; import { Linkify } from './Linkify'; import type { BodyRangesType, LocalizerType, RenderTextCallbackType, } from '../../types/Util'; type OpenConversationActionType = ( conversationId: string, messageId?: string ) => void; export type Props = { direction?: 'incoming' | 'outgoing'; text: string; textAttachment?: Pick; /** If set, all emoji will be the same size. Otherwise, just one emoji will be large. */ disableJumbomoji?: boolean; /** If set, links will be left alone instead of turned into clickable `` tags. */ disableLinks?: boolean; i18n: LocalizerType; bodyRanges?: BodyRangesType; onIncreaseTextLength?: () => unknown; openConversation?: OpenConversationActionType; kickOffBodyDownload?: () => void; }; const renderEmoji = ({ text, key, sizeClass, renderNonEmoji, }: { i18n: LocalizerType; text: string; key: number; sizeClass?: SizeClassType; renderNonEmoji: RenderTextCallbackType; }) => ( ); /** * This component makes it very easy to use all three of our message formatting * components: `Emojify`, `Linkify`, and `AddNewLines`. Because each of them is fully * configurable with their `renderXXX` props, this component will assemble all three of * them for you. */ export function MessageBody({ bodyRanges, direction, disableJumbomoji, disableLinks, i18n, onIncreaseTextLength, openConversation, text, textAttachment, kickOffBodyDownload, }: Props): JSX.Element { const hasReadMore = Boolean(onIncreaseTextLength); const textWithSuffix = textAttachment?.pending || hasReadMore ? `${text}...` : text; const sizeClass = disableJumbomoji ? undefined : getSizeClass(text); const processedText = AtMentionify.preprocessMentions( textWithSuffix, bodyRanges ); const renderNewLines: RenderTextCallbackType = ({ text: textWithNewLines, key, }) => { return ( ( )} /> ); }; let pendingContent: React.ReactNode; if (hasReadMore) { pendingContent = null; } else if (textAttachment?.pending) { pendingContent = ( {i18n('downloading')} ); } else if ( textAttachment && canBeDownloaded(textAttachment) && kickOffBodyDownload ) { pendingContent = ( {' '} ); } return ( {disableLinks ? ( renderEmoji({ i18n, text: processedText, sizeClass, key: 0, renderNonEmoji: renderNewLines, }) ) : ( { return renderEmoji({ i18n, text: nonLinkText, sizeClass, key, renderNonEmoji: renderNewLines, }); }} /> )} {pendingContent} {onIncreaseTextLength ? ( ) : null} ); }