import { writable } from "svelte/store"; import type { Writable } from "svelte/store"; import type { OpenDocument } from "@graphite/../wasm/pkg/graphite_wasm"; import type { Editor } from "@graphite/editor"; import { downloadFile, downloadFileBlob, upload } from "@graphite/utility-functions/files"; import { rasterizeSVG } from "@graphite/utility-functions/rasterization"; export type PortfolioStore = ReturnType; type PortfolioStoreState = { unsaved: boolean; documents: OpenDocument[]; activeDocumentIndex: number; dataPanelOpen: boolean; propertiesPanelOpen: boolean; layersPanelOpen: boolean; }; const initialState: PortfolioStoreState = { unsaved: false, documents: [], activeDocumentIndex: 0, dataPanelOpen: false, propertiesPanelOpen: true, layersPanelOpen: true, }; let editorRef: Editor | undefined = undefined; // Store state persisted across HMR to maintain reactive subscriptions in the component tree const store: Writable = import.meta.hot?.data?.store || writable(initialState); if (import.meta.hot) import.meta.hot.data.store = store; const { subscribe, update } = store; export function createPortfolioStore(editor: Editor) { destroyPortfolioStore(); editorRef = editor; editor.subscriptions.subscribeFrontendMessage("UpdateOpenDocumentsList", (data) => { update((state) => { state.documents = data.openDocuments; return state; }); }); editor.subscriptions.subscribeFrontendMessage("UpdateActiveDocument", (data) => { update((state) => { // Assume we receive a correct document id const activeId = state.documents.findIndex((doc) => doc.id === data.documentId); state.activeDocumentIndex = activeId; return state; }); }); editor.subscriptions.subscribeFrontendMessage("TriggerFetchAndOpenDocument", async (data) => { try { const url = new URL(`demo-artwork/${data.filename}`, document.location.href); const response = await fetch(url); editor.handle.openFile(data.filename, await response.bytes()); } catch { // Needs to be delayed until the end of the current call stack so the existing demo artwork dialog can be closed first, otherwise this dialog won't show setTimeout(() => { editor.handle.errorDialog("Failed to open document", "The file could not be reached over the internet. You may be offline, or it may be missing."); }, 0); } }); editor.subscriptions.subscribeFrontendMessage("TriggerOpen", async () => { const data = await upload(`image/*,.${editor.handle.fileExtension()}`, "data"); editor.handle.openFile(data.filename, data.content); }); editor.subscriptions.subscribeFrontendMessage("TriggerImport", async () => { // TODO: Use the same `accept` string as in the `TriggerOpen` handler once importing Graphite documents as nodes is supported const data = await upload("image/*", "data"); editor.handle.importFile(data.filename, data.content); }); editor.subscriptions.subscribeFrontendMessage("TriggerSaveDocument", (data) => { downloadFile(data.name, data.content); }); editor.subscriptions.subscribeFrontendMessage("TriggerSaveFile", (data) => { downloadFile(data.name, data.content); }); editor.subscriptions.subscribeFrontendMessage("TriggerExportImage", async (data) => { const { svg, name, mime, size } = data; // Fill the canvas with white if it'll be a JPEG (which does not support transparency and defaults to black) const backgroundColor = mime.endsWith("jpeg") ? "white" : undefined; // Rasterize the SVG to an image file try { const blob = await rasterizeSVG(svg, size[0], size[1], mime, backgroundColor); // Have the browser download the file to the user's disk downloadFileBlob(name, blob); } catch { // Fail silently if there's an error rasterizing the SVG, such as a zero-sized image } }); editor.subscriptions.subscribeFrontendMessage("UpdateDataPanelState", async (data) => { update((state) => { state.dataPanelOpen = data.open; return state; }); }); editor.subscriptions.subscribeFrontendMessage("UpdatePropertiesPanelState", async (data) => { update((state) => { state.propertiesPanelOpen = data.open; return state; }); }); editor.subscriptions.subscribeFrontendMessage("UpdateLayersPanelState", async (data) => { update((state) => { state.layersPanelOpen = data.open; return state; }); }); return { subscribe }; } export function destroyPortfolioStore() { const editor = editorRef; if (!editor) return; editor.subscriptions.unsubscribeFrontendMessage("UpdateOpenDocumentsList"); editor.subscriptions.unsubscribeFrontendMessage("UpdateActiveDocument"); editor.subscriptions.unsubscribeFrontendMessage("TriggerFetchAndOpenDocument"); editor.subscriptions.unsubscribeFrontendMessage("TriggerOpen"); editor.subscriptions.unsubscribeFrontendMessage("TriggerImport"); editor.subscriptions.unsubscribeFrontendMessage("TriggerSaveDocument"); editor.subscriptions.unsubscribeFrontendMessage("TriggerSaveFile"); editor.subscriptions.unsubscribeFrontendMessage("TriggerExportImage"); editor.subscriptions.unsubscribeFrontendMessage("UpdateDataPanelState"); editor.subscriptions.unsubscribeFrontendMessage("UpdatePropertiesPanelState"); editor.subscriptions.unsubscribeFrontendMessage("UpdateLayersPanelState"); }