import React from "react"; import { Card } from "../../src/components/Card"; import { ToolButton } from "../../src/components/ToolButton"; import { serializeAsJSON } from "../../src/data/json"; import { loadFirebaseStorage, saveFilesToFirebase } from "../data/firebase"; import { FileId, NonDeletedExcalidrawElement } from "../../src/element/types"; import { AppState, BinaryFileData, BinaryFiles } from "../../src/types"; import { nanoid } from "nanoid"; import { useI18n } from "../../src/i18n"; import { encryptData, generateEncryptionKey } from "../../src/data/encryption"; import { isInitializedImageElement } from "../../src/element/typeChecks"; import { FILE_UPLOAD_MAX_BYTES } from "../app_constants"; import { encodeFilesForUpload } from "../data/FileManager"; import { MIME_TYPES } from "../../src/constants"; import { trackEvent } from "../../src/analytics"; import { getFrame } from "../../src/utils"; import { ExcalidrawLogo } from "../../src/components/ExcalidrawLogo"; export const exportToExcalidrawPlus = async ( elements: readonly NonDeletedExcalidrawElement[], appState: Partial, files: BinaryFiles, ) => { const firebase = await loadFirebaseStorage(); const id = `${nanoid(12)}`; const encryptionKey = (await generateEncryptionKey())!; const encryptedData = await encryptData( encryptionKey, serializeAsJSON(elements, appState, files, "database"), ); const blob = new Blob( [encryptedData.iv, new Uint8Array(encryptedData.encryptedBuffer)], { type: MIME_TYPES.binary, }, ); await firebase .storage() .ref(`/migrations/scenes/${id}`) .put(blob, { customMetadata: { data: JSON.stringify({ version: 2, name: appState.name }), created: Date.now().toString(), }, }); const filesMap = new Map(); for (const element of elements) { if (isInitializedImageElement(element) && files[element.fileId]) { filesMap.set(element.fileId, files[element.fileId]); } } if (filesMap.size) { const filesToUpload = await encodeFilesForUpload({ files: filesMap, encryptionKey, maxBytes: FILE_UPLOAD_MAX_BYTES, }); await saveFilesToFirebase({ prefix: `/migrations/files/scenes/${id}`, files: filesToUpload, }); } window.open( `${ import.meta.env.VITE_APP_PLUS_APP }/import?excalidraw=${id},${encryptionKey}`, ); }; export const ExportToExcalidrawPlus: React.FC<{ elements: readonly NonDeletedExcalidrawElement[]; appState: Partial; files: BinaryFiles; onError: (error: Error) => void; }> = ({ elements, appState, files, onError }) => { const { t } = useI18n(); return (

Excalidraw+

{t("exportDialog.excalidrawplus_description")}
{ try { trackEvent("export", "eplus", `ui (${getFrame()})`); await exportToExcalidrawPlus(elements, appState, files); } catch (error: any) { console.error(error); if (error.name !== "AbortError") { onError(new Error(t("exportDialog.excalidrawplus_exportError"))); } } }} />
); };