Files
Momento/memento-note/components/block-action-menu.tsx
Antigravity 7fedfa8f50
Some checks failed
CI / Lint, Unit Tests & Build (push) Failing after 1m26s
CI / Deploy production (on server) (push) Has been skipped
feat: colonnes multi-colonnes (layout côte à côte style Notion)
- Architecture nested nodes: columns container → column children
- CSS seamless (pas de bordures, fin séparateur vertical entre colonnes)
- isolating: true sur les deux nœuds (curseur reste dans sa colonne)
- Commands addColumnBefore/addColumnAfter/deleteColumn
- Slash menu + drag handle + raccourci Mod+Shift+L
- i18n FR/EN complet
2026-06-14 18:49:15 +00:00

437 lines
17 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,
ArrowUp, ArrowDown, AlignLeft, ClipboardCopy, Sparkles,
ChevronsRightLeft, MessageSquareWarning, ListTree, Columns3,
Plus, Minus,
} from 'lucide-react'
import { replaceBlockWithStructuredView } from '@/components/tiptap-structured-view-block-extension'
import { insertToggleBlock, turnIntoToggleBlock } from '@/components/tiptap-toggle-extension'
import { turnIntoCalloutBlock } from '@/components/tiptap-callout-extension'
import { insertOutlineBlock } from '@/components/tiptap-outline-extension'
import { insertColumnsBlock } from '@/components/tiptap-columns-extension'
interface BlockActionMenuProps {
editor: Editor
onClose: () => void
anchorRect: DOMRect
blockPos: number
blockNode: PMNode | null
noteId?: string
sourceNoteTitle?: string
onBlockReferenceCopied?: (html: string) => void
}
type TurnIntoType =
| 'paragraph'
| 'heading1' | 'heading2' | 'heading3'
| 'bulletList' | 'orderedList' | 'taskList'
| 'blockquote' | 'codeBlock' | 'database'
interface TurnIntoOption {
id: TurnIntoType
icon: typeof Heading1
command?: (editor: Editor) => void
isDatabase?: boolean
}
/** Positionne le curseur dans le bloc avant d'appliquer une transformation */
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()
}
/**
* Applique une transformation en passant d'abord par paragraphe.
* Ceci corrige le cas heading → liste / blockquote / codeBlock qui échoue
* avec un toggle direct.
*/
function makeTurnCommand(cmd: (e: Editor) => void): (e: Editor) => void {
return (editor: Editor) => {
// clearNodes remet en paragraphe sans perte de texte
editor.chain().focus().clearNodes().run()
cmd(editor)
}
}
const TURN_INTO_OPTIONS: TurnIntoOption[] = [
{ id: 'paragraph', icon: AlignLeft, command: (e) => e.chain().focus().clearNodes().run() },
{ id: 'heading1', icon: Heading1, command: (e) => e.chain().focus().clearNodes().toggleHeading({ level: 1 }).run() },
{ id: 'heading2', icon: Heading2, command: (e) => e.chain().focus().clearNodes().toggleHeading({ level: 2 }).run() },
{ id: 'heading3', icon: Heading3, command: (e) => e.chain().focus().clearNodes().toggleHeading({ level: 3 }).run() },
{ id: 'bulletList', icon: List, command: makeTurnCommand((e) => e.chain().focus().toggleBulletList().run()) },
{ id: 'orderedList', icon: ListOrdered, command: makeTurnCommand((e) => e.chain().focus().toggleOrderedList().run()) },
{ id: 'taskList', icon: CheckSquare, command: makeTurnCommand((e) => e.chain().focus().toggleTaskList().run()) },
{ id: 'blockquote', icon: Quote, command: makeTurnCommand((e) => e.chain().focus().toggleBlockquote().run()) },
{ id: 'codeBlock', icon: CodeXml, command: makeTurnCommand((e) => e.chain().focus().toggleCodeBlock().run()) },
{ id: 'database', icon: Database, isDatabase: true },
]
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', '').trim()
}
return node.textContent?.trim() ?? ''
}
/** Déplace un bloc vers le haut (échange avec le frère précédent) */
function moveBlockUp(editor: Editor, blockPos: number, blockNode: PMNode): boolean {
const { doc } = editor.state
let prevPos = -1
let prevNode: PMNode | null = null
doc.forEach((node, offset) => {
if (offset + node.nodeSize === blockPos) {
prevPos = offset
prevNode = node
}
})
if (prevPos < 0 || !prevNode) return false
const safePrev = prevNode as PMNode
const tr = editor.state.tr
const from = prevPos
const to = blockPos + blockNode.nodeSize
tr.replaceWith(from, to, [blockNode.copy(blockNode.content), safePrev.copy(safePrev.content)])
editor.view.dispatch(tr)
// Repositionne le curseur dans le bloc déplacé (maintenant à prevPos)
const docSize = editor.state.doc.content.size
editor.chain().focus().setTextSelection(Math.min(prevPos + 1, docSize)).run()
return true
}
/** Déplace un bloc vers le bas (échange avec le frère suivant) */
function moveBlockDown(editor: Editor, blockPos: number, blockNode: PMNode): boolean {
const { doc } = editor.state
const nextPos = blockPos + blockNode.nodeSize
const nextNode = doc.nodeAt(nextPos)
if (!nextNode) return false
const tr = editor.state.tr
const from = blockPos
const to = nextPos + nextNode.nodeSize
tr.replaceWith(from, to, [nextNode.copy(nextNode.content), blockNode.copy(blockNode.content)])
editor.view.dispatch(tr)
const newPos = blockPos + nextNode.nodeSize
const docSize = editor.state.doc.content.size
editor.chain().focus().setTextSelection(Math.min(newPos + 1, docSize)).run()
return true
}
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 hoverTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null)
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 handleMoveUp = useCallback(() => {
if (blockNode && blockPos >= 0) {
const ok = moveBlockUp(editor, blockPos, blockNode)
if (!ok) toast(t('blockAction.moveUpFirst'))
}
onClose()
}, [editor, blockNode, blockPos, onClose, t])
const handleMoveDown = useCallback(() => {
if (blockNode && blockPos >= 0) {
const ok = moveBlockDown(editor, blockPos, blockNode)
if (!ok) toast(t('blockAction.moveDownLast'))
}
onClose()
}, [editor, blockNode, blockPos, onClose, t])
const handleCopyContent = useCallback(async () => {
const text = getBlockPlainContent(editor, blockPos, blockNode)
if (!text) { toast(t('blockAction.emptyBlock')); onClose(); return }
const ok = await copyTextToClipboard(text)
if (ok) toast.success(t('blockAction.contentCopied'))
else toast.error(t('blockAction.copyRefFailed'))
onClose()
}, [editor, blockPos, blockNode, onClose, t])
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 handleCreateDiagram = useCallback(async () => {
const text = getBlockPlainContent(editor, blockPos, blockNode)
if (!text || text.trim().length < 5) {
toast.error(t('blockAction.createDiagramEmpty') || "Le texte est trop court pour générer un diagramme.")
onClose()
return
}
onClose()
const toastId = toast.loading(t('blockAction.createDiagramLoading') || "Génération du diagramme Excalidraw par l'IA...")
try {
const { generateDiagramFromText } = await import('@/app/actions/diagram')
const res = await generateDiagramFromText(text)
if (!res.success || !res.canvasId) {
throw new Error(res.error || "La génération du diagramme a échoué.")
}
const canvasId = res.canvasId
const canvasRes = await fetch(`/api/canvas?id=${encodeURIComponent(canvasId)}`)
const canvasData = await canvasRes.json()
if (!canvasRes.ok || !canvasData.canvas?.data) {
throw new Error("Impossible de charger les données du diagramme généré.")
}
const { exportExcalidrawSceneToPngBlob } = await import('@/lib/client/excalidraw-export-image')
const blob = await exportExcalidrawSceneToPngBlob(canvasData.canvas.data)
if (!blob) {
throw new Error("Échec du rendu du diagramme en image PNG.")
}
const fd = new FormData()
fd.append('file', blob, `diagram-${canvasId.slice(-8)}.png`)
const up = await fetch('/api/upload', { method: 'POST', body: fd })
const upJson = await up.json()
if (!up.ok || !upJson.url) {
throw new Error("Échec du téléversement du diagramme généré.")
}
const imageUrl = upJson.url
const insertPos = blockPos + (blockNode ? blockNode.nodeSize : 0)
const htmlToInsert = `<p><a href="/lab?id=${canvasId}" target="_blank"><img src="${imageUrl}" alt="Diagramme Excalidraw" style="max-width: 100%; border-radius: 8px; border: 1px solid rgba(0,0,0,0.1);" /></a></p><p>🎨 <a href="/lab?id=${canvasId}" target="_blank"><strong>${t('blockAction.createDiagramSuccess') || "Éditer le diagramme dans le Lab Excalidraw"}</strong></a></p>`.trim();
editor.chain().focus().insertContentAt(insertPos, htmlToInsert).run()
toast.success(t('blockAction.createDiagramSuccess') || "Diagramme généré et inséré avec succès !", { id: toastId })
} catch (err: any) {
console.error('[handleCreateDiagram] Error:', err)
toast.error(err.message || "Une erreur est survenue lors de la génération.", { id: toastId })
}
}, [editor, blockPos, blockNode, onClose, 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])
const handleTurnIntoEnter = useCallback(() => {
if (hoverTimerRef.current) clearTimeout(hoverTimerRef.current)
setShowTurnInto(true)
}, [])
const handleTurnIntoLeave = useCallback(() => {
hoverTimerRef.current = setTimeout(() => setShowTurnInto(false), 200)
}, [])
const handleSubmenuEnter = useCallback(() => {
if (hoverTimerRef.current) clearTimeout(hoverTimerRef.current)
}, [])
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)
if (hoverTimerRef.current) clearTimeout(hoverTimerRef.current)
}
}, [onClose])
const menuLeft = anchorRect.right + 6
const menuTop = anchorRect.top - 4
const menuStyle: React.CSSProperties = {
position: 'fixed',
left: menuLeft > window.innerWidth - 230 ? anchorRect.left - 215 : menuLeft,
top: menuTop + 320 > window.innerHeight ? window.innerHeight - 330 : menuTop,
zIndex: 9999,
}
return createPortal(
<div ref={menuRef} style={menuStyle} className="block-action-menu">
{/* Actions de déplacement */}
<button type="button" className="block-action-item" onClick={handleMoveUp}>
<ArrowUp size={16} />
<span>{t('blockAction.moveUp')}</span>
</button>
<button type="button" className="block-action-item" onClick={handleMoveDown}>
<ArrowDown size={16} />
<span>{t('blockAction.moveDown')}</span>
</button>
<div className="block-action-separator" />
{/* Transformer en */}
<div
className="block-action-submenu-wrap"
onMouseLeave={handleTurnIntoLeave}
>
<button
type="button"
className="block-action-item block-action-submenu-trigger"
onClick={() => setShowTurnInto((v) => !v)}
onMouseEnter={handleTurnIntoEnter}
>
<Repeat size={16} />
<span>{t('blockAction.turnInto')}</span>
<ChevronRight size={14} className="ml-auto" />
</button>
{showTurnInto && (
<div className="block-action-submenu" onMouseEnter={handleSubmenuEnter}>
{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>
<div className="block-action-separator" />
{/* Section repliable */}
<button type="button" className="block-action-item" onClick={() => { turnIntoToggleBlock(editor, blockPos, blockNode); onClose() }}>
<ChevronsRightLeft size={16} />
<span>{t('richTextEditor.blockActionInsertToggle')}</span>
</button>
{/* Encadré */}
<button type="button" className="block-action-item" onClick={() => { turnIntoCalloutBlock(editor, blockPos, blockNode, 'info'); onClose() }}>
<MessageSquareWarning size={16} />
<span>{t('richTextEditor.blockActionInsertCallout')}</span>
</button>
{/* Sommaire */}
<button type="button" className="block-action-item" onClick={() => { insertOutlineBlock(editor); onClose() }}>
<ListTree size={16} />
<span>{t('richTextEditor.slashOutline')}</span>
</button>
{/* Colonnes */}
<button type="button" className="block-action-item" onClick={() => { insertColumnsBlock(editor, 2); onClose() }}>
<Columns3 size={16} />
<span>{t('richTextEditor.slashColumns')}</span>
</button>
<div className="block-action-separator" />
{/* Création de diagramme */}
<button type="button" className="block-action-item" onClick={() => { void handleCreateDiagram() }}>
<Sparkles size={16} className="text-amber-500 transition-all duration-200" />
<span className="font-medium text-amber-700 dark:text-amber-400">{t('blockAction.createDiagram')}</span>
</button>
<div className="block-action-separator" />
{/* Copier */}
<button type="button" className="block-action-item" onClick={() => { void handleCopyContent() }}>
<ClipboardCopy size={16} />
<span>{t('blockAction.copyContent')}</span>
</button>
<button type="button" className="block-action-item" onClick={() => { void handleCopyRef() }}>
<Link size={16} />
<span>{t('blockAction.copyRef')}</span>
</button>
<div className="block-action-separator" />
{/* Actions destructives */}
<button type="button" className="block-action-item" onClick={handleDuplicate}>
<Copy size={16} />
<span>{t('blockAction.duplicate')}</span>
</button>
<button type="button" className="block-action-item block-action-item--danger" onClick={handleDelete}>
<Trash2 size={16} />
<span>{t('blockAction.delete')}</span>
</button>
</div>,
document.body
)
}