Resize with negative width or height (#136)

* Disable resize for text, arrow and multiple selection

* Resize with negative width and height

* Fix resizing when leaves windows
This commit is contained in:
Paulo Menezes 2020-01-04 18:39:55 -03:00 committed by Christopher Chedeau
parent 79959cab34
commit dfb7faec30

View File

@ -170,12 +170,7 @@ function resizeTest(
): string | false { ): string | false {
if (element.type === "text" || element.type === "arrow") return false; if (element.type === "text" || element.type === "arrow") return false;
const x1 = getElementAbsoluteX1(element); const handlers = handlerRectangles(element, sceneState);
const x2 = getElementAbsoluteX2(element);
const y1 = getElementAbsoluteY1(element);
const y2 = getElementAbsoluteY2(element);
const handlers = handlerRectangles(x1, x2, y1, y2, sceneState);
const filter = Object.keys(handlers).filter(key => { const filter = Object.keys(handlers).filter(key => {
const handler = handlers[key]; const handler = handlers[key];
@ -270,43 +265,45 @@ function getScrollbars(
}; };
} }
function handlerRectangles( function handlerRectangles(element: ExcalidrawElement, sceneState: SceneState) {
elementX1: number, const elementX1 = element.x;
elementX2: number, const elementX2 = element.x + element.width;
elementY1: number, const elementY1 = element.y;
elementY2: number, const elementY2 = element.y + element.height;
sceneState: SceneState
) {
const margin = 4; const margin = 4;
const minimumSize = 40; const minimumSize = 40;
const handlers: { [handler: string]: number[] } = {}; const handlers: { [handler: string]: number[] } = {};
if (elementX2 - elementX1 > minimumSize) { const marginX = element.width < 0 ? 8 : -8;
const marginY = element.height < 0 ? 8 : -8;
if (Math.abs(elementX2 - elementX1) > minimumSize) {
handlers["n"] = [ handlers["n"] = [
elementX1 + (elementX2 - elementX1) / 2 + sceneState.scrollX - 4, elementX1 + (elementX2 - elementX1) / 2 + sceneState.scrollX - 4,
elementY1 - margin + sceneState.scrollY - 8, elementY1 - margin + sceneState.scrollY + marginY,
8, 8,
8 8
]; ];
handlers["s"] = [ handlers["s"] = [
elementX1 + (elementX2 - elementX1) / 2 + sceneState.scrollX - 4, elementX1 + (elementX2 - elementX1) / 2 + sceneState.scrollX - 4,
elementY2 - margin + sceneState.scrollY + 8, elementY2 - margin + sceneState.scrollY - marginY,
8, 8,
8 8
]; ];
} }
if (elementY2 - elementY1 > minimumSize) { if (Math.abs(elementY2 - elementY1) > minimumSize) {
handlers["w"] = [ handlers["w"] = [
elementX1 - margin + sceneState.scrollX - 8, elementX1 - margin + sceneState.scrollX + marginX,
elementY1 + (elementY2 - elementY1) / 2 + sceneState.scrollY - 4, elementY1 + (elementY2 - elementY1) / 2 + sceneState.scrollY - 4,
8, 8,
8 8
]; ];
handlers["e"] = [ handlers["e"] = [
elementX2 - margin + sceneState.scrollX + 8, elementX2 - margin + sceneState.scrollX - marginX,
elementY1 + (elementY2 - elementY1) / 2 + sceneState.scrollY - 4, elementY1 + (elementY2 - elementY1) / 2 + sceneState.scrollY - 4,
8, 8,
8 8
@ -314,26 +311,26 @@ function handlerRectangles(
} }
handlers["nw"] = [ handlers["nw"] = [
elementX1 - margin + sceneState.scrollX - 8, elementX1 - margin + sceneState.scrollX + marginX,
elementY1 - margin + sceneState.scrollY - 8, elementY1 - margin + sceneState.scrollY + marginY,
8, 8,
8 8
]; // nw ]; // nw
handlers["ne"] = [ handlers["ne"] = [
elementX2 - margin + sceneState.scrollX + 8, elementX2 - margin + sceneState.scrollX - marginX,
elementY1 - margin + sceneState.scrollY - 8, elementY1 - margin + sceneState.scrollY + marginY,
8, 8,
8 8
]; // ne ]; // ne
handlers["sw"] = [ handlers["sw"] = [
elementX1 - margin + sceneState.scrollX - 8, elementX1 - margin + sceneState.scrollX + marginX,
elementY2 - margin + sceneState.scrollY + 8, elementY2 - margin + sceneState.scrollY - marginY,
8, 8,
8 8
]; // sw ]; // sw
handlers["se"] = [ handlers["se"] = [
elementX2 - margin + sceneState.scrollX + 8, elementX2 - margin + sceneState.scrollX - marginX,
elementY2 - margin + sceneState.scrollY + 8, elementY2 - margin + sceneState.scrollY - marginY,
8, 8,
8 8
]; // se ]; // se
@ -402,13 +399,7 @@ function renderScene(
element.type !== "arrow" && element.type !== "arrow" &&
selectedIndices.length === 1 selectedIndices.length === 1
) { ) {
const handlers = handlerRectangles( const handlers = handlerRectangles(element, sceneState);
elementX1,
elementX2,
elementY1,
elementY2,
sceneState
);
Object.values(handlers).forEach(handler => { Object.values(handlers).forEach(handler => {
context.strokeRect(handler[0], handler[1], handler[2], handler[3]); context.strokeRect(handler[0], handler[1], handler[2], handler[3]);
}); });
@ -1157,7 +1148,6 @@ class App extends React.Component<{}, AppState> {
isResizingElements = true; isResizingElements = true;
} else { } else {
let hitElement = null; let hitElement = null;
// We need to to hit testing from front (end of the array) to back (beginning of the array) // We need to to hit testing from front (end of the array) to back (beginning of the array)
for (let i = elements.length - 1; i >= 0; --i) { for (let i = elements.length - 1; i >= 0; --i) {
if (hitTest(elements[i], x, y)) { if (hitTest(elements[i], x, y)) {
@ -1241,8 +1231,10 @@ class App extends React.Component<{}, AppState> {
const el = this.state.resizingElement; const el = this.state.resizingElement;
const selectedElements = elements.filter(el => el.isSelected); const selectedElements = elements.filter(el => el.isSelected);
if (selectedElements.length === 1) { if (selectedElements.length === 1) {
const x = e.clientX - target.offsetLeft - this.state.scrollX; const x =
const y = e.clientY - target.offsetTop - this.state.scrollY; e.clientX - CANVAS_WINDOW_OFFSET_LEFT - this.state.scrollX;
const y =
e.clientY - CANVAS_WINDOW_OFFSET_TOP - this.state.scrollY;
selectedElements.forEach(element => { selectedElements.forEach(element => {
switch (resizeHandle) { switch (resizeHandle) {
case "nw": case "nw":