fix restoring appState (#2182)

This commit is contained in:
David Luzar 2020-09-22 21:51:49 +02:00 committed by GitHub
parent b2822f3538
commit adb1ac5788
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 74 additions and 49 deletions

View File

@ -3927,7 +3927,7 @@ class App extends React.Component<ExcalidrawProps, AppState> {
this.setState({ shouldCacheIgnoreZoom: false }); this.setState({ shouldCacheIgnoreZoom: false });
}, 300); }, 300);
private getCanvasOffsets() { private getCanvasOffsets(): Pick<AppState, "offsetTop" | "offsetLeft"> {
if (this.excalidrawRef?.current) { if (this.excalidrawRef?.current) {
const parentElement = this.excalidrawRef.current.parentElement; const parentElement = this.excalidrawRef.current.parentElement;
const { left, top } = parentElement.getBoundingClientRect(); const { left, top } = parentElement.getBoundingClientRect();

View File

@ -1,8 +1,8 @@
import { getDefaultAppState, cleanAppStateForExport } from "../appState"; import { cleanAppStateForExport } from "../appState";
import { restore } from "./restore"; import { restore } from "./restore";
import { t } from "../i18n"; import { t } from "../i18n";
import { AppState } from "../types"; import { AppState } from "../types";
import { LibraryData } from "./types"; import { LibraryData, ImportedDataState } from "./types";
import { calculateScrollCenter } from "../scene"; import { calculateScrollCenter } from "../scene";
const loadFileContents = async (blob: any) => { const loadFileContents = async (blob: any) => {
@ -34,26 +34,24 @@ export const loadFromBlob = async (blob: any, appState?: AppState) => {
} }
const contents = await loadFileContents(blob); const contents = await loadFileContents(blob);
const defaultAppState = getDefaultAppState();
let elements = [];
let _appState = appState || defaultAppState;
try { try {
const data = JSON.parse(contents); const data: ImportedDataState = JSON.parse(contents);
if (data.type !== "excalidraw") { if (data.type !== "excalidraw") {
throw new Error(t("alerts.couldNotLoadInvalidFile")); throw new Error(t("alerts.couldNotLoadInvalidFile"));
} }
elements = data.elements || []; return restore({
_appState = { elements: data.elements,
...defaultAppState, appState: {
appearance: _appState.appearance, appearance: appState?.appearance,
...cleanAppStateForExport(data.appState as Partial<AppState>), ...cleanAppStateForExport(data.appState || {}),
...(appState ? calculateScrollCenter(elements, appState, null) : {}), ...(appState
}; ? calculateScrollCenter(data.elements || [], appState, null)
: {}),
},
});
} catch { } catch {
throw new Error(t("alerts.couldNotLoadInvalidFile")); throw new Error(t("alerts.couldNotLoadInvalidFile"));
} }
return restore(elements, _appState);
}; };
export const loadLibraryFromBlob = async (blob: any) => { export const loadLibraryFromBlob = async (blob: any) => {

View File

@ -18,7 +18,7 @@ import { serializeAsJSON } from "./json";
import { ExportType } from "../scene/types"; import { ExportType } from "../scene/types";
import { restore } from "./restore"; import { restore } from "./restore";
import { DataState } from "./types"; import { ImportedDataState } from "./types";
export { loadFromBlob } from "./blob"; export { loadFromBlob } from "./blob";
export { saveAsJSON, loadFromJSON } from "./json"; export { saveAsJSON, loadFromJSON } from "./json";
@ -232,7 +232,7 @@ export const exportToBackend = async (
const importFromBackend = async ( const importFromBackend = async (
id: string | null, id: string | null,
privateKey?: string | null, privateKey?: string | null,
) => { ): Promise<ImportedDataState> => {
let elements: readonly ExcalidrawElement[] = []; let elements: readonly ExcalidrawElement[] = [];
let appState = getDefaultAppState(); let appState = getDefaultAppState();
@ -364,19 +364,15 @@ export const exportCanvas = async (
export const loadScene = async ( export const loadScene = async (
id: string | null, id: string | null,
privateKey?: string | null, privateKey?: string | null,
initialData?: DataState, initialData?: ImportedDataState,
) => { ) => {
let data; let data;
if (id != null) { if (id != null) {
// the private key is used to decrypt the content from the server, take // the private key is used to decrypt the content from the server, take
// extra care not to leak it // extra care not to leak it
const { elements, appState } = await importFromBackend(id, privateKey); data = restore(await importFromBackend(id, privateKey));
data = restore(elements, appState);
} else { } else {
data = restore( data = restore(initialData || {});
initialData?.elements ?? [],
initialData?.appState ?? getDefaultAppState(),
);
} }
return { return {

View File

@ -22,7 +22,7 @@ export const loadLibrary = (): Promise<LibraryItems> => {
} }
const items = (JSON.parse(data) as LibraryItems).map( const items = (JSON.parse(data) as LibraryItems).map(
(elements) => restore(elements, null).elements, (elements) => restore({ elements, appState: null }).elements,
) as Mutable<LibraryItems>; ) as Mutable<LibraryItems>;
// clone to ensure we don't mutate the cached library elements in the app // clone to ensure we don't mutate the cached library elements in the app

View File

@ -4,7 +4,7 @@ import {
ExcalidrawSelectionElement, ExcalidrawSelectionElement,
} from "../element/types"; } from "../element/types";
import { AppState } from "../types"; import { AppState } from "../types";
import { DataState } from "./types"; import { DataState, ImportedDataState } from "./types";
import { isInvisiblySmallElement, getNormalizedDimensions } from "../element"; import { isInvisiblySmallElement, getNormalizedDimensions } from "../element";
import { isLinearElementType } from "../element/typeChecks"; import { isLinearElementType } from "../element/typeChecks";
import { randomId } from "../random"; import { randomId } from "../random";
@ -14,6 +14,7 @@ import {
DEFAULT_TEXT_ALIGN, DEFAULT_TEXT_ALIGN,
DEFAULT_VERTICAL_ALIGN, DEFAULT_VERTICAL_ALIGN,
} from "../constants"; } from "../constants";
import { getDefaultAppState } from "../appState";
const getFontFamilyByName = (fontFamilyName: string): FontFamily => { const getFontFamilyByName = (fontFamilyName: string): FontFamily => {
for (const [id, fontFamilyString] of Object.entries(FONT_FAMILY)) { for (const [id, fontFamilyString] of Object.entries(FONT_FAMILY)) {
@ -24,10 +25,10 @@ const getFontFamilyByName = (fontFamilyName: string): FontFamily => {
return DEFAULT_FONT_FAMILY; return DEFAULT_FONT_FAMILY;
}; };
function migrateElementWithProperties<T extends ExcalidrawElement>( const restoreElementWithProperties = <T extends ExcalidrawElement>(
element: Required<T>, element: Required<T>,
extra: Omit<Required<T>, keyof ExcalidrawElement>, extra: Omit<Required<T>, keyof ExcalidrawElement>,
): T { ): T => {
const base: Pick<T, keyof ExcalidrawElement> = { const base: Pick<T, keyof ExcalidrawElement> = {
type: element.type, type: element.type,
// all elements must have version > 0 so getDrawingVersion() will pick up // all elements must have version > 0 so getDrawingVersion() will pick up
@ -61,9 +62,9 @@ function migrateElementWithProperties<T extends ExcalidrawElement>(
...getNormalizedDimensions(base), ...getNormalizedDimensions(base),
...extra, ...extra,
} as T; } as T;
} };
const migrateElement = ( const restoreElement = (
element: Exclude<ExcalidrawElement, ExcalidrawSelectionElement>, element: Exclude<ExcalidrawElement, ExcalidrawSelectionElement>,
): typeof element => { ): typeof element => {
switch (element.type) { switch (element.type) {
@ -78,7 +79,7 @@ const migrateElement = (
fontSize = parseInt(fontPx, 10); fontSize = parseInt(fontPx, 10);
fontFamily = getFontFamilyByName(_fontFamily); fontFamily = getFontFamilyByName(_fontFamily);
} }
return migrateElementWithProperties(element, { return restoreElementWithProperties(element, {
fontSize, fontSize,
fontFamily, fontFamily,
text: element.text ?? "", text: element.text ?? "",
@ -89,7 +90,7 @@ const migrateElement = (
case "draw": case "draw":
case "line": case "line":
case "arrow": { case "arrow": {
return migrateElementWithProperties(element, { return restoreElementWithProperties(element, {
startBinding: element.startBinding, startBinding: element.startBinding,
endBinding: element.endBinding, endBinding: element.endBinding,
points: points:
@ -105,11 +106,11 @@ const migrateElement = (
} }
// generic elements // generic elements
case "ellipse": case "ellipse":
return migrateElementWithProperties(element, {}); return restoreElementWithProperties(element, {});
case "rectangle": case "rectangle":
return migrateElementWithProperties(element, {}); return restoreElementWithProperties(element, {});
case "diamond": case "diamond":
return migrateElementWithProperties(element, {}); return restoreElementWithProperties(element, {});
// don't use default case so as to catch a missing an element type case // don't use default case so as to catch a missing an element type case
// (we also don't want to throw, but instead return void so we // (we also don't want to throw, but instead return void so we
@ -117,24 +118,46 @@ const migrateElement = (
} }
}; };
export const restore = ( const restoreElements = (
savedElements: readonly ExcalidrawElement[], elements: ImportedDataState["elements"],
savedState: MarkOptional<AppState, "offsetTop" | "offsetLeft"> | null, ): ExcalidrawElement[] => {
): DataState => { return (elements || []).reduce((elements, element) => {
const elements = savedElements.reduce((elements, element) => {
// filtering out selection, which is legacy, no longer kept in elements, // filtering out selection, which is legacy, no longer kept in elements,
// and causing issues if retained // and causing issues if retained
if (element.type !== "selection" && !isInvisiblySmallElement(element)) { if (element.type !== "selection" && !isInvisiblySmallElement(element)) {
const migratedElement = migrateElement(element); const migratedElement = restoreElement(element);
if (migratedElement) { if (migratedElement) {
elements.push(migratedElement); elements.push(migratedElement);
} }
} }
return elements; return elements;
}, [] as ExcalidrawElement[]); }, [] as ExcalidrawElement[]);
};
const restoreAppState = (appState: ImportedDataState["appState"]): AppState => {
appState = appState || {};
const defaultAppState = getDefaultAppState();
const nextAppState = {} as typeof defaultAppState;
for (const [key, val] of Object.entries(defaultAppState)) {
if ((appState as any)[key] !== undefined) {
(nextAppState as any)[key] = (appState as any)[key];
} else {
(nextAppState as any)[key] = val;
}
}
return { return {
elements: elements, ...nextAppState,
appState: savedState, offsetLeft: appState.offsetLeft || 0,
offsetTop: appState.offsetTop || 0,
};
};
export const restore = (data: ImportedDataState): DataState => {
return {
elements: restoreElements(data.elements),
appState: restoreAppState(data.appState),
}; };
}; };

View File

@ -6,7 +6,15 @@ export interface DataState {
version?: string; version?: string;
source?: string; source?: string;
elements: readonly ExcalidrawElement[]; elements: readonly ExcalidrawElement[];
appState: MarkOptional<AppState, "offsetTop" | "offsetLeft"> | null; appState: MarkOptional<AppState, "offsetTop" | "offsetLeft">;
}
export interface ImportedDataState {
type?: string;
version?: string;
source?: string;
elements?: DataState["elements"] | null;
appState?: Partial<DataState["appState"]> | null;
} }
export interface LibraryData { export interface LibraryData {

View File

@ -17,7 +17,7 @@ import {
} from "./data/localStorage"; } from "./data/localStorage";
import { SAVE_TO_LOCAL_STORAGE_TIMEOUT } from "./time_constants"; import { SAVE_TO_LOCAL_STORAGE_TIMEOUT } from "./time_constants";
import { DataState } from "./data/types"; import { ImportedDataState } from "./data/types";
import { LoadingMessage } from "./components/LoadingMessage"; import { LoadingMessage } from "./components/LoadingMessage";
import { ExcalidrawElement } from "./element/types"; import { ExcalidrawElement } from "./element/types";
import { AppState } from "./types"; import { AppState } from "./types";
@ -112,7 +112,7 @@ function ExcalidrawApp() {
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------
const [initialState, setInitialState] = useState<{ const [initialState, setInitialState] = useState<{
data: DataState; data: ImportedDataState;
user: { user: {
name: string | null; name: string | null;
}; };

View File

@ -14,7 +14,7 @@ import { Point as RoughPoint } from "roughjs/bin/geometry";
import { SocketUpdateDataSource } from "./data"; import { SocketUpdateDataSource } from "./data";
import { LinearElementEditor } from "./element/linearElementEditor"; import { LinearElementEditor } from "./element/linearElementEditor";
import { SuggestedBinding } from "./element/binding"; import { SuggestedBinding } from "./element/binding";
import { DataState } from "./data/types"; import { ImportedDataState } from "./data/types";
export type FlooredNumber = number & { _brand: "FlooredNumber" }; export type FlooredNumber = number & { _brand: "FlooredNumber" };
export type Point = Readonly<RoughPoint>; export type Point = Readonly<RoughPoint>;
@ -127,7 +127,7 @@ export interface ExcalidrawProps {
elements: readonly ExcalidrawElement[], elements: readonly ExcalidrawElement[],
appState: AppState, appState: AppState,
) => void; ) => void;
initialData?: DataState; initialData?: ImportedDataState;
user?: { user?: {
name?: string | null; name?: string | null;
}; };