You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
106 lines
3.1 KiB
TypeScript
106 lines
3.1 KiB
TypeScript
import { newElementWith } from "../element/mutateElement";
|
|
import { isFrameLikeElement } from "../element/typeChecks";
|
|
import { ExcalidrawElement } from "../element/types";
|
|
import { KEYS } from "../keys";
|
|
import { arrayToMap } from "../utils";
|
|
import { register } from "./register";
|
|
|
|
const shouldLock = (elements: readonly ExcalidrawElement[]) =>
|
|
elements.every((el) => !el.locked);
|
|
|
|
export const actionToggleElementLock = register({
|
|
name: "toggleElementLock",
|
|
trackEvent: { category: "element" },
|
|
predicate: (elements, appState, _, app) => {
|
|
const selectedElements = app.scene.getSelectedElements(appState);
|
|
return !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,
|
|
},
|
|
commitToHistory: true,
|
|
};
|
|
},
|
|
contextItemLabel: (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";
|
|
},
|
|
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,
|
|
predicate: (elements) => {
|
|
return 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]),
|
|
),
|
|
},
|
|
commitToHistory: true,
|
|
};
|
|
},
|
|
contextItemLabel: "labels.elementLock.unlockAll",
|
|
});
|