2022-02-21 16:46:39 +05:30
|
|
|
import { BOUND_TEXT_PADDING, SHIFT_LOCKING_ANGLE } from "../constants";
|
2020-04-07 17:49:59 +09:00
|
|
|
import { rescalePoints } from "../points";
|
2020-04-10 01:10:35 +09:00
|
|
|
|
2020-12-06 22:39:31 +00:00
|
|
|
import {
|
|
|
|
rotate,
|
|
|
|
adjustXYWithRotation,
|
|
|
|
centerPoint,
|
|
|
|
rotatePoint,
|
|
|
|
} from "../math";
|
2020-04-08 09:49:52 -07:00
|
|
|
import {
|
|
|
|
ExcalidrawLinearElement,
|
2020-05-28 07:17:15 +09:00
|
|
|
ExcalidrawTextElement,
|
2020-04-08 09:49:52 -07:00
|
|
|
NonDeletedExcalidrawElement,
|
|
|
|
NonDeleted,
|
|
|
|
} from "./types";
|
2020-05-05 00:25:40 +09:00
|
|
|
import {
|
|
|
|
getElementAbsoluteCoords,
|
|
|
|
getCommonBounds,
|
|
|
|
getResizedElementAbsoluteCoords,
|
2022-08-13 22:53:10 +05:00
|
|
|
getCommonBoundingBox,
|
2020-05-05 00:25:40 +09:00
|
|
|
} from "./bounds";
|
2021-05-09 16:42:10 +01:00
|
|
|
import {
|
|
|
|
isFreeDrawElement,
|
|
|
|
isLinearElement,
|
|
|
|
isTextElement,
|
|
|
|
} from "./typeChecks";
|
2020-04-07 17:49:59 +09:00
|
|
|
import { mutateElement } from "./mutateElement";
|
2021-12-16 21:14:03 +05:30
|
|
|
import { getFontString } from "../utils";
|
2020-08-08 21:04:15 -07:00
|
|
|
import { updateBoundElements } from "./binding";
|
2020-08-10 14:16:39 +02:00
|
|
|
import {
|
|
|
|
TransformHandleType,
|
|
|
|
MaybeTransformHandleType,
|
2020-12-06 22:39:31 +00:00
|
|
|
TransformHandleDirection,
|
2020-08-10 14:16:39 +02:00
|
|
|
} from "./transformHandles";
|
2021-06-01 23:52:13 +05:30
|
|
|
import { Point, PointerDownState } from "../types";
|
2021-12-16 21:14:03 +05:30
|
|
|
import Scene from "../scene/Scene";
|
|
|
|
import {
|
2022-02-18 18:20:55 +05:30
|
|
|
getApproxMinLineHeight,
|
2021-12-16 21:14:03 +05:30
|
|
|
getApproxMinLineWidth,
|
2022-01-13 21:35:38 +05:30
|
|
|
getBoundTextElement,
|
2021-12-16 21:14:03 +05:30
|
|
|
getBoundTextElementId,
|
|
|
|
handleBindTextResize,
|
|
|
|
measureText,
|
|
|
|
} from "./textElement";
|
2020-04-07 17:49:59 +09:00
|
|
|
|
2021-03-26 11:45:08 -04:00
|
|
|
export const normalizeAngle = (angle: number): number => {
|
2020-07-26 19:21:38 +09:00
|
|
|
if (angle >= 2 * Math.PI) {
|
|
|
|
return angle - 2 * Math.PI;
|
|
|
|
}
|
|
|
|
return angle;
|
|
|
|
};
|
|
|
|
|
2020-08-28 17:20:06 +09:00
|
|
|
// Returns true when transform (resizing/rotation) happened
|
|
|
|
export const transformElements = (
|
2020-09-11 17:22:40 +02:00
|
|
|
pointerDownState: PointerDownState,
|
2020-08-10 14:16:39 +02:00
|
|
|
transformHandleType: MaybeTransformHandleType,
|
2020-07-26 19:21:38 +09:00
|
|
|
selectedElements: readonly NonDeletedExcalidrawElement[],
|
2020-05-11 00:41:36 +09:00
|
|
|
resizeArrowDirection: "origin" | "end",
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldRotateWithDiscreteAngle: boolean,
|
|
|
|
shouldResizeFromCenter: boolean,
|
|
|
|
shouldMaintainAspectRatio: boolean,
|
2020-05-09 17:57:00 +09:00
|
|
|
pointerX: number,
|
|
|
|
pointerY: number,
|
2020-07-26 19:21:38 +09:00
|
|
|
centerX: number,
|
|
|
|
centerY: number,
|
2020-05-05 00:25:40 +09:00
|
|
|
) => {
|
|
|
|
if (selectedElements.length === 1) {
|
|
|
|
const [element] = selectedElements;
|
2020-08-10 14:16:39 +02:00
|
|
|
if (transformHandleType === "rotation") {
|
2020-06-24 00:24:52 +09:00
|
|
|
rotateSingleElement(
|
|
|
|
element,
|
|
|
|
pointerX,
|
|
|
|
pointerY,
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldRotateWithDiscreteAngle,
|
2020-06-24 00:24:52 +09:00
|
|
|
);
|
2020-08-08 21:04:15 -07:00
|
|
|
updateBoundElements(element);
|
2020-05-28 07:17:15 +09:00
|
|
|
} else if (
|
2020-12-06 22:39:31 +00:00
|
|
|
isTextElement(element) &&
|
2020-08-10 14:16:39 +02:00
|
|
|
(transformHandleType === "nw" ||
|
|
|
|
transformHandleType === "ne" ||
|
|
|
|
transformHandleType === "sw" ||
|
|
|
|
transformHandleType === "se")
|
2020-05-28 07:17:15 +09:00
|
|
|
) {
|
|
|
|
resizeSingleTextElement(
|
|
|
|
element,
|
2020-08-10 14:16:39 +02:00
|
|
|
transformHandleType,
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldResizeFromCenter,
|
2020-05-28 07:17:15 +09:00
|
|
|
pointerX,
|
|
|
|
pointerY,
|
|
|
|
);
|
2020-09-08 12:03:49 -04:00
|
|
|
updateBoundElements(element);
|
2020-08-10 14:16:39 +02:00
|
|
|
} else if (transformHandleType) {
|
2021-01-16 18:49:13 +00:00
|
|
|
resizeSingleElement(
|
2022-02-22 18:45:59 +05:30
|
|
|
pointerDownState.originalElements,
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldMaintainAspectRatio,
|
2021-01-16 18:49:13 +00:00
|
|
|
element,
|
|
|
|
transformHandleType,
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldResizeFromCenter,
|
2021-01-16 18:49:13 +00:00
|
|
|
pointerX,
|
|
|
|
pointerY,
|
|
|
|
);
|
2020-05-05 00:25:40 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2020-07-26 19:21:38 +09:00
|
|
|
} else if (selectedElements.length > 1) {
|
2020-08-10 14:16:39 +02:00
|
|
|
if (transformHandleType === "rotation") {
|
2020-07-26 19:21:38 +09:00
|
|
|
rotateMultipleElements(
|
2020-09-11 17:22:40 +02:00
|
|
|
pointerDownState,
|
2020-07-26 19:21:38 +09:00
|
|
|
selectedElements,
|
|
|
|
pointerX,
|
|
|
|
pointerY,
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldRotateWithDiscreteAngle,
|
2020-07-26 19:21:38 +09:00
|
|
|
centerX,
|
|
|
|
centerY,
|
|
|
|
);
|
|
|
|
return true;
|
|
|
|
} else if (
|
2020-08-10 14:16:39 +02:00
|
|
|
transformHandleType === "nw" ||
|
|
|
|
transformHandleType === "ne" ||
|
|
|
|
transformHandleType === "sw" ||
|
|
|
|
transformHandleType === "se"
|
2020-07-26 19:21:38 +09:00
|
|
|
) {
|
|
|
|
resizeMultipleElements(
|
2022-08-13 22:53:10 +05:00
|
|
|
pointerDownState,
|
2020-07-26 19:21:38 +09:00
|
|
|
selectedElements,
|
2020-08-10 14:16:39 +02:00
|
|
|
transformHandleType,
|
2022-08-13 22:53:10 +05:00
|
|
|
shouldResizeFromCenter,
|
2020-07-26 19:21:38 +09:00
|
|
|
pointerX,
|
|
|
|
pointerY,
|
|
|
|
);
|
|
|
|
return true;
|
|
|
|
}
|
2020-05-05 00:25:40 +09:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
|
|
|
const rotateSingleElement = (
|
|
|
|
element: NonDeletedExcalidrawElement,
|
2020-05-09 17:57:00 +09:00
|
|
|
pointerX: number,
|
|
|
|
pointerY: number,
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldRotateWithDiscreteAngle: boolean,
|
2020-05-05 00:25:40 +09:00
|
|
|
) => {
|
|
|
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element);
|
|
|
|
const cx = (x1 + x2) / 2;
|
|
|
|
const cy = (y1 + y2) / 2;
|
2020-05-09 17:57:00 +09:00
|
|
|
let angle = (5 * Math.PI) / 2 + Math.atan2(pointerY - cy, pointerX - cx);
|
2021-10-21 22:05:48 +02:00
|
|
|
if (shouldRotateWithDiscreteAngle) {
|
2020-05-05 00:25:40 +09:00
|
|
|
angle += SHIFT_LOCKING_ANGLE / 2;
|
|
|
|
angle -= angle % SHIFT_LOCKING_ANGLE;
|
|
|
|
}
|
2020-07-26 19:21:38 +09:00
|
|
|
angle = normalizeAngle(angle);
|
2020-05-05 00:25:40 +09:00
|
|
|
mutateElement(element, { angle });
|
2021-12-16 21:14:03 +05:30
|
|
|
const boundTextElementId = getBoundTextElementId(element);
|
|
|
|
if (boundTextElementId) {
|
|
|
|
const textElement = Scene.getScene(element)!.getElement(boundTextElementId);
|
|
|
|
mutateElement(textElement!, { angle });
|
|
|
|
}
|
2020-05-05 00:25:40 +09:00
|
|
|
};
|
|
|
|
|
2020-05-14 23:56:14 +09:00
|
|
|
const rescalePointsInElement = (
|
|
|
|
element: NonDeletedExcalidrawElement,
|
|
|
|
width: number,
|
|
|
|
height: number,
|
2022-08-16 21:51:43 +02:00
|
|
|
normalizePoints: boolean,
|
2020-05-14 23:56:14 +09:00
|
|
|
) =>
|
2021-05-09 16:42:10 +01:00
|
|
|
isLinearElement(element) || isFreeDrawElement(element)
|
2020-05-14 23:56:14 +09:00
|
|
|
? {
|
|
|
|
points: rescalePoints(
|
|
|
|
0,
|
|
|
|
width,
|
2022-08-16 21:51:43 +02:00
|
|
|
rescalePoints(1, height, element.points, normalizePoints),
|
|
|
|
normalizePoints,
|
2020-05-14 23:56:14 +09:00
|
|
|
),
|
|
|
|
}
|
|
|
|
: {};
|
|
|
|
|
2020-07-09 22:22:10 +09:00
|
|
|
const MIN_FONT_SIZE = 1;
|
|
|
|
|
2020-06-08 18:25:20 +09:00
|
|
|
const measureFontSizeFromWH = (
|
|
|
|
element: NonDeleted<ExcalidrawTextElement>,
|
|
|
|
nextWidth: number,
|
|
|
|
nextHeight: number,
|
|
|
|
): { size: number; baseline: number } | null => {
|
2020-07-09 22:22:10 +09:00
|
|
|
// We only use width to scale font on resize
|
|
|
|
const nextFontSize = element.fontSize * (nextWidth / element.width);
|
|
|
|
if (nextFontSize < MIN_FONT_SIZE) {
|
|
|
|
return null;
|
2020-06-08 18:25:20 +09:00
|
|
|
}
|
2020-07-09 22:22:10 +09:00
|
|
|
const metrics = measureText(
|
2020-06-24 00:24:52 +09:00
|
|
|
element.text,
|
|
|
|
getFontString({ fontSize: nextFontSize, fontFamily: element.fontFamily }),
|
2021-12-16 21:14:03 +05:30
|
|
|
element.containerId ? element.width : null,
|
2020-06-24 00:24:52 +09:00
|
|
|
);
|
2020-07-09 22:22:10 +09:00
|
|
|
return {
|
|
|
|
size: nextFontSize,
|
|
|
|
baseline: metrics.baseline + (nextHeight - metrics.height),
|
|
|
|
};
|
2020-06-08 18:25:20 +09:00
|
|
|
};
|
|
|
|
|
2020-08-10 14:16:39 +02:00
|
|
|
const getSidesForTransformHandle = (
|
|
|
|
transformHandleType: TransformHandleType,
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldResizeFromCenter: boolean,
|
2020-06-25 21:21:27 +02:00
|
|
|
) => {
|
|
|
|
return {
|
|
|
|
n:
|
2020-08-10 14:16:39 +02:00
|
|
|
/^(n|ne|nw)$/.test(transformHandleType) ||
|
2021-10-21 22:05:48 +02:00
|
|
|
(shouldResizeFromCenter && /^(s|se|sw)$/.test(transformHandleType)),
|
2020-06-25 21:21:27 +02:00
|
|
|
s:
|
2020-08-10 14:16:39 +02:00
|
|
|
/^(s|se|sw)$/.test(transformHandleType) ||
|
2021-10-21 22:05:48 +02:00
|
|
|
(shouldResizeFromCenter && /^(n|ne|nw)$/.test(transformHandleType)),
|
2020-06-25 21:21:27 +02:00
|
|
|
w:
|
2020-08-10 14:16:39 +02:00
|
|
|
/^(w|nw|sw)$/.test(transformHandleType) ||
|
2021-10-21 22:05:48 +02:00
|
|
|
(shouldResizeFromCenter && /^(e|ne|se)$/.test(transformHandleType)),
|
2020-06-25 21:21:27 +02:00
|
|
|
e:
|
2020-08-10 14:16:39 +02:00
|
|
|
/^(e|ne|se)$/.test(transformHandleType) ||
|
2021-10-21 22:05:48 +02:00
|
|
|
(shouldResizeFromCenter && /^(w|nw|sw)$/.test(transformHandleType)),
|
2020-06-25 21:21:27 +02:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2020-05-28 07:17:15 +09:00
|
|
|
const resizeSingleTextElement = (
|
|
|
|
element: NonDeleted<ExcalidrawTextElement>,
|
2020-08-10 14:16:39 +02:00
|
|
|
transformHandleType: "nw" | "ne" | "sw" | "se",
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldResizeFromCenter: boolean,
|
2020-05-28 07:17:15 +09:00
|
|
|
pointerX: number,
|
|
|
|
pointerY: number,
|
|
|
|
) => {
|
|
|
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element);
|
|
|
|
const cx = (x1 + x2) / 2;
|
|
|
|
const cy = (y1 + y2) / 2;
|
|
|
|
// rotation pointer with reverse angle
|
|
|
|
const [rotatedX, rotatedY] = rotate(
|
|
|
|
pointerX,
|
|
|
|
pointerY,
|
|
|
|
cx,
|
|
|
|
cy,
|
|
|
|
-element.angle,
|
|
|
|
);
|
2020-11-06 22:06:30 +02:00
|
|
|
let scale: number;
|
2020-08-10 14:16:39 +02:00
|
|
|
switch (transformHandleType) {
|
2020-05-28 07:17:15 +09:00
|
|
|
case "se":
|
|
|
|
scale = Math.max(
|
|
|
|
(rotatedX - x1) / (x2 - x1),
|
|
|
|
(rotatedY - y1) / (y2 - y1),
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case "nw":
|
|
|
|
scale = Math.max(
|
|
|
|
(x2 - rotatedX) / (x2 - x1),
|
|
|
|
(y2 - rotatedY) / (y2 - y1),
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case "ne":
|
|
|
|
scale = Math.max(
|
|
|
|
(rotatedX - x1) / (x2 - x1),
|
|
|
|
(y2 - rotatedY) / (y2 - y1),
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
case "sw":
|
|
|
|
scale = Math.max(
|
|
|
|
(x2 - rotatedX) / (x2 - x1),
|
|
|
|
(rotatedY - y1) / (y2 - y1),
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (scale > 0) {
|
2020-06-08 18:25:20 +09:00
|
|
|
const nextWidth = element.width * scale;
|
|
|
|
const nextHeight = element.height * scale;
|
|
|
|
const nextFont = measureFontSizeFromWH(element, nextWidth, nextHeight);
|
|
|
|
if (nextFont === null) {
|
2020-05-28 07:17:15 +09:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
const [nextX1, nextY1, nextX2, nextY2] = getResizedElementAbsoluteCoords(
|
|
|
|
element,
|
2020-06-08 18:25:20 +09:00
|
|
|
nextWidth,
|
|
|
|
nextHeight,
|
2022-08-16 21:51:43 +02:00
|
|
|
false,
|
2020-05-28 07:17:15 +09:00
|
|
|
);
|
|
|
|
const deltaX1 = (x1 - nextX1) / 2;
|
|
|
|
const deltaY1 = (y1 - nextY1) / 2;
|
|
|
|
const deltaX2 = (x2 - nextX2) / 2;
|
|
|
|
const deltaY2 = (y2 - nextY2) / 2;
|
|
|
|
const [nextElementX, nextElementY] = adjustXYWithRotation(
|
2021-10-21 22:05:48 +02:00
|
|
|
getSidesForTransformHandle(transformHandleType, shouldResizeFromCenter),
|
2020-05-28 07:17:15 +09:00
|
|
|
element.x,
|
|
|
|
element.y,
|
|
|
|
element.angle,
|
|
|
|
deltaX1,
|
|
|
|
deltaY1,
|
|
|
|
deltaX2,
|
|
|
|
deltaY2,
|
|
|
|
);
|
|
|
|
mutateElement(element, {
|
2020-06-08 18:25:20 +09:00
|
|
|
fontSize: nextFont.size,
|
|
|
|
width: nextWidth,
|
|
|
|
height: nextHeight,
|
|
|
|
baseline: nextFont.baseline,
|
2020-05-28 07:17:15 +09:00
|
|
|
x: nextElementX,
|
|
|
|
y: nextElementY,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-03-26 11:45:08 -04:00
|
|
|
export const resizeSingleElement = (
|
2022-02-22 18:45:59 +05:30
|
|
|
originalElements: PointerDownState["originalElements"],
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldMaintainAspectRatio: boolean,
|
2020-05-05 00:25:40 +09:00
|
|
|
element: NonDeletedExcalidrawElement,
|
2020-12-06 22:39:31 +00:00
|
|
|
transformHandleDirection: TransformHandleDirection,
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldResizeFromCenter: boolean,
|
2020-12-06 22:39:31 +00:00
|
|
|
pointerX: number,
|
|
|
|
pointerY: number,
|
|
|
|
) => {
|
2022-02-22 18:45:59 +05:30
|
|
|
const stateAtResizeStart = originalElements.get(element.id)!;
|
2021-01-16 18:49:13 +00:00
|
|
|
// Gets bounds corners
|
|
|
|
const [x1, y1, x2, y2] = getResizedElementAbsoluteCoords(
|
|
|
|
stateAtResizeStart,
|
|
|
|
stateAtResizeStart.width,
|
|
|
|
stateAtResizeStart.height,
|
2022-08-16 21:51:43 +02:00
|
|
|
true,
|
2021-01-16 18:49:13 +00:00
|
|
|
);
|
2020-12-06 22:39:31 +00:00
|
|
|
const startTopLeft: Point = [x1, y1];
|
|
|
|
const startBottomRight: Point = [x2, y2];
|
|
|
|
const startCenter: Point = centerPoint(startTopLeft, startBottomRight);
|
|
|
|
|
|
|
|
// Calculate new dimensions based on cursor position
|
|
|
|
const rotatedPointer = rotatePoint(
|
|
|
|
[pointerX, pointerY],
|
|
|
|
startCenter,
|
|
|
|
-stateAtResizeStart.angle,
|
|
|
|
);
|
2021-01-16 18:49:13 +00:00
|
|
|
|
2021-05-09 16:42:10 +01:00
|
|
|
// Get bounds corners rendered on screen
|
2021-01-16 18:49:13 +00:00
|
|
|
const [esx1, esy1, esx2, esy2] = getResizedElementAbsoluteCoords(
|
|
|
|
element,
|
|
|
|
element.width,
|
|
|
|
element.height,
|
2022-08-16 21:51:43 +02:00
|
|
|
true,
|
2021-01-16 18:49:13 +00:00
|
|
|
);
|
2021-12-16 21:14:03 +05:30
|
|
|
|
2021-01-16 18:49:13 +00:00
|
|
|
const boundsCurrentWidth = esx2 - esx1;
|
|
|
|
const boundsCurrentHeight = esy2 - esy1;
|
|
|
|
|
|
|
|
// It's important we set the initial scale value based on the width and height at resize start,
|
|
|
|
// otherwise previous dimensions affected by modifiers will be taken into account.
|
|
|
|
const atStartBoundsWidth = startBottomRight[0] - startTopLeft[0];
|
|
|
|
const atStartBoundsHeight = startBottomRight[1] - startTopLeft[1];
|
|
|
|
let scaleX = atStartBoundsWidth / boundsCurrentWidth;
|
|
|
|
let scaleY = atStartBoundsHeight / boundsCurrentHeight;
|
|
|
|
|
2022-02-22 18:45:59 +05:30
|
|
|
let boundTextFont: { fontSize?: number; baseline?: number } = {};
|
|
|
|
const boundTextElement = getBoundTextElement(element);
|
|
|
|
|
2020-12-06 22:39:31 +00:00
|
|
|
if (transformHandleDirection.includes("e")) {
|
2021-01-16 18:49:13 +00:00
|
|
|
scaleX = (rotatedPointer[0] - startTopLeft[0]) / boundsCurrentWidth;
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
if (transformHandleDirection.includes("s")) {
|
2021-01-16 18:49:13 +00:00
|
|
|
scaleY = (rotatedPointer[1] - startTopLeft[1]) / boundsCurrentHeight;
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
if (transformHandleDirection.includes("w")) {
|
2021-01-16 18:49:13 +00:00
|
|
|
scaleX = (startBottomRight[0] - rotatedPointer[0]) / boundsCurrentWidth;
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
if (transformHandleDirection.includes("n")) {
|
2021-01-16 18:49:13 +00:00
|
|
|
scaleY = (startBottomRight[1] - rotatedPointer[1]) / boundsCurrentHeight;
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
2022-02-18 18:20:55 +05:30
|
|
|
|
2021-01-16 18:49:13 +00:00
|
|
|
// Linear elements dimensions differ from bounds dimensions
|
|
|
|
const eleInitialWidth = stateAtResizeStart.width;
|
|
|
|
const eleInitialHeight = stateAtResizeStart.height;
|
|
|
|
// We have to use dimensions of element on screen, otherwise the scaling of the
|
|
|
|
// dimensions won't match the cursor for linear elements.
|
|
|
|
let eleNewWidth = element.width * scaleX;
|
|
|
|
let eleNewHeight = element.height * scaleY;
|
2020-12-06 22:39:31 +00:00
|
|
|
|
|
|
|
// adjust dimensions for resizing from center
|
2021-10-21 22:05:48 +02:00
|
|
|
if (shouldResizeFromCenter) {
|
2021-01-16 18:49:13 +00:00
|
|
|
eleNewWidth = 2 * eleNewWidth - eleInitialWidth;
|
|
|
|
eleNewHeight = 2 * eleNewHeight - eleInitialHeight;
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// adjust dimensions to keep sides ratio
|
2021-10-21 22:05:48 +02:00
|
|
|
if (shouldMaintainAspectRatio) {
|
2021-01-16 18:49:13 +00:00
|
|
|
const widthRatio = Math.abs(eleNewWidth) / eleInitialWidth;
|
|
|
|
const heightRatio = Math.abs(eleNewHeight) / eleInitialHeight;
|
2020-12-06 22:39:31 +00:00
|
|
|
if (transformHandleDirection.length === 1) {
|
2021-01-16 18:49:13 +00:00
|
|
|
eleNewHeight *= widthRatio;
|
|
|
|
eleNewWidth *= heightRatio;
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
if (transformHandleDirection.length === 2) {
|
|
|
|
const ratio = Math.max(widthRatio, heightRatio);
|
2021-01-16 18:49:13 +00:00
|
|
|
eleNewWidth = eleInitialWidth * ratio * Math.sign(eleNewWidth);
|
|
|
|
eleNewHeight = eleInitialHeight * ratio * Math.sign(eleNewHeight);
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-18 18:20:55 +05:30
|
|
|
if (boundTextElement) {
|
2022-02-22 18:45:59 +05:30
|
|
|
const stateOfBoundTextElementAtResize = originalElements.get(
|
|
|
|
boundTextElement.id,
|
|
|
|
) as typeof boundTextElement | undefined;
|
|
|
|
if (stateOfBoundTextElementAtResize) {
|
|
|
|
boundTextFont = {
|
|
|
|
fontSize: stateOfBoundTextElementAtResize.fontSize,
|
|
|
|
baseline: stateOfBoundTextElementAtResize.baseline,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
if (shouldMaintainAspectRatio) {
|
|
|
|
const nextFont = measureFontSizeFromWH(
|
|
|
|
boundTextElement,
|
|
|
|
eleNewWidth - BOUND_TEXT_PADDING * 2,
|
|
|
|
eleNewHeight - BOUND_TEXT_PADDING * 2,
|
|
|
|
);
|
|
|
|
if (nextFont === null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
boundTextFont = {
|
|
|
|
fontSize: nextFont.size,
|
|
|
|
baseline: nextFont.baseline,
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
const minWidth = getApproxMinLineWidth(getFontString(boundTextElement));
|
|
|
|
const minHeight = getApproxMinLineHeight(getFontString(boundTextElement));
|
|
|
|
eleNewWidth = Math.ceil(Math.max(eleNewWidth, minWidth));
|
|
|
|
eleNewHeight = Math.ceil(Math.max(eleNewHeight, minHeight));
|
|
|
|
}
|
2022-02-18 18:20:55 +05:30
|
|
|
}
|
|
|
|
|
2021-11-01 15:24:05 +02:00
|
|
|
const [newBoundsX1, newBoundsY1, newBoundsX2, newBoundsY2] =
|
|
|
|
getResizedElementAbsoluteCoords(
|
|
|
|
stateAtResizeStart,
|
|
|
|
eleNewWidth,
|
|
|
|
eleNewHeight,
|
2022-08-16 21:51:43 +02:00
|
|
|
true,
|
2021-11-01 15:24:05 +02:00
|
|
|
);
|
2021-01-16 18:49:13 +00:00
|
|
|
const newBoundsWidth = newBoundsX2 - newBoundsX1;
|
|
|
|
const newBoundsHeight = newBoundsY2 - newBoundsY1;
|
|
|
|
|
2020-12-06 22:39:31 +00:00
|
|
|
// Calculate new topLeft based on fixed corner during resize
|
2021-01-16 18:49:13 +00:00
|
|
|
let newTopLeft = [...startTopLeft] as [number, number];
|
2020-12-06 22:39:31 +00:00
|
|
|
if (["n", "w", "nw"].includes(transformHandleDirection)) {
|
|
|
|
newTopLeft = [
|
2021-01-16 18:49:13 +00:00
|
|
|
startBottomRight[0] - Math.abs(newBoundsWidth),
|
|
|
|
startBottomRight[1] - Math.abs(newBoundsHeight),
|
2020-12-06 22:39:31 +00:00
|
|
|
];
|
|
|
|
}
|
|
|
|
if (transformHandleDirection === "ne") {
|
2021-01-16 18:49:13 +00:00
|
|
|
const bottomLeft = [startTopLeft[0], startBottomRight[1]];
|
|
|
|
newTopLeft = [bottomLeft[0], bottomLeft[1] - Math.abs(newBoundsHeight)];
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
if (transformHandleDirection === "sw") {
|
2021-01-16 18:49:13 +00:00
|
|
|
const topRight = [startBottomRight[0], startTopLeft[1]];
|
|
|
|
newTopLeft = [topRight[0] - Math.abs(newBoundsWidth), topRight[1]];
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Keeps opposite handle fixed during resize
|
2021-10-21 22:05:48 +02:00
|
|
|
if (shouldMaintainAspectRatio) {
|
2020-12-06 22:39:31 +00:00
|
|
|
if (["s", "n"].includes(transformHandleDirection)) {
|
2021-01-16 18:49:13 +00:00
|
|
|
newTopLeft[0] = startCenter[0] - newBoundsWidth / 2;
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
if (["e", "w"].includes(transformHandleDirection)) {
|
2021-01-16 18:49:13 +00:00
|
|
|
newTopLeft[1] = startCenter[1] - newBoundsHeight / 2;
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Flip horizontally
|
2021-01-16 18:49:13 +00:00
|
|
|
if (eleNewWidth < 0) {
|
2020-12-06 22:39:31 +00:00
|
|
|
if (transformHandleDirection.includes("e")) {
|
2021-01-16 18:49:13 +00:00
|
|
|
newTopLeft[0] -= Math.abs(newBoundsWidth);
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
if (transformHandleDirection.includes("w")) {
|
2021-01-16 18:49:13 +00:00
|
|
|
newTopLeft[0] += Math.abs(newBoundsWidth);
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// Flip vertically
|
2021-01-16 18:49:13 +00:00
|
|
|
if (eleNewHeight < 0) {
|
2020-12-06 22:39:31 +00:00
|
|
|
if (transformHandleDirection.includes("s")) {
|
2021-01-16 18:49:13 +00:00
|
|
|
newTopLeft[1] -= Math.abs(newBoundsHeight);
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
if (transformHandleDirection.includes("n")) {
|
2021-01-16 18:49:13 +00:00
|
|
|
newTopLeft[1] += Math.abs(newBoundsHeight);
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-21 22:05:48 +02:00
|
|
|
if (shouldResizeFromCenter) {
|
2021-01-16 18:49:13 +00:00
|
|
|
newTopLeft[0] = startCenter[0] - Math.abs(newBoundsWidth) / 2;
|
|
|
|
newTopLeft[1] = startCenter[1] - Math.abs(newBoundsHeight) / 2;
|
2020-12-06 22:39:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// adjust topLeft to new rotation point
|
|
|
|
const angle = stateAtResizeStart.angle;
|
|
|
|
const rotatedTopLeft = rotatePoint(newTopLeft, startCenter, angle);
|
|
|
|
const newCenter: Point = [
|
2021-01-16 18:49:13 +00:00
|
|
|
newTopLeft[0] + Math.abs(newBoundsWidth) / 2,
|
|
|
|
newTopLeft[1] + Math.abs(newBoundsHeight) / 2,
|
2020-12-06 22:39:31 +00:00
|
|
|
];
|
|
|
|
const rotatedNewCenter = rotatePoint(newCenter, startCenter, angle);
|
|
|
|
newTopLeft = rotatePoint(rotatedTopLeft, rotatedNewCenter, -angle);
|
|
|
|
|
2021-01-16 18:49:13 +00:00
|
|
|
// Readjust points for linear elements
|
|
|
|
const rescaledPoints = rescalePointsInElement(
|
|
|
|
stateAtResizeStart,
|
|
|
|
eleNewWidth,
|
|
|
|
eleNewHeight,
|
2022-08-16 21:51:43 +02:00
|
|
|
true,
|
2020-05-05 00:25:40 +09:00
|
|
|
);
|
2021-01-16 18:49:13 +00:00
|
|
|
// For linear elements (x,y) are the coordinates of the first drawn point not the top-left corner
|
|
|
|
// So we need to readjust (x,y) to be where the first point should be
|
|
|
|
const newOrigin = [...newTopLeft];
|
|
|
|
newOrigin[0] += stateAtResizeStart.x - newBoundsX1;
|
|
|
|
newOrigin[1] += stateAtResizeStart.y - newBoundsY1;
|
2020-12-06 22:39:31 +00:00
|
|
|
|
2021-01-16 18:49:13 +00:00
|
|
|
const resizedElement = {
|
|
|
|
width: Math.abs(eleNewWidth),
|
|
|
|
height: Math.abs(eleNewHeight),
|
|
|
|
x: newOrigin[0],
|
|
|
|
y: newOrigin[1],
|
|
|
|
...rescaledPoints,
|
|
|
|
};
|
2020-12-06 22:39:31 +00:00
|
|
|
|
2021-10-21 22:05:48 +02:00
|
|
|
if ("scale" in element && "scale" in stateAtResizeStart) {
|
|
|
|
mutateElement(element, {
|
|
|
|
scale: [
|
|
|
|
// defaulting because scaleX/Y can be 0/-0
|
|
|
|
(Math.sign(scaleX) || stateAtResizeStart.scale[0]) *
|
|
|
|
stateAtResizeStart.scale[0],
|
|
|
|
(Math.sign(scaleY) || stateAtResizeStart.scale[1]) *
|
|
|
|
stateAtResizeStart.scale[1],
|
|
|
|
],
|
|
|
|
});
|
|
|
|
}
|
2022-02-18 18:20:55 +05:30
|
|
|
|
2020-05-05 00:25:40 +09:00
|
|
|
if (
|
2022-02-18 18:20:55 +05:30
|
|
|
resizedElement.width !== 0 &&
|
2021-01-16 18:49:13 +00:00
|
|
|
resizedElement.height !== 0 &&
|
|
|
|
Number.isFinite(resizedElement.x) &&
|
|
|
|
Number.isFinite(resizedElement.y)
|
2020-05-05 00:25:40 +09:00
|
|
|
) {
|
2021-01-16 18:49:13 +00:00
|
|
|
updateBoundElements(element, {
|
|
|
|
newSize: { width: resizedElement.width, height: resizedElement.height },
|
2020-05-05 00:25:40 +09:00
|
|
|
});
|
2021-01-16 18:49:13 +00:00
|
|
|
mutateElement(element, resizedElement);
|
2022-02-22 18:45:59 +05:30
|
|
|
if (boundTextElement && boundTextFont) {
|
|
|
|
mutateElement(boundTextElement, { fontSize: boundTextFont.fontSize });
|
|
|
|
}
|
2022-02-21 17:15:29 +05:30
|
|
|
handleBindTextResize(element, transformHandleDirection);
|
2020-05-05 00:25:40 +09:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const resizeMultipleElements = (
|
2022-08-13 22:53:10 +05:00
|
|
|
pointerDownState: PointerDownState,
|
|
|
|
selectedElements: readonly NonDeletedExcalidrawElement[],
|
2020-08-10 14:16:39 +02:00
|
|
|
transformHandleType: "nw" | "ne" | "sw" | "se",
|
2022-08-13 22:53:10 +05:00
|
|
|
shouldResizeFromCenter: boolean,
|
2020-05-09 17:57:00 +09:00
|
|
|
pointerX: number,
|
|
|
|
pointerY: number,
|
2020-05-05 00:25:40 +09:00
|
|
|
) => {
|
2022-08-13 22:53:10 +05:00
|
|
|
// map selected elements to the original elements. While it never should
|
|
|
|
// happen that pointerDownState.originalElements won't contain the selected
|
|
|
|
// elements during resize, this coupling isn't guaranteed, so to ensure
|
|
|
|
// type safety we need to transform only those elements we filter.
|
|
|
|
const targetElements = selectedElements.reduce(
|
|
|
|
(
|
|
|
|
acc: {
|
|
|
|
/** element at resize start */
|
|
|
|
orig: NonDeletedExcalidrawElement;
|
|
|
|
/** latest element */
|
|
|
|
latest: NonDeletedExcalidrawElement;
|
|
|
|
}[],
|
|
|
|
element,
|
|
|
|
) => {
|
|
|
|
const origElement = pointerDownState.originalElements.get(element.id);
|
|
|
|
if (origElement) {
|
|
|
|
acc.push({ orig: origElement, latest: element });
|
|
|
|
}
|
|
|
|
return acc;
|
|
|
|
},
|
|
|
|
[],
|
|
|
|
);
|
|
|
|
|
|
|
|
const { minX, minY, maxX, maxY, midX, midY } = getCommonBoundingBox(
|
|
|
|
targetElements.map(({ orig }) => orig),
|
|
|
|
);
|
|
|
|
const direction = transformHandleType;
|
|
|
|
|
|
|
|
const mapDirectionsToAnchors: Record<typeof direction, Point> = {
|
|
|
|
ne: [minX, maxY],
|
|
|
|
se: [minX, minY],
|
|
|
|
sw: [maxX, minY],
|
|
|
|
nw: [maxX, maxY],
|
|
|
|
};
|
|
|
|
|
|
|
|
// anchor point must be on the opposite side of the dragged selection handle
|
|
|
|
// or be the center of the selection if alt is pressed
|
|
|
|
const [anchorX, anchorY]: Point = shouldResizeFromCenter
|
|
|
|
? [midX, midY]
|
|
|
|
: mapDirectionsToAnchors[direction];
|
|
|
|
|
|
|
|
const mapDirectionsToPointerSides: Record<
|
|
|
|
typeof direction,
|
|
|
|
[x: boolean, y: boolean]
|
|
|
|
> = {
|
|
|
|
ne: [pointerX >= anchorX, pointerY <= anchorY],
|
|
|
|
se: [pointerX >= anchorX, pointerY >= anchorY],
|
|
|
|
sw: [pointerX <= anchorX, pointerY >= anchorY],
|
|
|
|
nw: [pointerX <= anchorX, pointerY <= anchorY],
|
|
|
|
};
|
|
|
|
|
|
|
|
// pointer side relative to anchor
|
|
|
|
const [pointerSideX, pointerSideY] = mapDirectionsToPointerSides[
|
|
|
|
direction
|
|
|
|
].map((condition) => (condition ? 1 : -1));
|
|
|
|
|
|
|
|
// stop resizing if a pointer is on the other side of selection
|
|
|
|
if (pointerSideX < 0 && pointerSideY < 0) {
|
|
|
|
return;
|
2020-06-08 18:25:20 +09:00
|
|
|
}
|
2022-02-21 16:46:39 +05:30
|
|
|
|
2022-08-13 22:53:10 +05:00
|
|
|
const scale =
|
|
|
|
Math.max(
|
|
|
|
(pointerSideX * Math.abs(pointerX - anchorX)) / (maxX - minX),
|
|
|
|
(pointerSideY * Math.abs(pointerY - anchorY)) / (maxY - minY),
|
|
|
|
) * (shouldResizeFromCenter ? 2 : 1);
|
2022-02-21 16:46:39 +05:30
|
|
|
|
2022-10-29 16:01:38 +05:00
|
|
|
if (scale === 0) {
|
2022-08-13 22:53:10 +05:00
|
|
|
return;
|
|
|
|
}
|
2021-05-09 16:42:10 +01:00
|
|
|
|
2022-08-13 22:53:10 +05:00
|
|
|
targetElements.forEach((element) => {
|
|
|
|
const width = element.orig.width * scale;
|
|
|
|
const height = element.orig.height * scale;
|
|
|
|
const x = anchorX + (element.orig.x - anchorX) * scale;
|
|
|
|
const y = anchorY + (element.orig.y - anchorY) * scale;
|
|
|
|
|
|
|
|
// readjust points for linear & free draw elements
|
2022-08-16 21:51:43 +02:00
|
|
|
const rescaledPoints = rescalePointsInElement(
|
|
|
|
element.orig,
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
false,
|
|
|
|
);
|
2022-08-13 22:53:10 +05:00
|
|
|
|
|
|
|
const update: {
|
|
|
|
width: number;
|
|
|
|
height: number;
|
|
|
|
x: number;
|
|
|
|
y: number;
|
|
|
|
points?: Point[];
|
|
|
|
fontSize?: number;
|
|
|
|
baseline?: number;
|
|
|
|
} = {
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
x,
|
|
|
|
y,
|
|
|
|
...rescaledPoints,
|
|
|
|
};
|
2021-05-09 16:42:10 +01:00
|
|
|
|
2022-08-13 22:53:10 +05:00
|
|
|
let boundTextUpdates: { fontSize: number; baseline: number } | null = null;
|
2021-05-09 16:42:10 +01:00
|
|
|
|
2022-08-13 22:53:10 +05:00
|
|
|
const boundTextElement = getBoundTextElement(element.latest);
|
|
|
|
|
|
|
|
if (boundTextElement || isTextElement(element.orig)) {
|
|
|
|
const optionalPadding = boundTextElement ? BOUND_TEXT_PADDING * 2 : 0;
|
|
|
|
const textMeasurements = measureFontSizeFromWH(
|
|
|
|
boundTextElement ?? (element.orig as ExcalidrawTextElement),
|
|
|
|
width - optionalPadding,
|
|
|
|
height - optionalPadding,
|
|
|
|
);
|
2022-10-29 16:01:38 +05:00
|
|
|
|
|
|
|
if (!textMeasurements) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isTextElement(element.orig)) {
|
|
|
|
update.fontSize = textMeasurements.size;
|
|
|
|
update.baseline = textMeasurements.baseline;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (boundTextElement) {
|
|
|
|
boundTextUpdates = {
|
|
|
|
fontSize: textMeasurements.size,
|
|
|
|
baseline: textMeasurements.baseline,
|
|
|
|
};
|
2022-08-13 22:53:10 +05:00
|
|
|
}
|
2020-04-07 17:49:59 +09:00
|
|
|
}
|
2022-08-13 22:53:10 +05:00
|
|
|
|
2022-10-29 16:01:38 +05:00
|
|
|
updateBoundElements(element.latest, { newSize: { width, height } });
|
|
|
|
|
2022-08-13 22:53:10 +05:00
|
|
|
mutateElement(element.latest, update);
|
|
|
|
|
|
|
|
if (boundTextElement && boundTextUpdates) {
|
|
|
|
mutateElement(boundTextElement, boundTextUpdates);
|
|
|
|
handleBindTextResize(element.latest, transformHandleType);
|
|
|
|
}
|
|
|
|
});
|
2020-04-09 16:14:32 +01:00
|
|
|
};
|
2020-04-07 17:49:59 +09:00
|
|
|
|
2020-07-26 19:21:38 +09:00
|
|
|
const rotateMultipleElements = (
|
2020-09-11 17:22:40 +02:00
|
|
|
pointerDownState: PointerDownState,
|
2020-07-26 19:21:38 +09:00
|
|
|
elements: readonly NonDeletedExcalidrawElement[],
|
|
|
|
pointerX: number,
|
|
|
|
pointerY: number,
|
2021-10-21 22:05:48 +02:00
|
|
|
shouldRotateWithDiscreteAngle: boolean,
|
2020-07-26 19:21:38 +09:00
|
|
|
centerX: number,
|
|
|
|
centerY: number,
|
|
|
|
) => {
|
|
|
|
let centerAngle =
|
|
|
|
(5 * Math.PI) / 2 + Math.atan2(pointerY - centerY, pointerX - centerX);
|
2021-10-21 22:05:48 +02:00
|
|
|
if (shouldRotateWithDiscreteAngle) {
|
2020-07-26 19:21:38 +09:00
|
|
|
centerAngle += SHIFT_LOCKING_ANGLE / 2;
|
|
|
|
centerAngle -= centerAngle % SHIFT_LOCKING_ANGLE;
|
|
|
|
}
|
|
|
|
elements.forEach((element, index) => {
|
2020-08-28 17:20:06 +09:00
|
|
|
const [x1, y1, x2, y2] = getElementAbsoluteCoords(element);
|
|
|
|
const cx = (x1 + x2) / 2;
|
|
|
|
const cy = (y1 + y2) / 2;
|
2020-09-11 17:22:40 +02:00
|
|
|
const origAngle =
|
|
|
|
pointerDownState.originalElements.get(element.id)?.angle ?? element.angle;
|
2020-08-28 17:20:06 +09:00
|
|
|
const [rotatedCX, rotatedCY] = rotate(
|
|
|
|
cx,
|
|
|
|
cy,
|
|
|
|
centerX,
|
|
|
|
centerY,
|
2020-09-11 17:22:40 +02:00
|
|
|
centerAngle + origAngle - element.angle,
|
2020-08-28 17:20:06 +09:00
|
|
|
);
|
|
|
|
mutateElement(element, {
|
|
|
|
x: element.x + (rotatedCX - cx),
|
|
|
|
y: element.y + (rotatedCY - cy),
|
2020-09-11 17:22:40 +02:00
|
|
|
angle: normalizeAngle(centerAngle + origAngle),
|
2020-08-28 17:20:06 +09:00
|
|
|
});
|
2021-12-16 21:14:03 +05:30
|
|
|
const boundTextElementId = getBoundTextElementId(element);
|
|
|
|
if (boundTextElementId) {
|
|
|
|
const textElement =
|
|
|
|
Scene.getScene(element)!.getElement(boundTextElementId)!;
|
|
|
|
mutateElement(textElement, {
|
|
|
|
x: textElement.x + (rotatedCX - cx),
|
|
|
|
y: textElement.y + (rotatedCY - cy),
|
|
|
|
angle: normalizeAngle(centerAngle + origAngle),
|
|
|
|
});
|
|
|
|
}
|
2020-07-26 19:21:38 +09:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2020-05-09 17:57:00 +09:00
|
|
|
export const getResizeOffsetXY = (
|
2020-08-10 14:16:39 +02:00
|
|
|
transformHandleType: MaybeTransformHandleType,
|
2020-05-09 17:57:00 +09:00
|
|
|
selectedElements: NonDeletedExcalidrawElement[],
|
|
|
|
x: number,
|
|
|
|
y: number,
|
|
|
|
): [number, number] => {
|
|
|
|
const [x1, y1, x2, y2] =
|
|
|
|
selectedElements.length === 1
|
|
|
|
? getElementAbsoluteCoords(selectedElements[0])
|
|
|
|
: getCommonBounds(selectedElements);
|
|
|
|
const cx = (x1 + x2) / 2;
|
|
|
|
const cy = (y1 + y2) / 2;
|
|
|
|
const angle = selectedElements.length === 1 ? selectedElements[0].angle : 0;
|
|
|
|
[x, y] = rotate(x, y, cx, cy, -angle);
|
2020-08-10 14:16:39 +02:00
|
|
|
switch (transformHandleType) {
|
2020-05-09 17:57:00 +09:00
|
|
|
case "n":
|
|
|
|
return rotate(x - (x1 + x2) / 2, y - y1, 0, 0, angle);
|
|
|
|
case "s":
|
|
|
|
return rotate(x - (x1 + x2) / 2, y - y2, 0, 0, angle);
|
|
|
|
case "w":
|
|
|
|
return rotate(x - x1, y - (y1 + y2) / 2, 0, 0, angle);
|
|
|
|
case "e":
|
|
|
|
return rotate(x - x2, y - (y1 + y2) / 2, 0, 0, angle);
|
|
|
|
case "nw":
|
|
|
|
return rotate(x - x1, y - y1, 0, 0, angle);
|
|
|
|
case "ne":
|
|
|
|
return rotate(x - x2, y - y1, 0, 0, angle);
|
|
|
|
case "sw":
|
|
|
|
return rotate(x - x1, y - y2, 0, 0, angle);
|
|
|
|
case "se":
|
|
|
|
return rotate(x - x2, y - y2, 0, 0, angle);
|
|
|
|
default:
|
|
|
|
return [0, 0];
|
|
|
|
}
|
|
|
|
};
|
2020-05-11 00:41:36 +09:00
|
|
|
|
|
|
|
export const getResizeArrowDirection = (
|
2020-08-10 14:16:39 +02:00
|
|
|
transformHandleType: MaybeTransformHandleType,
|
2020-05-11 00:41:36 +09:00
|
|
|
element: NonDeleted<ExcalidrawLinearElement>,
|
|
|
|
): "origin" | "end" => {
|
|
|
|
const [, [px, py]] = element.points;
|
|
|
|
const isResizeEnd =
|
2020-08-10 14:16:39 +02:00
|
|
|
(transformHandleType === "nw" && (px < 0 || py < 0)) ||
|
|
|
|
(transformHandleType === "ne" && px >= 0) ||
|
|
|
|
(transformHandleType === "sw" && px <= 0) ||
|
|
|
|
(transformHandleType === "se" && (px > 0 || py > 0));
|
2020-05-11 00:41:36 +09:00
|
|
|
return isResizeEnd ? "end" : "origin";
|
|
|
|
};
|