
* resize single elements from the side * fix lint * do not resize texts from the sides (for we want to wrap/unwrap) * omit side handles for frames too * upgrade types * enable resizing from the sides for multiple elements as well * fix lint * maintain aspect ratio when elements are not of the same angle * lint * always resize proportionally for multiple elements * increase side resizing padding * code cleanup * adaptive handles * do not resize for linear elements with only two points * prioritize point dragging over edge resizing * lint * allow free resizing for multiple elements at degree 0 * always resize from the sides * reduce hit threshold * make small multiple elements movable * lint * show side handles on touch screen and mobile devices * differentiate touchscreens * keep proportional with text in multi-element resizing * update snapshot * update multi elements resizing logic * lint * reduce side resizing padding * bound texts do not scale in normal cases * lint * test sides for texts * wrap text * do not update text size when changing its alignment * keep text wrapped/unwrapped when editing * change wrapped size to auto size from context menu * fix test * lint * increase min width for wrapped texts * wrap wrapped text in container * unwrap when binding text to container * rename `wrapped` to `autoResize` * fix lint * revert: use `center` align when wrapping text in container * update snaps * fix lint * simplify logic on autoResize * lint and test * snapshots * remove unnecessary code * snapshots * fix: defaults not set correctly * tests for wrapping texts when resized * tests for text wrapping when edited * fix autoResize refactor * include autoResize flag check * refactor * feat: rename action label & change contextmenu position * fix: update version on `autoResize` action * fix infinite loop when editing text in a container * simplify * always maintain `width` if `!autoResize` * maintain `x` if `!autoResize` * maintain `y` pos after fontSize change if `!autoResize` * refactor * when editing, do not wrap text in textWysiwyg * simplify text editor * make test more readable * comment * rename action to match file name * revert function signature change * only update in app --------- Co-authored-by: dwelle <5153846+dwelle@users.noreply.github.com>
321 lines
7.7 KiB
TypeScript
321 lines
7.7 KiB
TypeScript
import type {
|
|
ElementsMap,
|
|
ExcalidrawElement,
|
|
NonDeletedExcalidrawElement,
|
|
PointerType,
|
|
} from "./types";
|
|
|
|
import type { Bounds } from "./bounds";
|
|
import { getElementAbsoluteCoords } from "./bounds";
|
|
import { rotate } from "../math";
|
|
import type { Device, InteractiveCanvasAppState, Zoom } from "../types";
|
|
import { isFrameLikeElement, isLinearElement } from "./typeChecks";
|
|
import {
|
|
DEFAULT_TRANSFORM_HANDLE_SPACING,
|
|
isAndroid,
|
|
isIOS,
|
|
} from "../constants";
|
|
|
|
export type TransformHandleDirection =
|
|
| "n"
|
|
| "s"
|
|
| "w"
|
|
| "e"
|
|
| "nw"
|
|
| "ne"
|
|
| "sw"
|
|
| "se";
|
|
|
|
export type TransformHandleType = TransformHandleDirection | "rotation";
|
|
|
|
export type TransformHandle = Bounds;
|
|
export type TransformHandles = Partial<{
|
|
[T in TransformHandleType]: TransformHandle;
|
|
}>;
|
|
export type MaybeTransformHandleType = TransformHandleType | false;
|
|
|
|
const transformHandleSizes: { [k in PointerType]: number } = {
|
|
mouse: 8,
|
|
pen: 16,
|
|
touch: 28,
|
|
};
|
|
|
|
const ROTATION_RESIZE_HANDLE_GAP = 16;
|
|
|
|
export const DEFAULT_OMIT_SIDES = {
|
|
e: true,
|
|
s: true,
|
|
n: true,
|
|
w: true,
|
|
};
|
|
|
|
export const OMIT_SIDES_FOR_MULTIPLE_ELEMENTS = {
|
|
e: true,
|
|
s: true,
|
|
n: true,
|
|
w: true,
|
|
};
|
|
|
|
export const OMIT_SIDES_FOR_FRAME = {
|
|
e: true,
|
|
s: true,
|
|
n: true,
|
|
w: true,
|
|
rotation: true,
|
|
};
|
|
|
|
const OMIT_SIDES_FOR_LINE_SLASH = {
|
|
e: true,
|
|
s: true,
|
|
n: true,
|
|
w: true,
|
|
nw: true,
|
|
se: true,
|
|
};
|
|
|
|
const OMIT_SIDES_FOR_LINE_BACKSLASH = {
|
|
e: true,
|
|
s: true,
|
|
n: true,
|
|
w: true,
|
|
};
|
|
|
|
const generateTransformHandle = (
|
|
x: number,
|
|
y: number,
|
|
width: number,
|
|
height: number,
|
|
cx: number,
|
|
cy: number,
|
|
angle: number,
|
|
): TransformHandle => {
|
|
const [xx, yy] = rotate(x + width / 2, y + height / 2, cx, cy, angle);
|
|
return [xx - width / 2, yy - height / 2, width, height];
|
|
};
|
|
|
|
export const canResizeFromSides = (device: Device) => {
|
|
if (device.viewport.isMobile) {
|
|
return false;
|
|
}
|
|
|
|
if (device.isTouchScreen && (isAndroid || isIOS)) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
export const getOmitSidesForDevice = (device: Device) => {
|
|
if (canResizeFromSides(device)) {
|
|
return DEFAULT_OMIT_SIDES;
|
|
}
|
|
|
|
return {};
|
|
};
|
|
|
|
export const getTransformHandlesFromCoords = (
|
|
[x1, y1, x2, y2, cx, cy]: [number, number, number, number, number, number],
|
|
angle: number,
|
|
zoom: Zoom,
|
|
pointerType: PointerType,
|
|
omitSides: { [T in TransformHandleType]?: boolean } = {},
|
|
margin = 4,
|
|
): TransformHandles => {
|
|
const size = transformHandleSizes[pointerType];
|
|
const handleWidth = size / zoom.value;
|
|
const handleHeight = size / zoom.value;
|
|
|
|
const handleMarginX = size / zoom.value;
|
|
const handleMarginY = size / zoom.value;
|
|
|
|
const width = x2 - x1;
|
|
const height = y2 - y1;
|
|
const dashedLineMargin = margin / zoom.value;
|
|
const centeringOffset =
|
|
(size - DEFAULT_TRANSFORM_HANDLE_SPACING * 2) / (2 * zoom.value);
|
|
|
|
const transformHandles: TransformHandles = {
|
|
nw: omitSides.nw
|
|
? undefined
|
|
: generateTransformHandle(
|
|
x1 - dashedLineMargin - handleMarginX + centeringOffset,
|
|
y1 - dashedLineMargin - handleMarginY + centeringOffset,
|
|
handleWidth,
|
|
handleHeight,
|
|
cx,
|
|
cy,
|
|
angle,
|
|
),
|
|
ne: omitSides.ne
|
|
? undefined
|
|
: generateTransformHandle(
|
|
x2 + dashedLineMargin - centeringOffset,
|
|
y1 - dashedLineMargin - handleMarginY + centeringOffset,
|
|
handleWidth,
|
|
handleHeight,
|
|
cx,
|
|
cy,
|
|
angle,
|
|
),
|
|
sw: omitSides.sw
|
|
? undefined
|
|
: generateTransformHandle(
|
|
x1 - dashedLineMargin - handleMarginX + centeringOffset,
|
|
y2 + dashedLineMargin - centeringOffset,
|
|
handleWidth,
|
|
handleHeight,
|
|
cx,
|
|
cy,
|
|
angle,
|
|
),
|
|
se: omitSides.se
|
|
? undefined
|
|
: generateTransformHandle(
|
|
x2 + dashedLineMargin - centeringOffset,
|
|
y2 + dashedLineMargin - centeringOffset,
|
|
handleWidth,
|
|
handleHeight,
|
|
cx,
|
|
cy,
|
|
angle,
|
|
),
|
|
rotation: omitSides.rotation
|
|
? undefined
|
|
: generateTransformHandle(
|
|
x1 + width / 2 - handleWidth / 2,
|
|
y1 -
|
|
dashedLineMargin -
|
|
handleMarginY +
|
|
centeringOffset -
|
|
ROTATION_RESIZE_HANDLE_GAP / zoom.value,
|
|
handleWidth,
|
|
handleHeight,
|
|
cx,
|
|
cy,
|
|
angle,
|
|
),
|
|
};
|
|
|
|
// We only want to show height handles (all cardinal directions) above a certain size
|
|
// Note: we render using "mouse" size so we should also use "mouse" size for this check
|
|
const minimumSizeForEightHandles =
|
|
(5 * transformHandleSizes.mouse) / zoom.value;
|
|
if (Math.abs(width) > minimumSizeForEightHandles) {
|
|
if (!omitSides.n) {
|
|
transformHandles.n = generateTransformHandle(
|
|
x1 + width / 2 - handleWidth / 2,
|
|
y1 - dashedLineMargin - handleMarginY + centeringOffset,
|
|
handleWidth,
|
|
handleHeight,
|
|
cx,
|
|
cy,
|
|
angle,
|
|
);
|
|
}
|
|
if (!omitSides.s) {
|
|
transformHandles.s = generateTransformHandle(
|
|
x1 + width / 2 - handleWidth / 2,
|
|
y2 + dashedLineMargin - centeringOffset,
|
|
handleWidth,
|
|
handleHeight,
|
|
cx,
|
|
cy,
|
|
angle,
|
|
);
|
|
}
|
|
}
|
|
if (Math.abs(height) > minimumSizeForEightHandles) {
|
|
if (!omitSides.w) {
|
|
transformHandles.w = generateTransformHandle(
|
|
x1 - dashedLineMargin - handleMarginX + centeringOffset,
|
|
y1 + height / 2 - handleHeight / 2,
|
|
handleWidth,
|
|
handleHeight,
|
|
cx,
|
|
cy,
|
|
angle,
|
|
);
|
|
}
|
|
if (!omitSides.e) {
|
|
transformHandles.e = generateTransformHandle(
|
|
x2 + dashedLineMargin - centeringOffset,
|
|
y1 + height / 2 - handleHeight / 2,
|
|
handleWidth,
|
|
handleHeight,
|
|
cx,
|
|
cy,
|
|
angle,
|
|
);
|
|
}
|
|
}
|
|
|
|
return transformHandles;
|
|
};
|
|
|
|
export const getTransformHandles = (
|
|
element: ExcalidrawElement,
|
|
zoom: Zoom,
|
|
elementsMap: ElementsMap,
|
|
pointerType: PointerType = "mouse",
|
|
omitSides: { [T in TransformHandleType]?: boolean } = DEFAULT_OMIT_SIDES,
|
|
): TransformHandles => {
|
|
// so that when locked element is selected (especially when you toggle lock
|
|
// via keyboard) the locked element is visually distinct, indicating
|
|
// you can't move/resize
|
|
if (element.locked) {
|
|
return {};
|
|
}
|
|
|
|
if (element.type === "freedraw" || isLinearElement(element)) {
|
|
if (element.points.length === 2) {
|
|
// only check the last point because starting point is always (0,0)
|
|
const [, p1] = element.points;
|
|
if (p1[0] === 0 || p1[1] === 0) {
|
|
omitSides = OMIT_SIDES_FOR_LINE_BACKSLASH;
|
|
} else if (p1[0] > 0 && p1[1] < 0) {
|
|
omitSides = OMIT_SIDES_FOR_LINE_SLASH;
|
|
} else if (p1[0] > 0 && p1[1] > 0) {
|
|
omitSides = OMIT_SIDES_FOR_LINE_BACKSLASH;
|
|
} else if (p1[0] < 0 && p1[1] > 0) {
|
|
omitSides = OMIT_SIDES_FOR_LINE_SLASH;
|
|
} else if (p1[0] < 0 && p1[1] < 0) {
|
|
omitSides = OMIT_SIDES_FOR_LINE_BACKSLASH;
|
|
}
|
|
}
|
|
} else if (isFrameLikeElement(element)) {
|
|
omitSides = {
|
|
...omitSides,
|
|
rotation: true,
|
|
};
|
|
}
|
|
const dashedLineMargin = isLinearElement(element)
|
|
? DEFAULT_TRANSFORM_HANDLE_SPACING + 8
|
|
: DEFAULT_TRANSFORM_HANDLE_SPACING;
|
|
return getTransformHandlesFromCoords(
|
|
getElementAbsoluteCoords(element, elementsMap, true),
|
|
element.angle,
|
|
zoom,
|
|
pointerType,
|
|
omitSides,
|
|
dashedLineMargin,
|
|
);
|
|
};
|
|
|
|
export const shouldShowBoundingBox = (
|
|
elements: readonly NonDeletedExcalidrawElement[],
|
|
appState: InteractiveCanvasAppState,
|
|
) => {
|
|
if (appState.editingLinearElement) {
|
|
return false;
|
|
}
|
|
if (elements.length > 1) {
|
|
return true;
|
|
}
|
|
const element = elements[0];
|
|
if (!isLinearElement(element)) {
|
|
return true;
|
|
}
|
|
|
|
return element.points.length > 2;
|
|
};
|