feat: Add option to flip single element on the context menu (#2520)
Co-authored-by: dwelle <luzar.david@gmail.com>pull/3320/head
parent
458e6d6c24
commit
b0d7ff290f
@ -0,0 +1,207 @@
|
||||
import { register } from "./register";
|
||||
import { getSelectedElements } from "../scene";
|
||||
import { getElementMap, getNonDeletedElements } from "../element";
|
||||
import { mutateElement } from "../element/mutateElement";
|
||||
import { ExcalidrawElement, NonDeleted } from "../element/types";
|
||||
import { normalizeAngle, resizeSingleElement } from "../element/resizeElements";
|
||||
import { AppState } from "../types";
|
||||
import { getTransformHandles } from "../element/transformHandles";
|
||||
import { isLinearElement } from "../element/typeChecks";
|
||||
import { updateBoundElements } from "../element/binding";
|
||||
import { LinearElementEditor } from "../element/linearElementEditor";
|
||||
|
||||
const enableActionFlipHorizontal = (
|
||||
elements: readonly ExcalidrawElement[],
|
||||
appState: AppState,
|
||||
) => {
|
||||
const eligibleElements = getSelectedElements(
|
||||
getNonDeletedElements(elements),
|
||||
appState,
|
||||
);
|
||||
return eligibleElements.length === 1 && eligibleElements[0].type !== "text";
|
||||
};
|
||||
|
||||
const enableActionFlipVertical = (
|
||||
elements: readonly ExcalidrawElement[],
|
||||
appState: AppState,
|
||||
) => {
|
||||
const eligibleElements = getSelectedElements(
|
||||
getNonDeletedElements(elements),
|
||||
appState,
|
||||
);
|
||||
return eligibleElements.length === 1;
|
||||
};
|
||||
|
||||
export const actionFlipHorizontal = register({
|
||||
name: "flipHorizontal",
|
||||
perform: (elements, appState) => {
|
||||
return {
|
||||
elements: flipSelectedElements(elements, appState, "horizontal"),
|
||||
appState,
|
||||
commitToHistory: true,
|
||||
};
|
||||
},
|
||||
keyTest: (event) => event.shiftKey && event.code === "KeyH",
|
||||
contextItemLabel: "labels.flipHorizontal",
|
||||
contextItemPredicate: (elements, appState) =>
|
||||
enableActionFlipHorizontal(elements, appState),
|
||||
});
|
||||
|
||||
export const actionFlipVertical = register({
|
||||
name: "flipVertical",
|
||||
perform: (elements, appState) => {
|
||||
return {
|
||||
elements: flipSelectedElements(elements, appState, "vertical"),
|
||||
appState,
|
||||
commitToHistory: true,
|
||||
};
|
||||
},
|
||||
keyTest: (event) => event.shiftKey && event.code === "KeyV",
|
||||
contextItemLabel: "labels.flipVertical",
|
||||
contextItemPredicate: (elements, appState) =>
|
||||
enableActionFlipVertical(elements, appState),
|
||||
});
|
||||
|
||||
const flipSelectedElements = (
|
||||
elements: readonly ExcalidrawElement[],
|
||||
appState: Readonly<AppState>,
|
||||
flipDirection: "horizontal" | "vertical",
|
||||
) => {
|
||||
const selectedElements = getSelectedElements(
|
||||
getNonDeletedElements(elements),
|
||||
appState,
|
||||
);
|
||||
|
||||
// remove once we allow for groups of elements to be flipped
|
||||
if (selectedElements.length > 1) {
|
||||
return elements;
|
||||
}
|
||||
|
||||
const updatedElements = flipElements(
|
||||
selectedElements,
|
||||
appState,
|
||||
flipDirection,
|
||||
);
|
||||
|
||||
const updatedElementsMap = getElementMap(updatedElements);
|
||||
|
||||
return elements.map((element) => updatedElementsMap[element.id] || element);
|
||||
};
|
||||
|
||||
const flipElements = (
|
||||
elements: NonDeleted<ExcalidrawElement>[],
|
||||
appState: AppState,
|
||||
flipDirection: "horizontal" | "vertical",
|
||||
): ExcalidrawElement[] => {
|
||||
for (let i = 0; i < elements.length; i++) {
|
||||
flipElement(elements[i], appState);
|
||||
// If vertical flip, rotate an extra 180
|
||||
if (flipDirection === "vertical") {
|
||||
rotateElement(elements[i], Math.PI);
|
||||
}
|
||||
}
|
||||
return elements;
|
||||
};
|
||||
|
||||
const flipElement = (
|
||||
element: NonDeleted<ExcalidrawElement>,
|
||||
appState: AppState,
|
||||
) => {
|
||||
const originalX = element.x;
|
||||
const originalY = element.y;
|
||||
const width = element.width;
|
||||
const height = element.height;
|
||||
const originalAngle = normalizeAngle(element.angle);
|
||||
|
||||
let finalOffsetX = 0;
|
||||
if (isLinearElement(element)) {
|
||||
finalOffsetX =
|
||||
element.points.reduce((max, point) => Math.max(max, point[0]), 0) * 2 -
|
||||
element.width;
|
||||
}
|
||||
|
||||
// Rotate back to zero, if necessary
|
||||
mutateElement(element, {
|
||||
angle: normalizeAngle(0),
|
||||
});
|
||||
// Flip unrotated by pulling TransformHandle to opposite side
|
||||
const transformHandles = getTransformHandles(element, appState.zoom);
|
||||
let usingNWHandle = true;
|
||||
let newNCoordsX = 0;
|
||||
let nHandle = transformHandles.nw;
|
||||
if (!nHandle) {
|
||||
// Use ne handle instead
|
||||
usingNWHandle = false;
|
||||
nHandle = transformHandles.ne;
|
||||
if (!nHandle) {
|
||||
mutateElement(element, {
|
||||
angle: originalAngle,
|
||||
});
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (isLinearElement(element)) {
|
||||
for (let i = 1; i < element.points.length; i++) {
|
||||
LinearElementEditor.movePoint(element, i, [
|
||||
-element.points[i][0],
|
||||
element.points[i][1],
|
||||
]);
|
||||
}
|
||||
LinearElementEditor.normalizePoints(element);
|
||||
} else {
|
||||
// calculate new x-coord for transformation
|
||||
newNCoordsX = usingNWHandle ? element.x + 2 * width : element.x - 2 * width;
|
||||
resizeSingleElement(
|
||||
element,
|
||||
true,
|
||||
element,
|
||||
usingNWHandle ? "nw" : "ne",
|
||||
false,
|
||||
newNCoordsX,
|
||||
nHandle[1],
|
||||
);
|
||||
// fix the size to account for handle sizes
|
||||
mutateElement(element, {
|
||||
width,
|
||||
height,
|
||||
});
|
||||
}
|
||||
|
||||
// Rotate by (360 degrees - original angle)
|
||||
let angle = normalizeAngle(2 * Math.PI - originalAngle);
|
||||
if (angle < 0) {
|
||||
// check, probably unnecessary
|
||||
angle = normalizeAngle(angle + 2 * Math.PI);
|
||||
}
|
||||
mutateElement(element, {
|
||||
angle,
|
||||
});
|
||||
|
||||
// Move back to original spot to appear "flipped in place"
|
||||
mutateElement(element, {
|
||||
x: originalX + finalOffsetX,
|
||||
y: originalY,
|
||||
});
|
||||
|
||||
updateBoundElements(element);
|
||||
};
|
||||
|
||||
const rotateElement = (element: ExcalidrawElement, rotationAngle: number) => {
|
||||
const originalX = element.x;
|
||||
const originalY = element.y;
|
||||
let angle = normalizeAngle(element.angle + rotationAngle);
|
||||
if (angle < 0) {
|
||||
// check, probably unnecessary
|
||||
angle = normalizeAngle(2 * Math.PI + angle);
|
||||
}
|
||||
mutateElement(element, {
|
||||
angle,
|
||||
});
|
||||
|
||||
// Move back to original spot
|
||||
mutateElement(element, {
|
||||
x: originalX,
|
||||
y: originalY,
|
||||
});
|
||||
};
|
@ -0,0 +1,615 @@
|
||||
import React from "react";
|
||||
import ReactDOM from "react-dom";
|
||||
import { render } from "./test-utils";
|
||||
import App from "../components/App";
|
||||
import { defaultLang, setLanguage } from "../i18n";
|
||||
import { UI, Pointer } from "./helpers/ui";
|
||||
import { API } from "./helpers/api";
|
||||
import { actionFlipHorizontal, actionFlipVertical } from "../actions";
|
||||
|
||||
const { h } = window;
|
||||
|
||||
const mouse = new Pointer("mouse");
|
||||
|
||||
beforeEach(async () => {
|
||||
// Unmount ReactDOM from root
|
||||
ReactDOM.unmountComponentAtNode(document.getElementById("root")!);
|
||||
mouse.reset();
|
||||
|
||||
await setLanguage(defaultLang);
|
||||
render(<App />);
|
||||
});
|
||||
|
||||
const createAndSelectOneRectangle = (angle: number = 0) => {
|
||||
UI.createElement("rectangle", {
|
||||
x: 0,
|
||||
y: 0,
|
||||
width: 100,
|
||||
height: 50,
|
||||
angle,
|
||||
});
|
||||
};
|
||||
|
||||
const createAndSelectOneDiamond = (angle: number = 0) => {
|
||||
UI.createElement("diamond", {
|
||||
x: 0,
|
||||
y: 0,
|
||||
width: 100,
|
||||
height: 50,
|
||||
angle,
|
||||
});
|
||||
};
|
||||
|
||||
const createAndSelectOneEllipse = (angle: number = 0) => {
|
||||
UI.createElement("ellipse", {
|
||||
x: 0,
|
||||
y: 0,
|
||||
width: 100,
|
||||
height: 50,
|
||||
angle,
|
||||
});
|
||||
};
|
||||
|
||||
const createAndSelectOneArrow = (angle: number = 0) => {
|
||||
UI.createElement("arrow", {
|
||||
x: 0,
|
||||
y: 0,
|
||||
width: 100,
|
||||
height: 50,
|
||||
angle,
|
||||
});
|
||||
};
|
||||
|
||||
const createAndSelectOneLine = (angle: number = 0) => {
|
||||
UI.createElement("line", {
|
||||
x: 0,
|
||||
y: 0,
|
||||
width: 100,
|
||||
height: 50,
|
||||
angle,
|
||||
});
|
||||
};
|
||||
|
||||
const createAndReturnOneDraw = (angle: number = 0) => {
|
||||
return UI.createElement("draw", {
|
||||
x: 0,
|
||||
y: 0,
|
||||
width: 50,
|
||||
height: 100,
|
||||
angle,
|
||||
});
|
||||
};
|
||||
|
||||
// Rectangle element
|
||||
|
||||
it("flips an unrotated rectangle horizontally correctly", () => {
|
||||
createAndSelectOneRectangle();
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips an unrotated rectangle vertically correctly", () => {
|
||||
createAndSelectOneRectangle();
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips a rotated rectangle horizontally correctly", () => {
|
||||
const originalAngle = (3 * Math.PI) / 4;
|
||||
const expectedAngle = (5 * Math.PI) / 4;
|
||||
|
||||
createAndSelectOneRectangle(originalAngle);
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(API.getSelectedElements()[0].angle).toBeCloseTo(expectedAngle);
|
||||
});
|
||||
|
||||
it("flips a rotated rectangle vertically correctly", () => {
|
||||
const originalAngle = (3 * Math.PI) / 4;
|
||||
const expectedAgnle = Math.PI / 4;
|
||||
|
||||
createAndSelectOneRectangle(originalAngle);
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(API.getSelectedElements()[0].angle).toBeCloseTo(expectedAgnle);
|
||||
});
|
||||
|
||||
// Diamond element
|
||||
|
||||
it("flips an unrotated diamond horizontally correctly", () => {
|
||||
createAndSelectOneDiamond();
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips an unrotated diamond vertically correctly", () => {
|
||||
createAndSelectOneDiamond();
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips a rotated diamond horizontally correctly", () => {
|
||||
const originalAngle = (5 * Math.PI) / 4;
|
||||
const expectedAngle = (3 * Math.PI) / 4;
|
||||
|
||||
createAndSelectOneDiamond(originalAngle);
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(API.getSelectedElements()[0].angle).toBeCloseTo(expectedAngle);
|
||||
});
|
||||
|
||||
it("flips a rotated diamond vertically correctly", () => {
|
||||
const originalAngle = (5 * Math.PI) / 4;
|
||||
const expectedAngle = (7 * Math.PI) / 4;
|
||||
|
||||
createAndSelectOneDiamond(originalAngle);
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(API.getSelectedElements()[0].angle).toBeCloseTo(expectedAngle);
|
||||
});
|
||||
|
||||
// Ellipse element
|
||||
|
||||
it("flips an unrotated ellipse horizontally correctly", () => {
|
||||
createAndSelectOneEllipse();
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips an unrotated ellipse vertically correctly", () => {
|
||||
createAndSelectOneEllipse();
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips a rotated ellipse horizontally correctly", () => {
|
||||
const originalAngle = (7 * Math.PI) / 4;
|
||||
const expectedAngle = Math.PI / 4;
|
||||
|
||||
createAndSelectOneEllipse(originalAngle);
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(API.getSelectedElements()[0].angle).toBeCloseTo(expectedAngle);
|
||||
});
|
||||
|
||||
it("flips a rotated ellipse vertically correctly", () => {
|
||||
const originalAngle = (7 * Math.PI) / 4;
|
||||
const expectedAngle = (5 * Math.PI) / 4;
|
||||
|
||||
createAndSelectOneEllipse(originalAngle);
|
||||
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if x position did not change
|
||||
expect(API.getSelectedElements()[0].x).toEqual(0);
|
||||
|
||||
expect(API.getSelectedElements()[0].y).toEqual(0);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(API.getSelectedElements()[0].angle).toBeCloseTo(expectedAngle);
|
||||
});
|
||||
|
||||
// Arrow element
|
||||
|
||||
it("flips an unrotated arrow horizontally correctly", () => {
|
||||
createAndSelectOneArrow();
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips an unrotated arrow vertically correctly", () => {
|
||||
createAndSelectOneArrow();
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips a rotated arrow horizontally correctly", () => {
|
||||
const originalAngle = Math.PI / 4;
|
||||
const expectedAngle = (7 * Math.PI) / 4;
|
||||
createAndSelectOneArrow(originalAngle);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(API.getSelectedElements()[0].angle).toBeCloseTo(expectedAngle);
|
||||
});
|
||||
|
||||
it("flips a rotated arrow vertically correctly", () => {
|
||||
const originalAngle = Math.PI / 4;
|
||||
const expectedAngle = (3 * Math.PI) / 4;
|
||||
createAndSelectOneArrow(originalAngle);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(API.getSelectedElements()[0].angle).toBeCloseTo(expectedAngle);
|
||||
});
|
||||
|
||||
// Line element
|
||||
|
||||
it("flips an unrotated line horizontally correctly", () => {
|
||||
createAndSelectOneLine();
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips an unrotated line vertically correctly", () => {
|
||||
createAndSelectOneLine();
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips a rotated line horizontally correctly", () => {
|
||||
const originalAngle = Math.PI / 4;
|
||||
const expectedAngle = (7 * Math.PI) / 4;
|
||||
|
||||
createAndSelectOneLine(originalAngle);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(API.getSelectedElements()[0].angle).toBeCloseTo(expectedAngle);
|
||||
});
|
||||
|
||||
it("flips a rotated line vertically correctly", () => {
|
||||
const originalAngle = Math.PI / 4;
|
||||
const expectedAngle = (3 * Math.PI) / 4;
|
||||
|
||||
createAndSelectOneLine(originalAngle);
|
||||
|
||||
const originalWidth = API.getSelectedElements()[0].width;
|
||||
const originalHeight = API.getSelectedElements()[0].height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElements()[0].width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElements()[0].height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(API.getSelectedElements()[0].angle).toBeCloseTo(expectedAngle);
|
||||
});
|
||||
|
||||
// Draw element
|
||||
|
||||
it("flips an unrotated drawing horizontally correctly", () => {
|
||||
const draw = createAndReturnOneDraw();
|
||||
// select draw, since not done automatically
|
||||
h.state.selectedElementIds[draw.id] = true;
|
||||
|
||||
const originalWidth = draw.width;
|
||||
const originalHeight = draw.height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(draw.width).toEqual(originalWidth);
|
||||
|
||||
expect(draw.height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips an unrotated drawing vertically correctly", () => {
|
||||
const draw = createAndReturnOneDraw();
|
||||
// select draw, since not done automatically
|
||||
h.state.selectedElementIds[draw.id] = true;
|
||||
|
||||
const originalWidth = draw.width;
|
||||
const originalHeight = draw.height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(draw.width).toEqual(originalWidth);
|
||||
|
||||
expect(draw.height).toEqual(originalHeight);
|
||||
});
|
||||
|
||||
it("flips a rotated drawing horizontally correctly", () => {
|
||||
const originalAngle = Math.PI / 4;
|
||||
const expectedAngle = (7 * Math.PI) / 4;
|
||||
|
||||
const draw = createAndReturnOneDraw(originalAngle);
|
||||
// select draw, since not done automatically
|
||||
h.state.selectedElementIds[draw.id] = true;
|
||||
|
||||
const originalWidth = draw.width;
|
||||
const originalHeight = draw.height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipHorizontal);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(draw.width).toEqual(originalWidth);
|
||||
|
||||
expect(draw.height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(draw.angle).toBeCloseTo(expectedAngle);
|
||||
});
|
||||
|
||||
it("flips a rotated drawing vertically correctly", () => {
|
||||
const originalAngle = Math.PI / 4;
|
||||
const expectedAngle = (3 * Math.PI) / 4;
|
||||
|
||||
const draw = createAndReturnOneDraw(originalAngle);
|
||||
// select draw, since not done automatically
|
||||
h.state.selectedElementIds[draw.id] = true;
|
||||
|
||||
const originalWidth = draw.width;
|
||||
const originalHeight = draw.height;
|
||||
|
||||
h.app.actionManager.executeAction(actionFlipVertical);
|
||||
|
||||
// Check if width and height did not change
|
||||
expect(API.getSelectedElement().width).toEqual(originalWidth);
|
||||
|
||||
expect(API.getSelectedElement().height).toEqual(originalHeight);
|
||||
|
||||
// Check angle
|
||||
expect(API.getSelectedElement().angle).toBeCloseTo(expectedAngle);
|
||||
});
|
Loading…
Reference in New Issue