
* lasso without 'real' shape detection * select a single linear el * improve ux * feed segments to worker * simplify path threshold adaptive to zoom * add a tiny threshold for checks * refactor code * lasso tests * fix: ts * do not capture lasso tool * try worker-loader in next config * update config * refactor * lint * feat: show active tool when using "more tools" * keep lasso if selected from toolbar * fix incorrect checks for resetting to selection * shift for additive selection * bound text related fixes * lint * keep alt toggled lasso selection if shift pressed * fix regression * fix 'dead' lassos * lint * use workerpool and polyfill * fix worker bundled with window related code * refactor * add file extension for worker constructor error * another attempt at constructor error * attempt at build issue * attempt with dynamic import * test not importing from math * narrow down imports * Reusing existing workers infrastructure (fallback to the main thread, type-safety) * Points on curve inside the shared chunk * Give up on experimental code splitting * Remove potentially unnecessary optimisation * Removing workers as the complexit is much worse, while perf. does not seem to be much better * fix selecting text containers and containing frames together * render fill directly from animated trail * do not re-render static when setting selected element ids in lasso * remove unnecessary property * tweak trail animation * slice points to remove notch * always start alt-lasso from initial point * revert build & worker changes (unused) * remove `lasso` from `hasStrokeColor` * label change * remove unused props * remove unsafe optimization * snaps --------- Co-authored-by: dwelle <5153846+dwelle@users.noreply.github.com> Co-authored-by: Marcel Mraz <marcel@excalidraw.com>
124 lines
3.7 KiB
TypeScript
124 lines
3.7 KiB
TypeScript
import { KEYS, arrayToMap } from "@excalidraw/common";
|
|
|
|
import { newElementWith } from "@excalidraw/element/mutateElement";
|
|
|
|
import { isFrameLikeElement } from "@excalidraw/element/typeChecks";
|
|
|
|
import type { ExcalidrawElement } from "@excalidraw/element/types";
|
|
|
|
import { LockedIcon, UnlockedIcon } from "../components/icons";
|
|
|
|
import { getSelectedElements } from "../scene";
|
|
import { CaptureUpdateAction } from "../store";
|
|
|
|
import { register } from "./register";
|
|
|
|
const shouldLock = (elements: readonly ExcalidrawElement[]) =>
|
|
elements.every((el) => !el.locked);
|
|
|
|
export const actionToggleElementLock = register({
|
|
name: "toggleElementLock",
|
|
label: (elements, appState, app) => {
|
|
const selected = app.scene.getSelectedElements({
|
|
selectedElementIds: appState.selectedElementIds,
|
|
includeBoundTextElement: false,
|
|
});
|
|
if (selected.length === 1 && !isFrameLikeElement(selected[0])) {
|
|
return selected[0].locked
|
|
? "labels.elementLock.unlock"
|
|
: "labels.elementLock.lock";
|
|
}
|
|
|
|
return shouldLock(selected)
|
|
? "labels.elementLock.lockAll"
|
|
: "labels.elementLock.unlockAll";
|
|
},
|
|
icon: (appState, elements) => {
|
|
const selectedElements = getSelectedElements(elements, appState);
|
|
return shouldLock(selectedElements) ? LockedIcon : UnlockedIcon;
|
|
},
|
|
trackEvent: { category: "element" },
|
|
predicate: (elements, appState, _, app) => {
|
|
const selectedElements = app.scene.getSelectedElements(appState);
|
|
return (
|
|
selectedElements.length > 0 &&
|
|
!selectedElements.some((element) => element.locked && element.frameId)
|
|
);
|
|
},
|
|
perform: (elements, appState, _, app) => {
|
|
const selectedElements = app.scene.getSelectedElements({
|
|
selectedElementIds: appState.selectedElementIds,
|
|
includeBoundTextElement: true,
|
|
includeElementsInFrames: true,
|
|
});
|
|
|
|
if (!selectedElements.length) {
|
|
return false;
|
|
}
|
|
|
|
const nextLockState = shouldLock(selectedElements);
|
|
const selectedElementsMap = arrayToMap(selectedElements);
|
|
return {
|
|
elements: elements.map((element) => {
|
|
if (!selectedElementsMap.has(element.id)) {
|
|
return element;
|
|
}
|
|
|
|
return newElementWith(element, { locked: nextLockState });
|
|
}),
|
|
appState: {
|
|
...appState,
|
|
selectedLinearElement: nextLockState
|
|
? null
|
|
: appState.selectedLinearElement,
|
|
},
|
|
captureUpdate: CaptureUpdateAction.IMMEDIATELY,
|
|
};
|
|
},
|
|
keyTest: (event, appState, elements, app) => {
|
|
return (
|
|
event.key.toLocaleLowerCase() === KEYS.L &&
|
|
event[KEYS.CTRL_OR_CMD] &&
|
|
event.shiftKey &&
|
|
app.scene.getSelectedElements({
|
|
selectedElementIds: appState.selectedElementIds,
|
|
includeBoundTextElement: false,
|
|
}).length > 0
|
|
);
|
|
},
|
|
});
|
|
|
|
export const actionUnlockAllElements = register({
|
|
name: "unlockAllElements",
|
|
trackEvent: { category: "canvas" },
|
|
viewMode: false,
|
|
icon: UnlockedIcon,
|
|
predicate: (elements, appState) => {
|
|
const selectedElements = getSelectedElements(elements, appState);
|
|
return (
|
|
selectedElements.length === 0 &&
|
|
elements.some((element) => element.locked)
|
|
);
|
|
},
|
|
perform: (elements, appState) => {
|
|
const lockedElements = elements.filter((el) => el.locked);
|
|
|
|
return {
|
|
elements: elements.map((element) => {
|
|
if (element.locked) {
|
|
return newElementWith(element, { locked: false });
|
|
}
|
|
return element;
|
|
}),
|
|
appState: {
|
|
...appState,
|
|
selectedElementIds: Object.fromEntries(
|
|
lockedElements.map((el) => [el.id, true]),
|
|
),
|
|
},
|
|
captureUpdate: CaptureUpdateAction.IMMEDIATELY,
|
|
};
|
|
},
|
|
label: "labels.elementLock.unlockAll",
|
|
});
|