ngx-vflow 1.14.0 → 1.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.
Files changed (44) hide show
  1. package/esm2022/lib/vflow/components/alignment-helper/alignment-helper.component.mjs +103 -0
  2. package/esm2022/lib/vflow/components/node/node.component.mjs +5 -4
  3. package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +8 -5
  4. package/esm2022/lib/vflow/interfaces/alignment-helper-settings.interface.mjs +2 -0
  5. package/esm2022/lib/vflow/interfaces/flow-entity.interface.mjs +1 -1
  6. package/esm2022/lib/vflow/interfaces/node.interface.mjs +13 -5
  7. package/esm2022/lib/vflow/interfaces/optimization.interface.mjs +2 -1
  8. package/esm2022/lib/vflow/interfaces/rect.mjs +10 -2
  9. package/esm2022/lib/vflow/interfaces/template-context.interface.mjs +1 -1
  10. package/esm2022/lib/vflow/models/edge.model.mjs +4 -9
  11. package/esm2022/lib/vflow/models/node.model.mjs +51 -7
  12. package/esm2022/lib/vflow/services/draggable.service.mjs +7 -1
  13. package/esm2022/lib/vflow/services/edge-rendering.service.mjs +1 -1
  14. package/esm2022/lib/vflow/services/flow-rendering.service.mjs +9 -1
  15. package/esm2022/lib/vflow/services/flow-status.service.mjs +13 -1
  16. package/esm2022/lib/vflow/utils/is-callable.mjs +10 -0
  17. package/esm2022/lib/vflow/utils/is-vflow-component.mjs +9 -0
  18. package/esm2022/public-api.mjs +3 -1
  19. package/esm2022/testing/component-mocks/vflow-mock.component.mjs +4 -3
  20. package/esm2022/testing/provide-custom-node-mocks.mjs +3 -2
  21. package/fesm2022/ngx-vflow-testing.mjs +5 -3
  22. package/fesm2022/ngx-vflow-testing.mjs.map +1 -1
  23. package/fesm2022/ngx-vflow.mjs +387 -201
  24. package/fesm2022/ngx-vflow.mjs.map +1 -1
  25. package/lib/vflow/components/alignment-helper/alignment-helper.component.d.ts +21 -0
  26. package/lib/vflow/components/vflow/vflow.component.d.ts +3 -1
  27. package/lib/vflow/interfaces/alignment-helper-settings.interface.d.ts +4 -0
  28. package/lib/vflow/interfaces/flow-entity.interface.d.ts +2 -1
  29. package/lib/vflow/interfaces/node.interface.d.ts +2 -2
  30. package/lib/vflow/interfaces/optimization.interface.d.ts +4 -0
  31. package/lib/vflow/interfaces/rect.d.ts +7 -0
  32. package/lib/vflow/interfaces/template-context.interface.d.ts +4 -0
  33. package/lib/vflow/models/edge.model.d.ts +2 -0
  34. package/lib/vflow/models/node.model.d.ts +4 -0
  35. package/lib/vflow/public-components/custom-template-edge/custom-template-edge.component.d.ts +1 -0
  36. package/lib/vflow/services/draggable.service.d.ts +1 -0
  37. package/lib/vflow/services/edge-rendering.service.d.ts +1 -1
  38. package/lib/vflow/services/flow-rendering.service.d.ts +4 -0
  39. package/lib/vflow/services/flow-status.service.d.ts +17 -1
  40. package/lib/vflow/utils/is-callable.d.ts +1 -0
  41. package/lib/vflow/utils/is-vflow-component.d.ts +4 -0
  42. package/package.json +1 -1
  43. package/public-api.d.ts +2 -0
  44. package/testing/component-mocks/vflow-mock.component.d.ts +3 -2
