@plait/core 0.24.0-next.2 → 0.24.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.
Files changed (102) hide show
  1. package/board/board.component.d.ts +1 -1
  2. package/core/children/children.component.d.ts +1 -1
  3. package/core/element/element.component.d.ts +1 -1
  4. package/core/element/plugin-element.d.ts +1 -1
  5. package/core/island/island-base.component.d.ts +1 -1
  6. package/{esm2020 → esm2022}/board/board.component.mjs +6 -6
  7. package/{esm2020 → esm2022}/core/children/children.component.mjs +6 -6
  8. package/{esm2020 → esm2022}/core/element/element.component.mjs +4 -4
  9. package/{esm2020 → esm2022}/core/element/plugin-element.mjs +4 -4
  10. package/{esm2020 → esm2022}/core/island/island-base.component.mjs +7 -7
  11. package/{esm2020 → esm2022}/interfaces/rectangle-client.mjs +11 -3
  12. package/{esm2020 → esm2022}/plait.module.mjs +5 -5
  13. package/esm2022/plugins/with-moving.mjs +98 -0
  14. package/{esm2020 → esm2022}/services/image-context.service.mjs +4 -4
  15. package/esm2022/utils/dom/common.mjs +68 -0
  16. package/esm2022/utils/element.mjs +53 -0
  17. package/esm2022/utils/math.mjs +128 -0
  18. package/{fesm2020 → fesm2022}/plait-core.mjs +108 -33
  19. package/fesm2022/plait-core.mjs.map +1 -0
  20. package/interfaces/rectangle-client.d.ts +1 -0
  21. package/package.json +14 -14
  22. package/utils/dom/common.d.ts +4 -0
  23. package/utils/element.d.ts +1 -0
  24. package/utils/math.d.ts +2 -0
  25. package/esm2020/plugins/with-moving.mjs +0 -97
  26. package/esm2020/utils/dom/common.mjs +0 -53
  27. package/esm2020/utils/element.mjs +0 -44
  28. package/esm2020/utils/math.mjs +0 -85
  29. package/fesm2015/plait-core.mjs +0 -3498
  30. package/fesm2015/plait-core.mjs.map +0 -1
  31. package/fesm2020/plait-core.mjs.map +0 -1
  32. /package/{esm2020 → esm2022}/board/board.component.interface.mjs +0 -0
  33. /package/{esm2020 → esm2022}/constants/index.mjs +0 -0
  34. /package/{esm2020 → esm2022}/constants/keycodes.mjs +0 -0
  35. /package/{esm2020 → esm2022}/constants/resize.mjs +0 -0
  36. /package/{esm2020 → esm2022}/constants/selection.mjs +0 -0
  37. /package/{esm2020 → esm2022}/core/children/effect.mjs +0 -0
  38. /package/{esm2020 → esm2022}/core/element/context-change.mjs +0 -0
  39. /package/{esm2020 → esm2022}/core/element/context.mjs +0 -0
  40. /package/{esm2020 → esm2022}/interfaces/board.mjs +0 -0
  41. /package/{esm2020 → esm2022}/interfaces/custom-types.mjs +0 -0
  42. /package/{esm2020 → esm2022}/interfaces/element.mjs +0 -0
  43. /package/{esm2020 → esm2022}/interfaces/history.mjs +0 -0
  44. /package/{esm2020 → esm2022}/interfaces/index.mjs +0 -0
  45. /package/{esm2020 → esm2022}/interfaces/node.mjs +0 -0
  46. /package/{esm2020 → esm2022}/interfaces/operation.mjs +0 -0
  47. /package/{esm2020 → esm2022}/interfaces/path-ref.mjs +0 -0
  48. /package/{esm2020 → esm2022}/interfaces/path.mjs +0 -0
  49. /package/{esm2020 → esm2022}/interfaces/plugin-key.mjs +0 -0
  50. /package/{esm2020 → esm2022}/interfaces/plugin.mjs +0 -0
  51. /package/{esm2020 → esm2022}/interfaces/point.mjs +0 -0
  52. /package/{esm2020 → esm2022}/interfaces/pointer.mjs +0 -0
  53. /package/{esm2020 → esm2022}/interfaces/selection.mjs +0 -0
  54. /package/{esm2020 → esm2022}/interfaces/theme.mjs +0 -0
  55. /package/{esm2020 → esm2022}/interfaces/viewport.mjs +0 -0
  56. /package/{esm2020 → esm2022}/plait-core.mjs +0 -0
  57. /package/{esm2020 → esm2022}/plugins/create-board.mjs +0 -0
  58. /package/{esm2020 → esm2022}/plugins/with-board.mjs +0 -0
  59. /package/{esm2020 → esm2022}/plugins/with-hand.mjs +0 -0
  60. /package/{esm2020 → esm2022}/plugins/with-history.mjs +0 -0
  61. /package/{esm2020 → esm2022}/plugins/with-hotkey.mjs +0 -0
  62. /package/{esm2020 → esm2022}/plugins/with-options.mjs +0 -0
  63. /package/{esm2020 → esm2022}/plugins/with-selection.mjs +0 -0
  64. /package/{esm2020 → esm2022}/plugins/with-viewport.mjs +0 -0
  65. /package/{esm2020 → esm2022}/public-api.mjs +0 -0
  66. /package/{esm2020 → esm2022}/testing/core/create-board.mjs +0 -0
  67. /package/{esm2020 → esm2022}/testing/core/fake-weak-map.mjs +0 -0
  68. /package/{esm2020 → esm2022}/testing/core/index.mjs +0 -0
  69. /package/{esm2020 → esm2022}/testing/fake-events/event-objects.mjs +0 -0
  70. /package/{esm2020 → esm2022}/testing/fake-events/index.mjs +0 -0
  71. /package/{esm2020 → esm2022}/testing/index.mjs +0 -0
  72. /package/{esm2020 → esm2022}/testing/test-element.mjs +0 -0
  73. /package/{esm2020 → esm2022}/transforms/board.mjs +0 -0
  74. /package/{esm2020 → esm2022}/transforms/general.mjs +0 -0
  75. /package/{esm2020 → esm2022}/transforms/index.mjs +0 -0
  76. /package/{esm2020 → esm2022}/transforms/node.mjs +0 -0
  77. /package/{esm2020 → esm2022}/transforms/selection.mjs +0 -0
  78. /package/{esm2020 → esm2022}/transforms/theme.mjs +0 -0
  79. /package/{esm2020 → esm2022}/transforms/viewport.mjs +0 -0
  80. /package/{esm2020 → esm2022}/utils/board.mjs +0 -0
  81. /package/{esm2020 → esm2022}/utils/clipboard.mjs +0 -0
  82. /package/{esm2020 → esm2022}/utils/common.mjs +0 -0
  83. /package/{esm2020 → esm2022}/utils/dom/environment.mjs +0 -0
  84. /package/{esm2020 → esm2022}/utils/dom/foreign.mjs +0 -0
  85. /package/{esm2020 → esm2022}/utils/dom/index.mjs +0 -0
  86. /package/{esm2020 → esm2022}/utils/draw/arrow.mjs +0 -0
  87. /package/{esm2020 → esm2022}/utils/draw/circle.mjs +0 -0
  88. /package/{esm2020 → esm2022}/utils/draw/line.mjs +0 -0
  89. /package/{esm2020 → esm2022}/utils/draw/rectangle.mjs +0 -0
  90. /package/{esm2020 → esm2022}/utils/environment.mjs +0 -0
  91. /package/{esm2020 → esm2022}/utils/helper.mjs +0 -0
  92. /package/{esm2020 → esm2022}/utils/history.mjs +0 -0
  93. /package/{esm2020 → esm2022}/utils/hotkeys.mjs +0 -0
  94. /package/{esm2020 → esm2022}/utils/id-creator.mjs +0 -0
  95. /package/{esm2020 → esm2022}/utils/index.mjs +0 -0
  96. /package/{esm2020 → esm2022}/utils/moving-element.mjs +0 -0
  97. /package/{esm2020 → esm2022}/utils/selected-element.mjs +0 -0
  98. /package/{esm2020 → esm2022}/utils/to-image.mjs +0 -0
  99. /package/{esm2020 → esm2022}/utils/touch.mjs +0 -0
  100. /package/{esm2020 → esm2022}/utils/tree.mjs +0 -0
  101. /package/{esm2020 → esm2022}/utils/viewport.mjs +0 -0
  102. /package/{esm2020 → esm2022}/utils/weak-maps.mjs +0 -0
