@plait/common 0.1.0-next.2 → 0.1.0-next.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/plugins/with-resize.mjs +101 -0
- package/esm2022/shape/common.mjs +14 -0
- package/esm2022/utils/line-path.mjs +173 -0
- package/esm2022/utils/rectangle.mjs +23 -0
- package/esm2022/utils/resize.mjs +61 -0
- package/{fesm2020 → fesm2022}/plait-common.mjs +108 -13
- package/fesm2022/plait-common.mjs.map +1 -0
- package/package.json +5 -11
- package/plugins/with-resize.d.ts +1 -0
- package/shape/common.d.ts +1 -1
- package/utils/line-path.d.ts +5 -0
- package/utils/rectangle.d.ts +6 -1
- package/utils/resize.d.ts +4 -3
- package/esm2020/plugins/with-resize.mjs +0 -100
- package/esm2020/shape/common.mjs +0 -6
- package/esm2020/utils/line-path.mjs +0 -96
- package/esm2020/utils/rectangle.mjs +0 -16
- package/esm2020/utils/resize.mjs +0 -58
- package/fesm2015/plait-common.mjs +0 -421
- package/fesm2015/plait-common.mjs.map +0 -1
- package/fesm2020/plait-common.mjs.map +0 -1
- /package/{esm2020 → esm2022}/constants/default.mjs +0 -0
- /package/{esm2020 → esm2022}/constants/index.mjs +0 -0
- /package/{esm2020 → esm2022}/constants/media.mjs +0 -0
- /package/{esm2020 → esm2022}/constants/resize.mjs +0 -0
- /package/{esm2020 → esm2022}/generator/active.generator.mjs +0 -0
- /package/{esm2020 → esm2022}/generator/generator.mjs +0 -0
- /package/{esm2020 → esm2022}/generator/index.mjs +0 -0
- /package/{esm2020 → esm2022}/plait-common.mjs +0 -0
- /package/{esm2020 → esm2022}/plugins/index.mjs +0 -0
- /package/{esm2020 → esm2022}/public-api.mjs +0 -0
- /package/{esm2020 → esm2022}/shape/index.mjs +0 -0
- /package/{esm2020 → esm2022}/transforms/element.mjs +0 -0
- /package/{esm2020 → esm2022}/transforms/index.mjs +0 -0
- /package/{esm2020 → esm2022}/utils/hot-key.mjs +0 -0
- /package/{esm2020 → esm2022}/utils/index.mjs +0 -0
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
import { MERGING, PRESS_AND_MOVE_BUFFER, PlaitBoard, PlaitPointerType, distanceBetweenPointAndPoint, isMainPointer, preventTouchMove, throttleRAF, toPoint, transformPoint } from '@plait/core';
|
|
2
|
+
import { addResizing, isResizing, removeResizing } from '../utils/resize';
|
|
3
|
+
const generalCanResize = (board, event) => {
|
|
4
|
+
return (PlaitBoard.isReadonly(board) ||
|
|
5
|
+
PlaitBoard.hasBeenTextEditing(board) ||
|
|
6
|
+
!PlaitBoard.isPointer(board, PlaitPointerType.hand) ||
|
|
7
|
+
!isMainPointer(event));
|
|
8
|
+
};
|
|
9
|
+
export const withResize = (board, options) => {
|
|
10
|
+
const { pointerDown, pointerMove, globalPointerUp } = board;
|
|
11
|
+
let resizeDetectResult = null;
|
|
12
|
+
let resizeRef = null;
|
|
13
|
+
let startPoint = null;
|
|
14
|
+
let hoveDetectResult = null;
|
|
15
|
+
board.pointerDown = (event) => {
|
|
16
|
+
if (!options.canResize() || !generalCanResize(board, event)) {
|
|
17
|
+
pointerDown(event);
|
|
18
|
+
return;
|
|
19
|
+
}
|
|
20
|
+
const point = transformPoint(board, toPoint(event.x, event.y, PlaitBoard.getHost(board)));
|
|
21
|
+
resizeDetectResult = options.detect(point);
|
|
22
|
+
if (resizeDetectResult) {
|
|
23
|
+
if (resizeDetectResult.cursorClass) {
|
|
24
|
+
PlaitBoard.getBoardContainer(board).classList.add(`${resizeDetectResult.cursorClass}`);
|
|
25
|
+
}
|
|
26
|
+
startPoint = [event.x, event.y];
|
|
27
|
+
resizeRef = {
|
|
28
|
+
path: PlaitBoard.findPath(board, resizeDetectResult.element),
|
|
29
|
+
element: resizeDetectResult.element,
|
|
30
|
+
handle: resizeDetectResult.handle
|
|
31
|
+
};
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
pointerDown(event);
|
|
35
|
+
};
|
|
36
|
+
board.pointerMove = (event) => {
|
|
37
|
+
if (!options.canResize() || !generalCanResize(board, event)) {
|
|
38
|
+
pointerMove(event);
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
if (startPoint && resizeDetectResult && !isResizing(board)) {
|
|
42
|
+
// prevent text from being selected
|
|
43
|
+
event.preventDefault();
|
|
44
|
+
preventTouchMove(board, true);
|
|
45
|
+
const endPoint = transformPoint(board, toPoint(event.x, event.y, PlaitBoard.getHost(board)));
|
|
46
|
+
const distance = distanceBetweenPointAndPoint(startPoint[0], startPoint[1], endPoint[0], endPoint[1]);
|
|
47
|
+
if (distance > PRESS_AND_MOVE_BUFFER) {
|
|
48
|
+
addResizing(board, resizeDetectResult.element, options.key);
|
|
49
|
+
MERGING.set(board, true);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
if (isResizing(board) && startPoint) {
|
|
53
|
+
// prevent text from being selected
|
|
54
|
+
event.preventDefault();
|
|
55
|
+
preventTouchMove(board, true);
|
|
56
|
+
const endTransformPoint = transformPoint(board, toPoint(event.x, event.y, PlaitBoard.getHost(board)));
|
|
57
|
+
throttleRAF(() => {
|
|
58
|
+
const endPoint = [event.x, event.y];
|
|
59
|
+
if (startPoint && resizeRef) {
|
|
60
|
+
const offsetX = endPoint[0] - startPoint[0];
|
|
61
|
+
const offsetY = endPoint[1] - startPoint[1];
|
|
62
|
+
options.onResize(resizeRef, { offsetX, offsetY, endTransformPoint });
|
|
63
|
+
}
|
|
64
|
+
});
|
|
65
|
+
return;
|
|
66
|
+
}
|
|
67
|
+
else {
|
|
68
|
+
const point = transformPoint(board, toPoint(event.x, event.y, PlaitBoard.getHost(board)));
|
|
69
|
+
const resizeDetectResult = options.detect(point);
|
|
70
|
+
if (resizeDetectResult) {
|
|
71
|
+
hoveDetectResult = resizeDetectResult;
|
|
72
|
+
if (hoveDetectResult.cursorClass) {
|
|
73
|
+
PlaitBoard.getBoardContainer(board).classList.add(`${hoveDetectResult.cursorClass}`);
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
else {
|
|
77
|
+
if (hoveDetectResult) {
|
|
78
|
+
if (hoveDetectResult.cursorClass) {
|
|
79
|
+
PlaitBoard.getBoardContainer(board).classList.remove(`${hoveDetectResult.cursorClass}`);
|
|
80
|
+
}
|
|
81
|
+
hoveDetectResult = null;
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
pointerMove(event);
|
|
86
|
+
};
|
|
87
|
+
board.globalPointerUp = (event) => {
|
|
88
|
+
globalPointerUp(event);
|
|
89
|
+
if (isResizing(board) || resizeDetectResult) {
|
|
90
|
+
options.endResize && options.endResize(resizeRef);
|
|
91
|
+
removeResizing(board, options.key);
|
|
92
|
+
startPoint = null;
|
|
93
|
+
resizeDetectResult = null;
|
|
94
|
+
resizeRef = null;
|
|
95
|
+
MERGING.set(board, false);
|
|
96
|
+
preventTouchMove(board, false);
|
|
97
|
+
}
|
|
98
|
+
};
|
|
99
|
+
return board;
|
|
100
|
+
};
|
|
101
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"with-resize.js","sourceRoot":"","sources":["../../../../packages/common/src/plugins/with-resize.ts"],"names":[],"mappings":"AAAA,OAAO,EACH,OAAO,EACP,qBAAqB,EAErB,UAAU,EAEV,gBAAgB,EAEhB,4BAA4B,EAC5B,aAAa,EACb,gBAAgB,EAChB,WAAW,EACX,OAAO,EACP,cAAc,EACjB,MAAM,aAAa,CAAC;AAErB,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AA4B1E,MAAM,gBAAgB,GAAG,CAAC,KAAiB,EAAE,KAAmB,EAAE,EAAE;IAChE,OAAO,CACH,UAAU,CAAC,UAAU,CAAC,KAAK,CAAC;QAC5B,UAAU,CAAC,kBAAkB,CAAC,KAAK,CAAC;QACpC,CAAC,UAAU,CAAC,SAAS,CAAC,KAAK,EAAE,gBAAgB,CAAC,IAAI,CAAC;QACnD,CAAC,aAAa,CAAC,KAAK,CAAC,CACxB,CAAC;AACN,CAAC,CAAC;AAEF,MAAM,CAAC,MAAM,UAAU,GAAG,CACtB,KAAiB,EACjB,OAAgC,EAClC,EAAE;IACA,MAAM,EAAE,WAAW,EAAE,WAAW,EAAE,eAAe,EAAE,GAAG,KAAK,CAAC;IAC5D,IAAI,kBAAkB,GAAoC,IAAI,CAAC;IAC/D,IAAI,SAAS,GAA2B,IAAI,CAAC;IAC7C,IAAI,UAAU,GAAiB,IAAI,CAAC;IACpC,IAAI,gBAAgB,GAAoC,IAAI,CAAC;IAE7D,KAAK,CAAC,WAAW,GAAG,CAAC,KAAmB,EAAE,EAAE;QACxC,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,KAAK,CAAC,EAAE;YACzD,WAAW,CAAC,KAAK,CAAC,CAAC;YACnB,OAAO;SACV;QACD,MAAM,KAAK,GAAG,cAAc,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,EAAE,UAAU,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QAC1F,kBAAkB,GAAG,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAC3C,IAAI,kBAAkB,EAAE;YACpB,IAAI,kBAAkB,CAAC,WAAW,EAAE;gBAChC,UAAU,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,kBAAkB,CAAC,WAAW,EAAE,CAAC,CAAC;aAC1F;YACD,UAAU,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC;YAChC,SAAS,GAAG;gBACR,IAAI,EAAE,UAAU,CAAC,QAAQ,CAAC,KAAK,EAAE,kBAAkB,CAAC,OAAO,CAAC;gBAC5D,OAAO,EAAE,kBAAkB,CAAC,OAAO;gBACnC,MAAM,EAAE,kBAAkB,CAAC,MAAM;aACpC,CAAC;YACF,OAAO;SACV;QACD,WAAW,CAAC,KAAK,CAAC,CAAC;IACvB,CAAC,CAAC;IAEF,KAAK,CAAC,WAAW,GAAG,CAAC,KAAmB,EAAE,EAAE;QACxC,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,KAAK,CAAC,EAAE;YACzD,WAAW,CAAC,KAAK,CAAC,CAAC;YACnB,OAAO;SACV;QACD,IAAI,UAAU,IAAI,kBAAkB,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE;YACxD,mCAAmC;YACnC,KAAK,CAAC,cAAc,EAAE,CAAC;YACvB,gBAAgB,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;YAE9B,MAAM,QAAQ,GAAG,cAAc,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,EAAE,UAAU,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAC7F,MAAM,QAAQ,GAAG,4BAA4B,CAAC,UAAU,CAAC,CAAC,CAAC,EAAE,UAAU,CAAC,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;YACtG,IAAI,QAAQ,GAAG,qBAAqB,EAAE;gBAClC,WAAW,CAAC,KAAK,EAAE,kBAAkB,CAAC,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC;gBAC5D,OAAO,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;aAC5B;SACJ;QAED,IAAI,UAAU,CAAC,KAAK,CAAC,IAAI,UAAU,EAAE;YACjC,mCAAmC;YACnC,KAAK,CAAC,cAAc,EAAE,CAAC;YACvB,gBAAgB,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;YAC9B,MAAM,iBAAiB,GAAG,cAAc,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,EAAE,UAAU,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YACtG,WAAW,CAAC,GAAG,EAAE;gBACb,MAAM,QAAQ,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC;gBACpC,IAAI,UAAU,IAAI,SAAS,EAAE;oBACzB,MAAM,OAAO,GAAG,QAAQ,CAAC,CAAC,CAAC,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC;oBAC5C,MAAM,OAAO,GAAG,QAAQ,CAAC,CAAC,CAAC,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC;oBAC5C,OAAO,CAAC,QAAQ,CAAC,SAAS,EAAE,EAAE,OAAO,EAAE,OAAO,EAAE,iBAAiB,EAAE,CAAC,CAAC;iBACxE;YACL,CAAC,CAAC,CAAC;YACH,OAAO;SACV;aAAM;YACH,MAAM,KAAK,GAAG,cAAc,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,EAAE,UAAU,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAC1F,MAAM,kBAAkB,GAAG,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YACjD,IAAI,kBAAkB,EAAE;gBACpB,gBAAgB,GAAG,kBAAkB,CAAC;gBACtC,IAAI,gBAAgB,CAAC,WAAW,EAAE;oBAC9B,UAAU,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,gBAAgB,CAAC,WAAW,EAAE,CAAC,CAAC;iBACxF;aACJ;iBAAM;gBACH,IAAI,gBAAgB,EAAE;oBAClB,IAAI,gBAAgB,CAAC,WAAW,EAAE;wBAC9B,UAAU,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,MAAM,CAAC,GAAG,gBAAgB,CAAC,WAAW,EAAE,CAAC,CAAC;qBAC3F;oBACD,gBAAgB,GAAG,IAAI,CAAC;iBAC3B;aACJ;SACJ;QACD,WAAW,CAAC,KAAK,CAAC,CAAC;IACvB,CAAC,CAAC;IAEF,KAAK,CAAC,eAAe,GAAG,CAAC,KAAmB,EAAE,EAAE;QAC5C,eAAe,CAAC,KAAK,CAAC,CAAC;QACvB,IAAI,UAAU,CAAC,KAAK,CAAC,IAAI,kBAAkB,EAAE;YACzC,OAAO,CAAC,SAAS,IAAI,OAAO,CAAC,SAAS,CAAC,SAAU,CAAC,CAAC;YACnD,cAAc,CAAC,KAAK,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC;YACnC,UAAU,GAAG,IAAI,CAAC;YAClB,kBAAkB,GAAG,IAAI,CAAC;YAC1B,SAAS,GAAG,IAAI,CAAC;YACjB,OAAO,CAAC,GAAG,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;YAC1B,gBAAgB,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;SAClC;IACL,CAAC,CAAC;IAEF,OAAO,KAAK,CAAC;AACjB,CAAC,CAAC","sourcesContent":["import {\n    MERGING,\n    PRESS_AND_MOVE_BUFFER,\n    Path,\n    PlaitBoard,\n    PlaitElement,\n    PlaitPointerType,\n    Point,\n    distanceBetweenPointAndPoint,\n    isMainPointer,\n    preventTouchMove,\n    throttleRAF,\n    toPoint,\n    transformPoint\n} from '@plait/core';\nimport { ResizeHandle, ResizeCursorClass } from '../constants/resize';\nimport { addResizing, isResizing, removeResizing } from '../utils/resize';\n\nexport interface WithResizeOptions<T extends PlaitElement = PlaitElement, K = ResizeHandle> {\n    key: string;\n    canResize: () => boolean;\n    detect: (point: Point) => ResizeDetectResult<T, K> | null;\n    onResize: (resizeRef: ResizeRef<T, K>, resizeState: ResizeState) => void;\n    endResize?: (resizeRef: ResizeRef<T, K>) => void;\n}\n\nexport interface ResizeDetectResult<T extends PlaitElement = PlaitElement, K = ResizeHandle> {\n    element: T;\n    handle: K;\n    cursorClass?: ResizeCursorClass;\n}\n\nexport interface ResizeRef<T extends PlaitElement = PlaitElement, K = ResizeHandle> {\n    element: T;\n    path: Path;\n    handle: K;\n}\n\nexport interface ResizeState {\n    offsetX: number;\n    offsetY: number;\n    endTransformPoint: Point;\n}\n\nconst generalCanResize = (board: PlaitBoard, event: PointerEvent) => {\n    return (\n        PlaitBoard.isReadonly(board) ||\n        PlaitBoard.hasBeenTextEditing(board) ||\n        !PlaitBoard.isPointer(board, PlaitPointerType.hand) ||\n        !isMainPointer(event)\n    );\n};\n\nexport const withResize = <T extends PlaitElement = PlaitElement, K = ResizeHandle>(\n    board: PlaitBoard,\n    options: WithResizeOptions<T, K>\n) => {\n    const { pointerDown, pointerMove, globalPointerUp } = board;\n    let resizeDetectResult: ResizeDetectResult<T, K> | null = null;\n    let resizeRef: ResizeRef<T, K> | null = null;\n    let startPoint: Point | null = null;\n    let hoveDetectResult: ResizeDetectResult<T, K> | null = null;\n\n    board.pointerDown = (event: PointerEvent) => {\n        if (!options.canResize() || !generalCanResize(board, event)) {\n            pointerDown(event);\n            return;\n        }\n        const point = transformPoint(board, toPoint(event.x, event.y, PlaitBoard.getHost(board)));\n        resizeDetectResult = options.detect(point);\n        if (resizeDetectResult) {\n            if (resizeDetectResult.cursorClass) {\n                PlaitBoard.getBoardContainer(board).classList.add(`${resizeDetectResult.cursorClass}`);\n            }\n            startPoint = [event.x, event.y];\n            resizeRef = {\n                path: PlaitBoard.findPath(board, resizeDetectResult.element),\n                element: resizeDetectResult.element,\n                handle: resizeDetectResult.handle\n            };\n            return;\n        }\n        pointerDown(event);\n    };\n\n    board.pointerMove = (event: PointerEvent) => {\n        if (!options.canResize() || !generalCanResize(board, event)) {\n            pointerMove(event);\n            return;\n        }\n        if (startPoint && resizeDetectResult && !isResizing(board)) {\n            // prevent text from being selected\n            event.preventDefault();\n            preventTouchMove(board, true);\n\n            const endPoint = transformPoint(board, toPoint(event.x, event.y, PlaitBoard.getHost(board)));\n            const distance = distanceBetweenPointAndPoint(startPoint[0], startPoint[1], endPoint[0], endPoint[1]);\n            if (distance > PRESS_AND_MOVE_BUFFER) {\n                addResizing(board, resizeDetectResult.element, options.key);\n                MERGING.set(board, true);\n            }\n        }\n\n        if (isResizing(board) && startPoint) {\n            // prevent text from being selected\n            event.preventDefault();\n            preventTouchMove(board, true);\n            const endTransformPoint = transformPoint(board, toPoint(event.x, event.y, PlaitBoard.getHost(board)));\n            throttleRAF(() => {\n                const endPoint = [event.x, event.y];\n                if (startPoint && resizeRef) {\n                    const offsetX = endPoint[0] - startPoint[0];\n                    const offsetY = endPoint[1] - startPoint[1];\n                    options.onResize(resizeRef, { offsetX, offsetY, endTransformPoint });\n                }\n            });\n            return;\n        } else {\n            const point = transformPoint(board, toPoint(event.x, event.y, PlaitBoard.getHost(board)));\n            const resizeDetectResult = options.detect(point);\n            if (resizeDetectResult) {\n                hoveDetectResult = resizeDetectResult;\n                if (hoveDetectResult.cursorClass) {\n                    PlaitBoard.getBoardContainer(board).classList.add(`${hoveDetectResult.cursorClass}`);\n                }\n            } else {\n                if (hoveDetectResult) {\n                    if (hoveDetectResult.cursorClass) {\n                        PlaitBoard.getBoardContainer(board).classList.remove(`${hoveDetectResult.cursorClass}`);\n                    }\n                    hoveDetectResult = null;\n                }\n            }\n        }\n        pointerMove(event);\n    };\n\n    board.globalPointerUp = (event: PointerEvent) => {\n        globalPointerUp(event);\n        if (isResizing(board) || resizeDetectResult) {\n            options.endResize && options.endResize(resizeRef!);\n            removeResizing(board, options.key);\n            startPoint = null;\n            resizeDetectResult = null;\n            resizeRef = null;\n            MERGING.set(board, false);\n            preventTouchMove(board, false);\n        }\n    };\n\n    return board;\n};\n"]}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
export const normalizeShapePoints = (points, shift = false) => {
|
|
2
|
+
let start = points[0];
|
|
3
|
+
let end = points[1];
|
|
4
|
+
if (shift) {
|
|
5
|
+
const width = Math.abs(start[0] - end[0]);
|
|
6
|
+
const height = Math.abs(start[1] - end[1]);
|
|
7
|
+
const edge = Math.max(height, width);
|
|
8
|
+
end = [start[0] + (end[0] > start[0] ? edge : -edge), start[1] + (end[1] > start[1] ? edge : -edge)];
|
|
9
|
+
}
|
|
10
|
+
const leftTopPoint = [Math.min(start[0], end[0]), Math.min(start[1], end[1])];
|
|
11
|
+
let rightBottomPoint = [Math.max(start[0], end[0]), Math.max(start[1], end[1])];
|
|
12
|
+
return [leftTopPoint, rightBottomPoint];
|
|
13
|
+
};
|
|
14
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcGFja2FnZXMvY29tbW9uL3NyYy9zaGFwZS9jb21tb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsTUFBTSxDQUFDLE1BQU0sb0JBQW9CLEdBQUcsQ0FBQyxNQUFzQixFQUFFLFFBQWlCLEtBQUssRUFBa0IsRUFBRTtJQUNuRyxJQUFJLEtBQUssR0FBRyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDdEIsSUFBSSxHQUFHLEdBQUcsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ3BCLElBQUksS0FBSyxFQUFFO1FBQ1AsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLEdBQUcsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDMUMsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLEdBQUcsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDM0MsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFDckMsR0FBRyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxHQUFHLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO0tBQ3hHO0lBRUQsTUFBTSxZQUFZLEdBQVUsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFBRSxJQUFJLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ3JGLElBQUksZ0JBQWdCLEdBQVUsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFBRSxJQUFJLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBRXZGLE9BQU8sQ0FBQyxZQUFZLEVBQUUsZ0JBQWdCLENBQUMsQ0FBQztBQUM1QyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBQb2ludCB9IGZyb20gJ0BwbGFpdC9jb3JlJztcblxuZXhwb3J0IGNvbnN0IG5vcm1hbGl6ZVNoYXBlUG9pbnRzID0gKHBvaW50czogW1BvaW50LCBQb2ludF0sIHNoaWZ0OiBib29sZWFuID0gZmFsc2UpOiBbUG9pbnQsIFBvaW50XSA9PiB7XG4gICAgbGV0IHN0YXJ0ID0gcG9pbnRzWzBdO1xuICAgIGxldCBlbmQgPSBwb2ludHNbMV07XG4gICAgaWYgKHNoaWZ0KSB7XG4gICAgICAgIGNvbnN0IHdpZHRoID0gTWF0aC5hYnMoc3RhcnRbMF0gLSBlbmRbMF0pO1xuICAgICAgICBjb25zdCBoZWlnaHQgPSBNYXRoLmFicyhzdGFydFsxXSAtIGVuZFsxXSk7XG4gICAgICAgIGNvbnN0IGVkZ2UgPSBNYXRoLm1heChoZWlnaHQsIHdpZHRoKTtcbiAgICAgICAgZW5kID0gW3N0YXJ0WzBdICsgKGVuZFswXSA+IHN0YXJ0WzBdID8gZWRnZSA6IC1lZGdlKSwgc3RhcnRbMV0gKyAoZW5kWzFdID4gc3RhcnRbMV0gPyBlZGdlIDogLWVkZ2UpXTtcbiAgICB9XG5cbiAgICBjb25zdCBsZWZ0VG9wUG9pbnQ6IFBvaW50ID0gW01hdGgubWluKHN0YXJ0WzBdLCBlbmRbMF0pLCBNYXRoLm1pbihzdGFydFsxXSwgZW5kWzFdKV07XG4gICAgbGV0IHJpZ2h0Qm90dG9tUG9pbnQ6IFBvaW50ID0gW01hdGgubWF4KHN0YXJ0WzBdLCBlbmRbMF0pLCBNYXRoLm1heChzdGFydFsxXSwgZW5kWzFdKV07XG5cbiAgICByZXR1cm4gW2xlZnRUb3BQb2ludCwgcmlnaHRCb3R0b21Qb2ludF07XG59O1xuIl19
|
|
@@ -0,0 +1,173 @@
|
|
|
1
|
+
import { distanceBetweenPointAndPoint } from '@plait/core';
|
|
2
|
+
export const getPoints = (source, sourcePosition, target, targetPosition, offset) => {
|
|
3
|
+
const sourceDir = handleDirections[sourcePosition];
|
|
4
|
+
const targetDir = handleDirections[targetPosition];
|
|
5
|
+
const sourceGapped = [source[0] + sourceDir.x * offset, source[1] + sourceDir.y * offset];
|
|
6
|
+
const targetGapped = [target[0] + targetDir.x * offset, target[1] + targetDir.y * offset];
|
|
7
|
+
const dir = getDirection(sourceGapped, sourcePosition, targetGapped);
|
|
8
|
+
const dirAccessor = dir.x !== 0 ? 'x' : 'y';
|
|
9
|
+
const currDir = dir[dirAccessor];
|
|
10
|
+
let points = [];
|
|
11
|
+
let centerX, centerY;
|
|
12
|
+
const [defaultCenterX, defaultCenterY] = getEdgeCenter({
|
|
13
|
+
sourceX: source[0],
|
|
14
|
+
sourceY: source[1],
|
|
15
|
+
targetX: target[0],
|
|
16
|
+
targetY: target[1]
|
|
17
|
+
});
|
|
18
|
+
// opposite handle positions, default case
|
|
19
|
+
if (sourceDir[dirAccessor] * targetDir[dirAccessor] === -1) {
|
|
20
|
+
centerX = defaultCenterX;
|
|
21
|
+
centerY = defaultCenterY;
|
|
22
|
+
// --->
|
|
23
|
+
// |
|
|
24
|
+
// >---
|
|
25
|
+
const verticalSplit = [
|
|
26
|
+
[centerX, sourceGapped[1]],
|
|
27
|
+
[centerX, targetGapped[1]]
|
|
28
|
+
];
|
|
29
|
+
// |
|
|
30
|
+
// ---
|
|
31
|
+
// |
|
|
32
|
+
const horizontalSplit = [
|
|
33
|
+
[sourceGapped[0], centerY],
|
|
34
|
+
[targetGapped[0], centerY]
|
|
35
|
+
];
|
|
36
|
+
if (sourceDir[dirAccessor] === currDir) {
|
|
37
|
+
points = dirAccessor === 'x' ? verticalSplit : horizontalSplit;
|
|
38
|
+
}
|
|
39
|
+
else {
|
|
40
|
+
points = dirAccessor === 'x' ? horizontalSplit : verticalSplit;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
else {
|
|
44
|
+
// sourceTarget means we take x from source and y from target, targetSource is the opposite
|
|
45
|
+
const sourceTarget = [[sourceGapped[0], targetGapped[1]]];
|
|
46
|
+
const targetSource = [[targetGapped[0], sourceGapped[1]]];
|
|
47
|
+
// this handles edges with same handle positions
|
|
48
|
+
if (dirAccessor === 'x') {
|
|
49
|
+
points = sourceDir.x === currDir ? targetSource : sourceTarget;
|
|
50
|
+
}
|
|
51
|
+
else {
|
|
52
|
+
points = sourceDir.y === currDir ? sourceTarget : targetSource;
|
|
53
|
+
}
|
|
54
|
+
// these are conditions for handling mixed handle positions like right -> bottom for example
|
|
55
|
+
let flipSourceTarget;
|
|
56
|
+
if (sourcePosition !== targetPosition) {
|
|
57
|
+
const dirAccessorOpposite = dirAccessor === 'x' ? 1 : 0;
|
|
58
|
+
const isSameDir = sourceDir[dirAccessor] === targetDir[dirAccessor === 'x' ? 'y' : 'x'];
|
|
59
|
+
const sourceGtTargetOppo = sourceGapped[dirAccessorOpposite] > targetGapped[dirAccessorOpposite];
|
|
60
|
+
const sourceLtTargetOppo = sourceGapped[dirAccessorOpposite] < targetGapped[dirAccessorOpposite];
|
|
61
|
+
flipSourceTarget =
|
|
62
|
+
(sourceDir[dirAccessor] === 1 && ((!isSameDir && sourceGtTargetOppo) || (isSameDir && sourceLtTargetOppo))) ||
|
|
63
|
+
(sourceDir[dirAccessor] !== 1 && ((!isSameDir && sourceLtTargetOppo) || (isSameDir && sourceGtTargetOppo)));
|
|
64
|
+
if (flipSourceTarget) {
|
|
65
|
+
points = dirAccessor === 'x' ? sourceTarget : targetSource;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
return [source, sourceGapped, ...points, targetGapped, target];
|
|
70
|
+
};
|
|
71
|
+
const getDirection = (source, sourcePosition = Direction.bottom, target) => {
|
|
72
|
+
if (sourcePosition === Direction.left || sourcePosition === Direction.right) {
|
|
73
|
+
return source[0] < target[0] ? { x: 1, y: 0 } : { x: -1, y: 0 };
|
|
74
|
+
}
|
|
75
|
+
return source[1] < target[1] ? { x: 0, y: 1 } : { x: 0, y: -1 };
|
|
76
|
+
};
|
|
77
|
+
export var Direction;
|
|
78
|
+
(function (Direction) {
|
|
79
|
+
Direction["left"] = "left";
|
|
80
|
+
Direction["top"] = "top";
|
|
81
|
+
Direction["right"] = "right";
|
|
82
|
+
Direction["bottom"] = "bottom";
|
|
83
|
+
})(Direction || (Direction = {}));
|
|
84
|
+
const handleDirections = {
|
|
85
|
+
[Direction.left]: { x: -1, y: 0 },
|
|
86
|
+
[Direction.right]: { x: 1, y: 0 },
|
|
87
|
+
[Direction.top]: { x: 0, y: -1 },
|
|
88
|
+
[Direction.bottom]: { x: 0, y: 1 }
|
|
89
|
+
};
|
|
90
|
+
export function getEdgeCenter({ sourceX, sourceY, targetX, targetY }) {
|
|
91
|
+
const xOffset = Math.abs(targetX - sourceX) / 2;
|
|
92
|
+
const centerX = targetX < sourceX ? targetX + xOffset : targetX - xOffset;
|
|
93
|
+
const yOffset = Math.abs(targetY - sourceY) / 2;
|
|
94
|
+
const centerY = targetY < sourceY ? targetY + yOffset : targetY - yOffset;
|
|
95
|
+
return [centerX, centerY, xOffset, yOffset];
|
|
96
|
+
}
|
|
97
|
+
export function getDirectionByPoint(point, defaultDirection) {
|
|
98
|
+
if (point[0] === 0) {
|
|
99
|
+
return Direction.left;
|
|
100
|
+
}
|
|
101
|
+
if (point[0] === 1) {
|
|
102
|
+
return Direction.right;
|
|
103
|
+
}
|
|
104
|
+
if (point[1] === 0) {
|
|
105
|
+
return Direction.top;
|
|
106
|
+
}
|
|
107
|
+
if (point[1] === 1) {
|
|
108
|
+
return Direction.bottom;
|
|
109
|
+
}
|
|
110
|
+
return defaultDirection;
|
|
111
|
+
}
|
|
112
|
+
export function getOppositeDirection(direction) {
|
|
113
|
+
switch (direction) {
|
|
114
|
+
case Direction.left:
|
|
115
|
+
return Direction.right;
|
|
116
|
+
case Direction.right:
|
|
117
|
+
return Direction.left;
|
|
118
|
+
case Direction.top:
|
|
119
|
+
return Direction.bottom;
|
|
120
|
+
case Direction.bottom:
|
|
121
|
+
return Direction.top;
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
export function getPointOnPolyline(points, ratio) {
|
|
125
|
+
const totalLength = calculatePolylineLength(points);
|
|
126
|
+
const targetDistance = totalLength * ratio;
|
|
127
|
+
let accumulatedDistance = 0;
|
|
128
|
+
for (let i = 0; i < points.length - 1; i++) {
|
|
129
|
+
const [x1, y1] = points[i];
|
|
130
|
+
const [x2, y2] = points[i + 1];
|
|
131
|
+
const segmentLength = distanceBetweenPointAndPoint(x1, y1, x2, y2);
|
|
132
|
+
if (accumulatedDistance + segmentLength >= targetDistance) {
|
|
133
|
+
const remainingDistance = targetDistance - accumulatedDistance;
|
|
134
|
+
const ratioInSegment = remainingDistance / segmentLength;
|
|
135
|
+
const targetX = x1 + (x2 - x1) * ratioInSegment;
|
|
136
|
+
const targetY = y1 + (y2 - y1) * ratioInSegment;
|
|
137
|
+
return [targetX, targetY];
|
|
138
|
+
}
|
|
139
|
+
accumulatedDistance += segmentLength;
|
|
140
|
+
}
|
|
141
|
+
return points[points.length - 1];
|
|
142
|
+
}
|
|
143
|
+
function calculatePolylineLength(points) {
|
|
144
|
+
let length = 0;
|
|
145
|
+
for (let i = 0; i < points.length - 1; i++) {
|
|
146
|
+
const [x1, y1] = points[i];
|
|
147
|
+
const [x2, y2] = points[i + 1];
|
|
148
|
+
length += distanceBetweenPointAndPoint(x1, y1, x2, y2);
|
|
149
|
+
}
|
|
150
|
+
return length;
|
|
151
|
+
}
|
|
152
|
+
export function getRatioByPoint(points, point) {
|
|
153
|
+
const totalLength = calculatePolylineLength(points);
|
|
154
|
+
let distance = 0;
|
|
155
|
+
for (let i = 0; i < points.length - 1; i++) {
|
|
156
|
+
const isOverlap = isPointOnLineSegment(point, points[i], points[i + 1]);
|
|
157
|
+
if (isOverlap) {
|
|
158
|
+
distance += distanceBetweenPointAndPoint(point[0], point[1], points[i][0], points[i][1]);
|
|
159
|
+
return distance / totalLength;
|
|
160
|
+
}
|
|
161
|
+
else {
|
|
162
|
+
distance += distanceBetweenPointAndPoint(points[i][0], points[i][1], points[i + 1][0], points[i + 1][1]);
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
throw new Error('Cannot get ratio by point');
|
|
166
|
+
}
|
|
167
|
+
export function isPointOnLineSegment(point, startPoint, endPoint) {
|
|
168
|
+
const distanceToStart = distanceBetweenPointAndPoint(point[0], point[1], startPoint[0], startPoint[1]);
|
|
169
|
+
const distanceToEnd = distanceBetweenPointAndPoint(point[0], point[1], endPoint[0], endPoint[1]);
|
|
170
|
+
const segmentLength = distanceBetweenPointAndPoint(startPoint[0], startPoint[1], endPoint[0], endPoint[1]);
|
|
171
|
+
return Math.abs(distanceToStart + distanceToEnd - segmentLength) < 0.1;
|
|
172
|
+
}
|
|
173
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"line-path.js","sourceRoot":"","sources":["../../../../packages/common/src/utils/line-path.ts"],"names":[],"mappings":"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","sourcesContent":["import { Point, distanceBetweenPointAndPoint } from '@plait/core';\n\nexport const getPoints = (source: Point, sourcePosition: Direction, target: Point, targetPosition: Direction, offset: number) => {\n    const sourceDir = handleDirections[sourcePosition];\n    const targetDir = handleDirections[targetPosition];\n    const sourceGapped: Point = [source[0] + sourceDir.x * offset, source[1] + sourceDir.y * offset];\n    const targetGapped: Point = [target[0] + targetDir.x * offset, target[1] + targetDir.y * offset];\n    const dir = getDirection(sourceGapped, sourcePosition, targetGapped);\n    const dirAccessor = dir.x !== 0 ? 'x' : 'y';\n    const currDir = dir[dirAccessor];\n\n    let points: Point[] = [];\n    let centerX, centerY;\n    const [defaultCenterX, defaultCenterY] = getEdgeCenter({\n        sourceX: source[0],\n        sourceY: source[1],\n        targetX: target[0],\n        targetY: target[1]\n    });\n    // opposite handle positions, default case\n    if (sourceDir[dirAccessor] * targetDir[dirAccessor] === -1) {\n        centerX = defaultCenterX;\n        centerY = defaultCenterY;\n        //    --->\n        //    |\n        // >---\n        const verticalSplit: Point[] = [\n            [centerX, sourceGapped[1]],\n            [centerX, targetGapped[1]]\n        ];\n        //    |\n        //  ---\n        //  |\n        const horizontalSplit: Point[] = [\n            [sourceGapped[0], centerY],\n            [targetGapped[0], centerY]\n        ];\n        if (sourceDir[dirAccessor] === currDir) {\n            points = dirAccessor === 'x' ? verticalSplit : horizontalSplit;\n        } else {\n            points = dirAccessor === 'x' ? horizontalSplit : verticalSplit;\n        }\n    } else {\n        // sourceTarget means we take x from source and y from target, targetSource is the opposite\n        const sourceTarget: Point[] = [[sourceGapped[0], targetGapped[1]]];\n        const targetSource: Point[] = [[targetGapped[0], sourceGapped[1]]];\n        // this handles edges with same handle positions\n        if (dirAccessor === 'x') {\n            points = sourceDir.x === currDir ? targetSource : sourceTarget;\n        } else {\n            points = sourceDir.y === currDir ? sourceTarget : targetSource;\n        }\n\n        // these are conditions for handling mixed handle positions like right -> bottom for example\n        let flipSourceTarget;\n        if (sourcePosition !== targetPosition) {\n            const dirAccessorOpposite = dirAccessor === 'x' ? 1 : 0;\n            const isSameDir = sourceDir[dirAccessor] === targetDir[dirAccessor === 'x' ? 'y' : 'x'];\n            const sourceGtTargetOppo = sourceGapped[dirAccessorOpposite] > targetGapped[dirAccessorOpposite];\n            const sourceLtTargetOppo = sourceGapped[dirAccessorOpposite] < targetGapped[dirAccessorOpposite];\n            flipSourceTarget =\n                (sourceDir[dirAccessor] === 1 && ((!isSameDir && sourceGtTargetOppo) || (isSameDir && sourceLtTargetOppo))) ||\n                (sourceDir[dirAccessor] !== 1 && ((!isSameDir && sourceLtTargetOppo) || (isSameDir && sourceGtTargetOppo)));\n\n            if (flipSourceTarget) {\n                points = dirAccessor === 'x' ? sourceTarget : targetSource;\n            }\n        }\n    }\n    return [source, sourceGapped, ...points, targetGapped, target];\n};\n\nconst getDirection = (source: Point, sourcePosition = Direction.bottom, target: Point) => {\n    if (sourcePosition === Direction.left || sourcePosition === Direction.right) {\n        return source[0] < target[0] ? { x: 1, y: 0 } : { x: -1, y: 0 };\n    }\n    return source[1] < target[1] ? { x: 0, y: 1 } : { x: 0, y: -1 };\n};\n\nexport enum Direction {\n    left = 'left',\n    top = 'top',\n    right = 'right',\n    bottom = 'bottom'\n}\n\nconst handleDirections = {\n    [Direction.left]: { x: -1, y: 0 },\n    [Direction.right]: { x: 1, y: 0 },\n    [Direction.top]: { x: 0, y: -1 },\n    [Direction.bottom]: { x: 0, y: 1 }\n};\n\nexport function getEdgeCenter({\n    sourceX,\n    sourceY,\n    targetX,\n    targetY\n}: {\n    sourceX: number;\n    sourceY: number;\n    targetX: number;\n    targetY: number;\n}): [number, number, number, number] {\n    const xOffset = Math.abs(targetX - sourceX) / 2;\n    const centerX = targetX < sourceX ? targetX + xOffset : targetX - xOffset;\n\n    const yOffset = Math.abs(targetY - sourceY) / 2;\n    const centerY = targetY < sourceY ? targetY + yOffset : targetY - yOffset;\n\n    return [centerX, centerY, xOffset, yOffset];\n}\n\nexport function getDirectionByPoint(point: Point, defaultDirection: Direction) {\n    if (point[0] === 0) {\n        return Direction.left;\n    }\n    if (point[0] === 1) {\n        return Direction.right;\n    }\n    if (point[1] === 0) {\n        return Direction.top;\n    }\n    if (point[1] === 1) {\n        return Direction.bottom;\n    }\n    return defaultDirection;\n}\n\nexport function getOppositeDirection(direction: Direction) {\n    switch (direction) {\n        case Direction.left:\n            return Direction.right;\n        case Direction.right:\n            return Direction.left;\n        case Direction.top:\n            return Direction.bottom;\n        case Direction.bottom:\n            return Direction.top;\n    }\n}\n\nexport function getPointOnPolyline(points: Point[], ratio: number) {\n    const totalLength = calculatePolylineLength(points);\n    const targetDistance = totalLength * ratio;\n\n    let accumulatedDistance = 0;\n    for (let i = 0; i < points.length - 1; i++) {\n        const [x1, y1] = points[i];\n        const [x2, y2] = points[i + 1];\n        const segmentLength = distanceBetweenPointAndPoint(x1, y1, x2, y2);\n\n        if (accumulatedDistance + segmentLength >= targetDistance) {\n            const remainingDistance = targetDistance - accumulatedDistance;\n            const ratioInSegment = remainingDistance / segmentLength;\n\n            const targetX = x1 + (x2 - x1) * ratioInSegment;\n            const targetY = y1 + (y2 - y1) * ratioInSegment;\n            return [targetX, targetY];\n        }\n\n        accumulatedDistance += segmentLength;\n    }\n\n    return points[points.length - 1];\n}\n\nfunction calculatePolylineLength(points: Point[]) {\n    let length = 0;\n    for (let i = 0; i < points.length - 1; i++) {\n        const [x1, y1] = points[i];\n        const [x2, y2] = points[i + 1];\n        length += distanceBetweenPointAndPoint(x1, y1, x2, y2);\n    }\n    return length;\n}\n\nexport function getRatioByPoint(points: Point[], point: Point) {\n    const totalLength = calculatePolylineLength(points);\n\n    let distance = 0;\n\n    for (let i = 0; i < points.length - 1; i++) {\n        const isOverlap = isPointOnLineSegment(point, points[i], points[i + 1]);\n\n        if (isOverlap) {\n            distance += distanceBetweenPointAndPoint(point[0], point[1], points[i][0], points[i][1]);\n            return distance / totalLength;\n        } else {\n            distance += distanceBetweenPointAndPoint(points[i][0], points[i][1], points[i + 1][0], points[i + 1][1]);\n        }\n    }\n    throw new Error('Cannot get ratio by point')\n}\n\nexport function isPointOnLineSegment(point: Point, startPoint: Point, endPoint: Point) {\n    const distanceToStart = distanceBetweenPointAndPoint(point[0], point[1], startPoint[0], startPoint[1]);\n    const distanceToEnd = distanceBetweenPointAndPoint(point[0], point[1], endPoint[0], endPoint[1]);\n\n    const segmentLength = distanceBetweenPointAndPoint(startPoint[0], startPoint[1], endPoint[0], endPoint[1]);\n\n    return Math.abs(distanceToStart + distanceToEnd - segmentLength) < 0.1;\n}\n"]}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import { PlaitBoard } from '@plait/core';
|
|
2
|
+
export const drawRectangle = (board, rectangle, options) => {
|
|
3
|
+
const roughSVG = PlaitBoard.getRoughSVG(board);
|
|
4
|
+
const rectangleG = roughSVG.rectangle(rectangle.x, rectangle.y, rectangle.width, rectangle.height, options);
|
|
5
|
+
rectangleG.querySelector('path').setAttribute('stroke-linecap', 'square');
|
|
6
|
+
return rectangleG;
|
|
7
|
+
};
|
|
8
|
+
export const getRectangleByPoints = (points) => {
|
|
9
|
+
let minX = Infinity, maxX = -Infinity, minY = Infinity, maxY = -Infinity;
|
|
10
|
+
points.forEach(point => {
|
|
11
|
+
minX = Math.min(point[0], minX);
|
|
12
|
+
maxX = Math.max(point[0], maxX);
|
|
13
|
+
minY = Math.min(point[1], minY);
|
|
14
|
+
maxY = Math.max(point[1], maxY);
|
|
15
|
+
});
|
|
16
|
+
return {
|
|
17
|
+
x: minX,
|
|
18
|
+
y: minY,
|
|
19
|
+
width: maxX - minX,
|
|
20
|
+
height: maxY - minY
|
|
21
|
+
};
|
|
22
|
+
};
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjdGFuZ2xlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcGFja2FnZXMvY29tbW9uL3NyYy91dGlscy9yZWN0YW5nbGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBMEIsTUFBTSxhQUFhLENBQUM7QUFHakUsTUFBTSxDQUFDLE1BQU0sYUFBYSxHQUFHLENBQUMsS0FBaUIsRUFBRSxTQUEwQixFQUFFLE9BQWdCLEVBQUUsRUFBRTtJQUM3RixNQUFNLFFBQVEsR0FBRyxVQUFVLENBQUMsV0FBVyxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQy9DLE1BQU0sVUFBVSxHQUFHLFFBQVEsQ0FBQyxTQUFTLENBQUMsU0FBUyxDQUFDLENBQUMsRUFBRSxTQUFTLENBQUMsQ0FBQyxFQUFFLFNBQVMsQ0FBQyxLQUFLLEVBQUUsU0FBUyxDQUFDLE1BQU0sRUFBRSxPQUFPLENBQUMsQ0FBQztJQUM1RyxVQUFVLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBRSxDQUFDLFlBQVksQ0FBQyxnQkFBZ0IsRUFBRSxRQUFRLENBQUMsQ0FBQztJQUMzRSxPQUFPLFVBQVUsQ0FBQztBQUN0QixDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxvQkFBb0IsR0FBRyxDQUFDLE1BQWUsRUFBRSxFQUFFO0lBQ3BELElBQUksSUFBSSxHQUFHLFFBQVEsRUFDZixJQUFJLEdBQUcsQ0FBQyxRQUFRLEVBQ2hCLElBQUksR0FBRyxRQUFRLEVBQ2YsSUFBSSxHQUFHLENBQUMsUUFBUSxDQUFDO0lBQ3JCLE1BQU0sQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLEVBQUU7UUFDbkIsSUFBSSxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxFQUFFLElBQUksQ0FBQyxDQUFDO1FBQ2hDLElBQUksR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsRUFBRSxJQUFJLENBQUMsQ0FBQztRQUNoQyxJQUFJLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLEVBQUUsSUFBSSxDQUFDLENBQUM7UUFDaEMsSUFBSSxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxFQUFFLElBQUksQ0FBQyxDQUFDO0lBQ3BDLENBQUMsQ0FBQyxDQUFDO0lBRUgsT0FBTztRQUNILENBQUMsRUFBRSxJQUFJO1FBQ1AsQ0FBQyxFQUFFLElBQUk7UUFDUCxLQUFLLEVBQUUsSUFBSSxHQUFHLElBQUk7UUFDbEIsTUFBTSxFQUFFLElBQUksR0FBRyxJQUFJO0tBQ3RCLENBQUM7QUFDTixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBQbGFpdEJvYXJkLCBQb2ludCwgUmVjdGFuZ2xlQ2xpZW50IH0gZnJvbSAnQHBsYWl0L2NvcmUnO1xuaW1wb3J0IHsgT3B0aW9ucyB9IGZyb20gJ3JvdWdoanMvYmluL2NvcmUnO1xuXG5leHBvcnQgY29uc3QgZHJhd1JlY3RhbmdsZSA9IChib2FyZDogUGxhaXRCb2FyZCwgcmVjdGFuZ2xlOiBSZWN0YW5nbGVDbGllbnQsIG9wdGlvbnM6IE9wdGlvbnMpID0+IHtcbiAgICBjb25zdCByb3VnaFNWRyA9IFBsYWl0Qm9hcmQuZ2V0Um91Z2hTVkcoYm9hcmQpO1xuICAgIGNvbnN0IHJlY3RhbmdsZUcgPSByb3VnaFNWRy5yZWN0YW5nbGUocmVjdGFuZ2xlLngsIHJlY3RhbmdsZS55LCByZWN0YW5nbGUud2lkdGgsIHJlY3RhbmdsZS5oZWlnaHQsIG9wdGlvbnMpO1xuICAgIHJlY3RhbmdsZUcucXVlcnlTZWxlY3RvcigncGF0aCcpIS5zZXRBdHRyaWJ1dGUoJ3N0cm9rZS1saW5lY2FwJywgJ3NxdWFyZScpO1xuICAgIHJldHVybiByZWN0YW5nbGVHO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFJlY3RhbmdsZUJ5UG9pbnRzID0gKHBvaW50czogUG9pbnRbXSkgPT4ge1xuICAgIGxldCBtaW5YID0gSW5maW5pdHksXG4gICAgICAgIG1heFggPSAtSW5maW5pdHksXG4gICAgICAgIG1pblkgPSBJbmZpbml0eSxcbiAgICAgICAgbWF4WSA9IC1JbmZpbml0eTtcbiAgICBwb2ludHMuZm9yRWFjaChwb2ludCA9PiB7XG4gICAgICAgIG1pblggPSBNYXRoLm1pbihwb2ludFswXSwgbWluWCk7XG4gICAgICAgIG1heFggPSBNYXRoLm1heChwb2ludFswXSwgbWF4WCk7XG4gICAgICAgIG1pblkgPSBNYXRoLm1pbihwb2ludFsxXSwgbWluWSk7XG4gICAgICAgIG1heFkgPSBNYXRoLm1heChwb2ludFsxXSwgbWF4WSk7XG4gICAgfSk7XG5cbiAgICByZXR1cm4ge1xuICAgICAgICB4OiBtaW5YLFxuICAgICAgICB5OiBtaW5ZLFxuICAgICAgICB3aWR0aDogbWF4WCAtIG1pblgsXG4gICAgICAgIGhlaWdodDogbWF4WSAtIG1pbllcbiAgICB9O1xufTtcbiJdfQ==
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { PlaitBoard, RectangleClient } from '@plait/core';
|
|
2
|
+
import { ResizeCursorClass, ResizeHandle } from '../constants/resize';
|
|
3
|
+
const getResizeHandleByIndex = (index) => {
|
|
4
|
+
switch (index) {
|
|
5
|
+
case 0:
|
|
6
|
+
return ResizeHandle.nw;
|
|
7
|
+
case 1:
|
|
8
|
+
return ResizeHandle.ne;
|
|
9
|
+
case 2:
|
|
10
|
+
return ResizeHandle.se;
|
|
11
|
+
case 3:
|
|
12
|
+
return ResizeHandle.sw;
|
|
13
|
+
default:
|
|
14
|
+
return null;
|
|
15
|
+
}
|
|
16
|
+
};
|
|
17
|
+
const getResizeCursorClassByIndex = (index) => {
|
|
18
|
+
switch (index) {
|
|
19
|
+
case 0:
|
|
20
|
+
return ResizeCursorClass.nwse;
|
|
21
|
+
case 1:
|
|
22
|
+
return ResizeCursorClass.nesw;
|
|
23
|
+
case 2:
|
|
24
|
+
return ResizeCursorClass.nwse;
|
|
25
|
+
case 3:
|
|
26
|
+
return ResizeCursorClass.nesw;
|
|
27
|
+
default:
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
};
|
|
31
|
+
export const getRectangleResizeHandleRefs = (rectangle, diameter) => {
|
|
32
|
+
const centers = RectangleClient.getCornerPoints(rectangle);
|
|
33
|
+
return centers.map((center, index) => {
|
|
34
|
+
return {
|
|
35
|
+
rectangle: {
|
|
36
|
+
x: center[0] - diameter / 2,
|
|
37
|
+
y: center[1] - diameter / 2,
|
|
38
|
+
width: diameter,
|
|
39
|
+
height: diameter
|
|
40
|
+
},
|
|
41
|
+
handle: getResizeHandleByIndex(index),
|
|
42
|
+
cursorClass: getResizeCursorClassByIndex(index)
|
|
43
|
+
};
|
|
44
|
+
});
|
|
45
|
+
};
|
|
46
|
+
export const IS_RESIZING = new WeakMap();
|
|
47
|
+
export const isResizing = (board) => {
|
|
48
|
+
return !!IS_RESIZING.get(board);
|
|
49
|
+
};
|
|
50
|
+
export const isResizingByCondition = (board, match) => {
|
|
51
|
+
return isResizing(board) && match(IS_RESIZING.get(board));
|
|
52
|
+
};
|
|
53
|
+
export const addResizing = (board, element, key) => {
|
|
54
|
+
PlaitBoard.getBoardContainer(board).classList.add(`${key}-resizing`);
|
|
55
|
+
IS_RESIZING.set(board, element);
|
|
56
|
+
};
|
|
57
|
+
export const removeResizing = (board, key) => {
|
|
58
|
+
PlaitBoard.getBoardContainer(board).classList.remove(`${key}-resizing`);
|
|
59
|
+
IS_RESIZING.delete(board);
|
|
60
|
+
};
|
|
61
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVzaXplLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcGFja2FnZXMvY29tbW9uL3NyYy91dGlscy9yZXNpemUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBdUIsZUFBZSxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBQy9FLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxZQUFZLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUV0RSxNQUFNLHNCQUFzQixHQUFHLENBQUMsS0FBYSxFQUFFLEVBQUU7SUFDN0MsUUFBUSxLQUFLLEVBQUU7UUFDWCxLQUFLLENBQUM7WUFDRixPQUFPLFlBQVksQ0FBQyxFQUFFLENBQUM7UUFDM0IsS0FBSyxDQUFDO1lBQ0YsT0FBTyxZQUFZLENBQUMsRUFBRSxDQUFDO1FBQzNCLEtBQUssQ0FBQztZQUNGLE9BQU8sWUFBWSxDQUFDLEVBQUUsQ0FBQztRQUMzQixLQUFLLENBQUM7WUFDRixPQUFPLFlBQVksQ0FBQyxFQUFFLENBQUM7UUFDM0I7WUFDSSxPQUFPLElBQUksQ0FBQztLQUNuQjtBQUNMLENBQUMsQ0FBQztBQUVGLE1BQU0sMkJBQTJCLEdBQUcsQ0FBQyxLQUFhLEVBQUUsRUFBRTtJQUNsRCxRQUFRLEtBQUssRUFBRTtRQUNYLEtBQUssQ0FBQztZQUNGLE9BQU8saUJBQWlCLENBQUMsSUFBSSxDQUFDO1FBQ2xDLEtBQUssQ0FBQztZQUNGLE9BQU8saUJBQWlCLENBQUMsSUFBSSxDQUFDO1FBQ2xDLEtBQUssQ0FBQztZQUNGLE9BQU8saUJBQWlCLENBQUMsSUFBSSxDQUFDO1FBQ2xDLEtBQUssQ0FBQztZQUNGLE9BQU8saUJBQWlCLENBQUMsSUFBSSxDQUFDO1FBQ2xDO1lBQ0ksT0FBTyxJQUFJLENBQUM7S0FDbkI7QUFDTCxDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSw0QkFBNEIsR0FBRyxDQUFDLFNBQTBCLEVBQUUsUUFBZ0IsRUFBRSxFQUFFO0lBQ3pGLE1BQU0sT0FBTyxHQUFHLGVBQWUsQ0FBQyxlQUFlLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDM0QsT0FBTyxPQUFPLENBQUMsR0FBRyxDQUFDLENBQUMsTUFBTSxFQUFFLEtBQWEsRUFBRSxFQUFFO1FBQ3pDLE9BQU87WUFDSCxTQUFTLEVBQUU7Z0JBQ1AsQ0FBQyxFQUFFLE1BQU0sQ0FBQyxDQUFDLENBQUMsR0FBRyxRQUFRLEdBQUcsQ0FBQztnQkFDM0IsQ0FBQyxFQUFFLE1BQU0sQ0FBQyxDQUFDLENBQUMsR0FBRyxRQUFRLEdBQUcsQ0FBQztnQkFDM0IsS0FBSyxFQUFFLFFBQVE7Z0JBQ2YsTUFBTSxFQUFFLFFBQVE7YUFDbkI7WUFDRCxNQUFNLEVBQUUsc0JBQXNCLENBQUMsS0FBSyxDQUFpQjtZQUNyRCxXQUFXLEVBQUUsMkJBQTJCLENBQUMsS0FBSyxDQUFzQjtTQUN2RSxDQUFDO0lBQ04sQ0FBQyxDQUFDLENBQUM7QUFDUCxDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxXQUFXLEdBQUcsSUFBSSxPQUFPLEVBQTRCLENBQUM7QUFFbkUsTUFBTSxDQUFDLE1BQU0sVUFBVSxHQUFHLENBQUMsS0FBaUIsRUFBRSxFQUFFO0lBQzVDLE9BQU8sQ0FBQyxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUM7QUFDcEMsQ0FBQyxDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0scUJBQXFCLEdBQUcsQ0FBQyxLQUFpQixFQUFFLEtBQXlDLEVBQUUsRUFBRTtJQUNsRyxPQUFPLFVBQVUsQ0FBQyxLQUFLLENBQUMsSUFBSSxLQUFLLENBQUMsV0FBVyxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQWlCLENBQUMsQ0FBQztBQUM5RSxDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxXQUFXLEdBQUcsQ0FBQyxLQUFpQixFQUFFLE9BQXFCLEVBQUUsR0FBVyxFQUFFLEVBQUU7SUFDakYsVUFBVSxDQUFDLGlCQUFpQixDQUFDLEtBQUssQ0FBQyxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsR0FBRyxHQUFHLFdBQVcsQ0FBQyxDQUFDO0lBQ3JFLFdBQVcsQ0FBQyxHQUFHLENBQUMsS0FBSyxFQUFFLE9BQU8sQ0FBQyxDQUFDO0FBQ3BDLENBQUMsQ0FBQztBQUVGLE1BQU0sQ0FBQyxNQUFNLGNBQWMsR0FBRyxDQUFDLEtBQWlCLEVBQUUsR0FBVyxFQUFFLEVBQUU7SUFDN0QsVUFBVSxDQUFDLGlCQUFpQixDQUFDLEtBQUssQ0FBQyxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsR0FBRyxHQUFHLFdBQVcsQ0FBQyxDQUFDO0lBQ3hFLFdBQVcsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUM7QUFDOUIsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUGxhaXRCb2FyZCwgUGxhaXRFbGVtZW50LCBQb2ludCwgUmVjdGFuZ2xlQ2xpZW50IH0gZnJvbSAnQHBsYWl0L2NvcmUnO1xuaW1wb3J0IHsgUmVzaXplQ3Vyc29yQ2xhc3MsIFJlc2l6ZUhhbmRsZSB9IGZyb20gJy4uL2NvbnN0YW50cy9yZXNpemUnO1xuXG5jb25zdCBnZXRSZXNpemVIYW5kbGVCeUluZGV4ID0gKGluZGV4OiBudW1iZXIpID0+IHtcbiAgICBzd2l0Y2ggKGluZGV4KSB7XG4gICAgICAgIGNhc2UgMDpcbiAgICAgICAgICAgIHJldHVybiBSZXNpemVIYW5kbGUubnc7XG4gICAgICAgIGNhc2UgMTpcbiAgICAgICAgICAgIHJldHVybiBSZXNpemVIYW5kbGUubmU7XG4gICAgICAgIGNhc2UgMjpcbiAgICAgICAgICAgIHJldHVybiBSZXNpemVIYW5kbGUuc2U7XG4gICAgICAgIGNhc2UgMzpcbiAgICAgICAgICAgIHJldHVybiBSZXNpemVIYW5kbGUuc3c7XG4gICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG59O1xuXG5jb25zdCBnZXRSZXNpemVDdXJzb3JDbGFzc0J5SW5kZXggPSAoaW5kZXg6IG51bWJlcikgPT4ge1xuICAgIHN3aXRjaCAoaW5kZXgpIHtcbiAgICAgICAgY2FzZSAwOlxuICAgICAgICAgICAgcmV0dXJuIFJlc2l6ZUN1cnNvckNsYXNzLm53c2U7XG4gICAgICAgIGNhc2UgMTpcbiAgICAgICAgICAgIHJldHVybiBSZXNpemVDdXJzb3JDbGFzcy5uZXN3O1xuICAgICAgICBjYXNlIDI6XG4gICAgICAgICAgICByZXR1cm4gUmVzaXplQ3Vyc29yQ2xhc3MubndzZTtcbiAgICAgICAgY2FzZSAzOlxuICAgICAgICAgICAgcmV0dXJuIFJlc2l6ZUN1cnNvckNsYXNzLm5lc3c7XG4gICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0UmVjdGFuZ2xlUmVzaXplSGFuZGxlUmVmcyA9IChyZWN0YW5nbGU6IFJlY3RhbmdsZUNsaWVudCwgZGlhbWV0ZXI6IG51bWJlcikgPT4ge1xuICAgIGNvbnN0IGNlbnRlcnMgPSBSZWN0YW5nbGVDbGllbnQuZ2V0Q29ybmVyUG9pbnRzKHJlY3RhbmdsZSk7XG4gICAgcmV0dXJuIGNlbnRlcnMubWFwKChjZW50ZXIsIGluZGV4OiBudW1iZXIpID0+IHtcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIHJlY3RhbmdsZToge1xuICAgICAgICAgICAgICAgIHg6IGNlbnRlclswXSAtIGRpYW1ldGVyIC8gMixcbiAgICAgICAgICAgICAgICB5OiBjZW50ZXJbMV0gLSBkaWFtZXRlciAvIDIsXG4gICAgICAgICAgICAgICAgd2lkdGg6IGRpYW1ldGVyLFxuICAgICAgICAgICAgICAgIGhlaWdodDogZGlhbWV0ZXJcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICBoYW5kbGU6IGdldFJlc2l6ZUhhbmRsZUJ5SW5kZXgoaW5kZXgpIGFzIFJlc2l6ZUhhbmRsZSxcbiAgICAgICAgICAgIGN1cnNvckNsYXNzOiBnZXRSZXNpemVDdXJzb3JDbGFzc0J5SW5kZXgoaW5kZXgpIGFzIFJlc2l6ZUN1cnNvckNsYXNzXG4gICAgICAgIH07XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgSVNfUkVTSVpJTkcgPSBuZXcgV2Vha01hcDxQbGFpdEJvYXJkLCBQbGFpdEVsZW1lbnQ+KCk7XG5cbmV4cG9ydCBjb25zdCBpc1Jlc2l6aW5nID0gKGJvYXJkOiBQbGFpdEJvYXJkKSA9PiB7XG4gICAgcmV0dXJuICEhSVNfUkVTSVpJTkcuZ2V0KGJvYXJkKTtcbn07XG5cbmV4cG9ydCBjb25zdCBpc1Jlc2l6aW5nQnlDb25kaXRpb24gPSAoYm9hcmQ6IFBsYWl0Qm9hcmQsIG1hdGNoOiAoZWxlbWVudDogUGxhaXRFbGVtZW50KSA9PiBib29sZWFuKSA9PiB7XG4gICAgcmV0dXJuIGlzUmVzaXppbmcoYm9hcmQpICYmIG1hdGNoKElTX1JFU0laSU5HLmdldChib2FyZCkgYXMgUGxhaXRFbGVtZW50KTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRSZXNpemluZyA9IChib2FyZDogUGxhaXRCb2FyZCwgZWxlbWVudDogUGxhaXRFbGVtZW50LCBrZXk6IHN0cmluZykgPT4ge1xuICAgIFBsYWl0Qm9hcmQuZ2V0Qm9hcmRDb250YWluZXIoYm9hcmQpLmNsYXNzTGlzdC5hZGQoYCR7a2V5fS1yZXNpemluZ2ApO1xuICAgIElTX1JFU0laSU5HLnNldChib2FyZCwgZWxlbWVudCk7XG59O1xuXG5leHBvcnQgY29uc3QgcmVtb3ZlUmVzaXppbmcgPSAoYm9hcmQ6IFBsYWl0Qm9hcmQsIGtleTogc3RyaW5nKSA9PiB7XG4gICAgUGxhaXRCb2FyZC5nZXRCb2FyZENvbnRhaW5lcihib2FyZCkuY2xhc3NMaXN0LnJlbW92ZShgJHtrZXl9LXJlc2l6aW5nYCk7XG4gICAgSVNfUkVTSVpJTkcuZGVsZXRlKGJvYXJkKTtcbn07XG4iXX0=
|
|
@@ -65,11 +65,18 @@ const drawRectangle = (board, rectangle, options) => {
|
|
|
65
65
|
return rectangleG;
|
|
66
66
|
};
|
|
67
67
|
const getRectangleByPoints = (points) => {
|
|
68
|
+
let minX = Infinity, maxX = -Infinity, minY = Infinity, maxY = -Infinity;
|
|
69
|
+
points.forEach(point => {
|
|
70
|
+
minX = Math.min(point[0], minX);
|
|
71
|
+
maxX = Math.max(point[0], maxX);
|
|
72
|
+
minY = Math.min(point[1], minY);
|
|
73
|
+
maxY = Math.max(point[1], maxY);
|
|
74
|
+
});
|
|
68
75
|
return {
|
|
69
|
-
x:
|
|
70
|
-
y:
|
|
71
|
-
width:
|
|
72
|
-
height:
|
|
76
|
+
x: minX,
|
|
77
|
+
y: minY,
|
|
78
|
+
width: maxX - minX,
|
|
79
|
+
height: maxY - minY
|
|
73
80
|
};
|
|
74
81
|
};
|
|
75
82
|
|
|
@@ -129,9 +136,17 @@ const CommonTransforms = {
|
|
|
129
136
|
removeElements
|
|
130
137
|
};
|
|
131
138
|
|
|
132
|
-
const normalizeShapePoints = (points) => {
|
|
133
|
-
|
|
134
|
-
|
|
139
|
+
const normalizeShapePoints = (points, shift = false) => {
|
|
140
|
+
let start = points[0];
|
|
141
|
+
let end = points[1];
|
|
142
|
+
if (shift) {
|
|
143
|
+
const width = Math.abs(start[0] - end[0]);
|
|
144
|
+
const height = Math.abs(start[1] - end[1]);
|
|
145
|
+
const edge = Math.max(height, width);
|
|
146
|
+
end = [start[0] + (end[0] > start[0] ? edge : -edge), start[1] + (end[1] > start[1] ? edge : -edge)];
|
|
147
|
+
}
|
|
148
|
+
const leftTopPoint = [Math.min(start[0], end[0]), Math.min(start[1], end[1])];
|
|
149
|
+
let rightBottomPoint = [Math.max(start[0], end[0]), Math.max(start[1], end[1])];
|
|
135
150
|
return [leftTopPoint, rightBottomPoint];
|
|
136
151
|
};
|
|
137
152
|
|
|
@@ -182,13 +197,16 @@ const IS_RESIZING = new WeakMap();
|
|
|
182
197
|
const isResizing = (board) => {
|
|
183
198
|
return !!IS_RESIZING.get(board);
|
|
184
199
|
};
|
|
185
|
-
const
|
|
200
|
+
const isResizingByCondition = (board, match) => {
|
|
201
|
+
return isResizing(board) && match(IS_RESIZING.get(board));
|
|
202
|
+
};
|
|
203
|
+
const addResizing = (board, element, key) => {
|
|
186
204
|
PlaitBoard.getBoardContainer(board).classList.add(`${key}-resizing`);
|
|
187
|
-
IS_RESIZING.set(board,
|
|
205
|
+
IS_RESIZING.set(board, element);
|
|
188
206
|
};
|
|
189
207
|
const removeResizing = (board, key) => {
|
|
190
208
|
PlaitBoard.getBoardContainer(board).classList.remove(`${key}-resizing`);
|
|
191
|
-
IS_RESIZING.
|
|
209
|
+
IS_RESIZING.delete(board);
|
|
192
210
|
};
|
|
193
211
|
|
|
194
212
|
const generalCanResize = (board, event) => {
|
|
@@ -236,7 +254,7 @@ const withResize = (board, options) => {
|
|
|
236
254
|
const endPoint = transformPoint(board, toPoint(event.x, event.y, PlaitBoard.getHost(board)));
|
|
237
255
|
const distance = distanceBetweenPointAndPoint(startPoint[0], startPoint[1], endPoint[0], endPoint[1]);
|
|
238
256
|
if (distance > PRESS_AND_MOVE_BUFFER) {
|
|
239
|
-
addResizing(board, options.key);
|
|
257
|
+
addResizing(board, resizeDetectResult.element, options.key);
|
|
240
258
|
MERGING.set(board, true);
|
|
241
259
|
}
|
|
242
260
|
}
|
|
@@ -278,6 +296,7 @@ const withResize = (board, options) => {
|
|
|
278
296
|
board.globalPointerUp = (event) => {
|
|
279
297
|
globalPointerUp(event);
|
|
280
298
|
if (isResizing(board) || resizeDetectResult) {
|
|
299
|
+
options.endResize && options.endResize(resizeRef);
|
|
281
300
|
removeResizing(board, options.key);
|
|
282
301
|
startPoint = null;
|
|
283
302
|
resizeDetectResult = null;
|
|
@@ -356,7 +375,7 @@ const getPoints = (source, sourcePosition, target, targetPosition, offset) => {
|
|
|
356
375
|
}
|
|
357
376
|
}
|
|
358
377
|
}
|
|
359
|
-
return [source, ...points, target];
|
|
378
|
+
return [source, sourceGapped, ...points, targetGapped, target];
|
|
360
379
|
};
|
|
361
380
|
const getDirection = (source, sourcePosition = Direction.bottom, target) => {
|
|
362
381
|
if (sourcePosition === Direction.left || sourcePosition === Direction.right) {
|
|
@@ -384,6 +403,82 @@ function getEdgeCenter({ sourceX, sourceY, targetX, targetY }) {
|
|
|
384
403
|
const centerY = targetY < sourceY ? targetY + yOffset : targetY - yOffset;
|
|
385
404
|
return [centerX, centerY, xOffset, yOffset];
|
|
386
405
|
}
|
|
406
|
+
function getDirectionByPoint(point, defaultDirection) {
|
|
407
|
+
if (point[0] === 0) {
|
|
408
|
+
return Direction.left;
|
|
409
|
+
}
|
|
410
|
+
if (point[0] === 1) {
|
|
411
|
+
return Direction.right;
|
|
412
|
+
}
|
|
413
|
+
if (point[1] === 0) {
|
|
414
|
+
return Direction.top;
|
|
415
|
+
}
|
|
416
|
+
if (point[1] === 1) {
|
|
417
|
+
return Direction.bottom;
|
|
418
|
+
}
|
|
419
|
+
return defaultDirection;
|
|
420
|
+
}
|
|
421
|
+
function getOppositeDirection(direction) {
|
|
422
|
+
switch (direction) {
|
|
423
|
+
case Direction.left:
|
|
424
|
+
return Direction.right;
|
|
425
|
+
case Direction.right:
|
|
426
|
+
return Direction.left;
|
|
427
|
+
case Direction.top:
|
|
428
|
+
return Direction.bottom;
|
|
429
|
+
case Direction.bottom:
|
|
430
|
+
return Direction.top;
|
|
431
|
+
}
|
|
432
|
+
}
|
|
433
|
+
function getPointOnPolyline(points, ratio) {
|
|
434
|
+
const totalLength = calculatePolylineLength(points);
|
|
435
|
+
const targetDistance = totalLength * ratio;
|
|
436
|
+
let accumulatedDistance = 0;
|
|
437
|
+
for (let i = 0; i < points.length - 1; i++) {
|
|
438
|
+
const [x1, y1] = points[i];
|
|
439
|
+
const [x2, y2] = points[i + 1];
|
|
440
|
+
const segmentLength = distanceBetweenPointAndPoint(x1, y1, x2, y2);
|
|
441
|
+
if (accumulatedDistance + segmentLength >= targetDistance) {
|
|
442
|
+
const remainingDistance = targetDistance - accumulatedDistance;
|
|
443
|
+
const ratioInSegment = remainingDistance / segmentLength;
|
|
444
|
+
const targetX = x1 + (x2 - x1) * ratioInSegment;
|
|
445
|
+
const targetY = y1 + (y2 - y1) * ratioInSegment;
|
|
446
|
+
return [targetX, targetY];
|
|
447
|
+
}
|
|
448
|
+
accumulatedDistance += segmentLength;
|
|
449
|
+
}
|
|
450
|
+
return points[points.length - 1];
|
|
451
|
+
}
|
|
452
|
+
function calculatePolylineLength(points) {
|
|
453
|
+
let length = 0;
|
|
454
|
+
for (let i = 0; i < points.length - 1; i++) {
|
|
455
|
+
const [x1, y1] = points[i];
|
|
456
|
+
const [x2, y2] = points[i + 1];
|
|
457
|
+
length += distanceBetweenPointAndPoint(x1, y1, x2, y2);
|
|
458
|
+
}
|
|
459
|
+
return length;
|
|
460
|
+
}
|
|
461
|
+
function getRatioByPoint(points, point) {
|
|
462
|
+
const totalLength = calculatePolylineLength(points);
|
|
463
|
+
let distance = 0;
|
|
464
|
+
for (let i = 0; i < points.length - 1; i++) {
|
|
465
|
+
const isOverlap = isPointOnLineSegment(point, points[i], points[i + 1]);
|
|
466
|
+
if (isOverlap) {
|
|
467
|
+
distance += distanceBetweenPointAndPoint(point[0], point[1], points[i][0], points[i][1]);
|
|
468
|
+
return distance / totalLength;
|
|
469
|
+
}
|
|
470
|
+
else {
|
|
471
|
+
distance += distanceBetweenPointAndPoint(points[i][0], points[i][1], points[i + 1][0], points[i + 1][1]);
|
|
472
|
+
}
|
|
473
|
+
}
|
|
474
|
+
throw new Error('Cannot get ratio by point');
|
|
475
|
+
}
|
|
476
|
+
function isPointOnLineSegment(point, startPoint, endPoint) {
|
|
477
|
+
const distanceToStart = distanceBetweenPointAndPoint(point[0], point[1], startPoint[0], startPoint[1]);
|
|
478
|
+
const distanceToEnd = distanceBetweenPointAndPoint(point[0], point[1], endPoint[0], endPoint[1]);
|
|
479
|
+
const segmentLength = distanceBetweenPointAndPoint(startPoint[0], startPoint[1], endPoint[0], endPoint[1]);
|
|
480
|
+
return Math.abs(distanceToStart + distanceToEnd - segmentLength) < 0.1;
|
|
481
|
+
}
|
|
387
482
|
|
|
388
483
|
function isVirtualKey(e) {
|
|
389
484
|
const isMod = e.ctrlKey || e.metaKey;
|
|
@@ -417,5 +512,5 @@ const isSpaceHotkey = (event) => {
|
|
|
417
512
|
* Generated bundle index. Do not edit.
|
|
418
513
|
*/
|
|
419
514
|
|
|
420
|
-
export { ActiveGenerator, BASE, CommonTransforms, Direction, Generator, IS_RESIZING, MediaKeys, PRIMARY_COLOR, RESIZE_HANDLE_DIAMETER, ResizeCursorClass, ResizeHandle, addResizing, drawRectangle, getEdgeCenter, getPoints, getRectangleByPoints, getRectangleResizeHandleRefs, hasAfterDraw, isEnterHotkey, isExpandHotkey, isResizing, isSpaceHotkey, isTabHotkey, isVirtualKey, normalizeShapePoints, removeResizing, withResize };
|
|
515
|
+
export { ActiveGenerator, BASE, CommonTransforms, Direction, Generator, IS_RESIZING, MediaKeys, PRIMARY_COLOR, RESIZE_HANDLE_DIAMETER, ResizeCursorClass, ResizeHandle, addResizing, drawRectangle, getDirectionByPoint, getEdgeCenter, getOppositeDirection, getPointOnPolyline, getPoints, getRatioByPoint, getRectangleByPoints, getRectangleResizeHandleRefs, hasAfterDraw, isEnterHotkey, isExpandHotkey, isPointOnLineSegment, isResizing, isResizingByCondition, isSpaceHotkey, isTabHotkey, isVirtualKey, normalizeShapePoints, removeResizing, withResize };
|
|
421
516
|
//# sourceMappingURL=plait-common.mjs.map
|