/**
* Copyright (c) Meta Platforms, Inc. and affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*
*/
import type { CSSProperties } from 'react';
import { useEffect, useMemo, useState, useTransition, useRef } from 'react';
import { LexicalComposer } from '@lexical/react/LexicalComposer';
import { PlainTextPlugin } from '@lexical/react/LexicalPlainTextPlugin';
import { RichTextPlugin } from '@lexical/react/LexicalRichTextPlugin';
import { ContentEditable } from '@lexical/react/LexicalContentEditable';
import { HistoryPlugin } from '@lexical/react/LexicalHistoryPlugin';
import { OnChangePlugin } from '@lexical/react/LexicalOnChangePlugin';
import { ListPlugin } from '@lexical/react/LexicalListPlugin';
import { CheckListPlugin } from '@lexical/react/LexicalCheckListPlugin';
import { TabIndentationPlugin } from '@lexical/react/LexicalTabIndentationPlugin';
import LexicalErrorBoundary from '@lexical/react/LexicalErrorBoundary';
import { HeadingNode, QuoteNode } from '@lexical/rich-text';
import { ListItemNode, ListNode } from '@lexical/list';
import { CodeHighlightNode, CodeNode } from '@lexical/code';
import VariableLabelPickerPlugin from './plugins/VariableLabelPickerPlugin';
import ListDisplayFixPlugin from './plugins/ListDisplayFixPlugin';
import { Box, Flex } from '@chakra-ui/react';
import styles from './index.module.scss';
import VariablePlugin from './plugins/VariablePlugin';
import { VariableNode } from './plugins/VariablePlugin/node';
import type { EditorState, LexicalEditor } from 'lexical';
import OnBlurPlugin from './plugins/OnBlurPlugin';
import type { FormPropsType } from './type';
import { type EditorVariableLabelPickerType, type EditorVariablePickerType } from './type';
import { getNanoid } from '@fastgpt/global/common/string/tools';
import FocusPlugin from './plugins/FocusPlugin';
import { textToEditorState, editorStateToText } from './utils';
import { MaxLengthPlugin } from './plugins/MaxLengthPlugin';
import { VariableLabelNode } from './plugins/VariableLabelPlugin/node';
import VariableLabelPlugin from './plugins/VariableLabelPlugin';
import { useDeepCompareEffect } from 'ahooks';
import MarkdownPlugin from './plugins/MarkdownPlugin';
import MyIcon from '../../Icon';
import ListExitPlugin from './plugins/ListExitPlugin';
import KeyDownPlugin from './plugins/KeyDownPlugin';
import EditablePlugin from './plugins/EditablePlugin';
import SkillPickerPlugin from './plugins/SkillPickerPlugin';
import type { SkillLabelItemType } from './plugins/SkillLabelPlugin';
import SkillLabelPlugin from './plugins/SkillLabelPlugin';
import { SkillNode } from './plugins/SkillLabelPlugin/node';
import type { SkillOptionItemType } from './plugins/SkillPickerPlugin';
const Placeholder = ({ children, padding }: { children: React.ReactNode; padding: string }) => (
{children}
);
export type EditorProps = {
isRichText?: boolean;
variables?: EditorVariablePickerType[];
variableLabels?: EditorVariableLabelPickerType[];
onAddToolFromEditor?: (toolKey: string) => Promise;
onRemoveToolFromEditor?: (toolId: string) => void;
onConfigureTool?: (toolId: string) => void;
value: string;
skillOption?: SkillOptionItemType;
onRemoveSkill?: (id: string) => void;
onClickSkill?: (id: string) => void;
selectedSkills?: SkillLabelItemType[];
showOpenModal?: boolean;
minH?: number;
maxH?: number;
maxLength?: number;
placeholder?: string;
placeholderPadding?: string;
isInvalid?: boolean;
isDisabled?: boolean;
onKeyDown?: (e: React.KeyboardEvent) => void;
ExtensionPopover?: ((e: {
onChangeText: (text: string) => void;
iconButtonStyle: Record;
}) => React.ReactNode)[];
boxStyle?: CSSProperties;
};
export default function Editor({
isRichText = false,
minH = 200,
maxH = 400,
maxLength,
showOpenModal = true,
onOpenModal,
// {{}} 类型,已弃用
variables = [],
// /选择变量
variableLabels = [],
// @选择技能
skillOption,
selectedSkills,
onClickSkill,
onRemoveSkill,
onChange,
onChangeText,
onBlur,
value = '',
placeholder = '',
placeholderPadding = '12px 14px',
bg = 'white',
isInvalid,
isDisabled = false,
onKeyDown,
ExtensionPopover,
boxStyle
}: EditorProps &
FormPropsType & {
onOpenModal?: () => void;
onChange: (editor: LexicalEditor) => void;
onChangeText?: ((text: string) => void) | undefined;
onBlur?: (editor: LexicalEditor) => void;
}) {
const [key, setKey] = useState(getNanoid(6));
const [_, startSts] = useTransition();
const [focus, setFocus] = useState(false);
const [scrollHeight, setScrollHeight] = useState(0);
const editorOutputRef = useRef(value);
const initialConfig = {
namespace: isRichText ? 'richPromptEditor' : 'promptEditor',
nodes: [
VariableNode,
VariableLabelNode,
SkillNode,
// Only register rich text nodes when in rich text mode
...(isRichText
? [HeadingNode, ListNode, ListItemNode, QuoteNode, CodeNode, CodeHighlightNode]
: [])
],
editorState: textToEditorState(value, isRichText),
onError: (error: Error) => {
console.error('Lexical errror', error);
}
};
useDeepCompareEffect(() => {
if (focus && value === editorOutputRef.current) return;
setKey(getNanoid(6));
}, [value, variables, variableLabels, skillOption, selectedSkills]);
const showFullScreenIcon = useMemo(() => {
return showOpenModal && scrollHeight > maxH;
}, [showOpenModal, scrollHeight, maxH]);
const iconButtonStyle = useMemo(
() => ({
position: 'absolute' as const,
bottom: 1,
right: showFullScreenIcon ? '34px' : 2,
zIndex: 10,
cursor: 'pointer',
borderRadius: '6px',
background: 'rgba(255, 255, 255, 0.01)',
backdropFilter: 'blur(6.6666669845581055px)',
alignItems: 'center',
justifyContent: 'center',
w: 6,
h: 6
}),
[showFullScreenIcon]
);
return (
{/* Text type */}
{isRichText ? (
}
placeholder={{placeholder}}
ErrorBoundary={LexicalErrorBoundary}
/>
) : (
}
placeholder={{placeholder}}
ErrorBoundary={LexicalErrorBoundary}
/>
)}
<>
{/* Basic Plugin */}
<>
{
editorOutputRef.current = editorStateToText(editor);
const rootElement = editor.getRootElement();
setScrollHeight(rootElement?.scrollHeight || 0);
startSts(() => {
onChange?.(editor);
});
}}
/>
>
{/* 定制交互插件 */}
{variables.length > 0 && (
<>
{/* */}
>
)}
{variableLabels.length > 0 && (
<>
>
)}
{skillOption && onClickSkill && onRemoveSkill && selectedSkills && (
<>
>
)}
{isRichText && (
<>
>
)}
>
{onChangeText &&
ExtensionPopover?.map((Item, index) => (
))}
{showFullScreenIcon && (
)}
);
}