* Scroll content to the center when loading from backend * spread * Load from file * Return type
303 lines
8.0 KiB
TypeScript
303 lines
8.0 KiB
TypeScript
import { ExcalidrawElement } from "../element/types";
|
|
|
|
import { getDefaultAppState } from "../appState";
|
|
|
|
import { AppState } from "../types";
|
|
import { ExportType } from "./types";
|
|
import { getExportCanvasPreview } from "./getExportCanvasPreview";
|
|
import nanoid from "nanoid";
|
|
import { fileOpen, fileSave } from "browser-nativefs";
|
|
|
|
import i18n from "../i18n";
|
|
|
|
const LOCAL_STORAGE_KEY = "excalidraw";
|
|
const LOCAL_STORAGE_KEY_STATE = "excalidraw-state";
|
|
const BACKEND_POST = "https://json.excalidraw.com/api/v1/post/";
|
|
const BACKEND_GET = "https://json.excalidraw.com/api/v1/";
|
|
|
|
// TODO: Defined globally, since file handles aren't yet serializable.
|
|
// Once `FileSystemFileHandle` can be serialized, make this
|
|
// part of `AppState`.
|
|
(window as any).handle = null;
|
|
|
|
interface DataState {
|
|
elements: readonly ExcalidrawElement[];
|
|
appState: AppState;
|
|
}
|
|
|
|
export function serializeAsJSON(
|
|
elements: readonly ExcalidrawElement[],
|
|
appState?: AppState,
|
|
): string {
|
|
return JSON.stringify({
|
|
version: 1,
|
|
source: window.location.origin,
|
|
elements: elements.map(({ shape, ...el }) => el),
|
|
appState: appState || getDefaultAppState(),
|
|
});
|
|
}
|
|
|
|
function calculateScroll(
|
|
elements: readonly ExcalidrawElement[],
|
|
): { scrollX: number; scrollY: number } {
|
|
// Bounding box of all elements
|
|
let top = Number.MAX_SAFE_INTEGER;
|
|
let left = Number.MAX_SAFE_INTEGER;
|
|
let bottom = -Number.MAX_SAFE_INTEGER;
|
|
let right = -Number.MAX_SAFE_INTEGER;
|
|
|
|
for (const element of elements) {
|
|
left = Math.min(
|
|
left,
|
|
element.width > 0 ? element.x : element.x + element.width,
|
|
);
|
|
top = Math.min(
|
|
top,
|
|
element.height > 0 ? element.y : element.y + element.height,
|
|
);
|
|
right = Math.max(
|
|
right,
|
|
element.width > 0 ? element.x + element.width : element.x,
|
|
);
|
|
bottom = Math.max(
|
|
bottom,
|
|
element.height > 0 ? element.y + element.height : element.y,
|
|
);
|
|
}
|
|
const centerX = left + (right - left) / 2;
|
|
const centerY = top + (bottom - top) / 2;
|
|
return {
|
|
scrollX: window.innerWidth / 2 - centerX,
|
|
scrollY: window.innerHeight / 2 - centerY,
|
|
};
|
|
}
|
|
|
|
export async function saveAsJSON(
|
|
elements: readonly ExcalidrawElement[],
|
|
appState: AppState,
|
|
) {
|
|
const serialized = serializeAsJSON(elements, appState);
|
|
|
|
const name = `${appState.name}.json`;
|
|
await fileSave(
|
|
new Blob([serialized], { type: "application/json" }),
|
|
{
|
|
fileName: name,
|
|
description: "Excalidraw file",
|
|
},
|
|
(window as any).handle,
|
|
);
|
|
}
|
|
|
|
export async function loadFromJSON() {
|
|
const updateAppState = (contents: string) => {
|
|
const defaultAppState = getDefaultAppState();
|
|
let elements = [];
|
|
let appState = defaultAppState;
|
|
try {
|
|
const data = JSON.parse(contents);
|
|
elements = data.elements || [];
|
|
appState = { ...defaultAppState, ...data.appState };
|
|
} catch (e) {
|
|
// Do nothing because elements array is already empty
|
|
}
|
|
return { elements, appState };
|
|
};
|
|
|
|
const blob = await fileOpen({
|
|
description: "Excalidraw files",
|
|
extensions: ["json"],
|
|
mimeTypes: ["application/json"],
|
|
});
|
|
if (blob.handle) {
|
|
(window as any).handle = blob.handle;
|
|
}
|
|
let contents;
|
|
if ("text" in Blob) {
|
|
contents = await blob.text();
|
|
} else {
|
|
contents = await (async () => {
|
|
return new Promise(resolve => {
|
|
const reader = new FileReader();
|
|
reader.readAsText(blob, "utf8");
|
|
reader.onloadend = () => {
|
|
if (reader.readyState === FileReader.DONE) {
|
|
resolve(reader.result as string);
|
|
}
|
|
};
|
|
});
|
|
})();
|
|
}
|
|
const { elements, appState } = updateAppState(contents);
|
|
return new Promise<DataState>(resolve => {
|
|
resolve(restore(elements, { ...appState, ...calculateScroll(elements) }));
|
|
});
|
|
}
|
|
|
|
export async function exportToBackend(
|
|
elements: readonly ExcalidrawElement[],
|
|
appState: AppState,
|
|
) {
|
|
const response = await fetch(BACKEND_POST, {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: serializeAsJSON(elements, appState),
|
|
});
|
|
const json = await response.json();
|
|
if (json.id) {
|
|
const url = new URL(window.location.href);
|
|
url.searchParams.append("id", json.id);
|
|
|
|
await navigator.clipboard.writeText(url.toString());
|
|
window.alert(
|
|
i18n.t("alerts.copiedToClipboard", {
|
|
url: url.toString(),
|
|
interpolation: { escapeValue: false },
|
|
}),
|
|
);
|
|
} else {
|
|
window.alert(i18n.t("alerts.couldNotCreateShareableLink"));
|
|
}
|
|
}
|
|
|
|
export async function importFromBackend(id: string | null) {
|
|
let elements: readonly ExcalidrawElement[] = [];
|
|
let appState: AppState = getDefaultAppState();
|
|
const response = await fetch(`${BACKEND_GET}${id}.json`).then(data =>
|
|
data.clone().json(),
|
|
);
|
|
if (response != null) {
|
|
try {
|
|
elements = response.elements || elements;
|
|
appState = response.appState || appState;
|
|
} catch (error) {
|
|
window.alert(i18n.t("alerts.importBackendFailed"));
|
|
console.error(error);
|
|
}
|
|
}
|
|
return restore(elements, { ...appState, ...calculateScroll(elements) });
|
|
}
|
|
|
|
export async function exportCanvas(
|
|
type: ExportType,
|
|
elements: readonly ExcalidrawElement[],
|
|
canvas: HTMLCanvasElement,
|
|
{
|
|
exportBackground,
|
|
exportPadding = 10,
|
|
viewBackgroundColor,
|
|
name,
|
|
scale = 1,
|
|
}: {
|
|
exportBackground: boolean;
|
|
exportPadding?: number;
|
|
viewBackgroundColor: string;
|
|
name: string;
|
|
scale?: number;
|
|
},
|
|
) {
|
|
if (!elements.length)
|
|
return window.alert(i18n.t("alerts.cannotExportEmptyCanvas"));
|
|
// calculate smallest area to fit the contents in
|
|
|
|
const tempCanvas = getExportCanvasPreview(elements, {
|
|
exportBackground,
|
|
viewBackgroundColor,
|
|
exportPadding,
|
|
scale,
|
|
});
|
|
tempCanvas.style.display = "none";
|
|
document.body.appendChild(tempCanvas);
|
|
|
|
if (type === "png") {
|
|
const fileName = `${name}.png`;
|
|
tempCanvas.toBlob(async (blob: any) => {
|
|
if (blob) {
|
|
await fileSave(blob, {
|
|
fileName: fileName,
|
|
description: "Excalidraw image",
|
|
});
|
|
}
|
|
});
|
|
} else if (type === "clipboard") {
|
|
const errorMsg = i18n.t("alerts.couldNotCopyToClipboard");
|
|
try {
|
|
tempCanvas.toBlob(async function(blob: any) {
|
|
try {
|
|
await navigator.clipboard.write([
|
|
new window.ClipboardItem({ "image/png": blob }),
|
|
]);
|
|
} catch (err) {
|
|
window.alert(errorMsg);
|
|
}
|
|
});
|
|
} catch (err) {
|
|
window.alert(errorMsg);
|
|
}
|
|
} else if (type === "backend") {
|
|
const appState = getDefaultAppState();
|
|
if (exportBackground) {
|
|
appState.viewBackgroundColor = viewBackgroundColor;
|
|
}
|
|
exportToBackend(elements, appState);
|
|
}
|
|
|
|
// clean up the DOM
|
|
if (tempCanvas !== canvas) tempCanvas.remove();
|
|
}
|
|
|
|
function restore(
|
|
savedElements: readonly ExcalidrawElement[],
|
|
savedState: AppState,
|
|
): DataState {
|
|
return {
|
|
elements: savedElements.map(element => ({
|
|
...element,
|
|
id: element.id || nanoid(),
|
|
fillStyle: element.fillStyle || "hachure",
|
|
strokeWidth: element.strokeWidth || 1,
|
|
roughness: element.roughness || 1,
|
|
opacity:
|
|
element.opacity === null || element.opacity === undefined
|
|
? 100
|
|
: element.opacity,
|
|
})),
|
|
appState: savedState,
|
|
};
|
|
}
|
|
|
|
export function restoreFromLocalStorage() {
|
|
const savedElements = localStorage.getItem(LOCAL_STORAGE_KEY);
|
|
const savedState = localStorage.getItem(LOCAL_STORAGE_KEY_STATE);
|
|
|
|
let elements = [];
|
|
if (savedElements) {
|
|
try {
|
|
elements = JSON.parse(savedElements).map(
|
|
({ shape, ...element }: ExcalidrawElement) => element,
|
|
);
|
|
} catch (e) {
|
|
// Do nothing because elements array is already empty
|
|
}
|
|
}
|
|
|
|
let appState = null;
|
|
if (savedState) {
|
|
try {
|
|
appState = JSON.parse(savedState);
|
|
} catch (e) {
|
|
// Do nothing because appState is already null
|
|
}
|
|
}
|
|
|
|
return restore(elements, appState);
|
|
}
|
|
|
|
export function saveToLocalStorage(
|
|
elements: readonly ExcalidrawElement[],
|
|
state: AppState,
|
|
) {
|
|
localStorage.setItem(LOCAL_STORAGE_KEY, JSON.stringify(elements));
|
|
localStorage.setItem(LOCAL_STORAGE_KEY_STATE, JSON.stringify(state));
|
|
}
|