Files
Momento/memento-note/components/block-action-menu.tsx

227 lines
7.6 KiB
TypeScript

'use client'
import { useEffect, useRef, useState, useCallback } from 'react'
import { createPortal } from 'react-dom'
import { useLanguage } from '@/lib/i18n'
import type { Editor } from '@tiptap/core'
import type { Node as PMNode } from '@tiptap/pm/model'
import { copyTextToClipboard } from '@/lib/editor/copy-text-to-clipboard'
import { ensureBlockReferenceId } from '@/lib/editor/block-reference-id'
import { rememberBlockReference } from '@/lib/editor/parse-block-reference'
import { toast } from 'sonner'
import {
Trash2, Copy, Repeat, Link, ChevronRight,
Heading1, Heading2, Heading3, List, ListOrdered,
CheckSquare, Quote, CodeXml, Database,
} from 'lucide-react'
import { replaceBlockWithStructuredView } from '@/components/tiptap-structured-view-block-extension'
interface BlockActionMenuProps {
editor: Editor
onClose: () => void
anchorRect: DOMRect
blockPos: number
blockNode: PMNode | null
noteId?: string
sourceNoteTitle?: string
onBlockReferenceCopied?: (html: string) => void
}
type TurnIntoType =
| 'heading1' | 'heading2' | 'heading3'
| 'bulletList' | 'orderedList' | 'taskList'
| 'blockquote' | 'codeBlock' | 'database'
interface TurnIntoOption {
id: TurnIntoType
icon: typeof Heading1
command?: (editor: Editor) => void
isDatabase?: boolean
}
const TURN_INTO_OPTIONS: TurnIntoOption[] = [
{ id: 'heading1', icon: Heading1, command: (e) => e.chain().focus().toggleHeading({ level: 1 }).run() },
{ id: 'heading2', icon: Heading2, command: (e) => e.chain().focus().toggleHeading({ level: 2 }).run() },
{ id: 'heading3', icon: Heading3, command: (e) => e.chain().focus().toggleHeading({ level: 3 }).run() },
{ id: 'bulletList', icon: List, command: (e) => e.chain().focus().toggleBulletList().run() },
{ id: 'orderedList', icon: ListOrdered, command: (e) => e.chain().focus().toggleOrderedList().run() },
{ id: 'taskList', icon: CheckSquare, command: (e) => e.chain().focus().toggleTaskList().run() },
{ id: 'blockquote', icon: Quote, command: (e) => e.chain().focus().toggleBlockquote().run() },
{ id: 'codeBlock', icon: CodeXml, command: (e) => e.chain().focus().toggleCodeBlock().run() },
{ id: 'database', icon: Database, isDatabase: true },
]
function focusBlock(editor: Editor, blockPos: number) {
const docSize = editor.state.doc.content.size
const cursorPos = Math.min(blockPos + 1, docSize)
editor.chain().focus().setTextSelection(cursorPos).run()
}
function getBlockPlainContent(editor: Editor, blockPos: number, blockNode: PMNode | null): string {
const node = blockNode ?? (blockPos >= 0 ? editor.state.doc.nodeAt(blockPos) : null)
if (!node || blockPos < 0) return ''
const from = blockPos + 1
const to = blockPos + node.nodeSize - 1
if (to > from) {
return editor.state.doc.textBetween(from, to, '\n', '\0').trim()
}
return node.textContent?.trim() ?? ''
}
export function BlockActionMenu({
editor,
onClose,
anchorRect,
blockPos,
blockNode,
noteId,
sourceNoteTitle,
onBlockReferenceCopied,
}: BlockActionMenuProps) {
const { t } = useLanguage()
const menuRef = useRef<HTMLDivElement>(null)
const [showTurnInto, setShowTurnInto] = useState(false)
const handleDelete = useCallback(() => {
if (blockNode && blockPos >= 0) {
editor.chain().focus().deleteRange({ from: blockPos, to: blockPos + blockNode.nodeSize }).run()
}
onClose()
}, [editor, blockNode, blockPos, onClose])
const handleDuplicate = useCallback(() => {
if (blockNode && blockPos >= 0) {
const insertPos = blockPos + blockNode.nodeSize
editor.view.dispatch(
editor.state.tr.insert(insertPos, blockNode.copy())
)
editor.commands.focus()
}
onClose()
}, [editor, blockNode, blockPos, onClose])
const handleCopyRef = useCallback(async () => {
if (!noteId?.trim()) {
toast.error(t('blockAction.copyRefNoNote'))
onClose()
return
}
const blockId = ensureBlockReferenceId(editor, blockPos, blockNode)
if (!blockId) {
toast.error(t('blockAction.copyRefUnsupported'))
onClose()
return
}
const html = editor.getHTML()
const blockContent = getBlockPlainContent(editor, blockPos, blockNode)
onBlockReferenceCopied?.(html)
const ref = `${window.location.origin}/home?openNote=${encodeURIComponent(noteId)}#block-${encodeURIComponent(blockId)}`
const copied = await copyTextToClipboard(ref)
if (copied) {
rememberBlockReference(ref, { blockContent, sourceNoteTitle })
toast.success(t('blockAction.copied'))
} else {
toast.error(t('blockAction.copyRefFailed'))
}
onClose()
}, [blockNode, blockPos, editor, noteId, onBlockReferenceCopied, onClose, sourceNoteTitle, t])
const handleTurnInto = useCallback((option: TurnIntoOption) => {
if (blockPos >= 0 && blockNode) {
if (option.isDatabase) {
const notebookId = (editor.storage as any).structuredViewBlock?.notebookId as string | null
replaceBlockWithStructuredView(editor, blockPos, blockNode, notebookId)
} else if (option.command) {
focusBlock(editor, blockPos)
option.command(editor)
}
}
setShowTurnInto(false)
onClose()
}, [editor, blockNode, blockPos, onClose])
useEffect(() => {
function handleClickOutside(e: MouseEvent) {
if (menuRef.current && !menuRef.current.contains(e.target as Node)) {
onClose()
}
}
function handleKeyDown(e: KeyboardEvent) {
if (e.key === 'Escape') onClose()
}
document.addEventListener('mousedown', handleClickOutside)
document.addEventListener('keydown', handleKeyDown)
return () => {
document.removeEventListener('mousedown', handleClickOutside)
document.removeEventListener('keydown', handleKeyDown)
}
}, [onClose])
const menuStyle: React.CSSProperties = {
position: 'fixed',
left: anchorRect.right + 6,
top: anchorRect.top - 4,
zIndex: 9999,
}
if (Number(menuStyle.left) > window.innerWidth - 220) {
menuStyle.left = anchorRect.left - 210
}
if (Number(menuStyle.top) + 300 > window.innerHeight) {
menuStyle.top = window.innerHeight - 310
}
return createPortal(
<div ref={menuRef} style={menuStyle} className="block-action-menu">
<button type="button" className="block-action-item" onClick={handleDelete}>
<Trash2 size={16} />
<span>{t('blockAction.delete')}</span>
</button>
<button type="button" className="block-action-item" onClick={handleDuplicate}>
<Copy size={16} />
<span>{t('blockAction.duplicate')}</span>
</button>
<div className="block-action-separator" />
<button
type="button"
className="block-action-item block-action-submenu-trigger"
onClick={() => setShowTurnInto(!showTurnInto)}
onMouseEnter={() => setShowTurnInto(true)}
>
<Repeat size={16} />
<span>{t('blockAction.turnInto')}</span>
<ChevronRight size={14} className="ml-auto" />
</button>
{showTurnInto && (
<div className="block-action-submenu">
{TURN_INTO_OPTIONS.map((opt) => (
<button
key={opt.id}
type="button"
className="block-action-item"
onClick={() => handleTurnInto(opt)}
>
<opt.icon size={16} />
<span>{t(`blockAction.turnInto_${opt.id}`)}</span>
</button>
))}
</div>
)}
<div className="block-action-separator" />
<button type="button" className="block-action-item" onClick={() => { void handleCopyRef() }}>
<Link size={16} />
<span>{t('blockAction.copyRef')}</span>
</button>
</div>,
document.body
)
}