lexical: remove MentionPlugin, only insert mention when selected from autosuggest

This commit is contained in:
Alex Gleason 2023-09-25 14:49:00 -05:00
parent 71bb8cc73e
commit e88b952a6f
No known key found for this signature in database
GPG key ID: 7211D1F99744FBB7
4 changed files with 25 additions and 64 deletions

View file

@ -23,7 +23,6 @@ import { useAppDispatch } from 'soapbox/hooks';
import { useNodes } from './nodes'; import { useNodes } from './nodes';
import AutosuggestPlugin from './plugins/autosuggest-plugin'; import AutosuggestPlugin from './plugins/autosuggest-plugin';
import FocusPlugin from './plugins/focus-plugin'; import FocusPlugin from './plugins/focus-plugin';
import MentionPlugin from './plugins/mention-plugin';
import RefPlugin from './plugins/ref-plugin'; import RefPlugin from './plugins/ref-plugin';
import StatePlugin from './plugins/state-plugin'; import StatePlugin from './plugins/state-plugin';
@ -162,7 +161,6 @@ const ComposeEditor = React.forwardRef<LexicalEditor, IComposeEditor>(({
/> />
<HistoryPlugin /> <HistoryPlugin />
<HashtagPlugin /> <HashtagPlugin />
<MentionPlugin />
<AutosuggestPlugin composeId={composeId} suggestionsHidden={suggestionsHidden} setSuggestionsHidden={setSuggestionsHidden} /> <AutosuggestPlugin composeId={composeId} suggestionsHidden={suggestionsHidden} setSuggestionsHidden={setSuggestionsHidden} />
<AutoLinkPlugin matchers={LINK_MATCHERS} /> <AutoLinkPlugin matchers={LINK_MATCHERS} />
<StatePlugin composeId={composeId} handleSubmit={handleSubmit} /> <StatePlugin composeId={composeId} handleSubmit={handleSubmit} />

View file

@ -60,7 +60,11 @@ class MentionNode extends TextNode {
} }
const $createMentionNode = (text = ''): MentionNode => $applyNodeReplacement(new MentionNode(text)); function $createMentionNode(text: string): MentionNode {
const node = new MentionNode(text);
node.setMode('token').toggleDirectionless();
return $applyNodeReplacement(node);
}
const $isMentionNode = ( const $isMentionNode = (
node: LexicalNode | null | undefined, node: LexicalNode | null | undefined,

View file

@ -19,6 +19,7 @@ import {
KEY_TAB_COMMAND, KEY_TAB_COMMAND,
LexicalEditor, LexicalEditor,
RangeSelection, RangeSelection,
TextNode,
} from 'lexical'; } from 'lexical';
import React, { import React, {
MutableRefObject, MutableRefObject,
@ -39,6 +40,7 @@ import { selectAccount } from 'soapbox/selectors';
import { textAtCursorMatchesToken } from 'soapbox/utils/suggestions'; import { textAtCursorMatchesToken } from 'soapbox/utils/suggestions';
import AutosuggestAccount from '../../components/autosuggest-account'; import AutosuggestAccount from '../../components/autosuggest-account';
import { $createMentionNode } from '../nodes/mention-node';
import type { AutoSuggestion } from 'soapbox/components/autosuggest-input'; import type { AutoSuggestion } from 'soapbox/components/autosuggest-input';
@ -303,27 +305,29 @@ const AutosuggestPlugin = ({
dispatch((dispatch, getState) => { dispatch((dispatch, getState) => {
const state = editor.getEditorState(); const state = editor.getEditorState();
const node = (state._selection as RangeSelection)?.anchor?.getNode(); const node = (state._selection as RangeSelection)?.anchor?.getNode();
const { leadOffset, matchingString } = resolution!.match;
/** Offset for the beginning of the matched text, including the token. */
const offset = leadOffset - 1;
if (typeof suggestion === 'object') { if (typeof suggestion === 'object') {
if (!suggestion.id) return; if (!suggestion.id) return;
dispatch(useEmoji(suggestion)); // eslint-disable-line react-hooks/rules-of-hooks dispatch(useEmoji(suggestion)); // eslint-disable-line react-hooks/rules-of-hooks
const { leadOffset, matchingString } = resolution!.match;
if (isNativeEmoji(suggestion)) { if (isNativeEmoji(suggestion)) {
node.spliceText(leadOffset - 1, matchingString.length, `${suggestion.native} `, true); node.spliceText(offset, matchingString.length, `${suggestion.native} `, true);
} else { } else {
node.spliceText(leadOffset - 1, matchingString.length, `${suggestion.colons} `, true); node.spliceText(offset, matchingString.length, `${suggestion.colons} `, true);
} }
} else if (suggestion[0] === '#') { } else if (suggestion[0] === '#') {
node.setTextContent(`${suggestion} `); node.setTextContent(`${suggestion} `);
node.select(); node.select();
} else { } else {
const content = selectAccount(getState(), suggestion)!.acct; const acct = selectAccount(getState(), suggestion)!.acct;
const result = (node as TextNode).splitText(offset, offset + matchingString.length);
node.setTextContent(`@${content} `); const textNode = result[1] ?? result[0];
node.select(); const mentionNode = textNode?.replace($createMentionNode(`@${acct}`));
mentionNode.insertAfter(new TextNode(' '));
mentionNode.selectNext();
} }
dispatch(clearComposeSuggestions(composeId)); dispatch(clearComposeSuggestions(composeId));
@ -337,13 +341,18 @@ const AutosuggestPlugin = ({
if (!node) return null; if (!node) return null;
if (['mention', 'hashtag'].includes(node.getType())) { if (['hashtag'].includes(node.getType())) {
const matchingString = node.getTextContent(); const matchingString = node.getTextContent();
return { leadOffset: 0, matchingString }; return { leadOffset: 0, matchingString };
} }
if (node.getType() === 'text') { if (node.getType() === 'text') {
const [leadOffset, matchingString] = textAtCursorMatchesToken(node.getTextContent(), (state._selection as RangeSelection)?.anchor?.offset, [':']); const [leadOffset, matchingString] = textAtCursorMatchesToken(
node.getTextContent(),
(state._selection as RangeSelection)?.anchor?.offset,
[':', '@'],
);
if (!leadOffset || !matchingString) return null; if (!leadOffset || !matchingString) return null;
return { leadOffset, matchingString }; return { leadOffset, matchingString };
} }

View file

@ -1,50 +0,0 @@
/**
* This source code is derived from code from Meta Platforms, Inc.
* and affiliates, licensed under the MIT license located in the
* LICENSE file in the /src/features/compose/editor directory.
*/
import { useLexicalComposerContext } from '@lexical/react/LexicalComposerContext';
import { useLexicalTextEntity } from '@lexical/react/useLexicalTextEntity';
import { useCallback, useEffect } from 'react';
import { $createMentionNode, MentionNode } from '../nodes/mention-node';
import type { TextNode } from 'lexical';
const MENTION_REGEX = /(?:^|\s)@[^\s]+/i;
const MentionPlugin = (): JSX.Element | null => {
const [editor] = useLexicalComposerContext();
useEffect(() => {
if (!editor.hasNodes([MentionNode])) {
throw new Error('MentionPlugin: MentionNode not registered on editor');
}
}, [editor]);
const createNode = useCallback((textNode: TextNode): MentionNode => {
return $createMentionNode(textNode.getTextContent());
}, []);
const getMatch = useCallback((text: string) => {
const match = MENTION_REGEX.exec(text);
if (!match) return null;
const length = match[0].length;
const start = match.index;
const end = start + length;
return { start, end };
}, []);
useLexicalTextEntity<MentionNode>(
getMatch,
MentionNode,
createNode,
);
return null;
};
export default MentionPlugin;