@@ -0,0 +1,103 @@
1
+ import { ChangeDetectionStrategy, Component, computed, inject, input } from '@angular/core';
2
+ import { nodeToRect } from '../../utils/nodes';
3
+ import { FlowStatusService, isNodeDragEndStatus, isNodeDragStartStatus } from '../../services/flow-status.service';
4
+ import { rectToRectWithSides } from '../../interfaces/rect';
5
+ import { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';
6
+ import { filter, map, tap } from 'rxjs';
7
+ import { extendedComputed } from '../../utils/signals/extended-computed';
8
+ import { NodeRenderingService } from '../../services/node-rendering.service';
9
+ import { getSpacePoints } from '../../utils/get-space-points';
10
+ import * as i0 from "@angular/core";
11
+ export class AlignmentHelperComponent {
12
+ constructor() {
13
+ this.nodeRenderingService = inject(NodeRenderingService);
14
+ this.flowStatus = inject(FlowStatusService);
15
+ this.tolerance = input(10);
16
+ this.lineColor = input('#1b262c');
17
+ this.isNodeDragging = computed(() => isNodeDragStartStatus(this.flowStatus.status()));
18
+ this.intersections = extendedComputed((lastValue) => {
19
+ const status = this.flowStatus.status();
20
+ if (isNodeDragStartStatus(status)) {
21
+ const node = status.payload.node;
22
+ const d = rectToRectWithSides(nodeToRect(node));
23
+ const otherRects = this.nodeRenderingService
24
+ .viewportNodes()
25
+ .filter((n) => n !== node)
26
+ // do not check children of the dragged node
27
+ .filter((n) => !node.children().includes(n))
28
+ .map((n) => rectToRectWithSides(nodeToRect(n)));
29
+ const lines = [];
30
+ let snappedX = d.x;
31
+ let snappedY = d.y;
32
+ let closestXDiff = Infinity;
33
+ let closestYDiff = Infinity;
34
+ otherRects.forEach((o) => {
35
+ const dCenterX = d.left + d.width / 2;
36
+ const oCenterX = o.left + o.width / 2;
37
+ for (const [dX, oX, snapX, isCenter] of [
38
+ // center check
39
+ [dCenterX, oCenterX, oCenterX - d.width / 2, true],
40
+ [d.left, o.left, o.left, false],
41
+ [d.left, o.right, o.right, false],
42
+ [d.right, o.left, o.left - d.width, false],
43
+ [d.right, o.right, o.right - d.width, false],
44
+ ]) {
45
+ const diff = Math.abs(dX - oX);
46
+ if (diff <= this.tolerance()) {
47
+ const y = Math.min(d.top, o.top);
48
+ const y2 = Math.max(d.bottom, o.bottom);
49
+ lines.push({ x: oX, y, x2: oX, y2, isCenter });
50
+ if (diff < closestXDiff) {
51
+ closestXDiff = diff;
52
+ snappedX = snapX;
53
+ }
54
+ if (isCenter)
55
+ break;
56
+ }
57
+ }
58
+ const dCenterY = d.top + d.height / 2;
59
+ const oCenterY = o.top + o.height / 2;
60
+ for (const [dY, oY, snapY, isCenter] of [
61
+ // center check
62
+ [dCenterY, oCenterY, oCenterY - d.height / 2, true],
63
+ [d.top, o.top, o.top, false],
64
+ [d.top, o.bottom, o.bottom, false],
65
+ [d.bottom, o.top, o.top - d.height, false],
66
+ [d.bottom, o.bottom, o.bottom - d.height, false],
67
+ ]) {
68
+ const diff = Math.abs(dY - oY);
69
+ if (diff <= this.tolerance()) {
70
+ const x = Math.min(d.left, o.left);
71
+ const x2 = Math.max(d.right, o.right);
72
+ lines.push({ x, y: oY, x2, y2: oY, isCenter });
73
+ if (diff < closestYDiff) {
74
+ closestYDiff = diff;
75
+ snappedY = snapY;
76
+ }
77
+ if (isCenter)
78
+ break;
79
+ }
80
+ }
81
+ });
82
+ return { lines, snappedX, snappedY };
83
+ }
84
+ return lastValue;
85
+ });
86
+ toObservable(this.flowStatus.status)
87
+ .pipe(filter(isNodeDragEndStatus), map((status) => status.payload.node), map((node) => [node, this.intersections()]), tap(([node, intersections]) => {
88
+ if (intersections) {
89
+ const snapped = { x: intersections.snappedX, y: intersections.snappedY };
90
+ const parentIfExists = node.parent() ? [node.parent()] : [];
91
+ node.setPoint(getSpacePoints(snapped, parentIfExists)[0]);
92
+ }
93
+ }), takeUntilDestroyed())
94
+ .subscribe();
95
+ }
96
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AlignmentHelperComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
97
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: AlignmentHelperComponent, isStandalone: true, selector: "g[alignmentHelper]", inputs: { tolerance: { classPropertyName: "tolerance", publicName: "tolerance", isSignal: true, isRequired: false, transformFunction: null }, lineColor: { classPropertyName: "lineColor", publicName: "lineColor", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "@if (isNodeDragging()) {\n @if (intersections(); as intersections) {\n @for (intersection of intersections.lines; track $index) {\n <svg:line\n [attr.stroke]=\"lineColor()\"\n [attr.stroke-dasharray]=\"intersection.isCenter ? 4 : null\"\n [attr.x1]=\"intersection.x\"\n [attr.y1]=\"intersection.y\"\n [attr.x2]=\"intersection.x2\"\n [attr.y2]=\"intersection.y2\" />\n }\n }\n}\n", changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
98
+ }
99
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AlignmentHelperComponent, decorators: [{
100
+ type: Component,
101
+ args: [{ selector: 'g[alignmentHelper]', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, template: "@if (isNodeDragging()) {\n @if (intersections(); as intersections) {\n @for (intersection of intersections.lines; track $index) {\n <svg:line\n [attr.stroke]=\"lineColor()\"\n [attr.stroke-dasharray]=\"intersection.isCenter ? 4 : null\"\n [attr.x1]=\"intersection.x\"\n [attr.y1]=\"intersection.y\"\n [attr.x2]=\"intersection.x2\"\n [attr.y2]=\"intersection.y2\" />\n }\n }\n}\n" }]
102
+ }], ctorParameters: () => [] });
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alignment-helper.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/alignment-helper/alignment-helper.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/alignment-helper/alignment-helper.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, inject, input } from '@angular/core';\nimport { nodeToRect } from '../../utils/nodes';\nimport { FlowStatusService, isNodeDragEndStatus, isNodeDragStartStatus } from '../../services/flow-status.service';\nimport { rectToRectWithSides } from '../../interfaces/rect';\nimport { Box } from '../../interfaces/box';\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';\nimport { filter, map, tap } from 'rxjs';\nimport { extendedComputed } from '../../utils/signals/extended-computed';\nimport { NodeRenderingService } from '../../services/node-rendering.service';\nimport { getSpacePoints } from '../../utils/get-space-points';\n\ninterface Intersection {\n  lines: (Box & { isCenter: boolean })[];\n  snappedX: number;\n  snappedY: number;\n}\n\n@Component({\n  selector: 'g[alignmentHelper]',\n  templateUrl: './alignment-helper.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n})\nexport class AlignmentHelperComponent {\n  private nodeRenderingService = inject(NodeRenderingService);\n  private flowStatus = inject(FlowStatusService);\n\n  readonly tolerance = input(10);\n  readonly lineColor = input('#1b262c');\n\n  protected isNodeDragging = computed(() => isNodeDragStartStatus(this.flowStatus.status()));\n\n  protected readonly intersections = extendedComputed<Intersection>((lastValue) => {\n    const status = this.flowStatus.status();\n\n    if (isNodeDragStartStatus(status)) {\n      const node = status.payload.node;\n\n      const d = rectToRectWithSides(nodeToRect(node));\n      const otherRects = this.nodeRenderingService\n        .viewportNodes()\n        .filter((n) => n !== node)\n        // do not check children of the dragged node\n        .filter((n) => !node.children().includes(n))\n        .map((n) => rectToRectWithSides(nodeToRect(n)));\n\n      const lines: Intersection['lines'] = [];\n\n      let snappedX = d.x;\n      let snappedY = d.y;\n      let closestXDiff = Infinity;\n      let closestYDiff = Infinity;\n\n      otherRects.forEach((o) => {\n        const dCenterX = d.left + d.width / 2;\n        const oCenterX = o.left + o.width / 2;\n\n        for (const [dX, oX, snapX, isCenter] of [\n          // center check\n          [dCenterX, oCenterX, oCenterX - d.width / 2, true] as const,\n          [d.left, o.left, o.left, false] as const,\n          [d.left, o.right, o.right, false] as const,\n          [d.right, o.left, o.left - d.width, false] as const,\n          [d.right, o.right, o.right - d.width, false] as const,\n        ]) {\n          const diff = Math.abs(dX - oX);\n\n          if (diff <= this.tolerance()) {\n            const y = Math.min(d.top, o.top);\n            const y2 = Math.max(d.bottom, o.bottom);\n\n            lines.push({ x: oX, y, x2: oX, y2, isCenter });\n\n            if (diff < closestXDiff) {\n              closestXDiff = diff;\n              snappedX = snapX;\n            }\n\n            if (isCenter) break;\n          }\n        }\n\n        const dCenterY = d.top + d.height / 2;\n        const oCenterY = o.top + o.height / 2;\n\n        for (const [dY, oY, snapY, isCenter] of [\n          // center check\n          [dCenterY, oCenterY, oCenterY - d.height / 2, true] as const,\n          [d.top, o.top, o.top, false] as const,\n          [d.top, o.bottom, o.bottom, false] as const,\n          [d.bottom, o.top, o.top - d.height, false] as const,\n          [d.bottom, o.bottom, o.bottom - d.height, false] as const,\n        ]) {\n          const diff = Math.abs(dY - oY);\n\n          if (diff <= this.tolerance()) {\n            const x = Math.min(d.left, o.left);\n            const x2 = Math.max(d.right, o.right);\n\n            lines.push({ x, y: oY, x2, y2: oY, isCenter });\n\n            if (diff < closestYDiff) {\n              closestYDiff = diff;\n              snappedY = snapY;\n            }\n\n            if (isCenter) break;\n          }\n        }\n      });\n\n      return { lines, snappedX, snappedY };\n    }\n\n    return lastValue;\n  });\n\n  constructor() {\n    toObservable(this.flowStatus.status)\n      .pipe(\n        filter(isNodeDragEndStatus),\n        map((status) => status.payload.node),\n        map((node) => [node, this.intersections()] as const),\n        tap(([node, intersections]) => {\n          if (intersections) {\n            const snapped = { x: intersections.snappedX, y: intersections.snappedY };\n            const parentIfExists = node.parent() ? [node.parent()!] : [];\n\n            node.setPoint(getSpacePoints(snapped, parentIfExists)[0]);\n          }\n        }),\n        takeUntilDestroyed(),\n      )\n      .subscribe();\n  }\n}\n","@if (isNodeDragging()) {\n  @if (intersections(); as intersections) {\n    @for (intersection of intersections.lines; track $index) {\n      <svg:line\n        [attr.stroke]=\"lineColor()\"\n        [attr.stroke-dasharray]=\"intersection.isCenter ? 4 : null\"\n        [attr.x1]=\"intersection.x\"\n        [attr.y1]=\"intersection.y\"\n        [attr.x2]=\"intersection.x2\"\n        [attr.y2]=\"intersection.y2\" />\n    }\n  }\n}\n"]}
@@ -9,7 +9,7 @@ import { ConnectionControllerDirective } from '../../directives/connection-contr
9
9
  import { NodeAccessorService } from '../../services/node-accessor.service';
10
10
  import { OverlaysService } from '../../services/overlays.service';
11
11
  import { HandleSizeControllerDirective } from '../../directives/handle-size-controller.directive';
12
- import { NgTemplateOutlet, NgComponentOutlet } from '@angular/common';
12
+ import { NgTemplateOutlet, NgComponentOutlet, AsyncPipe } from '@angular/common';
13
13
  import { DefaultNodeComponent } from '../default-node/default-node.component';
14
14
  import { PointerDirective } from '../../directives/pointer.directive';
15
15
  // TODO: fix loading of these by @defer (should work in Angular 18+)
@@ -85,7 +85,7 @@ export class NodeComponent {
85
85
  }
86
86
  }
87
87
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NodeComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
88
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: NodeComponent, isStandalone: true, selector: "g[node]", inputs: { model: { classPropertyName: "model", publicName: "model", isSignal: true, isRequired: true, transformFunction: null }, nodeTemplate: { classPropertyName: "nodeTemplate", publicName: "nodeTemplate", isSignal: true, isRequired: false, transformFunction: null }, nodeSvgTemplate: { classPropertyName: "nodeSvgTemplate", publicName: "nodeSvgTemplate", isSignal: true, isRequired: false, transformFunction: null }, groupNodeTemplate: { classPropertyName: "groupNodeTemplate", publicName: "groupNodeTemplate", isSignal: true, isRequired: false, transformFunction: null } }, host: { classAttribute: "vflow-node" }, providers: [HandleService, NodeAccessorService], ngImport: i0, template: "<!-- Default node -->\n@if (model().rawNode.type === 'default') {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode(); selectNode()\">\n <default-node\n nodeHandlesController\n [selected]=\"model().selected()\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\"\n [style.max-width]=\"model().styleWidth()\"\n [style.max-height]=\"model().styleHeight()\">\n <div [outerHTML]=\"model().text()\"></div>\n\n <handle type=\"source\" position=\"right\" />\n <handle type=\"target\" position=\"left\" />\n </default-node>\n </svg:foreignObject>\n}\n\n<!-- HTML Template node -->\n@if (model().rawNode.type === 'html-template' && nodeTemplate()) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode()\">\n <div\n nodeHandlesController\n nodeResizeController\n class=\"wrapper\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\">\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </div>\n </svg:foreignObject>\n}\n\n<!-- SVG Template node -->\n@if (model().rawNode.type === 'svg-template' && nodeSvgTemplate()) {\n <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n <ng-container\n [ngTemplateOutlet]=\"nodeSvgTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </svg:g>\n}\n\n<!-- Component node -->\n@if (model().isComponentType) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode()\">\n <div\n nodeHandlesController\n nodeResizeController\n class=\"wrapper\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\">\n <ng-container\n [ngComponentOutlet]=\"$any(model().rawNode.type)\"\n [ngComponentOutletInputs]=\"model().componentTypeInputs\"\n [ngComponentOutletInjector]=\"injector\" />\n </div>\n </svg:foreignObject>\n}\n\n<!-- Default group node -->\n@if (model().rawNode.type === 'default-group') {\n <svg:rect\n class=\"default-group-node\"\n rx=\"5\"\n ry=\"5\"\n [resizable]=\"model().resizable()\"\n [gap]=\"3\"\n [resizerColor]=\"model().color()\"\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 (click)=\"pullNode(); selectNode()\" />\n}\n\n<!-- Template group node -->\n@if (model().rawNode.type === 'template-group' && groupNodeTemplate()) {\n <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n <ng-container\n [ngTemplateOutlet]=\"groupNodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </svg:g>\n}\n\n<!-- Resizer -->\n@if (model().resizerTemplate(); as template) {\n @if (model().resizable()) {\n <ng-template [ngTemplateOutlet]=\"template\" />\n }\n}\n\n<!-- Handles -->\n@for (handle of model().handles(); track handle) {\n @if (handle.template === undefined) {\n <svg:circle\n class=\"default-handle\"\n r=\"5\"\n [attr.cx]=\"handle.hostOffset().x\"\n [attr.cy]=\"handle.hostOffset().y\"\n [attr.stroke-width]=\"handle.strokeWidth\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\" />\n }\n\n @if (handle.template === null) {\n <svg:g\n [handleSizeController]=\"handle\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\" />\n }\n\n @if (handle.template) {\n <svg:g\n [handleSizeController]=\"handle\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\">\n <ng-container *ngTemplateOutlet=\"handle.template; context: handle.templateContext\" />\n </svg:g>\n }\n\n @if (showMagnet()) {\n <svg:circle\n class=\"magnet\"\n [attr.r]=\"model().magnetRadius\"\n [attr.cx]=\"handle.hostOffset().x\"\n [attr.cy]=\"handle.hostOffset().y\"\n (pointerEnd)=\"endConnection(); resetValidateConnection(handle)\"\n (pointerOver)=\"validateConnection(handle)\"\n (pointerOut)=\"resetValidateConnection(handle)\" />\n }\n}\n\n<!-- Toolbar -->\n@for (toolbar of toolbars(); track toolbar) {\n <svg:foreignObject\n [attr.width]=\"toolbar.size().width\"\n [attr.height]=\"toolbar.size().height\"\n [attr.transform]=\"toolbar.transform()\">\n <ng-container [ngTemplateOutlet]=\"toolbar.template()\" />\n </svg:foreignObject>\n}\n", styles: [".magnet{opacity:0}.wrapper{display:table-cell}.default-group-node{stroke-width:1.5px;fill-opacity:.05}.default-group-node_selected{stroke-width:2px}.default-handle{stroke:#fff;fill:#1b262c}\n"], dependencies: [{ kind: "directive", type: PointerDirective, selector: "[pointerStart], [pointerEnd], [pointerOver], [pointerOut]", outputs: ["pointerOver", "pointerOut", "pointerStart", "pointerEnd"] }, { kind: "component", type: DefaultNodeComponent, selector: "default-node", inputs: ["selected"] }, { kind: "component", type: HandleComponent, selector: "handle", inputs: ["position", "type", "id", "template", "offsetX", "offsetY"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "component", type: ResizableComponent, selector: "[resizable]", inputs: ["resizable", "resizerColor", "gap"] }, { kind: "directive", type: HandleSizeControllerDirective, selector: "[handleSizeController]", inputs: ["handleSizeController"] }, { kind: "directive", type: NodeHandlesControllerDirective, selector: "[nodeHandlesController]" }, { kind: "directive", type: NodeResizeControllerDirective, selector: "[nodeResizeController]" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
88
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: NodeComponent, isStandalone: true, selector: "g[node]", inputs: { model: { classPropertyName: "model", publicName: "model", isSignal: true, isRequired: true, transformFunction: null }, nodeTemplate: { classPropertyName: "nodeTemplate", publicName: "nodeTemplate", isSignal: true, isRequired: false, transformFunction: null }, nodeSvgTemplate: { classPropertyName: "nodeSvgTemplate", publicName: "nodeSvgTemplate", isSignal: true, isRequired: false, transformFunction: null }, groupNodeTemplate: { classPropertyName: "groupNodeTemplate", publicName: "groupNodeTemplate", isSignal: true, isRequired: false, transformFunction: null } }, host: { classAttribute: "vflow-node" }, providers: [HandleService, NodeAccessorService], ngImport: i0, template: "<!-- Default node -->\n@if (model().rawNode.type === 'default') {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode(); selectNode()\">\n <default-node\n nodeHandlesController\n [selected]=\"model().selected()\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\"\n [style.max-width]=\"model().styleWidth()\"\n [style.max-height]=\"model().styleHeight()\">\n <div [outerHTML]=\"model().text()\"></div>\n\n <handle type=\"source\" position=\"right\" />\n <handle type=\"target\" position=\"left\" />\n </default-node>\n </svg:foreignObject>\n}\n\n<!-- HTML Template node -->\n@if (model().rawNode.type === 'html-template' && nodeTemplate()) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode()\">\n <div\n nodeHandlesController\n nodeResizeController\n class=\"wrapper\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\">\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </div>\n </svg:foreignObject>\n}\n\n<!-- SVG Template node -->\n@if (model().rawNode.type === 'svg-template' && nodeSvgTemplate()) {\n <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n <ng-container\n [ngTemplateOutlet]=\"nodeSvgTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </svg:g>\n}\n\n<!-- Component node -->\n@if (model().isComponentType) {\n @if (model().componentInstance$ | async; as component) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode()\">\n <div\n nodeHandlesController\n nodeResizeController\n class=\"wrapper\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\">\n <ng-container\n [ngComponentOutlet]=\"$any(component)\"\n [ngComponentOutletInputs]=\"model().componentTypeInputs\"\n [ngComponentOutletInjector]=\"injector\" />\n </div>\n </svg:foreignObject>\n }\n}\n\n<!-- Default group node -->\n@if (model().rawNode.type === 'default-group') {\n <svg:rect\n class=\"default-group-node\"\n rx=\"5\"\n ry=\"5\"\n [resizable]=\"model().resizable()\"\n [gap]=\"3\"\n [resizerColor]=\"model().color()\"\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 (click)=\"pullNode(); selectNode()\" />\n}\n\n<!-- Template group node -->\n@if (model().rawNode.type === 'template-group' && groupNodeTemplate()) {\n <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n <ng-container\n [ngTemplateOutlet]=\"groupNodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </svg:g>\n}\n\n<!-- Resizer -->\n@if (model().resizerTemplate(); as template) {\n @if (model().resizable()) {\n <ng-template [ngTemplateOutlet]=\"template\" />\n }\n}\n\n<!-- Handles -->\n@for (handle of model().handles(); track handle) {\n @if (handle.template === undefined) {\n <svg:circle\n class=\"default-handle\"\n r=\"5\"\n [attr.cx]=\"handle.hostOffset().x\"\n [attr.cy]=\"handle.hostOffset().y\"\n [attr.stroke-width]=\"handle.strokeWidth\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\" />\n }\n\n @if (handle.template === null) {\n <svg:g\n [handleSizeController]=\"handle\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\" />\n }\n\n @if (handle.template) {\n <svg:g\n [handleSizeController]=\"handle\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\">\n <ng-container *ngTemplateOutlet=\"handle.template; context: handle.templateContext\" />\n </svg:g>\n }\n\n @if (showMagnet()) {\n <svg:circle\n class=\"magnet\"\n [attr.r]=\"model().magnetRadius\"\n [attr.cx]=\"handle.hostOffset().x\"\n [attr.cy]=\"handle.hostOffset().y\"\n (pointerEnd)=\"endConnection(); resetValidateConnection(handle)\"\n (pointerOver)=\"validateConnection(handle)\"\n (pointerOut)=\"resetValidateConnection(handle)\" />\n }\n}\n\n<!-- Toolbar -->\n@for (toolbar of toolbars(); track toolbar) {\n <svg:foreignObject\n [attr.width]=\"toolbar.size().width\"\n [attr.height]=\"toolbar.size().height\"\n [attr.transform]=\"toolbar.transform()\">\n <ng-container [ngTemplateOutlet]=\"toolbar.template()\" />\n </svg:foreignObject>\n}\n", styles: [".magnet{opacity:0}.wrapper{display:table-cell}.default-group-node{stroke-width:1.5px;fill-opacity:.05}.default-group-node_selected{stroke-width:2px}.default-handle{stroke:#fff;fill:#1b262c}\n"], dependencies: [{ kind: "directive", type: PointerDirective, selector: "[pointerStart], [pointerEnd], [pointerOver], [pointerOut]", outputs: ["pointerOver", "pointerOut", "pointerStart", "pointerEnd"] }, { kind: "component", type: DefaultNodeComponent, selector: "default-node", inputs: ["selected"] }, { kind: "component", type: HandleComponent, selector: "handle", inputs: ["position", "type", "id", "template", "offsetX", "offsetY"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "component", type: ResizableComponent, selector: "[resizable]", inputs: ["resizable", "resizerColor", "gap"] }, { kind: "directive", type: HandleSizeControllerDirective, selector: "[handleSizeController]", inputs: ["handleSizeController"] }, { kind: "directive", type: NodeHandlesControllerDirective, selector: "[nodeHandlesController]" }, { kind: "directive", type: NodeResizeControllerDirective, selector: "[nodeResizeController]" }, { kind: "pipe", type: AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
89
89
  }
90
90
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NodeComponent, decorators: [{
91
91
  type: Component,
@@ -101,6 +101,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
101
101
  HandleSizeControllerDirective,
102
102
  NodeHandlesControllerDirective,
103
103
  NodeResizeControllerDirective,
104
- ], template: "<!-- Default node -->\n@if (model().rawNode.type === 'default') {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode(); selectNode()\">\n <default-node\n nodeHandlesController\n [selected]=\"model().selected()\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\"\n [style.max-width]=\"model().styleWidth()\"\n [style.max-height]=\"model().styleHeight()\">\n <div [outerHTML]=\"model().text()\"></div>\n\n <handle type=\"source\" position=\"right\" />\n <handle type=\"target\" position=\"left\" />\n </default-node>\n </svg:foreignObject>\n}\n\n<!-- HTML Template node -->\n@if (model().rawNode.type === 'html-template' && nodeTemplate()) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode()\">\n <div\n nodeHandlesController\n nodeResizeController\n class=\"wrapper\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\">\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </div>\n </svg:foreignObject>\n}\n\n<!-- SVG Template node -->\n@if (model().rawNode.type === 'svg-template' && nodeSvgTemplate()) {\n <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n <ng-container\n [ngTemplateOutlet]=\"nodeSvgTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </svg:g>\n}\n\n<!-- Component node -->\n@if (model().isComponentType) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode()\">\n <div\n nodeHandlesController\n nodeResizeController\n class=\"wrapper\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\">\n <ng-container\n [ngComponentOutlet]=\"$any(model().rawNode.type)\"\n [ngComponentOutletInputs]=\"model().componentTypeInputs\"\n [ngComponentOutletInjector]=\"injector\" />\n </div>\n </svg:foreignObject>\n}\n\n<!-- Default group node -->\n@if (model().rawNode.type === 'default-group') {\n <svg:rect\n class=\"default-group-node\"\n rx=\"5\"\n ry=\"5\"\n [resizable]=\"model().resizable()\"\n [gap]=\"3\"\n [resizerColor]=\"model().color()\"\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 (click)=\"pullNode(); selectNode()\" />\n}\n\n<!-- Template group node -->\n@if (model().rawNode.type === 'template-group' && groupNodeTemplate()) {\n <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n <ng-container\n [ngTemplateOutlet]=\"groupNodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </svg:g>\n}\n\n<!-- Resizer -->\n@if (model().resizerTemplate(); as template) {\n @if (model().resizable()) {\n <ng-template [ngTemplateOutlet]=\"template\" />\n }\n}\n\n<!-- Handles -->\n@for (handle of model().handles(); track handle) {\n @if (handle.template === undefined) {\n <svg:circle\n class=\"default-handle\"\n r=\"5\"\n [attr.cx]=\"handle.hostOffset().x\"\n [attr.cy]=\"handle.hostOffset().y\"\n [attr.stroke-width]=\"handle.strokeWidth\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\" />\n }\n\n @if (handle.template === null) {\n <svg:g\n [handleSizeController]=\"handle\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\" />\n }\n\n @if (handle.template) {\n <svg:g\n [handleSizeController]=\"handle\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\">\n <ng-container *ngTemplateOutlet=\"handle.template; context: handle.templateContext\" />\n </svg:g>\n }\n\n @if (showMagnet()) {\n <svg:circle\n class=\"magnet\"\n [attr.r]=\"model().magnetRadius\"\n [attr.cx]=\"handle.hostOffset().x\"\n [attr.cy]=\"handle.hostOffset().y\"\n (pointerEnd)=\"endConnection(); resetValidateConnection(handle)\"\n (pointerOver)=\"validateConnection(handle)\"\n (pointerOut)=\"resetValidateConnection(handle)\" />\n }\n}\n\n<!-- Toolbar -->\n@for (toolbar of toolbars(); track toolbar) {\n <svg:foreignObject\n [attr.width]=\"toolbar.size().width\"\n [attr.height]=\"toolbar.size().height\"\n [attr.transform]=\"toolbar.transform()\">\n <ng-container [ngTemplateOutlet]=\"toolbar.template()\" />\n </svg:foreignObject>\n}\n", styles: [".magnet{opacity:0}.wrapper{display:table-cell}.default-group-node{stroke-width:1.5px;fill-opacity:.05}.default-group-node_selected{stroke-width:2px}.default-handle{stroke:#fff;fill:#1b262c}\n"] }]
104
+ AsyncPipe,
105
+ ], template: "<!-- Default node -->\n@if (model().rawNode.type === 'default') {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode(); selectNode()\">\n <default-node\n nodeHandlesController\n [selected]=\"model().selected()\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\"\n [style.max-width]=\"model().styleWidth()\"\n [style.max-height]=\"model().styleHeight()\">\n <div [outerHTML]=\"model().text()\"></div>\n\n <handle type=\"source\" position=\"right\" />\n <handle type=\"target\" position=\"left\" />\n </default-node>\n </svg:foreignObject>\n}\n\n<!-- HTML Template node -->\n@if (model().rawNode.type === 'html-template' && nodeTemplate()) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode()\">\n <div\n nodeHandlesController\n nodeResizeController\n class=\"wrapper\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\">\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </div>\n </svg:foreignObject>\n}\n\n<!-- SVG Template node -->\n@if (model().rawNode.type === 'svg-template' && nodeSvgTemplate()) {\n <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n <ng-container\n [ngTemplateOutlet]=\"nodeSvgTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </svg:g>\n}\n\n<!-- Component node -->\n@if (model().isComponentType) {\n @if (model().componentInstance$ | async; as component) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"model().foWidth()\"\n [attr.height]=\"model().foHeight()\"\n (click)=\"pullNode()\">\n <div\n nodeHandlesController\n nodeResizeController\n class=\"wrapper\"\n [style.width]=\"model().styleWidth()\"\n [style.height]=\"model().styleHeight()\">\n <ng-container\n [ngComponentOutlet]=\"$any(component)\"\n [ngComponentOutletInputs]=\"model().componentTypeInputs\"\n [ngComponentOutletInjector]=\"injector\" />\n </div>\n </svg:foreignObject>\n }\n}\n\n<!-- Default group node -->\n@if (model().rawNode.type === 'default-group') {\n <svg:rect\n class=\"default-group-node\"\n rx=\"5\"\n ry=\"5\"\n [resizable]=\"model().resizable()\"\n [gap]=\"3\"\n [resizerColor]=\"model().color()\"\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 (click)=\"pullNode(); selectNode()\" />\n}\n\n<!-- Template group node -->\n@if (model().rawNode.type === 'template-group' && groupNodeTemplate()) {\n <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n <ng-container\n [ngTemplateOutlet]=\"groupNodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"model().context\"\n [ngTemplateOutletInjector]=\"injector\" />\n </svg:g>\n}\n\n<!-- Resizer -->\n@if (model().resizerTemplate(); as template) {\n @if (model().resizable()) {\n <ng-template [ngTemplateOutlet]=\"template\" />\n }\n}\n\n<!-- Handles -->\n@for (handle of model().handles(); track handle) {\n @if (handle.template === undefined) {\n <svg:circle\n class=\"default-handle\"\n r=\"5\"\n [attr.cx]=\"handle.hostOffset().x\"\n [attr.cy]=\"handle.hostOffset().y\"\n [attr.stroke-width]=\"handle.strokeWidth\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\" />\n }\n\n @if (handle.template === null) {\n <svg:g\n [handleSizeController]=\"handle\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\" />\n }\n\n @if (handle.template) {\n <svg:g\n [handleSizeController]=\"handle\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection()\">\n <ng-container *ngTemplateOutlet=\"handle.template; context: handle.templateContext\" />\n </svg:g>\n }\n\n @if (showMagnet()) {\n <svg:circle\n class=\"magnet\"\n [attr.r]=\"model().magnetRadius\"\n [attr.cx]=\"handle.hostOffset().x\"\n [attr.cy]=\"handle.hostOffset().y\"\n (pointerEnd)=\"endConnection(); resetValidateConnection(handle)\"\n (pointerOver)=\"validateConnection(handle)\"\n (pointerOut)=\"resetValidateConnection(handle)\" />\n }\n}\n\n<!-- Toolbar -->\n@for (toolbar of toolbars(); track toolbar) {\n <svg:foreignObject\n [attr.width]=\"toolbar.size().width\"\n [attr.height]=\"toolbar.size().height\"\n [attr.transform]=\"toolbar.transform()\">\n <ng-container [ngTemplateOutlet]=\"toolbar.template()\" />\n </svg:foreignObject>\n}\n", styles: [".magnet{opacity:0}.wrapper{display:table-cell}.default-group-node{stroke-width:1.5px;fill-opacity:.05}.default-group-node_selected{stroke-width:2px}.default-handle{stroke:#fff;fill:#1b262c}\n"] }]
105
106
  }] });
