'use client'; import { useEffect, useMemo } from 'react'; import { AIChatPlugin, AIPlugin } from '@udecode/plate-ai/react'; import { type SlateEditor, getAncestorNode, getEndPoint, getNodeString, } from '@udecode/plate-common'; import { type PlateEditor, focusEditor, useEditorPlugin, } from '@udecode/plate-common/react'; import { useIsSelecting } from '@udecode/plate-selection/react'; import { Album, BadgeHelp, Check, CornerUpLeft, FeatherIcon, ListEnd, ListMinus, ListPlus, PenLine, Wand, X, } from 'lucide-react'; import { CommandGroup, CommandItem } from './command'; export type EditorChatState = | 'cursorCommand' | 'cursorSuggestion' | 'selectionCommand' | 'selectionSuggestion'; export const aiChatItems = { accept: { icon: <Check />, label: 'Accept', value: 'accept', onSelect: ({ editor }) => { editor.getTransforms(AIChatPlugin).aiChat.accept(); focusEditor(editor, getEndPoint(editor, editor.selection!)); }, }, continueWrite: { icon: <PenLine />, label: 'Continue writing', value: 'continueWrite', onSelect: ({ editor }) => { const ancestorNode = getAncestorNode(editor); if (!ancestorNode) return; const isEmpty = getNodeString(ancestorNode[0]).trim().length === 0; void editor.getApi(AIChatPlugin).aiChat.submit({ mode: 'insert', prompt: isEmpty ? `<Document> {editor} </Document> Start writing a new paragraph AFTER <Document> ONLY ONE SENTENCE` : 'Continue writing AFTER <Block> ONLY ONE SENTENCE. DONT REPEAT THE TEXT.', }); }, }, discard: { icon: <X />, label: 'Discard', shortcut: 'Escape', value: 'discard', onSelect: ({ editor }) => { editor.getTransforms(AIPlugin).ai.undo(); editor.getApi(AIChatPlugin).aiChat.hide(); }, }, explain: { icon: <BadgeHelp />, label: 'Explain', value: 'explain', onSelect: ({ editor }) => { void editor.getApi(AIChatPlugin).aiChat.submit({ prompt: { default: 'Explain {editor}', selecting: 'Explain', }, }); }, }, fixSpelling: { icon: <Check />, label: 'Fix spelling & grammar', value: 'fixSpelling', onSelect: ({ editor }) => { void editor.getApi(AIChatPlugin).aiChat.submit({ prompt: 'Fix spelling and grammar', }); }, }, improveWriting: { icon: <Wand />, label: 'Improve writing', value: 'improveWriting', onSelect: ({ editor }) => { void editor.getApi(AIChatPlugin).aiChat.submit({ prompt: 'Improve the writing', }); }, }, insertBelow: { icon: <ListEnd />, label: 'Insert below', value: 'insertBelow', onSelect: ({ aiEditor, editor }) => { void editor.getTransforms(AIChatPlugin).aiChat.insertBelow(aiEditor); }, }, makeLonger: { icon: <ListPlus />, label: 'Make longer', value: 'makeLonger', onSelect: ({ editor }) => { void editor.getApi(AIChatPlugin).aiChat.submit({ prompt: 'Make longer', }); }, }, makeShorter: { icon: <ListMinus />, label: 'Make shorter', value: 'makeShorter', onSelect: ({ editor }) => { void editor.getApi(AIChatPlugin).aiChat.submit({ prompt: 'Make shorter', }); }, }, replace: { icon: <Check />, label: 'Replace selection', value: 'replace', onSelect: ({ aiEditor, editor }) => { void editor.getTransforms(AIChatPlugin).aiChat.replaceSelection(aiEditor); }, }, simplifyLanguage: { icon: <FeatherIcon />, label: 'Simplify language', value: 'simplifyLanguage', onSelect: ({ editor }) => { void editor.getApi(AIChatPlugin).aiChat.submit({ prompt: 'Simplify the language', }); }, }, summarize: { icon: <Album />, label: 'Add a summary', value: 'summarize', onSelect: ({ editor }) => { void editor.getApi(AIChatPlugin).aiChat.submit({ mode: 'insert', prompt: { default: 'Summarize {editor}', selecting: 'Summarize', }, }); }, }, tryAgain: { icon: <CornerUpLeft />, label: 'Try again', value: 'tryAgain', onSelect: ({ editor }) => { void editor.getApi(AIChatPlugin).aiChat.reload(); }, }, } satisfies Record< string, { icon: React.ReactNode; label: string; value: string; component?: React.ComponentType<{ menuState: EditorChatState }>; filterItems?: boolean; items?: { label: string; value: string }[]; shortcut?: string; onSelect?: ({ aiEditor, editor, }: { aiEditor: SlateEditor; editor: PlateEditor; }) => void; } >; const menuStateItems: Record< EditorChatState, { items: (typeof aiChatItems)[keyof typeof aiChatItems][]; heading?: string; }[] > = { cursorCommand: [ { items: [ aiChatItems.continueWrite, aiChatItems.summarize, aiChatItems.explain, ], }, ], cursorSuggestion: [ { items: [aiChatItems.accept, aiChatItems.discard, aiChatItems.tryAgain], }, ], selectionCommand: [ { items: [ aiChatItems.improveWriting, aiChatItems.makeLonger, aiChatItems.makeShorter, aiChatItems.fixSpelling, aiChatItems.simplifyLanguage, ], }, ], selectionSuggestion: [ { items: [ aiChatItems.replace, aiChatItems.insertBelow, aiChatItems.discard, aiChatItems.tryAgain, ], }, ], }; export const AIMenuItems = ({ aiEditorRef, setValue, }: { aiEditorRef: React.MutableRefObject<SlateEditor | null>; setValue: (value: string) => void; }) => { const { editor, useOption } = useEditorPlugin(AIChatPlugin); const { messages } = useOption('chat'); const isSelecting = useIsSelecting(); const menuState = useMemo(() => { if (messages && messages.length > 0) { return isSelecting ? 'selectionSuggestion' : 'cursorSuggestion'; } return isSelecting ? 'selectionCommand' : 'cursorCommand'; }, [isSelecting, messages]); const menuGroups = useMemo(() => { const items = menuStateItems[menuState]; return items; }, [menuState]); useEffect(() => { if (menuGroups.length > 0 && menuGroups[0].items.length > 0) { setValue(menuGroups[0].items[0].value); } }, [menuGroups, setValue]); return ( <> {menuGroups.map((group, index) => ( <CommandGroup key={index} heading={group.heading}> {group.items.map((menuItem) => ( <CommandItem key={menuItem.value} className='[&_svg]:text-zinc-500 dark:[&_svg]:text-zinc-400' value={menuItem.value} onSelect={() => { menuItem.onSelect?.({ aiEditor: aiEditorRef.current!, editor: editor, }); }} > {menuItem.icon} <span>{menuItem.label}</span> </CommandItem> ))} </CommandGroup> ))} </> ); };