fix: repair element bindings on restore (#5956)
* fix: repair element bindings on restore * fix dropping non-text bound elements * be more conservative
This commit is contained in:
parent
760fd7b3a6
commit
fffd4957db
@ -1,6 +1,7 @@
|
|||||||
import {
|
import {
|
||||||
ExcalidrawElement,
|
ExcalidrawElement,
|
||||||
ExcalidrawSelectionElement,
|
ExcalidrawSelectionElement,
|
||||||
|
ExcalidrawTextElement,
|
||||||
FontFamilyValues,
|
FontFamilyValues,
|
||||||
} from "../element/types";
|
} from "../element/types";
|
||||||
import {
|
import {
|
||||||
@ -16,7 +17,7 @@ import {
|
|||||||
isInvisiblySmallElement,
|
isInvisiblySmallElement,
|
||||||
refreshTextDimensions,
|
refreshTextDimensions,
|
||||||
} from "../element";
|
} from "../element";
|
||||||
import { isLinearElementType } from "../element/typeChecks";
|
import { isLinearElementType, isTextElement } from "../element/typeChecks";
|
||||||
import { randomId } from "../random";
|
import { randomId } from "../random";
|
||||||
import {
|
import {
|
||||||
DEFAULT_FONT_FAMILY,
|
DEFAULT_FONT_FAMILY,
|
||||||
@ -235,6 +236,82 @@ const restoreElement = (
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Repairs contaienr element's boundElements array by removing duplicates and
|
||||||
|
* fixing containerId of bound elements if not present. Also removes any
|
||||||
|
* bound elements that do not exist in the elements array.
|
||||||
|
*
|
||||||
|
* NOTE mutates elements.
|
||||||
|
*/
|
||||||
|
const repairContainerElement = (
|
||||||
|
container: Mutable<ExcalidrawElement>,
|
||||||
|
elementsMap: Map<string, Mutable<ExcalidrawElement>>,
|
||||||
|
) => {
|
||||||
|
if (container.boundElements) {
|
||||||
|
// copy because we're not cloning on restore, and we don't want to mutate upstream
|
||||||
|
const boundElements = container.boundElements.slice();
|
||||||
|
|
||||||
|
// dedupe bindings & fix boundElement.containerId if not set already
|
||||||
|
const boundIds = new Set<ExcalidrawElement["id"]>();
|
||||||
|
container.boundElements = boundElements.reduce(
|
||||||
|
(
|
||||||
|
acc: Mutable<NonNullable<ExcalidrawElement["boundElements"]>>,
|
||||||
|
binding,
|
||||||
|
) => {
|
||||||
|
const boundElement = elementsMap.get(binding.id);
|
||||||
|
if (boundElement && !boundIds.has(binding.id)) {
|
||||||
|
if (
|
||||||
|
isTextElement(boundElement) &&
|
||||||
|
// being slightly conservative here, preserving existing containerId
|
||||||
|
// if defined, lest boundElements is stale
|
||||||
|
!boundElement.containerId
|
||||||
|
) {
|
||||||
|
(boundElement as Mutable<ExcalidrawTextElement>).containerId =
|
||||||
|
container.id;
|
||||||
|
}
|
||||||
|
|
||||||
|
acc.push(binding);
|
||||||
|
boundIds.add(binding.id);
|
||||||
|
}
|
||||||
|
return acc;
|
||||||
|
},
|
||||||
|
[],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Repairs target bound element's container's boundElements array,
|
||||||
|
* or removes contaienrId if container does not exist.
|
||||||
|
*
|
||||||
|
* NOTE mutates elements.
|
||||||
|
*/
|
||||||
|
const repairBoundElement = (
|
||||||
|
boundElement: Mutable<ExcalidrawTextElement>,
|
||||||
|
elementsMap: Map<string, Mutable<ExcalidrawElement>>,
|
||||||
|
) => {
|
||||||
|
const container = boundElement.containerId
|
||||||
|
? elementsMap.get(boundElement.containerId)
|
||||||
|
: null;
|
||||||
|
|
||||||
|
if (!container) {
|
||||||
|
boundElement.containerId = null;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
container.boundElements &&
|
||||||
|
!container.boundElements.find((binding) => binding.id === boundElement.id)
|
||||||
|
) {
|
||||||
|
// copy because we're not cloning on restore, and we don't want to mutate upstream
|
||||||
|
const boundElements = (
|
||||||
|
container.boundElements || (container.boundElements = [])
|
||||||
|
).slice();
|
||||||
|
boundElements.push({ type: "text", id: boundElement.id });
|
||||||
|
container.boundElements = boundElements;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
export const restoreElements = (
|
export const restoreElements = (
|
||||||
elements: ImportedDataState["elements"],
|
elements: ImportedDataState["elements"],
|
||||||
/** NOTE doesn't serve for reconciliation */
|
/** NOTE doesn't serve for reconciliation */
|
||||||
@ -242,7 +319,7 @@ export const restoreElements = (
|
|||||||
refreshDimensions = false,
|
refreshDimensions = false,
|
||||||
): ExcalidrawElement[] => {
|
): ExcalidrawElement[] => {
|
||||||
const localElementsMap = localElements ? arrayToMap(localElements) : null;
|
const localElementsMap = localElements ? arrayToMap(localElements) : null;
|
||||||
return (elements || []).reduce((elements, element) => {
|
const restoredElements = (elements || []).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)) {
|
||||||
@ -260,6 +337,18 @@ export const restoreElements = (
|
|||||||
}
|
}
|
||||||
return elements;
|
return elements;
|
||||||
}, [] as ExcalidrawElement[]);
|
}, [] as ExcalidrawElement[]);
|
||||||
|
|
||||||
|
// repair binding. Mutates elements.
|
||||||
|
const restoredElementsMap = arrayToMap(restoredElements);
|
||||||
|
for (const element of restoredElements) {
|
||||||
|
if (isTextElement(element) && element.containerId) {
|
||||||
|
repairBoundElement(element, restoredElementsMap);
|
||||||
|
} else if (element.boundElements) {
|
||||||
|
repairContainerElement(element, restoredElementsMap);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return restoredElements;
|
||||||
};
|
};
|
||||||
|
|
||||||
const coalesceAppStateValue = <
|
const coalesceAppStateValue = <
|
||||||
|
Loading…
x
Reference in New Issue
Block a user