2020-05-26 13:07:46 -07:00
|
|
|
import { KEYS } from "../keys";
|
|
|
|
import { register } from "./register";
|
|
|
|
import { newElementWith } from "../element/mutateElement";
|
|
|
|
import { getSelectedElements } from "../scene";
|
|
|
|
import {
|
|
|
|
getSelectedGroupIds,
|
|
|
|
selectGroup,
|
|
|
|
selectGroupsForSelectedElements,
|
|
|
|
getElementsInGroup,
|
|
|
|
addToGroup,
|
|
|
|
removeFromSelectedGroups,
|
2020-05-26 13:56:22 -07:00
|
|
|
isElementInGroup,
|
2020-05-26 13:07:46 -07:00
|
|
|
} from "../groups";
|
|
|
|
import { getNonDeletedElements } from "../element";
|
2020-05-28 01:56:18 -07:00
|
|
|
import { randomId } from "../random";
|
2020-05-26 13:07:46 -07:00
|
|
|
|
|
|
|
export const actionGroup = register({
|
|
|
|
name: "group",
|
|
|
|
perform: (elements, appState) => {
|
|
|
|
const selectedElements = getSelectedElements(
|
|
|
|
getNonDeletedElements(elements),
|
|
|
|
appState,
|
|
|
|
);
|
|
|
|
if (selectedElements.length < 2) {
|
|
|
|
// nothing to group
|
|
|
|
return { appState, elements, commitToHistory: false };
|
|
|
|
}
|
|
|
|
// if everything is already grouped into 1 group, there is nothing to do
|
|
|
|
const selectedGroupIds = getSelectedGroupIds(appState);
|
|
|
|
if (selectedGroupIds.length === 1) {
|
|
|
|
const selectedGroupId = selectedGroupIds[0];
|
|
|
|
const elementIdsInGroup = new Set(
|
|
|
|
getElementsInGroup(elements, selectedGroupId).map(
|
|
|
|
(element) => element.id,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
const selectedElementIds = new Set(
|
|
|
|
selectedElements.map((element) => element.id),
|
|
|
|
);
|
|
|
|
const combinedSet = new Set([
|
|
|
|
...Array.from(elementIdsInGroup),
|
|
|
|
...Array.from(selectedElementIds),
|
|
|
|
]);
|
|
|
|
if (combinedSet.size === elementIdsInGroup.size) {
|
|
|
|
// no incremental ids in the selected ids
|
|
|
|
return { appState, elements, commitToHistory: false };
|
|
|
|
}
|
|
|
|
}
|
2020-05-28 01:56:18 -07:00
|
|
|
const newGroupId = randomId();
|
2020-05-26 13:07:46 -07:00
|
|
|
const updatedElements = elements.map((element) => {
|
|
|
|
if (!appState.selectedElementIds[element.id]) {
|
|
|
|
return element;
|
|
|
|
}
|
|
|
|
return newElementWith(element, {
|
|
|
|
groupIds: addToGroup(
|
|
|
|
element.groupIds,
|
|
|
|
newGroupId,
|
|
|
|
appState.editingGroupId,
|
|
|
|
),
|
|
|
|
});
|
|
|
|
});
|
2020-05-26 13:56:22 -07:00
|
|
|
// keep the z order within the group the same, but move them
|
|
|
|
// to the z order of the highest element in the layer stack
|
|
|
|
const elementsInGroup = getElementsInGroup(updatedElements, newGroupId);
|
|
|
|
const lastElementInGroup = elementsInGroup[elementsInGroup.length - 1];
|
|
|
|
const lastGroupElementIndex = updatedElements.lastIndexOf(
|
|
|
|
lastElementInGroup,
|
|
|
|
);
|
|
|
|
const elementsAfterGroup = updatedElements.slice(lastGroupElementIndex + 1);
|
|
|
|
const elementsBeforeGroup = updatedElements
|
|
|
|
.slice(0, lastGroupElementIndex)
|
|
|
|
.filter(
|
|
|
|
(updatedElement) => !isElementInGroup(updatedElement, newGroupId),
|
|
|
|
);
|
|
|
|
const updatedElementsInOrder = [
|
|
|
|
...elementsBeforeGroup,
|
|
|
|
...elementsInGroup,
|
|
|
|
...elementsAfterGroup,
|
|
|
|
];
|
|
|
|
|
2020-05-26 13:07:46 -07:00
|
|
|
return {
|
|
|
|
appState: selectGroup(
|
|
|
|
newGroupId,
|
|
|
|
{ ...appState, selectedGroupIds: {} },
|
2020-05-26 13:56:22 -07:00
|
|
|
getNonDeletedElements(updatedElementsInOrder),
|
2020-05-26 13:07:46 -07:00
|
|
|
),
|
2020-05-26 13:56:22 -07:00
|
|
|
elements: updatedElementsInOrder,
|
2020-05-26 13:07:46 -07:00
|
|
|
commitToHistory: true,
|
|
|
|
};
|
|
|
|
},
|
|
|
|
contextMenuOrder: 4,
|
|
|
|
contextItemLabel: "labels.group",
|
|
|
|
keyTest: (event) => {
|
|
|
|
return (
|
|
|
|
!event.shiftKey &&
|
|
|
|
event[KEYS.CTRL_OR_CMD] &&
|
|
|
|
event.keyCode === KEYS.G_KEY_CODE
|
|
|
|
);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
export const actionUngroup = register({
|
|
|
|
name: "ungroup",
|
|
|
|
perform: (elements, appState) => {
|
|
|
|
const groupIds = getSelectedGroupIds(appState);
|
|
|
|
if (groupIds.length === 0) {
|
|
|
|
return { appState, elements, commitToHistory: false };
|
|
|
|
}
|
|
|
|
const nextElements = elements.map((element) => {
|
|
|
|
const nextGroupIds = removeFromSelectedGroups(
|
|
|
|
element.groupIds,
|
|
|
|
appState.selectedGroupIds,
|
|
|
|
);
|
|
|
|
if (nextGroupIds.length === element.groupIds.length) {
|
|
|
|
return element;
|
|
|
|
}
|
|
|
|
return newElementWith(element, {
|
|
|
|
groupIds: nextGroupIds,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
return {
|
|
|
|
appState: selectGroupsForSelectedElements(
|
|
|
|
{ ...appState, selectedGroupIds: {} },
|
|
|
|
getNonDeletedElements(nextElements),
|
|
|
|
),
|
|
|
|
elements: nextElements,
|
|
|
|
commitToHistory: true,
|
|
|
|
};
|
|
|
|
},
|
|
|
|
keyTest: (event) => {
|
|
|
|
return (
|
|
|
|
event.shiftKey &&
|
|
|
|
event[KEYS.CTRL_OR_CMD] &&
|
|
|
|
event.keyCode === KEYS.G_KEY_CODE
|
|
|
|
);
|
|
|
|
},
|
|
|
|
contextMenuOrder: 5,
|
|
|
|
contextItemLabel: "labels.ungroup",
|
|
|
|
});
|