2023-01-15 13:49:33 -08:00
|
|
|
/*
|
|
|
|
MIT License
|
|
|
|
|
|
|
|
Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
|
|
|
|
|
|
This source code is licensed under the MIT license found in the
|
|
|
|
LICENSE file in the /app/soapbox/features/compose/editor directory.
|
|
|
|
*/
|
2023-03-02 10:42:31 -08:00
|
|
|
import { $convertFromMarkdownString, $convertToMarkdownString, TRANSFORMERS } from '@lexical/markdown';
|
2023-03-19 12:17:09 -07:00
|
|
|
import { AutoFocusPlugin } from '@lexical/react/LexicalAutoFocusPlugin';
|
2023-01-01 14:01:21 -08:00
|
|
|
import { LexicalComposer, InitialConfigType } from '@lexical/react/LexicalComposer';
|
2023-03-02 10:42:31 -08:00
|
|
|
import { useLexicalComposerContext } from '@lexical/react/LexicalComposerContext';
|
2022-12-01 12:32:55 -08:00
|
|
|
import { ContentEditable } from '@lexical/react/LexicalContentEditable';
|
|
|
|
import LexicalErrorBoundary from '@lexical/react/LexicalErrorBoundary';
|
2023-03-08 13:34:57 -08:00
|
|
|
import { HashtagPlugin } from '@lexical/react/LexicalHashtagPlugin';
|
2022-12-01 12:32:55 -08:00
|
|
|
import { HistoryPlugin } from '@lexical/react/LexicalHistoryPlugin';
|
|
|
|
import { LinkPlugin } from '@lexical/react/LexicalLinkPlugin';
|
2023-04-01 04:40:15 -07:00
|
|
|
import { ListPlugin } from '@lexical/react/LexicalListPlugin';
|
2022-12-01 12:32:55 -08:00
|
|
|
import { OnChangePlugin } from '@lexical/react/LexicalOnChangePlugin';
|
|
|
|
import { RichTextPlugin } from '@lexical/react/LexicalRichTextPlugin';
|
2023-03-02 10:42:31 -08:00
|
|
|
import clsx from 'clsx';
|
|
|
|
import { $createParagraphNode, $createTextNode, $getRoot } from 'lexical';
|
|
|
|
import React, { useEffect, useMemo, useState } from 'react';
|
2022-12-01 12:32:55 -08:00
|
|
|
import { FormattedMessage } from 'react-intl';
|
|
|
|
|
2023-03-02 10:42:31 -08:00
|
|
|
import { setEditorState } from 'soapbox/actions/compose';
|
|
|
|
import { useAppDispatch, useFeatures } from 'soapbox/hooks';
|
2023-01-01 14:01:21 -08:00
|
|
|
|
2022-12-01 12:32:55 -08:00
|
|
|
import nodes from './nodes';
|
2023-03-30 09:17:12 -07:00
|
|
|
import { AutosuggestPlugin } from './plugins/autosuggest-plugin';
|
2023-03-14 09:22:23 -07:00
|
|
|
import DraggableBlockPlugin from './plugins/draggable-block-plugin';
|
2022-12-01 12:32:55 -08:00
|
|
|
import FloatingLinkEditorPlugin from './plugins/floating-link-editor-plugin';
|
|
|
|
import FloatingTextFormatToolbarPlugin from './plugins/floating-text-format-toolbar-plugin';
|
2023-03-10 15:17:54 -08:00
|
|
|
import { MentionPlugin } from './plugins/mention-plugin';
|
2022-12-01 12:32:55 -08:00
|
|
|
|
2023-03-19 12:17:09 -07:00
|
|
|
const StatePlugin = ({ composeId }: { composeId: string }) => {
|
2023-03-02 10:42:31 -08:00
|
|
|
const dispatch = useAppDispatch();
|
|
|
|
const [editor] = useLexicalComposerContext();
|
|
|
|
|
2023-03-19 12:17:09 -07:00
|
|
|
useEffect(() => {
|
2023-03-02 10:42:31 -08:00
|
|
|
editor.registerUpdateListener(({ editorState }) => {
|
|
|
|
dispatch(setEditorState(composeId, editorState.isEmpty() ? null : JSON.stringify(editorState.toJSON())));
|
|
|
|
});
|
|
|
|
}, [editor]);
|
|
|
|
|
|
|
|
return null;
|
2022-12-01 12:32:55 -08:00
|
|
|
};
|
|
|
|
|
2023-03-02 10:42:31 -08:00
|
|
|
const ComposeEditor = React.forwardRef<string, any>(({ composeId, condensed, onFocus, autoFocus }, editorStateRef) => {
|
|
|
|
const dispatch = useAppDispatch();
|
2023-01-01 14:01:21 -08:00
|
|
|
const features = useFeatures();
|
|
|
|
|
2023-03-27 11:54:28 -07:00
|
|
|
const [suggestionsHidden, setSuggestionsHidden] = useState(true);
|
|
|
|
|
2023-03-02 10:42:31 -08:00
|
|
|
const initialConfig: InitialConfigType = useMemo(function() {
|
|
|
|
return {
|
|
|
|
namespace: 'ComposeForm',
|
|
|
|
onError: console.error,
|
|
|
|
nodes,
|
|
|
|
theme: {
|
2023-03-08 13:34:57 -08:00
|
|
|
hashtag: 'hover:underline text-primary-600 dark:text-accent-blue hover:text-primary-800 dark:hover:text-accent-blue',
|
2023-03-10 15:17:54 -08:00
|
|
|
mention: 'hover:underline text-primary-600 dark:text-accent-blue hover:text-primary-800 dark:hover:text-accent-blue',
|
2023-03-02 10:42:31 -08:00
|
|
|
text: {
|
|
|
|
bold: 'font-bold',
|
|
|
|
code: 'font-mono',
|
|
|
|
italic: 'italic',
|
|
|
|
strikethrough: 'line-through',
|
|
|
|
underline: 'underline',
|
|
|
|
underlineStrikethrough: 'underline-line-through',
|
|
|
|
},
|
2023-03-19 12:17:09 -07:00
|
|
|
heading: {
|
|
|
|
h1: 'text-2xl font-bold',
|
|
|
|
h2: 'text-xl font-bold',
|
|
|
|
h3: 'text-lg font-semibold',
|
|
|
|
},
|
2023-03-02 10:42:31 -08:00
|
|
|
},
|
|
|
|
editorState: dispatch((_, getState) => {
|
|
|
|
const state = getState();
|
|
|
|
const compose = state.compose.get(composeId);
|
|
|
|
|
|
|
|
if (!compose) return;
|
|
|
|
|
|
|
|
if (compose.editorState) {
|
|
|
|
return compose.editorState;
|
|
|
|
}
|
|
|
|
|
|
|
|
return function() {
|
|
|
|
if (compose.content_type === 'text/markdown') {
|
|
|
|
$convertFromMarkdownString(compose.text, TRANSFORMERS);
|
|
|
|
} else {
|
|
|
|
const paragraph = $createParagraphNode();
|
|
|
|
const textNode = $createTextNode(compose.text);
|
|
|
|
|
|
|
|
paragraph.append(textNode);
|
|
|
|
|
|
|
|
$getRoot()
|
|
|
|
.clear()
|
|
|
|
.append(paragraph);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}),
|
|
|
|
};
|
|
|
|
}, []);
|
|
|
|
|
2022-12-01 12:32:55 -08:00
|
|
|
const [floatingAnchorElem, setFloatingAnchorElem] =
|
|
|
|
useState<HTMLDivElement | null>(null);
|
|
|
|
|
|
|
|
const onRef = (_floatingAnchorElem: HTMLDivElement) => {
|
|
|
|
if (_floatingAnchorElem !== null) {
|
|
|
|
setFloatingAnchorElem(_floatingAnchorElem);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
|
|
|
<LexicalComposer initialConfig={initialConfig}>
|
2023-01-01 14:01:21 -08:00
|
|
|
<div className='lexical relative' data-markup>
|
2022-12-01 12:32:55 -08:00
|
|
|
<RichTextPlugin
|
|
|
|
contentEditable={
|
2023-01-01 14:01:21 -08:00
|
|
|
<div className='editor' ref={onRef} onFocus={onFocus}>
|
|
|
|
<ContentEditable
|
2023-03-14 09:22:23 -07:00
|
|
|
className={clsx('mr-4 py-2 outline-none transition-[min-height] motion-reduce:transition-none', {
|
2023-03-19 12:17:09 -07:00
|
|
|
'min-fh-[40px]': condensed,
|
2023-01-01 14:01:21 -08:00
|
|
|
'min-h-[100px]': !condensed,
|
|
|
|
})}
|
2023-03-19 12:17:09 -07:00
|
|
|
autoFocus={autoFocus}
|
2023-01-01 14:01:21 -08:00
|
|
|
/>
|
2022-12-01 12:32:55 -08:00
|
|
|
</div>
|
|
|
|
}
|
|
|
|
placeholder={(
|
2023-03-02 10:42:31 -08:00
|
|
|
<div className='pointer-events-none absolute top-2 select-none text-gray-600 dark:placeholder:text-gray-600'>
|
2022-12-01 12:32:55 -08:00
|
|
|
<FormattedMessage id='compose_form.placeholder' defaultMessage="What's on your mind" />
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
ErrorBoundary={LexicalErrorBoundary}
|
|
|
|
/>
|
2023-03-19 12:17:09 -07:00
|
|
|
{autoFocus && <AutoFocusPlugin />}
|
2022-12-01 12:32:55 -08:00
|
|
|
<OnChangePlugin onChange={(_, editor) => {
|
|
|
|
editor.update(() => {
|
|
|
|
if (editorStateRef) (editorStateRef as any).current = $convertToMarkdownString(TRANSFORMERS);
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
<HistoryPlugin />
|
2023-03-08 13:34:57 -08:00
|
|
|
<HashtagPlugin />
|
2023-03-30 09:17:12 -07:00
|
|
|
<MentionPlugin />
|
|
|
|
<AutosuggestPlugin composeId={composeId} suggestionsHidden={suggestionsHidden} setSuggestionsHidden={setSuggestionsHidden} />
|
2023-01-01 14:01:21 -08:00
|
|
|
{features.richText && <LinkPlugin />}
|
2023-04-01 04:40:15 -07:00
|
|
|
{features.richText && <ListPlugin />}
|
2023-01-01 14:01:21 -08:00
|
|
|
{features.richText && floatingAnchorElem && (
|
2022-12-01 12:32:55 -08:00
|
|
|
<>
|
2023-03-14 09:22:23 -07:00
|
|
|
<DraggableBlockPlugin anchorElem={floatingAnchorElem} />
|
2022-12-01 12:32:55 -08:00
|
|
|
<FloatingTextFormatToolbarPlugin anchorElem={floatingAnchorElem} />
|
|
|
|
<FloatingLinkEditorPlugin anchorElem={floatingAnchorElem} />
|
|
|
|
</>
|
2023-01-01 14:01:21 -08:00
|
|
|
)}
|
2023-03-19 12:17:09 -07:00
|
|
|
<StatePlugin composeId={composeId} />
|
2022-12-01 12:32:55 -08:00
|
|
|
</div>
|
|
|
|
</LexicalComposer>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
export default ComposeEditor;
|