"use client"; import { FindReplacePlugin } from "@platejs/find-replace"; import { useAtomValue, useSetAtom } from "jotai"; import { Check, Copy, Download, FileQuestionMark, FileText, Pencil, RefreshCw, XIcon, } from "lucide-react"; import dynamic from "next/dynamic"; import { useCallback, useEffect, useRef, useState } from "react"; import { toast } from "sonner"; import { pendingChunkHighlightAtom } from "@/atoms/document-viewer/pending-chunk-highlight.atom"; import { closeEditorPanelAtom, editorPanelAtom } from "@/atoms/editor/editor-panel.atom"; import { VersionHistoryButton } from "@/components/documents/version-history"; import type { PlateEditorInstance } from "@/components/editor/plate-editor"; import { SourceCodeEditor } from "@/components/editor/source-code-editor"; import { MarkdownViewer } from "@/components/markdown-viewer"; import { Alert, AlertDescription } from "@/components/ui/alert"; import { Button } from "@/components/ui/button"; import { Drawer, DrawerContent, DrawerHandle, DrawerTitle } from "@/components/ui/drawer"; import { CITATION_HIGHLIGHT_CLASS } from "@/components/ui/search-highlight-node"; import { Spinner } from "@/components/ui/spinner"; import { useMediaQuery } from "@/hooks/use-media-query"; import { useElectronAPI } from "@/hooks/use-platform"; import { authenticatedFetch, getBearerToken, redirectToLogin } from "@/lib/auth-utils"; import { buildCitationSearchCandidates } from "@/lib/citation-search"; import { inferMonacoLanguageFromPath } from "@/lib/editor-language"; const PlateEditor = dynamic( () => import("@/components/editor/plate-editor").then((m) => ({ default: m.PlateEditor })), { ssr: false, loading: () => } ); type CitationHighlightStatus = "exact" | "miss"; const LARGE_DOCUMENT_THRESHOLD = 2 * 1024 * 1024; // 2MB const CITATION_MAX_LENGTH = 16 * 1024 * 1024; // 16MB on-demand cap for citation jumps interface EditorContent { document_id: number; title: string; document_type?: string; source_markdown: string; content_size_bytes?: number; chunk_count?: number; truncated?: boolean; } const EDITABLE_DOCUMENT_TYPES = new Set(["FILE", "NOTE"]); type EditorRenderMode = "rich_markdown" | "source_code"; type AgentFilesystemMount = { mount: string; rootPath: string; }; function normalizeLocalVirtualPathForEditor( candidatePath: string, mounts: AgentFilesystemMount[] ): string { const normalizedCandidate = candidatePath.trim().replace(/\\/g, "/").replace(/\/+/g, "/"); if (!normalizedCandidate) return candidatePath; const defaultMount = mounts[0]?.mount; if (!defaultMount) { return normalizedCandidate.startsWith("/") ? normalizedCandidate : `/${normalizedCandidate.replace(/^\/+/, "")}`; } const mountNames = new Set(mounts.map((entry) => entry.mount)); if (normalizedCandidate.startsWith("/")) { const relative = normalizedCandidate.replace(/^\/+/, ""); const [firstSegment] = relative.split("/", 1); if (mountNames.has(firstSegment)) { return `/${relative}`; } return `/${defaultMount}/${relative}`; } const relative = normalizedCandidate.replace(/^\/+/, ""); const [firstSegment] = relative.split("/", 1); if (mountNames.has(firstSegment)) { return `/${relative}`; } return `/${defaultMount}/${relative}`; } function EditorPanelSkeleton() { return (
); } export function EditorPanelContent({ kind = "document", documentId, localFilePath, searchSpaceId, title, onClose, }: { kind?: "document" | "local_file"; documentId?: number; localFilePath?: string; searchSpaceId?: number; title: string | null; onClose?: () => void; }) { const electronAPI = useElectronAPI(); const [editorDoc, setEditorDoc] = useState(null); const [isLoading, setIsLoading] = useState(true); const [error, setError] = useState(null); const [saving, setSaving] = useState(false); const [downloading, setDownloading] = useState(false); const [isEditing, setIsEditing] = useState(false); const [editedMarkdown, setEditedMarkdown] = useState(null); const [localFileContent, setLocalFileContent] = useState(""); const [hasCopied, setHasCopied] = useState(false); const markdownRef = useRef(""); const copyResetTimeoutRef = useRef | null>(null); const initialLoadDone = useRef(false); const changeCountRef = useRef(0); const [displayTitle, setDisplayTitle] = useState(title || "Untitled"); const isLocalFileMode = kind === "local_file"; const editorRenderMode: EditorRenderMode = isLocalFileMode ? "source_code" : "rich_markdown"; // --- Citation-jump highlight wiring ---------------------------------- // `EditorPanelContent` is the consumer of `pendingChunkHighlightAtom`: when // a citation badge is clicked, the badge stages `{documentId, chunkId, // chunkText}` and opens this panel. We drive Plate's `FindReplacePlugin` // (registered in every preset) to highlight the cited text natively via // Slate decorations — no DOM walking, no Range gymnastics. The state // machine below escalates the document fetch from 2MB → 16MB once if no // candidate snippet matched in the preview, and surfaces miss outcomes // via an inline alert. const pending = useAtomValue(pendingChunkHighlightAtom); const setPendingHighlight = useSetAtom(pendingChunkHighlightAtom); const [fetchKey, setFetchKey] = useState(0); const [maxLengthOverride, setMaxLengthOverride] = useState(null); const [highlightResult, setHighlightResult] = useState(null); const editorRef = useRef(null); const escalatedForRef = useRef(null); const lastAppliedChunkIdRef = useRef(null); // Tracks whether a citation highlight is currently decorated in the // editor. We use a ref (not state) because the click-to-dismiss handler // runs in a stable callback that would otherwise close over stale state. const isHighlightActiveRef = useRef(false); // Once a citation jump targets this doc we have to keep `PlateEditor` // mounted for the *rest of the doc session* — even after the highlight // effect clears `pendingChunkHighlightAtom` (which it does as soon as // the decoration is applied, so a follow-up citation on the same chunk // can re-trigger). Without this latch, non-editable docs would re-render // back into `MarkdownViewer` the instant `pending` is released, tearing // down the Plate decorations and dropping the highlight after a frame. const [stickyPlateMode, setStickyPlateMode] = useState(false); const clearCitationSearch = useCallback(() => { isHighlightActiveRef.current = false; const editor = editorRef.current; if (!editor) return; try { editor.setOption(FindReplacePlugin, "search", ""); editor.api.redecorate(); } catch (err) { console.warn("[EditorPanelContent] clearCitationSearch failed:", err); } }, []); // Dismiss the highlight when the user interacts with the editor surface. // `onPointerDown` fires before focus / selection changes so the click // itself feels responsive — the highlight clears in the same event tick // that places the cursor. No-op when nothing is highlighted, so we don't // thrash `redecorate` on every click in normal editing. const handleEditorPointerDown = useCallback(() => { if (!isHighlightActiveRef.current) return; clearCitationSearch(); setHighlightResult(null); }, [clearCitationSearch]); const isCitationTarget = !!pending && !isLocalFileMode && pending.documentId === documentId; const resolveLocalVirtualPath = useCallback( async (candidatePath: string): Promise => { if (!electronAPI?.getAgentFilesystemMounts) { return candidatePath; } try { const mounts = (await electronAPI.getAgentFilesystemMounts( searchSpaceId )) as AgentFilesystemMount[]; return normalizeLocalVirtualPathForEditor(candidatePath, mounts); } catch { return candidatePath; } }, [electronAPI, searchSpaceId] ); const isLargeDocument = (editorDoc?.content_size_bytes ?? 0) > LARGE_DOCUMENT_THRESHOLD; // `fetchKey` is an explicit re-fetch trigger (escalation bumps it to force // a new request even when documentId/searchSpaceId haven't changed). useEffect(() => { const controller = new AbortController(); setIsLoading(true); setError(null); setEditorDoc(null); setEditedMarkdown(null); setLocalFileContent(""); setHasCopied(false); setIsEditing(false); initialLoadDone.current = false; changeCountRef.current = 0; // Clear any in-flight FindReplacePlugin search before the editor // re-mounts on new content (a fresh editor key is generated below // from documentId + isEditing, so the previous editor + its // decorations are about to be discarded anyway, but we belt-and- // brace here for the case where only `fetchKey` changed). clearCitationSearch(); const doFetch = async () => { try { if (isLocalFileMode) { if (!localFilePath) { throw new Error("Missing local file path"); } if (!electronAPI?.readAgentLocalFileText) { throw new Error("Local file editor is available only in desktop mode."); } const resolvedLocalPath = await resolveLocalVirtualPath(localFilePath); const readResult = await electronAPI.readAgentLocalFileText( resolvedLocalPath, searchSpaceId ); if (!readResult.ok) { throw new Error(readResult.error || "Failed to read local file"); } const inferredTitle = resolvedLocalPath.split("/").pop() || resolvedLocalPath; const content: EditorContent = { document_id: -1, title: inferredTitle, document_type: "NOTE", source_markdown: readResult.content, }; markdownRef.current = content.source_markdown; setLocalFileContent(content.source_markdown); setDisplayTitle(title || inferredTitle); setEditorDoc(content); initialLoadDone.current = true; return; } if (!documentId || !searchSpaceId) { throw new Error("Missing document context"); } const token = getBearerToken(); if (!token) { redirectToLogin(); return; } const url = new URL( `${process.env.NEXT_PUBLIC_FASTAPI_BACKEND_URL}/api/v1/search-spaces/${searchSpaceId}/documents/${documentId}/editor-content` ); url.searchParams.set("max_length", String(maxLengthOverride ?? LARGE_DOCUMENT_THRESHOLD)); // `fetchKey` participates here so biome's noUnusedVariables sees it // as consumed; bumping it forces a fresh request even when the URL // is otherwise identical. if (fetchKey > 0) url.searchParams.set("_n", String(fetchKey)); const response = await authenticatedFetch(url.toString(), { method: "GET" }); if (controller.signal.aborted) return; if (!response.ok) { const errorData = await response .json() .catch(() => ({ detail: "Failed to fetch document" })); throw new Error(errorData.detail || "Failed to fetch document"); } const data = await response.json(); if (data.source_markdown === undefined || data.source_markdown === null) { setError( "This document does not have editable content. Please re-upload to enable editing." ); setIsLoading(false); return; } markdownRef.current = data.source_markdown; setDisplayTitle(data.title || title || "Untitled"); setEditorDoc(data); initialLoadDone.current = true; } catch (err) { if (controller.signal.aborted) return; console.error("Error fetching document:", err); setError(err instanceof Error ? err.message : "Failed to fetch document"); } finally { if (!controller.signal.aborted) setIsLoading(false); } }; doFetch().catch(() => {}); return () => controller.abort(); }, [ documentId, electronAPI, isLocalFileMode, localFilePath, resolveLocalVirtualPath, searchSpaceId, title, fetchKey, maxLengthOverride, clearCitationSearch, ]); // Reset citation-jump bookkeeping whenever the panel switches to a different // document (or local file). Body only writes setters — the deps are the // real triggers we want to react to. // biome-ignore lint/correctness/useExhaustiveDependencies: documentId/localFilePath are intentional triggers. useEffect(() => { clearCitationSearch(); escalatedForRef.current = null; lastAppliedChunkIdRef.current = null; setHighlightResult(null); setMaxLengthOverride(null); setFetchKey(0); // Drop sticky Plate mode when the panel moves to a different doc // — the next doc starts in its preferred render mode (Plate for // editable, MarkdownViewer for everything else) until/unless a // citation jump targets it. setStickyPlateMode(false); }, [documentId, localFilePath, clearCitationSearch]); // Latch sticky Plate mode the first time a citation jump targets this // doc. We keep it sticky for the remainder of this doc session so the // highlight effect's `setPendingHighlight(null)` doesn't unmount the // editor mid-flight (see comment on `stickyPlateMode` declaration). useEffect(() => { if (isCitationTarget) setStickyPlateMode(true); }, [isCitationTarget]); // `isEditorReady` is what `useEffect` actually depends on — `editorRef` // is a ref so changes don't trigger re-runs. We flip this to `true` once // `PlateEditor` calls back with its live editor instance (its // `usePlateEditor` value-init runs synchronously, so by the time this // flips true the markdown is already deserialized into the Slate tree). const [isEditorReady, setIsEditorReady] = useState(false); const handleEditorReady = useCallback((editor: PlateEditorInstance | null) => { console.log("[citation:editor] handleEditorReady", { ready: !!editor }); editorRef.current = editor; setIsEditorReady(!!editor); }, []); // --- Citation jump highlight effect ----------------------------------- // Drives Plate's FindReplacePlugin to highlight the cited chunk: // 1. Build candidate snippets from the chunk text (first sentence, // first 8 words, full chunk if short). Plate's decorate runs per- // block and won't cross block boundaries, so the shorter // candidates exist to give us something that fits in one // paragraph / heading. // 2. For each candidate: setOption('search', ...) → redecorate → // wait two animation frames for React to flush → query the editor // DOM for `.${CITATION_HIGHLIGHT_CLASS}`. First hit wins. // // Why a className and not a `data-*` attribute? Plate's // `PlateLeaf` runs its props through `useNodeAttributes`, which // only forwards `attributes`, `className`, `ref`, and `style` — // arbitrary `data-*` attributes are silently dropped. `className` // is the only escape hatch guaranteed to survive into the DOM. // 3. On hit: smooth-scroll the first match into view, mark the // highlight active (so a click inside the editor can dismiss it), // release the pending atom. // 4. On terminal miss: if the doc was truncated and we haven't // escalated yet, bump the fetch's `max_length` to the citation // cap and re-fetch — the post-refetch render will re-run this // effect against the larger preview. Otherwise, release the // atom and show the miss alert. useEffect(() => { console.log("[citation:effect] fired", { isCitationTarget, pendingDocId: pending?.documentId, pendingChunkId: pending?.chunkId, pendingChunkTextLen: pending?.chunkText?.length, documentId, isLocalFileMode, isEditing, hasMarkdown: !!editorDoc?.source_markdown, markdownLen: editorDoc?.source_markdown?.length, truncated: editorDoc?.truncated, isEditorReady, editorRefSet: !!editorRef.current, maxLengthOverride, }); if (!isCitationTarget || !pending) { console.log("[citation:effect] guard ✗ no citation target / no pending"); return; } if (isLocalFileMode || isEditing) { console.log("[citation:effect] guard ✗ localFileMode/editing"); return; } if (!editorDoc?.source_markdown) { console.log("[citation:effect] guard ✗ source_markdown not ready"); return; } if (!isEditorReady) { console.log("[citation:effect] guard ✗ editor not ready yet"); return; } const editor = editorRef.current; if (!editor) { console.log("[citation:effect] guard ✗ editorRef.current is null"); return; } if (lastAppliedChunkIdRef.current !== pending.chunkId) { lastAppliedChunkIdRef.current = pending.chunkId; } let cancelled = false; const finishMiss = () => { console.log("[citation:effect] terminal miss — no candidate matched"); try { editor.setOption(FindReplacePlugin, "search", ""); editor.api.redecorate(); } catch (err) { console.warn("[EditorPanelContent] reset search after miss failed:", err); } const canEscalate = editorDoc.truncated === true && (maxLengthOverride ?? LARGE_DOCUMENT_THRESHOLD) < CITATION_MAX_LENGTH && escalatedForRef.current !== pending.chunkId; console.log("[citation:effect] miss decision", { truncated: editorDoc.truncated, currentMaxLength: maxLengthOverride ?? LARGE_DOCUMENT_THRESHOLD, canEscalate, }); if (canEscalate) { escalatedForRef.current = pending.chunkId; setMaxLengthOverride(CITATION_MAX_LENGTH); setFetchKey((k) => k + 1); // Keep the atom set so the post-refetch render re-runs. return; } setHighlightResult("miss"); setPendingHighlight(null); }; const tryCandidates = async () => { const candidates = buildCitationSearchCandidates(pending.chunkText); console.log("[citation:effect] candidates built", { count: candidates.length, previews: candidates.map((c) => c.slice(0, 60)), }); if (candidates.length === 0) { if (!cancelled) finishMiss(); return; } // Resolve the editor's rendered DOM root via Slate's stable // `[data-slate-editor="true"]` attribute (set by slate-react's // ``). Scoping queries to this root prevents // `` elements rendered elsewhere on the page (e.g. chat // search-highlight leaves in another mounted PlateEditor) from // being mistaken for citation hits. const editorRoot = document.querySelector('[data-slate-editor="true"]'); console.log("[citation:effect] editor root", { hasRoot: !!editorRoot, }); const root: ParentNode = editorRoot ?? document; for (let i = 0; i < candidates.length; i++) { const candidate = candidates[i]; if (cancelled) return; try { editor.setOption(FindReplacePlugin, "search", candidate); editor.api.redecorate(); console.log(`[citation:effect] try #${i} setOption + redecorate`, { len: candidate.length, preview: candidate.slice(0, 80), }); } catch (err) { console.warn("[EditorPanelContent] setOption/redecorate failed:", err); continue; } // Two rAFs: first lets Slate flush its onChange, second lets // React commit the decoration leaves into the DOM. await new Promise((resolve) => requestAnimationFrame(() => requestAnimationFrame(() => resolve())) ); if (cancelled) return; // Primary probe: by our stable class on the rendered . let el = root.querySelector(`.${CITATION_HIGHLIGHT_CLASS}`); const classMarkCount = root.querySelectorAll(`.${CITATION_HIGHLIGHT_CLASS}`).length; // Diagnostic fallback: any inside the editor root. // If we ever see allMarks > 0 but classMarkCount === 0, // the className was stripped again and we need to revisit // `useNodeAttributes` filtering. const allMarkCount = root.querySelectorAll("mark").length; if (!el && allMarkCount > 0) { el = root.querySelector("mark"); } console.log(`[citation:effect] try #${i} DOM probe`, { foundEl: !!el, classMarkCount, allMarkCount, usedFallback: !!el && classMarkCount === 0, }); if (el) { try { el.scrollIntoView({ block: "center", behavior: "smooth" }); } catch { el.scrollIntoView(); } isHighlightActiveRef.current = true; setHighlightResult("exact"); console.log(`[citation:effect] ✓ exact via candidate #${i} — atom released`); // No auto-clear timer — the highlight is intentionally // permanent until the user clicks inside the editor (see // `handleEditorPointerDown`) or another dismissal trigger // fires (doc switch, edit-mode toggle, panel unmount, // next citation jump). Sticky Plate mode keeps the // editor mounted after the atom clears. setPendingHighlight(null); return; } } if (!cancelled) finishMiss(); }; void tryCandidates(); return () => { cancelled = true; }; }, [ isCitationTarget, pending, documentId, editorDoc?.source_markdown, editorDoc?.truncated, isLocalFileMode, isEditing, isEditorReady, maxLengthOverride, clearCitationSearch, setPendingHighlight, ]); // Cleanup any active highlight on unmount. useEffect(() => { return () => clearCitationSearch(); }, [clearCitationSearch]); // Toggling into edit mode swaps Plate out of readOnly. Clear the citation // search so stale leaves don't linger in the editing surface. useEffect(() => { if (isEditing) { clearCitationSearch(); setHighlightResult(null); } }, [isEditing, clearCitationSearch]); useEffect(() => { return () => { if (copyResetTimeoutRef.current) { clearTimeout(copyResetTimeoutRef.current); } }; }, []); const handleMarkdownChange = useCallback((md: string) => { markdownRef.current = md; if (!initialLoadDone.current) return; changeCountRef.current += 1; if (changeCountRef.current <= 1) return; setEditedMarkdown(md); }, []); const handleCopy = useCallback(async () => { try { const textToCopy = markdownRef.current ?? editorDoc?.source_markdown ?? ""; await navigator.clipboard.writeText(textToCopy); setHasCopied(true); if (copyResetTimeoutRef.current) { clearTimeout(copyResetTimeoutRef.current); } copyResetTimeoutRef.current = setTimeout(() => { setHasCopied(false); }, 1400); } catch (err) { console.error("Error copying content:", err); } }, [editorDoc?.source_markdown]); const handleSave = useCallback( async (_options?: { silent?: boolean }) => { setSaving(true); try { if (isLocalFileMode) { if (!localFilePath) { throw new Error("Missing local file path"); } if (!electronAPI?.writeAgentLocalFileText) { throw new Error("Local file editor is available only in desktop mode."); } const resolvedLocalPath = await resolveLocalVirtualPath(localFilePath); const contentToSave = markdownRef.current; const writeResult = await electronAPI.writeAgentLocalFileText( resolvedLocalPath, contentToSave, searchSpaceId ); if (!writeResult.ok) { throw new Error(writeResult.error || "Failed to save local file"); } setEditorDoc((prev) => (prev ? { ...prev, source_markdown: contentToSave } : prev)); setEditedMarkdown(markdownRef.current === contentToSave ? null : markdownRef.current); return true; } if (!searchSpaceId || !documentId) { throw new Error("Missing document context"); } const token = getBearerToken(); if (!token) { toast.error("Please login to save"); redirectToLogin(); return; } const response = await authenticatedFetch( `${process.env.NEXT_PUBLIC_FASTAPI_BACKEND_URL}/api/v1/search-spaces/${searchSpaceId}/documents/${documentId}/save`, { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ source_markdown: markdownRef.current }), } ); if (!response.ok) { const errorData = await response .json() .catch(() => ({ detail: "Failed to save document" })); throw new Error(errorData.detail || "Failed to save document"); } setEditorDoc((prev) => (prev ? { ...prev, source_markdown: markdownRef.current } : prev)); setEditedMarkdown(null); toast.success("Document saved! Reindexing in background..."); return true; } catch (err) { console.error("Error saving document:", err); toast.error(err instanceof Error ? err.message : "Failed to save document"); return false; } finally { setSaving(false); } }, [ documentId, electronAPI, isLocalFileMode, localFilePath, resolveLocalVirtualPath, searchSpaceId, ] ); const isEditableType = editorDoc ? (editorRenderMode === "source_code" || EDITABLE_DOCUMENT_TYPES.has(editorDoc.document_type ?? "")) && !isLargeDocument : false; // Use PlateEditor for any of: // - Editable doc types (FILE/NOTE) — existing editing UX. // - Active citation jump in flight (`isCitationTarget`) — covers the // mount in the very first render where the atom is set but the // sticky effect hasn't fired yet. // - Sticky Plate mode latched on a previous citation jump — keeps // the editor mounted (with its decorations) after the highlight // effect clears the atom. Resets when the doc changes. const renderInPlateEditor = isEditableType || isCitationTarget || stickyPlateMode; const hasUnsavedChanges = editedMarkdown !== null; const showDesktopHeader = !!onClose; const showEditingActions = isEditableType && isEditing; const localFileLanguage = inferMonacoLanguageFromPath(localFilePath); const handleCancelEditing = useCallback(() => { const savedContent = editorDoc?.source_markdown ?? ""; markdownRef.current = savedContent; setLocalFileContent(savedContent); setEditedMarkdown(null); changeCountRef.current = 0; setIsEditing(false); }, [editorDoc?.source_markdown]); const handleDownloadMarkdown = useCallback(async () => { if (!searchSpaceId || !documentId) return; setDownloading(true); try { const response = await authenticatedFetch( `${process.env.NEXT_PUBLIC_FASTAPI_BACKEND_URL}/api/v1/search-spaces/${searchSpaceId}/documents/${documentId}/download-markdown`, { method: "GET" } ); if (!response.ok) throw new Error("Download failed"); const blob = await response.blob(); const url = URL.createObjectURL(blob); const a = document.createElement("a"); a.href = url; const disposition = response.headers.get("content-disposition"); const match = disposition?.match(/filename="(.+)"/); a.download = match?.[1] ?? `${editorDoc?.title || "document"}.md`; document.body.appendChild(a); a.click(); a.remove(); URL.revokeObjectURL(url); toast.success("Download started"); } catch { toast.error("Failed to download document"); } finally { setDownloading(false); } }, [documentId, editorDoc?.title, searchSpaceId]); // We no longer surface an "approximate" status — Plate's FindReplacePlugin // either decorates an exact match or it doesn't, and the candidate snippet // strategy (first sentence → first 8 words → full chunk) means we either // land on the citation start or fall through to the miss alert. const showMissAlert = isCitationTarget && highlightResult === "miss"; const citationAlerts = showMissAlert && ( Cited section couldn't be located in this view. {editorDoc?.truncated && ( )} ); const largeDocAlert = isLargeDocument && !isLocalFileMode && editorDoc && ( This document is too large for the editor ( {Math.round((editorDoc.content_size_bytes ?? 0) / 1024 / 1024)}MB,{" "} {editorDoc.chunk_count ?? 0} chunks). Showing a preview below. ); return ( <> {showDesktopHeader ? (

File

{displayTitle}

{showEditingActions ? ( <> ) : ( <> {!isLocalFileMode && editorDoc?.document_type && documentId && ( )} {isEditableType && ( )} )}
) : (

{displayTitle}

{showEditingActions ? ( <> ) : ( <> {!isLocalFileMode && editorDoc?.document_type && documentId && ( )} {isEditableType && ( )} )}
)}
{isLoading ? ( ) : error || !editorDoc ? (
{error?.toLowerCase().includes("still being processed") ? (
) : (
)}

{error?.toLowerCase().includes("still being processed") ? "Document is processing" : "Document unavailable"}

{error || "An unknown error occurred"}

) : editorRenderMode === "source_code" ? (
{ void handleSave({ silent: true }); }} readOnly={!isEditing} onChange={(next) => { markdownRef.current = next; setLocalFileContent(next); if (!initialLoadDone.current) return; setEditedMarkdown(next === (editorDoc?.source_markdown ?? "") ? null : next); }} />
) : isLargeDocument && !isLocalFileMode && !isCitationTarget ? ( // Large doc, no active citation — fast Streamdown preview // + download CTA. We only fall back to MarkdownViewer here // because Plate is heavy on multi-MB docs and the user // isn't waiting on a specific citation to render.
{largeDocAlert}
) : renderInPlateEditor ? ( // Editable doc (FILE/NOTE) OR active citation jump (any // doc type). The citation path uses Plate's // FindReplacePlugin for native, decoration-based // highlighting — see the citation-jump highlight effect // above for how `editorRef` and `handleEditorReady` are // wired.
{(citationAlerts || (isLargeDocument && isCitationTarget && !isLocalFileMode)) && (
{isLargeDocument && isCitationTarget && largeDocAlert} {citationAlerts}
)}
) : (
)}
); } function DesktopEditorPanel() { const panelState = useAtomValue(editorPanelAtom); const closePanel = useSetAtom(closeEditorPanelAtom); useEffect(() => { const handleKeyDown = (e: KeyboardEvent) => { if (e.key === "Escape") closePanel(); }; document.addEventListener("keydown", handleKeyDown); return () => document.removeEventListener("keydown", handleKeyDown); }, [closePanel]); const hasTarget = panelState.kind === "document" ? !!panelState.documentId && !!panelState.searchSpaceId : !!panelState.localFilePath; if (!panelState.isOpen || !hasTarget) return null; return (
); } function MobileEditorDrawer() { const panelState = useAtomValue(editorPanelAtom); const closePanel = useSetAtom(closeEditorPanelAtom); if (panelState.kind === "local_file") return null; const hasTarget = panelState.kind === "document" ? !!panelState.documentId && !!panelState.searchSpaceId : !!panelState.localFilePath; if (!hasTarget) return null; return ( { if (!open) closePanel(); }} shouldScaleBackground={false} > {panelState.title || "Editor"}
); } export function EditorPanel() { const panelState = useAtomValue(editorPanelAtom); const isDesktop = useMediaQuery("(min-width: 1024px)"); const hasTarget = panelState.kind === "document" ? !!panelState.documentId && !!panelState.searchSpaceId : !!panelState.localFilePath; if (!panelState.isOpen || !hasTarget) return null; if (!isDesktop && panelState.kind === "local_file") return null; if (isDesktop) { return ; } return ; } export function MobileEditorPanel() { const panelState = useAtomValue(editorPanelAtom); const isDesktop = useMediaQuery("(min-width: 1024px)"); const hasTarget = panelState.kind === "document" ? !!panelState.documentId && !!panelState.searchSpaceId : !!panelState.localFilePath; if (isDesktop || !panelState.isOpen || !hasTarget || panelState.kind === "local_file") return null; return ; }