import { NodeViewProps, NodeViewWrapper } from "@tiptap/react"; import { ActionIcon, Card, Image, Modal, Text, useComputedColorScheme, } from "@mantine/core"; import { useRef, useState } from "react"; import { uploadFile } from "@/features/page/services/page-service.ts"; import { useDisclosure } from "@mantine/hooks"; import { getDrawioUrl, getFileUrl } from "@/lib/config.ts"; import { DrawIoEmbed, DrawIoEmbedRef, EventExit, EventSave, } from "react-drawio"; import { IAttachment } from "@/lib/types"; import { decodeBase64ToSvgString, svgStringToFile } from "@/lib/utils"; import clsx from "clsx"; import { IconEdit } from "@tabler/icons-react"; import { useTranslation } from "react-i18next"; export default function DrawioView(props: NodeViewProps) { const { t } = useTranslation(); const { node, updateAttributes, editor, selected } = props; const { src, title, width, attachmentId } = node.attrs; const drawioRef = useRef(null); const [initialXML, setInitialXML] = useState(""); const [opened, { open, close }] = useDisclosure(false); const computedColorScheme = useComputedColorScheme(); const handleOpen = async () => { if (!editor.isEditable) { return; } try { if (src) { const url = getFileUrl(src); const request = await fetch(url, { credentials: "include", cache: "no-store", }); const blob = await request.blob(); const reader = new FileReader(); reader.readAsDataURL(blob); reader.onloadend = () => { const base64data = (reader.result || "") as string; setInitialXML(base64data); }; } } catch (err) { console.error(err); } finally { open(); } }; const handleSave = async (data: EventSave) => { const svgString = decodeBase64ToSvgString(data.xml); const fileName = "diagram.drawio.svg"; const drawioSVGFile = await svgStringToFile(svgString, fileName); const pageId = editor.storage?.pageId; let attachment: IAttachment = null; if (attachmentId) { attachment = await uploadFile(drawioSVGFile, pageId, attachmentId); } else { attachment = await uploadFile(drawioSVGFile, pageId); } updateAttributes({ src: `/api/files/${attachment.id}/${attachment.fileName}?t=${new Date(attachment.updatedAt).getTime()}`, title: attachment.fileName, size: attachment.fileSize, attachmentId: attachment.id, }); close(); }; return (
{ // If the save is triggered by another event, then do nothing if (data.parentEvent !== "save") { return; } handleSave(data); }} onClose={(data: EventExit) => { // If the exit is triggered by another event, then do nothing if (data.parentEvent) { return; } close(); }} />
{src ? (
e.detail === 2 && handleOpen()} radius="md" fit="contain" w={width} src={getFileUrl(src)} alt={title} className={clsx( selected ? "ProseMirror-selectednode" : "", "alignCenter", )} /> {selected && ( )}
) : ( e.detail === 2 && handleOpen()} p="xs" style={{ display: "flex", justifyContent: "center", alignItems: "center", }} withBorder className={clsx(selected ? "ProseMirror-selectednode" : "")} >
{t("Double-click to edit Draw.io diagram")}
)}
); }