ngx-vflow 1.16.4 → 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.
Files changed (181) hide show
  1. package/README.md +35 -69
  2. package/fesm2022/ngx-vflow-testing.mjs +77 -67
  3. package/fesm2022/ngx-vflow-testing.mjs.map +1 -1
  4. package/fesm2022/ngx-vflow.mjs +926 -559
  5. package/fesm2022/ngx-vflow.mjs.map +1 -1
  6. package/lib/vflow/components/custom-node-base/custom-node-base.component.d.ts +1 -1
  7. package/lib/vflow/components/vflow/vflow.component.d.ts +18 -9
  8. package/lib/vflow/directives/auto-pan.directive.d.ts +16 -0
  9. package/lib/vflow/directives/changes-controller.directive.d.ts +13 -87
  10. package/lib/vflow/directives/connection-controller.directive.d.ts +8 -10
  11. package/lib/vflow/directives/space-point-context.directive.d.ts +2 -1
  12. package/lib/vflow/interfaces/component-node-event.interface.d.ts +1 -2
  13. package/lib/vflow/interfaces/connection-events.interface.d.ts +66 -0
  14. package/lib/vflow/interfaces/connection-settings.interface.d.ts +7 -1
  15. package/lib/vflow/interfaces/connection.interface.d.ts +0 -5
  16. package/lib/vflow/interfaces/curve-factory.interface.d.ts +2 -2
  17. package/lib/vflow/interfaces/edge-label.interface.d.ts +0 -4
  18. package/lib/vflow/interfaces/edge.interface.d.ts +36 -8
  19. package/lib/vflow/interfaces/node.interface.d.ts +44 -63
  20. package/lib/vflow/interfaces/selection-strategy.interface.d.ts +16 -0
  21. package/lib/vflow/interfaces/template-context.interface.d.ts +8 -3
  22. package/lib/vflow/models/connection.model.d.ts +1 -1
  23. package/lib/vflow/models/edge.model.d.ts +20 -23
  24. package/lib/vflow/models/handle.model.d.ts +1 -1
  25. package/lib/vflow/models/node.model.d.ts +4 -7
  26. package/lib/vflow/public-components/custom-node/custom-node.component.d.ts +1 -1
  27. package/lib/vflow/public-components/custom-template-edge/custom-template-edge.component.d.ts +2 -1
  28. package/lib/vflow/public-components/minimap/minimap.component.d.ts +1 -1
  29. package/lib/vflow/services/draggable.service.d.ts +3 -0
  30. package/lib/vflow/services/flow-entities.service.d.ts +2 -2
  31. package/lib/vflow/services/flow-settings.service.d.ts +3 -0
  32. package/lib/vflow/services/flow-status.service.d.ts +35 -7
  33. package/lib/vflow/services/selection.service.d.ts +4 -1
  34. package/lib/vflow/strategies/default-selection.strategy.d.ts +6 -0
  35. package/lib/vflow/strategies/manual-selection.strategy.d.ts +5 -0
  36. package/lib/vflow/types/selection-mode.type.d.ts +1 -0
  37. package/lib/vflow/types/unwrap-signal.type.d.ts +4 -0
  38. package/lib/vflow/utils/adjust-direction.d.ts +2 -6
  39. package/lib/vflow/utils/identity-checker/reference-identity-checker.d.ts +2 -2
  40. package/lib/vflow/utils/is-vflow-component.d.ts +0 -2
  41. package/lib/vflow/utils/signals/to-lazy-signal.d.ts +4 -4
  42. package/package.json +3 -7
  43. package/public-api.d.ts +3 -2
  44. package/testing/component-mocks/vflow-mock.component.d.ts +8 -6
  45. package/testing/directive-mocks/connection-controller-mock.directive.d.ts +8 -6
  46. package/esm2022/lib/vflow/components/alignment-helper/alignment-helper.component.mjs +0 -103
  47. package/esm2022/lib/vflow/components/background/background.component.mjs +0 -121
  48. package/esm2022/lib/vflow/components/connection/connection.component.mjs +0 -157
  49. package/esm2022/lib/vflow/components/custom-node-base/custom-node-base.component.mjs +0 -57
  50. package/esm2022/lib/vflow/components/default-node/default-node.component.mjs +0 -16
  51. package/esm2022/lib/vflow/components/defs/defs.component.mjs +0 -16
  52. package/esm2022/lib/vflow/components/edge/edge.component.mjs +0 -54
  53. package/esm2022/lib/vflow/components/edge-label/edge-label.component.mjs +0 -76
  54. package/esm2022/lib/vflow/components/node/node.component.mjs +0 -107
  55. package/esm2022/lib/vflow/components/preview-flow/draw-node.mjs +0 -100
  56. package/esm2022/lib/vflow/components/preview-flow/preview-flow.component.mjs +0 -62
  57. package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +0 -437
  58. package/esm2022/lib/vflow/constants/magic-number-to-fix-glitch-in-chrome.constant.mjs +0 -4
  59. package/esm2022/lib/vflow/decorators/microtask.decorator.mjs +0 -11
  60. package/esm2022/lib/vflow/directives/changes-controller.directive.mjs +0 -165
  61. package/esm2022/lib/vflow/directives/connection-controller.directive.mjs +0 -145
  62. package/esm2022/lib/vflow/directives/drag-handle.directive.mjs +0 -28
  63. package/esm2022/lib/vflow/directives/flow-size-controller.directive.mjs +0 -40
  64. package/esm2022/lib/vflow/directives/handle-size-controller.directive.mjs +0 -41
  65. package/esm2022/lib/vflow/directives/map-context.directive.mjs +0 -116
  66. package/esm2022/lib/vflow/directives/node-handles-controller.directive.mjs +0 -33
  67. package/esm2022/lib/vflow/directives/node-resize-controller.directive.mjs +0 -37
  68. package/esm2022/lib/vflow/directives/pointer.directive.mjs +0 -84
  69. package/esm2022/lib/vflow/directives/reference.directive.mjs +0 -17
  70. package/esm2022/lib/vflow/directives/root-pointer.directive.mjs +0 -58
  71. package/esm2022/lib/vflow/directives/root-svg-context.directive.mjs +0 -35
  72. package/esm2022/lib/vflow/directives/selectable.directive.mjs +0 -48
  73. package/esm2022/lib/vflow/directives/space-point-context.directive.mjs +0 -42
  74. package/esm2022/lib/vflow/directives/template.directive.mjs +0 -119
  75. package/esm2022/lib/vflow/interfaces/alignment-helper-settings.interface.mjs +0 -2
  76. package/esm2022/lib/vflow/interfaces/box.mjs +0 -2
  77. package/esm2022/lib/vflow/interfaces/component-node-event.interface.mjs +0 -2
  78. package/esm2022/lib/vflow/interfaces/connection-settings.interface.mjs +0 -2
  79. package/esm2022/lib/vflow/interfaces/connection.interface.mjs +0 -2
  80. package/esm2022/lib/vflow/interfaces/connection.internal.interface.mjs +0 -2
  81. package/esm2022/lib/vflow/interfaces/contextable.interface.mjs +0 -2
  82. package/esm2022/lib/vflow/interfaces/curve-factory.interface.mjs +0 -2
  83. package/esm2022/lib/vflow/interfaces/edge-label.interface.mjs +0 -2
  84. package/esm2022/lib/vflow/interfaces/edge.interface.mjs +0 -2
  85. package/esm2022/lib/vflow/interfaces/fit-view-options.interface.mjs +0 -2
  86. package/esm2022/lib/vflow/interfaces/flow-entity.interface.mjs +0 -2
  87. package/esm2022/lib/vflow/interfaces/intersecting-nodes-options.interface.mjs +0 -2
  88. package/esm2022/lib/vflow/interfaces/marker.interface.mjs +0 -2
  89. package/esm2022/lib/vflow/interfaces/node-preview.interface.mjs +0 -2
  90. package/esm2022/lib/vflow/interfaces/node.interface.mjs +0 -53
  91. package/esm2022/lib/vflow/interfaces/optimization.interface.mjs +0 -7
  92. package/esm2022/lib/vflow/interfaces/point.interface.mjs +0 -2
  93. package/esm2022/lib/vflow/interfaces/rect.mjs +0 -10
  94. package/esm2022/lib/vflow/interfaces/template-context.interface.mjs +0 -2
  95. package/esm2022/lib/vflow/interfaces/viewport.interface.mjs +0 -2
  96. package/esm2022/lib/vflow/math/edge-path/bezier-path.mjs +0 -66
  97. package/esm2022/lib/vflow/math/edge-path/smooth-step-path.mjs +0 -226
  98. package/esm2022/lib/vflow/math/edge-path/straigh-path.mjs +0 -12
  99. package/esm2022/lib/vflow/math/point-on-line-by-ratio.mjs +0 -12
  100. package/esm2022/lib/vflow/models/connection.model.mjs +0 -31
  101. package/esm2022/lib/vflow/models/edge-label.model.mjs +0 -8
  102. package/esm2022/lib/vflow/models/edge.model.mjs +0 -208
  103. package/esm2022/lib/vflow/models/handle.model.mjs +0 -98
  104. package/esm2022/lib/vflow/models/minimap.model.mjs +0 -7
  105. package/esm2022/lib/vflow/models/node.model.mjs +0 -181
  106. package/esm2022/lib/vflow/models/toolbar.model.mjs +0 -36
  107. package/esm2022/lib/vflow/public-components/custom-dynamic-node/custom-dynamic-node.component.mjs +0 -25
  108. package/esm2022/lib/vflow/public-components/custom-node/custom-node.component.mjs +0 -24
  109. package/esm2022/lib/vflow/public-components/custom-template-edge/custom-template-edge.component.mjs +0 -29
  110. package/esm2022/lib/vflow/public-components/handle/handle.component.mjs +0 -53
  111. package/esm2022/lib/vflow/public-components/minimap/minimap.component.mjs +0 -108
  112. package/esm2022/lib/vflow/public-components/node-toolbar/node-toolbar.component.mjs +0 -73
  113. package/esm2022/lib/vflow/public-components/resizable/resizable.component.mjs +0 -272
  114. package/esm2022/lib/vflow/services/component-event-bus.service.mjs +0 -18
  115. package/esm2022/lib/vflow/services/draggable.service.mjs +0 -124
  116. package/esm2022/lib/vflow/services/edge-changes.service.mjs +0 -43
  117. package/esm2022/lib/vflow/services/edge-rendering.service.mjs +0 -40
  118. package/esm2022/lib/vflow/services/flow-entities.service.mjs +0 -56
  119. package/esm2022/lib/vflow/services/flow-rendering.service.mjs +0 -41
  120. package/esm2022/lib/vflow/services/flow-settings.service.mjs +0 -33
  121. package/esm2022/lib/vflow/services/flow-status.service.mjs +0 -49
  122. package/esm2022/lib/vflow/services/handle.service.mjs +0 -30
  123. package/esm2022/lib/vflow/services/keyboard.service.mjs +0 -47
  124. package/esm2022/lib/vflow/services/node-accessor.service.mjs +0 -16
  125. package/esm2022/lib/vflow/services/node-changes.service.mjs +0 -41
  126. package/esm2022/lib/vflow/services/node-rendering.service.mjs +0 -66
  127. package/esm2022/lib/vflow/services/overlays.service.mjs +0 -35
  128. package/esm2022/lib/vflow/services/preview-flow-render-strategy.service.mjs +0 -21
  129. package/esm2022/lib/vflow/services/selection.service.mjs +0 -54
  130. package/esm2022/lib/vflow/services/viewport.service.mjs +0 -64
  131. package/esm2022/lib/vflow/types/background.type.mjs +0 -2
  132. package/esm2022/lib/vflow/types/connection-mode.type.mjs +0 -2
  133. package/esm2022/lib/vflow/types/edge-change.type.mjs +0 -2
  134. package/esm2022/lib/vflow/types/handle-type.type.mjs +0 -2
  135. package/esm2022/lib/vflow/types/keyboard-action.type.mjs +0 -2
  136. package/esm2022/lib/vflow/types/node-change.type.mjs +0 -2
  137. package/esm2022/lib/vflow/types/position.type.mjs +0 -2
  138. package/esm2022/lib/vflow/types/viewport-change-type.type.mjs +0 -2
  139. package/esm2022/lib/vflow/utils/add-nodes-to-edges.mjs +0 -11
  140. package/esm2022/lib/vflow/utils/adjust-direction.mjs +0 -30
  141. package/esm2022/lib/vflow/utils/align-number.mjs +0 -4
  142. package/esm2022/lib/vflow/utils/assert-injector.mjs +0 -27
  143. package/esm2022/lib/vflow/utils/event.mjs +0 -4
  144. package/esm2022/lib/vflow/utils/get-os.mjs +0 -24
  145. package/esm2022/lib/vflow/utils/get-overlapping-area.mjs +0 -6
  146. package/esm2022/lib/vflow/utils/get-space-points.mjs +0 -25
  147. package/esm2022/lib/vflow/utils/hash.mjs +0 -7
  148. package/esm2022/lib/vflow/utils/id.mjs +0 -5
  149. package/esm2022/lib/vflow/utils/identity-checker/reference-identity-checker.mjs +0 -28
  150. package/esm2022/lib/vflow/utils/is-callable.mjs +0 -10
  151. package/esm2022/lib/vflow/utils/is-defined.mjs +0 -4
  152. package/esm2022/lib/vflow/utils/is-group-node.mjs +0 -4
  153. package/esm2022/lib/vflow/utils/is-vflow-component.mjs +0 -9
  154. package/esm2022/lib/vflow/utils/nodes.mjs +0 -60
  155. package/esm2022/lib/vflow/utils/resizable.mjs +0 -11
  156. package/esm2022/lib/vflow/utils/round.mjs +0 -2
  157. package/esm2022/lib/vflow/utils/signals/extended-computed.mjs +0 -15
  158. package/esm2022/lib/vflow/utils/signals/to-lazy-signal.mjs +0 -35
  159. package/esm2022/lib/vflow/utils/to-unified-node.mjs +0 -24
  160. package/esm2022/lib/vflow/utils/transform-background.mjs +0 -4
  161. package/esm2022/lib/vflow/utils/viewport.mjs +0 -51
  162. package/esm2022/lib/vflow/vflow.mjs +0 -29
  163. package/esm2022/ngx-vflow.mjs +0 -5
  164. package/esm2022/public-api.mjs +0 -55
  165. package/esm2022/testing/component-mocks/custom-template-edge-mock.component.mjs +0 -16
  166. package/esm2022/testing/component-mocks/handle-mock.component.mjs +0 -26
  167. package/esm2022/testing/component-mocks/minimap-mock.component.mjs +0 -24
  168. package/esm2022/testing/component-mocks/node-toolbar-mock.component.mjs +0 -23
  169. package/esm2022/testing/component-mocks/resizable-mock.component.mjs +0 -27
  170. package/esm2022/testing/component-mocks/vflow-mock.component.mjs +0 -299
  171. package/esm2022/testing/directive-mocks/connection-controller-mock.directive.mjs +0 -29
  172. package/esm2022/testing/directive-mocks/drag-handle-mock.directive.mjs +0 -11
  173. package/esm2022/testing/directive-mocks/selectable-mock.directive.mjs +0 -14
  174. package/esm2022/testing/directive-mocks/template-mock.directive.mjs +0 -101
  175. package/esm2022/testing/ngx-vflow-testing.mjs +0 -5
  176. package/esm2022/testing/provide-custom-node-mocks.mjs +0 -60
  177. package/esm2022/testing/public-api.mjs +0 -13
  178. package/esm2022/testing/types.mjs +0 -2
  179. package/esm2022/testing/vflow-mocks.mjs +0 -28
  180. package/lib/vflow/public-components/custom-dynamic-node/custom-dynamic-node.component.d.ts +0 -13
  181. 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"]}