"use client" import type React from "react" import { createContext, useContext, useRef, useState } from "react" import type { DrawIoEmbedRef } from "react-drawio" import { STORAGE_DIAGRAM_XML_KEY } from "@/components/chat-panel" import type { ExportFormat } from "@/components/save-dialog" import { extractDiagramXML, validateAndFixXml } from "../lib/utils" interface DiagramContextType { chartXML: string latestSvg: string diagramHistory: { svg: string; xml: string }[] loadDiagram: (chart: string, skipValidation?: boolean) => string | null handleExport: () => void handleExportWithoutHistory: () => void resolverRef: React.Ref<((value: string) => void) | null> drawioRef: React.Ref handleDiagramExport: (data: any) => void clearDiagram: () => void saveDiagramToFile: ( filename: string, format: ExportFormat, sessionId?: string, ) => void saveDiagramToStorage: () => Promise isDrawioReady: boolean onDrawioLoad: () => void resetDrawioReady: () => void } const DiagramContext = createContext(undefined) export function DiagramProvider({ children }: { children: React.ReactNode }) { const [chartXML, setChartXML] = useState("") const [latestSvg, setLatestSvg] = useState("") const [diagramHistory, setDiagramHistory] = useState< { svg: string; xml: string }[] >([]) const [isDrawioReady, setIsDrawioReady] = useState(false) const hasCalledOnLoadRef = useRef(false) const drawioRef = useRef(null) const resolverRef = useRef<((value: string) => void) | null>(null) // Track if we're expecting an export for history (user-initiated) const expectHistoryExportRef = useRef(false) const onDrawioLoad = () => { // Only set ready state once to prevent infinite loops if (hasCalledOnLoadRef.current) return hasCalledOnLoadRef.current = true // console.log("[DiagramContext] DrawIO loaded, setting ready state") setIsDrawioReady(true) } const resetDrawioReady = () => { // console.log("[DiagramContext] Resetting DrawIO ready state") hasCalledOnLoadRef.current = false setIsDrawioReady(false) } // Track if we're expecting an export for file save (stores raw export data) const saveResolverRef = useRef<{ resolver: ((data: string) => void) | null format: ExportFormat | null }>({ resolver: null, format: null }) const handleExport = () => { if (drawioRef.current) { // Mark that this export should be saved to history expectHistoryExportRef.current = true drawioRef.current.exportDiagram({ format: "xmlsvg", }) } } const handleExportWithoutHistory = () => { if (drawioRef.current) { // Export without saving to history (for edit_diagram fetching current state) drawioRef.current.exportDiagram({ format: "xmlsvg", }) } } // Save current diagram to localStorage (used before theme/UI changes) const saveDiagramToStorage = async (): Promise => { if (!drawioRef.current) return try { const currentXml = await Promise.race([ new Promise((resolve) => { resolverRef.current = resolve drawioRef.current?.exportDiagram({ format: "xmlsvg" }) }), new Promise((_, reject) => setTimeout(() => reject(new Error("Export timeout")), 2000), ), ]) // Only save if diagram has meaningful content (not empty template) if (currentXml && currentXml.length > 300) { localStorage.setItem(STORAGE_DIAGRAM_XML_KEY, currentXml) } } catch (error) { console.error("Failed to save diagram to storage:", error) } } const loadDiagram = ( chart: string, skipValidation?: boolean, ): string | null => { let xmlToLoad = chart // Validate XML structure before loading (unless skipped for internal use) if (!skipValidation) { const validation = validateAndFixXml(chart) if (!validation.valid) { console.warn( "[loadDiagram] Validation error:", validation.error, ) return validation.error } // Use fixed XML if auto-fix was applied if (validation.fixed) { console.log( "[loadDiagram] Auto-fixed XML issues:", validation.fixes, ) xmlToLoad = validation.fixed } } // Keep chartXML in sync even when diagrams are injected (e.g., display_diagram tool) setChartXML(xmlToLoad) if (drawioRef.current) { drawioRef.current.load({ xml: xmlToLoad, }) } return null } const handleDiagramExport = (data: any) => { // Handle save to file if requested (process raw data before extraction) if (saveResolverRef.current.resolver) { const format = saveResolverRef.current.format saveResolverRef.current.resolver(data.data) saveResolverRef.current = { resolver: null, format: null } // For non-xmlsvg formats, skip XML extraction as it will fail // Only drawio (which uses xmlsvg internally) has the content attribute if (format === "png" || format === "svg") { return } } const extractedXML = extractDiagramXML(data.data) setChartXML(extractedXML) setLatestSvg(data.data) // Only add to history if this was a user-initiated export // Limit to 20 entries to prevent memory leaks during long sessions const MAX_HISTORY_SIZE = 20 if (expectHistoryExportRef.current) { setDiagramHistory((prev) => { const newHistory = [ ...prev, { svg: data.data, xml: extractedXML, }, ] // Keep only the last MAX_HISTORY_SIZE entries (circular buffer) return newHistory.slice(-MAX_HISTORY_SIZE) }) expectHistoryExportRef.current = false } if (resolverRef.current) { resolverRef.current(extractedXML) resolverRef.current = null } } const clearDiagram = () => { const emptyDiagram = `` // Skip validation for trusted internal template (loadDiagram also sets chartXML) loadDiagram(emptyDiagram, true) setLatestSvg("") setDiagramHistory([]) } const saveDiagramToFile = ( filename: string, format: ExportFormat, sessionId?: string, ) => { if (!drawioRef.current) { console.warn("Draw.io editor not ready") return } // Map format to draw.io export format const drawioFormat = format === "drawio" ? "xmlsvg" : format // Set up the resolver before triggering export saveResolverRef.current = { resolver: (exportData: string) => { let fileContent: string | Blob let mimeType: string let extension: string if (format === "drawio") { // Extract XML from SVG for .drawio format const xml = extractDiagramXML(exportData) let xmlContent = xml if (!xml.includes("${xml}` } fileContent = xmlContent mimeType = "application/xml" extension = ".drawio" // Save to localStorage when user manually saves localStorage.setItem(STORAGE_DIAGRAM_XML_KEY, xmlContent) } else if (format === "png") { // PNG data comes as base64 data URL fileContent = exportData mimeType = "image/png" extension = ".png" } else { // SVG format fileContent = exportData mimeType = "image/svg+xml" extension = ".svg" } // Log save event to Langfuse (flags the trace) logSaveToLangfuse(filename, format, sessionId) // Handle download let url: string if ( typeof fileContent === "string" && fileContent.startsWith("data:") ) { // Already a data URL (PNG) url = fileContent } else { const blob = new Blob([fileContent], { type: mimeType }) url = URL.createObjectURL(blob) } const a = document.createElement("a") a.href = url a.download = `${filename}${extension}` document.body.appendChild(a) a.click() document.body.removeChild(a) // Delay URL revocation to ensure download completes if (!url.startsWith("data:")) { setTimeout(() => URL.revokeObjectURL(url), 100) } }, format, } // Export diagram - callback will be handled in handleDiagramExport drawioRef.current.exportDiagram({ format: drawioFormat }) } // Log save event to Langfuse (just flags the trace, doesn't send content) const logSaveToLangfuse = async ( filename: string, format: string, sessionId?: string, ) => { try { await fetch("/api/log-save", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ filename, format, sessionId }), }) } catch (error) { console.warn("Failed to log save to Langfuse:", error) } } return ( {children} ) } export function useDiagram() { const context = useContext(DiagramContext) if (context === undefined) { throw new Error("useDiagram must be used within a DiagramProvider") } return context }