signal-desktop/ts/components/CompositionInput.tsx

697 lines
19 KiB
TypeScript
Raw Normal View History

2022-03-04 21:14:52 +00:00
// Copyright 2019-2022 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';
import type { KeyboardStatic, RangeStatic } from 'quill';
import Quill from 'quill';
2020-11-03 01:19:52 +00:00
import { MentionCompletion } from '../quill/mentions/completion';
import { EmojiBlot, EmojiCompletion } from '../quill/emoji';
import type { EmojiPickDataType } from './emoji/EmojiPicker';
import { convertShortName } from './emoji/lib';
2022-11-10 04:59:36 +00:00
import type {
LocalizerType,
DraftBodyRangesType,
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,
matchReactEmoji,
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,
getTextAndMentionsFromOps,
isMentionBlot,
getDeltaToRestartMention,
insertMentionOps,
insertEmojiOps,
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';
2022-10-04 23:17:15 +00:00
import { useRefMerger } from '../hooks/useRefMerger';
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('modules/emojiCompletion', EmojiCompletion);
2020-11-03 01:19:52 +00:00
Quill.register('modules/mentionCompletion', MentionCompletion);
Quill.register('modules/signalClipboard', SignalClipboard);
type HistoryStatic = {
undo(): void;
clear(): void;
};
export type InputApi = {
focus: () => void;
insertEmoji: (e: EmojiPickDataType) => void;
2022-10-04 23:17:15 +00:00
setText: (text: string, 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;
getPreferredBadge: PreferredBadgeSelectorType;
large?: boolean;
inputApi?: React.MutableRefObject<InputApi | undefined>;
skinTone?: EmojiPickDataType['skinTone'];
draftText?: string;
2022-11-10 04:59:36 +00:00
draftBodyRanges?: DraftBodyRangesType;
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;
2020-11-03 01:19:52 +00:00
onEditorStateChange?(
conversationId: string | undefined,
2020-11-03 01:19:52 +00:00
messageText: string,
2022-11-10 04:59:36 +00:00
bodyRanges: DraftBodyRangesType,
2020-11-03 01:19:52 +00:00
caretLocation?: number
): unknown;
onTextTooLong(): unknown;
onPickEmoji(o: EmojiPickDataType): unknown;
2021-07-30 18:37:03 +00:00
onSubmit(
message: string,
2022-11-10 04:59:36 +00:00
mentions: DraftBodyRangesType,
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;
2022-03-04 21:14:52 +00:00
getQuotedMessage?(): unknown;
clearQuotedMessage?(): 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,
conversationId,
i18n,
disabled,
large,
inputApi,
2021-04-27 22:35:35 +00:00
moduleClassName,
onPickEmoji,
onSubmit,
2022-10-04 23:17:15 +00:00
onScroll,
2022-03-04 21:14:52 +00:00
placeholder,
skinTone,
2020-11-03 01:19:52 +00:00
draftText,
draftBodyRanges,
2021-11-17 18:38:52 +00:00
getPreferredBadge,
2020-11-03 01:19:52 +00:00
getQuotedMessage,
clearQuotedMessage,
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 [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()
);
2020-11-03 01:19:52 +00:00
const generateDelta = (
text: string,
2022-11-10 04:59:36 +00:00
bodyRanges: DraftBodyRangesType
2020-11-03 01:19:52 +00:00
): Delta => {
const initialOps = [{ insert: text }];
const opsWithMentions = insertMentionOps(initialOps, bodyRanges);
const opsWithEmojis = insertEmojiOps(opsWithMentions);
2020-11-03 01:19:52 +00:00
return new Delta(opsWithEmojis);
};
2020-09-12 00:46:52 +00:00
2022-11-10 04:59:36 +00:00
const getTextAndMentions = (): [string, DraftBodyRangesType] => {
const quill = quillRef.current;
2020-09-12 00:46:52 +00:00
if (quill === undefined) {
2020-11-03 01:19:52 +00:00
return ['', []];
}
2020-09-12 00:46:52 +00:00
const contents = quill.getContents();
2020-09-12 00:46:52 +00:00
if (contents === undefined) {
2020-11-03 01:19:52 +00:00
return ['', []];
}
2020-09-12 00:46:52 +00:00
const { ops } = contents;
2020-09-12 00:46:52 +00:00
if (ops === undefined) {
2020-11-03 01:19:52 +00:00
return ['', []];
}
2020-09-12 00:46:52 +00:00
2020-11-04 02:04:22 +00:00
return getTextAndMentionsFromOps(ops);
};
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();
};
2022-10-04 23:17:15 +00:00
const setText = (text: string, cursorToEnd?: boolean) => {
const quill = quillRef.current;
if (quill === undefined) {
return;
}
canSendRef.current = true;
quill.setText(text);
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;
}
2020-11-03 01:19:52 +00:00
const [text, mentions] = getTextAndMentions();
log.info(
2021-07-30 18:37:03 +00:00
`CompositionInput: Submitting message ${timestamp} with ${mentions.length} mentions`
);
canSendRef.current = false;
2021-07-30 18:37:03 +00:00
onSubmit(text, mentions, timestamp);
};
if (inputApi) {
inputApi.current = {
focus,
insertEmoji,
2022-10-04 23:17:15 +00:00
setText,
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;
};
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;
2020-11-03 01:19:52 +00:00
const [text, mentions] = getTextAndMentions();
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(
conversationId,
text,
mentions,
selection ? selection.index : undefined
);
}, 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 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],
2020-11-03 01:19:52 +00:00
['SPAN', matchReactEmoji],
[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,
},
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
},
}}
2020-11-03 01:19:52 +00:00
formats={['emoji', 'mention']}
2022-03-04 21:14:52 +00:00
placeholder={placeholder || i18n('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);
return (
<Manager>
<Reference>
{({ ref }) => (
2021-05-11 00:50:43 +00:00
<div className={getClassName('__input')} ref={ref}>
{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'),
2022-03-04 21:14:52 +00:00
large ? getClassName('__input__scroller--large') : null,
children ? getClassName('__input--with-children') : null
)}
>
{reactQuill}
{emojiCompletionElement}
2020-11-03 01:19:52 +00:00
{mentionCompletionElement}
</div>
</div>
)}
</Reference>
</Manager>
);
2021-08-11 19:29:07 +00:00
}