@@ -21,4 +21,5 @@ export declare const RectangleClient: {
21
21
  };
22
22
  isEqual: (rectangle: RectangleClient, otherRectangle: RectangleClient) => boolean;
23
23
  getCornerPoints: (rectangle: RectangleClient) => [Point, Point, Point, Point];
24
+ getEdgeCenterPoints: (rectangle: RectangleClient) => [Point, Point, Point, Point];
24
25
  };
package/package.json CHANGED
@@ -1,31 +1,31 @@
1
1
  {
2
2
  "name": "@plait/core",
3
- "version": "0.24.0-next.2",
3
+ "version": "0.24.0-next.3",
4
4
  "peerDependencies": {
5
- "@angular/common": "^15.2.2",
6
- "@angular/core": "^15.2.2"
5
+ "@angular/common": "^16.0.0",
6
+ "@angular/core": "^16.0.0"
7
7
  },
8
8
  "dependencies": {
9
9
  "tslib": "^2.3.0"
10
10
  },
11
- "module": "fesm2015/plait-core.mjs",
12
- "es2020": "fesm2020/plait-core.mjs",
13
- "esm2020": "esm2020/plait-core.mjs",
14
- "fesm2020": "fesm2020/plait-core.mjs",
15
- "fesm2015": "fesm2015/plait-core.mjs",
16
- "typings": "index.d.ts",
17
11
  "exports": {
12
+ "./styles/styles": {
13
+ "sass": "./styles/styles.scss"
14
+ },
15
+ "./styles/*": {
16
+ "sass": "./styles/*"
17
+ },
18
18
  "./package.json": {
19
19
  "default": "./package.json"
20
20
  },
21
21
  ".": {
22
22
  "types": "./index.d.ts",
23
- "esm2020": "./esm2020/plait-core.mjs",
24
- "es2020": "./fesm2020/plait-core.mjs",
25
- "es2015": "./fesm2015/plait-core.mjs",
26
- "node": "./fesm2015/plait-core.mjs",
27
- "default": "./fesm2020/plait-core.mjs"
23
+ "esm2022": "./esm2022/plait-core.mjs",
24
+ "esm": "./esm2022/plait-core.mjs",
25
+ "default": "./fesm2022/plait-core.mjs"
28
26
  }
29
27
  },
28
+ "module": "fesm2022/plait-core.mjs",
29
+ "typings": "index.d.ts",
30
30
  "sideEffects": false
31
31
  }
@@ -1,8 +1,12 @@
1
+ import { Options } from 'roughjs/bin/core';
2
+ import { RectangleClient } from '../../interfaces';
1
3
  import { Point } from '../../interfaces/point';
2
4
  export declare const NS = "http://www.w3.org/2000/svg";
3
5
  export declare function toPoint(x: number, y: number, container: SVGElement): Point;
4
6
  export declare function createG(): SVGGElement;
5
7
  export declare function createPath(): SVGPathElement;
8
+ export declare function createRect(rectangle: RectangleClient, options?: Options): SVGRectElement;
9
+ export declare function createMask(): SVGMaskElement;
6
10
  export declare function createSVG(): SVGSVGElement;
7
11
  export declare function createText(x: number, y: number, fill: string, textContent: string): SVGTextElement;
