Compare commits

...

1 Commits

Author SHA1 Message Date
dwelle
add575a419 feat: stop storing selection element into appState.draggingElement 2023-10-15 23:01:19 +02:00
8 changed files with 370 additions and 452 deletions

View File

@@ -170,6 +170,7 @@ export const actionFinalize = register({
: activeTool, : activeTool,
activeEmbeddable: null, activeEmbeddable: null,
draggingElement: null, draggingElement: null,
selectionElement: null,
multiElement: null, multiElement: null,
editingElement: null, editingElement: null,
startBoundElement: null, startBoundElement: null,
@@ -196,7 +197,9 @@ export const actionFinalize = register({
keyTest: (event, appState) => keyTest: (event, appState) =>
(event.key === KEYS.ESCAPE && (event.key === KEYS.ESCAPE &&
(appState.editingLinearElement !== null || (appState.editingLinearElement !== null ||
(!appState.draggingElement && appState.multiElement === null))) || (!appState.selectionElement &&
!appState.draggingElement &&
appState.multiElement === null))) ||
((event.key === KEYS.ESCAPE || event.key === KEYS.ENTER) && ((event.key === KEYS.ESCAPE || event.key === KEYS.ENTER) &&
appState.multiElement !== null), appState.multiElement !== null),
PanelComponent: ({ appState, updateData, data }) => ( PanelComponent: ({ appState, updateData, data }) => (

View File

@@ -21,6 +21,7 @@ const writeData = (
!appState.multiElement && !appState.multiElement &&
!appState.resizingElement && !appState.resizingElement &&
!appState.editingElement && !appState.editingElement &&
!appState.selectionElement &&
!appState.draggingElement !appState.draggingElement
) { ) {
const data = updater(); const data = updater();

View File

@@ -3014,7 +3014,8 @@ class App extends React.Component<AppProps, AppState> {
!event.ctrlKey && !event.ctrlKey &&
!event.altKey && !event.altKey &&
!event.metaKey && !event.metaKey &&
this.state.draggingElement === null !this.state.draggingElement &&
!this.state.selectionElement
) { ) {
const shape = findShapeByKey(event.key); const shape = findShapeByKey(event.key);
if (shape) { if (shape) {
@@ -3343,6 +3344,7 @@ class App extends React.Component<AppProps, AppState> {
this.setState({ this.setState({
draggingElement: null, draggingElement: null,
selectionElement: null,
editingElement: null, editingElement: null,
}); });
if (this.state.activeTool.locked) { if (this.state.activeTool.locked) {
@@ -4421,8 +4423,7 @@ class App extends React.Component<AppProps, AppState> {
// finger is lifted // finger is lifted
if ( if (
event.pointerType === "touch" && event.pointerType === "touch" &&
this.state.draggingElement && this.state.draggingElement?.type === "freedraw"
this.state.draggingElement.type === "freedraw"
) { ) {
const element = this.state.draggingElement as ExcalidrawFreeDrawElement; const element = this.state.draggingElement as ExcalidrawFreeDrawElement;
this.updateScene({ this.updateScene({
@@ -4434,6 +4435,7 @@ class App extends React.Component<AppProps, AppState> {
} }
: {}), : {}),
appState: { appState: {
selectionElement: null,
draggingElement: null, draggingElement: null,
editingElement: null, editingElement: null,
startBoundElement: null, startBoundElement: null,
@@ -4561,13 +4563,16 @@ class App extends React.Component<AppProps, AppState> {
// retrieve the latest element as the state may be stale // retrieve the latest element as the state may be stale
const pendingImageElement = const pendingImageElement =
this.state.pendingImageElementId && this.state.pendingImageElementId &&
this.scene.getElement(this.state.pendingImageElementId); this.scene.getElement<ExcalidrawImageElement>(
this.state.pendingImageElementId,
);
if (!pendingImageElement) { if (!pendingImageElement) {
return; return;
} }
this.setState({ this.setState({
selectionElement: null,
draggingElement: pendingImageElement, draggingElement: pendingImageElement,
editingElement: pendingImageElement, editingElement: pendingImageElement,
pendingImageElementId: null, pendingImageElementId: null,
@@ -5374,6 +5379,7 @@ class App extends React.Component<AppProps, AppState> {
); );
this.scene.addNewElement(element); this.scene.addNewElement(element);
this.setState({ this.setState({
selectionElement: null,
draggingElement: element, draggingElement: element,
editingElement: element, editingElement: element,
startBoundElement: boundElement, startBoundElement: boundElement,
@@ -5593,6 +5599,7 @@ class App extends React.Component<AppProps, AppState> {
this.scene.addNewElement(element); this.scene.addNewElement(element);
this.setState({ this.setState({
selectionElement: null,
draggingElement: element, draggingElement: element,
editingElement: element, editingElement: element,
startBoundElement: boundElement, startBoundElement: boundElement,
@@ -5667,12 +5674,13 @@ class App extends React.Component<AppProps, AppState> {
if (element.type === "selection") { if (element.type === "selection") {
this.setState({ this.setState({
selectionElement: element, selectionElement: element,
draggingElement: element, draggingElement: null,
}); });
} else { } else {
this.scene.addNewElement(element); this.scene.addNewElement(element);
this.setState({ this.setState({
multiElement: null, multiElement: null,
selectionElement: null,
draggingElement: element, draggingElement: element,
editingElement: element, editingElement: element,
}); });
@@ -5705,6 +5713,7 @@ class App extends React.Component<AppProps, AppState> {
this.setState({ this.setState({
multiElement: null, multiElement: null,
selectionElement: null,
draggingElement: frame, draggingElement: frame,
editingElement: frame, editingElement: frame,
}); });
@@ -5763,7 +5772,9 @@ class App extends React.Component<AppProps, AppState> {
if (this.maybeHandleResize(pointerDownState, event)) { if (this.maybeHandleResize(pointerDownState, event)) {
return; return;
} }
if (!this.maybeUpdateSelectionElement(pointerDownState, event)) {
this.maybeDragNewGenericElement(pointerDownState, event); this.maybeDragNewGenericElement(pointerDownState, event);
}
}); });
} }
@@ -5776,7 +5787,9 @@ class App extends React.Component<AppProps, AppState> {
if (this.maybeHandleResize(pointerDownState, event)) { if (this.maybeHandleResize(pointerDownState, event)) {
return; return;
} }
if (!this.maybeUpdateSelectionElement(pointerDownState, event)) {
this.maybeDragNewGenericElement(pointerDownState, event); this.maybeDragNewGenericElement(pointerDownState, event);
}
}); });
} }
@@ -6132,6 +6145,13 @@ class App extends React.Component<AppProps, AppState> {
} }
} }
pointerDownState.lastCoords.x = pointerCoords.x;
pointerDownState.lastCoords.y = pointerCoords.y;
if (this.maybeHandleBoxSelection(pointerDownState, event)) {
return;
}
// It is very important to read this.state within each move event, // It is very important to read this.state within each move event,
// otherwise we would read a stale one! // otherwise we would read a stale one!
const draggingElement = this.state.draggingElement; const draggingElement = this.state.draggingElement;
@@ -6199,105 +6219,6 @@ class App extends React.Component<AppProps, AppState> {
pointerDownState.lastCoords.y = pointerCoords.y; pointerDownState.lastCoords.y = pointerCoords.y;
this.maybeDragNewGenericElement(pointerDownState, event); this.maybeDragNewGenericElement(pointerDownState, event);
} }
if (this.state.activeTool.type === "selection") {
pointerDownState.boxSelection.hasOccurred = true;
const elements = this.scene.getNonDeletedElements();
// box-select line editor points
if (this.state.editingLinearElement) {
LinearElementEditor.handleBoxSelection(
event,
this.state,
this.setState.bind(this),
);
// regular box-select
} else {
let shouldReuseSelection = true;
if (!event.shiftKey && isSomeElementSelected(elements, this.state)) {
if (
pointerDownState.withCmdOrCtrl &&
pointerDownState.hit.element
) {
this.setState((prevState) =>
selectGroupsForSelectedElements(
{
...prevState,
selectedElementIds: {
[pointerDownState.hit.element!.id]: true,
},
},
this.scene.getNonDeletedElements(),
prevState,
this,
),
);
} else {
shouldReuseSelection = false;
}
}
const elementsWithinSelection = getElementsWithinSelection(
elements,
draggingElement,
);
this.setState((prevState) => {
const nextSelectedElementIds = {
...(shouldReuseSelection && prevState.selectedElementIds),
...elementsWithinSelection.reduce(
(acc: Record<ExcalidrawElement["id"], true>, element) => {
acc[element.id] = true;
return acc;
},
{},
),
};
if (pointerDownState.hit.element) {
// if using ctrl/cmd, select the hitElement only if we
// haven't box-selected anything else
if (!elementsWithinSelection.length) {
nextSelectedElementIds[pointerDownState.hit.element.id] = true;
} else {
delete nextSelectedElementIds[pointerDownState.hit.element.id];
}
}
prevState = !shouldReuseSelection
? { ...prevState, selectedGroupIds: {}, editingGroupId: null }
: prevState;
return {
...selectGroupsForSelectedElements(
{
editingGroupId: prevState.editingGroupId,
selectedElementIds: nextSelectedElementIds,
},
this.scene.getNonDeletedElements(),
prevState,
this,
),
// select linear element only when we haven't box-selected anything else
selectedLinearElement:
elementsWithinSelection.length === 1 &&
isLinearElement(elementsWithinSelection[0])
? new LinearElementEditor(
elementsWithinSelection[0],
this.scene,
)
: null,
showHyperlinkPopup:
elementsWithinSelection.length === 1 &&
(elementsWithinSelection[0].link ||
isEmbeddableElement(elementsWithinSelection[0]))
? "info"
: false,
};
});
}
}
}); });
} }
@@ -6558,6 +6479,7 @@ class App extends React.Component<AppProps, AppState> {
resetCursor(this.interactiveCanvas); resetCursor(this.interactiveCanvas);
this.setState((prevState) => ({ this.setState((prevState) => ({
draggingElement: null, draggingElement: null,
selectionElement: null,
activeTool: updateActiveTool(this.state, { activeTool: updateActiveTool(this.state, {
type: "selection", type: "selection",
}), }),
@@ -6576,6 +6498,7 @@ class App extends React.Component<AppProps, AppState> {
} else { } else {
this.setState((prevState) => ({ this.setState((prevState) => ({
draggingElement: null, draggingElement: null,
selectionElement: null,
})); }));
} }
} }
@@ -6595,16 +6518,13 @@ class App extends React.Component<AppProps, AppState> {
); );
this.setState({ this.setState({
draggingElement: null, draggingElement: null,
selectionElement: null,
}); });
return; return;
} }
if (draggingElement) {
if (pointerDownState.drag.hasOccurred) { if (pointerDownState.drag.hasOccurred) {
const sceneCoords = viewportCoordsToSceneCoords( const sceneCoords = viewportCoordsToSceneCoords(childEvent, this.state);
childEvent,
this.state,
);
// when editing the points of a linear element, we check if the // when editing the points of a linear element, we check if the
// linear element still is in the frame afterwards // linear element still is in the frame afterwards
@@ -6640,8 +6560,7 @@ class App extends React.Component<AppProps, AppState> {
} }
} else { } else {
// update the relationships between selected elements and frames // update the relationships between selected elements and frames
const topLayerFrame = const topLayerFrame = this.getTopLayerFrameAtSceneCoords(sceneCoords);
this.getTopLayerFrameAtSceneCoords(sceneCoords);
const selectedElements = this.scene.getSelectedElements(this.state); const selectedElements = this.scene.getSelectedElements(this.state);
let nextElements = this.scene.getElementsIncludingDeleted(); let nextElements = this.scene.getElementsIncludingDeleted();
@@ -6729,6 +6648,7 @@ class App extends React.Component<AppProps, AppState> {
} }
} }
if (draggingElement) {
if (draggingElement.type === "frame") { if (draggingElement.type === "frame") {
const elementsInsideFrame = getElementsInNewFrame( const elementsInsideFrame = getElementsInNewFrame(
this.scene.getElementsIncludingDeleted(), this.scene.getElementsIncludingDeleted(),
@@ -7032,8 +6952,7 @@ class App extends React.Component<AppProps, AppState> {
if ( if (
!activeTool.locked && !activeTool.locked &&
activeTool.type !== "freedraw" && activeTool.type !== "freedraw" &&
draggingElement && draggingElement
draggingElement.type !== "selection"
) { ) {
this.setState((prevState) => ({ this.setState((prevState) => ({
selectedElementIds: makeNextSelectedElementIds( selectedElementIds: makeNextSelectedElementIds(
@@ -7072,12 +6991,14 @@ class App extends React.Component<AppProps, AppState> {
resetCursor(this.interactiveCanvas); resetCursor(this.interactiveCanvas);
this.setState({ this.setState({
draggingElement: null, draggingElement: null,
selectionElement: null,
suggestedBindings: [], suggestedBindings: [],
activeTool: updateActiveTool(this.state, { type: "selection" }), activeTool: updateActiveTool(this.state, { type: "selection" }),
}); });
} else { } else {
this.setState({ this.setState({
draggingElement: null, draggingElement: null,
selectionElement: null,
suggestedBindings: [], suggestedBindings: [],
}); });
} }
@@ -7876,21 +7797,20 @@ class App extends React.Component<AppProps, AppState> {
); );
}; };
private maybeDragNewGenericElement = ( private maybeUpdateSelectionElement = (
pointerDownState: PointerDownState, pointerDownState: PointerDownState,
event: MouseEvent | KeyboardEvent, event: PointerEvent | KeyboardEvent,
): void => { ): boolean => {
const draggingElement = this.state.draggingElement; const { selectionElement } = this.state;
const pointerCoords = pointerDownState.lastCoords;
if (!draggingElement) { if (!selectionElement || this.state.activeTool.type !== "selection") {
return; return false;
} }
if (
draggingElement.type === "selection" && const pointerCoords = pointerDownState.lastCoords;
this.state.activeTool.type !== "eraser"
) {
dragNewElement( dragNewElement(
draggingElement, selectionElement,
this.state.activeTool.type, this.state.activeTool.type,
pointerDownState.origin.x, pointerDownState.origin.x,
pointerDownState.origin.y, pointerDownState.origin.y,
@@ -7901,7 +7821,124 @@ class App extends React.Component<AppProps, AppState> {
shouldMaintainAspectRatio(event), shouldMaintainAspectRatio(event),
shouldResizeFromCenter(event), shouldResizeFromCenter(event),
); );
return true;
};
private maybeHandleBoxSelection = (
pointerDownState: PointerDownState,
event: PointerEvent,
): boolean => {
const { selectionElement } = this.state;
if (!selectionElement || this.state.activeTool.type !== "selection") {
return false;
}
this.maybeUpdateSelectionElement(pointerDownState, event);
pointerDownState.boxSelection.hasOccurred = true;
const elements = this.scene.getNonDeletedElements();
// box-select line editor points
if (this.state.editingLinearElement) {
LinearElementEditor.handleBoxSelection(
event,
this.state,
this.setState.bind(this),
);
// regular box-select
} else { } else {
let shouldReuseSelection = true;
if (!event.shiftKey && isSomeElementSelected(elements, this.state)) {
if (pointerDownState.withCmdOrCtrl && pointerDownState.hit.element) {
this.setState((prevState) =>
selectGroupsForSelectedElements(
{
...prevState,
selectedElementIds: {
[pointerDownState.hit.element!.id]: true,
},
},
this.scene.getNonDeletedElements(),
prevState,
this,
),
);
} else {
shouldReuseSelection = false;
}
}
const elementsWithinSelection = getElementsWithinSelection(
elements,
selectionElement,
);
this.setState((prevState) => {
const nextSelectedElementIds = {
...(shouldReuseSelection && prevState.selectedElementIds),
...elementsWithinSelection.reduce(
(acc: Record<ExcalidrawElement["id"], true>, element) => {
acc[element.id] = true;
return acc;
},
{},
),
};
if (pointerDownState.hit.element) {
// if using ctrl/cmd, select the hitElement only if we
// haven't box-selected anything else
if (!elementsWithinSelection.length) {
nextSelectedElementIds[pointerDownState.hit.element.id] = true;
} else {
delete nextSelectedElementIds[pointerDownState.hit.element.id];
}
}
prevState = !shouldReuseSelection
? { ...prevState, selectedGroupIds: {}, editingGroupId: null }
: prevState;
return {
...selectGroupsForSelectedElements(
{
editingGroupId: prevState.editingGroupId,
selectedElementIds: nextSelectedElementIds,
},
this.scene.getNonDeletedElements(),
prevState,
this,
),
// select linear element only when we haven't box-selected anything else
selectedLinearElement:
elementsWithinSelection.length === 1 &&
isLinearElement(elementsWithinSelection[0])
? new LinearElementEditor(elementsWithinSelection[0], this.scene)
: null,
showHyperlinkPopup:
elementsWithinSelection.length === 1 &&
(elementsWithinSelection[0].link ||
isEmbeddableElement(elementsWithinSelection[0]))
? "info"
: false,
};
});
}
return true;
};
private maybeDragNewGenericElement = (
pointerDownState: PointerDownState,
event: MouseEvent | KeyboardEvent,
): void => {
const draggingElement = this.state.draggingElement;
const pointerCoords = pointerDownState.lastCoords;
if (!draggingElement) {
return;
}
let [gridX, gridY] = getGridPoint( let [gridX, gridY] = getGridPoint(
pointerCoords.x, pointerCoords.x,
pointerCoords.y, pointerCoords.y,
@@ -7912,9 +7949,7 @@ class App extends React.Component<AppProps, AppState> {
isInitializedImageElement(draggingElement) && isInitializedImageElement(draggingElement) &&
this.imageCache.get(draggingElement.fileId)?.image; this.imageCache.get(draggingElement.fileId)?.image;
const aspectRatio = const aspectRatio =
image && !(image instanceof Promise) image && !(image instanceof Promise) ? image.width / image.height : null;
? image.width / image.height
: null;
this.maybeCacheReferenceSnapPoints(event, [draggingElement]); this.maybeCacheReferenceSnapPoints(event, [draggingElement]);
@@ -7972,7 +8007,6 @@ class App extends React.Component<AppProps, AppState> {
), ),
}); });
} }
}
}; };
private maybeHandleResize = ( private maybeHandleResize = (

View File

@@ -82,8 +82,9 @@ const getHints = ({ appState, isMobile, device, app }: HintViewerProps) => {
if (activeTool.type === "selection") { if (activeTool.type === "selection") {
if ( if (
appState.draggingElement?.type === "selection" && appState.selectionElement &&
!selectedElements.length && !selectedElements.length &&
!appState.draggingElement &&
!appState.editingElement && !appState.editingElement &&
!appState.editingLinearElement !appState.editingLinearElement
) { ) {

View File

@@ -210,6 +210,7 @@ export const Hyperlink = ({
}; };
const { x, y } = getCoordsForPopover(element, appState); const { x, y } = getCoordsForPopover(element, appState);
if ( if (
appState.selectionElement ||
appState.draggingElement || appState.draggingElement ||
appState.resizingElement || appState.resizingElement ||
appState.isRotating || appState.isRotating ||

View File

@@ -134,10 +134,7 @@ export class LinearElementEditor {
appState: AppState, appState: AppState,
setState: React.Component<any, AppState>["setState"], setState: React.Component<any, AppState>["setState"],
) { ) {
if ( if (!appState.editingLinearElement || !appState.selectionElement) {
!appState.editingLinearElement ||
appState.draggingElement?.type !== "selection"
) {
return false; return false;
} }
const { editingLinearElement } = appState; const { editingLinearElement } = appState;
@@ -149,7 +146,7 @@ export class LinearElementEditor {
} }
const [selectionX1, selectionY1, selectionX2, selectionY2] = const [selectionX1, selectionY1, selectionX2, selectionY2] =
getElementAbsoluteCoords(appState.draggingElement); getElementAbsoluteCoords(appState.selectionElement);
const pointsSceneCoords = const pointsSceneCoords =
LinearElementEditor.getPointsGlobalCoordinates(element); LinearElementEditor.getPointsGlobalCoordinates(element);

View File

@@ -5152,35 +5152,7 @@ exports[`regression tests > deselects group of selected elements on pointer down
"currentItemTextAlign": "left", "currentItemTextAlign": "left",
"cursorButton": "down", "cursorButton": "down",
"defaultSidebarDockedPreference": false, "defaultSidebarDockedPreference": false,
"draggingElement": { "draggingElement": null,
"angle": 0,
"backgroundColor": "transparent",
"boundElements": null,
"fillStyle": "hachure",
"frameId": null,
"groupIds": [],
"height": 0,
"id": "id3",
"isDeleted": false,
"link": null,
"locked": false,
"opacity": 100,
"roughness": 1,
"roundness": {
"type": 2,
},
"seed": 400692809,
"strokeColor": "#1e1e1e",
"strokeStyle": "solid",
"strokeWidth": 1,
"type": "selection",
"updated": 1,
"version": 1,
"versionNonce": 0,
"width": 0,
"x": 500,
"y": 500,
},
"editingElement": null, "editingElement": null,
"editingFrame": null, "editingFrame": null,
"editingGroupId": null, "editingGroupId": null,
@@ -5449,35 +5421,7 @@ exports[`regression tests > deselects group of selected elements on pointer up w
"currentItemTextAlign": "left", "currentItemTextAlign": "left",
"cursorButton": "up", "cursorButton": "up",
"defaultSidebarDockedPreference": false, "defaultSidebarDockedPreference": false,
"draggingElement": { "draggingElement": null,
"angle": 0,
"backgroundColor": "transparent",
"boundElements": null,
"fillStyle": "hachure",
"frameId": null,
"groupIds": [],
"height": 0,
"id": "id3",
"isDeleted": false,
"link": null,
"locked": false,
"opacity": 100,
"roughness": 1,
"roundness": {
"type": 2,
},
"seed": 400692809,
"strokeColor": "#1e1e1e",
"strokeStyle": "solid",
"strokeWidth": 1,
"type": "selection",
"updated": 1,
"version": 1,
"versionNonce": 0,
"width": 0,
"x": 50,
"y": 50,
},
"editingElement": null, "editingElement": null,
"editingFrame": null, "editingFrame": null,
"editingGroupId": null, "editingGroupId": null,
@@ -5718,35 +5662,7 @@ exports[`regression tests > deselects selected element on pointer down when poin
"currentItemTextAlign": "left", "currentItemTextAlign": "left",
"cursorButton": "down", "cursorButton": "down",
"defaultSidebarDockedPreference": false, "defaultSidebarDockedPreference": false,
"draggingElement": { "draggingElement": null,
"angle": 0,
"backgroundColor": "transparent",
"boundElements": null,
"fillStyle": "hachure",
"frameId": null,
"groupIds": [],
"height": 0,
"id": "id1",
"isDeleted": false,
"link": null,
"locked": false,
"opacity": 100,
"roughness": 1,
"roundness": {
"type": 2,
},
"seed": 2019559783,
"strokeColor": "#1e1e1e",
"strokeStyle": "solid",
"strokeWidth": 1,
"type": "selection",
"updated": 1,
"version": 1,
"versionNonce": 0,
"width": 0,
"x": 110,
"y": 110,
},
"editingElement": null, "editingElement": null,
"editingFrame": null, "editingFrame": null,
"editingGroupId": null, "editingGroupId": null,
@@ -16262,35 +16178,7 @@ exports[`regression tests > switches from group of selected elements to another
"currentItemTextAlign": "left", "currentItemTextAlign": "left",
"cursorButton": "down", "cursorButton": "down",
"defaultSidebarDockedPreference": false, "defaultSidebarDockedPreference": false,
"draggingElement": { "draggingElement": null,
"angle": 0,
"backgroundColor": "transparent",
"boundElements": null,
"fillStyle": "hachure",
"frameId": null,
"groupIds": [],
"height": 0,
"id": "id4",
"isDeleted": false,
"link": null,
"locked": false,
"opacity": 100,
"roughness": 1,
"roundness": {
"type": 2,
},
"seed": 493213705,
"strokeColor": "#1e1e1e",
"strokeStyle": "solid",
"strokeWidth": 1,
"type": "selection",
"updated": 1,
"version": 1,
"versionNonce": 0,
"width": 0,
"x": 0,
"y": 0,
},
"editingElement": null, "editingElement": null,
"editingFrame": null, "editingFrame": null,
"editingGroupId": null, "editingGroupId": null,
@@ -16662,35 +16550,7 @@ exports[`regression tests > switches selected element on pointer down > [end of
"currentItemTextAlign": "left", "currentItemTextAlign": "left",
"cursorButton": "down", "cursorButton": "down",
"defaultSidebarDockedPreference": false, "defaultSidebarDockedPreference": false,
"draggingElement": { "draggingElement": null,
"angle": 0,
"backgroundColor": "transparent",
"boundElements": null,
"fillStyle": "hachure",
"frameId": null,
"groupIds": [],
"height": 0,
"id": "id2",
"isDeleted": false,
"link": null,
"locked": false,
"opacity": 100,
"roughness": 1,
"roundness": {
"type": 2,
},
"seed": 238820263,
"strokeColor": "#1e1e1e",
"strokeStyle": "solid",
"strokeWidth": 1,
"type": "selection",
"updated": 1,
"version": 1,
"versionNonce": 0,
"width": 0,
"x": 0,
"y": 0,
},
"editingElement": null, "editingElement": null,
"editingFrame": null, "editingFrame": null,
"editingGroupId": null, "editingGroupId": null,

View File

@@ -17,6 +17,7 @@ import {
StrokeRoundness, StrokeRoundness,
ExcalidrawFrameElement, ExcalidrawFrameElement,
ExcalidrawEmbeddableElement, ExcalidrawEmbeddableElement,
ExcalidrawSelectionElement,
} from "./element/types"; } from "./element/types";
import { Point as RoughPoint } from "roughjs/bin/geometry"; import { Point as RoughPoint } from "roughjs/bin/geometry";
import { LinearElementEditor } from "./element/linearElementEditor"; import { LinearElementEditor } from "./element/linearElementEditor";
@@ -182,10 +183,25 @@ export type AppState = {
element: NonDeletedExcalidrawElement; element: NonDeletedExcalidrawElement;
state: "hover" | "active"; state: "hover" | "active";
} | null; } | null;
draggingElement: NonDeletedExcalidrawElement | null; /** element that's being dragged or created */
draggingElement: Exclude<
NonDeletedExcalidrawElement,
ExcalidrawSelectionElement
> | null;
/**
* Element that's being resized.
* NOTE not set when resizing a group or linear element
*/
resizingElement: NonDeletedExcalidrawElement | null; resizingElement: NonDeletedExcalidrawElement | null;
/** multi-point linear element when it's being created */
multiElement: NonDeleted<ExcalidrawLinearElement> | null; multiElement: NonDeleted<ExcalidrawLinearElement> | null;
selectionElement: NonDeletedExcalidrawElement | null; /**
* The selection box (we currently use an excalidraw element).
*
* Checking for this attribute is a good way to determine whether the user is
* selecting.
*/
selectionElement: ExcalidrawSelectionElement | null;
isBindingEnabled: boolean; isBindingEnabled: boolean;
startBoundElement: NonDeleted<ExcalidrawBindableElement> | null; startBoundElement: NonDeleted<ExcalidrawBindableElement> | null;
suggestedBindings: SuggestedBinding[]; suggestedBindings: SuggestedBinding[];
@@ -198,9 +214,14 @@ export type AppState = {
}; };
editingFrame: string | null; editingFrame: string | null;
elementsToHighlight: NonDeleted<ExcalidrawElement>[] | null; elementsToHighlight: NonDeleted<ExcalidrawElement>[] | null;
// element being edited, but not necessarily added to elements array yet /**
// (e.g. text element when typing into the input) * Text that's being element, or new element being created.
*/
editingElement: NonDeletedExcalidrawElement | null; editingElement: NonDeletedExcalidrawElement | null;
/**
* Linear element that's being edited (when in the linear element editor).
* Not set when creating multi-point linear element.
*/
editingLinearElement: LinearElementEditor | null; editingLinearElement: LinearElementEditor | null;
activeTool: { activeTool: {
/** /**