// Copyright 2019 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React, { memo, useCallback, useEffect, useRef, useState } from 'react'; import classNames from 'classnames'; import type { ReadonlyDeep } from 'type-fest'; import type { DraftBodyRanges, HydratedBodyRangesType, } from '../types/BodyRange'; import type { LocalizerType, ThemeType } from '../types/Util'; import type { ErrorDialogAudioRecorderType } from '../types/AudioRecorder'; import { RecordingState } from '../types/AudioRecorder'; import type { imageToBlurHash } from '../util/imageToBlurHash'; import { Spinner } from './Spinner'; import type { Props as EmojiButtonProps, EmojiButtonAPI, } from './emoji/EmojiButton'; import { EmojiButton } from './emoji/EmojiButton'; import type { Props as StickerButtonProps } from './stickers/StickerButton'; import { StickerButton } from './stickers/StickerButton'; import type { InputApi, Props as CompositionInputProps, } from './CompositionInput'; import { CompositionInput } from './CompositionInput'; import type { Props as MessageRequestActionsProps } from './conversation/MessageRequestActions'; import { MessageRequestActions } from './conversation/MessageRequestActions'; import type { PropsType as GroupV1DisabledActionsPropsType } from './conversation/GroupV1DisabledActions'; import { GroupV1DisabledActions } from './conversation/GroupV1DisabledActions'; import type { PropsType as GroupV2PendingApprovalActionsPropsType } from './conversation/GroupV2PendingApprovalActions'; import { GroupV2PendingApprovalActions } from './conversation/GroupV2PendingApprovalActions'; import { AnnouncementsOnlyGroupBanner } from './AnnouncementsOnlyGroupBanner'; import { AttachmentList } from './conversation/AttachmentList'; import type { AttachmentDraftType, InMemoryAttachmentDraftType, } from '../types/Attachment'; import { isImageAttachment, isVoiceMessage } from '../types/Attachment'; import type { AciString } from '../types/ServiceId'; import { AudioCapture } from './conversation/AudioCapture'; import { CompositionUpload } from './CompositionUpload'; import type { ConversationRemovalStage, ConversationType, PushPanelForConversationActionType, ShowConversationType, } from '../state/ducks/conversations'; import type { EmojiPickDataType } from './emoji/EmojiPicker'; import type { LinkPreviewType } from '../types/message/LinkPreviews'; import { MandatoryProfileSharingActions } from './conversation/MandatoryProfileSharingActions'; import { MediaQualitySelector } from './MediaQualitySelector'; import type { Props as QuoteProps } from './conversation/Quote'; import { Quote } from './conversation/Quote'; import { countStickers } from './stickers/lib'; import { useAttachFileShortcut, useEditLastMessageSent, useKeyboardShortcutsConditionally, } from '../hooks/useKeyboardShortcuts'; import { MediaEditor } from './MediaEditor'; import { isImageTypeSupported } from '../util/GoogleChrome'; import * as KeyboardLayout from '../services/keyboardLayout'; import { usePrevious } from '../hooks/usePrevious'; import { PanelType } from '../types/Panels'; import type { SmartCompositionRecordingDraftProps } from '../state/smart/CompositionRecordingDraft'; import { useEscapeHandling } from '../hooks/useEscapeHandling'; import type { SmartCompositionRecordingProps } from '../state/smart/CompositionRecording'; import SelectModeActions from './conversation/SelectModeActions'; import type { ShowToastAction } from '../state/ducks/toast'; import type { DraftEditMessageType } from '../model-types.d'; import type { ForwardMessagesPayload } from '../state/ducks/globalModals'; import { ForwardMessagesModalType } from './ForwardMessagesModal'; export type OwnProps = Readonly<{ acceptedMessageRequest: boolean | null; removalStage: ConversationRemovalStage | null; addAttachment: ( conversationId: string, attachment: InMemoryAttachmentDraftType ) => unknown; announcementsOnly: boolean | null; areWeAdmin: boolean | null; areWePending: boolean | null; areWePendingApproval: boolean | null; cancelRecording: () => unknown; completeRecording: ( conversationId: string, onRecordingComplete: (rec: InMemoryAttachmentDraftType) => unknown ) => unknown; convertDraftBodyRangesIntoHydrated: ( bodyRanges: DraftBodyRanges | undefined ) => HydratedBodyRangesType | undefined; conversationId: string; discardEditMessage: (id: string) => unknown; draftEditMessage: DraftEditMessageType | null; draftAttachments: ReadonlyArray; errorDialogAudioRecorderType: ErrorDialogAudioRecorderType | null; errorRecording: (e: ErrorDialogAudioRecorderType) => unknown; focusCounter: number; groupAdmins: Array; groupVersion: 1 | 2 | null; i18n: LocalizerType; imageToBlurHash: typeof imageToBlurHash; isDisabled: boolean; isFetchingUUID: boolean | null; isFormattingEnabled: boolean; isGroupV1AndDisabled: boolean | null; isMissingMandatoryProfileSharing: boolean | null; isSignalConversation: boolean | null; isActive: boolean; lastEditableMessageId: string | null; recordingState: RecordingState; messageCompositionId: string; shouldHidePopovers: boolean | null; isSMSOnly: boolean | null; left: boolean | null; linkPreviewLoading: boolean; linkPreviewResult: LinkPreviewType | null; onClearAttachments(conversationId: string): unknown; onCloseLinkPreview(conversationId: string): unknown; platform: string; showToast: ShowToastAction; processAttachments: (options: { conversationId: string; files: ReadonlyArray; }) => unknown; setMediaQualitySetting(conversationId: string, isHQ: boolean): unknown; sendStickerMessage( id: string, opts: { packId: string; stickerId: number } ): unknown; sendEditedMessage( conversationId: string, options: { bodyRanges?: DraftBodyRanges; message?: string; quoteAuthorAci?: AciString; quoteSentAt?: number; targetMessageId: string; } ): unknown; sendMultiMediaMessage( conversationId: string, options: { draftAttachments?: ReadonlyArray; bodyRanges?: DraftBodyRanges; message?: string; timestamp?: number; voiceNoteAttachment?: InMemoryAttachmentDraftType; } ): unknown; quotedMessageId: string | null; quotedMessageProps: null | ReadonlyDeep< Omit< QuoteProps, 'i18n' | 'onClick' | 'onClose' | 'withContentAbove' | 'isCompose' > >; quotedMessageAuthorAci: AciString | null; quotedMessageSentAt: number | null; removeAttachment: (conversationId: string, filePath: string) => unknown; scrollToMessage: (conversationId: string, messageId: string) => unknown; setComposerFocus: (conversationId: string) => unknown; setMessageToEdit(conversationId: string, messageId: string): unknown; setQuoteByMessageId( conversationId: string, messageId: string | undefined ): unknown; shouldSendHighQualityAttachments: boolean; showConversation: ShowConversationType; startRecording: (id: string) => unknown; theme: ThemeType; renderSmartCompositionRecording: ( props: SmartCompositionRecordingProps ) => JSX.Element; renderSmartCompositionRecordingDraft: ( props: SmartCompositionRecordingDraftProps ) => JSX.Element | null; selectedMessageIds: ReadonlyArray | undefined; toggleSelectMode: (on: boolean) => void; toggleForwardMessagesModal: ( payload: ForwardMessagesPayload, onForward: () => void ) => void; }>; export type Props = Pick< CompositionInputProps, | 'clearQuotedMessage' | 'draftText' | 'draftBodyRanges' | 'getPreferredBadge' | 'getQuotedMessage' | 'onEditorStateChange' | 'onTextTooLong' | 'sendCounter' | 'sortedGroupMembers' > & Pick< EmojiButtonProps, 'onPickEmoji' | 'onSetSkinTone' | 'recentEmojis' | 'skinTone' > & Pick< StickerButtonProps, | 'knownPacks' | 'receivedPacks' | 'installedPack' | 'installedPacks' | 'blessedPacks' | 'recentStickers' | 'clearInstalledStickerPack' | 'showIntroduction' | 'clearShowIntroduction' | 'showPickerHint' | 'clearShowPickerHint' > & MessageRequestActionsProps & Pick & Pick & { pushPanelForConversation: PushPanelForConversationActionType; } & OwnProps; export const CompositionArea = memo(function CompositionArea({ // Base props addAttachment, conversationId, convertDraftBodyRangesIntoHydrated, discardEditMessage, draftEditMessage, focusCounter, i18n, imageToBlurHash, isDisabled, isSignalConversation, isActive, lastEditableMessageId, messageCompositionId, pushPanelForConversation, platform, processAttachments, removeAttachment, sendEditedMessage, sendMultiMediaMessage, setComposerFocus, setMessageToEdit, setQuoteByMessageId, shouldHidePopovers, showToast, theme, // AttachmentList draftAttachments, onClearAttachments, // AudioCapture recordingState, startRecording, // StagedLinkPreview linkPreviewLoading, linkPreviewResult, onCloseLinkPreview, // Quote quotedMessageId, quotedMessageProps, quotedMessageAuthorAci, quotedMessageSentAt, scrollToMessage, // MediaQualitySelector setMediaQualitySetting, shouldSendHighQualityAttachments, // CompositionInput clearQuotedMessage, draftBodyRanges, draftText, getPreferredBadge, getQuotedMessage, isFormattingEnabled, onEditorStateChange, onTextTooLong, sendCounter, sortedGroupMembers, // EmojiButton onPickEmoji, onSetSkinTone, recentEmojis, skinTone, // StickerButton knownPacks, receivedPacks, installedPack, installedPacks, blessedPacks, recentStickers, clearInstalledStickerPack, sendStickerMessage, showIntroduction, clearShowIntroduction, showPickerHint, clearShowPickerHint, // Message Requests acceptedMessageRequest, areWePending, areWePendingApproval, conversationType, groupVersion, isBlocked, isHidden, isReported, isMissingMandatoryProfileSharing, left, removalStage, acceptConversation, blockConversation, reportSpam, blockAndReportSpam, deleteConversation, conversationName, addedByName, // GroupV1 Disabled Actions isGroupV1AndDisabled, showGV2MigrationDialog, // GroupV2 announcementsOnly, areWeAdmin, groupAdmins, cancelJoinRequest, showConversation, // SMS-only contacts isSMSOnly, isFetchingUUID, renderSmartCompositionRecording, renderSmartCompositionRecordingDraft, // Selected messages selectedMessageIds, toggleSelectMode, toggleForwardMessagesModal, }: Props): JSX.Element | null { const [dirty, setDirty] = useState(false); const [large, setLarge] = useState(false); const [attachmentToEdit, setAttachmentToEdit] = useState< AttachmentDraftType | undefined >(); const inputApiRef = useRef(); const emojiButtonRef = useRef(); const fileInputRef = useRef(null); const handleForceSend = useCallback(() => { setLarge(false); if (inputApiRef.current) { inputApiRef.current.submit(); } }, [inputApiRef, setLarge]); const draftEditMessageBody = draftEditMessage?.body; const editedMessageId = draftEditMessage?.targetMessageId; const handleSubmit = useCallback( ( message: string, bodyRanges: DraftBodyRanges, timestamp: number ): boolean => { if (!dirty) { return false; } emojiButtonRef.current?.close(); if (editedMessageId) { sendEditedMessage(conversationId, { bodyRanges, message, // sent timestamp for the quote quoteSentAt: quotedMessageSentAt ?? undefined, quoteAuthorAci: quotedMessageAuthorAci ?? undefined, targetMessageId: editedMessageId, }); } else { sendMultiMediaMessage(conversationId, { draftAttachments, bodyRanges, message, timestamp, }); } setLarge(false); return true; }, [ conversationId, dirty, draftAttachments, editedMessageId, quotedMessageSentAt, quotedMessageAuthorAci, sendEditedMessage, sendMultiMediaMessage, setLarge, ] ); const launchAttachmentPicker = useCallback(() => { const fileInput = fileInputRef.current; if (fileInput) { // Setting the value to empty so that onChange always fires in case // you add multiple photos. fileInput.value = ''; fileInput.click(); } }, []); function maybeEditAttachment(attachment: AttachmentDraftType) { if (!isImageTypeSupported(attachment.contentType)) { return; } setAttachmentToEdit(attachment); } const isComposerEmpty = !draftAttachments.length && !draftText && !draftEditMessage; const maybeEditMessage = useCallback(() => { if (!isComposerEmpty || !lastEditableMessageId) { return false; } setMessageToEdit(conversationId, lastEditableMessageId); return true; }, [ conversationId, isComposerEmpty, lastEditableMessageId, setMessageToEdit, ]); const [hasFocus, setHasFocus] = useState(false); const attachFileShortcut = useAttachFileShortcut(launchAttachmentPicker); const editLastMessageSent = useEditLastMessageSent(maybeEditMessage); useKeyboardShortcutsConditionally( hasFocus, attachFileShortcut, editLastMessageSent ); // Focus input on first mount const previousFocusCounter = usePrevious( focusCounter, focusCounter ); useEffect(() => { if (inputApiRef.current) { inputApiRef.current.focus(); setHasFocus(true); } }, []); // Focus input whenever explicitly requested useEffect(() => { if (focusCounter !== previousFocusCounter && inputApiRef.current) { inputApiRef.current.focus(); setHasFocus(true); } }, [inputApiRef, focusCounter, previousFocusCounter]); const withStickers = countStickers({ knownPacks, blessedPacks, installedPacks, receivedPacks, }) > 0; const previousMessageCompositionId = usePrevious( messageCompositionId, messageCompositionId ); const previousSendCounter = usePrevious(sendCounter, sendCounter); useEffect(() => { if (!inputApiRef.current) { return; } if ( previousMessageCompositionId !== messageCompositionId || previousSendCounter !== sendCounter ) { inputApiRef.current.reset(); } }, [messageCompositionId, sendCounter, previousMessageCompositionId, previousSendCounter]); const insertEmoji = useCallback( (e: EmojiPickDataType) => { if (inputApiRef.current) { inputApiRef.current.insertEmoji(e); onPickEmoji(e); } }, [inputApiRef, onPickEmoji] ); // We want to reset the state of Quill only if: // // - Our other device edits the message (edit history length would change) // - User begins editing another message. const editHistoryLength = draftEditMessage?.editHistoryLength; const hasEditHistoryChanged = usePrevious(editHistoryLength, editHistoryLength) !== editHistoryLength; const hasEditedMessageChanged = usePrevious(editedMessageId, editedMessageId) !== editedMessageId; const hasEditDraftChanged = hasEditHistoryChanged || hasEditedMessageChanged; useEffect(() => { if (!hasEditDraftChanged) { return; } inputApiRef.current?.setContents( draftEditMessageBody ?? '', draftBodyRanges ?? undefined, true ); }, [draftBodyRanges, draftEditMessageBody, hasEditDraftChanged]); const previousConversationId = usePrevious(conversationId, conversationId); useEffect(() => { if (conversationId === previousConversationId) { return; } if (!draftText) { inputApiRef.current?.setContents(''); return; } inputApiRef.current?.setContents( draftText, draftBodyRanges ?? undefined, true ); }, [conversationId, draftBodyRanges, draftText, previousConversationId]); const handleToggleLarge = useCallback(() => { setLarge(l => !l); }, [setLarge]); const shouldShowMicrophone = !large && isComposerEmpty; const showMediaQualitySelector = draftAttachments.some(isImageAttachment); const leftHandSideButtonsFragment = ( <>
setComposerFocus(conversationId)} recentEmojis={recentEmojis} skinTone={skinTone} onSetSkinTone={onSetSkinTone} />
{showMediaQualitySelector ? (
) : null} ); const micButtonFragment = shouldShowMicrophone ? (
) : null; const editMessageFragment = draftEditMessage ? ( <> {large &&
}
) : null; const isRecording = recordingState === RecordingState.Recording; const attButton = draftEditMessage || linkPreviewResult || isRecording ? undefined : (
); const sendButtonFragment = !draftEditMessage ? ( <>
) : null; const stickerButtonPlacement = large ? 'top-start' : 'top-end'; const stickerButtonFragment = !draftEditMessage && withStickers ? (
pushPanelForConversation({ type: PanelType.StickerManager, }) } onPickSticker={(packId, stickerId) => sendStickerMessage(conversationId, { packId, stickerId }) } showIntroduction={showIntroduction} clearShowIntroduction={clearShowIntroduction} showPickerHint={showPickerHint} clearShowPickerHint={clearShowPickerHint} position={stickerButtonPlacement} />
) : null; // Listen for cmd/ctrl-shift-x to toggle large composition mode useEffect(() => { const handler = (e: KeyboardEvent) => { const { shiftKey, ctrlKey, metaKey } = e; const key = KeyboardLayout.lookup(e); // When using the ctrl key, `key` is `'K'`. When using the cmd key, `key` is `'k'` const targetKey = key === 'k' || key === 'K'; const commandKey = platform === 'darwin' && metaKey; const controlKey = platform !== 'darwin' && ctrlKey; const commandOrCtrl = commandKey || controlKey; // cmd/ctrl-shift-k if (targetKey && shiftKey && commandOrCtrl) { e.preventDefault(); setLarge(x => !x); } }; document.addEventListener('keydown', handler); return () => { document.removeEventListener('keydown', handler); }; }, [platform, setLarge]); const handleRecordingBeforeSend = useCallback(() => { emojiButtonRef.current?.close(); }, [emojiButtonRef]); const handleEscape = useCallback(() => { if (linkPreviewResult) { onCloseLinkPreview(conversationId); } else if (draftEditMessage) { discardEditMessage(conversationId); } else if (quotedMessageId) { setQuoteByMessageId(conversationId, undefined); } }, [ conversationId, discardEditMessage, draftEditMessage, linkPreviewResult, onCloseLinkPreview, quotedMessageId, setQuoteByMessageId, ]); useEscapeHandling(handleEscape); if (isSignalConversation) { // TODO DESKTOP-4547 return
; } if (selectedMessageIds != null) { return ( { toggleSelectMode(false); }} onDeleteMessages={() => { window.reduxActions.globalModals.toggleDeleteMessagesModal({ conversationId, messageIds: selectedMessageIds, onDelete() { toggleSelectMode(false); }, }); }} onForwardMessages={() => { if (selectedMessageIds.length > 0) { toggleForwardMessagesModal( { type: ForwardMessagesModalType.Forward, messageIds: selectedMessageIds, }, () => { toggleSelectMode(false); } ); } }} showToast={showToast} /> ); } if ( isBlocked || areWePending || (!acceptedMessageRequest && removalStage !== 'justNotification') ) { return ( ); } if (conversationType === 'direct' && isSMSOnly) { return (
{isFetchingUUID ? ( ) : ( <>

{i18n('icu:CompositionArea--sms-only__title')}

{i18n('icu:CompositionArea--sms-only__body')}

)}
); } // If no message request, but we haven't shared profile yet, we show profile-sharing UI if ( !left && (conversationType === 'direct' || (conversationType === 'group' && groupVersion === 1)) && isMissingMandatoryProfileSharing ) { return ( ); } // If this is a V1 group, now disabled entirely, we show UI to help them upgrade if (!left && isGroupV1AndDisabled) { return ( ); } if (areWePendingApproval) { return ( ); } if (announcementsOnly && !areWeAdmin) { return ( ); } if (isRecording) { return renderSmartCompositionRecording({ onBeforeSend: handleRecordingBeforeSend, }); } if (draftAttachments.length === 1 && isVoiceMessage(draftAttachments[0])) { const voiceNoteAttachment = draftAttachments[0]; if (!voiceNoteAttachment.pending && voiceNoteAttachment.url) { return renderSmartCompositionRecordingDraft({ voiceNoteAttachment }); } } return (
{attachmentToEdit && 'url' in attachmentToEdit && attachmentToEdit.url && ( setAttachmentToEdit(undefined)} onDone={({ caption, captionBodyRanges, data, contentType, blurHash, }) => { const newAttachment = { ...attachmentToEdit, contentType, blurHash, data, size: data.byteLength, }; addAttachment(conversationId, newAttachment); setAttachmentToEdit(undefined); onEditorStateChange?.({ bodyRanges: captionBodyRanges ?? [], conversationId, messageText: caption ?? '', sendCounter, }); inputApiRef.current?.setContents( caption ?? '', convertDraftBodyRangesIntoHydrated(captionBodyRanges), true ); }} onPickEmoji={onPickEmoji} onTextTooLong={onTextTooLong} platform={platform} recentStickers={recentStickers} skinTone={skinTone} sortedGroupMembers={sortedGroupMembers} /> )}
{quotedMessageProps && (
scrollToMessage(conversationId, quotedMessageId) : undefined } onClose={() => { setQuoteByMessageId(conversationId, undefined); }} />
)} {draftAttachments.length ? (
onClearAttachments(conversationId)} onCloseAttachment={attachment => { if (attachment.path) { removeAttachment(conversationId, attachment.path); } }} />
) : null}
{!large ? leftHandSideButtonsFragment : null}
setHasFocus(false)} onFocus={() => setHasFocus(true)} onCloseLinkPreview={onCloseLinkPreview} onDirtyChange={setDirty} onEditorStateChange={onEditorStateChange} onPickEmoji={onPickEmoji} onSubmit={handleSubmit} onTextTooLong={onTextTooLong} platform={platform} sendCounter={sendCounter} shouldHidePopovers={shouldHidePopovers} skinTone={skinTone ?? null} sortedGroupMembers={sortedGroupMembers} theme={theme} />
{!large ? ( <> {stickerButtonFragment} {!dirty ? micButtonFragment : null} {editMessageFragment} {attButton} ) : null}
{large ? (
{leftHandSideButtonsFragment} {stickerButtonFragment} {attButton} {!dirty ? micButtonFragment : null} {editMessageFragment} {dirty || !shouldShowMicrophone ? sendButtonFragment : null}
) : null}
); });