8
12
  /**
@@ -1,3 +1,4 @@
1
1
  import { PlaitBoard, PlaitElement, RectangleClient } from '../interfaces';
2
2
  export declare function getRectangleByElements(board: PlaitBoard, elements: PlaitElement[], recursion: boolean): RectangleClient;
3
3
  export declare function getBoardRectangle(board: PlaitBoard): RectangleClient;
4
+ export declare function getElementById<T extends PlaitElement = PlaitElement>(board: PlaitBoard, id: string): T | null;
package/utils/math.d.ts CHANGED
@@ -1,7 +1,9 @@
1
1
  import { Point } from '../interfaces';
2
2
  import { RectangleClient } from '../interfaces/rectangle-client';
3
3
  export declare function distanceBetweenPointAndSegment(x: number, y: number, x1: number, y1: number, x2: number, y2: number): number;
4
+ export declare function getNearestPointBetweenPointAndSegment(point: Point, linePoints: [Point, Point]): Point;
4
5
  export declare function distanceBetweenPointAndSegments(points: Point[], point: Point): number;
6
+ export declare function getNearestPointBetweenPointAndSegments(point: Point, points: Point[]): Point;
5
7
  export declare function rotate(x1: number, y1: number, x2: number, y2: number, angle: number): number[];
6
8
  export declare function distanceBetweenPointAndPoint(x1: number, y1: number, x2: number, y2: number): number;
7
9
  export declare function distanceBetweenPointAndRectangle(x: number, y: number, rect: RectangleClient): number;
@@ -1,97 +0,0 @@
1
- import { BOARD_TO_HOST } from '../utils/weak-maps';
2
- import { PlaitBoard } from '../interfaces/board';
3
- import { isInPlaitBoard, transformPoint } from '../utils/board';
4
- import { toPoint } from '../utils/dom/common';
5
- import { Transforms } from '../transforms';
6
- import { getHitElementOfRoot, getSelectedElements } from '../utils/selected-element';
7
- import { PlaitNode } from '../interfaces/node';
8
- import { throttleRAF } from '../utils/common';
9
- import { addMovingElements, removeMovingElements } from '../utils/moving-element';
10
- import { MERGING } from '../interfaces/history';
11
- export function withMoving(board) {
12
- const { pointerDown, pointerMove, globalPointerUp, globalPointerMove } = board;
13
- let offsetX = 0;
14
- let offsetY = 0;
15
- let isPreventDefault = false;
16
- let startPoint;
17
- let activeElements = [];
18
- board.pointerDown = (event) => {
19
- const host = BOARD_TO_HOST.get(board);
20
- const point = transformPoint(board, toPoint(event.x, event.y, host));
21
- const range = { anchor: point, focus: point };
22
- let movableElements = board.children.filter(item => board.isMovable(item));
23
- if (movableElements.length) {
24
- startPoint = point;
25
- const selectedRootElements = getSelectedElements(board).filter(item => movableElements.includes(item));
26
- const hitElement = getHitElementOfRoot(board, movableElements, range);
27
- if (hitElement && selectedRootElements.includes(hitElement)) {
28
- activeElements = selectedRootElements;
29
- }
30
- else if (hitElement) {
31
- activeElements = [hitElement];
32
- }
33
- }
34
- pointerDown(event);
35
- };
36
- board.pointerMove = (event) => {
37
- if (startPoint && activeElements.length && !PlaitBoard.hasBeenTextEditing(board)) {
38
- if (!isPreventDefault) {
39
- isPreventDefault = true;
40
- }
41
- const host = BOARD_TO_HOST.get(board);
42
- const endPoint = transformPoint(board, toPoint(event.x, event.y, host));
43
- offsetX = endPoint[0] - startPoint[0];
44
- offsetY = endPoint[1] - startPoint[1];
45
- const offsetBuffer = 5;
46
- if (Math.abs(offsetX) > offsetBuffer || Math.abs(offsetY) > offsetBuffer) {
47
- throttleRAF(() => {
48
- const currentElements = activeElements.map(activeElement => {
49
- const points = activeElement.points || [];
50
- const [x, y] = activeElement.points[0];
51
- const newPoints = points.map(p => [p[0] + offsetX, p[1] + offsetY]);
52
- const index = board.children.findIndex(item => item.id === activeElement.id);
53
- Transforms.setNode(board, {
54
- points: newPoints
55
- }, [index]);
56
- MERGING.set(board, true);
57
- return PlaitNode.get(board, [index]);
58
- });
59
- PlaitBoard.getBoardContainer(board).classList.add('element-moving');
60
- addMovingElements(board, currentElements);
61
- });
62
- }
63
- }
64
- if (isPreventDefault) {
65
- // 阻止 move 过程中触发画布滚动行为
66
- event.preventDefault();
67
- }
68
- pointerMove(event);
69
- };
70
- board.globalPointerMove = (event) => {
71
- if (startPoint) {
72
- const inPlaitBoardElement = isInPlaitBoard(board, event.x, event.y);
73
- if (!inPlaitBoardElement) {
74
- cancelMove(board);
75
- }
76
- }
77
- globalPointerMove(event);
78
- };
79
- board.globalPointerUp = event => {
80
- isPreventDefault = false;
81
- if (startPoint) {
82
- cancelMove(board);
83
- }
84
- globalPointerUp(event);
85
- };
86
- function cancelMove(board) {
87
- startPoint = null;
88
- offsetX = 0;
89
- offsetY = 0;
90
- activeElements = [];
91
- removeMovingElements(board);
92
- MERGING.set(board, false);
93
- PlaitBoard.getBoardContainer(board).classList.remove('element-moving');
94
- }
95
- return board;
96
- }
97
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"with-moving.js","sourceRoot":"","sources":["../../../../packages/core/src/plugins/with-moving.ts"],"names":[],"mappings":"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","sourcesContent":["import { BOARD_TO_HOST } from '../utils/weak-maps';\nimport { PlaitBoard } from '../interfaces/board';\nimport { isInPlaitBoard, transformPoint } from '../utils/board';\nimport { toPoint } from '../utils/dom/common';\nimport { Point } from '../interfaces/point';\nimport { Transforms } from '../transforms';\nimport { PlaitElement } from '../interfaces/element';\nimport { getHitElementOfRoot, getSelectedElements } from '../utils/selected-element';\nimport { PlaitNode } from '../interfaces/node';\nimport { throttleRAF } from '../utils/common';\nimport { addMovingElements, removeMovingElements } from '../utils/moving-element';\nimport { MERGING } from '../interfaces/history';\nimport { Range } from '../interfaces';\n\nexport function withMoving(board: PlaitBoard) {\n    const { pointerDown, pointerMove, globalPointerUp, globalPointerMove } = board;\n\n    let offsetX = 0;\n    let offsetY = 0;\n    let isPreventDefault = false;\n    let startPoint: Point | null;\n    let activeElements: PlaitElement[] = [];\n\n    board.pointerDown = (event: PointerEvent) => {\n        const host = BOARD_TO_HOST.get(board);\n        const point = transformPoint(board, toPoint(event.x, event.y, host!));\n        const range = { anchor: point, focus: point } as Range;\n        let movableElements = board.children.filter(item => board.isMovable(item));\n        if (movableElements.length) {\n            startPoint = point;\n            const selectedRootElements = getSelectedElements(board).filter(item => movableElements.includes(item));\n            const hitElement = getHitElementOfRoot(board, movableElements, range);\n            if (hitElement && selectedRootElements.includes(hitElement)) {\n                activeElements = selectedRootElements;\n            } else if (hitElement) {\n                activeElements = [hitElement];\n            }\n        }\n\n        pointerDown(event);\n    };\n\n    board.pointerMove = (event: PointerEvent) => {\n        if (startPoint && activeElements.length && !PlaitBoard.hasBeenTextEditing(board)) {\n            if (!isPreventDefault) {\n                isPreventDefault = true;\n            }\n            const host = BOARD_TO_HOST.get(board);\n            const endPoint = transformPoint(board, toPoint(event.x, event.y, host!));\n            offsetX = endPoint[0] - startPoint[0];\n            offsetY = endPoint[1] - startPoint[1];\n            const offsetBuffer = 5;\n            if (Math.abs(offsetX) > offsetBuffer || Math.abs(offsetY) > offsetBuffer) {\n                throttleRAF(() => {\n                    const currentElements = activeElements.map(activeElement => {\n                        const points = activeElement.points || [];\n                        const [x, y] = activeElement.points![0];\n                        const newPoints = points.map(p => [p[0] + offsetX, p[1] + offsetY]) as Point[];\n                        const index = board.children.findIndex(item => item.id === activeElement.id);\n                        Transforms.setNode(\n                            board,\n                            {\n                                points: newPoints\n                            },\n                            [index]\n                        );\n                        MERGING.set(board, true);\n                        return PlaitNode.get(board, [index]);\n                    });\n                    PlaitBoard.getBoardContainer(board).classList.add('element-moving');\n                    addMovingElements(board, currentElements as PlaitElement[]);\n                });\n            }\n        }\n        if (isPreventDefault) {\n            // 阻止 move 过程中触发画布滚动行为\n            event.preventDefault();\n        }\n        pointerMove(event);\n    };\n\n    board.globalPointerMove = (event: PointerEvent) => {\n        if (startPoint) {\n            const inPlaitBoardElement = isInPlaitBoard(board, event.x, event.y);\n            if (!inPlaitBoardElement) {\n                cancelMove(board);\n            }\n        }\n        globalPointerMove(event);\n    };\n\n    board.globalPointerUp = event => {\n        isPreventDefault = false;\n        if (startPoint) {\n            cancelMove(board);\n        }\n        globalPointerUp(event);\n    };\n\n    function cancelMove(board: PlaitBoard) {\n        startPoint = null;\n        offsetX = 0;\n        offsetY = 0;\n        activeElements = [];\n        removeMovingElements(board);\n        MERGING.set(board, false);\n        PlaitBoard.getBoardContainer(board).classList.remove('element-moving');\n    }\n\n    return board;\n}\n"]}
@@ -1,53 +0,0 @@
1
- import { POINTER_BUTTON } from '../../constants';
2
- export const NS = 'http://www.w3.org/2000/svg';
3
- export function toPoint(x, y, container) {
4
- const rect = container.getBoundingClientRect();
5
- return [x - rect.x, y - rect.y];
6
- }
7
- export function createG() {
8
- const newG = document.createElementNS(NS, 'g');
9
- return newG;
10
- }
11
- export function createPath() {
12
- const newG = document.createElementNS(NS, 'path');
13
- return newG;
14
- }
15
- export function createSVG() {
16
- const svg = document.createElementNS(NS, 'svg');
17
- return svg;
18
- }
19
- export function createText(x, y, fill, textContent) {
20
- var text = document.createElementNS(NS, 'text');
21
- text.setAttribute('x', `${x}`);
22
- text.setAttribute('y', `${y}`);
23
- text.setAttribute('fill', fill);
24
- text.textContent = textContent;
25
- return text;
26
- }
27
- /**
28
- * Check if a DOM node is an element node.
29
- */
30
- export const isDOMElement = (value) => {
31
- return isDOMNode(value) && value.nodeType === 1;
32
- };
33
- /**
34
- * Check if a value is a DOM node.
35
- */
36
- export const isDOMNode = (value) => {
37
- return value instanceof window.Node;
38
- };
39
- export const hasInputOrTextareaTarget = (target) => {
40
- if (isDOMElement(target)) {
41
- if (target.tagName === 'INPUT' || target.tagName === 'TEXTAREA') {
42
- return true;
43
- }
44
- }
45
- return false;
46
- };
47
- export const isSecondaryPointer = (event) => {
48
- return event.button === POINTER_BUTTON.SECONDARY;
49
- };
50
- export const isMainPointer = (event) => {
51
- return event.button === POINTER_BUTTON.MAIN;
52
- };
53
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvY29yZS9zcmMvdXRpbHMvZG9tL2NvbW1vbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFHakQsTUFBTSxDQUFDLE1BQU0sRUFBRSxHQUFHLDRCQUE0QixDQUFDO0FBRS9DLE1BQU0sVUFBVSxPQUFPLENBQUMsQ0FBUyxFQUFFLENBQVMsRUFBRSxTQUFxQjtJQUMvRCxNQUFNLElBQUksR0FBRyxTQUFTLENBQUMscUJBQXFCLEVBQUUsQ0FBQztJQUMvQyxPQUFPLENBQUMsQ0FBQyxHQUFHLElBQUksQ0FBQyxDQUFDLEVBQUUsQ0FBQyxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztBQUNwQyxDQUFDO0FBRUQsTUFBTSxVQUFVLE9BQU87SUFDbkIsTUFBTSxJQUFJLEdBQUcsUUFBUSxDQUFDLGVBQWUsQ0FBQyxFQUFFLEVBQUUsR0FBRyxDQUFDLENBQUM7SUFDL0MsT0FBTyxJQUFJLENBQUM7QUFDaEIsQ0FBQztBQUVELE1BQU0sVUFBVSxVQUFVO0lBQ3RCLE1BQU0sSUFBSSxHQUFHLFFBQVEsQ0FBQyxlQUFlLENBQUMsRUFBRSxFQUFFLE1BQU0sQ0FBQyxDQUFDO0lBQ2xELE9BQU8sSUFBSSxDQUFDO0FBQ2hCLENBQUM7QUFFRCxNQUFNLFVBQVUsU0FBUztJQUNyQixNQUFNLEdBQUcsR0FBRyxRQUFRLENBQUMsZUFBZSxDQUFDLEVBQUUsRUFBRSxLQUFLLENBQUMsQ0FBQztJQUNoRCxPQUFPLEdBQUcsQ0FBQztBQUNmLENBQUM7QUFFRCxNQUFNLFVBQVUsVUFBVSxDQUFDLENBQVMsRUFBRSxDQUFTLEVBQUUsSUFBWSxFQUFFLFdBQW1CO0lBQzlFLElBQUksSUFBSSxHQUFHLFFBQVEsQ0FBQyxlQUFlLENBQUMsRUFBRSxFQUFFLE1BQU0sQ0FBQyxDQUFDO0lBQ2hELElBQUksQ0FBQyxZQUFZLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsQ0FBQztJQUMvQixJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLENBQUM7SUFDL0IsSUFBSSxDQUFDLFlBQVksQ0FBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDaEMsSUFBSSxDQUFDLFdBQVcsR0FBRyxXQUFXLENBQUM7SUFDL0IsT0FBTyxJQUFJLENBQUM7QUFDaEIsQ0FBQztBQUVEOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sWUFBWSxHQUFHLENBQUMsS0FBVSxFQUFvQixFQUFFO0lBQ3pELE9BQU8sU0FBUyxDQUFDLEtBQUssQ0FBQyxJQUFJLEtBQUssQ0FBQyxRQUFRLEtBQUssQ0FBQyxDQUFDO0FBQ3BELENBQUMsQ0FBQztBQUVGOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sU0FBUyxHQUFHLENBQUMsS0FBVSxFQUFpQixFQUFFO0lBQ25ELE9BQU8sS0FBSyxZQUFZLE1BQU0sQ0FBQyxJQUFJLENBQUM7QUFDeEMsQ0FBQyxDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0sd0JBQXdCLEdBQUcsQ0FBQyxNQUEwQixFQUFFLEVBQUU7SUFDbkUsSUFBSSxZQUFZLENBQUMsTUFBTSxDQUFDLEVBQUU7UUFDdEIsSUFBSSxNQUFNLENBQUMsT0FBTyxLQUFLLE9BQU8sSUFBSSxNQUFNLENBQUMsT0FBTyxLQUFLLFVBQVUsRUFBRTtZQUM3RCxPQUFPLElBQUksQ0FBQztTQUNmO0tBQ0o7SUFDRCxPQUFPLEtBQUssQ0FBQztBQUNqQixDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxrQkFBa0IsR0FBRyxDQUFDLEtBQWlCLEVBQUUsRUFBRTtJQUNwRCxPQUFPLEtBQUssQ0FBQyxNQUFNLEtBQUssY0FBYyxDQUFDLFNBQVMsQ0FBQztBQUNyRCxDQUFDLENBQUE7QUFFRCxNQUFNLENBQUMsTUFBTSxhQUFhLEdBQUcsQ0FBQyxLQUFpQixFQUFFLEVBQUU7SUFDL0MsT0FBTyxLQUFLLENBQUMsTUFBTSxLQUFLLGNBQWMsQ0FBQyxJQUFJLENBQUM7QUFDaEQsQ0FBQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUE9JTlRFUl9CVVRUT04gfSBmcm9tICcuLi8uLi9jb25zdGFudHMnO1xuaW1wb3J0IHsgUG9pbnQgfSBmcm9tICcuLi8uLi9pbnRlcmZhY2VzL3BvaW50JztcblxuZXhwb3J0IGNvbnN0IE5TID0gJ2h0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnJztcblxuZXhwb3J0IGZ1bmN0aW9uIHRvUG9pbnQoeDogbnVtYmVyLCB5OiBudW1iZXIsIGNvbnRhaW5lcjogU1ZHRWxlbWVudCk6IFBvaW50IHtcbiAgICBjb25zdCByZWN0ID0gY29udGFpbmVyLmdldEJvdW5kaW5nQ2xpZW50UmVjdCgpO1xuICAgIHJldHVybiBbeCAtIHJlY3QueCwgeSAtIHJlY3QueV07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVHKCkge1xuICAgIGNvbnN0IG5ld0cgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50TlMoTlMsICdnJyk7XG4gICAgcmV0dXJuIG5ld0c7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVQYXRoKCkge1xuICAgIGNvbnN0IG5ld0cgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50TlMoTlMsICdwYXRoJyk7XG4gICAgcmV0dXJuIG5ld0c7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVTVkcoKSB7XG4gICAgY29uc3Qgc3ZnID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKE5TLCAnc3ZnJyk7XG4gICAgcmV0dXJuIHN2Zztcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZVRleHQoeDogbnVtYmVyLCB5OiBudW1iZXIsIGZpbGw6IHN0cmluZywgdGV4dENvbnRlbnQ6IHN0cmluZykge1xuICAgIHZhciB0ZXh0ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKE5TLCAndGV4dCcpO1xuICAgIHRleHQuc2V0QXR0cmlidXRlKCd4JywgYCR7eH1gKTtcbiAgICB0ZXh0LnNldEF0dHJpYnV0ZSgneScsIGAke3l9YCk7XG4gICAgdGV4dC5zZXRBdHRyaWJ1dGUoJ2ZpbGwnLCBmaWxsKTtcbiAgICB0ZXh0LnRleHRDb250ZW50ID0gdGV4dENvbnRlbnQ7XG4gICAgcmV0dXJuIHRleHQ7XG59XG5cbi8qKlxuICogQ2hlY2sgaWYgYSBET00gbm9kZSBpcyBhbiBlbGVtZW50IG5vZGUuXG4gKi9cbmV4cG9ydCBjb25zdCBpc0RPTUVsZW1lbnQgPSAodmFsdWU6IGFueSk6IHZhbHVlIGlzIEVsZW1lbnQgPT4ge1xuICAgIHJldHVybiBpc0RPTU5vZGUodmFsdWUpICYmIHZhbHVlLm5vZGVUeXBlID09PSAxO1xufTtcblxuLyoqXG4gKiBDaGVjayBpZiBhIHZhbHVlIGlzIGEgRE9NIG5vZGUuXG4gKi9cbmV4cG9ydCBjb25zdCBpc0RPTU5vZGUgPSAodmFsdWU6IGFueSk6IHZhbHVlIGlzIE5vZGUgPT4ge1xuICAgIHJldHVybiB2YWx1ZSBpbnN0YW5jZW9mIHdpbmRvdy5Ob2RlO1xufTtcblxuZXhwb3J0IGNvbnN0IGhhc0lucHV0T3JUZXh0YXJlYVRhcmdldCA9ICh0YXJnZXQ6IEV2ZW50VGFyZ2V0IHwgbnVsbCkgPT4ge1xuICAgIGlmIChpc0RPTUVsZW1lbnQodGFyZ2V0KSkge1xuICAgICAgICBpZiAodGFyZ2V0LnRhZ05hbWUgPT09ICdJTlBVVCcgfHwgdGFyZ2V0LnRhZ05hbWUgPT09ICdURVhUQVJFQScpIHtcbiAgICAgICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgICB9XG4gICAgfVxuICAgIHJldHVybiBmYWxzZTtcbn07XG5cbmV4cG9ydCBjb25zdCBpc1NlY29uZGFyeVBvaW50ZXIgPSAoZXZlbnQ6IE1vdXNlRXZlbnQpID0+IHtcbiAgICByZXR1cm4gZXZlbnQuYnV0dG9uID09PSBQT0lOVEVSX0JVVFRPTi5TRUNPTkRBUlk7XG59XG5cbmV4cG9ydCBjb25zdCBpc01haW5Qb2ludGVyID0gKGV2ZW50OiBNb3VzZUV2ZW50KSA9PiB7XG4gICAgcmV0dXJuIGV2ZW50LmJ1dHRvbiA9PT0gUE9JTlRFUl9CVVRUT04uTUFJTjtcbn1cbiJdfQ==
@@ -1,44 +0,0 @@
1
- import { depthFirstRecursion } from './tree';
2
- export function getRectangleByElements(board, elements, recursion) {
3
- const boundaryBox = {
4
- left: Number.MAX_VALUE,
5
- top: Number.MAX_VALUE,
6
- right: Number.NEGATIVE_INFINITY,
7
- bottom: Number.NEGATIVE_INFINITY
8
- };
9
- const calcRectangleClient = (node) => {
10
- const nodeRectangle = board.getRectangle(node);
11
- if (nodeRectangle) {
12
- boundaryBox.left = Math.min(boundaryBox.left, nodeRectangle.x);
13
- boundaryBox.top = Math.min(boundaryBox.top, nodeRectangle.y);
14
- boundaryBox.right = Math.max(boundaryBox.right, nodeRectangle.x + nodeRectangle.width);
15
- boundaryBox.bottom = Math.max(boundaryBox.bottom, nodeRectangle.y + nodeRectangle.height);
16
- }
17
- };
18
- elements.forEach(element => {
19
- if (recursion) {
20
- depthFirstRecursion(element, node => calcRectangleClient(node), node => board.isRecursion(node));
21
- }
22
- else {
23
- calcRectangleClient(element);
24
- }
25
- });
26
- if (boundaryBox.left === Number.MAX_VALUE) {
27
- return {
28
- x: 0,
29
- y: 0,
30
- width: 0,
31
- height: 0
32
- };
33
- }
34
- return {
35
- x: boundaryBox.left,
36
- y: boundaryBox.top,
37
- width: boundaryBox.right - boundaryBox.left,
38
- height: boundaryBox.bottom - boundaryBox.top
39
- };
40
- }
41
- export function getBoardRectangle(board) {
42
- return getRectangleByElements(board, board.children, true);
43
- }
44
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWxlbWVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3BhY2thZ2VzL2NvcmUvc3JjL3V0aWxzL2VsZW1lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sUUFBUSxDQUFDO0FBRTdDLE1BQU0sVUFBVSxzQkFBc0IsQ0FBQyxLQUFpQixFQUFFLFFBQXdCLEVBQUUsU0FBa0I7SUFDbEcsTUFBTSxXQUFXLEdBQUc7UUFDaEIsSUFBSSxFQUFFLE1BQU0sQ0FBQyxTQUFTO1FBQ3RCLEdBQUcsRUFBRSxNQUFNLENBQUMsU0FBUztRQUNyQixLQUFLLEVBQUUsTUFBTSxDQUFDLGlCQUFpQjtRQUMvQixNQUFNLEVBQUUsTUFBTSxDQUFDLGlCQUFpQjtLQUNuQyxDQUFDO0lBRUYsTUFBTSxtQkFBbUIsR0FBRyxDQUFDLElBQWtCLEVBQUUsRUFBRTtRQUMvQyxNQUFNLGFBQWEsR0FBRyxLQUFLLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQy9DLElBQUksYUFBYSxFQUFFO1lBQ2YsV0FBVyxDQUFDLElBQUksR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLFdBQVcsQ0FBQyxJQUFJLEVBQUUsYUFBYSxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQy9ELFdBQVcsQ0FBQyxHQUFHLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxXQUFXLENBQUMsR0FBRyxFQUFFLGFBQWEsQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUM3RCxXQUFXLENBQUMsS0FBSyxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsV0FBVyxDQUFDLEtBQUssRUFBRSxhQUFhLENBQUMsQ0FBQyxHQUFHLGFBQWEsQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUN2RixXQUFXLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsV0FBVyxDQUFDLE1BQU0sRUFBRSxhQUFhLENBQUMsQ0FBQyxHQUFHLGFBQWEsQ0FBQyxNQUFNLENBQUMsQ0FBQztTQUM3RjtJQUNMLENBQUMsQ0FBQztJQUVGLFFBQVEsQ0FBQyxPQUFPLENBQUMsT0FBTyxDQUFDLEVBQUU7UUFDdkIsSUFBSSxTQUFTLEVBQUU7WUFDWCxtQkFBbUIsQ0FDZixPQUFPLEVBQ1AsSUFBSSxDQUFDLEVBQUUsQ0FBQyxtQkFBbUIsQ0FBQyxJQUFJLENBQUMsRUFDakMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxDQUNsQyxDQUFDO1NBQ0w7YUFBTTtZQUNILG1CQUFtQixDQUFDLE9BQU8sQ0FBQyxDQUFDO1NBQ2hDO0lBQ0wsQ0FBQyxDQUFDLENBQUM7SUFFSCxJQUFJLFdBQVcsQ0FBQyxJQUFJLEtBQUssTUFBTSxDQUFDLFNBQVMsRUFBRTtRQUN2QyxPQUFPO1lBQ0gsQ0FBQyxFQUFFLENBQUM7WUFDSixDQUFDLEVBQUUsQ0FBQztZQUNKLEtBQUssRUFBRSxDQUFDO1lBQ1IsTUFBTSxFQUFFLENBQUM7U0FDWixDQUFDO0tBQ0w7SUFFRCxPQUFPO1FBQ0gsQ0FBQyxFQUFFLFdBQVcsQ0FBQyxJQUFJO1FBQ25CLENBQUMsRUFBRSxXQUFXLENBQUMsR0FBRztRQUNsQixLQUFLLEVBQUUsV0FBVyxDQUFDLEtBQUssR0FBRyxXQUFXLENBQUMsSUFBSTtRQUMzQyxNQUFNLEVBQUUsV0FBVyxDQUFDLE1BQU0sR0FBRyxXQUFXLENBQUMsR0FBRztLQUMvQyxDQUFDO0FBQ04sQ0FBQztBQUVELE1BQU0sVUFBVSxpQkFBaUIsQ0FBQyxLQUFpQjtJQUMvQyxPQUFPLHNCQUFzQixDQUFDLEtBQUssRUFBRSxLQUFLLENBQUMsUUFBUSxFQUFFLElBQUksQ0FBQyxDQUFDO0FBQy9ELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBQbGFpdEJvYXJkLCBQbGFpdEVsZW1lbnQsIFJlY3RhbmdsZUNsaWVudCB9IGZyb20gJy4uL2ludGVyZmFjZXMnO1xuaW1wb3J0IHsgZGVwdGhGaXJzdFJlY3Vyc2lvbiB9IGZyb20gJy4vdHJlZSc7XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRSZWN0YW5nbGVCeUVsZW1lbnRzKGJvYXJkOiBQbGFpdEJvYXJkLCBlbGVtZW50czogUGxhaXRFbGVtZW50W10sIHJlY3Vyc2lvbjogYm9vbGVhbik6IFJlY3RhbmdsZUNsaWVudCB7XG4gICAgY29uc3QgYm91bmRhcnlCb3ggPSB7XG4gICAgICAgIGxlZnQ6IE51bWJlci5NQVhfVkFMVUUsXG4gICAgICAgIHRvcDogTnVtYmVyLk1BWF9WQUxVRSxcbiAgICAgICAgcmlnaHQ6IE51bWJlci5ORUdBVElWRV9JTkZJTklUWSxcbiAgICAgICAgYm90dG9tOiBOdW1iZXIuTkVHQVRJVkVfSU5GSU5JVFlcbiAgICB9O1xuXG4gICAgY29uc3QgY2FsY1JlY3RhbmdsZUNsaWVudCA9IChub2RlOiBQbGFpdEVsZW1lbnQpID0+IHtcbiAgICAgICAgY29uc3Qgbm9kZVJlY3RhbmdsZSA9IGJvYXJkLmdldFJlY3RhbmdsZShub2RlKTtcbiAgICAgICAgaWYgKG5vZGVSZWN0YW5nbGUpIHtcbiAgICAgICAgICAgIGJvdW5kYXJ5Qm94LmxlZnQgPSBNYXRoLm1pbihib3VuZGFyeUJveC5sZWZ0LCBub2RlUmVjdGFuZ2xlLngpO1xuICAgICAgICAgICAgYm91bmRhcnlCb3gudG9wID0gTWF0aC5taW4oYm91bmRhcnlCb3gudG9wLCBub2RlUmVjdGFuZ2xlLnkpO1xuICAgICAgICAgICAgYm91bmRhcnlCb3gucmlnaHQgPSBNYXRoLm1heChib3VuZGFyeUJveC5yaWdodCwgbm9kZVJlY3RhbmdsZS54ICsgbm9kZVJlY3RhbmdsZS53aWR0aCk7XG4gICAgICAgICAgICBib3VuZGFyeUJveC5ib3R0b20gPSBNYXRoLm1heChib3VuZGFyeUJveC5ib3R0b20sIG5vZGVSZWN0YW5nbGUueSArIG5vZGVSZWN0YW5nbGUuaGVpZ2h0KTtcbiAgICAgICAgfVxuICAgIH07XG5cbiAgICBlbGVtZW50cy5mb3JFYWNoKGVsZW1lbnQgPT4ge1xuICAgICAgICBpZiAocmVjdXJzaW9uKSB7XG4gICAgICAgICAgICBkZXB0aEZpcnN0UmVjdXJzaW9uKFxuICAgICAgICAgICAgICAgIGVsZW1lbnQsXG4gICAgICAgICAgICAgICAgbm9kZSA9PiBjYWxjUmVjdGFuZ2xlQ2xpZW50KG5vZGUpLFxuICAgICAgICAgICAgICAgIG5vZGUgPT4gYm9hcmQuaXNSZWN1cnNpb24obm9kZSlcbiAgICAgICAgICAgICk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBjYWxjUmVjdGFuZ2xlQ2xpZW50KGVsZW1lbnQpO1xuICAgICAgICB9XG4gICAgfSk7XG5cbiAgICBpZiAoYm91bmRhcnlCb3gubGVmdCA9PT0gTnVtYmVyLk1BWF9WQUxVRSkge1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgeDogMCxcbiAgICAgICAgICAgIHk6IDAsXG4gICAgICAgICAgICB3aWR0aDogMCxcbiAgICAgICAgICAgIGhlaWdodDogMFxuICAgICAgICB9O1xuICAgIH1cblxuICAgIHJldHVybiB7XG4gICAgICAgIHg6IGJvdW5kYXJ5Qm94LmxlZnQsXG4gICAgICAgIHk6IGJvdW5kYXJ5Qm94LnRvcCxcbiAgICAgICAgd2lkdGg6IGJvdW5kYXJ5Qm94LnJpZ2h0IC0gYm91bmRhcnlCb3gubGVmdCxcbiAgICAgICAgaGVpZ2h0OiBib3VuZGFyeUJveC5ib3R0b20gLSBib3VuZGFyeUJveC50b3BcbiAgICB9O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0Qm9hcmRSZWN0YW5nbGUoYm9hcmQ6IFBsYWl0Qm9hcmQpOiBSZWN0YW5nbGVDbGllbnQge1xuICAgIHJldHVybiBnZXRSZWN0YW5nbGVCeUVsZW1lbnRzKGJvYXJkLCBib2FyZC5jaGlsZHJlbiwgdHJ1ZSk7XG59XG4iXX0=
@@ -1,85 +0,0 @@
1
- import { RectangleClient } from '../interfaces/rectangle-client';
2
- // https://stackoverflow.com/a/6853926/232122
3
- export function distanceBetweenPointAndSegment(x, y, x1, y1, x2, y2) {
4
- const A = x - x1;
5
- const B = y - y1;
6
- const C = x2 - x1;
7
- const D = y2 - y1;
8
- const dot = A * C + B * D;
9
- const lenSquare = C * C + D * D;
10
- let param = -1;
11
- if (lenSquare !== 0) {
12
- // in case of 0 length line
13
- param = dot / lenSquare;
14
- }
15
- let xx, yy;
16
- if (param < 0) {
17
- xx = x1;
18
- yy = y1;
19
- }
20
- else if (param > 1) {
21
- xx = x2;
22
- yy = y2;
23
- }
24
- else {
25
- xx = x1 + param * C;
26
- yy = y1 + param * D;
27
- }
28
- const dx = x - xx;
29
- const dy = y - yy;
30
- return Math.hypot(dx, dy);
31
- }
32
- export function distanceBetweenPointAndSegments(points, point) {
33
- const len = points.length;
34
- let distance = Infinity;
35
- for (let i = 0; i < len - 1; i++) {
36
- const p = points[i];
37
- const p2 = points[i + 1];
38
- const currentDistance = distanceBetweenPointAndSegment(point[0], point[1], p[0], p[1], p2[0], p2[1]);
39
- if (currentDistance < distance) {
40
- distance = currentDistance;
41
- }
42
- }
43
- return distance;
44
- }
45
- export function rotate(x1, y1, x2, y2, angle) {
46
- // 𝑎′𝑥=(𝑎𝑥−𝑐𝑥)cos𝜃−(𝑎𝑦−𝑐𝑦)sin𝜃+𝑐𝑥
47
- // 𝑎′𝑦=(𝑎𝑥−𝑐𝑥)sin𝜃+(𝑎𝑦−𝑐𝑦)cos𝜃+𝑐𝑦.
48
- // https://math.stackexchange.com/questions/2204520/how-do-i-rotate-a-line-segment-in-a-specific-point-on-the-line
49
- return [(x1 - x2) * Math.cos(angle) - (y1 - y2) * Math.sin(angle) + x2, (x1 - x2) * Math.sin(angle) + (y1 - y2) * Math.cos(angle) + y2];
50
- }
51
- export function distanceBetweenPointAndPoint(x1, y1, x2, y2) {
52
- const dx = x1 - x2;
53
- const dy = y1 - y2;
54
- return Math.hypot(dx, dy);
55
- }
56
- // https://stackoverflow.com/questions/5254838/calculating-distance-between-a-point-and-a-rectangular-box-nearest-point
57
- export function distanceBetweenPointAndRectangle(x, y, rect) {
58
- var dx = Math.max(rect.x - x, 0, x - (rect.x + rect.width));
59
- var dy = Math.max(rect.y - y, 0, y - (rect.y + rect.height));
60
- return Math.sqrt(dx * dx + dy * dy);
61
- }
62
- export const isLineHitLine = (a, b, c, d) => {
63
- const crossProduct = (v1, v2) => v1[0] * v2[1] - v1[1] * v2[0];
64
- const ab = [b[0] - a[0], b[1] - a[1]];
65
- const ac = [c[0] - a[0], c[1] - a[1]];
66
- const ad = [d[0] - a[0], d[1] - a[1]];
67
- const ca = [a[0] - c[0], a[1] - c[1]];
68
- const cb = [b[0] - c[0], b[1] - c[1]];
69
- const cd = [d[0] - c[0], d[1] - c[1]];
70
- return crossProduct(ab, ac) * crossProduct(ab, ad) <= 0 && crossProduct(cd, ca) * crossProduct(cd, cb) <= 0;
71
- };
72
- export const isPolylineHitRectangle = (points, rectangle) => {
73
- const rectanglePoints = RectangleClient.getCornerPoints(rectangle);
74
- for (let i = 1; i < points.length; i++) {
75
- const isIntersect = isLineHitLine(points[i], points[i - 1], rectanglePoints[0], rectanglePoints[1]) ||
76
- isLineHitLine(points[i], points[i - 1], rectanglePoints[1], rectanglePoints[2]) ||
77
- isLineHitLine(points[i], points[i - 1], rectanglePoints[2], rectanglePoints[3]) ||
78
- isLineHitLine(points[i], points[i - 1], rectanglePoints[3], rectanglePoints[0]);
79
- if (isIntersect) {
80
- return true;
81
- }
82
- }
83
- return false;
84
- };
85
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"math.js","sourceRoot":"","sources":["../../../../packages/core/src/utils/math.ts"],"names":[],"mappings":"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","sourcesContent":["import { Point } from '../interfaces';\nimport { RectangleClient } from '../interfaces/rectangle-client';\n\n// https://stackoverflow.com/a/6853926/232122\nexport function distanceBetweenPointAndSegment(x: number, y: number, x1: number, y1: number, x2: number, y2: number) {\n    const A = x - x1;\n    const B = y - y1;\n    const C = x2 - x1;\n    const D = y2 - y1;\n\n    const dot = A * C + B * D;\n    const lenSquare = C * C + D * D;\n    let param = -1;\n    if (lenSquare !== 0) {\n        // in case of 0 length line\n        param = dot / lenSquare;\n    }\n\n    let xx, yy;\n    if (param < 0) {\n        xx = x1;\n        yy = y1;\n    } else if (param > 1) {\n        xx = x2;\n        yy = y2;\n    } else {\n        xx = x1 + param * C;\n        yy = y1 + param * D;\n    }\n\n    const dx = x - xx;\n    const dy = y - yy;\n    return Math.hypot(dx, dy);\n}\n\nexport function distanceBetweenPointAndSegments(points: Point[], point: Point) {\n    const len = points.length;\n    let distance = Infinity;\n    for (let i = 0; i < len - 1; i++) {\n        const p = points[i];\n        const p2 = points[i + 1];\n        const currentDistance = distanceBetweenPointAndSegment(point[0], point[1], p[0], p[1], p2[0], p2[1]);\n        if (currentDistance < distance) {\n            distance = currentDistance;\n        }\n    }\n    return distance;\n}\n\nexport function rotate(x1: number, y1: number, x2: number, y2: number, angle: number) {\n    // 𝑎′𝑥=(𝑎𝑥−𝑐𝑥)cos𝜃−(𝑎𝑦−𝑐𝑦)sin𝜃+𝑐𝑥\n    // 𝑎′𝑦=(𝑎𝑥−𝑐𝑥)sin𝜃+(𝑎𝑦−𝑐𝑦)cos𝜃+𝑐𝑦.\n    // https://math.stackexchange.com/questions/2204520/how-do-i-rotate-a-line-segment-in-a-specific-point-on-the-line\n    return [(x1 - x2) * Math.cos(angle) - (y1 - y2) * Math.sin(angle) + x2, (x1 - x2) * Math.sin(angle) + (y1 - y2) * Math.cos(angle) + y2];\n}\n\nexport function distanceBetweenPointAndPoint(x1: number, y1: number, x2: number, y2: number) {\n    const dx = x1 - x2;\n    const dy = y1 - y2;\n    return Math.hypot(dx, dy);\n}\n\n// https://stackoverflow.com/questions/5254838/calculating-distance-between-a-point-and-a-rectangular-box-nearest-point\nexport function distanceBetweenPointAndRectangle(x: number, y: number, rect: RectangleClient) {\n    var dx = Math.max(rect.x - x, 0, x - (rect.x + rect.width));\n    var dy = Math.max(rect.y - y, 0, y - (rect.y + rect.height));\n    return Math.sqrt(dx * dx + dy * dy);\n}\n\nexport const isLineHitLine = (a: Point, b: Point, c: Point, d: Point): boolean => {\n    const crossProduct = (v1: Point, v2: Point) => v1[0] * v2[1] - v1[1] * v2[0];\n\n    const ab: Point = [b[0] - a[0], b[1] - a[1]];\n    const ac: Point = [c[0] - a[0], c[1] - a[1]];\n    const ad: Point = [d[0] - a[0], d[1] - a[1]];\n\n    const ca: Point = [a[0] - c[0], a[1] - c[1]];\n    const cb: Point = [b[0] - c[0], b[1] - c[1]];\n    const cd: Point = [d[0] - c[0], d[1] - c[1]];\n\n    return crossProduct(ab, ac) * crossProduct(ab, ad) <= 0 && crossProduct(cd, ca) * crossProduct(cd, cb) <= 0;\n};\n\nexport const isPolylineHitRectangle = (points: Point[], rectangle: RectangleClient) => {\n    const rectanglePoints = RectangleClient.getCornerPoints(rectangle);\n\n    for (let i = 1; i < points.length; i++) {\n        const isIntersect =\n            isLineHitLine(points[i], points[i - 1], rectanglePoints[0], rectanglePoints[1]) ||\n            isLineHitLine(points[i], points[i - 1], rectanglePoints[1], rectanglePoints[2]) ||\n            isLineHitLine(points[i], points[i - 1], rectanglePoints[2], rectanglePoints[3]) ||\n            isLineHitLine(points[i], points[i - 1], rectanglePoints[3], rectanglePoints[0]);\n        if (isIntersect) {\n            return true;\n        }\n    }\n\n    return false;\n};\n"]}