ngx-vflow 0.15.0 → 0.16.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/esm2022/lib/vflow/components/background/background.component.mjs +72 -4
- package/esm2022/lib/vflow/components/connection/connection.component.mjs +6 -1
- package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +7 -4
- package/esm2022/lib/vflow/interfaces/edge.interface.mjs +1 -1
- package/esm2022/lib/vflow/interfaces/optimization.interface.mjs +1 -1
- package/esm2022/lib/vflow/math/edge-path/smooth-step-path.mjs +170 -0
- package/esm2022/lib/vflow/models/edge.model.mjs +6 -1
- package/esm2022/lib/vflow/public-components/minimap/minimap.component.mjs +8 -2
- package/esm2022/lib/vflow/services/node-rendering.service.mjs +8 -1
- package/esm2022/lib/vflow/types/background.type.mjs +1 -1
- package/esm2022/lib/vflow/utils/is-group-node.mjs +4 -0
- package/fesm2022/ngx-vflow.mjs +270 -7
- package/fesm2022/ngx-vflow.mjs.map +1 -1
- package/lib/vflow/components/background/background.component.d.ts +13 -0
- package/lib/vflow/components/vflow/vflow.component.d.ts +2 -0
- package/lib/vflow/interfaces/edge.interface.d.ts +1 -1
- package/lib/vflow/interfaces/optimization.interface.d.ts +13 -0
- package/lib/vflow/math/edge-path/smooth-step-path.d.ts +5 -0
- package/lib/vflow/services/node-rendering.service.d.ts +2 -0
- package/lib/vflow/types/background.type.d.ts +24 -1
- package/lib/vflow/utils/is-group-node.d.ts +2 -0
- package/package.json +3 -3
|
@@ -0,0 +1,170 @@
|
|
|
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(source, target, sourcePosition, targetPosition, borderRadius = 5) {
|
|
142
|
+
const [points, labelX, labelY] = getPoints({
|
|
143
|
+
source,
|
|
144
|
+
sourcePosition,
|
|
145
|
+
target,
|
|
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
|
+
return {
|
|
161
|
+
path,
|
|
162
|
+
points: {
|
|
163
|
+
// TODO start and end points temporary unavailable for this path
|
|
164
|
+
start: { x: labelX, y: labelY },
|
|
165
|
+
center: { x: labelX, y: labelY },
|
|
166
|
+
end: { x: labelX, y: labelY },
|
|
167
|
+
}
|
|
168
|
+
};
|
|
169
|
+
}
|
|
170
|
+
//# 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 { PathData } from '../../interfaces/path-data.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  source: Point,\n  target: Point,\n  sourcePosition: Position,\n  targetPosition: Position,\n  borderRadius: number = 5\n): PathData {\n  const [points, labelX, labelY] = getPoints({\n    source,\n    sourcePosition,\n    target,\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  return {\n    path,\n    points: {\n      // TODO start and end points temporary unavailable for this path\n      start: { x: labelX, y: labelY },\n      center: { x: labelX, y: labelY },\n      end: { x: labelX, y: labelY },\n    }\n  }\n}\n"]}
|
|
@@ -3,6 +3,7 @@ import { EdgeLabelModel } from "./edge-label.model";
|
|
|
3
3
|
import { straightPath } from "../math/edge-path/straigh-path";
|
|
4
4
|
import { bezierPath } from "../math/edge-path/bezier-path";
|
|
5
5
|
import { toObservable } from "@angular/core/rxjs-interop";
|
|
6
|
+
import { smoothStepPath } from "../math/edge-path/smooth-step-path";
|
|
6
7
|
export class EdgeModel {
|
|
7
8
|
constructor(edge) {
|
|
8
9
|
this.edge = edge;
|
|
@@ -72,6 +73,10 @@ export class EdgeModel {
|
|
|
72
73
|
return straightPath(source.pointAbsolute(), target.pointAbsolute(), this.usingPoints);
|
|
73
74
|
case 'bezier':
|
|
74
75
|
return bezierPath(source.pointAbsolute(), target.pointAbsolute(), source.rawHandle.position, target.rawHandle.position, this.usingPoints);
|
|
76
|
+
case 'smooth-step':
|
|
77
|
+
return smoothStepPath(source.pointAbsolute(), target.pointAbsolute(), source.rawHandle.position, target.rawHandle.position);
|
|
78
|
+
case 'step':
|
|
79
|
+
return smoothStepPath(source.pointAbsolute(), target.pointAbsolute(), source.rawHandle.position, target.rawHandle.position, 0);
|
|
75
80
|
}
|
|
76
81
|
});
|
|
77
82
|
this.edgeLabels = {};
|
|
@@ -86,4 +91,4 @@ export class EdgeModel {
|
|
|
86
91
|
this.usingPoints = [!!this.edgeLabels.start, !!this.edgeLabels.center, !!this.edgeLabels.end];
|
|
87
92
|
}
|
|
88
93
|
}
|
|
89
|
-
//# 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 { WritableSignal, computed, 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 { UsingPoints } from \"../types/using-points.type\";\nimport { HandleModel } from \"./handle.model\";\nimport { toObservable } from \"@angular/core/rxjs-interop\";\nimport { FlowEntity } from \"../interfaces/flow-entity.interface\";\n\nexport class EdgeModel implements FlowEntity {\n  public source = signal<NodeModel | undefined>(undefined)\n  public target = signal<NodeModel | undefined>(undefined)\n  public curve: Curve\n  public type: EdgeType\n\n  public selected = signal(false);\n  public selected$ = toObservable(this.selected);\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()\n        .find(handle => handle.rawHandle.id === this.edge.sourceHandle)\n    } else {\n      existsSourceHandle = !!source.handles()\n        .find(handle => handle.rawHandle.type === 'source')\n    }\n\n    if (this.edge.targetHandle) {\n      existsTargetHandle = !!target.handles()\n        .find(handle => handle.rawHandle.id === this.edge.targetHandle)\n    } else {\n      existsTargetHandle = !!target.handles()\n        .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    let source: HandleModel | undefined\n    if (this.edge.sourceHandle) {\n      source = this.source()?.handles()\n        .find(handle => handle.rawHandle.id === this.edge.sourceHandle)\n    } else {\n      source = this.source()?.handles()\n        .find(handle => handle.rawHandle.type === 'source')\n    }\n\n    let target: HandleModel | undefined\n    if (this.edge.targetHandle) {\n      target = this.target()?.handles()\n        .find(handle => handle.rawHandle.id === this.edge.targetHandle)\n    } else {\n      target = this.target()?.handles()\n        .find(handle => handle.rawHandle.type === 'target')\n    }\n\n    // TODO: don't like this\n    if (!source || !target) {\n      return {\n        path: '',\n        points: {\n          start: { x: 0, y: 0 },\n          center: { x: 0, y: 0 },\n          end: { x: 0, y: 0 }\n        }\n      }\n    }\n\n    switch (this.curve) {\n      case 'straight':\n        return straightPath(source.pointAbsolute(), target.pointAbsolute(), this.usingPoints)\n      case 'bezier':\n        return bezierPath(\n          source.pointAbsolute(), target.pointAbsolute(),\n          source.rawHandle.position,\n          target.rawHandle.position,\n          this.usingPoints\n        )\n    }\n  })\n\n  public edgeLabels: { [position in EdgeLabelPosition]?: EdgeLabelModel } = {}\n\n  private usingPoints: UsingPoints\n\n  constructor(public edge: Edge) {\n    this.type = edge.type ?? 'default'\n    this.curve = edge.curve ?? 'bezier'\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    this.usingPoints = [!!this.edgeLabels.start, !!this.edgeLabels.center, !!this.edgeLabels.end]\n  }\n}\n"]}
|
|
94
|
+
//# 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 { WritableSignal, computed, 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 { UsingPoints } from \"../types/using-points.type\";\nimport { HandleModel } from \"./handle.model\";\nimport { toObservable } from \"@angular/core/rxjs-interop\";\nimport { FlowEntity } from \"../interfaces/flow-entity.interface\";\nimport { smoothStepPath } from \"../math/edge-path/smooth-step-path\";\n\nexport class EdgeModel implements FlowEntity {\n  public source = signal<NodeModel | undefined>(undefined)\n  public target = signal<NodeModel | undefined>(undefined)\n  public curve: Curve\n  public type: EdgeType\n\n  public selected = signal(false);\n  public selected$ = toObservable(this.selected);\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()\n        .find(handle => handle.rawHandle.id === this.edge.sourceHandle)\n    } else {\n      existsSourceHandle = !!source.handles()\n        .find(handle => handle.rawHandle.type === 'source')\n    }\n\n    if (this.edge.targetHandle) {\n      existsTargetHandle = !!target.handles()\n        .find(handle => handle.rawHandle.id === this.edge.targetHandle)\n    } else {\n      existsTargetHandle = !!target.handles()\n        .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    let source: HandleModel | undefined\n    if (this.edge.sourceHandle) {\n      source = this.source()?.handles()\n        .find(handle => handle.rawHandle.id === this.edge.sourceHandle)\n    } else {\n      source = this.source()?.handles()\n        .find(handle => handle.rawHandle.type === 'source')\n    }\n\n    let target: HandleModel | undefined\n    if (this.edge.targetHandle) {\n      target = this.target()?.handles()\n        .find(handle => handle.rawHandle.id === this.edge.targetHandle)\n    } else {\n      target = this.target()?.handles()\n        .find(handle => handle.rawHandle.type === 'target')\n    }\n\n    // TODO: don't like this\n    if (!source || !target) {\n      return {\n        path: '',\n        points: {\n          start: { x: 0, y: 0 },\n          center: { x: 0, y: 0 },\n          end: { x: 0, y: 0 }\n        }\n      }\n    }\n\n    switch (this.curve) {\n      case 'straight':\n        return straightPath(source.pointAbsolute(), target.pointAbsolute(), this.usingPoints)\n      case 'bezier':\n        return bezierPath(\n          source.pointAbsolute(), target.pointAbsolute(),\n          source.rawHandle.position,\n          target.rawHandle.position,\n          this.usingPoints\n        )\n      case 'smooth-step':\n        return smoothStepPath(\n          source.pointAbsolute(), target.pointAbsolute(),\n          source.rawHandle.position,\n          target.rawHandle.position,\n        )\n      case 'step':\n        return smoothStepPath(\n          source.pointAbsolute(), target.pointAbsolute(),\n          source.rawHandle.position,\n          target.rawHandle.position,\n          0\n        )\n    }\n  })\n\n  public edgeLabels: { [position in EdgeLabelPosition]?: EdgeLabelModel } = {}\n\n  private usingPoints: UsingPoints\n\n  constructor(public edge: Edge) {\n    this.type = edge.type ?? 'default'\n    this.curve = edge.curve ?? 'bezier'\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    this.usingPoints = [!!this.edgeLabels.start, !!this.edgeLabels.center, !!this.edgeLabels.end]\n  }\n}\n"]}
|
|
@@ -29,7 +29,13 @@ export class MiniMapComponent {
|
|
|
29
29
|
}
|
|
30
30
|
return 0.2;
|
|
31
31
|
});
|
|
32
|
-
this.viewportColor = computed(() =>
|
|
32
|
+
this.viewportColor = computed(() => {
|
|
33
|
+
const bg = this.flowSettingsService.background();
|
|
34
|
+
if (bg.type === 'dots' || bg.type === 'solid') {
|
|
35
|
+
return bg.color ?? '#fff';
|
|
36
|
+
}
|
|
37
|
+
return '#fff';
|
|
38
|
+
});
|
|
33
39
|
this.hovered = signal(false);
|
|
34
40
|
this.minimapPoint = computed(() => {
|
|
35
41
|
switch (this.minimapPosition()) {
|
|
@@ -116,4 +122,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
116
122
|
type: ViewChild,
|
|
117
123
|
args: ['minimap', { static: true }]
|
|
118
124
|
}] } });
|
|
119
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"minimap.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/minimap/minimap.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/minimap/minimap.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, computed, inject, Injector, Input, OnInit, signal, TemplateRef, ViewChild } from '@angular/core';\nimport { FlowEntitiesService } from '../../services/flow-entities.service';\nimport { MinimapModel } from '../../models/minimap.model';\nimport { NodeModel } from '../../models/node.model';\nimport { FlowSettingsService } from '../../services/flow-settings.service';\nimport { getViewportForBounds } from '../../utils/viewport';\nimport { getNodesBounds } from '../../utils/nodes';\nimport { ViewportService } from '../../services/viewport.service';\n\nexport type MiniMapPosition = 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right'\n\n@Component({\n  selector: 'mini-map',\n  templateUrl: './minimap.component.html',\n  styleUrls: [`./minimap.component.scss`]\n})\nexport class MiniMapComponent implements OnInit {\n  protected entitiesService = inject(FlowEntitiesService)\n  protected flowSettingsService = inject(FlowSettingsService)\n  protected viewportService = inject(ViewportService)\n  protected injector = inject(Injector)\n\n  /**\n   * The corner of the flow where to render a mini-map\n   */\n  @Input()\n  public set position(value: MiniMapPosition) {\n    this.minimapPosition.set(value)\n  }\n\n  /**\n   * The color outside the viewport (invisible area)\n   */\n  @Input()\n  public maskColor = `rgba(215, 215, 215, 0.6)`\n\n  /**\n   * The minimap stroke color\n   */\n  @Input()\n  public strokeColor = `rgb(200, 200, 200)`\n\n  /**\n   * Make a minimap bigger on hover\n   */\n  @Input()\n  public set scaleOnHover(value: boolean) {\n    this.scaleOnHoverSignal.set(value)\n  }\n\n  @ViewChild('minimap', { static: true })\n  private minimap!: TemplateRef<unknown>\n\n  private readonly minimapOffset = 10\n\n  private readonly minimapScale = computed(() => {\n    if (this.scaleOnHoverSignal()) {\n      return this.hovered() ? 0.4 : 0.2\n    }\n\n    return 0.2\n  })\n\n  protected viewportColor = computed(() => this.flowSettingsService.background().color ?? '#fff')\n\n  protected hovered = signal(false)\n\n  protected minimapPoint = computed(() => {\n    switch (this.minimapPosition()) {\n      case 'top-left':\n        return { x: this.minimapOffset, y: this.minimapOffset }\n      case 'top-right':\n        return {\n          x: this.flowSettingsService.computedFlowWidth() - this.minimapWidth() - this.minimapOffset,\n          y: this.minimapOffset\n        }\n      case 'bottom-left':\n        return {\n          x: this.minimapOffset,\n          y: this.flowSettingsService.computedFlowHeight() - this.minimapHeight() - this.minimapOffset\n        }\n      case 'bottom-right':\n        return {\n          x: this.flowSettingsService.computedFlowWidth() - this.minimapWidth() - this.minimapOffset,\n          y: this.flowSettingsService.computedFlowHeight() - this.minimapHeight() - this.minimapOffset\n        }\n    }\n  })\n\n  protected minimapWidth = computed(() =>\n    this.flowSettingsService.computedFlowWidth() * this.minimapScale()\n  )\n  protected minimapHeight = computed(() =>\n    this.flowSettingsService.computedFlowHeight() * this.minimapScale()\n  )\n\n  protected viewportTransform = computed(() => {\n    const viewport = this.viewportService.readableViewport();\n    let scale = 1 / viewport.zoom\n\n    let x = -(viewport.x * this.minimapScale()) * scale\n    x /= this.minimapScale()\n\n    let y = -(viewport.y * this.minimapScale()) * scale\n    y /= this.minimapScale()\n\n    scale /= this.minimapScale()\n\n    return `translate(${x}, ${y}) scale(${scale})`;\n  });\n\n\n  protected boundsViewport = computed(() => {\n    const nodes = this.entitiesService.nodes()\n\n    return getViewportForBounds(\n      getNodesBounds(nodes),\n      this.flowSettingsService.computedFlowWidth(),\n      this.flowSettingsService.computedFlowHeight(),\n      -Infinity,\n      1.5,\n      0\n    )\n\n  })\n\n  protected minimapTransform = computed(() => {\n    const vport = this.boundsViewport()\n\n    const x = vport.x * this.minimapScale()\n    const y = vport.y * this.minimapScale()\n    const scale = vport.zoom * this.minimapScale()\n\n    return `translate(${x} ${y}) scale(${scale})`\n  })\n\n  private minimapPosition = signal<MiniMapPosition>('bottom-right')\n\n  private scaleOnHoverSignal = signal(false)\n\n  public ngOnInit(): void {\n    const model = new MinimapModel()\n    model.template.set(this.minimap)\n\n    this.entitiesService.minimap.set(model)\n  }\n\n  protected trackNodes(idx: number, { node }: NodeModel) {\n    return node\n  }\n}\n","<ng-template #minimap>\n  <svg:rect\n    [attr.x]=\"minimapPoint().x\"\n    [attr.y]=\"minimapPoint().y\"\n    [attr.width]=\"minimapWidth()\"\n    [attr.height]=\"minimapHeight()\"\n    [attr.stroke]=\"strokeColor\"\n    fill=\"none\"\n  />\n\n  <svg:svg\n    [attr.x]=\"minimapPoint().x\"\n    [attr.y]=\"minimapPoint().y\"\n    [attr.width]=\"minimapWidth()\"\n    [attr.height]=\"minimapHeight()\"\n    (mouseover)=\"hovered.set(true)\"\n    (mouseleave)=\"hovered.set(false)\"\n  >\n    <svg:rect\n      [attr.width]=\"minimapWidth()\"\n      [attr.height]=\"minimapHeight()\"\n      [attr.fill]=\"maskColor\"\n    />\n\n    <svg:g [attr.transform]=\"minimapTransform()\">\n      <svg:rect\n        [attr.fill]=\"viewportColor()\"\n        [attr.transform]=\"viewportTransform()\"\n        [attr.width]=\"minimapWidth()\"\n        [attr.height]=\"minimapHeight()\"\n      />\n\n      <ng-container\n        *ngFor=\"let model of entitiesService.nodes(); trackBy: trackNodes\"\n      >\n        <svg:foreignObject\n          *ngIf=\"model.node.type === 'default' || model.node.type === 'html-template' || model.isComponentType\"\n          [attr.transform]=\"model.pointTransform()\"\n          [attr.width]=\"model.size().width\"\n          [attr.height]=\"model.size().height\"\n        >\n          <default-node\n            [selected]=\"model.selected()\"\n            [style.width.px]=\"model.size().width\"\n            [style.height.px]=\"model.size().height\"\n            [style.max-width.px]=\"model.size().width\"\n            [style.max-height.px]=\"model.size().height\"\n          >\n            <div [outerHTML]=\"model.text()\"></div>\n          </default-node>\n        </svg:foreignObject>\n\n        <svg:rect\n          *ngIf=\"model.node.type === 'default-group' || model.node.type === 'template-group'\"\n          class=\"default-group-node\"\n          rx=\"5\"\n          ry=\"5\"\n          [attr.transform]=\"model.pointTransform()\"\n          [class.default-group-node_selected]=\"model.selected()\"\n          [attr.width]=\"model.size().width\"\n          [attr.height]=\"model.size().height\"\n          [style.stroke]=\"model.color()\"\n          [style.fill]=\"model.color()\"\n        />\n\n      </ng-container>\n    </svg:g>\n  </svg:svg>\n</ng-template>\n"]}
|
|
125
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"minimap.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/minimap/minimap.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/minimap/minimap.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, computed, inject, Injector, Input, OnInit, signal, TemplateRef, ViewChild } from '@angular/core';\nimport { FlowEntitiesService } from '../../services/flow-entities.service';\nimport { MinimapModel } from '../../models/minimap.model';\nimport { NodeModel } from '../../models/node.model';\nimport { FlowSettingsService } from '../../services/flow-settings.service';\nimport { getViewportForBounds } from '../../utils/viewport';\nimport { getNodesBounds } from '../../utils/nodes';\nimport { ViewportService } from '../../services/viewport.service';\n\nexport type MiniMapPosition = 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right'\n\n@Component({\n  selector: 'mini-map',\n  templateUrl: './minimap.component.html',\n  styleUrls: [`./minimap.component.scss`]\n})\nexport class MiniMapComponent implements OnInit {\n  protected entitiesService = inject(FlowEntitiesService)\n  protected flowSettingsService = inject(FlowSettingsService)\n  protected viewportService = inject(ViewportService)\n  protected injector = inject(Injector)\n\n  /**\n   * The corner of the flow where to render a mini-map\n   */\n  @Input()\n  public set position(value: MiniMapPosition) {\n    this.minimapPosition.set(value)\n  }\n\n  /**\n   * The color outside the viewport (invisible area)\n   */\n  @Input()\n  public maskColor = `rgba(215, 215, 215, 0.6)`\n\n  /**\n   * The minimap stroke color\n   */\n  @Input()\n  public strokeColor = `rgb(200, 200, 200)`\n\n  /**\n   * Make a minimap bigger on hover\n   */\n  @Input()\n  public set scaleOnHover(value: boolean) {\n    this.scaleOnHoverSignal.set(value)\n  }\n\n  @ViewChild('minimap', { static: true })\n  private minimap!: TemplateRef<unknown>\n\n  private readonly minimapOffset = 10\n\n  private readonly minimapScale = computed(() => {\n    if (this.scaleOnHoverSignal()) {\n      return this.hovered() ? 0.4 : 0.2\n    }\n\n    return 0.2\n  })\n\n  protected viewportColor = computed(() => {\n    const bg = this.flowSettingsService.background()\n\n    if (bg.type === 'dots' || bg.type === 'solid') {\n      return bg.color ?? '#fff'\n    }\n\n    return '#fff'\n  })\n\n  protected hovered = signal(false)\n\n  protected minimapPoint = computed(() => {\n    switch (this.minimapPosition()) {\n      case 'top-left':\n        return { x: this.minimapOffset, y: this.minimapOffset }\n      case 'top-right':\n        return {\n          x: this.flowSettingsService.computedFlowWidth() - this.minimapWidth() - this.minimapOffset,\n          y: this.minimapOffset\n        }\n      case 'bottom-left':\n        return {\n          x: this.minimapOffset,\n          y: this.flowSettingsService.computedFlowHeight() - this.minimapHeight() - this.minimapOffset\n        }\n      case 'bottom-right':\n        return {\n          x: this.flowSettingsService.computedFlowWidth() - this.minimapWidth() - this.minimapOffset,\n          y: this.flowSettingsService.computedFlowHeight() - this.minimapHeight() - this.minimapOffset\n        }\n    }\n  })\n\n  protected minimapWidth = computed(() =>\n    this.flowSettingsService.computedFlowWidth() * this.minimapScale()\n  )\n  protected minimapHeight = computed(() =>\n    this.flowSettingsService.computedFlowHeight() * this.minimapScale()\n  )\n\n  protected viewportTransform = computed(() => {\n    const viewport = this.viewportService.readableViewport();\n    let scale = 1 / viewport.zoom\n\n    let x = -(viewport.x * this.minimapScale()) * scale\n    x /= this.minimapScale()\n\n    let y = -(viewport.y * this.minimapScale()) * scale\n    y /= this.minimapScale()\n\n    scale /= this.minimapScale()\n\n    return `translate(${x}, ${y}) scale(${scale})`;\n  });\n\n\n  protected boundsViewport = computed(() => {\n    const nodes = this.entitiesService.nodes()\n\n    return getViewportForBounds(\n      getNodesBounds(nodes),\n      this.flowSettingsService.computedFlowWidth(),\n      this.flowSettingsService.computedFlowHeight(),\n      -Infinity,\n      1.5,\n      0\n    )\n\n  })\n\n  protected minimapTransform = computed(() => {\n    const vport = this.boundsViewport()\n\n    const x = vport.x * this.minimapScale()\n    const y = vport.y * this.minimapScale()\n    const scale = vport.zoom * this.minimapScale()\n\n    return `translate(${x} ${y}) scale(${scale})`\n  })\n\n  private minimapPosition = signal<MiniMapPosition>('bottom-right')\n\n  private scaleOnHoverSignal = signal(false)\n\n  public ngOnInit(): void {\n    const model = new MinimapModel()\n    model.template.set(this.minimap)\n\n    this.entitiesService.minimap.set(model)\n  }\n\n  protected trackNodes(idx: number, { node }: NodeModel) {\n    return node\n  }\n}\n","<ng-template #minimap>\n  <svg:rect\n    [attr.x]=\"minimapPoint().x\"\n    [attr.y]=\"minimapPoint().y\"\n    [attr.width]=\"minimapWidth()\"\n    [attr.height]=\"minimapHeight()\"\n    [attr.stroke]=\"strokeColor\"\n    fill=\"none\"\n  />\n\n  <svg:svg\n    [attr.x]=\"minimapPoint().x\"\n    [attr.y]=\"minimapPoint().y\"\n    [attr.width]=\"minimapWidth()\"\n    [attr.height]=\"minimapHeight()\"\n    (mouseover)=\"hovered.set(true)\"\n    (mouseleave)=\"hovered.set(false)\"\n  >\n    <svg:rect\n      [attr.width]=\"minimapWidth()\"\n      [attr.height]=\"minimapHeight()\"\n      [attr.fill]=\"maskColor\"\n    />\n\n    <svg:g [attr.transform]=\"minimapTransform()\">\n      <svg:rect\n        [attr.fill]=\"viewportColor()\"\n        [attr.transform]=\"viewportTransform()\"\n        [attr.width]=\"minimapWidth()\"\n        [attr.height]=\"minimapHeight()\"\n      />\n\n      <ng-container\n        *ngFor=\"let model of entitiesService.nodes(); trackBy: trackNodes\"\n      >\n        <svg:foreignObject\n          *ngIf=\"model.node.type === 'default' || model.node.type === 'html-template' || model.isComponentType\"\n          [attr.transform]=\"model.pointTransform()\"\n          [attr.width]=\"model.size().width\"\n          [attr.height]=\"model.size().height\"\n        >\n          <default-node\n            [selected]=\"model.selected()\"\n            [style.width.px]=\"model.size().width\"\n            [style.height.px]=\"model.size().height\"\n            [style.max-width.px]=\"model.size().width\"\n            [style.max-height.px]=\"model.size().height\"\n          >\n            <div [outerHTML]=\"model.text()\"></div>\n          </default-node>\n        </svg:foreignObject>\n\n        <svg:rect\n          *ngIf=\"model.node.type === 'default-group' || model.node.type === 'template-group'\"\n          class=\"default-group-node\"\n          rx=\"5\"\n          ry=\"5\"\n          [attr.transform]=\"model.pointTransform()\"\n          [class.default-group-node_selected]=\"model.selected()\"\n          [attr.width]=\"model.size().width\"\n          [attr.height]=\"model.size().height\"\n          [style.stroke]=\"model.color()\"\n          [style.fill]=\"model.color()\"\n        />\n\n      </ng-container>\n    </svg:g>\n  </svg:svg>\n</ng-template>\n"]}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { Injectable, computed, inject } from '@angular/core';
|
|
2
2
|
import { FlowEntitiesService } from './flow-entities.service';
|
|
3
|
+
import { isGroupNode } from '../utils/is-group-node';
|
|
3
4
|
import * as i0 from "@angular/core";
|
|
4
5
|
export class NodeRenderingService {
|
|
5
6
|
constructor() {
|
|
@@ -8,6 +9,12 @@ export class NodeRenderingService {
|
|
|
8
9
|
return this.flowEntitiesService.nodes()
|
|
9
10
|
.sort((aNode, bNode) => aNode.renderOrder() - bNode.renderOrder());
|
|
10
11
|
});
|
|
12
|
+
this.groups = computed(() => {
|
|
13
|
+
return this.nodes().filter(n => isGroupNode(n));
|
|
14
|
+
});
|
|
15
|
+
this.nonGroups = computed(() => {
|
|
16
|
+
return this.nodes().filter(n => !isGroupNode(n));
|
|
17
|
+
});
|
|
11
18
|
this.maxOrder = computed(() => {
|
|
12
19
|
return Math.max(...this.flowEntitiesService.nodes().map((n) => n.renderOrder()));
|
|
13
20
|
});
|
|
@@ -28,4 +35,4 @@ export class NodeRenderingService {
|
|
|
28
35
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NodeRenderingService, decorators: [{
|
|
29
36
|
type: Injectable
|
|
30
37
|
}] });
|
|
31
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
38
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibm9kZS1yZW5kZXJpbmcuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9zZXJ2aWNlcy9ub2RlLXJlbmRlcmluZy5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsUUFBUSxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUM3RCxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUU5RCxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sd0JBQXdCLENBQUM7O0FBR3JELE1BQU0sT0FBTyxvQkFBb0I7SUFEakM7UUFFVSx3QkFBbUIsR0FBRyxNQUFNLENBQUMsbUJBQW1CLENBQUMsQ0FBQTtRQUV6QyxVQUFLLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRTtZQUNwQyxPQUFPLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxLQUFLLEVBQUU7aUJBQ3BDLElBQUksQ0FBQyxDQUFDLEtBQUssRUFBRSxLQUFLLEVBQUUsRUFBRSxDQUFDLEtBQUssQ0FBQyxXQUFXLEVBQUUsR0FBRyxLQUFLLENBQUMsV0FBVyxFQUFFLENBQUMsQ0FBQTtRQUN0RSxDQUFDLENBQUMsQ0FBQTtRQUVjLFdBQU0sR0FBRyxRQUFRLENBQUMsR0FBRyxFQUFFO1lBQ3JDLE9BQU8sSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFBO1FBQ2pELENBQUMsQ0FBQyxDQUFDO1FBRWEsY0FBUyxHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUU7WUFDeEMsT0FBTyxJQUFJLENBQUMsS0FBSyxFQUFFLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQTtRQUNsRCxDQUFDLENBQUMsQ0FBQztRQUVLLGFBQVEsR0FBRyxRQUFRLENBQUMsR0FBRyxFQUFFO1lBQy9CLE9BQU8sSUFBSSxDQUFDLEdBQUcsQ0FDYixHQUFHLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxLQUFLLEVBQUUsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxDQUNoRSxDQUFBO1FBQ0gsQ0FBQyxDQUFDLENBQUE7S0FZSDtJQVZRLFFBQVEsQ0FBQyxJQUFlO1FBQzdCLG1EQUFtRDtRQUNuRCxZQUFZO1FBQ1osSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLFFBQVEsRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFBO1FBQ3pDLGdCQUFnQjtRQUNoQixJQUFJLENBQUMsbUJBQW1CO2FBQ3JCLEtBQUssRUFBRTthQUNQLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxNQUFNLEVBQUUsS0FBSyxJQUFJLENBQUM7YUFDaEMsT0FBTyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFBO0lBQ25DLENBQUM7K0dBL0JVLG9CQUFvQjttSEFBcEIsb0JBQW9COzs0RkFBcEIsb0JBQW9CO2tCQURoQyxVQUFVIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSwgY29tcHV0ZWQsIGluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRmxvd0VudGl0aWVzU2VydmljZSB9IGZyb20gJy4vZmxvdy1lbnRpdGllcy5zZXJ2aWNlJztcbmltcG9ydCB7IE5vZGVNb2RlbCB9IGZyb20gJy4uL21vZGVscy9ub2RlLm1vZGVsJztcbmltcG9ydCB7IGlzR3JvdXBOb2RlIH0gZnJvbSAnLi4vdXRpbHMvaXMtZ3JvdXAtbm9kZSc7XG5cbkBJbmplY3RhYmxlKClcbmV4cG9ydCBjbGFzcyBOb2RlUmVuZGVyaW5nU2VydmljZSB7XG4gIHByaXZhdGUgZmxvd0VudGl0aWVzU2VydmljZSA9IGluamVjdChGbG93RW50aXRpZXNTZXJ2aWNlKVxuXG4gIHB1YmxpYyByZWFkb25seSBub2RlcyA9IGNvbXB1dGVkKCgpID0+IHtcbiAgICByZXR1cm4gdGhpcy5mbG93RW50aXRpZXNTZXJ2aWNlLm5vZGVzKClcbiAgICAgIC5zb3J0KChhTm9kZSwgYk5vZGUpID0+IGFOb2RlLnJlbmRlck9yZGVyKCkgLSBiTm9kZS5yZW5kZXJPcmRlcigpKVxuICB9KVxuXG4gIHB1YmxpYyByZWFkb25seSBncm91cHMgPSBjb21wdXRlZCgoKSA9PiB7XG4gICAgcmV0dXJuIHRoaXMubm9kZXMoKS5maWx0ZXIobiA9PiBpc0dyb3VwTm9kZShuKSlcbiAgfSk7XG5cbiAgcHVibGljIHJlYWRvbmx5IG5vbkdyb3VwcyA9IGNvbXB1dGVkKCgpID0+IHtcbiAgICByZXR1cm4gdGhpcy5ub2RlcygpLmZpbHRlcihuID0+ICFpc0dyb3VwTm9kZShuKSlcbiAgfSk7XG5cbiAgcHJpdmF0ZSBtYXhPcmRlciA9IGNvbXB1dGVkKCgpID0+IHtcbiAgICByZXR1cm4gTWF0aC5tYXgoXG4gICAgICAuLi50aGlzLmZsb3dFbnRpdGllc1NlcnZpY2Uubm9kZXMoKS5tYXAoKG4pID0+IG4ucmVuZGVyT3JkZXIoKSlcbiAgICApXG4gIH0pXG5cbiAgcHVibGljIHB1bGxOb2RlKG5vZGU6IE5vZGVNb2RlbCkge1xuICAgIC8vIFRPRE8gZG8gbm90IHB1bGwgd2hlbiB0aGUgbm9kZSBpcyBhbHJlYWR5IG9uIHRvcFxuICAgIC8vIHB1bGwgbm9kZVxuICAgIG5vZGUucmVuZGVyT3JkZXIuc2V0KHRoaXMubWF4T3JkZXIoKSArIDEpXG4gICAgLy8gcHVsbCBjaGlsZHJlblxuICAgIHRoaXMuZmxvd0VudGl0aWVzU2VydmljZVxuICAgICAgLm5vZGVzKClcbiAgICAgIC5maWx0ZXIobiA9PiBuLnBhcmVudCgpID09PSBub2RlKVxuICAgICAgLmZvckVhY2gobiA9PiB0aGlzLnB1bGxOb2RlKG4pKVxuICB9XG59XG4iXX0=
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFja2dyb3VuZC50eXBlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LXZmbG93LWxpYi9zcmMvbGliL3ZmbG93L3R5cGVzL2JhY2tncm91bmQudHlwZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgQmFja2dyb3VuZCA9IENvbG9yQmFja2dyb3VuZCB8IERvdHNCYWNrZ3JvdW5kIHwgSW1hZ2VCYWNrZ3JvdW5kXG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29sb3JCYWNrZ3JvdW5kIHtcbiAgdHlwZTogJ3NvbGlkJ1xuICBjb2xvcjogc3RyaW5nXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRG90c0JhY2tncm91bmQge1xuICB0eXBlOiAnZG90cydcbiAgLyoqXG4gICAqIEdhcCBiZXR3ZWVuIGRvdHNcbiAgICovXG4gIGdhcD86IG51bWJlclxuXG4gIC8qKlxuICAgKiBDb2xvciBvZiB0aGUgZG90XG4gICAqL1xuICBjb2xvcj86IHN0cmluZ1xuXG4gIC8qKlxuICAgKiBEaWFtZXRlciBvZiB0aGUgZG90XG4gICAqL1xuICBzaXplPzogbnVtYmVyXG5cbiAgLyoqXG4gICAqIENvbG9yIGJlaGluZCB0aGEgZG90IHBhdHRlcm5cbiAgICovXG4gIGJhY2tncm91bmRDb2xvcj86IHN0cmluZ1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEltYWdlQmFja2dyb3VuZCB7XG4gIHR5cGU6ICdpbWFnZSdcbiAgLyoqXG4gICAqIFBhdGggdG8gaW1hZ2UuIFBhc3NlcyBhcyBhIGhyZWYgdG8gdGhpcyBlbGVtZW50OlxuICAgKlxuICAgKiBodHRwczovL2RldmVsb3Blci5tb3ppbGxhLm9yZy9lbi1VUy9kb2NzL1dlYi9TVkcvRWxlbWVudC9pbWFnZVxuICAgKi9cbiAgc3JjOiBzdHJpbmdcblxuICAvKipcbiAgICogSWYgdHJ1ZSwgaW1hZ2Ugd2lsbCBiZSBmaXhlZCBvbiB0aGUgYmFja2dyb3VuZCAobm90IG1vdmluZyB3aXRoIHRoZSBmbG93LCB6b29tIGFsc28gZG9lc24ndCBhZmZlY3QgaXQpXG4gICAqL1xuICBmaXhlZD86IGJvb2xlYW5cblxuICAvKipcbiAgICogSWYgdHJ1ZSwgaW1hZ2Ugd2lsbCBiZSByZXBlYXRlZFxuICAgKi9cbiAgcmVwZWF0PzogYm9vbGVhblxuXG4gIC8qKlxuICAgKiBTY2FsZSByZWxhdGl2ZSB0byBuYXR1cmFsIGltYWdlIHNpemUuXG4gICAqXG4gICAqIEZvciBleGFtcGxlLCBpZiBpbWFnZSBpcyAxMDB4MTAwIGFuZCBzY2FsZSBpcyAwLjUsIGl0IHdpbGwgYmUgcmVuZGVyZWQgYXMgNTB4NTBcbiAgICovXG4gIHNjYWxlPzogbnVtYmVyXG59XG4iXX0=
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
export function isGroupNode(node) {
|
|
2
|
+
return node.node.type === 'default-group' || node.node.type === 'template-group';
|
|
3
|
+
}
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaXMtZ3JvdXAtbm9kZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy91dGlscy9pcy1ncm91cC1ub2RlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBLE1BQU0sVUFBVSxXQUFXLENBQUMsSUFBZTtJQUN6QyxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxLQUFLLGVBQWUsSUFBSSxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksS0FBSyxnQkFBZ0IsQ0FBQztBQUVuRixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTm9kZU1vZGVsIH0gZnJvbSBcIi4uL21vZGVscy9ub2RlLm1vZGVsXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBpc0dyb3VwTm9kZShub2RlOiBOb2RlTW9kZWwpOiBib29sZWFuIHtcbiAgcmV0dXJuIG5vZGUubm9kZS50eXBlID09PSAnZGVmYXVsdC1ncm91cCcgfHwgbm9kZS5ub2RlLnR5cGUgPT09ICd0ZW1wbGF0ZS1ncm91cCc7XG5cbn1cbiJdfQ==
|