feat: Find & Replace dans l'éditeur (Ctrl+F) + corrections Toggle/Callout/Outline
Some checks failed
CI / Lint, Unit Tests & Build (push) Failing after 1m31s
CI / Deploy production (on server) (push) Has been skipped

- Find & Replace : barre flottante Ctrl+F, recherche instantanée synchrone
  - Highlights ProseMirror (jaune = match, orange = actif)
  - Scroll vers le match sans voler le focus de l'input
  - Options: sensible à la casse, regex
  - Remplacer / Tout remplacer
  - i18n FR/EN complet
- Toggle/Callout/Outline: corrections bugs + design
This commit is contained in:
Antigravity
2026-06-14 17:19:51 +00:00
parent 2723e06b80
commit 5246ed41e9
5 changed files with 348 additions and 0 deletions

View File

@@ -0,0 +1,292 @@
'use client'
import { useState, useCallback, useEffect, useRef } from 'react'
import { createPortal } from 'react-dom'
import { X, ChevronDown, ChevronUp, Replace, CaseSensitive, Regex } from 'lucide-react'
import { useLanguage } from '@/lib/i18n'
import { cn } from '@/lib/utils'
import { Extension } from '@tiptap/core'
import { Editor } from '@tiptap/core'
import { Decoration, DecorationSet } from '@tiptap/pm/view'
import { Plugin, PluginKey, TextSelection } from '@tiptap/pm/state'
interface Match {
from: number
to: number
}
const pluginKey = new PluginKey<DecorationState>('findReplaceDecorations')
interface DecorationState {
decorations: DecorationSet
currentIndex: number
}
function findMatches(doc: any, query: string, caseSensitive: boolean, useRegex: boolean): Match[] {
if (!query) return []
let pattern: RegExp
try {
if (useRegex) {
pattern = new RegExp(query, caseSensitive ? 'g' : 'gi')
} else {
pattern = new RegExp(query.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'), caseSensitive ? 'g' : 'gi')
}
} catch { return [] }
const results: Match[] = []
doc.descendants((node: any, pos: number) => {
if (!node.isText || !node.text) return
pattern.lastIndex = 0
let m: RegExpExecArray | null
while ((m = pattern.exec(node.text)) !== null) {
if (m[0].length === 0) { pattern.lastIndex++; continue }
results.push({ from: pos + m.index, to: pos + m.index + m[0].length })
}
})
return results
}
export const FindReplaceExtension = Extension.create({
name: 'findReplace',
addProseMirrorPlugins() {
return [
new Plugin({
key: pluginKey,
state: {
init(): DecorationState {
return { decorations: DecorationSet.empty, currentIndex: -1 }
},
apply(tr, old: DecorationState): DecorationState {
const meta = tr.getMeta(pluginKey)
if (meta) return meta
if (tr.docChanged) {
return { decorations: DecorationSet.empty, currentIndex: -1 }
}
return old || { decorations: DecorationSet.empty, currentIndex: -1 }
},
},
props: {
decorations(state) {
return pluginKey.getState(state)?.decorations || null
},
},
}),
]
},
})
export function FindReplaceBar({ editor, onClose }: { editor: Editor; onClose: () => void }) {
const { t } = useLanguage()
const [searchText, setSearchText] = useState('')
const [replaceText, setReplaceText] = useState('')
const [caseSensitive, setCaseSensitive] = useState(false)
const [useRegex, setUseRegex] = useState(false)
const [count, setCount] = useState(0)
const [currentIndex, setCurrentIndex] = useState(-1)
const [showReplace, setShowReplace] = useState(false)
const inputRef = useRef<HTMLInputElement>(null)
const matchesRef = useRef<Match[]>([])
const indexRef = useRef(-1)
const applyDecorations = useCallback((matches: Match[], activeIdx: number) => {
const decos = matches.map((m, i) =>
Decoration.inline(m.from, m.to, {
class: i === activeIdx ? 'fr-active' : 'fr-match',
})
)
const set = DecorationSet.create(editor.state.doc, decos)
const tr = editor.state.tr.setMeta(pluginKey, { decorations: set, currentIndex: activeIdx })
tr.setMeta('addToHistory', false)
editor.view.dispatch(tr)
}, [editor])
const scrollToMatch = useCallback((m: Match) => {
try {
const dom = editor.view.domAtPos(m.from)
const el = dom.node instanceof HTMLElement ? dom.node : (dom.node as any).parentElement
el?.scrollIntoView({ behavior: 'smooth', block: 'center' })
} catch {}
}, [editor])
const runSearch = useCallback((text: string) => {
if (!text.trim()) {
matchesRef.current = []
setCount(0)
setCurrentIndex(-1)
indexRef.current = -1
applyDecorations([], -1)
return
}
const found = findMatches(editor.state.doc, text, caseSensitive, useRegex)
matchesRef.current = found
setCount(found.length)
const idx = found.length > 0 ? 0 : -1
setCurrentIndex(idx)
indexRef.current = idx
applyDecorations(found, idx)
if (idx >= 0) scrollToMatch(found[idx])
}, [caseSensitive, useRegex, editor, applyDecorations, scrollToMatch])
useEffect(() => { inputRef.current?.focus() }, [])
const next = useCallback(() => {
const ms = matchesRef.current
if (ms.length === 0) return
const ni = (indexRef.current + 1) % ms.length
indexRef.current = ni
setCurrentIndex(ni)
applyDecorations(ms, ni)
scrollToMatch(ms[ni])
}, [applyDecorations, scrollToMatch])
const prev = useCallback(() => {
const ms = matchesRef.current
if (ms.length === 0) return
const ni = (indexRef.current - 1 + ms.length) % ms.length
indexRef.current = ni
setCurrentIndex(ni)
applyDecorations(ms, ni)
scrollToMatch(ms[ni])
}, [applyDecorations, scrollToMatch])
const replaceOne = useCallback(() => {
const ms = matchesRef.current
const idx = indexRef.current
if (idx < 0 || ms.length === 0) return
const m = ms[idx]
editor.chain().focus().insertContentAt({ from: m.from, to: m.to }, replaceText).run()
setTimeout(() => {
const found = findMatches(editor.state.doc, searchText, caseSensitive, useRegex)
matchesRef.current = found
setCount(found.length)
const ni = found.length > 0 ? Math.min(idx, found.length - 1) : -1
indexRef.current = ni
setCurrentIndex(ni)
applyDecorations(found, ni)
if (ni >= 0) scrollToMatch(found[ni])
}, 50)
}, [editor, replaceText, searchText, caseSensitive, useRegex, applyDecorations, scrollToMatch])
const replaceAll = useCallback(() => {
const ms = matchesRef.current
if (ms.length === 0) return
const sorted = [...ms].sort((a, b) => b.from - a.from)
editor.chain().focus()
sorted.forEach(m => editor.chain().insertContentAt({ from: m.from, to: m.to }, replaceText).run())
matchesRef.current = []
setCount(0)
setCurrentIndex(-1)
indexRef.current = -1
applyDecorations([], -1)
}, [editor, replaceText, applyDecorations])
return createPortal(
<div
className="fixed top-4 right-4 z-[9999] w-80 rounded-xl border border-border bg-popover shadow-xl"
dir="auto"
onMouseDown={(e) => e.stopPropagation()}
>
<div className="flex items-center justify-between px-3 py-2 border-b border-border/50">
<span className="text-sm font-medium">{t('richTextEditor.findReplaceTitle')}</span>
<button onClick={onClose} className="p-0.5 rounded hover:bg-muted text-muted-foreground">
<X className="h-4 w-4" />
</button>
</div>
<div className="p-3 space-y-2">
<div className="flex items-center gap-1">
<input
ref={inputRef}
type="text"
value={searchText}
onChange={(e) => {
setSearchText(e.target.value)
runSearch(e.target.value)
}}
onKeyDown={(e) => {
e.stopPropagation()
if (e.key === 'Enter') { e.preventDefault(); e.shiftKey ? prev() : next() }
if (e.key === 'Escape') onClose()
}}
placeholder={t('richTextEditor.findPlaceholder')}
className="flex-1 min-w-0 rounded-md border border-border bg-background px-2.5 py-1.5 text-sm outline-none focus:ring-2 focus:ring-primary/30"
/>
<span className="text-xs text-muted-foreground tabular-nums whitespace-nowrap px-1">
{count > 0 ? `${currentIndex + 1}/${count}` : `0/0`}
</span>
</div>
<div className="flex items-center gap-0.5">
<button
onClick={() => setCaseSensitive(v => !v)}
className={cn('p-1.5 rounded hover:bg-muted text-muted-foreground transition-colors', caseSensitive && 'bg-primary/10 text-primary')}
title={t('richTextEditor.findCaseSensitive')}
>
<CaseSensitive className="h-4 w-4" />
</button>
<button
onClick={() => setUseRegex(v => !v)}
className={cn('p-1.5 rounded hover:bg-muted text-muted-foreground transition-colors', useRegex && 'bg-primary/10 text-primary')}
title={t('richTextEditor.findRegex')}
>
<Regex className="h-4 w-4" />
</button>
<button
onClick={() => setShowReplace(v => !v)}
className={cn('p-1.5 rounded hover:bg-muted text-muted-foreground transition-colors', showReplace && 'bg-muted text-foreground')}
title={t('richTextEditor.findReplaceToggle')}
>
<Replace className="h-4 w-4" />
</button>
<div className="flex-1" />
<button
onClick={prev}
disabled={count === 0}
className="p-1.5 rounded hover:bg-muted text-muted-foreground disabled:opacity-40 disabled:cursor-not-allowed"
title={t('richTextEditor.findPrev')}
>
<ChevronUp className="h-4 w-4" />
</button>
<button
onClick={next}
disabled={count === 0}
className="p-1.5 rounded hover:bg-muted text-muted-foreground disabled:opacity-40 disabled:cursor-not-allowed"
title={t('richTextEditor.findNext')}
>
<ChevronDown className="h-4 w-4" />
</button>
</div>
{showReplace && (
<div className="flex items-center gap-1 pt-1 border-t border-border/30">
<input
type="text"
value={replaceText}
onChange={(e) => setReplaceText(e.target.value)}
onKeyDown={(e) => e.stopPropagation()}
placeholder={t('richTextEditor.replacePlaceholder')}
className="flex-1 min-w-0 rounded-md border border-border bg-background px-2.5 py-1.5 text-sm outline-none focus:ring-2 focus:ring-primary/30"
/>
<button
onClick={replaceOne}
disabled={currentIndex < 0}
className="px-2 py-1 text-xs rounded-md hover:bg-muted text-muted-foreground disabled:opacity-40 disabled:cursor-not-allowed whitespace-nowrap"
>
{t('richTextEditor.replace')}
</button>
<button
onClick={replaceAll}
disabled={count === 0}
className="px-2 py-1 text-xs rounded-md bg-primary/10 text-primary hover:bg-primary/20 disabled:opacity-40 disabled:cursor-not-allowed whitespace-nowrap font-medium"
>
{t('richTextEditor.replaceAll')}
</button>
</div>
)}
</div>
</div>,
document.body
)
}

View File

@@ -29,6 +29,7 @@ import { StructuredViewBlockExtension, insertStructuredViewBlockAtSelection } fr
import { ToggleExtension, insertToggleBlock } from './tiptap-toggle-extension'
import { CalloutExtension, insertCalloutBlock } from './tiptap-callout-extension'
import { OutlineExtension, insertOutlineBlock } from './tiptap-outline-extension'
import { FindReplaceBar, FindReplaceExtension } from './editor-find-replace-bar'
import { RtlPreserveExtension } from './tiptap-rtl-preserve-extension'
import { ClipArticleExtension } from './tiptap-clip-article-extension'
import { BlockPicker, type BlockSuggestion } from './block-picker'
@@ -306,6 +307,20 @@ export const RichTextEditor = forwardRef<RichTextEditorHandle, RichTextEditorPro
} | null>(null)
const [isMobile, setIsMobile] = useState(false)
const [actionSheetOpen, setActionSheetOpen] = useState(false)
const [showFindReplace, setShowFindReplace] = useState(false)
useEffect(() => {
const handleFindShortcut = (e: KeyboardEvent) => {
if ((e.metaKey || e.ctrlKey) && e.key === 'f' && !e.shiftKey) {
const target = e.target as HTMLElement
if (target.tagName === 'INPUT' || target.tagName === 'TEXTAREA') return
e.preventDefault()
setShowFindReplace(v => !v)
}
}
document.addEventListener('keydown', handleFindShortcut)
return () => document.removeEventListener('keydown', handleFindShortcut)
}, [])
const [noteLinkPickerOpen, setNoteLinkPickerOpen] = useState(false)
const [noteLinkQuery, setNoteLinkQuery] = useState('')
const noteLinkRangeRef = useRef<{ from: number; to: number } | null>(null)
@@ -442,6 +457,7 @@ export const RichTextEditor = forwardRef<RichTextEditorHandle, RichTextEditorPro
ToggleExtension,
CalloutExtension,
OutlineExtension,
FindReplaceExtension,
ClipArticleExtension,
RtlPreserveExtension,
Placeholder.configure({
@@ -1098,6 +1114,10 @@ export const RichTextEditor = forwardRef<RichTextEditorHandle, RichTextEditorPro
<EditorContent editor={editor} />
{editor && showFindReplace && (
<FindReplaceBar editor={editor} onClose={() => setShowFindReplace(false)} />
)}
{editor && blockMenuState && (
<BlockActionMenu
editor={editor}