ngx-vflow 1.16.3 → 2.0.0
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/README.md +35 -69
- package/fesm2022/ngx-vflow-testing.mjs +77 -67
- package/fesm2022/ngx-vflow-testing.mjs.map +1 -1
- package/fesm2022/ngx-vflow.mjs +926 -559
- package/fesm2022/ngx-vflow.mjs.map +1 -1
- package/lib/vflow/components/custom-node-base/custom-node-base.component.d.ts +1 -1
- package/lib/vflow/components/vflow/vflow.component.d.ts +18 -9
- package/lib/vflow/directives/auto-pan.directive.d.ts +16 -0
- package/lib/vflow/directives/changes-controller.directive.d.ts +13 -87
- package/lib/vflow/directives/connection-controller.directive.d.ts +8 -10
- package/lib/vflow/directives/space-point-context.directive.d.ts +2 -1
- package/lib/vflow/interfaces/component-node-event.interface.d.ts +1 -2
- package/lib/vflow/interfaces/connection-events.interface.d.ts +66 -0
- package/lib/vflow/interfaces/connection-settings.interface.d.ts +7 -1
- package/lib/vflow/interfaces/connection.interface.d.ts +0 -5
- package/lib/vflow/interfaces/curve-factory.interface.d.ts +2 -2
- package/lib/vflow/interfaces/edge-label.interface.d.ts +0 -4
- package/lib/vflow/interfaces/edge.interface.d.ts +36 -8
- package/lib/vflow/interfaces/node.interface.d.ts +44 -63
- package/lib/vflow/interfaces/selection-strategy.interface.d.ts +16 -0
- package/lib/vflow/interfaces/template-context.interface.d.ts +8 -3
- package/lib/vflow/models/connection.model.d.ts +1 -1
- package/lib/vflow/models/edge.model.d.ts +20 -23
- package/lib/vflow/models/handle.model.d.ts +1 -1
- package/lib/vflow/models/node.model.d.ts +4 -7
- package/lib/vflow/public-components/custom-node/custom-node.component.d.ts +1 -1
- package/lib/vflow/public-components/custom-template-edge/custom-template-edge.component.d.ts +2 -1
- package/lib/vflow/public-components/minimap/minimap.component.d.ts +1 -1
- package/lib/vflow/services/draggable.service.d.ts +3 -0
- package/lib/vflow/services/flow-entities.service.d.ts +2 -2
- package/lib/vflow/services/flow-settings.service.d.ts +3 -0
- package/lib/vflow/services/flow-status.service.d.ts +35 -7
- package/lib/vflow/services/selection.service.d.ts +4 -1
- package/lib/vflow/strategies/default-selection.strategy.d.ts +6 -0
- package/lib/vflow/strategies/manual-selection.strategy.d.ts +5 -0
- package/lib/vflow/types/selection-mode.type.d.ts +1 -0
- package/lib/vflow/types/unwrap-signal.type.d.ts +4 -0
- package/lib/vflow/utils/adjust-direction.d.ts +2 -6
- package/lib/vflow/utils/identity-checker/reference-identity-checker.d.ts +2 -2
- package/lib/vflow/utils/is-vflow-component.d.ts +0 -2
- package/lib/vflow/utils/signals/to-lazy-signal.d.ts +4 -4
- package/package.json +3 -7
- package/public-api.d.ts +3 -2
- package/testing/component-mocks/vflow-mock.component.d.ts +8 -6
- package/testing/directive-mocks/connection-controller-mock.directive.d.ts +8 -6
- package/esm2022/lib/vflow/components/alignment-helper/alignment-helper.component.mjs +0 -103
- package/esm2022/lib/vflow/components/background/background.component.mjs +0 -121
- package/esm2022/lib/vflow/components/connection/connection.component.mjs +0 -157
- package/esm2022/lib/vflow/components/custom-node-base/custom-node-base.component.mjs +0 -57
- package/esm2022/lib/vflow/components/default-node/default-node.component.mjs +0 -16
- package/esm2022/lib/vflow/components/defs/defs.component.mjs +0 -16
- package/esm2022/lib/vflow/components/edge/edge.component.mjs +0 -54
- package/esm2022/lib/vflow/components/edge-label/edge-label.component.mjs +0 -76
- package/esm2022/lib/vflow/components/node/node.component.mjs +0 -107
- package/esm2022/lib/vflow/components/preview-flow/draw-node.mjs +0 -100
- package/esm2022/lib/vflow/components/preview-flow/preview-flow.component.mjs +0 -62
- package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +0 -437
- package/esm2022/lib/vflow/constants/magic-number-to-fix-glitch-in-chrome.constant.mjs +0 -4
- package/esm2022/lib/vflow/decorators/microtask.decorator.mjs +0 -11
- package/esm2022/lib/vflow/directives/changes-controller.directive.mjs +0 -165
- package/esm2022/lib/vflow/directives/connection-controller.directive.mjs +0 -145
- package/esm2022/lib/vflow/directives/drag-handle.directive.mjs +0 -28
- package/esm2022/lib/vflow/directives/flow-size-controller.directive.mjs +0 -40
- package/esm2022/lib/vflow/directives/handle-size-controller.directive.mjs +0 -41
- package/esm2022/lib/vflow/directives/map-context.directive.mjs +0 -116
- package/esm2022/lib/vflow/directives/node-handles-controller.directive.mjs +0 -33
- package/esm2022/lib/vflow/directives/node-resize-controller.directive.mjs +0 -37
- package/esm2022/lib/vflow/directives/pointer.directive.mjs +0 -84
- package/esm2022/lib/vflow/directives/reference.directive.mjs +0 -17
- package/esm2022/lib/vflow/directives/root-pointer.directive.mjs +0 -58
- package/esm2022/lib/vflow/directives/root-svg-context.directive.mjs +0 -35
- package/esm2022/lib/vflow/directives/selectable.directive.mjs +0 -48
- package/esm2022/lib/vflow/directives/space-point-context.directive.mjs +0 -42
- package/esm2022/lib/vflow/directives/template.directive.mjs +0 -119
- package/esm2022/lib/vflow/interfaces/alignment-helper-settings.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/box.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/component-node-event.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/connection-settings.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/connection.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/connection.internal.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/contextable.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/curve-factory.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/edge-label.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/edge.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/fit-view-options.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/flow-entity.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/intersecting-nodes-options.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/marker.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/node-preview.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/node.interface.mjs +0 -53
- package/esm2022/lib/vflow/interfaces/optimization.interface.mjs +0 -7
- package/esm2022/lib/vflow/interfaces/point.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/rect.mjs +0 -10
- package/esm2022/lib/vflow/interfaces/template-context.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/viewport.interface.mjs +0 -2
- package/esm2022/lib/vflow/math/edge-path/bezier-path.mjs +0 -66
- package/esm2022/lib/vflow/math/edge-path/smooth-step-path.mjs +0 -226
- package/esm2022/lib/vflow/math/edge-path/straigh-path.mjs +0 -12
- package/esm2022/lib/vflow/math/point-on-line-by-ratio.mjs +0 -12
- package/esm2022/lib/vflow/models/connection.model.mjs +0 -31
- package/esm2022/lib/vflow/models/edge-label.model.mjs +0 -8
- package/esm2022/lib/vflow/models/edge.model.mjs +0 -208
- package/esm2022/lib/vflow/models/handle.model.mjs +0 -98
- package/esm2022/lib/vflow/models/minimap.model.mjs +0 -7
- package/esm2022/lib/vflow/models/node.model.mjs +0 -181
- package/esm2022/lib/vflow/models/toolbar.model.mjs +0 -36
- package/esm2022/lib/vflow/public-components/custom-dynamic-node/custom-dynamic-node.component.mjs +0 -25
- package/esm2022/lib/vflow/public-components/custom-node/custom-node.component.mjs +0 -24
- package/esm2022/lib/vflow/public-components/custom-template-edge/custom-template-edge.component.mjs +0 -29
- package/esm2022/lib/vflow/public-components/handle/handle.component.mjs +0 -53
- package/esm2022/lib/vflow/public-components/minimap/minimap.component.mjs +0 -108
- package/esm2022/lib/vflow/public-components/node-toolbar/node-toolbar.component.mjs +0 -73
- package/esm2022/lib/vflow/public-components/resizable/resizable.component.mjs +0 -272
- package/esm2022/lib/vflow/services/component-event-bus.service.mjs +0 -18
- package/esm2022/lib/vflow/services/draggable.service.mjs +0 -124
- package/esm2022/lib/vflow/services/edge-changes.service.mjs +0 -43
- package/esm2022/lib/vflow/services/edge-rendering.service.mjs +0 -40
- package/esm2022/lib/vflow/services/flow-entities.service.mjs +0 -56
- package/esm2022/lib/vflow/services/flow-rendering.service.mjs +0 -41
- package/esm2022/lib/vflow/services/flow-settings.service.mjs +0 -33
- package/esm2022/lib/vflow/services/flow-status.service.mjs +0 -49
- package/esm2022/lib/vflow/services/handle.service.mjs +0 -30
- package/esm2022/lib/vflow/services/keyboard.service.mjs +0 -47
- package/esm2022/lib/vflow/services/node-accessor.service.mjs +0 -16
- package/esm2022/lib/vflow/services/node-changes.service.mjs +0 -41
- package/esm2022/lib/vflow/services/node-rendering.service.mjs +0 -66
- package/esm2022/lib/vflow/services/overlays.service.mjs +0 -35
- package/esm2022/lib/vflow/services/preview-flow-render-strategy.service.mjs +0 -21
- package/esm2022/lib/vflow/services/selection.service.mjs +0 -54
- package/esm2022/lib/vflow/services/viewport.service.mjs +0 -64
- package/esm2022/lib/vflow/types/background.type.mjs +0 -2
- package/esm2022/lib/vflow/types/connection-mode.type.mjs +0 -2
- package/esm2022/lib/vflow/types/edge-change.type.mjs +0 -2
- package/esm2022/lib/vflow/types/handle-type.type.mjs +0 -2
- package/esm2022/lib/vflow/types/keyboard-action.type.mjs +0 -2
- package/esm2022/lib/vflow/types/node-change.type.mjs +0 -2
- package/esm2022/lib/vflow/types/position.type.mjs +0 -2
- package/esm2022/lib/vflow/types/viewport-change-type.type.mjs +0 -2
- package/esm2022/lib/vflow/utils/add-nodes-to-edges.mjs +0 -11
- package/esm2022/lib/vflow/utils/adjust-direction.mjs +0 -30
- package/esm2022/lib/vflow/utils/align-number.mjs +0 -4
- package/esm2022/lib/vflow/utils/assert-injector.mjs +0 -27
- package/esm2022/lib/vflow/utils/event.mjs +0 -4
- package/esm2022/lib/vflow/utils/get-os.mjs +0 -24
- package/esm2022/lib/vflow/utils/get-overlapping-area.mjs +0 -6
- package/esm2022/lib/vflow/utils/get-space-points.mjs +0 -25
- package/esm2022/lib/vflow/utils/hash.mjs +0 -7
- package/esm2022/lib/vflow/utils/id.mjs +0 -5
- package/esm2022/lib/vflow/utils/identity-checker/reference-identity-checker.mjs +0 -28
- package/esm2022/lib/vflow/utils/is-callable.mjs +0 -10
- package/esm2022/lib/vflow/utils/is-defined.mjs +0 -4
- package/esm2022/lib/vflow/utils/is-group-node.mjs +0 -4
- package/esm2022/lib/vflow/utils/is-vflow-component.mjs +0 -9
- package/esm2022/lib/vflow/utils/nodes.mjs +0 -60
- package/esm2022/lib/vflow/utils/resizable.mjs +0 -11
- package/esm2022/lib/vflow/utils/round.mjs +0 -2
- package/esm2022/lib/vflow/utils/signals/extended-computed.mjs +0 -15
- package/esm2022/lib/vflow/utils/signals/to-lazy-signal.mjs +0 -35
- package/esm2022/lib/vflow/utils/to-unified-node.mjs +0 -24
- package/esm2022/lib/vflow/utils/transform-background.mjs +0 -4
- package/esm2022/lib/vflow/utils/viewport.mjs +0 -51
- package/esm2022/lib/vflow/vflow.mjs +0 -29
- package/esm2022/ngx-vflow.mjs +0 -5
- package/esm2022/public-api.mjs +0 -55
- package/esm2022/testing/component-mocks/custom-template-edge-mock.component.mjs +0 -16
- package/esm2022/testing/component-mocks/handle-mock.component.mjs +0 -26
- package/esm2022/testing/component-mocks/minimap-mock.component.mjs +0 -24
- package/esm2022/testing/component-mocks/node-toolbar-mock.component.mjs +0 -23
- package/esm2022/testing/component-mocks/resizable-mock.component.mjs +0 -27
- package/esm2022/testing/component-mocks/vflow-mock.component.mjs +0 -299
- package/esm2022/testing/directive-mocks/connection-controller-mock.directive.mjs +0 -29
- package/esm2022/testing/directive-mocks/drag-handle-mock.directive.mjs +0 -11
- package/esm2022/testing/directive-mocks/selectable-mock.directive.mjs +0 -14
- package/esm2022/testing/directive-mocks/template-mock.directive.mjs +0 -101
- package/esm2022/testing/ngx-vflow-testing.mjs +0 -5
- package/esm2022/testing/provide-custom-node-mocks.mjs +0 -60
- package/esm2022/testing/public-api.mjs +0 -13
- package/esm2022/testing/types.mjs +0 -2
- package/esm2022/testing/vflow-mocks.mjs +0 -28
- package/lib/vflow/public-components/custom-dynamic-node/custom-dynamic-node.component.d.ts +0 -13
- package/lib/vflow/utils/to-unified-node.d.ts +0 -2
|
@@ -1,226 +0,0 @@
|
|
|
1
|
-
const handleDirections = {
|
|
2
|
-
left: { x: -1, y: 0 },
|
|
3
|
-
right: { x: 1, y: 0 },
|
|
4
|
-
top: { x: 0, y: -1 },
|
|
5
|
-
bottom: { x: 0, y: 1 },
|
|
6
|
-
};
|
|
7
|
-
export function getEdgeCenter(source, target) {
|
|
8
|
-
const xOffset = Math.abs(target.x - source.x) / 2;
|
|
9
|
-
const centerX = target.x < source.x ? target.x + xOffset : target.x - xOffset;
|
|
10
|
-
const yOffset = Math.abs(target.y - source.y) / 2;
|
|
11
|
-
const centerY = target.y < source.y ? target.y + yOffset : target.y - yOffset;
|
|
12
|
-
return [centerX, centerY, xOffset, yOffset];
|
|
13
|
-
}
|
|
14
|
-
const getDirection = ({ source, sourcePosition = 'bottom', target, }) => {
|
|
15
|
-
if (sourcePosition === 'left' || sourcePosition === 'right') {
|
|
16
|
-
return source.x < target.x ? { x: 1, y: 0 } : { x: -1, y: 0 };
|
|
17
|
-
}
|
|
18
|
-
return source.y < target.y ? { x: 0, y: 1 } : { x: 0, y: -1 };
|
|
19
|
-
};
|
|
20
|
-
const distance = (a, b) => Math.sqrt(Math.pow(b.x - a.x, 2) + Math.pow(b.y - a.y, 2));
|
|
21
|
-
// ith this function we try to mimic a orthogonal edge routing behaviour
|
|
22
|
-
// It's not as good as a real orthogonal edge routing but it's faster and good enough as a default for step and smooth step edges
|
|
23
|
-
function getPoints({ source, sourcePosition = 'bottom', target, targetPosition = 'top', offset, }) {
|
|
24
|
-
const sourceDir = handleDirections[sourcePosition];
|
|
25
|
-
const targetDir = handleDirections[targetPosition];
|
|
26
|
-
const sourceGapped = { x: source.x + sourceDir.x * offset, y: source.y + sourceDir.y * offset };
|
|
27
|
-
const targetGapped = { x: target.x + targetDir.x * offset, y: target.y + targetDir.y * offset };
|
|
28
|
-
const dir = getDirection({
|
|
29
|
-
source: sourceGapped,
|
|
30
|
-
sourcePosition,
|
|
31
|
-
target: targetGapped,
|
|
32
|
-
});
|
|
33
|
-
const dirAccessor = dir.x !== 0 ? 'x' : 'y';
|
|
34
|
-
const currDir = dir[dirAccessor];
|
|
35
|
-
let points = [];
|
|
36
|
-
let centerX, centerY;
|
|
37
|
-
const sourceGapOffset = { x: 0, y: 0 };
|
|
38
|
-
const targetGapOffset = { x: 0, y: 0 };
|
|
39
|
-
const [defaultCenterX, defaultCenterY] = getEdgeCenter(source, target);
|
|
40
|
-
// opposite handle positions, default case
|
|
41
|
-
if (sourceDir[dirAccessor] * targetDir[dirAccessor] === -1) {
|
|
42
|
-
centerX = defaultCenterX;
|
|
43
|
-
centerY = defaultCenterY;
|
|
44
|
-
// --->
|
|
45
|
-
// |
|
|
46
|
-
// >---
|
|
47
|
-
const verticalSplit = [
|
|
48
|
-
{ x: centerX, y: sourceGapped.y },
|
|
49
|
-
{ x: centerX, y: targetGapped.y },
|
|
50
|
-
];
|
|
51
|
-
// |
|
|
52
|
-
// ---
|
|
53
|
-
// |
|
|
54
|
-
const horizontalSplit = [
|
|
55
|
-
{ x: sourceGapped.x, y: centerY },
|
|
56
|
-
{ x: targetGapped.x, y: centerY },
|
|
57
|
-
];
|
|
58
|
-
if (sourceDir[dirAccessor] === currDir) {
|
|
59
|
-
points = dirAccessor === 'x' ? verticalSplit : horizontalSplit;
|
|
60
|
-
}
|
|
61
|
-
else {
|
|
62
|
-
points = dirAccessor === 'x' ? horizontalSplit : verticalSplit;
|
|
63
|
-
}
|
|
64
|
-
}
|
|
65
|
-
else {
|
|
66
|
-
// sourceTarget means we take x from source and y from target, targetSource is the opposite
|
|
67
|
-
const sourceTarget = [{ x: sourceGapped.x, y: targetGapped.y }];
|
|
68
|
-
const targetSource = [{ x: targetGapped.x, y: sourceGapped.y }];
|
|
69
|
-
// this handles edges with same handle positions
|
|
70
|
-
if (dirAccessor === 'x') {
|
|
71
|
-
points = sourceDir.x === currDir ? targetSource : sourceTarget;
|
|
72
|
-
}
|
|
73
|
-
else {
|
|
74
|
-
points = sourceDir.y === currDir ? sourceTarget : targetSource;
|
|
75
|
-
}
|
|
76
|
-
if (sourcePosition === targetPosition) {
|
|
77
|
-
const diff = Math.abs(source[dirAccessor] - target[dirAccessor]);
|
|
78
|
-
// if an edge goes from right to right for example (sourcePosition === targetPosition) and the distance between source.x and target.x is less than the offset, the added point and the gapped source/target will overlap. This leads to a weird edge path. To avoid this we add a gapOffset to the source/target
|
|
79
|
-
if (diff <= offset) {
|
|
80
|
-
const gapOffset = Math.min(offset - 1, offset - diff);
|
|
81
|
-
if (sourceDir[dirAccessor] === currDir) {
|
|
82
|
-
sourceGapOffset[dirAccessor] = (sourceGapped[dirAccessor] > source[dirAccessor] ? -1 : 1) * gapOffset;
|
|
83
|
-
}
|
|
84
|
-
else {
|
|
85
|
-
targetGapOffset[dirAccessor] = (targetGapped[dirAccessor] > target[dirAccessor] ? -1 : 1) * gapOffset;
|
|
86
|
-
}
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
// these are conditions for handling mixed handle positions like Right -> Bottom for example
|
|
90
|
-
if (sourcePosition !== targetPosition) {
|
|
91
|
-
const dirAccessorOpposite = dirAccessor === 'x' ? 'y' : 'x';
|
|
92
|
-
const isSameDir = sourceDir[dirAccessor] === targetDir[dirAccessorOpposite];
|
|
93
|
-
const sourceGtTargetOppo = sourceGapped[dirAccessorOpposite] > targetGapped[dirAccessorOpposite];
|
|
94
|
-
const sourceLtTargetOppo = sourceGapped[dirAccessorOpposite] < targetGapped[dirAccessorOpposite];
|
|
95
|
-
const flipSourceTarget = (sourceDir[dirAccessor] === 1 && ((!isSameDir && sourceGtTargetOppo) || (isSameDir && sourceLtTargetOppo))) ||
|
|
96
|
-
(sourceDir[dirAccessor] !== 1 && ((!isSameDir && sourceLtTargetOppo) || (isSameDir && sourceGtTargetOppo)));
|
|
97
|
-
if (flipSourceTarget) {
|
|
98
|
-
points = dirAccessor === 'x' ? sourceTarget : targetSource;
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
const sourceGapPoint = { x: sourceGapped.x + sourceGapOffset.x, y: sourceGapped.y + sourceGapOffset.y };
|
|
102
|
-
const targetGapPoint = { x: targetGapped.x + targetGapOffset.x, y: targetGapped.y + targetGapOffset.y };
|
|
103
|
-
const maxXDistance = Math.max(Math.abs(sourceGapPoint.x - points[0].x), Math.abs(targetGapPoint.x - points[0].x));
|
|
104
|
-
const maxYDistance = Math.max(Math.abs(sourceGapPoint.y - points[0].y), Math.abs(targetGapPoint.y - points[0].y));
|
|
105
|
-
// we want to place the label on the longest segment of the edge
|
|
106
|
-
if (maxXDistance >= maxYDistance) {
|
|
107
|
-
centerX = (sourceGapPoint.x + targetGapPoint.x) / 2;
|
|
108
|
-
centerY = points[0].y;
|
|
109
|
-
}
|
|
110
|
-
else {
|
|
111
|
-
centerX = points[0].x;
|
|
112
|
-
centerY = (sourceGapPoint.y + targetGapPoint.y) / 2;
|
|
113
|
-
}
|
|
114
|
-
}
|
|
115
|
-
const pathPoints = [
|
|
116
|
-
source,
|
|
117
|
-
{ x: sourceGapped.x + sourceGapOffset.x, y: sourceGapped.y + sourceGapOffset.y },
|
|
118
|
-
...points,
|
|
119
|
-
{ x: targetGapped.x + targetGapOffset.x, y: targetGapped.y + targetGapOffset.y },
|
|
120
|
-
target,
|
|
121
|
-
];
|
|
122
|
-
return [pathPoints, centerX, centerY];
|
|
123
|
-
}
|
|
124
|
-
function getBend(a, b, c, size) {
|
|
125
|
-
const bendSize = Math.min(distance(a, b) / 2, distance(b, c) / 2, size);
|
|
126
|
-
const { x, y } = b;
|
|
127
|
-
// no bend
|
|
128
|
-
if ((a.x === x && x === c.x) || (a.y === y && y === c.y)) {
|
|
129
|
-
return `L${x} ${y}`;
|
|
130
|
-
}
|
|
131
|
-
// first segment is horizontal
|
|
132
|
-
if (a.y === y) {
|
|
133
|
-
const xDir = a.x < c.x ? -1 : 1;
|
|
134
|
-
const yDir = a.y < c.y ? 1 : -1;
|
|
135
|
-
return `L ${x + bendSize * xDir},${y}Q ${x},${y} ${x},${y + bendSize * yDir}`;
|
|
136
|
-
}
|
|
137
|
-
const xDir = a.x < c.x ? 1 : -1;
|
|
138
|
-
const yDir = a.y < c.y ? -1 : 1;
|
|
139
|
-
return `L ${x},${y + bendSize * yDir}Q ${x},${y} ${x + bendSize * xDir},${y}`;
|
|
140
|
-
}
|
|
141
|
-
export function smoothStepPath({ sourcePoint, targetPoint, sourcePosition, targetPosition }, borderRadius = 5) {
|
|
142
|
-
const [points, labelX, labelY] = getPoints({
|
|
143
|
-
source: sourcePoint,
|
|
144
|
-
sourcePosition,
|
|
145
|
-
target: targetPoint,
|
|
146
|
-
targetPosition,
|
|
147
|
-
offset: 20,
|
|
148
|
-
});
|
|
149
|
-
const path = points.reduce((res, p, i) => {
|
|
150
|
-
let segment = '';
|
|
151
|
-
if (i > 0 && i < points.length - 1) {
|
|
152
|
-
segment = getBend(points[i - 1], p, points[i + 1], borderRadius);
|
|
153
|
-
}
|
|
154
|
-
else {
|
|
155
|
-
segment = `${i === 0 ? 'M' : 'L'}${p.x} ${p.y}`;
|
|
156
|
-
}
|
|
157
|
-
res += segment;
|
|
158
|
-
return res;
|
|
159
|
-
}, '');
|
|
160
|
-
// Performance optimization: Pre-calculate cumulative distances and use binary search
|
|
161
|
-
const n = points.length;
|
|
162
|
-
if (n < 2) {
|
|
163
|
-
return {
|
|
164
|
-
path,
|
|
165
|
-
labelPoints: {
|
|
166
|
-
start: { x: labelX, y: labelY },
|
|
167
|
-
center: { x: labelX, y: labelY },
|
|
168
|
-
end: { x: labelX, y: labelY },
|
|
169
|
-
},
|
|
170
|
-
};
|
|
171
|
-
}
|
|
172
|
-
// Pre-calculate segment lengths and cumulative distances in a single loop
|
|
173
|
-
const segmentLengths = new Array(n - 1);
|
|
174
|
-
const cumulativeDistances = new Array(n);
|
|
175
|
-
cumulativeDistances[0] = 0;
|
|
176
|
-
let totalLength = 0;
|
|
177
|
-
for (let i = 0; i < n - 1; i++) {
|
|
178
|
-
const dx = points[i + 1].x - points[i].x;
|
|
179
|
-
const dy = points[i + 1].y - points[i].y;
|
|
180
|
-
const len = Math.sqrt(dx * dx + dy * dy);
|
|
181
|
-
segmentLengths[i] = len;
|
|
182
|
-
totalLength += len;
|
|
183
|
-
cumulativeDistances[i + 1] = totalLength;
|
|
184
|
-
}
|
|
185
|
-
// Optimized helper function using binary search
|
|
186
|
-
const getPointAtRatio = (ratio) => {
|
|
187
|
-
const targetDistance = totalLength * ratio;
|
|
188
|
-
// Edge cases
|
|
189
|
-
if (targetDistance <= 0)
|
|
190
|
-
return points[0];
|
|
191
|
-
if (targetDistance >= totalLength)
|
|
192
|
-
return points[n - 1];
|
|
193
|
-
// Binary search for the correct segment
|
|
194
|
-
let low = 0;
|
|
195
|
-
let high = n - 1;
|
|
196
|
-
while (low < high - 1) {
|
|
197
|
-
const mid = (low + high) >>> 1; // Bitwise right shift is faster than Math.floor
|
|
198
|
-
if (cumulativeDistances[mid] < targetDistance) {
|
|
199
|
-
low = mid;
|
|
200
|
-
}
|
|
201
|
-
else {
|
|
202
|
-
high = mid;
|
|
203
|
-
}
|
|
204
|
-
}
|
|
205
|
-
// Calculate position within the segment
|
|
206
|
-
const segmentStartDistance = cumulativeDistances[low];
|
|
207
|
-
const localDistance = targetDistance - segmentStartDistance;
|
|
208
|
-
const t = localDistance / segmentLengths[low];
|
|
209
|
-
// Linear interpolation
|
|
210
|
-
const start = points[low];
|
|
211
|
-
const end = points[low + 1];
|
|
212
|
-
return {
|
|
213
|
-
x: start.x + (end.x - start.x) * t,
|
|
214
|
-
y: start.y + (end.y - start.y) * t,
|
|
215
|
-
};
|
|
216
|
-
};
|
|
217
|
-
return {
|
|
218
|
-
path,
|
|
219
|
-
labelPoints: {
|
|
220
|
-
start: getPointAtRatio(0.15),
|
|
221
|
-
center: { x: labelX, y: labelY },
|
|
222
|
-
end: getPointAtRatio(0.85),
|
|
223
|
-
},
|
|
224
|
-
};
|
|
225
|
-
}
|
|
226
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"smooth-step-path.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/math/edge-path/smooth-step-path.ts"],"names":[],"mappings":"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","sourcesContent":["import { CurveFactoryParams, CurveLayout } from '../../interfaces/curve-factory.interface';\nimport { Point } from '../../interfaces/point.interface';\nimport { Position } from '../../types/position.type';\n\nconst handleDirections = {\n  left: { x: -1, y: 0 },\n  right: { x: 1, y: 0 },\n  top: { x: 0, y: -1 },\n  bottom: { x: 0, y: 1 },\n};\n\nexport function getEdgeCenter(source: Point, target: Point): [number, number, number, number] {\n  const xOffset = Math.abs(target.x - source.x) / 2;\n  const centerX = target.x < source.x ? target.x + xOffset : target.x - xOffset;\n\n  const yOffset = Math.abs(target.y - source.y) / 2;\n  const centerY = target.y < source.y ? target.y + yOffset : target.y - yOffset;\n\n  return [centerX, centerY, xOffset, yOffset];\n}\n\nconst getDirection = ({\n  source,\n  sourcePosition = 'bottom',\n  target,\n}: {\n  source: Point;\n  sourcePosition: Position;\n  target: Point;\n}): Point => {\n  if (sourcePosition === 'left' || sourcePosition === 'right') {\n    return source.x < target.x ? { x: 1, y: 0 } : { x: -1, y: 0 };\n  }\n  return source.y < target.y ? { x: 0, y: 1 } : { x: 0, y: -1 };\n};\n\nconst distance = (a: Point, b: Point) => Math.sqrt(Math.pow(b.x - a.x, 2) + Math.pow(b.y - a.y, 2));\n\n// ith this function we try to mimic a orthogonal edge routing behaviour\n// It's not as good as a real orthogonal edge routing but it's faster and good enough as a default for step and smooth step edges\nfunction getPoints({\n  source,\n  sourcePosition = 'bottom',\n  target,\n  targetPosition = 'top',\n  offset,\n}: {\n  source: Point;\n  sourcePosition: Position;\n  target: Point;\n  targetPosition: Position;\n  offset: number;\n}): [Point[], number, number] {\n  const sourceDir = handleDirections[sourcePosition];\n  const targetDir = handleDirections[targetPosition];\n  const sourceGapped: Point = { x: source.x + sourceDir.x * offset, y: source.y + sourceDir.y * offset };\n  const targetGapped: Point = { x: target.x + targetDir.x * offset, y: target.y + targetDir.y * offset };\n  const dir = getDirection({\n    source: sourceGapped,\n    sourcePosition,\n    target: targetGapped,\n  });\n  const dirAccessor = dir.x !== 0 ? 'x' : 'y';\n  const currDir = dir[dirAccessor];\n\n  let points: Point[] = [];\n  let centerX, centerY;\n  const sourceGapOffset = { x: 0, y: 0 };\n  const targetGapOffset = { x: 0, y: 0 };\n\n  const [defaultCenterX, defaultCenterY] = getEdgeCenter(source, target);\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      { x: centerX, y: sourceGapped.y },\n      { x: centerX, y: targetGapped.y },\n    ];\n    //    |\n    //  ---\n    //  |\n    const horizontalSplit: Point[] = [\n      { x: sourceGapped.x, y: centerY },\n      { x: targetGapped.x, y: centerY },\n    ];\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[] = [{ x: sourceGapped.x, y: targetGapped.y }];\n    const targetSource: Point[] = [{ x: targetGapped.x, y: sourceGapped.y }];\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    if (sourcePosition === targetPosition) {\n      const diff = Math.abs(source[dirAccessor] - target[dirAccessor]);\n\n      // if an edge goes from right to right for example (sourcePosition === targetPosition) and the distance between source.x and target.x is less than the offset, the added point and the gapped source/target will overlap. This leads to a weird edge path. To avoid this we add a gapOffset to the source/target\n      if (diff <= offset) {\n        const gapOffset = Math.min(offset - 1, offset - diff);\n        if (sourceDir[dirAccessor] === currDir) {\n          sourceGapOffset[dirAccessor] = (sourceGapped[dirAccessor] > source[dirAccessor] ? -1 : 1) * gapOffset;\n        } else {\n          targetGapOffset[dirAccessor] = (targetGapped[dirAccessor] > target[dirAccessor] ? -1 : 1) * gapOffset;\n        }\n      }\n    }\n\n    // these are conditions for handling mixed handle positions like Right -> Bottom for example\n    if (sourcePosition !== targetPosition) {\n      const dirAccessorOpposite = dirAccessor === 'x' ? 'y' : 'x';\n      const isSameDir = sourceDir[dirAccessor] === targetDir[dirAccessorOpposite];\n      const sourceGtTargetOppo = sourceGapped[dirAccessorOpposite] > targetGapped[dirAccessorOpposite];\n      const sourceLtTargetOppo = sourceGapped[dirAccessorOpposite] < targetGapped[dirAccessorOpposite];\n      const 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    const sourceGapPoint = { x: sourceGapped.x + sourceGapOffset.x, y: sourceGapped.y + sourceGapOffset.y };\n    const targetGapPoint = { x: targetGapped.x + targetGapOffset.x, y: targetGapped.y + targetGapOffset.y };\n    const maxXDistance = Math.max(Math.abs(sourceGapPoint.x - points[0].x), Math.abs(targetGapPoint.x - points[0].x));\n    const maxYDistance = Math.max(Math.abs(sourceGapPoint.y - points[0].y), Math.abs(targetGapPoint.y - points[0].y));\n\n    // we want to place the label on the longest segment of the edge\n    if (maxXDistance >= maxYDistance) {\n      centerX = (sourceGapPoint.x + targetGapPoint.x) / 2;\n      centerY = points[0].y;\n    } else {\n      centerX = points[0].x;\n      centerY = (sourceGapPoint.y + targetGapPoint.y) / 2;\n    }\n  }\n\n  const pathPoints = [\n    source,\n    { x: sourceGapped.x + sourceGapOffset.x, y: sourceGapped.y + sourceGapOffset.y },\n    ...points,\n    { x: targetGapped.x + targetGapOffset.x, y: targetGapped.y + targetGapOffset.y },\n    target,\n  ];\n\n  return [pathPoints, centerX, centerY];\n}\n\nfunction getBend(a: Point, b: Point, c: Point, size: number): string {\n  const bendSize = Math.min(distance(a, b) / 2, distance(b, c) / 2, size);\n  const { x, y } = b;\n\n  // no bend\n  if ((a.x === x && x === c.x) || (a.y === y && y === c.y)) {\n    return `L${x} ${y}`;\n  }\n\n  // first segment is horizontal\n  if (a.y === y) {\n    const xDir = a.x < c.x ? -1 : 1;\n    const yDir = a.y < c.y ? 1 : -1;\n    return `L ${x + bendSize * xDir},${y}Q ${x},${y} ${x},${y + bendSize * yDir}`;\n  }\n\n  const xDir = a.x < c.x ? 1 : -1;\n  const yDir = a.y < c.y ? -1 : 1;\n  return `L ${x},${y + bendSize * yDir}Q ${x},${y} ${x + bendSize * xDir},${y}`;\n}\n\nexport function smoothStepPath(\n  { sourcePoint, targetPoint, sourcePosition, targetPosition }: CurveFactoryParams,\n  borderRadius: number = 5,\n): CurveLayout {\n  const [points, labelX, labelY] = getPoints({\n    source: sourcePoint,\n    sourcePosition,\n    target: targetPoint,\n    targetPosition,\n    offset: 20,\n  });\n\n  const path = points.reduce<string>((res, p, i) => {\n    let segment = '';\n\n    if (i > 0 && i < points.length - 1) {\n      segment = getBend(points[i - 1], p, points[i + 1], borderRadius);\n    } else {\n      segment = `${i === 0 ? 'M' : 'L'}${p.x} ${p.y}`;\n    }\n\n    res += segment;\n\n    return res;\n  }, '');\n\n  // Performance optimization: Pre-calculate cumulative distances and use binary search\n  const n = points.length;\n  if (n < 2) {\n    return {\n      path,\n      labelPoints: {\n        start: { x: labelX, y: labelY },\n        center: { x: labelX, y: labelY },\n        end: { x: labelX, y: labelY },\n      },\n    };\n  }\n\n  // Pre-calculate segment lengths and cumulative distances in a single loop\n  const segmentLengths: number[] = new Array(n - 1);\n  const cumulativeDistances: number[] = new Array(n);\n  cumulativeDistances[0] = 0;\n\n  let totalLength = 0;\n\n  for (let i = 0; i < n - 1; i++) {\n    const dx = points[i + 1].x - points[i].x;\n    const dy = points[i + 1].y - points[i].y;\n    const len = Math.sqrt(dx * dx + dy * dy);\n    segmentLengths[i] = len;\n    totalLength += len;\n    cumulativeDistances[i + 1] = totalLength;\n  }\n\n  // Optimized helper function using binary search\n  const getPointAtRatio = (ratio: number): Point => {\n    const targetDistance = totalLength * ratio;\n\n    // Edge cases\n    if (targetDistance <= 0) return points[0];\n    if (targetDistance >= totalLength) return points[n - 1];\n\n    // Binary search for the correct segment\n    let low = 0;\n    let high = n - 1;\n\n    while (low < high - 1) {\n      const mid = (low + high) >>> 1; // Bitwise right shift is faster than Math.floor\n      if (cumulativeDistances[mid] < targetDistance) {\n        low = mid;\n      } else {\n        high = mid;\n      }\n    }\n\n    // Calculate position within the segment\n    const segmentStartDistance = cumulativeDistances[low];\n    const localDistance = targetDistance - segmentStartDistance;\n    const t = localDistance / segmentLengths[low];\n\n    // Linear interpolation\n    const start = points[low];\n    const end = points[low + 1];\n\n    return {\n      x: start.x + (end.x - start.x) * t,\n      y: start.y + (end.y - start.y) * t,\n    };\n  };\n\n  return {\n    path,\n    labelPoints: {\n      start: getPointAtRatio(0.15),\n      center: { x: labelX, y: labelY },\n      end: getPointAtRatio(0.85),\n    },\n  };\n}\n"]}
|
|
@@ -1,12 +0,0 @@
|
|
|
1
|
-
import { getPointOnLineByRatio } from '../point-on-line-by-ratio';
|
|
2
|
-
export function straightPath({ sourcePoint, targetPoint }) {
|
|
3
|
-
return {
|
|
4
|
-
path: `M ${sourcePoint.x},${sourcePoint.y}L ${targetPoint.x},${targetPoint.y}`,
|
|
5
|
-
labelPoints: {
|
|
6
|
-
start: getPointOnLineByRatio(sourcePoint, targetPoint, 0.15),
|
|
7
|
-
center: getPointOnLineByRatio(sourcePoint, targetPoint, 0.5),
|
|
8
|
-
end: getPointOnLineByRatio(sourcePoint, targetPoint, 0.85),
|
|
9
|
-
},
|
|
10
|
-
};
|
|
11
|
-
}
|
|
12
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RyYWlnaC1wYXRoLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LXZmbG93LWxpYi9zcmMvbGliL3ZmbG93L21hdGgvZWRnZS1wYXRoL3N0cmFpZ2gtcGF0aC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUVsRSxNQUFNLFVBQVUsWUFBWSxDQUFDLEVBQUUsV0FBVyxFQUFFLFdBQVcsRUFBc0I7SUFDM0UsT0FBTztRQUNMLElBQUksRUFBRSxLQUFLLFdBQVcsQ0FBQyxDQUFDLElBQUksV0FBVyxDQUFDLENBQUMsS0FBSyxXQUFXLENBQUMsQ0FBQyxJQUFJLFdBQVcsQ0FBQyxDQUFDLEVBQUU7UUFDOUUsV0FBVyxFQUFFO1lBQ1gsS0FBSyxFQUFFLHFCQUFxQixDQUFDLFdBQVcsRUFBRSxXQUFXLEVBQUUsSUFBSSxDQUFDO1lBQzVELE1BQU0sRUFBRSxxQkFBcUIsQ0FBQyxXQUFXLEVBQUUsV0FBVyxFQUFFLEdBQUcsQ0FBQztZQUM1RCxHQUFHLEVBQUUscUJBQXFCLENBQUMsV0FBVyxFQUFFLFdBQVcsRUFBRSxJQUFJLENBQUM7U0FDM0Q7S0FDRixDQUFDO0FBQ0osQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEN1cnZlRmFjdG9yeVBhcmFtcywgQ3VydmVMYXlvdXQgfSBmcm9tICcuLi8uLi9pbnRlcmZhY2VzL2N1cnZlLWZhY3RvcnkuaW50ZXJmYWNlJztcbmltcG9ydCB7IGdldFBvaW50T25MaW5lQnlSYXRpbyB9IGZyb20gJy4uL3BvaW50LW9uLWxpbmUtYnktcmF0aW8nO1xuXG5leHBvcnQgZnVuY3Rpb24gc3RyYWlnaHRQYXRoKHsgc291cmNlUG9pbnQsIHRhcmdldFBvaW50IH06IEN1cnZlRmFjdG9yeVBhcmFtcyk6IEN1cnZlTGF5b3V0IHtcbiAgcmV0dXJuIHtcbiAgICBwYXRoOiBgTSAke3NvdXJjZVBvaW50Lnh9LCR7c291cmNlUG9pbnQueX1MICR7dGFyZ2V0UG9pbnQueH0sJHt0YXJnZXRQb2ludC55fWAsXG4gICAgbGFiZWxQb2ludHM6IHtcbiAgICAgIHN0YXJ0OiBnZXRQb2ludE9uTGluZUJ5UmF0aW8oc291cmNlUG9pbnQsIHRhcmdldFBvaW50LCAwLjE1KSxcbiAgICAgIGNlbnRlcjogZ2V0UG9pbnRPbkxpbmVCeVJhdGlvKHNvdXJjZVBvaW50LCB0YXJnZXRQb2ludCwgMC41KSxcbiAgICAgIGVuZDogZ2V0UG9pbnRPbkxpbmVCeVJhdGlvKHNvdXJjZVBvaW50LCB0YXJnZXRQb2ludCwgMC44NSksXG4gICAgfSxcbiAgfTtcbn1cbiJdfQ==
|
|
@@ -1,12 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Get point on line
|
|
3
|
-
*
|
|
4
|
-
* https://math.stackexchange.com/questions/563566/how-do-i-find-the-middle1-2-1-3-1-4-etc-of-a-line
|
|
5
|
-
*/
|
|
6
|
-
export function getPointOnLineByRatio(start, end, ratio) {
|
|
7
|
-
return {
|
|
8
|
-
x: (1 - ratio) * start.x + ratio * end.x,
|
|
9
|
-
y: (1 - ratio) * start.y + ratio * end.y,
|
|
10
|
-
};
|
|
11
|
-
}
|
|
12
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG9pbnQtb24tbGluZS1ieS1yYXRpby5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9tYXRoL3BvaW50LW9uLWxpbmUtYnktcmF0aW8udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUE7Ozs7R0FJRztBQUNILE1BQU0sVUFBVSxxQkFBcUIsQ0FBQyxLQUFZLEVBQUUsR0FBVSxFQUFFLEtBQWE7SUFDM0UsT0FBTztRQUNMLENBQUMsRUFBRSxDQUFDLENBQUMsR0FBRyxLQUFLLENBQUMsR0FBRyxLQUFLLENBQUMsQ0FBQyxHQUFHLEtBQUssR0FBRyxHQUFHLENBQUMsQ0FBQztRQUN4QyxDQUFDLEVBQUUsQ0FBQyxDQUFDLEdBQUcsS0FBSyxDQUFDLEdBQUcsS0FBSyxDQUFDLENBQUMsR0FBRyxLQUFLLEdBQUcsR0FBRyxDQUFDLENBQUM7S0FDekMsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBQb2ludCB9IGZyb20gJy4uL2ludGVyZmFjZXMvcG9pbnQuaW50ZXJmYWNlJztcblxuLyoqXG4gKiBHZXQgcG9pbnQgb24gbGluZVxuICpcbiAqIGh0dHBzOi8vbWF0aC5zdGFja2V4Y2hhbmdlLmNvbS9xdWVzdGlvbnMvNTYzNTY2L2hvdy1kby1pLWZpbmQtdGhlLW1pZGRsZTEtMi0xLTMtMS00LWV0Yy1vZi1hLWxpbmVcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldFBvaW50T25MaW5lQnlSYXRpbyhzdGFydDogUG9pbnQsIGVuZDogUG9pbnQsIHJhdGlvOiBudW1iZXIpOiBQb2ludCB7XG4gIHJldHVybiB7XG4gICAgeDogKDEgLSByYXRpbykgKiBzdGFydC54ICsgcmF0aW8gKiBlbmQueCxcbiAgICB5OiAoMSAtIHJhdGlvKSAqIHN0YXJ0LnkgKyByYXRpbyAqIGVuZC55LFxuICB9O1xufVxuIl19
|
|
@@ -1,31 +0,0 @@
|
|
|
1
|
-
export class ConnectionModel {
|
|
2
|
-
constructor(settings) {
|
|
3
|
-
this.settings = settings;
|
|
4
|
-
this.curve = settings.curve ?? 'bezier';
|
|
5
|
-
this.type = settings.type ?? 'default';
|
|
6
|
-
this.mode = settings.mode ?? 'strict';
|
|
7
|
-
const validatorsToRun = this.getValidators(settings);
|
|
8
|
-
this.validator = (connection) => validatorsToRun.every((v) => v(connection));
|
|
9
|
-
}
|
|
10
|
-
getValidators(settings) {
|
|
11
|
-
const validators = [];
|
|
12
|
-
validators.push(notSelfValidator);
|
|
13
|
-
if (this.mode === 'loose') {
|
|
14
|
-
validators.push(hasSourceAndTargetHandleValidator);
|
|
15
|
-
}
|
|
16
|
-
if (settings.validator) {
|
|
17
|
-
validators.push(settings.validator);
|
|
18
|
-
}
|
|
19
|
-
return validators;
|
|
20
|
-
}
|
|
21
|
-
}
|
|
22
|
-
/**
|
|
23
|
-
* Internal validator that not allows self connections
|
|
24
|
-
*/
|
|
25
|
-
const notSelfValidator = (connection) => {
|
|
26
|
-
return connection.source !== connection.target;
|
|
27
|
-
};
|
|
28
|
-
const hasSourceAndTargetHandleValidator = (connection) => {
|
|
29
|
-
return connection.sourceHandle !== undefined && connection.targetHandle !== undefined;
|
|
30
|
-
};
|
|
31
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29ubmVjdGlvbi5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9tb2RlbHMvY29ubmVjdGlvbi5tb2RlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLQSxNQUFNLE9BQU8sZUFBZTtJQU0xQixZQUFtQixRQUE0QjtRQUE1QixhQUFRLEdBQVIsUUFBUSxDQUFvQjtRQUM3QyxJQUFJLENBQUMsS0FBSyxHQUFHLFFBQVEsQ0FBQyxLQUFLLElBQUksUUFBUSxDQUFDO1FBQ3hDLElBQUksQ0FBQyxJQUFJLEdBQUcsUUFBUSxDQUFDLElBQUksSUFBSSxTQUFTLENBQUM7UUFDdkMsSUFBSSxDQUFDLElBQUksR0FBRyxRQUFRLENBQUMsSUFBSSxJQUFJLFFBQVEsQ0FBQztRQUV0QyxNQUFNLGVBQWUsR0FBRyxJQUFJLENBQUMsYUFBYSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQ3JELElBQUksQ0FBQyxTQUFTLEdBQUcsQ0FBQyxVQUFVLEVBQUUsRUFBRSxDQUFDLGVBQWUsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDO0lBQy9FLENBQUM7SUFFTyxhQUFhLENBQUMsUUFBNEI7UUFDaEQsTUFBTSxVQUFVLEdBQUcsRUFBRSxDQUFDO1FBRXRCLFVBQVUsQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztRQUVsQyxJQUFJLElBQUksQ0FBQyxJQUFJLEtBQUssT0FBTyxFQUFFLENBQUM7WUFDMUIsVUFBVSxDQUFDLElBQUksQ0FBQyxpQ0FBaUMsQ0FBQyxDQUFDO1FBQ3JELENBQUM7UUFFRCxJQUFJLFFBQVEsQ0FBQyxTQUFTLEVBQUUsQ0FBQztZQUN2QixVQUFVLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUN0QyxDQUFDO1FBRUQsT0FBTyxVQUFVLENBQUM7SUFDcEIsQ0FBQztDQUNGO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLGdCQUFnQixHQUFHLENBQUMsVUFBc0IsRUFBRSxFQUFFO0lBQ2xELE9BQU8sVUFBVSxDQUFDLE1BQU0sS0FBSyxVQUFVLENBQUMsTUFBTSxDQUFDO0FBQ2pELENBQUMsQ0FBQztBQUVGLE1BQU0saUNBQWlDLEdBQUcsQ0FBQyxVQUFzQixFQUFFLEVBQUU7SUFDbkUsT0FBTyxVQUFVLENBQUMsWUFBWSxLQUFLLFNBQVMsSUFBSSxVQUFVLENBQUMsWUFBWSxLQUFLLFNBQVMsQ0FBQztBQUN4RixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb25uZWN0aW9uU2V0dGluZ3MsIENvbm5lY3Rpb25WYWxpZGF0b3JGbiB9IGZyb20gJy4uL2ludGVyZmFjZXMvY29ubmVjdGlvbi1zZXR0aW5ncy5pbnRlcmZhY2UnO1xuaW1wb3J0IHsgQ29ubmVjdGlvbiB9IGZyb20gJy4uL2ludGVyZmFjZXMvY29ubmVjdGlvbi5pbnRlcmZhY2UnO1xuaW1wb3J0IHsgQ3VydmUsIEVkZ2VUeXBlIH0gZnJvbSAnLi4vaW50ZXJmYWNlcy9lZGdlLmludGVyZmFjZSc7XG5pbXBvcnQgeyBDb25uZWN0aW9uTW9kZSB9IGZyb20gJy4uL3R5cGVzL2Nvbm5lY3Rpb24tbW9kZS50eXBlJztcblxuZXhwb3J0IGNsYXNzIENvbm5lY3Rpb25Nb2RlbCB7XG4gIHB1YmxpYyBjdXJ2ZTogQ3VydmU7XG4gIHB1YmxpYyB0eXBlOiBFZGdlVHlwZTtcbiAgcHVibGljIHZhbGlkYXRvcjogQ29ubmVjdGlvblZhbGlkYXRvckZuO1xuICBwdWJsaWMgbW9kZTogQ29ubmVjdGlvbk1vZGU7XG5cbiAgY29uc3RydWN0b3IocHVibGljIHNldHRpbmdzOiBDb25uZWN0aW9uU2V0dGluZ3MpIHtcbiAgICB0aGlzLmN1cnZlID0gc2V0dGluZ3MuY3VydmUgPz8gJ2Jlemllcic7XG4gICAgdGhpcy50eXBlID0gc2V0dGluZ3MudHlwZSA/PyAnZGVmYXVsdCc7XG4gICAgdGhpcy5tb2RlID0gc2V0dGluZ3MubW9kZSA/PyAnc3RyaWN0JztcblxuICAgIGNvbnN0IHZhbGlkYXRvcnNUb1J1biA9IHRoaXMuZ2V0VmFsaWRhdG9ycyhzZXR0aW5ncyk7XG4gICAgdGhpcy52YWxpZGF0b3IgPSAoY29ubmVjdGlvbikgPT4gdmFsaWRhdG9yc1RvUnVuLmV2ZXJ5KCh2KSA9PiB2KGNvbm5lY3Rpb24pKTtcbiAgfVxuXG4gIHByaXZhdGUgZ2V0VmFsaWRhdG9ycyhzZXR0aW5nczogQ29ubmVjdGlvblNldHRpbmdzKSB7XG4gICAgY29uc3QgdmFsaWRhdG9ycyA9IFtdO1xuXG4gICAgdmFsaWRhdG9ycy5wdXNoKG5vdFNlbGZWYWxpZGF0b3IpO1xuXG4gICAgaWYgKHRoaXMubW9kZSA9PT0gJ2xvb3NlJykge1xuICAgICAgdmFsaWRhdG9ycy5wdXNoKGhhc1NvdXJjZUFuZFRhcmdldEhhbmRsZVZhbGlkYXRvcik7XG4gICAgfVxuXG4gICAgaWYgKHNldHRpbmdzLnZhbGlkYXRvcikge1xuICAgICAgdmFsaWRhdG9ycy5wdXNoKHNldHRpbmdzLnZhbGlkYXRvcik7XG4gICAgfVxuXG4gICAgcmV0dXJuIHZhbGlkYXRvcnM7XG4gIH1cbn1cblxuLyoqXG4gKiBJbnRlcm5hbCB2YWxpZGF0b3IgdGhhdCBub3QgYWxsb3dzIHNlbGYgY29ubmVjdGlvbnNcbiAqL1xuY29uc3Qgbm90U2VsZlZhbGlkYXRvciA9IChjb25uZWN0aW9uOiBDb25uZWN0aW9uKSA9PiB7XG4gIHJldHVybiBjb25uZWN0aW9uLnNvdXJjZSAhPT0gY29ubmVjdGlvbi50YXJnZXQ7XG59O1xuXG5jb25zdCBoYXNTb3VyY2VBbmRUYXJnZXRIYW5kbGVWYWxpZGF0b3IgPSAoY29ubmVjdGlvbjogQ29ubmVjdGlvbikgPT4ge1xuICByZXR1cm4gY29ubmVjdGlvbi5zb3VyY2VIYW5kbGUgIT09IHVuZGVmaW5lZCAmJiBjb25uZWN0aW9uLnRhcmdldEhhbmRsZSAhPT0gdW5kZWZpbmVkO1xufTtcbiJdfQ==
|
|
@@ -1,8 +0,0 @@
|
|
|
1
|
-
import { signal } from '@angular/core';
|
|
2
|
-
export class EdgeLabelModel {
|
|
3
|
-
constructor(edgeLabel) {
|
|
4
|
-
this.edgeLabel = edgeLabel;
|
|
5
|
-
this.size = signal({ width: 0, height: 0 });
|
|
6
|
-
}
|
|
7
|
-
}
|
|
8
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWRnZS1sYWJlbC5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9tb2RlbHMvZWRnZS1sYWJlbC5tb2RlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBR3ZDLE1BQU0sT0FBTyxjQUFjO0lBR3pCLFlBQW1CLFNBQW9CO1FBQXBCLGNBQVMsR0FBVCxTQUFTLENBQVc7UUFGaEMsU0FBSSxHQUFHLE1BQU0sQ0FBQyxFQUFFLEtBQUssRUFBRSxDQUFDLEVBQUUsTUFBTSxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUM7SUFFSixDQUFDO0NBQzVDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgc2lnbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBFZGdlTGFiZWwgfSBmcm9tICcuLi9pbnRlcmZhY2VzL2VkZ2UtbGFiZWwuaW50ZXJmYWNlJztcblxuZXhwb3J0IGNsYXNzIEVkZ2VMYWJlbE1vZGVsIHtcbiAgcHVibGljIHNpemUgPSBzaWduYWwoeyB3aWR0aDogMCwgaGVpZ2h0OiAwIH0pO1xuXG4gIGNvbnN0cnVjdG9yKHB1YmxpYyBlZGdlTGFiZWw6IEVkZ2VMYWJlbCkge31cbn1cbiJdfQ==
|
|
@@ -1,208 +0,0 @@
|
|
|
1
|
-
import { computed, inject, signal } from '@angular/core';
|
|
2
|
-
import { EdgeLabelModel } from './edge-label.model';
|
|
3
|
-
import { straightPath } from '../math/edge-path/straigh-path';
|
|
4
|
-
import { bezierPath } from '../math/edge-path/bezier-path';
|
|
5
|
-
import { toObservable } from '@angular/core/rxjs-interop';
|
|
6
|
-
import { smoothStepPath } from '../math/edge-path/smooth-step-path';
|
|
7
|
-
import { hashCode } from '../utils/hash';
|
|
8
|
-
import { FlowEntitiesService } from '../services/flow-entities.service';
|
|
9
|
-
import { extendedComputed } from '../utils/signals/extended-computed';
|
|
10
|
-
export class EdgeModel {
|
|
11
|
-
constructor(edge) {
|
|
12
|
-
this.edge = edge;
|
|
13
|
-
this.flowEntitiesService = inject(FlowEntitiesService);
|
|
14
|
-
this.source = signal(undefined);
|
|
15
|
-
this.target = signal(undefined);
|
|
16
|
-
this.selected = signal(false);
|
|
17
|
-
this.selected$ = toObservable(this.selected);
|
|
18
|
-
this.shouldLoad = computed(() => (this.source()?.shouldLoad() ?? false) && (this.target()?.shouldLoad() ?? false));
|
|
19
|
-
this.renderOrder = signal(0);
|
|
20
|
-
this.detached = computed(() => {
|
|
21
|
-
const source = this.source();
|
|
22
|
-
const target = this.target();
|
|
23
|
-
if (!source || !target) {
|
|
24
|
-
return true;
|
|
25
|
-
}
|
|
26
|
-
let existsSourceHandle = false;
|
|
27
|
-
let existsTargetHandle = false;
|
|
28
|
-
if (this.edge.sourceHandle) {
|
|
29
|
-
existsSourceHandle = !!source.handles().find((handle) => handle.rawHandle.id === this.edge.sourceHandle);
|
|
30
|
-
}
|
|
31
|
-
else {
|
|
32
|
-
existsSourceHandle = !!source.handles().find((handle) => handle.rawHandle.type === 'source');
|
|
33
|
-
}
|
|
34
|
-
if (this.edge.targetHandle) {
|
|
35
|
-
existsTargetHandle = !!target.handles().find((handle) => handle.rawHandle.id === this.edge.targetHandle);
|
|
36
|
-
}
|
|
37
|
-
else {
|
|
38
|
-
existsTargetHandle = !!target.handles().find((handle) => handle.rawHandle.type === 'target');
|
|
39
|
-
}
|
|
40
|
-
return !existsSourceHandle || !existsTargetHandle;
|
|
41
|
-
});
|
|
42
|
-
this.detached$ = toObservable(this.detached);
|
|
43
|
-
this.path = computed(() => {
|
|
44
|
-
const source = this.sourceHandle();
|
|
45
|
-
const target = this.targetHandle();
|
|
46
|
-
// TODO: don't like this
|
|
47
|
-
if (!source || !target) {
|
|
48
|
-
return { path: '' };
|
|
49
|
-
}
|
|
50
|
-
const params = this.getPathFactoryParams(source, target);
|
|
51
|
-
switch (this.curve) {
|
|
52
|
-
case 'straight':
|
|
53
|
-
return straightPath(params);
|
|
54
|
-
case 'bezier':
|
|
55
|
-
return bezierPath(params);
|
|
56
|
-
case 'smooth-step':
|
|
57
|
-
return smoothStepPath(params);
|
|
58
|
-
case 'step':
|
|
59
|
-
return smoothStepPath(params, 0);
|
|
60
|
-
default:
|
|
61
|
-
return this.curve(params);
|
|
62
|
-
}
|
|
63
|
-
});
|
|
64
|
-
this.sourceHandle = extendedComputed((previousHandle) => {
|
|
65
|
-
let handle = null;
|
|
66
|
-
if (this.floating) {
|
|
67
|
-
handle = this.closestHandles().sourceHandle;
|
|
68
|
-
}
|
|
69
|
-
else {
|
|
70
|
-
if (this.edge.sourceHandle) {
|
|
71
|
-
handle =
|
|
72
|
-
this.source()
|
|
73
|
-
?.handles()
|
|
74
|
-
.find((handle) => handle.rawHandle.id === this.edge.sourceHandle) ?? null;
|
|
75
|
-
}
|
|
76
|
-
else {
|
|
77
|
-
handle =
|
|
78
|
-
this.source()
|
|
79
|
-
?.handles()
|
|
80
|
-
.find((handle) => handle.rawHandle.type === 'source') ?? null;
|
|
81
|
-
}
|
|
82
|
-
}
|
|
83
|
-
// In case of virtual scrolling, if the node is scrolled out of view the handle may disappear
|
|
84
|
-
// which could lead to the edge not being rendered
|
|
85
|
-
// so we return the previous handle if the current one is null
|
|
86
|
-
// TODO: check if this breaks anything
|
|
87
|
-
if (handle === null) {
|
|
88
|
-
return previousHandle;
|
|
89
|
-
}
|
|
90
|
-
return handle;
|
|
91
|
-
});
|
|
92
|
-
this.targetHandle = extendedComputed((previousHandle) => {
|
|
93
|
-
let handle = null;
|
|
94
|
-
if (this.floating) {
|
|
95
|
-
handle = this.closestHandles().targetHandle;
|
|
96
|
-
}
|
|
97
|
-
else {
|
|
98
|
-
if (this.edge.targetHandle) {
|
|
99
|
-
handle =
|
|
100
|
-
this.target()
|
|
101
|
-
?.handles()
|
|
102
|
-
.find((handle) => handle.rawHandle.id === this.edge.targetHandle) ?? null;
|
|
103
|
-
}
|
|
104
|
-
else {
|
|
105
|
-
handle =
|
|
106
|
-
this.target()
|
|
107
|
-
?.handles()
|
|
108
|
-
.find((handle) => handle.rawHandle.type === 'target') ?? null;
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
// In case of virtual scrolling, if the node is scrolled out of view the handle may disappear
|
|
112
|
-
// which could lead to the edge not being rendered
|
|
113
|
-
// so we return the previous handle if the current one is null
|
|
114
|
-
// TODO: check if this breaks anything
|
|
115
|
-
if (handle === null) {
|
|
116
|
-
return previousHandle;
|
|
117
|
-
}
|
|
118
|
-
return handle;
|
|
119
|
-
});
|
|
120
|
-
this.closestHandles = computed(() => {
|
|
121
|
-
const source = this.source();
|
|
122
|
-
const target = this.target();
|
|
123
|
-
if (!source || !target) {
|
|
124
|
-
return { sourceHandle: null, targetHandle: null };
|
|
125
|
-
}
|
|
126
|
-
// Get all source handles from source node
|
|
127
|
-
const sourceHandles = this.flowEntitiesService.connection().mode === 'strict'
|
|
128
|
-
? source.handles().filter((h) => h.rawHandle.type === 'source')
|
|
129
|
-
: source.handles();
|
|
130
|
-
// Get all target handles from target node
|
|
131
|
-
const targetHandles = this.flowEntitiesService.connection().mode === 'strict'
|
|
132
|
-
? target.handles().filter((h) => h.rawHandle.type === 'target')
|
|
133
|
-
: target.handles();
|
|
134
|
-
if (sourceHandles.length === 0 || targetHandles.length === 0) {
|
|
135
|
-
return { sourceHandle: null, targetHandle: null };
|
|
136
|
-
}
|
|
137
|
-
let minDistance = Infinity;
|
|
138
|
-
let closestSourceHandle = null;
|
|
139
|
-
let closestTargetHandle = null;
|
|
140
|
-
// Check all combinations of source and target handles
|
|
141
|
-
for (const sourceHandle of sourceHandles) {
|
|
142
|
-
for (const targetHandle of targetHandles) {
|
|
143
|
-
const sourcePoint = sourceHandle.pointAbsolute();
|
|
144
|
-
const targetPoint = targetHandle.pointAbsolute();
|
|
145
|
-
const distance = Math.sqrt(Math.pow(sourcePoint.x - targetPoint.x, 2) + Math.pow(sourcePoint.y - targetPoint.y, 2));
|
|
146
|
-
if (distance < minDistance) {
|
|
147
|
-
minDistance = distance;
|
|
148
|
-
closestSourceHandle = sourceHandle;
|
|
149
|
-
closestTargetHandle = targetHandle;
|
|
150
|
-
}
|
|
151
|
-
}
|
|
152
|
-
}
|
|
153
|
-
return {
|
|
154
|
-
sourceHandle: closestSourceHandle,
|
|
155
|
-
targetHandle: closestTargetHandle,
|
|
156
|
-
};
|
|
157
|
-
});
|
|
158
|
-
/**
|
|
159
|
-
* TODO: not reactive
|
|
160
|
-
*/
|
|
161
|
-
this.markerStartUrl = computed(() => {
|
|
162
|
-
const marker = this.edge.markers?.start;
|
|
163
|
-
return marker ? `url(#${hashCode(JSON.stringify(marker))})` : '';
|
|
164
|
-
});
|
|
165
|
-
/**
|
|
166
|
-
* TODO: not reactive
|
|
167
|
-
*/
|
|
168
|
-
this.markerEndUrl = computed(() => {
|
|
169
|
-
const marker = this.edge.markers?.end;
|
|
170
|
-
return marker ? `url(#${hashCode(JSON.stringify(marker))})` : '';
|
|
171
|
-
});
|
|
172
|
-
this.context = {
|
|
173
|
-
$implicit: {
|
|
174
|
-
// TODO: check if edge could change
|
|
175
|
-
edge: this.edge,
|
|
176
|
-
path: computed(() => this.path().path),
|
|
177
|
-
markerStart: this.markerStartUrl,
|
|
178
|
-
markerEnd: this.markerEndUrl,
|
|
179
|
-
selected: this.selected.asReadonly(),
|
|
180
|
-
shouldLoad: this.shouldLoad,
|
|
181
|
-
},
|
|
182
|
-
};
|
|
183
|
-
this.edgeLabels = {};
|
|
184
|
-
this.type = edge.type ?? 'default';
|
|
185
|
-
this.curve = edge.curve ?? 'bezier';
|
|
186
|
-
this.reconnectable = edge.reconnectable ?? false;
|
|
187
|
-
this.floating = edge.floating ?? false;
|
|
188
|
-
if (edge.edgeLabels?.start)
|
|
189
|
-
this.edgeLabels.start = new EdgeLabelModel(edge.edgeLabels.start);
|
|
190
|
-
if (edge.edgeLabels?.center)
|
|
191
|
-
this.edgeLabels.center = new EdgeLabelModel(edge.edgeLabels.center);
|
|
192
|
-
if (edge.edgeLabels?.end)
|
|
193
|
-
this.edgeLabels.end = new EdgeLabelModel(edge.edgeLabels.end);
|
|
194
|
-
}
|
|
195
|
-
getPathFactoryParams(source, target) {
|
|
196
|
-
return {
|
|
197
|
-
mode: 'edge',
|
|
198
|
-
edge: this.edge,
|
|
199
|
-
sourcePoint: source.pointAbsolute(),
|
|
200
|
-
targetPoint: target.pointAbsolute(),
|
|
201
|
-
sourcePosition: source.rawHandle.position,
|
|
202
|
-
targetPosition: target.rawHandle.position,
|
|
203
|
-
allEdges: this.flowEntitiesService.rawEdges(),
|
|
204
|
-
allNodes: this.flowEntitiesService.rawNodes(),
|
|
205
|
-
};
|
|
206
|
-
}
|
|
207
|
-
}
|
|
208
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"edge.model.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/models/edge.model.ts"],"names":[],"mappings":"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","sourcesContent":["import { computed, inject, signal } from '@angular/core';\nimport { EdgeLabelPosition } from '../interfaces/edge-label.interface';\nimport { Edge, Curve, EdgeType } from '../interfaces/edge.interface';\nimport { EdgeLabelModel } from './edge-label.model';\nimport { NodeModel } from './node.model';\nimport { straightPath } from '../math/edge-path/straigh-path';\nimport { bezierPath } from '../math/edge-path/bezier-path';\nimport { toObservable } from '@angular/core/rxjs-interop';\nimport { FlowEntity } from '../interfaces/flow-entity.interface';\nimport { smoothStepPath } from '../math/edge-path/smooth-step-path';\nimport { hashCode } from '../utils/hash';\nimport { Contextable } from '../interfaces/contextable.interface';\nimport { EdgeContext } from '../interfaces/template-context.interface';\nimport { HandleModel } from './handle.model';\nimport { CurveFactoryParams } from '../interfaces/curve-factory.interface';\nimport { FlowEntitiesService } from '../services/flow-entities.service';\nimport { extendedComputed } from '../utils/signals/extended-computed';\n\nexport class EdgeModel implements FlowEntity, Contextable<EdgeContext> {\n  private readonly flowEntitiesService = inject(FlowEntitiesService);\n\n  public source = signal<NodeModel | undefined>(undefined);\n  public target = signal<NodeModel | undefined>(undefined);\n  public curve: Curve;\n  public type: EdgeType;\n  public reconnectable: boolean | 'source' | 'target';\n  public floating: boolean;\n\n  public selected = signal(false);\n  public selected$ = toObservable(this.selected);\n\n  public shouldLoad = computed(() => (this.source()?.shouldLoad() ?? false) && (this.target()?.shouldLoad() ?? false));\n\n  public renderOrder = signal(0);\n\n  public detached = computed(() => {\n    const source = this.source();\n    const target = this.target();\n\n    if (!source || !target) {\n      return true;\n    }\n\n    let existsSourceHandle = false;\n    let existsTargetHandle = false;\n\n    if (this.edge.sourceHandle) {\n      existsSourceHandle = !!source.handles().find((handle) => handle.rawHandle.id === this.edge.sourceHandle);\n    } else {\n      existsSourceHandle = !!source.handles().find((handle) => handle.rawHandle.type === 'source');\n    }\n\n    if (this.edge.targetHandle) {\n      existsTargetHandle = !!target.handles().find((handle) => handle.rawHandle.id === this.edge.targetHandle);\n    } else {\n      existsTargetHandle = !!target.handles().find((handle) => handle.rawHandle.type === 'target');\n    }\n\n    return !existsSourceHandle || !existsTargetHandle;\n  });\n\n  public detached$ = toObservable(this.detached);\n\n  public path = computed(() => {\n    const source = this.sourceHandle();\n    const target = this.targetHandle();\n\n    // TODO: don't like this\n    if (!source || !target) {\n      return { path: '' };\n    }\n\n    const params = this.getPathFactoryParams(source, target);\n\n    switch (this.curve) {\n      case 'straight':\n        return straightPath(params);\n      case 'bezier':\n        return bezierPath(params);\n      case 'smooth-step':\n        return smoothStepPath(params);\n      case 'step':\n        return smoothStepPath(params, 0);\n      default:\n        return this.curve(params);\n    }\n  });\n\n  public sourceHandle = extendedComputed<HandleModel | null>((previousHandle) => {\n    let handle: HandleModel | null = null;\n\n    if (this.floating) {\n      handle = this.closestHandles().sourceHandle;\n    } else {\n      if (this.edge.sourceHandle) {\n        handle =\n          this.source()\n            ?.handles()\n            .find((handle) => handle.rawHandle.id === this.edge.sourceHandle) ?? null;\n      } else {\n        handle =\n          this.source()\n            ?.handles()\n            .find((handle) => handle.rawHandle.type === 'source') ?? null;\n      }\n    }\n\n    // In case of virtual scrolling, if the node is scrolled out of view the handle may disappear\n    // which could lead to the edge not being rendered\n    // so we return the previous handle if the current one is null\n    // TODO: check if this breaks anything\n    if (handle === null) {\n      return previousHandle;\n    }\n\n    return handle;\n  });\n\n  public targetHandle = extendedComputed<HandleModel | null>((previousHandle) => {\n    let handle: HandleModel | null = null;\n\n    if (this.floating) {\n      handle = this.closestHandles().targetHandle;\n    } else {\n      if (this.edge.targetHandle) {\n        handle =\n          this.target()\n            ?.handles()\n            .find((handle) => handle.rawHandle.id === this.edge.targetHandle) ?? null;\n      } else {\n        handle =\n          this.target()\n            ?.handles()\n            .find((handle) => handle.rawHandle.type === 'target') ?? null;\n      }\n    }\n\n    // In case of virtual scrolling, if the node is scrolled out of view the handle may disappear\n    // which could lead to the edge not being rendered\n    // so we return the previous handle if the current one is null\n    // TODO: check if this breaks anything\n    if (handle === null) {\n      return previousHandle;\n    }\n\n    return handle;\n  });\n\n  public closestHandles = computed(() => {\n    const source = this.source();\n    const target = this.target();\n\n    if (!source || !target) {\n      return { sourceHandle: null, targetHandle: null };\n    }\n\n    // Get all source handles from source node\n    const sourceHandles =\n      this.flowEntitiesService.connection().mode === 'strict'\n        ? source.handles().filter((h) => h.rawHandle.type === 'source')\n        : source.handles();\n    // Get all target handles from target node\n    const targetHandles =\n      this.flowEntitiesService.connection().mode === 'strict'\n        ? target.handles().filter((h) => h.rawHandle.type === 'target')\n        : target.handles();\n\n    if (sourceHandles.length === 0 || targetHandles.length === 0) {\n      return { sourceHandle: null, targetHandle: null };\n    }\n\n    let minDistance = Infinity;\n    let closestSourceHandle: HandleModel | null = null;\n    let closestTargetHandle: HandleModel | null = null;\n\n    // Check all combinations of source and target handles\n    for (const sourceHandle of sourceHandles) {\n      for (const targetHandle of targetHandles) {\n        const sourcePoint = sourceHandle.pointAbsolute();\n        const targetPoint = targetHandle.pointAbsolute();\n\n        const distance = Math.sqrt(\n          Math.pow(sourcePoint.x - targetPoint.x, 2) + Math.pow(sourcePoint.y - targetPoint.y, 2),\n        );\n\n        if (distance < minDistance) {\n          minDistance = distance;\n          closestSourceHandle = sourceHandle;\n          closestTargetHandle = targetHandle;\n        }\n      }\n    }\n\n    return {\n      sourceHandle: closestSourceHandle,\n      targetHandle: closestTargetHandle,\n    };\n  });\n\n  /**\n   * TODO: not reactive\n   */\n  public markerStartUrl = computed(() => {\n    const marker = this.edge.markers?.start;\n\n    return marker ? `url(#${hashCode(JSON.stringify(marker))})` : '';\n  });\n\n  /**\n   * TODO: not reactive\n   */\n  public markerEndUrl = computed(() => {\n    const marker = this.edge.markers?.end;\n\n    return marker ? `url(#${hashCode(JSON.stringify(marker))})` : '';\n  });\n\n  public context = {\n    $implicit: {\n      // TODO: check if edge could change\n      edge: this.edge,\n      path: computed(() => this.path().path),\n      markerStart: this.markerStartUrl,\n      markerEnd: this.markerEndUrl,\n      selected: this.selected.asReadonly(),\n      shouldLoad: this.shouldLoad,\n    },\n  };\n\n  public edgeLabels: { [position in EdgeLabelPosition]?: EdgeLabelModel } = {};\n\n  constructor(public edge: Edge) {\n    this.type = edge.type ?? 'default';\n    this.curve = edge.curve ?? 'bezier';\n    this.reconnectable = edge.reconnectable ?? false;\n    this.floating = edge.floating ?? false;\n\n    if (edge.edgeLabels?.start) this.edgeLabels.start = new EdgeLabelModel(edge.edgeLabels.start);\n    if (edge.edgeLabels?.center) this.edgeLabels.center = new EdgeLabelModel(edge.edgeLabels.center);\n    if (edge.edgeLabels?.end) this.edgeLabels.end = new EdgeLabelModel(edge.edgeLabels.end);\n  }\n\n  private getPathFactoryParams(source: HandleModel, target: HandleModel): CurveFactoryParams {\n    return {\n      mode: 'edge',\n      edge: this.edge,\n      sourcePoint: source.pointAbsolute(),\n      targetPoint: target.pointAbsolute(),\n      sourcePosition: source.rawHandle.position,\n      targetPosition: target.rawHandle.position,\n      allEdges: this.flowEntitiesService.rawEdges(),\n      allNodes: this.flowEntitiesService.rawNodes(),\n    };\n  }\n}\n"]}
|