106
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/node/node.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/node/node.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  Injector,\n  OnDestroy,\n  OnInit,\n  TemplateRef,\n  computed,\n  effect,\n  inject,\n  input,\n} from '@angular/core';\nimport { DraggableService } from '../../services/draggable.service';\nimport { NodeModel } from '../../models/node.model';\nimport { FlowStatusService } from '../../services/flow-status.service';\nimport { HandleService } from '../../services/handle.service';\nimport { HandleModel } from '../../models/handle.model';\nimport { NodeRenderingService } from '../../services/node-rendering.service';\nimport { FlowSettingsService } from '../../services/flow-settings.service';\nimport { SelectionService } from '../../services/selection.service';\nimport { ConnectionControllerDirective } from '../../directives/connection-controller.directive';\nimport { NodeAccessorService } from '../../services/node-accessor.service';\nimport { OverlaysService } from '../../services/overlays.service';\nimport { HandleSizeControllerDirective } from '../../directives/handle-size-controller.directive';\nimport { NgTemplateOutlet, NgComponentOutlet } from '@angular/common';\nimport { DefaultNodeComponent } from '../default-node/default-node.component';\nimport { PointerDirective } from '../../directives/pointer.directive';\n\n// TODO: fix loading of these by @defer (should work in Angular 18+)\n// public components that uses in default node (loaded by defer)\nimport { ResizableComponent } from '../../public-components/resizable/resizable.component';\nimport { HandleComponent } from '../../public-components/handle/handle.component';\nimport { NodeHandlesControllerDirective } from '../../directives/node-handles-controller.directive';\nimport { NodeResizeControllerDirective } from '../../directives/node-resize-controller.directive';\n\nexport type HandleState = 'valid' | 'invalid' | 'idle';\n\n@Component({\n  standalone: true,\n  selector: 'g[node]',\n  templateUrl: './node.component.html',\n  styleUrls: ['./node.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [HandleService, NodeAccessorService],\n  host: {\n    class: 'vflow-node',\n  },\n  imports: [\n    PointerDirective,\n    DefaultNodeComponent,\n    HandleComponent,\n    NgTemplateOutlet,\n    NgComponentOutlet,\n    ResizableComponent,\n    HandleSizeControllerDirective,\n    NodeHandlesControllerDirective,\n    NodeResizeControllerDirective,\n  ],\n})\nexport class NodeComponent implements OnInit, OnDestroy {\n  protected injector = inject(Injector);\n  private handleService = inject(HandleService);\n  private draggableService = inject(DraggableService);\n  private flowStatusService = inject(FlowStatusService);\n  private nodeRenderingService = inject(NodeRenderingService);\n  private flowSettingsService = inject(FlowSettingsService);\n  private selectionService = inject(SelectionService);\n  private hostRef = inject<ElementRef<SVGElement>>(ElementRef);\n  private nodeAccessor = inject(NodeAccessorService);\n  private overlaysService = inject(OverlaysService);\n\n  // TODO remove dependency from this directive\n  private connectionController = inject(ConnectionControllerDirective, { optional: true });\n\n  public model = input.required<NodeModel>();\n\n  public nodeTemplate = input<TemplateRef<any>>();\n\n  public nodeSvgTemplate = input<TemplateRef<any>>();\n\n  public groupNodeTemplate = input<TemplateRef<any>>();\n\n  protected showMagnet = computed(\n    () =>\n      this.flowStatusService.status().state === 'connection-start' ||\n      this.flowStatusService.status().state === 'connection-validation' ||\n      this.flowStatusService.status().state === 'reconnection-start' ||\n      this.flowStatusService.status().state === 'reconnection-validation',\n  );\n\n  protected toolbars = computed(() => this.overlaysService.nodeToolbarsMap().get(this.model()));\n\n  public ngOnInit() {\n    this.model().isVisible.set(true);\n\n    this.nodeAccessor.model.set(this.model());\n    this.handleService.node.set(this.model());\n\n    effect(\n      () => {\n        if (this.model().draggable()) {\n          this.draggableService.enable(this.hostRef.nativeElement, this.model());\n        } else {\n          this.draggableService.disable(this.hostRef.nativeElement);\n        }\n      },\n      { injector: this.injector },\n    );\n  }\n\n  public ngOnDestroy(): void {\n    this.model().isVisible.set(false);\n\n    this.draggableService.destroy(this.hostRef.nativeElement);\n  }\n\n  protected startConnection(event: Event, handle: HandleModel) {\n    // ignore drag by stopping propagation\n    event.stopPropagation();\n\n    this.connectionController?.startConnection(handle);\n  }\n\n  protected validateConnection(handle: HandleModel) {\n    this.connectionController?.validateConnection(handle);\n  }\n\n  protected resetValidateConnection(targetHandle: HandleModel) {\n    this.connectionController?.resetValidateConnection(targetHandle);\n  }\n\n  protected endConnection() {\n    this.connectionController?.endConnection();\n  }\n\n  protected pullNode() {\n    if (this.flowSettingsService.elevateNodesOnSelect()) {\n      this.nodeRenderingService.pullNode(this.model());\n    }\n  }\n\n  protected selectNode() {\n    if (this.flowSettingsService.entitiesSelectable()) {\n      this.selectionService.select(this.model());\n    }\n  }\n}\n","<!-- Default node -->\n@if (model().rawNode.type === 'default') {\n  <svg:foreignObject\n    class=\"selectable\"\n    [attr.width]=\"model().foWidth()\"\n    [attr.height]=\"model().foHeight()\"\n    (click)=\"pullNode(); selectNode()\">\n    <default-node\n      nodeHandlesController\n      [selected]=\"model().selected()\"\n      [style.width]=\"model().styleWidth()\"\n      [style.height]=\"model().styleHeight()\"\n      [style.max-width]=\"model().styleWidth()\"\n      [style.max-height]=\"model().styleHeight()\">\n      <div [outerHTML]=\"model().text()\"></div>\n\n      <handle type=\"source\" position=\"right\" />\n      <handle type=\"target\" position=\"left\" />\n    </default-node>\n  </svg:foreignObject>\n}\n\n<!-- HTML Template node -->\n@if (model().rawNode.type === 'html-template' && nodeTemplate()) {\n  <svg:foreignObject\n    class=\"selectable\"\n    [attr.width]=\"model().foWidth()\"\n    [attr.height]=\"model().foHeight()\"\n    (click)=\"pullNode()\">\n    <div\n      nodeHandlesController\n      nodeResizeController\n      class=\"wrapper\"\n      [style.width]=\"model().styleWidth()\"\n      [style.height]=\"model().styleHeight()\">\n      <ng-container\n        [ngTemplateOutlet]=\"nodeTemplate() ?? null\"\n        [ngTemplateOutletContext]=\"model().context\"\n        [ngTemplateOutletInjector]=\"injector\" />\n    </div>\n  </svg:foreignObject>\n}\n\n<!-- SVG Template node -->\n@if (model().rawNode.type === 'svg-template' && nodeSvgTemplate()) {\n  <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n    <ng-container\n      [ngTemplateOutlet]=\"nodeSvgTemplate() ?? null\"\n      [ngTemplateOutletContext]=\"model().context\"\n      [ngTemplateOutletInjector]=\"injector\" />\n  </svg:g>\n}\n\n<!-- Component node -->\n@if (model().isComponentType) {\n  <svg:foreignObject\n    class=\"selectable\"\n    [attr.width]=\"model().foWidth()\"\n    [attr.height]=\"model().foHeight()\"\n    (click)=\"pullNode()\">\n    <div\n      nodeHandlesController\n      nodeResizeController\n      class=\"wrapper\"\n      [style.width]=\"model().styleWidth()\"\n      [style.height]=\"model().styleHeight()\">\n      <ng-container\n        [ngComponentOutlet]=\"$any(model().rawNode.type)\"\n        [ngComponentOutletInputs]=\"model().componentTypeInputs\"\n        [ngComponentOutletInjector]=\"injector\" />\n    </div>\n  </svg:foreignObject>\n}\n\n<!-- Default group node -->\n@if (model().rawNode.type === 'default-group') {\n  <svg:rect\n    class=\"default-group-node\"\n    rx=\"5\"\n    ry=\"5\"\n    [resizable]=\"model().resizable()\"\n    [gap]=\"3\"\n    [resizerColor]=\"model().color()\"\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    (click)=\"pullNode(); selectNode()\" />\n}\n\n<!-- Template group node  -->\n@if (model().rawNode.type === 'template-group' && groupNodeTemplate()) {\n  <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n    <ng-container\n      [ngTemplateOutlet]=\"groupNodeTemplate() ?? null\"\n      [ngTemplateOutletContext]=\"model().context\"\n      [ngTemplateOutletInjector]=\"injector\" />\n  </svg:g>\n}\n\n<!-- Resizer -->\n@if (model().resizerTemplate(); as template) {\n  @if (model().resizable()) {\n    <ng-template [ngTemplateOutlet]=\"template\" />\n  }\n}\n\n<!-- Handles -->\n@for (handle of model().handles(); track handle) {\n  @if (handle.template === undefined) {\n    <svg:circle\n      class=\"default-handle\"\n      r=\"5\"\n      [attr.cx]=\"handle.hostOffset().x\"\n      [attr.cy]=\"handle.hostOffset().y\"\n      [attr.stroke-width]=\"handle.strokeWidth\"\n      (pointerStart)=\"startConnection($event, handle)\"\n      (pointerEnd)=\"endConnection()\" />\n  }\n\n  @if (handle.template === null) {\n    <svg:g\n      [handleSizeController]=\"handle\"\n      (pointerStart)=\"startConnection($event, handle)\"\n      (pointerEnd)=\"endConnection()\" />\n  }\n\n  @if (handle.template) {\n    <svg:g\n      [handleSizeController]=\"handle\"\n      (pointerStart)=\"startConnection($event, handle)\"\n      (pointerEnd)=\"endConnection()\">\n      <ng-container *ngTemplateOutlet=\"handle.template; context: handle.templateContext\" />\n    </svg:g>\n  }\n\n  @if (showMagnet()) {\n    <svg:circle\n      class=\"magnet\"\n      [attr.r]=\"model().magnetRadius\"\n      [attr.cx]=\"handle.hostOffset().x\"\n      [attr.cy]=\"handle.hostOffset().y\"\n      (pointerEnd)=\"endConnection(); resetValidateConnection(handle)\"\n      (pointerOver)=\"validateConnection(handle)\"\n      (pointerOut)=\"resetValidateConnection(handle)\" />\n  }\n}\n\n<!-- Toolbar -->\n@for (toolbar of toolbars(); track toolbar) {\n  <svg:foreignObject\n    [attr.width]=\"toolbar.size().width\"\n    [attr.height]=\"toolbar.size().height\"\n    [attr.transform]=\"toolbar.transform()\">\n    <ng-container [ngTemplateOutlet]=\"toolbar.template()\" />\n  </svg:foreignObject>\n}\n"]}
107
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/node/node.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/node/node.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,uBAAuB,EACvB,SAAS,EACT,UAAU,EACV,QAAQ,EAIR,QAAQ,EACR,MAAM,EACN,MAAM,EACN,KAAK,GACN,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,gBAAgB,EAAE,MAAM,kCAAkC,CAAC;AAEpE,OAAO,EAAE,iBAAiB,EAAE,MAAM,oCAAoC,CAAC;AACvE,OAAO,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAE9D,OAAO,EAAE,oBAAoB,EAAE,MAAM,uCAAuC,CAAC;AAC7E,OAAO,EAAE,mBAAmB,EAAE,MAAM,sCAAsC,CAAC;AAC3E,OAAO,EAAE,gBAAgB,EAAE,MAAM,kCAAkC,CAAC;AACpE,OAAO,EAAE,6BAA6B,EAAE,MAAM,kDAAkD,CAAC;AACjG,OAAO,EAAE,mBAAmB,EAAE,MAAM,sCAAsC,CAAC;AAC3E,OAAO,EAAE,eAAe,EAAE,MAAM,iCAAiC,CAAC;AAClE,OAAO,EAAE,6BAA6B,EAAE,MAAM,mDAAmD,CAAC;AAClG,OAAO,EAAE,gBAAgB,EAAE,iBAAiB,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AACjF,OAAO,EAAE,oBAAoB,EAAE,MAAM,wCAAwC,CAAC;AAC9E,OAAO,EAAE,gBAAgB,EAAE,MAAM,oCAAoC,CAAC;AAEtE,oEAAoE;AACpE,gEAAgE;AAChE,OAAO,EAAE,kBAAkB,EAAE,MAAM,uDAAuD,CAAC;AAC3F,OAAO,EAAE,eAAe,EAAE,MAAM,iDAAiD,CAAC;AAClF,OAAO,EAAE,8BAA8B,EAAE,MAAM,oDAAoD,CAAC;AACpG,OAAO,EAAE,6BAA6B,EAAE,MAAM,mDAAmD,CAAC;;AA2BlG,MAAM,OAAO,aAAa;IAvB1B;QAwBY,aAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;QAC9B,kBAAa,GAAG,MAAM,CAAC,aAAa,CAAC,CAAC;QACtC,qBAAgB,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;QAC5C,sBAAiB,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAC9C,yBAAoB,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;QACpD,wBAAmB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAClD,qBAAgB,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;QAC5C,YAAO,GAAG,MAAM,CAAyB,UAAU,CAAC,CAAC;QACrD,iBAAY,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAC3C,oBAAe,GAAG,MAAM,CAAC,eAAe,CAAC,CAAC;QAElD,6CAA6C;QACrC,yBAAoB,GAAG,MAAM,CAAC,6BAA6B,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;QAElF,UAAK,GAAG,KAAK,CAAC,QAAQ,EAAa,CAAC;QAEpC,iBAAY,GAAG,KAAK,EAAoB,CAAC;QAEzC,oBAAe,GAAG,KAAK,EAAoB,CAAC;QAE5C,sBAAiB,GAAG,KAAK,EAAoB,CAAC;QAE3C,eAAU,GAAG,QAAQ,CAC7B,GAAG,EAAE,CACH,IAAI,CAAC,iBAAiB,CAAC,MAAM,EAAE,CAAC,KAAK,KAAK,kBAAkB;YAC5D,IAAI,CAAC,iBAAiB,CAAC,MAAM,EAAE,CAAC,KAAK,KAAK,uBAAuB;YACjE,IAAI,CAAC,iBAAiB,CAAC,MAAM,EAAE,CAAC,KAAK,KAAK,oBAAoB;YAC9D,IAAI,CAAC,iBAAiB,CAAC,MAAM,EAAE,CAAC,KAAK,KAAK,yBAAyB,CACtE,CAAC;QAEQ,aAAQ,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,eAAe,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;KAwD/F;IAtDQ,QAAQ;QACb,IAAI,CAAC,KAAK,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAEjC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QAC1C,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QAE1C,MAAM,CACJ,GAAG,EAAE;YACH,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC,SAAS,EAAE,EAAE,CAAC;gBAC7B,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;YACzE,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;YAC5D,CAAC;QACH,CAAC,EACD,EAAE,QAAQ,EAAE,IAAI,CAAC,QAAQ,EAAE,CAC5B,CAAC;IACJ,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,KAAK,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAElC,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;IAC5D,CAAC;IAES,eAAe,CAAC,KAAY,EAAE,MAAmB;QACzD,sCAAsC;QACtC,KAAK,CAAC,eAAe,EAAE,CAAC;QAExB,IAAI,CAAC,oBAAoB,EAAE,eAAe,CAAC,MAAM,CAAC,CAAC;IACrD,CAAC;IAES,kBAAkB,CAAC,MAAmB;QAC9C,IAAI,CAAC,oBAAoB,EAAE,kBAAkB,CAAC,MAAM,CAAC,CAAC;IACxD,CAAC;IAES,uBAAuB,CAAC,YAAyB;QACzD,IAAI,CAAC,oBAAoB,EAAE,uBAAuB,CAAC,YAAY,CAAC,CAAC;IACnE,CAAC;IAES,aAAa;QACrB,IAAI,CAAC,oBAAoB,EAAE,aAAa,EAAE,CAAC;IAC7C,CAAC;IAES,QAAQ;QAChB,IAAI,IAAI,CAAC,mBAAmB,CAAC,oBAAoB,EAAE,EAAE,CAAC;YACpD,IAAI,CAAC,oBAAoB,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QACnD,CAAC;IACH,CAAC;IAES,UAAU;QAClB,IAAI,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,EAAE,EAAE,CAAC;YAClD,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QAC7C,CAAC;IACH,CAAC;+GAtFU,aAAa;mGAAb,aAAa,gqBAjBb,CAAC,aAAa,EAAE,mBAAmB,CAAC,0BC5CjD,mkKAgKA,yPD/GI,gBAAgB,4KAChB,oBAAoB,+EACpB,eAAe,yHACf,gBAAgB,oJAChB,iBAAiB,oPACjB,kBAAkB,sGAClB,6BAA6B,qGAC7B,8BAA8B,oEAC9B,6BAA6B,8DAC7B,SAAS;;4FAGA,aAAa;kBAvBzB,SAAS;iCACI,IAAI,YACN,SAAS,mBAGF,uBAAuB,CAAC,MAAM,aACpC,CAAC,aAAa,EAAE,mBAAmB,CAAC,QACzC;wBACJ,KAAK,EAAE,YAAY;qBACpB,WACQ;wBACP,gBAAgB;wBAChB,oBAAoB;wBACpB,eAAe;wBACf,gBAAgB;wBAChB,iBAAiB;wBACjB,kBAAkB;wBAClB,6BAA6B;wBAC7B,8BAA8B;wBAC9B,6BAA6B;wBAC7B,SAAS;qBACV","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  Injector,\n  OnDestroy,\n  OnInit,\n  TemplateRef,\n  computed,\n  effect,\n  inject,\n  input,\n} from '@angular/core';\nimport { DraggableService } from '../../services/draggable.service';\nimport { NodeModel } from '../../models/node.model';\nimport { FlowStatusService } from '../../services/flow-status.service';\nimport { HandleService } from '../../services/handle.service';\nimport { HandleModel } from '../../models/handle.model';\nimport { NodeRenderingService } from '../../services/node-rendering.service';\nimport { FlowSettingsService } from '../../services/flow-settings.service';\nimport { SelectionService } from '../../services/selection.service';\nimport { ConnectionControllerDirective } from '../../directives/connection-controller.directive';\nimport { NodeAccessorService } from '../../services/node-accessor.service';\nimport { OverlaysService } from '../../services/overlays.service';\nimport { HandleSizeControllerDirective } from '../../directives/handle-size-controller.directive';\nimport { NgTemplateOutlet, NgComponentOutlet, AsyncPipe } from '@angular/common';\nimport { DefaultNodeComponent } from '../default-node/default-node.component';\nimport { PointerDirective } from '../../directives/pointer.directive';\n\n// TODO: fix loading of these by @defer (should work in Angular 18+)\n// public components that uses in default node (loaded by defer)\nimport { ResizableComponent } from '../../public-components/resizable/resizable.component';\nimport { HandleComponent } from '../../public-components/handle/handle.component';\nimport { NodeHandlesControllerDirective } from '../../directives/node-handles-controller.directive';\nimport { NodeResizeControllerDirective } from '../../directives/node-resize-controller.directive';\n\nexport type HandleState = 'valid' | 'invalid' | 'idle';\n\n@Component({\n  standalone: true,\n  selector: 'g[node]',\n  templateUrl: './node.component.html',\n  styleUrls: ['./node.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [HandleService, NodeAccessorService],\n  host: {\n    class: 'vflow-node',\n  },\n  imports: [\n    PointerDirective,\n    DefaultNodeComponent,\n    HandleComponent,\n    NgTemplateOutlet,\n    NgComponentOutlet,\n    ResizableComponent,\n    HandleSizeControllerDirective,\n    NodeHandlesControllerDirective,\n    NodeResizeControllerDirective,\n    AsyncPipe,\n  ],\n})\nexport class NodeComponent implements OnInit, OnDestroy {\n  protected injector = inject(Injector);\n  private handleService = inject(HandleService);\n  private draggableService = inject(DraggableService);\n  private flowStatusService = inject(FlowStatusService);\n  private nodeRenderingService = inject(NodeRenderingService);\n  private flowSettingsService = inject(FlowSettingsService);\n  private selectionService = inject(SelectionService);\n  private hostRef = inject<ElementRef<SVGElement>>(ElementRef);\n  private nodeAccessor = inject(NodeAccessorService);\n  private overlaysService = inject(OverlaysService);\n\n  // TODO remove dependency from this directive\n  private connectionController = inject(ConnectionControllerDirective, { optional: true });\n\n  public model = input.required<NodeModel>();\n\n  public nodeTemplate = input<TemplateRef<any>>();\n\n  public nodeSvgTemplate = input<TemplateRef<any>>();\n\n  public groupNodeTemplate = input<TemplateRef<any>>();\n\n  protected showMagnet = computed(\n    () =>\n      this.flowStatusService.status().state === 'connection-start' ||\n      this.flowStatusService.status().state === 'connection-validation' ||\n      this.flowStatusService.status().state === 'reconnection-start' ||\n      this.flowStatusService.status().state === 'reconnection-validation',\n  );\n\n  protected toolbars = computed(() => this.overlaysService.nodeToolbarsMap().get(this.model()));\n\n  public ngOnInit() {\n    this.model().isVisible.set(true);\n\n    this.nodeAccessor.model.set(this.model());\n    this.handleService.node.set(this.model());\n\n    effect(\n      () => {\n        if (this.model().draggable()) {\n          this.draggableService.enable(this.hostRef.nativeElement, this.model());\n        } else {\n          this.draggableService.disable(this.hostRef.nativeElement);\n        }\n      },\n      { injector: this.injector },\n    );\n  }\n\n  public ngOnDestroy(): void {\n    this.model().isVisible.set(false);\n\n    this.draggableService.destroy(this.hostRef.nativeElement);\n  }\n\n  protected startConnection(event: Event, handle: HandleModel) {\n    // ignore drag by stopping propagation\n    event.stopPropagation();\n\n    this.connectionController?.startConnection(handle);\n  }\n\n  protected validateConnection(handle: HandleModel) {\n    this.connectionController?.validateConnection(handle);\n  }\n\n  protected resetValidateConnection(targetHandle: HandleModel) {\n    this.connectionController?.resetValidateConnection(targetHandle);\n  }\n\n  protected endConnection() {\n    this.connectionController?.endConnection();\n  }\n\n  protected pullNode() {\n    if (this.flowSettingsService.elevateNodesOnSelect()) {\n      this.nodeRenderingService.pullNode(this.model());\n    }\n  }\n\n  protected selectNode() {\n    if (this.flowSettingsService.entitiesSelectable()) {\n      this.selectionService.select(this.model());\n    }\n  }\n}\n","<!-- Default node -->\n@if (model().rawNode.type === 'default') {\n  <svg:foreignObject\n    class=\"selectable\"\n    [attr.width]=\"model().foWidth()\"\n    [attr.height]=\"model().foHeight()\"\n    (click)=\"pullNode(); selectNode()\">\n    <default-node\n      nodeHandlesController\n      [selected]=\"model().selected()\"\n      [style.width]=\"model().styleWidth()\"\n      [style.height]=\"model().styleHeight()\"\n      [style.max-width]=\"model().styleWidth()\"\n      [style.max-height]=\"model().styleHeight()\">\n      <div [outerHTML]=\"model().text()\"></div>\n\n      <handle type=\"source\" position=\"right\" />\n      <handle type=\"target\" position=\"left\" />\n    </default-node>\n  </svg:foreignObject>\n}\n\n<!-- HTML Template node -->\n@if (model().rawNode.type === 'html-template' && nodeTemplate()) {\n  <svg:foreignObject\n    class=\"selectable\"\n    [attr.width]=\"model().foWidth()\"\n    [attr.height]=\"model().foHeight()\"\n    (click)=\"pullNode()\">\n    <div\n      nodeHandlesController\n      nodeResizeController\n      class=\"wrapper\"\n      [style.width]=\"model().styleWidth()\"\n      [style.height]=\"model().styleHeight()\">\n      <ng-container\n        [ngTemplateOutlet]=\"nodeTemplate() ?? null\"\n        [ngTemplateOutletContext]=\"model().context\"\n        [ngTemplateOutletInjector]=\"injector\" />\n    </div>\n  </svg:foreignObject>\n}\n\n<!-- SVG Template node -->\n@if (model().rawNode.type === 'svg-template' && nodeSvgTemplate()) {\n  <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n    <ng-container\n      [ngTemplateOutlet]=\"nodeSvgTemplate() ?? null\"\n      [ngTemplateOutletContext]=\"model().context\"\n      [ngTemplateOutletInjector]=\"injector\" />\n  </svg:g>\n}\n\n<!-- Component node -->\n@if (model().isComponentType) {\n  @if (model().componentInstance$ | async; as component) {\n    <svg:foreignObject\n      class=\"selectable\"\n      [attr.width]=\"model().foWidth()\"\n      [attr.height]=\"model().foHeight()\"\n      (click)=\"pullNode()\">\n      <div\n        nodeHandlesController\n        nodeResizeController\n        class=\"wrapper\"\n        [style.width]=\"model().styleWidth()\"\n        [style.height]=\"model().styleHeight()\">\n        <ng-container\n          [ngComponentOutlet]=\"$any(component)\"\n          [ngComponentOutletInputs]=\"model().componentTypeInputs\"\n          [ngComponentOutletInjector]=\"injector\" />\n      </div>\n    </svg:foreignObject>\n  }\n}\n\n<!-- Default group node -->\n@if (model().rawNode.type === 'default-group') {\n  <svg:rect\n    class=\"default-group-node\"\n    rx=\"5\"\n    ry=\"5\"\n    [resizable]=\"model().resizable()\"\n    [gap]=\"3\"\n    [resizerColor]=\"model().color()\"\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    (click)=\"pullNode(); selectNode()\" />\n}\n\n<!-- Template group node  -->\n@if (model().rawNode.type === 'template-group' && groupNodeTemplate()) {\n  <svg:g class=\"selectable\" nodeHandlesController (click)=\"pullNode()\">\n    <ng-container\n      [ngTemplateOutlet]=\"groupNodeTemplate() ?? null\"\n      [ngTemplateOutletContext]=\"model().context\"\n      [ngTemplateOutletInjector]=\"injector\" />\n  </svg:g>\n}\n\n<!-- Resizer -->\n@if (model().resizerTemplate(); as template) {\n  @if (model().resizable()) {\n    <ng-template [ngTemplateOutlet]=\"template\" />\n  }\n}\n\n<!-- Handles -->\n@for (handle of model().handles(); track handle) {\n  @if (handle.template === undefined) {\n    <svg:circle\n      class=\"default-handle\"\n      r=\"5\"\n      [attr.cx]=\"handle.hostOffset().x\"\n      [attr.cy]=\"handle.hostOffset().y\"\n      [attr.stroke-width]=\"handle.strokeWidth\"\n      (pointerStart)=\"startConnection($event, handle)\"\n      (pointerEnd)=\"endConnection()\" />\n  }\n\n  @if (handle.template === null) {\n    <svg:g\n      [handleSizeController]=\"handle\"\n      (pointerStart)=\"startConnection($event, handle)\"\n      (pointerEnd)=\"endConnection()\" />\n  }\n\n  @if (handle.template) {\n    <svg:g\n      [handleSizeController]=\"handle\"\n      (pointerStart)=\"startConnection($event, handle)\"\n      (pointerEnd)=\"endConnection()\">\n      <ng-container *ngTemplateOutlet=\"handle.template; context: handle.templateContext\" />\n    </svg:g>\n  }\n\n  @if (showMagnet()) {\n    <svg:circle\n      class=\"magnet\"\n      [attr.r]=\"model().magnetRadius\"\n      [attr.cx]=\"handle.hostOffset().x\"\n      [attr.cy]=\"handle.hostOffset().y\"\n      (pointerEnd)=\"endConnection(); resetValidateConnection(handle)\"\n      (pointerOver)=\"validateConnection(handle)\"\n      (pointerOut)=\"resetValidateConnection(handle)\" />\n  }\n}\n\n<!-- Toolbar -->\n@for (toolbar of toolbars(); track toolbar) {\n  <svg:foreignObject\n    [attr.width]=\"toolbar.size().width\"\n    [attr.height]=\"toolbar.size().height\"\n    [attr.transform]=\"toolbar.transform()\">\n    <ng-container [ngTemplateOutlet]=\"toolbar.template()\" />\n  </svg:foreignObject>\n}\n"]}