signal-desktop/ts/components/CompositionInput.tsx

979 lines
28 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2019 Signal Messenger, LLC
2020-10-30 20:34:04 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
import * as React from 'react';
import Delta from 'quill-delta';
import ReactQuill from 'react-quill';
import classNames from 'classnames';
import { Manager, Reference } from 'react-popper';
2023-01-19 00:59:47 +00:00
import type { DeltaStatic, KeyboardStatic, RangeStatic } from 'quill';
import Quill from 'quill';
2020-11-03 01:19:52 +00:00
import { MentionCompletion } from '../quill/mentions/completion';
import { FormattingMenu, QuillFormattingStyle } from '../quill/formatting/menu';
import { MonospaceBlot } from '../quill/formatting/monospaceBlot';
import { SpoilerBlot } from '../quill/formatting/spoilerBlot';
import { EmojiBlot, EmojiCompletion } from '../quill/emoji';
import type { EmojiPickDataType } from './emoji/EmojiPicker';
import { convertShortName } from './emoji/lib';
import type {
DraftBodyRanges,
HydratedBodyRangesType,
RangeNode,
} from '../types/BodyRange';
import { BodyRange, collapseRangeTree, insertRange } from '../types/BodyRange';
import type { LocalizerType, ThemeType } from '../types/Util';
import type { ConversationType } from '../state/ducks/conversations';
2021-11-17 18:38:52 +00:00
import type { PreferredBadgeSelectorType } from '../state/selectors/badges';
2021-10-26 22:59:08 +00:00
import { isValidUuid } from '../types/UUID';
2020-11-03 01:19:52 +00:00
import { MentionBlot } from '../quill/mentions/blot';
import {
matchEmojiImage,
matchEmojiBlot,
matchEmojiText,
2020-11-03 01:19:52 +00:00
} from '../quill/emoji/matchers';
import { matchMention } from '../quill/mentions/matchers';
import { MemberRepository } from '../quill/memberRepository';
2020-11-04 02:04:22 +00:00
import {
getDeltaToRemoveStaleMentions,
getTextAndRangesFromOps,
isMentionBlot,
getDeltaToRestartMention,
insertEmojiOps,
insertFormattingAndMentionsOps,
2020-11-04 02:04:22 +00:00
} from '../quill/util';
import { SignalClipboard } from '../quill/signal-clipboard';
2020-11-21 00:03:16 +00:00
import { DirectionalBlot } from '../quill/block/blot';
2021-05-11 00:50:43 +00:00
import { getClassNamesFor } from '../util/getClassNamesFor';
import * as log from '../logging/log';
2023-05-10 01:23:56 +00:00
import * as Errors from '../types/errors';
2022-10-04 23:17:15 +00:00
import { useRefMerger } from '../hooks/useRefMerger';
2023-01-30 20:16:09 +00:00
import type { LinkPreviewType } from '../types/message/LinkPreviews';
import { StagedLinkPreview } from './conversation/StagedLinkPreview';
import type { DraftEditMessageType } from '../model-types.d';
import { usePrevious } from '../hooks/usePrevious';
import {
matchBold,
matchItalic,
matchMonospace,
matchSpoiler,
matchStrikethrough,
} from '../quill/formatting/matchers';
import { missingCaseError } from '../util/missingCaseError';
Quill.register('formats/emoji', EmojiBlot);
2020-11-03 01:19:52 +00:00
Quill.register('formats/mention', MentionBlot);
2020-11-21 00:03:16 +00:00
Quill.register('formats/block', DirectionalBlot);
Quill.register('formats/monospace', MonospaceBlot);
Quill.register('formats/spoiler', SpoilerBlot);
Quill.register('modules/emojiCompletion', EmojiCompletion);
2020-11-03 01:19:52 +00:00
Quill.register('modules/mentionCompletion', MentionCompletion);
Quill.register('modules/formattingMenu', FormattingMenu);
Quill.register('modules/signalClipboard', SignalClipboard);
type HistoryStatic = {
undo(): void;
clear(): void;
};
export type InputApi = {
focus: () => void;
insertEmoji: (e: EmojiPickDataType) => void;
2023-01-19 00:59:47 +00:00
setContents: (
text: string,
draftBodyRanges?: HydratedBodyRangesType,
2023-01-19 00:59:47 +00:00
cursorToEnd?: boolean
) => void;
reset: () => void;
submit: () => void;
};
export type Props = Readonly<{
2022-03-04 21:14:52 +00:00
children?: React.ReactNode;
conversationId?: string;
i18n: LocalizerType;
disabled?: boolean;
draftEditMessage?: DraftEditMessageType;
getPreferredBadge: PreferredBadgeSelectorType;
large?: boolean;
inputApi?: React.MutableRefObject<InputApi | undefined>;
isFormattingEnabled: boolean;
isFormattingFlagEnabled: boolean;
isFormattingSpoilersFlagEnabled: boolean;
sendCounter: number;
skinTone?: EmojiPickDataType['skinTone'];
draftText?: string;
draftBodyRanges?: HydratedBodyRangesType;
moduleClassName?: string;
theme: ThemeType;
placeholder?: string;
sortedGroupMembers?: ReadonlyArray<ConversationType>;
2022-10-04 23:17:15 +00:00
scrollerRef?: React.RefObject<HTMLDivElement>;
onDirtyChange?(dirty: boolean): unknown;
onEditorStateChange?(options: {
bodyRanges: DraftBodyRanges;
caretLocation?: number;
conversationId: string | undefined;
messageText: string;
sendCounter: number;
}): unknown;
onTextTooLong(): unknown;
onPickEmoji(o: EmojiPickDataType): unknown;
onBlur?: () => unknown;
onFocus?: () => unknown;
2021-07-30 18:37:03 +00:00
onSubmit(
message: string,
bodyRanges: DraftBodyRanges,
2021-07-30 18:37:03 +00:00
timestamp: number
): unknown;
2022-10-04 23:17:15 +00:00
onScroll?: (ev: React.UIEvent<HTMLElement>) => void;
platform: string;
2023-05-10 01:23:56 +00:00
shouldHidePopovers?: boolean;
2022-03-04 21:14:52 +00:00
getQuotedMessage?(): unknown;
clearQuotedMessage?(): unknown;
2023-01-30 20:16:09 +00:00
linkPreviewLoading?: boolean;
linkPreviewResult?: LinkPreviewType;
onCloseLinkPreview?(conversationId: string): unknown;
}>;
const MAX_LENGTH = 64 * 1024;
2021-05-11 00:50:43 +00:00
const BASE_CLASS_NAME = 'module-composition-input';
2021-04-27 22:35:35 +00:00
2021-08-11 19:29:07 +00:00
export function CompositionInput(props: Props): React.ReactElement {
const {
2022-03-04 21:14:52 +00:00
children,
2023-01-30 20:16:09 +00:00
clearQuotedMessage,
conversationId,
disabled,
2023-01-30 20:16:09 +00:00
draftBodyRanges,
draftEditMessage,
2023-01-30 20:16:09 +00:00
draftText,
getPreferredBadge,
getQuotedMessage,
i18n,
inputApi,
isFormattingEnabled,
isFormattingFlagEnabled,
isFormattingSpoilersFlagEnabled,
2023-01-30 20:16:09 +00:00
large,
linkPreviewLoading,
linkPreviewResult,
2021-04-27 22:35:35 +00:00
moduleClassName,
2023-01-30 20:16:09 +00:00
onCloseLinkPreview,
onBlur,
onFocus,
onPickEmoji,
2022-10-04 23:17:15 +00:00
onScroll,
2023-01-30 20:16:09 +00:00
onSubmit,
2022-03-04 21:14:52 +00:00
placeholder,
platform,
2023-05-10 01:23:56 +00:00
shouldHidePopovers,
skinTone,
sendCounter,
sortedGroupMembers,
2021-11-17 18:38:52 +00:00
theme,
} = props;
2022-10-04 23:17:15 +00:00
const refMerger = useRefMerger();
2021-11-11 22:43:05 +00:00
const [emojiCompletionElement, setEmojiCompletionElement] =
React.useState<JSX.Element>();
const [formattingChooserElement, setFormattingChooserElement] =
React.useState<JSX.Element>();
2021-11-11 22:43:05 +00:00
const [lastSelectionRange, setLastSelectionRange] =
React.useState<RangeStatic | null>(null);
const [mentionCompletionElement, setMentionCompletionElement] =
React.useState<JSX.Element>();
const emojiCompletionRef = React.useRef<EmojiCompletion>();
2020-11-03 01:19:52 +00:00
const mentionCompletionRef = React.useRef<MentionCompletion>();
const quillRef = React.useRef<Quill>();
2022-10-04 23:17:15 +00:00
const scrollerRefInner = React.useRef<HTMLDivElement>(null);
const propsRef = React.useRef<Props>(props);
const canSendRef = React.useRef<boolean>(false);
2020-11-03 01:19:52 +00:00
const memberRepositoryRef = React.useRef<MemberRepository>(
new MemberRepository()
);
2023-05-10 01:23:56 +00:00
const [isMouseDown, setIsMouseDown] = React.useState<boolean>(false);
2020-11-03 01:19:52 +00:00
const generateDelta = (
text: string,
bodyRanges: HydratedBodyRangesType
2020-11-03 01:19:52 +00:00
): Delta => {
const textLength = text.length;
const tree = bodyRanges.reduce<ReadonlyArray<RangeNode>>((acc, range) => {
if (range.start < textLength) {
return insertRange(range, acc);
}
return acc;
}, []);
const nodes = collapseRangeTree({ tree, text });
const opsWithFormattingAndMentions = insertFormattingAndMentionsOps(nodes);
const opsWithEmojis = insertEmojiOps(opsWithFormattingAndMentions);
2020-11-03 01:19:52 +00:00
return new Delta(opsWithEmojis);
};
2020-09-12 00:46:52 +00:00
const getTextAndRanges = (): {
text: string;
bodyRanges: DraftBodyRanges;
} => {
const quill = quillRef.current;
2020-09-12 00:46:52 +00:00
if (quill === undefined) {
return { text: '', bodyRanges: [] };
}
2020-09-12 00:46:52 +00:00
const contents = quill.getContents();
2020-09-12 00:46:52 +00:00
if (contents === undefined) {
return { text: '', bodyRanges: [] };
}
2020-09-12 00:46:52 +00:00
const { ops } = contents;
2020-09-12 00:46:52 +00:00
if (ops === undefined) {
return { text: '', bodyRanges: [] };
}
2020-09-12 00:46:52 +00:00
const { text, bodyRanges } = getTextAndRangesFromOps(ops);
return {
text,
bodyRanges: bodyRanges.filter(range => {
if (BodyRange.isMention(range)) {
return true;
}
if (BodyRange.isFormatting(range)) {
if (!isFormattingFlagEnabled) {
return false;
}
if (
range.style === BodyRange.Style.SPOILER &&
!isFormattingSpoilersFlagEnabled
) {
return false;
}
return true;
}
throw missingCaseError(range);
}),
};
};
2020-09-12 00:46:52 +00:00
const focus = () => {
const quill = quillRef.current;
2020-09-12 00:46:52 +00:00
if (quill === undefined) {
return;
}
2020-09-12 00:46:52 +00:00
quill.focus();
};
2020-09-12 00:46:52 +00:00
const insertEmoji = (e: EmojiPickDataType) => {
const quill = quillRef.current;
if (quill === undefined) {
return;
}
const range = quill.getSelection();
const insertionRange = range || lastSelectionRange;
if (insertionRange == null) {
return;
2020-01-08 17:44:54 +00:00
}
const emoji = convertShortName(e.shortName, e.skinTone);
const delta = new Delta()
.retain(insertionRange.index)
.delete(insertionRange.length)
.insert({ emoji });
quill.updateContents(delta, 'user');
quill.setSelection(insertionRange.index + 1, 0, 'user');
};
const reset = () => {
const quill = quillRef.current;
if (quill === undefined) {
return;
2020-01-08 17:44:54 +00:00
}
canSendRef.current = true;
quill.setText('');
const historyModule = quill.getModule('history');
if (historyModule === undefined) {
return;
}
historyModule.clear();
};
2023-01-19 00:59:47 +00:00
const setContents = (
text: string,
bodyRanges?: HydratedBodyRangesType,
2023-01-19 00:59:47 +00:00
cursorToEnd?: boolean
) => {
2022-10-04 23:17:15 +00:00
const quill = quillRef.current;
if (quill === undefined) {
return;
}
const delta = generateDelta(text || '', bodyRanges || []);
2023-01-19 00:59:47 +00:00
2022-10-04 23:17:15 +00:00
canSendRef.current = true;
2023-01-19 00:59:47 +00:00
// We need to cast here because we use @types/quill@1.3.10 which has types
// for quill-delta even though quill-delta is written in TS and has its own
// types. @types/quill@2.0.0 fixes the issue but react-quill has a peer-dep
// on the older quill types.
quill.setContents(delta as unknown as DeltaStatic);
2022-10-04 23:17:15 +00:00
if (cursorToEnd) {
quill.setSelection(quill.getLength(), 0);
}
};
const submit = () => {
2021-07-30 18:37:03 +00:00
const timestamp = Date.now();
const quill = quillRef.current;
if (quill === undefined) {
return;
}
if (!canSendRef.current) {
log.warn(
'CompositionInput: Not submitting message - cannot send right now'
);
return;
}
const { text, bodyRanges } = getTextAndRanges();
2020-11-03 01:19:52 +00:00
log.info(
`CompositionInput: Submitting message ${timestamp} with ${bodyRanges.length} ranges`
2021-07-30 18:37:03 +00:00
);
canSendRef.current = false;
onSubmit(text, bodyRanges, timestamp);
};
if (inputApi) {
inputApi.current = {
focus,
insertEmoji,
2023-01-19 00:59:47 +00:00
setContents,
reset,
submit,
};
}
React.useEffect(() => {
propsRef.current = props;
}, [props]);
React.useEffect(() => {
canSendRef.current = !disabled;
}, [disabled]);
2021-04-27 22:35:35 +00:00
const onShortKeyEnter = (): boolean => {
submit();
return false;
};
const previousFormattingEnabled = usePrevious(
isFormattingEnabled,
isFormattingEnabled
);
const previousFormattingFlagEnabled = usePrevious(
isFormattingFlagEnabled,
isFormattingFlagEnabled
);
const previousFormattingSpoilersFlagEnabled = usePrevious(
isFormattingSpoilersFlagEnabled,
isFormattingSpoilersFlagEnabled
);
2023-05-10 01:23:56 +00:00
const previousIsMouseDown = usePrevious(isMouseDown, isMouseDown);
React.useEffect(() => {
const formattingChanged =
typeof previousFormattingEnabled === 'boolean' &&
previousFormattingEnabled !== isFormattingEnabled;
const flagChanged =
typeof previousFormattingFlagEnabled === 'boolean' &&
previousFormattingFlagEnabled !== isFormattingFlagEnabled;
const spoilersFlagChanged =
typeof previousFormattingSpoilersFlagEnabled === 'boolean' &&
previousFormattingSpoilersFlagEnabled !== isFormattingSpoilersFlagEnabled;
2023-05-10 01:23:56 +00:00
const mouseDownChanged = previousIsMouseDown !== isMouseDown;
const quill = quillRef.current;
2023-05-10 01:23:56 +00:00
const changed =
formattingChanged ||
flagChanged ||
spoilersFlagChanged ||
mouseDownChanged;
if (quill && changed) {
quill.getModule('formattingMenu').updateOptions({
isMenuEnabled: isFormattingEnabled,
2023-05-10 01:23:56 +00:00
isMouseDown,
isEnabled: isFormattingFlagEnabled,
isSpoilersEnabled: isFormattingSpoilersFlagEnabled,
});
quill.options.formats = getQuillFormats({
isFormattingFlagEnabled,
isFormattingSpoilersFlagEnabled,
});
}
}, [
isFormattingEnabled,
isFormattingFlagEnabled,
isFormattingSpoilersFlagEnabled,
2023-05-10 01:23:56 +00:00
isMouseDown,
previousFormattingEnabled,
previousFormattingFlagEnabled,
previousFormattingSpoilersFlagEnabled,
2023-05-10 01:23:56 +00:00
previousIsMouseDown,
quillRef,
]);
2021-04-27 22:35:35 +00:00
const onEnter = (): boolean => {
const quill = quillRef.current;
const emojiCompletion = emojiCompletionRef.current;
2020-11-03 01:19:52 +00:00
const mentionCompletion = mentionCompletionRef.current;
if (quill === undefined) {
return false;
}
2020-11-03 01:19:52 +00:00
if (emojiCompletion === undefined || mentionCompletion === undefined) {
return false;
}
if (emojiCompletion.results.length) {
emojiCompletion.completeEmoji();
return false;
}
2020-11-03 01:19:52 +00:00
if (mentionCompletion.results.length) {
mentionCompletion.completeMention();
return false;
}
2020-11-04 02:04:22 +00:00
if (propsRef.current.large) {
return true;
}
submit();
return false;
};
2021-04-27 22:35:35 +00:00
const onTab = (): boolean => {
const quill = quillRef.current;
const emojiCompletion = emojiCompletionRef.current;
2020-11-03 01:19:52 +00:00
const mentionCompletion = mentionCompletionRef.current;
if (quill === undefined) {
return false;
}
2019-08-06 19:18:37 +00:00
2020-11-03 01:19:52 +00:00
if (emojiCompletion === undefined || mentionCompletion === undefined) {
return false;
}
if (emojiCompletion.results.length) {
emojiCompletion.completeEmoji();
return false;
}
2020-11-03 01:19:52 +00:00
if (mentionCompletion.results.length) {
mentionCompletion.completeMention();
return false;
}
return true;
};
2021-04-27 22:35:35 +00:00
const onEscape = (): boolean => {
const quill = quillRef.current;
if (quill === undefined) {
return false;
}
const emojiCompletion = emojiCompletionRef.current;
2020-11-03 01:19:52 +00:00
const mentionCompletion = mentionCompletionRef.current;
if (emojiCompletion) {
if (emojiCompletion.results.length) {
emojiCompletion.reset();
return false;
}
}
2020-11-03 01:19:52 +00:00
if (mentionCompletion) {
if (mentionCompletion.results.length) {
mentionCompletion.clearResults();
2020-11-03 01:19:52 +00:00
return false;
}
}
2022-03-04 21:14:52 +00:00
if (getQuotedMessage?.()) {
clearQuotedMessage?.();
return false;
}
return true;
};
2021-04-27 22:35:35 +00:00
const onBackspace = (): boolean => {
const quill = quillRef.current;
if (quill === undefined) {
return true;
}
const selection = quill.getSelection();
if (!selection || selection.length > 0) {
return true;
}
const [blotToDelete] = quill.getLeaf(selection.index);
if (!isMentionBlot(blotToDelete)) {
return true;
}
const contents = quill.getContents(0, selection.index - 1);
const restartDelta = getDeltaToRestartMention(contents.ops);
quill.updateContents(restartDelta);
quill.setSelection(selection.index, 0);
return false;
};
2021-04-27 22:35:35 +00:00
const onChange = (): void => {
const quill = quillRef.current;
const { text, bodyRanges } = getTextAndRanges();
2020-11-03 01:19:52 +00:00
if (quill !== undefined) {
const historyModule: HistoryStatic = quill.getModule('history');
if (text.length > MAX_LENGTH) {
historyModule.undo();
propsRef.current.onTextTooLong();
return;
}
const { onEditorStateChange } = propsRef.current;
if (onEditorStateChange) {
// `getSelection` inside the `onChange` event handler will be the
// selection value _before_ the change occurs. `setTimeout` 0 here will
// let `getSelection` return the selection after the change takes place.
// this is necessary for `maybeGrabLinkPreview` as it needs the correct
// `caretLocation` from the post-change selection index value.
setTimeout(() => {
const selection = quill.getSelection();
onEditorStateChange({
bodyRanges,
caretLocation: selection ? selection.index : undefined,
conversationId,
messageText: text,
sendCounter,
});
}, 0);
}
}
if (propsRef.current.onDirtyChange) {
propsRef.current.onDirtyChange(text.length > 0);
}
};
2020-01-08 17:44:54 +00:00
React.useEffect(() => {
const quill = quillRef.current;
if (quill === undefined) {
return;
2020-01-08 17:44:54 +00:00
}
quill.enable(!disabled);
quill.focus();
}, [disabled]);
React.useEffect(() => {
const quill = quillRef.current;
if (quill === undefined) {
return;
}
function handleFocus() {
onFocus?.();
}
function handleBlur() {
onBlur?.();
}
quill.root.addEventListener('focus', handleFocus);
quill.root.addEventListener('blur', handleBlur);
return () => {
quill.root.removeEventListener('focus', handleFocus);
quill.root.removeEventListener('blur', handleBlur);
};
}, [onFocus, onBlur]);
React.useEffect(() => {
const emojiCompletion = emojiCompletionRef.current;
if (emojiCompletion === undefined || skinTone === undefined) {
return;
2020-01-08 17:44:54 +00:00
}
emojiCompletion.options.skinTone = skinTone;
}, [skinTone]);
React.useEffect(
() => () => {
const emojiCompletion = emojiCompletionRef.current;
2020-11-03 01:19:52 +00:00
const mentionCompletion = mentionCompletionRef.current;
2020-11-03 01:19:52 +00:00
if (emojiCompletion !== undefined) {
emojiCompletion.destroy();
}
2020-11-03 01:19:52 +00:00
if (mentionCompletion !== undefined) {
mentionCompletion.destroy();
}
},
[]
);
const removeStaleMentions = (
currentMembers: ReadonlyArray<ConversationType>
) => {
2020-11-03 01:19:52 +00:00
const quill = quillRef.current;
if (quill === undefined) {
return;
}
const { ops } = quill.getContents();
if (ops === undefined) {
return;
}
const currentMemberUuids = currentMembers
.map(m => m.uuid)
2021-10-26 22:59:08 +00:00
.filter(isValidUuid);
2020-11-03 01:19:52 +00:00
const newDelta = getDeltaToRemoveStaleMentions(ops, currentMemberUuids);
// eslint-disable-next-line @typescript-eslint/no-explicit-any
quill.updateContents(newDelta as any);
};
const memberIds = sortedGroupMembers ? sortedGroupMembers.map(m => m.id) : [];
2020-11-03 01:19:52 +00:00
React.useEffect(() => {
memberRepositoryRef.current.updateMembers(sortedGroupMembers || []);
removeStaleMentions(sortedGroupMembers || []);
2020-11-03 01:19:52 +00:00
// We are still depending on members, but ESLint can't tell
// Comparing the actual members list does not work for a couple reasons:
// * Arrays with the same objects are not "equal" to React
// * We only care about added/removed members, ignoring other attributes
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [JSON.stringify(memberIds)]);
2021-04-27 22:35:35 +00:00
// Placing all of these callbacks inside of a ref since Quill is not able
// to re-render. We want to make sure that all these callbacks are fresh
// so that the consumers of this component won't deal with stale props or
// stale state as the result of calling them.
const unstaleCallbacks = {
onBackspace,
onChange,
onEnter,
onEscape,
onPickEmoji,
onShortKeyEnter,
onTab,
};
const callbacksRef = React.useRef(unstaleCallbacks);
callbacksRef.current = unstaleCallbacks;
const reactQuill = React.useMemo(
() => {
2020-11-03 01:19:52 +00:00
const delta = generateDelta(draftText || '', draftBodyRanges || []);
return (
<ReactQuill
2021-05-11 00:50:43 +00:00
className={`${BASE_CLASS_NAME}__quill`}
2021-04-27 22:35:35 +00:00
onChange={() => callbacksRef.current.onChange()}
defaultValue={delta}
modules={{
toolbar: false,
signalClipboard: true,
clipboard: {
matchers: [
['IMG', matchEmojiImage],
['IMG', matchEmojiBlot],
['STRONG', matchBold],
['EM', matchItalic],
['SPAN', matchMonospace],
['S', matchStrikethrough],
['SPAN', matchSpoiler],
[Node.TEXT_NODE, matchEmojiText],
2020-11-03 01:19:52 +00:00
['SPAN', matchMention(memberRepositoryRef)],
],
},
keyboard: {
bindings: {
2021-04-27 22:35:35 +00:00
onEnter: {
key: 13,
handler: () => callbacksRef.current.onEnter(),
}, // 13 = Enter
onShortKeyEnter: {
key: 13, // 13 = Enter
shortKey: true,
2021-04-27 22:35:35 +00:00
handler: () => callbacksRef.current.onShortKeyEnter(),
},
2021-04-27 22:35:35 +00:00
onEscape: {
key: 27,
handler: () => callbacksRef.current.onEscape(),
}, // 27 = Escape
onBackspace: {
key: 8,
handler: () => callbacksRef.current.onBackspace(),
}, // 8 = Backspace
},
},
emojiCompletion: {
setEmojiPickerElement: setEmojiCompletionElement,
2021-04-27 22:35:35 +00:00
onPickEmoji: (emoji: EmojiPickDataType) =>
callbacksRef.current.onPickEmoji(emoji),
skinTone,
},
formattingMenu: {
i18n,
isMenuEnabled: isFormattingEnabled,
isEnabled: isFormattingFlagEnabled,
isSpoilersEnabled: isFormattingSpoilersFlagEnabled,
platform,
setFormattingChooserElement,
},
2020-11-03 01:19:52 +00:00
mentionCompletion: {
2021-11-17 18:38:52 +00:00
getPreferredBadge,
me: sortedGroupMembers
? sortedGroupMembers.find(foo => foo.isMe)
: undefined,
2020-11-03 01:19:52 +00:00
memberRepositoryRef,
setMentionPickerElement: setMentionCompletionElement,
i18n,
2021-11-17 18:38:52 +00:00
theme,
2020-11-03 01:19:52 +00:00
},
}}
formats={getQuillFormats({
isFormattingFlagEnabled,
isFormattingSpoilersFlagEnabled,
})}
2023-03-30 00:03:25 +00:00
placeholder={placeholder || i18n('icu:sendMessage')}
readOnly={disabled}
ref={element => {
if (element) {
const quill = element.getEditor();
const keyboard = quill.getModule('keyboard') as KeyboardStatic;
// force the tab handler to be prepended, otherwise it won't be
// executed: https://github.com/quilljs/quill/issues/1967
2021-04-27 22:35:35 +00:00
keyboard.bindings[9].unshift({
key: 9,
handler: () => callbacksRef.current.onTab(),
}); // 9 = Tab
// also, remove the default \t insertion binding
keyboard.bindings[9].pop();
// When loading a multi-line message out of a draft, the cursor
// position needs to be pushed to the end of the input manually.
quill.once('editor-change', () => {
2022-10-04 23:17:15 +00:00
const scroller = scrollerRefInner.current;
if (scroller != null) {
quill.scrollingContainer = scroller;
}
2020-11-03 01:19:52 +00:00
setTimeout(() => {
quill.setSelection(quill.getLength(), 0);
2020-11-04 02:04:22 +00:00
quill.root.classList.add('ql-editor--loaded');
2020-11-03 01:19:52 +00:00
}, 0);
});
quill.on(
'selection-change',
(newRange: RangeStatic, oldRange: RangeStatic) => {
// If we lose focus, store the last edit point for emoji insertion
if (newRange == null) {
setLastSelectionRange(oldRange);
}
}
);
quillRef.current = quill;
emojiCompletionRef.current = quill.getModule('emojiCompletion');
2021-11-11 22:43:05 +00:00
mentionCompletionRef.current =
quill.getModule('mentionCompletion');
}
}}
/>
);
},
// quill shouldn't re-render, all changes should take place exclusively
// through mutating the quill state directly instead of through props
// eslint-disable-next-line react-hooks/exhaustive-deps
[]
);
// The onClick handler below is only to make it easier for mouse users to focus the
// message box. In 'large' mode, the actual Quill text box can be one line while the
// visual text box is much larger. Clicking that should allow you to start typing,
// hence the click handler.
// eslint-disable-next-line max-len
/* eslint-disable jsx-a11y/click-events-have-key-events, jsx-a11y/no-static-element-interactions */
2021-05-11 00:50:43 +00:00
const getClassName = getClassNamesFor(BASE_CLASS_NAME, moduleClassName);
2023-05-10 01:23:56 +00:00
const onMouseDown = React.useCallback(
event => {
const target = event.target as HTMLElement;
try {
// If the user is actually clicking the format menu, we drop this event
if (target.closest('.module-composition-input__format-menu')) {
return;
}
setIsMouseDown(true);
2023-05-12 20:48:14 +00:00
const onMouseUp = () => {
setIsMouseDown(false);
window.removeEventListener('mouseup', onMouseUp);
};
window.addEventListener('mouseup', onMouseUp);
2023-05-10 01:23:56 +00:00
} catch (error) {
log.error(
'CompositionInput.onMouseDown: Failed to check event target',
Errors.toLogFormat(error)
);
}
setIsMouseDown(true);
},
[setIsMouseDown]
);
return (
<Manager>
<Reference>
{({ ref }) => (
2023-01-13 00:24:59 +00:00
<div
className={getClassName('__input')}
2023-05-09 15:52:03 +00:00
data-supertab
2023-01-13 00:24:59 +00:00
ref={ref}
data-testid="CompositionInput"
data-enabled={disabled ? 'false' : 'true'}
2023-05-10 01:23:56 +00:00
onMouseDown={onMouseDown}
2023-01-13 00:24:59 +00:00
>
{draftEditMessage && (
<div className={getClassName('__editing-message')}>
{i18n('icu:CompositionInput__editing-message')}
</div>
)}
{draftEditMessage?.attachmentThumbnail && (
<div className={getClassName('__editing-message__attachment')}>
<img
alt={i18n('icu:stagedImageAttachment', {
path: draftEditMessage.attachmentThumbnail,
})}
src={draftEditMessage.attachmentThumbnail}
/>
</div>
)}
2023-02-03 22:21:07 +00:00
{conversationId && linkPreviewLoading && linkPreviewResult && (
<StagedLinkPreview
{...linkPreviewResult}
moduleClassName="CompositionInput__link-preview"
i18n={i18n}
onClose={() => onCloseLinkPreview?.(conversationId)}
/>
)}
{children}
<div
2022-10-04 23:17:15 +00:00
ref={
props.scrollerRef
? refMerger(scrollerRefInner, props.scrollerRef)
: scrollerRefInner
}
onClick={focus}
2022-10-04 23:17:15 +00:00
onScroll={onScroll}
className={classNames(
2021-05-11 00:50:43 +00:00
getClassName('__input__scroller'),
2023-02-03 22:21:07 +00:00
!large && linkPreviewResult
2023-01-30 20:16:09 +00:00
? getClassName('__input__scroller--link-preview')
: null,
2022-03-04 21:14:52 +00:00
large ? getClassName('__input__scroller--large') : null,
children ? getClassName('__input--with-children') : null
)}
>
{reactQuill}
2023-05-10 01:23:56 +00:00
{shouldHidePopovers ? null : (
<>
{emojiCompletionElement}
{mentionCompletionElement}
{formattingChooserElement}
</>
)}
</div>
</div>
)}
</Reference>
</Manager>
);
2021-08-11 19:29:07 +00:00
}
function getQuillFormats({
isFormattingFlagEnabled,
isFormattingSpoilersFlagEnabled,
}: {
isFormattingFlagEnabled: boolean;
isFormattingSpoilersFlagEnabled: boolean;
}): Array<string> {
return [
// For image replacement (local-only)
'emoji',
// @mentions
'mention',
...(isFormattingFlagEnabled
? [
// Custom
...(isFormattingSpoilersFlagEnabled
? [QuillFormattingStyle.spoiler]
: []),
QuillFormattingStyle.monospace,
// Built-in
QuillFormattingStyle.bold,
QuillFormattingStyle.italic,
QuillFormattingStyle.strike,
]
: []),
];
}