Refactoring in pointer down event handler, step 2 (#1887)
* Refactor: introduce pointer down state to replace implicit closure state with an explicit object * Refactor: use pointer down state for resize handle * Refactor: use pointer down state for isResizing * Refactor: use pointer down state for resizing offset * Refactor: use pointer down state for hit element * Refactor: move selection handling out of pointer down event handler * Refactor: move text handling out of pointer down event handler * Refactor: move linear tools handling out of pointer down event handler * Refactor: move element creation out of pointer down handler
This commit is contained in:
parent
6cc6e13892
commit
5664de0459
@ -61,6 +61,7 @@ import {
|
|||||||
ExcalidrawProps,
|
ExcalidrawProps,
|
||||||
ExcalidrawTextElement,
|
ExcalidrawTextElement,
|
||||||
NonDeleted,
|
NonDeleted,
|
||||||
|
ExcalidrawGenericElement,
|
||||||
} from "../element/types";
|
} from "../element/types";
|
||||||
|
|
||||||
import { distance2d, isPathALoop, getGridPoint } from "../math";
|
import { distance2d, isPathALoop, getGridPoint } from "../math";
|
||||||
@ -75,6 +76,7 @@ import {
|
|||||||
viewportCoordsToSceneCoords,
|
viewportCoordsToSceneCoords,
|
||||||
sceneCoordsToViewportCoords,
|
sceneCoordsToViewportCoords,
|
||||||
setCursorForShape,
|
setCursorForShape,
|
||||||
|
tupleToCoors,
|
||||||
} from "../utils";
|
} from "../utils";
|
||||||
import {
|
import {
|
||||||
KEYS,
|
KEYS,
|
||||||
@ -186,6 +188,30 @@ const gesture: Gesture = {
|
|||||||
initialScale: null,
|
initialScale: null,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
type PointerDownState = Readonly<{
|
||||||
|
// The first position at which pointerDown happened
|
||||||
|
origin: Readonly<{ x: number; y: number }>;
|
||||||
|
// The previous pointer position
|
||||||
|
lastCoords: { x: number; y: number };
|
||||||
|
resize: {
|
||||||
|
// Handle when resizing, might change during the pointer interaction
|
||||||
|
handle: ReturnType<typeof resizeTest>;
|
||||||
|
// This is determined on the initial pointer down event
|
||||||
|
isResizing: boolean;
|
||||||
|
// This is determined on the initial pointer down event
|
||||||
|
offset: { x: number; y: number };
|
||||||
|
// This is determined on the initial pointer down event
|
||||||
|
arrowDirection: "origin" | "end";
|
||||||
|
};
|
||||||
|
hit: {
|
||||||
|
// The element the pointer is "hitting", is determined on the initial
|
||||||
|
// pointer down event
|
||||||
|
element: ExcalidrawElement | null;
|
||||||
|
// This is determined on the initial pointer down event
|
||||||
|
wasAddedToSelection: boolean;
|
||||||
|
};
|
||||||
|
}>;
|
||||||
|
|
||||||
class App extends React.Component<ExcalidrawProps, AppState> {
|
class App extends React.Component<ExcalidrawProps, AppState> {
|
||||||
canvas: HTMLCanvasElement | null = null;
|
canvas: HTMLCanvasElement | null = null;
|
||||||
rc: RoughCanvas | null = null;
|
rc: RoughCanvas | null = null;
|
||||||
@ -1952,301 +1978,70 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
isOverVerticalScrollBar,
|
isOverVerticalScrollBar,
|
||||||
} = isOverScrollBarsNow;
|
} = isOverScrollBarsNow;
|
||||||
|
|
||||||
const { x, y } = viewportCoordsToSceneCoords(
|
const origin = viewportCoordsToSceneCoords(
|
||||||
event,
|
event,
|
||||||
this.state,
|
this.state,
|
||||||
this.canvas,
|
this.canvas,
|
||||||
window.devicePixelRatio,
|
window.devicePixelRatio,
|
||||||
);
|
);
|
||||||
const lastCoords = { x, y };
|
|
||||||
|
|
||||||
if (this.handleDraggingScrollBar(event, lastCoords, isOverScrollBarsNow)) {
|
// State for the duration of a pointer interaction, which starts with a
|
||||||
|
// pointerDown event, ends with a pointerUp event (or another pointerDown)
|
||||||
|
const pointerDownState: PointerDownState = {
|
||||||
|
origin,
|
||||||
|
// we need to duplicate because we'll be updating this state
|
||||||
|
lastCoords: { ...origin },
|
||||||
|
resize: {
|
||||||
|
handle: false as ReturnType<typeof resizeTest>,
|
||||||
|
isResizing: false,
|
||||||
|
offset: { x: 0, y: 0 },
|
||||||
|
arrowDirection: "origin",
|
||||||
|
},
|
||||||
|
hit: {
|
||||||
|
element: null,
|
||||||
|
wasAddedToSelection: false,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
if (
|
||||||
|
this.handleDraggingScrollBar(event, pointerDownState, isOverScrollBarsNow)
|
||||||
|
) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const originX = x;
|
|
||||||
const originY = y;
|
|
||||||
const [originGridX, originGridY] = getGridPoint(
|
const [originGridX, originGridY] = getGridPoint(
|
||||||
originX,
|
pointerDownState.origin.x,
|
||||||
originY,
|
pointerDownState.origin.y,
|
||||||
this.state.gridSize,
|
this.state.gridSize,
|
||||||
);
|
);
|
||||||
|
|
||||||
type ResizeTestType = ReturnType<typeof resizeTest>;
|
|
||||||
let resizeHandle: ResizeTestType = false;
|
|
||||||
const setResizeHandle = (nextResizeHandle: ResizeTestType) => {
|
|
||||||
resizeHandle = nextResizeHandle;
|
|
||||||
};
|
|
||||||
let resizeOffsetXY: [number, number] = [0, 0];
|
|
||||||
let resizeArrowDirection: "origin" | "end" = "origin";
|
|
||||||
let isResizingElements = false;
|
|
||||||
let draggingOccurred = false;
|
let draggingOccurred = false;
|
||||||
let dragOffsetXY: [number, number] | null = null;
|
let dragOffsetXY: [number, number] | null = null;
|
||||||
let hitElement: ExcalidrawElement | null = null;
|
|
||||||
let hitElementWasAddedToSelection = false;
|
|
||||||
|
|
||||||
if (this.state.elementType !== "selection") {
|
this.clearSelectionIfNotUsingSelection();
|
||||||
this.setState({
|
|
||||||
selectedElementIds: {},
|
|
||||||
selectedGroupIds: {},
|
|
||||||
editingGroupId: null,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.state.elementType === "selection") {
|
if (this.handleSelectionOnPointerDown(event, pointerDownState)) {
|
||||||
const elements = globalSceneState.getElements();
|
|
||||||
const selectedElements = getSelectedElements(elements, this.state);
|
|
||||||
if (selectedElements.length === 1 && !this.state.editingLinearElement) {
|
|
||||||
const elementWithResizeHandler = getElementWithResizeHandler(
|
|
||||||
elements,
|
|
||||||
this.state,
|
|
||||||
x,
|
|
||||||
y,
|
|
||||||
this.state.zoom,
|
|
||||||
event.pointerType,
|
|
||||||
);
|
|
||||||
if (elementWithResizeHandler != null) {
|
|
||||||
this.setState({ resizingElement: elementWithResizeHandler.element });
|
|
||||||
resizeHandle = elementWithResizeHandler.resizeHandle;
|
|
||||||
}
|
|
||||||
} else if (selectedElements.length > 1) {
|
|
||||||
resizeHandle = getResizeHandlerFromCoords(
|
|
||||||
getCommonBounds(selectedElements),
|
|
||||||
x,
|
|
||||||
y,
|
|
||||||
this.state.zoom,
|
|
||||||
event.pointerType,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if (resizeHandle) {
|
|
||||||
document.documentElement.style.cursor = getCursorForResizingElement({
|
|
||||||
resizeHandle,
|
|
||||||
});
|
|
||||||
isResizingElements = true;
|
|
||||||
resizeOffsetXY = getResizeOffsetXY(
|
|
||||||
resizeHandle,
|
|
||||||
selectedElements,
|
|
||||||
x,
|
|
||||||
y,
|
|
||||||
);
|
|
||||||
if (
|
|
||||||
selectedElements.length === 1 &&
|
|
||||||
isLinearElement(selectedElements[0]) &&
|
|
||||||
selectedElements[0].points.length === 2
|
|
||||||
) {
|
|
||||||
resizeArrowDirection = getResizeArrowDirection(
|
|
||||||
resizeHandle,
|
|
||||||
selectedElements[0],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (this.state.editingLinearElement) {
|
|
||||||
const ret = LinearElementEditor.handlePointerDown(
|
|
||||||
event,
|
|
||||||
this.state,
|
|
||||||
(appState) => this.setState(appState),
|
|
||||||
history,
|
|
||||||
x,
|
|
||||||
y,
|
|
||||||
);
|
|
||||||
if (ret.hitElement) {
|
|
||||||
hitElement = ret.hitElement;
|
|
||||||
}
|
|
||||||
if (ret.didAddPoint) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// hitElement may already be set above, so check first
|
|
||||||
hitElement =
|
|
||||||
hitElement ||
|
|
||||||
getElementAtPosition(elements, this.state, x, y, this.state.zoom);
|
|
||||||
|
|
||||||
this.maybeClearSelectionWhenHittingElement(event, hitElement);
|
|
||||||
|
|
||||||
// If we click on something
|
|
||||||
if (hitElement) {
|
|
||||||
// deselect if item is selected
|
|
||||||
// if shift is not clicked, this will always return true
|
|
||||||
// otherwise, it will trigger selection based on current
|
|
||||||
// state of the box
|
|
||||||
if (!this.state.selectedElementIds[hitElement.id]) {
|
|
||||||
// if we are currently editing a group, treat all selections outside of the group
|
|
||||||
// as exiting editing mode.
|
|
||||||
if (
|
|
||||||
this.state.editingGroupId &&
|
|
||||||
!isElementInGroup(hitElement, this.state.editingGroupId)
|
|
||||||
) {
|
|
||||||
this.setState({
|
|
||||||
selectedElementIds: {},
|
|
||||||
selectedGroupIds: {},
|
|
||||||
editingGroupId: null,
|
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this.setState((prevState) => {
|
|
||||||
return selectGroupsForSelectedElements(
|
|
||||||
{
|
|
||||||
...prevState,
|
|
||||||
selectedElementIds: {
|
|
||||||
...prevState.selectedElementIds,
|
|
||||||
[hitElement!.id]: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
globalSceneState.getElements(),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
// TODO: this is strange...
|
|
||||||
globalSceneState.replaceAllElements(
|
|
||||||
globalSceneState.getElementsIncludingDeleted(),
|
|
||||||
);
|
|
||||||
hitElementWasAddedToSelection = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const { selectedElementIds } = this.state;
|
|
||||||
this.setState({
|
|
||||||
previousSelectedElementIds: selectedElementIds,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.state.elementType === "text") {
|
if (this.state.elementType === "text") {
|
||||||
// if we're currently still editing text, clicking outside
|
this.handleTextOnPointerDown(event, pointerDownState);
|
||||||
// should only finalize it, not create another (irrespective
|
|
||||||
// of state.elementLocked)
|
|
||||||
if (this.state.editingElement?.type === "text") {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.startTextEditing({
|
|
||||||
sceneX: x,
|
|
||||||
sceneY: y,
|
|
||||||
insertAtParentCenter: !event.altKey,
|
|
||||||
});
|
|
||||||
|
|
||||||
resetCursor();
|
|
||||||
if (!this.state.elementLocked) {
|
|
||||||
this.setState({
|
|
||||||
elementType: "selection",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
return;
|
return;
|
||||||
} else if (
|
} else if (
|
||||||
this.state.elementType === "arrow" ||
|
this.state.elementType === "arrow" ||
|
||||||
this.state.elementType === "draw" ||
|
this.state.elementType === "draw" ||
|
||||||
this.state.elementType === "line"
|
this.state.elementType === "line"
|
||||||
) {
|
) {
|
||||||
if (this.state.multiElement) {
|
this.handleLinearElementOnPointerDown(
|
||||||
const { multiElement } = this.state;
|
event,
|
||||||
|
this.state.elementType,
|
||||||
// finalize if completing a loop
|
pointerDownState,
|
||||||
if (multiElement.type === "line" && isPathALoop(multiElement.points)) {
|
|
||||||
mutateElement(multiElement, {
|
|
||||||
lastCommittedPoint:
|
|
||||||
multiElement.points[multiElement.points.length - 1],
|
|
||||||
});
|
|
||||||
this.actionManager.executeAction(actionFinalize);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { x: rx, y: ry, lastCommittedPoint } = multiElement;
|
|
||||||
|
|
||||||
// clicking inside commit zone → finalize arrow
|
|
||||||
if (
|
|
||||||
multiElement.points.length > 1 &&
|
|
||||||
lastCommittedPoint &&
|
|
||||||
distance2d(
|
|
||||||
x - rx,
|
|
||||||
y - ry,
|
|
||||||
lastCommittedPoint[0],
|
|
||||||
lastCommittedPoint[1],
|
|
||||||
) < LINE_CONFIRM_THRESHOLD
|
|
||||||
) {
|
|
||||||
this.actionManager.executeAction(actionFinalize);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.setState((prevState) => ({
|
|
||||||
selectedElementIds: {
|
|
||||||
...prevState.selectedElementIds,
|
|
||||||
[multiElement.id]: true,
|
|
||||||
},
|
|
||||||
}));
|
|
||||||
// clicking outside commit zone → update reference for last committed
|
|
||||||
// point
|
|
||||||
mutateElement(multiElement, {
|
|
||||||
lastCommittedPoint:
|
|
||||||
multiElement.points[multiElement.points.length - 1],
|
|
||||||
});
|
|
||||||
document.documentElement.style.cursor = CURSOR_TYPE.POINTER;
|
|
||||||
} else {
|
|
||||||
const [gridX, gridY] = getGridPoint(
|
|
||||||
x,
|
|
||||||
y,
|
|
||||||
this.state.elementType === "draw" ? null : this.state.gridSize,
|
|
||||||
);
|
);
|
||||||
const element = newLinearElement({
|
|
||||||
type: this.state.elementType,
|
|
||||||
x: gridX,
|
|
||||||
y: gridY,
|
|
||||||
strokeColor: this.state.currentItemStrokeColor,
|
|
||||||
backgroundColor: this.state.currentItemBackgroundColor,
|
|
||||||
fillStyle: this.state.currentItemFillStyle,
|
|
||||||
strokeWidth: this.state.currentItemStrokeWidth,
|
|
||||||
strokeStyle: this.state.currentItemStrokeStyle,
|
|
||||||
roughness: this.state.currentItemRoughness,
|
|
||||||
opacity: this.state.currentItemOpacity,
|
|
||||||
});
|
|
||||||
this.setState((prevState) => ({
|
|
||||||
selectedElementIds: {
|
|
||||||
...prevState.selectedElementIds,
|
|
||||||
[element.id]: false,
|
|
||||||
},
|
|
||||||
}));
|
|
||||||
mutateElement(element, {
|
|
||||||
points: [...element.points, [0, 0]],
|
|
||||||
});
|
|
||||||
globalSceneState.replaceAllElements([
|
|
||||||
...globalSceneState.getElementsIncludingDeleted(),
|
|
||||||
element,
|
|
||||||
]);
|
|
||||||
this.setState({
|
|
||||||
draggingElement: element,
|
|
||||||
editingElement: element,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
const [gridX, gridY] = getGridPoint(x, y, this.state.gridSize);
|
this.createGenericElementOnPointerDown(
|
||||||
const element = newElement({
|
this.state.elementType,
|
||||||
type: this.state.elementType,
|
pointerDownState,
|
||||||
x: gridX,
|
);
|
||||||
y: gridY,
|
|
||||||
strokeColor: this.state.currentItemStrokeColor,
|
|
||||||
backgroundColor: this.state.currentItemBackgroundColor,
|
|
||||||
fillStyle: this.state.currentItemFillStyle,
|
|
||||||
strokeWidth: this.state.currentItemStrokeWidth,
|
|
||||||
strokeStyle: this.state.currentItemStrokeStyle,
|
|
||||||
roughness: this.state.currentItemRoughness,
|
|
||||||
opacity: this.state.currentItemOpacity,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (element.type === "selection") {
|
|
||||||
this.setState({
|
|
||||||
selectionElement: element,
|
|
||||||
draggingElement: element,
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
globalSceneState.replaceAllElements([
|
|
||||||
...globalSceneState.getElementsIncludingDeleted(),
|
|
||||||
element,
|
|
||||||
]);
|
|
||||||
this.setState({
|
|
||||||
multiElement: null,
|
|
||||||
draggingElement: element,
|
|
||||||
editingElement: element,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let selectedElementWasDuplicated = false;
|
let selectedElementWasDuplicated = false;
|
||||||
@ -2259,8 +2054,8 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
if (dragOffsetXY === null) {
|
if (dragOffsetXY === null) {
|
||||||
dragOffsetXY = getDragOffsetXY(
|
dragOffsetXY = getDragOffsetXY(
|
||||||
getSelectedElements(globalSceneState.getElements(), this.state),
|
getSelectedElements(globalSceneState.getElements(), this.state),
|
||||||
originX,
|
pointerDownState.origin.x,
|
||||||
originY,
|
pointerDownState.origin.y,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2271,21 +2066,21 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
|
|
||||||
if (isOverHorizontalScrollBar) {
|
if (isOverHorizontalScrollBar) {
|
||||||
const x = event.clientX;
|
const x = event.clientX;
|
||||||
const dx = x - lastCoords.x;
|
const dx = x - pointerDownState.lastCoords.x;
|
||||||
this.setState({
|
this.setState({
|
||||||
scrollX: normalizeScroll(this.state.scrollX - dx / this.state.zoom),
|
scrollX: normalizeScroll(this.state.scrollX - dx / this.state.zoom),
|
||||||
});
|
});
|
||||||
lastCoords.x = x;
|
pointerDownState.lastCoords.x = x;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isOverVerticalScrollBar) {
|
if (isOverVerticalScrollBar) {
|
||||||
const y = event.clientY;
|
const y = event.clientY;
|
||||||
const dy = y - lastCoords.y;
|
const dy = y - pointerDownState.lastCoords.y;
|
||||||
this.setState({
|
this.setState({
|
||||||
scrollY: normalizeScroll(this.state.scrollY - dy / this.state.zoom),
|
scrollY: normalizeScroll(this.state.scrollY - dy / this.state.zoom),
|
||||||
});
|
});
|
||||||
lastCoords.y = y;
|
pointerDownState.lastCoords.y = y;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2306,31 +2101,44 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
(this.state.elementType === "arrow" ||
|
(this.state.elementType === "arrow" ||
|
||||||
this.state.elementType === "line")
|
this.state.elementType === "line")
|
||||||
) {
|
) {
|
||||||
if (distance2d(x, y, originX, originY) < DRAGGING_THRESHOLD) {
|
if (
|
||||||
|
distance2d(
|
||||||
|
x,
|
||||||
|
y,
|
||||||
|
pointerDownState.origin.x,
|
||||||
|
pointerDownState.origin.y,
|
||||||
|
) < DRAGGING_THRESHOLD
|
||||||
|
) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isResizingElements) {
|
if (pointerDownState.resize.isResizing) {
|
||||||
const selectedElements = getSelectedElements(
|
const selectedElements = getSelectedElements(
|
||||||
globalSceneState.getElements(),
|
globalSceneState.getElements(),
|
||||||
this.state,
|
this.state,
|
||||||
);
|
);
|
||||||
|
const resizeHandle = pointerDownState.resize.handle;
|
||||||
this.setState({
|
this.setState({
|
||||||
|
// TODO: rename this state field to "isScaling" to distinguish
|
||||||
|
// it from the generic "isResizing" which includes scaling and
|
||||||
|
// rotating
|
||||||
isResizing: resizeHandle && resizeHandle !== "rotation",
|
isResizing: resizeHandle && resizeHandle !== "rotation",
|
||||||
isRotating: resizeHandle === "rotation",
|
isRotating: resizeHandle === "rotation",
|
||||||
});
|
});
|
||||||
const [resizeX, resizeY] = getGridPoint(
|
const [resizeX, resizeY] = getGridPoint(
|
||||||
x - resizeOffsetXY[0],
|
x - pointerDownState.resize.offset.x,
|
||||||
y - resizeOffsetXY[1],
|
y - pointerDownState.resize.offset.y,
|
||||||
this.state.gridSize,
|
this.state.gridSize,
|
||||||
);
|
);
|
||||||
if (
|
if (
|
||||||
resizeElements(
|
resizeElements(
|
||||||
resizeHandle,
|
resizeHandle,
|
||||||
setResizeHandle,
|
(newResizeHandle) => {
|
||||||
|
pointerDownState.resize.handle = newResizeHandle;
|
||||||
|
},
|
||||||
selectedElements,
|
selectedElements,
|
||||||
resizeArrowDirection,
|
pointerDownState.resize.arrowDirection,
|
||||||
getRotateWithDiscreteAngleKey(event),
|
getRotateWithDiscreteAngleKey(event),
|
||||||
getResizeWithSidesSameLengthKey(event),
|
getResizeWithSidesSameLengthKey(event),
|
||||||
getResizeCenterPointKey(event),
|
getResizeCenterPointKey(event),
|
||||||
@ -2348,17 +2156,18 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
(appState) => this.setState(appState),
|
(appState) => this.setState(appState),
|
||||||
x,
|
x,
|
||||||
y,
|
y,
|
||||||
lastCoords.x,
|
pointerDownState.lastCoords.x,
|
||||||
lastCoords.y,
|
pointerDownState.lastCoords.y,
|
||||||
);
|
);
|
||||||
|
|
||||||
if (didDrag) {
|
if (didDrag) {
|
||||||
lastCoords.x = x;
|
pointerDownState.lastCoords.x = x;
|
||||||
lastCoords.y = y;
|
pointerDownState.lastCoords.y = y;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const hitElement = pointerDownState.hit.element;
|
||||||
if (hitElement && this.state.selectedElementIds[hitElement.id]) {
|
if (hitElement && this.state.selectedElementIds[hitElement.id]) {
|
||||||
// Marking that click was used for dragging to check
|
// Marking that click was used for dragging to check
|
||||||
// if elements should be deselected on pointerup
|
// if elements should be deselected on pointerup
|
||||||
@ -2391,7 +2200,8 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
this.state.selectedElementIds[element.id] ||
|
this.state.selectedElementIds[element.id] ||
|
||||||
// case: the state.selectedElementIds might not have been
|
// case: the state.selectedElementIds might not have been
|
||||||
// updated yet by the time this mousemove event is fired
|
// updated yet by the time this mousemove event is fired
|
||||||
(element.id === hitElement.id && hitElementWasAddedToSelection)
|
(element.id === hitElement.id &&
|
||||||
|
pointerDownState.hit.wasAddedToSelection)
|
||||||
) {
|
) {
|
||||||
const duplicatedElement = duplicateElement(
|
const duplicatedElement = duplicateElement(
|
||||||
this.state.editingGroupId,
|
this.state.editingGroupId,
|
||||||
@ -2399,8 +2209,8 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
element,
|
element,
|
||||||
);
|
);
|
||||||
const [originDragX, originDragY] = getGridPoint(
|
const [originDragX, originDragY] = getGridPoint(
|
||||||
originX - dragOffsetXY[0],
|
pointerDownState.origin.x - dragOffsetXY[0],
|
||||||
originY - dragOffsetXY[1],
|
pointerDownState.origin.y - dragOffsetXY[1],
|
||||||
this.state.gridSize,
|
this.state.gridSize,
|
||||||
);
|
);
|
||||||
mutateElement(duplicatedElement, {
|
mutateElement(duplicatedElement, {
|
||||||
@ -2467,12 +2277,12 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
dragNewElement(
|
dragNewElement(
|
||||||
draggingElement,
|
draggingElement,
|
||||||
this.state.elementType,
|
this.state.elementType,
|
||||||
originX,
|
pointerDownState.origin.x,
|
||||||
originY,
|
pointerDownState.origin.y,
|
||||||
x,
|
x,
|
||||||
y,
|
y,
|
||||||
distance(originX, x),
|
distance(pointerDownState.origin.x, x),
|
||||||
distance(originY, y),
|
distance(pointerDownState.origin.y, y),
|
||||||
getResizeWithSidesSameLengthKey(event),
|
getResizeWithSidesSameLengthKey(event),
|
||||||
getResizeCenterPointKey(event),
|
getResizeCenterPointKey(event),
|
||||||
);
|
);
|
||||||
@ -2654,11 +2464,12 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
// If click occurred and elements were dragged or some element
|
// If click occurred and elements were dragged or some element
|
||||||
// was added to selection (on pointerdown phase) we need to keep
|
// was added to selection (on pointerdown phase) we need to keep
|
||||||
// selection unchanged
|
// selection unchanged
|
||||||
|
const hitElement = pointerDownState.hit.element;
|
||||||
if (
|
if (
|
||||||
getSelectedGroupIds(this.state).length === 0 &&
|
getSelectedGroupIds(this.state).length === 0 &&
|
||||||
hitElement &&
|
hitElement &&
|
||||||
!draggingOccurred &&
|
!draggingOccurred &&
|
||||||
!hitElementWasAddedToSelection
|
!pointerDownState.hit.wasAddedToSelection
|
||||||
) {
|
) {
|
||||||
if (childEvent.shiftKey) {
|
if (childEvent.shiftKey) {
|
||||||
this.setState((prevState) => ({
|
this.setState((prevState) => ({
|
||||||
@ -2859,7 +2670,7 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
// Returns whether the event is a dragging a scrollbar
|
// Returns whether the event is a dragging a scrollbar
|
||||||
private handleDraggingScrollBar(
|
private handleDraggingScrollBar(
|
||||||
event: React.PointerEvent<HTMLCanvasElement>,
|
event: React.PointerEvent<HTMLCanvasElement>,
|
||||||
lastCoords: { x: number; y: number },
|
pointerDownState: PointerDownState,
|
||||||
{
|
{
|
||||||
isOverHorizontalScrollBar,
|
isOverHorizontalScrollBar,
|
||||||
isOverVerticalScrollBar,
|
isOverVerticalScrollBar,
|
||||||
@ -2877,8 +2688,8 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
isDraggingScrollBar = true;
|
isDraggingScrollBar = true;
|
||||||
lastCoords.x = event.clientX;
|
pointerDownState.lastCoords.x = event.clientX;
|
||||||
lastCoords.y = event.clientY;
|
pointerDownState.lastCoords.y = event.clientY;
|
||||||
const onPointerMove = withBatchedUpdates((event: PointerEvent) => {
|
const onPointerMove = withBatchedUpdates((event: PointerEvent) => {
|
||||||
const target = event.target;
|
const target = event.target;
|
||||||
if (!(target instanceof HTMLElement)) {
|
if (!(target instanceof HTMLElement)) {
|
||||||
@ -2887,21 +2698,21 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
|
|
||||||
if (isOverHorizontalScrollBar) {
|
if (isOverHorizontalScrollBar) {
|
||||||
const x = event.clientX;
|
const x = event.clientX;
|
||||||
const dx = x - lastCoords.x;
|
const dx = x - pointerDownState.lastCoords.x;
|
||||||
this.setState({
|
this.setState({
|
||||||
scrollX: normalizeScroll(this.state.scrollX - dx / this.state.zoom),
|
scrollX: normalizeScroll(this.state.scrollX - dx / this.state.zoom),
|
||||||
});
|
});
|
||||||
lastCoords.x = x;
|
pointerDownState.lastCoords.x = x;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isOverVerticalScrollBar) {
|
if (isOverVerticalScrollBar) {
|
||||||
const y = event.clientY;
|
const y = event.clientY;
|
||||||
const dy = y - lastCoords.y;
|
const dy = y - pointerDownState.lastCoords.y;
|
||||||
this.setState({
|
this.setState({
|
||||||
scrollY: normalizeScroll(this.state.scrollY - dy / this.state.zoom),
|
scrollY: normalizeScroll(this.state.scrollY - dy / this.state.zoom),
|
||||||
});
|
});
|
||||||
lastCoords.y = y;
|
pointerDownState.lastCoords.y = y;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -2924,6 +2735,306 @@ class App extends React.Component<ExcalidrawProps, AppState> {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private clearSelectionIfNotUsingSelection = (): void => {
|
||||||
|
if (this.state.elementType !== "selection") {
|
||||||
|
this.setState({
|
||||||
|
selectedElementIds: {},
|
||||||
|
selectedGroupIds: {},
|
||||||
|
editingGroupId: null,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Returns whether the pointer event has been completely handled
|
||||||
|
private handleSelectionOnPointerDown = (
|
||||||
|
event: React.PointerEvent<HTMLCanvasElement>,
|
||||||
|
pointerDownState: PointerDownState,
|
||||||
|
): boolean => {
|
||||||
|
if (this.state.elementType === "selection") {
|
||||||
|
const elements = globalSceneState.getElements();
|
||||||
|
const selectedElements = getSelectedElements(elements, this.state);
|
||||||
|
if (selectedElements.length === 1 && !this.state.editingLinearElement) {
|
||||||
|
const elementWithResizeHandler = getElementWithResizeHandler(
|
||||||
|
elements,
|
||||||
|
this.state,
|
||||||
|
pointerDownState.origin.x,
|
||||||
|
pointerDownState.origin.y,
|
||||||
|
this.state.zoom,
|
||||||
|
event.pointerType,
|
||||||
|
);
|
||||||
|
if (elementWithResizeHandler != null) {
|
||||||
|
this.setState({
|
||||||
|
resizingElement: elementWithResizeHandler.element,
|
||||||
|
});
|
||||||
|
pointerDownState.resize.handle =
|
||||||
|
elementWithResizeHandler.resizeHandle;
|
||||||
|
}
|
||||||
|
} else if (selectedElements.length > 1) {
|
||||||
|
pointerDownState.resize.handle = getResizeHandlerFromCoords(
|
||||||
|
getCommonBounds(selectedElements),
|
||||||
|
pointerDownState.origin.x,
|
||||||
|
pointerDownState.origin.y,
|
||||||
|
this.state.zoom,
|
||||||
|
event.pointerType,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (pointerDownState.resize.handle) {
|
||||||
|
document.documentElement.style.cursor = getCursorForResizingElement({
|
||||||
|
resizeHandle: pointerDownState.resize.handle,
|
||||||
|
});
|
||||||
|
pointerDownState.resize.isResizing = true;
|
||||||
|
pointerDownState.resize.offset = tupleToCoors(
|
||||||
|
getResizeOffsetXY(
|
||||||
|
pointerDownState.resize.handle,
|
||||||
|
selectedElements,
|
||||||
|
pointerDownState.origin.x,
|
||||||
|
pointerDownState.origin.y,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
if (
|
||||||
|
selectedElements.length === 1 &&
|
||||||
|
isLinearElement(selectedElements[0]) &&
|
||||||
|
selectedElements[0].points.length === 2
|
||||||
|
) {
|
||||||
|
pointerDownState.resize.arrowDirection = getResizeArrowDirection(
|
||||||
|
pointerDownState.resize.handle,
|
||||||
|
selectedElements[0],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (this.state.editingLinearElement) {
|
||||||
|
const ret = LinearElementEditor.handlePointerDown(
|
||||||
|
event,
|
||||||
|
this.state,
|
||||||
|
(appState) => this.setState(appState),
|
||||||
|
history,
|
||||||
|
pointerDownState.origin.x,
|
||||||
|
pointerDownState.origin.y,
|
||||||
|
);
|
||||||
|
if (ret.hitElement) {
|
||||||
|
pointerDownState.hit.element = ret.hitElement;
|
||||||
|
}
|
||||||
|
if (ret.didAddPoint) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// hitElement may already be set above, so check first
|
||||||
|
pointerDownState.hit.element =
|
||||||
|
pointerDownState.hit.element ??
|
||||||
|
getElementAtPosition(
|
||||||
|
elements,
|
||||||
|
this.state,
|
||||||
|
pointerDownState.origin.x,
|
||||||
|
pointerDownState.origin.y,
|
||||||
|
this.state.zoom,
|
||||||
|
);
|
||||||
|
|
||||||
|
this.maybeClearSelectionWhenHittingElement(
|
||||||
|
event,
|
||||||
|
pointerDownState.hit.element,
|
||||||
|
);
|
||||||
|
|
||||||
|
// If we click on something
|
||||||
|
const hitElement = pointerDownState.hit.element;
|
||||||
|
if (hitElement != null) {
|
||||||
|
// deselect if item is selected
|
||||||
|
// if shift is not clicked, this will always return true
|
||||||
|
// otherwise, it will trigger selection based on current
|
||||||
|
// state of the box
|
||||||
|
if (!this.state.selectedElementIds[hitElement.id]) {
|
||||||
|
// if we are currently editing a group, treat all selections outside of the group
|
||||||
|
// as exiting editing mode.
|
||||||
|
if (
|
||||||
|
this.state.editingGroupId &&
|
||||||
|
!isElementInGroup(hitElement, this.state.editingGroupId)
|
||||||
|
) {
|
||||||
|
this.setState({
|
||||||
|
selectedElementIds: {},
|
||||||
|
selectedGroupIds: {},
|
||||||
|
editingGroupId: null,
|
||||||
|
});
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
this.setState((prevState) => {
|
||||||
|
return selectGroupsForSelectedElements(
|
||||||
|
{
|
||||||
|
...prevState,
|
||||||
|
selectedElementIds: {
|
||||||
|
...prevState.selectedElementIds,
|
||||||
|
[hitElement!.id]: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
globalSceneState.getElements(),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
// TODO: this is strange...
|
||||||
|
globalSceneState.replaceAllElements(
|
||||||
|
globalSceneState.getElementsIncludingDeleted(),
|
||||||
|
);
|
||||||
|
pointerDownState.hit.wasAddedToSelection = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const { selectedElementIds } = this.state;
|
||||||
|
this.setState({
|
||||||
|
previousSelectedElementIds: selectedElementIds,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
private handleTextOnPointerDown = (
|
||||||
|
event: React.PointerEvent<HTMLCanvasElement>,
|
||||||
|
pointerDownState: PointerDownState,
|
||||||
|
): void => {
|
||||||
|
// if we're currently still editing text, clicking outside
|
||||||
|
// should only finalize it, not create another (irrespective
|
||||||
|
// of state.elementLocked)
|
||||||
|
if (this.state.editingElement?.type === "text") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.startTextEditing({
|
||||||
|
sceneX: pointerDownState.origin.x,
|
||||||
|
sceneY: pointerDownState.origin.y,
|
||||||
|
insertAtParentCenter: !event.altKey,
|
||||||
|
});
|
||||||
|
|
||||||
|
resetCursor();
|
||||||
|
if (!this.state.elementLocked) {
|
||||||
|
this.setState({
|
||||||
|
elementType: "selection",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
private handleLinearElementOnPointerDown = (
|
||||||
|
event: React.PointerEvent<HTMLCanvasElement>,
|
||||||
|
elementType: "draw" | "line" | "arrow",
|
||||||
|
pointerDownState: PointerDownState,
|
||||||
|
): void => {
|
||||||
|
if (this.state.multiElement) {
|
||||||
|
const { multiElement } = this.state;
|
||||||
|
|
||||||
|
// finalize if completing a loop
|
||||||
|
if (multiElement.type === "line" && isPathALoop(multiElement.points)) {
|
||||||
|
mutateElement(multiElement, {
|
||||||
|
lastCommittedPoint:
|
||||||
|
multiElement.points[multiElement.points.length - 1],
|
||||||
|
});
|
||||||
|
this.actionManager.executeAction(actionFinalize);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { x: rx, y: ry, lastCommittedPoint } = multiElement;
|
||||||
|
|
||||||
|
// clicking inside commit zone → finalize arrow
|
||||||
|
if (
|
||||||
|
multiElement.points.length > 1 &&
|
||||||
|
lastCommittedPoint &&
|
||||||
|
distance2d(
|
||||||
|
pointerDownState.origin.x - rx,
|
||||||
|
pointerDownState.origin.y - ry,
|
||||||
|
lastCommittedPoint[0],
|
||||||
|
lastCommittedPoint[1],
|
||||||
|
) < LINE_CONFIRM_THRESHOLD
|
||||||
|
) {
|
||||||
|
this.actionManager.executeAction(actionFinalize);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.setState((prevState) => ({
|
||||||
|
selectedElementIds: {
|
||||||
|
...prevState.selectedElementIds,
|
||||||
|
[multiElement.id]: true,
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
// clicking outside commit zone → update reference for last committed
|
||||||
|
// point
|
||||||
|
mutateElement(multiElement, {
|
||||||
|
lastCommittedPoint: multiElement.points[multiElement.points.length - 1],
|
||||||
|
});
|
||||||
|
document.documentElement.style.cursor = CURSOR_TYPE.POINTER;
|
||||||
|
} else {
|
||||||
|
const [gridX, gridY] = getGridPoint(
|
||||||
|
pointerDownState.origin.x,
|
||||||
|
pointerDownState.origin.y,
|
||||||
|
elementType === "draw" ? null : this.state.gridSize,
|
||||||
|
);
|
||||||
|
const element = newLinearElement({
|
||||||
|
type: elementType,
|
||||||
|
x: gridX,
|
||||||
|
y: gridY,
|
||||||
|
strokeColor: this.state.currentItemStrokeColor,
|
||||||
|
backgroundColor: this.state.currentItemBackgroundColor,
|
||||||
|
fillStyle: this.state.currentItemFillStyle,
|
||||||
|
strokeWidth: this.state.currentItemStrokeWidth,
|
||||||
|
strokeStyle: this.state.currentItemStrokeStyle,
|
||||||
|
roughness: this.state.currentItemRoughness,
|
||||||
|
opacity: this.state.currentItemOpacity,
|
||||||
|
});
|
||||||
|
this.setState((prevState) => ({
|
||||||
|
selectedElementIds: {
|
||||||
|
...prevState.selectedElementIds,
|
||||||
|
[element.id]: false,
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
mutateElement(element, {
|
||||||
|
points: [...element.points, [0, 0]],
|
||||||
|
});
|
||||||
|
globalSceneState.replaceAllElements([
|
||||||
|
...globalSceneState.getElementsIncludingDeleted(),
|
||||||
|
element,
|
||||||
|
]);
|
||||||
|
this.setState({
|
||||||
|
draggingElement: element,
|
||||||
|
editingElement: element,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
private createGenericElementOnPointerDown = (
|
||||||
|
elementType: ExcalidrawGenericElement["type"],
|
||||||
|
pointerDownState: PointerDownState,
|
||||||
|
): void => {
|
||||||
|
const [gridX, gridY] = getGridPoint(
|
||||||
|
pointerDownState.origin.x,
|
||||||
|
pointerDownState.origin.y,
|
||||||
|
this.state.gridSize,
|
||||||
|
);
|
||||||
|
const element = newElement({
|
||||||
|
type: elementType,
|
||||||
|
x: gridX,
|
||||||
|
y: gridY,
|
||||||
|
strokeColor: this.state.currentItemStrokeColor,
|
||||||
|
backgroundColor: this.state.currentItemBackgroundColor,
|
||||||
|
fillStyle: this.state.currentItemFillStyle,
|
||||||
|
strokeWidth: this.state.currentItemStrokeWidth,
|
||||||
|
strokeStyle: this.state.currentItemStrokeStyle,
|
||||||
|
roughness: this.state.currentItemRoughness,
|
||||||
|
opacity: this.state.currentItemOpacity,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (element.type === "selection") {
|
||||||
|
this.setState({
|
||||||
|
selectionElement: element,
|
||||||
|
draggingElement: element,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
globalSceneState.replaceAllElements([
|
||||||
|
...globalSceneState.getElementsIncludingDeleted(),
|
||||||
|
element,
|
||||||
|
]);
|
||||||
|
this.setState({
|
||||||
|
multiElement: null,
|
||||||
|
draggingElement: element,
|
||||||
|
editingElement: element,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
private maybeClearSelectionWhenHittingElement(
|
private maybeClearSelectionWhenHittingElement(
|
||||||
event: React.PointerEvent<HTMLCanvasElement>,
|
event: React.PointerEvent<HTMLCanvasElement>,
|
||||||
hitElement: ExcalidrawElement | null,
|
hitElement: ExcalidrawElement | null,
|
||||||
|
@ -241,3 +241,10 @@ const RE_RTL_CHECK = new RegExp(`^[^${RS_LTR_CHARS}]*[${RS_RTL_CHARS}]`);
|
|||||||
export const isRTL = (text: string) => {
|
export const isRTL = (text: string) => {
|
||||||
return RE_RTL_CHECK.test(text);
|
return RE_RTL_CHECK.test(text);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export function tupleToCoors(
|
||||||
|
xyTuple: [number, number],
|
||||||
|
): { x: number; y: number } {
|
||||||
|
const [x, y] = xyTuple;
|
||||||
|
return { x, y };
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user