2020-03-14 21:48:51 -07:00
|
|
|
import { Point } from "../types";
|
2020-03-07 10:20:38 -05:00
|
|
|
|
2020-05-27 15:14:50 +02:00
|
|
|
import {
|
|
|
|
ExcalidrawElement,
|
|
|
|
ExcalidrawTextElement,
|
|
|
|
FontFamily,
|
|
|
|
} from "../element/types";
|
2020-03-07 10:20:38 -05:00
|
|
|
import { AppState } from "../types";
|
|
|
|
import { DataState } from "./types";
|
2020-04-08 21:00:27 +01:00
|
|
|
import {
|
|
|
|
isInvisiblySmallElement,
|
|
|
|
normalizeDimensions,
|
|
|
|
isTextElement,
|
|
|
|
} from "../element";
|
2020-03-07 10:20:38 -05:00
|
|
|
import { calculateScrollCenter } from "../scene";
|
2020-03-23 16:38:41 -07:00
|
|
|
import { randomId } from "../random";
|
2020-05-27 15:14:50 +02:00
|
|
|
import { DEFAULT_TEXT_ALIGN, DEFAULT_FONT_FAMILY } from "../appState";
|
|
|
|
import { FONT_FAMILY } from "../constants";
|
|
|
|
|
|
|
|
const getFontFamilyByName = (fontFamilyName: string): FontFamily => {
|
|
|
|
for (const [id, fontFamilyString] of Object.entries(FONT_FAMILY)) {
|
|
|
|
if (fontFamilyString.includes(fontFamilyName)) {
|
|
|
|
return parseInt(id) as FontFamily;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return DEFAULT_FONT_FAMILY;
|
|
|
|
};
|
2020-03-07 10:20:38 -05:00
|
|
|
|
2020-05-20 16:21:37 +03:00
|
|
|
export const restore = (
|
2020-03-17 20:55:40 +01:00
|
|
|
// we're making the elements mutable for this API because we want to
|
|
|
|
// efficiently remove/tweak properties on them (to migrate old scenes)
|
|
|
|
savedElements: readonly Mutable<ExcalidrawElement>[],
|
2020-03-07 10:20:38 -05:00
|
|
|
savedState: AppState | null,
|
|
|
|
opts?: { scrollToContent: boolean },
|
2020-05-20 16:21:37 +03:00
|
|
|
): DataState => {
|
2020-03-07 10:20:38 -05:00
|
|
|
const elements = savedElements
|
2020-03-23 13:05:07 +02:00
|
|
|
.filter((el) => {
|
2020-03-10 19:41:41 +01:00
|
|
|
// filtering out selection, which is legacy, no longer kept in elements,
|
|
|
|
// and causing issues if retained
|
|
|
|
return el.type !== "selection" && !isInvisiblySmallElement(el);
|
|
|
|
})
|
2020-03-23 13:05:07 +02:00
|
|
|
.map((element) => {
|
2020-03-07 10:20:38 -05:00
|
|
|
let points: Point[] = [];
|
|
|
|
if (element.type === "arrow") {
|
|
|
|
if (Array.isArray(element.points)) {
|
|
|
|
// if point array is empty, add one point to the arrow
|
|
|
|
// this is used as fail safe to convert incoming data to a valid
|
|
|
|
// arrow. In the new arrow, width and height are not being usde
|
|
|
|
points = element.points.length > 0 ? element.points : [[0, 0]];
|
|
|
|
} else {
|
|
|
|
// convert old arrow type to a new one
|
|
|
|
// old arrow spec used width and height
|
|
|
|
// to determine the endpoints
|
|
|
|
points = [
|
|
|
|
[0, 0],
|
|
|
|
[element.width, element.height],
|
|
|
|
];
|
|
|
|
}
|
2020-03-17 20:55:40 +01:00
|
|
|
element.points = points;
|
2020-05-12 20:10:11 +01:00
|
|
|
} else if (element.type === "line" || element.type === "draw") {
|
2020-03-07 10:20:38 -05:00
|
|
|
// old spec, pre-arrows
|
|
|
|
// old spec, post-arrows
|
|
|
|
if (!Array.isArray(element.points) || element.points.length === 0) {
|
|
|
|
points = [
|
|
|
|
[0, 0],
|
|
|
|
[element.width, element.height],
|
|
|
|
];
|
|
|
|
} else {
|
|
|
|
points = element.points;
|
|
|
|
}
|
2020-03-17 20:55:40 +01:00
|
|
|
element.points = points;
|
2020-03-07 10:20:38 -05:00
|
|
|
} else {
|
2020-04-08 21:00:27 +01:00
|
|
|
if (isTextElement(element)) {
|
2020-05-27 15:14:50 +02:00
|
|
|
if ("font" in element) {
|
|
|
|
const [fontPx, fontFamily]: [
|
|
|
|
string,
|
|
|
|
string,
|
|
|
|
] = (element as any).font.split(" ");
|
|
|
|
(element as Mutable<ExcalidrawTextElement>).fontSize = parseInt(
|
|
|
|
fontPx,
|
|
|
|
10,
|
|
|
|
);
|
|
|
|
(element as Mutable<
|
|
|
|
ExcalidrawTextElement
|
|
|
|
>).fontFamily = getFontFamilyByName(fontFamily);
|
|
|
|
delete (element as any).font;
|
|
|
|
}
|
2020-04-11 17:59:22 +09:00
|
|
|
if (!element.textAlign) {
|
|
|
|
element.textAlign = DEFAULT_TEXT_ALIGN;
|
|
|
|
}
|
2020-04-08 21:00:27 +01:00
|
|
|
}
|
|
|
|
|
2020-03-07 10:20:38 -05:00
|
|
|
normalizeDimensions(element);
|
2020-03-17 20:55:40 +01:00
|
|
|
// old spec, where non-linear elements used to have empty points arrays
|
|
|
|
if ("points" in element) {
|
|
|
|
delete element.points;
|
|
|
|
}
|
2020-03-07 10:20:38 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
...element,
|
2020-05-26 13:07:46 -07:00
|
|
|
// all elements must have version > 0 so getDrawingVersion() will pick
|
|
|
|
// up newly added elements
|
2020-03-14 20:46:57 -07:00
|
|
|
version: element.version || 1,
|
2020-03-23 16:38:41 -07:00
|
|
|
id: element.id || randomId(),
|
2020-05-23 07:26:59 +02:00
|
|
|
isDeleted: false,
|
2020-03-07 10:20:38 -05:00
|
|
|
fillStyle: element.fillStyle || "hachure",
|
|
|
|
strokeWidth: element.strokeWidth || 1,
|
2020-05-14 17:04:33 +02:00
|
|
|
strokeStyle: element.strokeStyle ?? "solid",
|
2020-03-12 17:49:11 +01:00
|
|
|
roughness: element.roughness ?? 1,
|
2020-03-07 10:20:38 -05:00
|
|
|
opacity:
|
|
|
|
element.opacity === null || element.opacity === undefined
|
|
|
|
? 100
|
|
|
|
: element.opacity,
|
2020-04-02 17:40:26 +09:00
|
|
|
angle: element.angle ?? 0,
|
2020-05-26 13:07:46 -07:00
|
|
|
groupIds: element.groupIds || [],
|
2020-03-07 10:20:38 -05:00
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
if (opts?.scrollToContent && savedState) {
|
|
|
|
savedState = { ...savedState, ...calculateScrollCenter(elements) };
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
elements: elements,
|
|
|
|
appState: savedState,
|
|
|
|
};
|
2020-05-20 16:21:37 +03:00
|
|
|
};
|