Files
Momento/memento-note/components/tiptap-live-block-extension.tsx
Antigravity f46654f574 feat: editor improvements and architectural grid prototype
Multiple feature additions and improvements across the application:

- NextGen Editor: drag handles, smart paste, block actions
- Structured views: Kanban and table layouts for notes
- Architectural Grid: new brainstorming/agent interface prototype
- Flashcards: SM-2 revision algorithm with AI generation
- MCP server: robustness improvements
- Graph/PDF chat: fix click propagation and copy behavior
- Various UI/UX enhancements and bug fixes

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-27 19:45:15 +00:00

248 lines
9.8 KiB
TypeScript

'use client'
import { Node, mergeAttributes } from '@tiptap/core'
import { ReactNodeViewRenderer, NodeViewWrapper, type NodeViewProps } from '@tiptap/react'
import { useEffect, useRef, useState, useCallback } from 'react'
import { Zap, AlertCircle, Unlink, ArrowRight, Trash2 } from 'lucide-react'
import { useLanguage } from '@/lib/i18n'
import { NOTE_REQUEST_SAVE_EVENT } from '@/lib/note-change-sync'
import { openNotePeek } from '@/lib/note-peek-sync'
declare module '@tiptap/core' {
interface Storage {
liveBlock: {
hostNoteId: string | null
}
}
}
// ---------------------------------------------------------------------------
// LiveBlock Node View
// ---------------------------------------------------------------------------
function LiveBlockView({ node, updateAttributes, deleteNode, editor, getPos }: NodeViewProps) {
const { t } = useLanguage()
const { sourceNoteId, blockId, snapshotContent, sourceNoteTitle } = node.attrs
const [localContent, setLocalContent] = useState(snapshotContent || '')
const [isDeleted, setIsDeleted] = useState(false)
const [isOffline, setIsOffline] = useState(false)
const [pulse, setPulse] = useState(false)
const pulseTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null)
const requestSave = useCallback(() => {
const hostNoteId = editor.storage.liveBlock?.hostNoteId
if (hostNoteId) {
window.dispatchEvent(new CustomEvent(NOTE_REQUEST_SAVE_EVENT, {
detail: { noteId: hostNoteId, reason: 'live-block-mutation' },
}))
}
}, [editor])
// Fetch current block status on mount
useEffect(() => {
if (!sourceNoteId || !blockId) return
fetch(`/api/blocks/${encodeURIComponent(blockId)}/status?sourceNoteId=${sourceNoteId}`)
.then(r => r.json())
.then((data: { exists: boolean; content: string; sourceNoteTitle: string }) => {
if (!data.exists) {
if (snapshotContent?.trim()) {
setLocalContent(snapshotContent)
return
}
setIsDeleted(true)
} else {
setLocalContent(data.content)
updateAttributes({ snapshotContent: data.content, sourceNoteTitle: data.sourceNoteTitle })
}
})
.catch(() => setIsOffline(true))
}, [sourceNoteId, blockId, snapshotContent, updateAttributes])
// Listen for real-time block update events
useEffect(() => {
const handleBlockUpdate = (e: CustomEvent) => {
if (e.detail?.blockId !== blockId) return
setLocalContent(e.detail.content)
updateAttributes({ snapshotContent: e.detail.content })
setPulse(true)
if (pulseTimerRef.current) clearTimeout(pulseTimerRef.current)
pulseTimerRef.current = setTimeout(() => setPulse(false), 1200)
}
const handleBlockDeleted = (e: CustomEvent) => {
if (e.detail?.blockId !== blockId) return
setIsDeleted(true)
}
window.addEventListener('live-block:update', handleBlockUpdate as EventListener)
window.addEventListener('live-block:deleted', handleBlockDeleted as EventListener)
return () => {
window.removeEventListener('live-block:update', handleBlockUpdate as EventListener)
window.removeEventListener('live-block:deleted', handleBlockDeleted as EventListener)
if (pulseTimerRef.current) clearTimeout(pulseTimerRef.current)
}
}, [blockId, updateAttributes])
/** Convertit le bloc live en paragraphe local (conserve le texte affiché). */
const handleDetach = useCallback(() => {
const pos = getPos()
if (typeof pos !== 'number') return
const currentNode = editor.state.doc.nodeAt(pos)
if (!currentNode) return
const text = localContent.trim()
const paragraph = editor.state.schema.nodes.paragraph.create(
null,
text ? editor.state.schema.text(text) : undefined,
)
editor.view.dispatch(editor.state.tr.replaceWith(pos, pos + currentNode.nodeSize, paragraph))
editor.commands.focus()
requestSave()
}, [editor, getPos, localContent, requestSave])
const handleRemove = useCallback(() => {
deleteNode()
requestSave()
}, [deleteNode, requestSave])
const handleOpenSource = useCallback((event: React.MouseEvent) => {
event.preventDefault()
event.stopPropagation()
openNotePeek({ noteId: sourceNoteId, blockId: blockId || undefined })
}, [sourceNoteId, blockId])
const borderClass = isDeleted
? 'border-l-rose-500 border-y-rose-200 border-r-rose-200 bg-rose-50/20 dark:border-l-red-700 dark:border-y-red-900/40 dark:border-r-red-900/40 dark:bg-red-950/5'
: isOffline
? 'border-l-amber-500 border-y-amber-200 border-r-amber-200 bg-amber-50/10 dark:border-l-amber-600 dark:border-y-amber-800/40 dark:border-r-amber-800/40'
: pulse
? 'border-l-blue-500 border-y-blue-300 border-r-blue-300 bg-blue-50/20 shadow-md shadow-blue-500/15 dark:bg-blue-950/10'
: 'border-l-blue-500 border-y-[#E8E6E3] border-r-[#E8E6E3] bg-blue-50/5 dark:border-y-zinc-800 dark:border-r-zinc-800 dark:bg-blue-950/5'
const headerTitle = isDeleted
? t('liveBlock.sourceDisconnected')
: (sourceNoteTitle || t('liveBlock.connectedNote'))
const actionButtonClass =
'text-[9.5px] font-bold flex items-center gap-1 transition-all shrink-0'
return (
<NodeViewWrapper>
<div className="group/liveblock my-4 not-prose">
<div className={`w-full rounded-xl border-l-[3px] border-y border-r transition-all duration-300 overflow-hidden ${borderClass}`}>
{/* Header */}
<div className="px-4 py-1.5 flex items-center justify-between bg-black/[0.015] dark:bg-white/[0.01] border-b border-black/[0.03] dark:border-white/[0.02]">
<div className="flex items-center gap-2 min-w-0">
{isDeleted ? (
<AlertCircle size={10} className="text-rose-500 shrink-0" />
) : (
<Zap size={10} className={`shrink-0 ${isOffline ? 'text-amber-500' : 'text-blue-500 fill-blue-500/20'}`} />
)}
<span className="text-[10px] font-sans font-medium text-[var(--color-concrete)] hover:text-[var(--color-ink)] transition-colors cursor-default max-w-[200px] truncate">
{headerTitle}
</span>
{isDeleted ? (
<span className="bg-rose-500/10 text-rose-600 dark:text-rose-400 font-bold px-1.5 rounded text-[8px] uppercase tracking-wider">
{t('liveBlock.statusDisconnected')}
</span>
) : isOffline ? (
<span className="bg-amber-500/10 text-amber-600 dark:text-amber-400 font-bold px-1.5 rounded text-[8px] uppercase tracking-wider">
{t('liveBlock.statusOffline')}
</span>
) : (
<span className="bg-blue-500/10 text-blue-600 dark:text-blue-400 font-bold px-1.5 rounded text-[8px] uppercase tracking-wider animate-pulse">
{t('liveBlock.statusLive')}
</span>
)}
</div>
<div
className={`flex items-center gap-2 shrink-0 ${isDeleted ? '' : 'opacity-0 group-hover/liveblock:opacity-100'} transition-opacity`}
>
<button
type="button"
onClick={handleDetach}
title={t('liveBlock.detachHelp')}
className={`${actionButtonClass} ${
isDeleted
? 'text-rose-600 hover:text-rose-500 dark:text-rose-400 hover:underline'
: 'text-[var(--color-concrete)] hover:text-[var(--color-ink)] dark:hover:text-[var(--color-dark-ink)]'
}`}
contentEditable={false}
>
<Unlink size={10} />
{t('liveBlock.detachLink')}
</button>
{!isDeleted && (
<button
type="button"
onClick={handleOpenSource}
className={`${actionButtonClass} text-blue-600 dark:text-blue-400 hover:underline`}
contentEditable={false}
>
{t('liveBlock.openSource')} <ArrowRight size={10} />
</button>
)}
<button
type="button"
onClick={handleRemove}
title={t('liveBlock.removeBlock')}
className={`${actionButtonClass} text-rose-600/80 hover:text-rose-600 dark:text-rose-400/80 dark:hover:text-rose-400`}
contentEditable={false}
>
<Trash2 size={10} />
</button>
</div>
</div>
{/* Content */}
<div className="px-4 py-3 bg-blue-500/[0.015] dark:bg-blue-500/[0.005]">
<p
className="text-sm leading-relaxed text-[var(--color-ink)] opacity-80 dark:text-[var(--color-dark-ink)] font-sans whitespace-pre-wrap"
contentEditable={false}
>
{localContent || t('liveBlock.emptyContent')}
</p>
</div>
</div>
</div>
</NodeViewWrapper>
)
}
// ---------------------------------------------------------------------------
// TipTap Node Definition
// ---------------------------------------------------------------------------
export const LiveBlockExtension = Node.create({
name: 'liveBlock',
group: 'block',
atom: true,
draggable: true,
selectable: true,
addStorage() {
return {
hostNoteId: null as string | null,
}
},
addAttributes() {
return {
sourceNoteId: { default: '' },
blockId: { default: '' },
snapshotContent: { default: '' },
sourceNoteTitle: { default: '' },
}
},
parseHTML() {
return [{ tag: 'div[data-live-block]' }]
},
renderHTML({ HTMLAttributes }) {
return ['div', mergeAttributes(HTMLAttributes, { 'data-live-block': 'true' })]
},
addNodeView() {
return ReactNodeViewRenderer(LiveBlockView)
},
})