ngx-vflow 0.15.0 → 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +8 -1
- package/esm2022/lib/vflow/components/background/background.component.mjs +75 -8
- package/esm2022/lib/vflow/components/connection/connection.component.mjs +67 -49
- package/esm2022/lib/vflow/components/custom-node-base/custom-node-base.component.mjs +32 -25
- package/esm2022/lib/vflow/components/default-node/default-node.component.mjs +9 -11
- package/esm2022/lib/vflow/components/defs/defs.component.mjs +9 -12
- package/esm2022/lib/vflow/components/edge/edge.component.mjs +21 -24
- package/esm2022/lib/vflow/components/edge-label/edge-label.component.mjs +26 -31
- package/esm2022/lib/vflow/components/node/node.component.mjs +54 -50
- package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +95 -90
- package/esm2022/lib/vflow/decorators/run-in-injection-context.decorator.mjs +1 -1
- package/esm2022/lib/vflow/directives/changes-controller.directive.mjs +41 -125
- package/esm2022/lib/vflow/directives/connection-controller.directive.mjs +16 -16
- package/esm2022/lib/vflow/directives/drag-handle.directive.mjs +7 -6
- package/esm2022/lib/vflow/directives/flow-size-controller.directive.mjs +7 -6
- package/esm2022/lib/vflow/directives/handle-size-controller.directive.mjs +15 -12
- package/esm2022/lib/vflow/directives/map-context.directive.mjs +8 -5
- package/esm2022/lib/vflow/directives/pointer.directive.mjs +18 -19
- package/esm2022/lib/vflow/directives/reference.directive.mjs +6 -5
- package/esm2022/lib/vflow/directives/root-pointer.directive.mjs +8 -5
- package/esm2022/lib/vflow/directives/root-svg-context.directive.mjs +8 -5
- package/esm2022/lib/vflow/directives/selectable.directive.mjs +10 -7
- package/esm2022/lib/vflow/directives/space-point-context.directive.mjs +8 -5
- package/esm2022/lib/vflow/directives/template.directive.mjs +43 -25
- package/esm2022/lib/vflow/interfaces/component-node-event.interface.mjs +1 -1
- package/esm2022/lib/vflow/interfaces/edge.interface.mjs +1 -1
- package/esm2022/lib/vflow/interfaces/node.interface.mjs +1 -1
- package/esm2022/lib/vflow/interfaces/optimization.interface.mjs +1 -1
- package/esm2022/lib/vflow/math/edge-path/bezier-path.mjs +1 -1
- package/esm2022/lib/vflow/math/edge-path/smooth-step-path.mjs +170 -0
- package/esm2022/lib/vflow/models/connection.model.mjs +1 -1
- package/esm2022/lib/vflow/models/edge.model.mjs +6 -1
- package/esm2022/lib/vflow/models/handle.model.mjs +2 -2
- package/esm2022/lib/vflow/models/node.model.mjs +4 -10
- package/esm2022/lib/vflow/models/toolbar.model.mjs +1 -1
- package/esm2022/lib/vflow/public-components/custom-dynamic-node/custom-dynamic-node.component.mjs +16 -10
- package/esm2022/lib/vflow/public-components/custom-node/custom-node.component.mjs +15 -10
- package/esm2022/lib/vflow/public-components/handle/handle.component.mjs +52 -0
- package/esm2022/lib/vflow/public-components/minimap/minimap.component.mjs +42 -45
- package/esm2022/lib/vflow/public-components/node-toolbar/node-toolbar.component.mjs +24 -27
- package/esm2022/lib/vflow/public-components/resizable/resizable.component.mjs +60 -45
- package/esm2022/lib/vflow/services/component-event-bus.service.mjs +3 -3
- package/esm2022/lib/vflow/services/draggable.service.mjs +4 -4
- package/esm2022/lib/vflow/services/edge-changes.service.mjs +3 -3
- package/esm2022/lib/vflow/services/flow-entities.service.mjs +4 -4
- package/esm2022/lib/vflow/services/flow-settings.service.mjs +4 -10
- package/esm2022/lib/vflow/services/flow-status.service.mjs +4 -4
- package/esm2022/lib/vflow/services/handle.service.mjs +4 -4
- package/esm2022/lib/vflow/services/keyboard.service.mjs +5 -5
- package/esm2022/lib/vflow/services/node-accessor.service.mjs +3 -3
- package/esm2022/lib/vflow/services/node-changes.service.mjs +3 -3
- package/esm2022/lib/vflow/services/node-rendering.service.mjs +11 -4
- package/esm2022/lib/vflow/services/overlays.service.mjs +3 -3
- package/esm2022/lib/vflow/services/selection.service.mjs +4 -4
- package/esm2022/lib/vflow/services/viewport.service.mjs +3 -3
- package/esm2022/lib/vflow/types/background.type.mjs +1 -1
- package/esm2022/lib/vflow/utils/adjust-direction.mjs +1 -1
- package/esm2022/lib/vflow/utils/get-os.mjs +1 -1
- package/esm2022/lib/vflow/utils/is-group-node.mjs +4 -0
- package/esm2022/lib/vflow/utils/nodes.mjs +1 -1
- package/esm2022/lib/vflow/vflow.mjs +24 -0
- package/esm2022/public-api.mjs +4 -5
- package/fesm2022/ngx-vflow.mjs +1160 -1005
- package/fesm2022/ngx-vflow.mjs.map +1 -1
- package/lib/vflow/components/background/background.component.d.ts +14 -1
- package/lib/vflow/components/connection/connection.component.d.ts +3 -3
- package/lib/vflow/components/custom-node-base/custom-node-base.component.d.ts +4 -9
- package/lib/vflow/components/default-node/default-node.component.d.ts +2 -2
- package/lib/vflow/components/defs/defs.component.d.ts +2 -2
- package/lib/vflow/components/edge/edge.component.d.ts +7 -7
- package/lib/vflow/components/edge-label/edge-label.component.d.ts +9 -8
- package/lib/vflow/components/node/node.component.d.ts +5 -6
- package/lib/vflow/components/vflow/vflow.component.d.ts +19 -27
- package/lib/vflow/directives/changes-controller.directive.d.ts +29 -30
- package/lib/vflow/directives/connection-controller.directive.d.ts +1 -2
- package/lib/vflow/directives/drag-handle.directive.d.ts +1 -1
- package/lib/vflow/directives/flow-size-controller.directive.d.ts +1 -1
- package/lib/vflow/directives/handle-size-controller.directive.d.ts +2 -2
- package/lib/vflow/directives/map-context.directive.d.ts +1 -1
- package/lib/vflow/directives/pointer.directive.d.ts +9 -6
- package/lib/vflow/directives/reference.directive.d.ts +1 -1
- package/lib/vflow/directives/root-pointer.directive.d.ts +1 -1
- package/lib/vflow/directives/root-svg-context.directive.d.ts +1 -1
- package/lib/vflow/directives/selectable.directive.d.ts +1 -1
- package/lib/vflow/directives/space-point-context.directive.d.ts +1 -1
- package/lib/vflow/directives/template.directive.d.ts +6 -6
- package/lib/vflow/interfaces/component-node-event.interface.d.ts +3 -3
- package/lib/vflow/interfaces/edge.interface.d.ts +1 -1
- package/lib/vflow/interfaces/node.interface.d.ts +14 -14
- package/lib/vflow/interfaces/optimization.interface.d.ts +7 -1
- package/lib/vflow/math/edge-path/smooth-step-path.d.ts +5 -0
- package/lib/vflow/models/edge.model.d.ts +1 -17
- package/lib/vflow/models/handle.model.d.ts +4 -4
- package/lib/vflow/models/node.model.d.ts +2 -5
- package/lib/vflow/public-components/custom-dynamic-node/custom-dynamic-node.component.d.ts +3 -3
- package/lib/vflow/public-components/custom-node/custom-node.component.d.ts +3 -3
- package/lib/vflow/{components → public-components}/handle/handle.component.d.ts +5 -5
- package/lib/vflow/public-components/minimap/minimap.component.d.ts +8 -10
- package/lib/vflow/public-components/node-toolbar/node-toolbar.component.d.ts +6 -5
- package/lib/vflow/public-components/resizable/resizable.component.d.ts +5 -4
- package/lib/vflow/services/flow-settings.service.d.ts +0 -7
- 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/lib/vflow/utils/reference-keeper.d.ts +1 -1
- package/lib/vflow/vflow.d.ts +9 -0
- package/package.json +3 -3
- package/public-api.d.ts +2 -3
- package/esm2022/lib/vflow/components/handle/handle.component.mjs +0 -49
- package/esm2022/lib/vflow/interfaces/handle-positions.interface.mjs +0 -2
- package/esm2022/lib/vflow/vflow.module.mjs +0 -121
- package/lib/vflow/interfaces/handle-positions.interface.d.ts +0 -5
- package/lib/vflow/vflow.module.d.ts +0 -30
|
@@ -1,25 +1,26 @@
|
|
|
1
1
|
import { __decorate } from "tslib";
|
|
2
|
-
import { ChangeDetectionStrategy, Component, ElementRef, Injector,
|
|
2
|
+
import { ChangeDetectionStrategy, Component, ElementRef, Injector, NgZone, computed, effect, inject, input, viewChild, } from '@angular/core';
|
|
3
|
+
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
3
4
|
import { DraggableService } from '../../services/draggable.service';
|
|
4
5
|
import { FlowStatusService } from '../../services/flow-status.service';
|
|
5
6
|
import { HandleService } from '../../services/handle.service';
|
|
6
7
|
import { resizable } from '../../utils/resizable';
|
|
7
8
|
import { filter, map, startWith, switchMap, tap } from 'rxjs';
|
|
8
|
-
import { InjectionContext } from '../../decorators/run-in-injection-context.decorator';
|
|
9
|
+
import { InjectionContext, } from '../../decorators/run-in-injection-context.decorator';
|
|
9
10
|
import { NodeRenderingService } from '../../services/node-rendering.service';
|
|
10
11
|
import { FlowSettingsService } from '../../services/flow-settings.service';
|
|
11
12
|
import { SelectionService } from '../../services/selection.service';
|
|
12
13
|
import { ConnectionControllerDirective } from '../../directives/connection-controller.directive';
|
|
13
|
-
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
14
14
|
import { NodeAccessorService } from '../../services/node-accessor.service';
|
|
15
15
|
import { OverlaysService } from '../../services/overlays.service';
|
|
16
|
+
import { HandleSizeControllerDirective } from '../../directives/handle-size-controller.directive';
|
|
17
|
+
import { NgTemplateOutlet, NgComponentOutlet } from '@angular/common';
|
|
18
|
+
import { DefaultNodeComponent } from '../default-node/default-node.component';
|
|
19
|
+
import { PointerDirective } from '../../directives/pointer.directive';
|
|
20
|
+
// public components that uses in default node (loaded by defer)
|
|
21
|
+
import { ResizableComponent } from '../../public-components/resizable/resizable.component';
|
|
22
|
+
import { HandleComponent } from '../../public-components/handle/handle.component';
|
|
16
23
|
import * as i0 from "@angular/core";
|
|
17
|
-
import * as i1 from "@angular/common";
|
|
18
|
-
import * as i2 from "../default-node/default-node.component";
|
|
19
|
-
import * as i3 from "../handle/handle.component";
|
|
20
|
-
import * as i4 from "../../public-components/resizable/resizable.component";
|
|
21
|
-
import * as i5 from "../../directives/handle-size-controller.directive";
|
|
22
|
-
import * as i6 from "../../directives/pointer.directive";
|
|
23
24
|
export class NodeComponent {
|
|
24
25
|
constructor() {
|
|
25
26
|
this.injector = inject(Injector);
|
|
@@ -34,40 +35,47 @@ export class NodeComponent {
|
|
|
34
35
|
this.nodeAccessor = inject(NodeAccessorService);
|
|
35
36
|
this.overlaysService = inject(OverlaysService);
|
|
36
37
|
this.zone = inject(NgZone);
|
|
38
|
+
this.nodeModel = input.required();
|
|
39
|
+
this.nodeTemplate = input();
|
|
40
|
+
this.groupNodeTemplate = input();
|
|
41
|
+
this.htmlWrapperRef = viewChild.required('htmlWrapper');
|
|
37
42
|
this.showMagnet = computed(() => this.flowStatusService.status().state === 'connection-start' ||
|
|
38
43
|
this.flowStatusService.status().state === 'connection-validation');
|
|
39
|
-
this.styleWidth = computed(() => `${this.nodeModel.size().width}px`);
|
|
40
|
-
this.styleHeight = computed(() => `${this.nodeModel.size().height}px`);
|
|
41
|
-
this.toolbar = computed(() => this.overlaysService.nodeToolbars().get(this.nodeModel));
|
|
44
|
+
this.styleWidth = computed(() => `${this.nodeModel().size().width}px`);
|
|
45
|
+
this.styleHeight = computed(() => `${this.nodeModel().size().height}px`);
|
|
46
|
+
this.toolbar = computed(() => this.overlaysService.nodeToolbars().get(this.nodeModel()));
|
|
42
47
|
}
|
|
43
48
|
ngOnInit() {
|
|
44
|
-
this.nodeAccessor.model.set(this.nodeModel);
|
|
45
|
-
this.handleService.node.set(this.nodeModel);
|
|
49
|
+
this.nodeAccessor.model.set(this.nodeModel());
|
|
50
|
+
this.handleService.node.set(this.nodeModel());
|
|
46
51
|
effect(() => {
|
|
47
|
-
if (this.nodeModel.draggable()) {
|
|
48
|
-
this.draggableService.enable(this.hostRef.nativeElement, this.nodeModel);
|
|
52
|
+
if (this.nodeModel().draggable()) {
|
|
53
|
+
this.draggableService.enable(this.hostRef.nativeElement, this.nodeModel());
|
|
49
54
|
}
|
|
50
55
|
else {
|
|
51
56
|
this.draggableService.disable(this.hostRef.nativeElement);
|
|
52
57
|
}
|
|
53
58
|
});
|
|
54
|
-
this.nodeModel
|
|
55
|
-
.pipe(switchMap((handles) => resizable(handles.map(h => h.parentReference), this.zone)
|
|
56
|
-
.pipe(map(() => handles))), tap((handles) => {
|
|
59
|
+
this.nodeModel()
|
|
60
|
+
.handles$.pipe(switchMap((handles) => resizable(handles.map((h) => h.parentReference), this.zone).pipe(map(() => handles))), tap((handles) => {
|
|
57
61
|
// TODO (performance) inspect how to avoid calls of this when flow initially rendered
|
|
58
|
-
handles.forEach(h => h.updateParent());
|
|
62
|
+
handles.forEach((h) => h.updateParent());
|
|
59
63
|
}), takeUntilDestroyed())
|
|
60
64
|
.subscribe();
|
|
61
65
|
}
|
|
62
66
|
ngAfterViewInit() {
|
|
63
|
-
this.nodeModel.linkDefaultNodeSizeWithModelSize();
|
|
64
|
-
if (this.nodeModel.node.type === 'html-template' ||
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
this.nodeModel.
|
|
70
|
-
|
|
67
|
+
this.nodeModel().linkDefaultNodeSizeWithModelSize();
|
|
68
|
+
if (this.nodeModel().node.type === 'html-template' ||
|
|
69
|
+
this.nodeModel().isComponentType) {
|
|
70
|
+
resizable([this.htmlWrapperRef().nativeElement], this.zone)
|
|
71
|
+
.pipe(startWith(null), tap(() => this.nodeModel()
|
|
72
|
+
.handles()
|
|
73
|
+
.forEach((h) => h.updateParent())), filter(() => !this.nodeModel().resizing()), tap(() => {
|
|
74
|
+
const width = this.htmlWrapperRef().nativeElement.clientWidth;
|
|
75
|
+
const height = this.htmlWrapperRef().nativeElement.clientHeight;
|
|
76
|
+
this.nodeModel().size.set({ width, height });
|
|
77
|
+
}), takeUntilDestroyed())
|
|
78
|
+
.subscribe();
|
|
71
79
|
}
|
|
72
80
|
}
|
|
73
81
|
ngOnDestroy() {
|
|
@@ -88,15 +96,15 @@ export class NodeComponent {
|
|
|
88
96
|
this.connectionController.endConnection(handle);
|
|
89
97
|
}
|
|
90
98
|
pullNode() {
|
|
91
|
-
this.nodeRenderingService.pullNode(this.nodeModel);
|
|
99
|
+
this.nodeRenderingService.pullNode(this.nodeModel());
|
|
92
100
|
}
|
|
93
101
|
selectNode() {
|
|
94
102
|
if (this.flowSettingsService.entitiesSelectable()) {
|
|
95
|
-
this.selectionService.select(this.nodeModel);
|
|
103
|
+
this.selectionService.select(this.nodeModel());
|
|
96
104
|
}
|
|
97
105
|
}
|
|
98
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "
|
|
99
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "
|
|
106
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NodeComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
107
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: NodeComponent, isStandalone: true, selector: "g[node]", inputs: { nodeModel: { classPropertyName: "nodeModel", publicName: "nodeModel", isSignal: true, isRequired: true, transformFunction: null }, nodeTemplate: { classPropertyName: "nodeTemplate", publicName: "nodeTemplate", isSignal: true, isRequired: false, transformFunction: null }, groupNodeTemplate: { classPropertyName: "groupNodeTemplate", publicName: "groupNodeTemplate", isSignal: true, isRequired: false, transformFunction: null } }, host: { classAttribute: "vflow-node" }, providers: [HandleService, NodeAccessorService], viewQueries: [{ propertyName: "htmlWrapperRef", first: true, predicate: ["htmlWrapper"], descendants: true, isSignal: true }], ngImport: i0, template: "<!-- Default node -->\n@if (nodeModel().node.type === \"default\") {\n @defer {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"nodeModel().size().width\"\n [attr.height]=\"nodeModel().size().height\"\n (pointerStart)=\"pullNode(); selectNode()\"\n >\n <default-node\n #htmlWrapper\n [selected]=\"nodeModel().selected()\"\n [style.width]=\"styleWidth()\"\n [style.height]=\"styleHeight()\"\n [style.max-width]=\"styleWidth()\"\n [style.max-height]=\"styleHeight()\"\n >\n <div [outerHTML]=\"nodeModel().text()\"></div>\n\n <handle type=\"source\" position=\"right\"/>\n <handle type=\"target\" position=\"left\" />\n </default-node>\n </svg:foreignObject>\n }\n}\n\n<!-- Template node -->\n@if (nodeModel().node.type === \"html-template\" && nodeTemplate()) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"nodeModel().size().width\"\n [attr.height]=\"nodeModel().size().height\"\n (pointerStart)=\"pullNode()\"\n >\n <div\n #htmlWrapper\n class=\"wrapper\"\n [style.width]=\"styleWidth()\"\n [style.height]=\"styleHeight()\"\n >\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"{\n $implicit: { node: nodeModel().node, selected: nodeModel().selected },\n }\"\n [ngTemplateOutletInjector]=\"injector\"\n />\n </div>\n </svg:foreignObject>\n}\n\n<!-- Component node -->\n@if (nodeModel().isComponentType) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"nodeModel().size().width\"\n [attr.height]=\"nodeModel().size().height\"\n (pointerStart)=\"pullNode()\"\n >\n <div\n #htmlWrapper\n class=\"wrapper\"\n [style.width]=\"styleWidth()\"\n [style.height]=\"styleHeight()\"\n >\n <ng-container\n [ngComponentOutlet]=\"$any(nodeModel().node.type)\"\n [ngComponentOutletInputs]=\"nodeModel().componentTypeInputs\"\n [ngComponentOutletInjector]=\"injector\"\n />\n </div>\n </svg:foreignObject>\n}\n\n<!-- Default group node -->\n@if (nodeModel().node.type === \"default-group\") {\n @defer {\n <svg:rect\n [resizable]=\"nodeModel().resizable()\"\n [gap]=\"3\"\n [resizerColor]=\"nodeModel().color()\"\n class=\"default-group-node\"\n rx=\"5\"\n ry=\"5\"\n [class.default-group-node_selected]=\"nodeModel().selected()\"\n [attr.width]=\"nodeModel().size().width\"\n [attr.height]=\"nodeModel().size().height\"\n [style.stroke]=\"nodeModel().color()\"\n [style.fill]=\"nodeModel().color()\"\n (pointerStart)=\"pullNode(); selectNode()\"\n />\n }\n}\n\n<!-- Template group node -->\n@if (nodeModel().node.type === \"template-group\" && groupNodeTemplate()) {\n <svg:g class=\"selectable\" (pointerStart)=\"pullNode()\">\n <ng-container\n [ngTemplateOutlet]=\"groupNodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"{\n $implicit: {\n node: nodeModel().node,\n selected: nodeModel().selected,\n width: nodeModel().width,\n height: nodeModel().height,\n },\n }\"\n [ngTemplateOutletInjector]=\"injector\"\n />\n </svg:g>\n}\n\n<!-- Resizer -->\n@if (nodeModel().resizerTemplate(); as template) {\n @if (nodeModel().resizable()) {\n <ng-template [ngTemplateOutlet]=\"template\" />\n }\n}\n\n<!-- Handles -->\n@for (handle of nodeModel().handles(); track handle) {\n @if (!handle.template) {\n <svg:circle\n class=\"default-handle\"\n [attr.cx]=\"handle.offset().x\"\n [attr.cy]=\"handle.offset().y\"\n [attr.stroke-width]=\"handle.strokeWidth\"\n r=\"5\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection(handle)\"\n />\n }\n\n @if (handle.template) {\n <svg:g\n [handleSizeController]=\"handle\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection(handle)\"\n >\n <ng-container\n *ngTemplateOutlet=\"handle.template; context: handle.templateContext\"\n />\n </svg:g>\n }\n\n @if (showMagnet()) {\n <svg:circle\n class=\"magnet\"\n [attr.r]=\"nodeModel().magnetRadius\"\n [attr.cx]=\"handle.offset().x\"\n [attr.cy]=\"handle.offset().y\"\n (pointerEnd)=\"endConnection(handle); resetValidateConnection(handle)\"\n (pointerOver)=\"validateConnection(handle)\"\n (pointerOut)=\"resetValidateConnection(handle)\"\n />\n }\n}\n\n<!-- Toolbar -->\n@if (toolbar(); as toolbar) {\n <svg:foreignObject\n [attr.width]=\"toolbar.size().width\"\n [attr.height]=\"toolbar.size().height\"\n [attr.transform]=\"toolbar.transform()\"\n >\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: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "directive", type: HandleSizeControllerDirective, selector: "[handleSizeController]", inputs: ["handleSizeController"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
100
108
|
}
|
|
101
109
|
__decorate([
|
|
102
110
|
InjectionContext
|
|
@@ -104,22 +112,18 @@ __decorate([
|
|
|
104
112
|
__decorate([
|
|
105
113
|
InjectionContext
|
|
106
114
|
], NodeComponent.prototype, "ngAfterViewInit", null);
|
|
107
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "
|
|
115
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NodeComponent, decorators: [{
|
|
108
116
|
type: Component,
|
|
109
|
-
args: [{ selector: 'g[node]', changeDetection: ChangeDetectionStrategy.OnPush, providers: [HandleService, NodeAccessorService], host: {
|
|
110
|
-
|
|
111
|
-
},
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
}], htmlWrapperRef: [{
|
|
122
|
-
type: ViewChild,
|
|
123
|
-
args: ['htmlWrapper']
|
|
124
|
-
}], ngOnInit: [], ngAfterViewInit: [] } });
|
|
125
|
-
//# 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 { AfterViewInit, ChangeDetectionStrategy, Component, ElementRef, Injector, Input, NgZone, OnDestroy, OnInit, TemplateRef, ViewChild, computed, effect, inject } 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 { resizable } from '../../utils/resizable';\nimport { filter, first, map, startWith, switchMap, tap } from 'rxjs';\nimport { InjectionContext, WithInjector } from '../../decorators/run-in-injection-context.decorator';\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 { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { NodeAccessorService } from '../../services/node-accessor.service';\nimport { OverlaysService } from '../../services/overlays.service';\n\nexport type HandleState = 'valid' | 'invalid' | 'idle'\n\n@Component({\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})\nexport class NodeComponent implements OnInit, AfterViewInit, OnDestroy, WithInjector {\n  public 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 connectionController = inject(ConnectionControllerDirective)\n  private nodeAccessor = inject(NodeAccessorService)\n  private overlaysService = inject(OverlaysService)\n  private zone = inject(NgZone)\n\n  @Input()\n  public nodeModel!: NodeModel\n\n  @Input()\n  public nodeTemplate?: TemplateRef<any>\n\n  @Input()\n  public groupNodeTemplate?: TemplateRef<any>\n\n  @ViewChild('nodeContent')\n  public nodeContentRef!: ElementRef<SVGGraphicsElement>\n\n  @ViewChild('htmlWrapper')\n  public htmlWrapperRef!: ElementRef<HTMLDivElement>\n\n  protected showMagnet = computed(() =>\n    this.flowStatusService.status().state === 'connection-start' ||\n    this.flowStatusService.status().state === 'connection-validation'\n  )\n\n  protected styleWidth = computed(() => `${this.nodeModel.size().width}px`)\n  protected styleHeight = computed(() => `${this.nodeModel.size().height}px`)\n\n  protected toolbar = computed(() => this.overlaysService.nodeToolbars().get(this.nodeModel))\n\n  @InjectionContext\n  public ngOnInit() {\n    this.nodeAccessor.model.set(this.nodeModel);\n\n    this.handleService.node.set(this.nodeModel);\n\n    effect(() => {\n      if (this.nodeModel.draggable()) {\n        this.draggableService.enable(this.hostRef.nativeElement, this.nodeModel)\n      } else {\n        this.draggableService.disable(this.hostRef.nativeElement)\n      }\n    })\n\n    this.nodeModel.handles$\n      .pipe(\n        switchMap((handles) =>\n          resizable(handles.map(h => h.parentReference!), this.zone)\n            .pipe(map(() => handles))\n        ),\n        tap((handles) => {\n          // TODO (performance) inspect how to avoid calls of this when flow initially rendered\n          handles.forEach(h => h.updateParent())\n        }),\n        takeUntilDestroyed()\n      )\n      .subscribe()\n  }\n\n  @InjectionContext\n  public ngAfterViewInit(): void {\n    this.nodeModel.linkDefaultNodeSizeWithModelSize()\n\n    if (this.nodeModel.node.type === 'html-template' || this.nodeModel.isComponentType) {\n      resizable([this.htmlWrapperRef.nativeElement], this.zone)\n        .pipe(\n          startWith(null),\n          tap(() => this.nodeModel.handles().forEach(h => h.updateParent())),\n          filter(() => !this.nodeModel.resizing()),\n          tap(() => {\n            const width = this.htmlWrapperRef.nativeElement.clientWidth\n            const height = this.htmlWrapperRef.nativeElement.clientHeight\n\n            this.nodeModel.size.set({ width, height })\n          }),\n          takeUntilDestroyed()\n        ).subscribe()\n    }\n  }\n\n  public ngOnDestroy(): void {\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(handle: HandleModel) {\n    this.connectionController.endConnection(handle)\n  }\n\n  protected pullNode() {\n    this.nodeRenderingService.pullNode(this.nodeModel)\n  }\n\n  protected selectNode() {\n    if (this.flowSettingsService.entitiesSelectable()) {\n      this.selectionService.select(this.nodeModel)\n    }\n  }\n}\n","<!-- Default node -->\n<svg:foreignObject\n  *ngIf=\"nodeModel.node.type === 'default'\"\n  class=\"selectable\"\n  #nodeContent\n  [attr.width]=\"nodeModel.size().width\"\n  [attr.height]=\"nodeModel.size().height\"\n  (pointerStart)=\"pullNode(); selectNode()\"\n>\n  <default-node\n    #htmlWrapper\n    [selected]=\"nodeModel.selected()\"\n    [style.width]=\"styleWidth()\"\n    [style.height]=\"styleHeight()\"\n    [style.max-width]=\"styleWidth()\"\n    [style.max-height]=\"styleHeight()\"\n  >\n    <div [outerHTML]=\"nodeModel.text()\"></div>\n\n    <handle type=\"source\" [position]=\"nodeModel.sourcePosition()\" />\n    <handle type=\"target\" [position]=\"nodeModel.targetPosition()\" />\n  </default-node>\n</svg:foreignObject>\n\n<!-- Template node -->\n<svg:foreignObject\n  *ngIf=\"nodeModel.node.type === 'html-template' && nodeTemplate\"\n  class=\"selectable\"\n  [attr.width]=\"nodeModel.size().width\"\n  [attr.height]=\"nodeModel.size().height\"\n  (pointerStart)=\"pullNode()\"\n>\n  <div\n    #htmlWrapper\n    class=\"wrapper\"\n    [style.width]=\"styleWidth()\"\n    [style.height]=\"styleHeight()\"\n  >\n    <ng-container\n      [ngTemplateOutlet]=\"nodeTemplate\"\n      [ngTemplateOutletContext]=\"{ $implicit: { node: nodeModel.node, selected: nodeModel.selected } }\"\n      [ngTemplateOutletInjector]=\"injector\"\n    />\n  </div>\n</svg:foreignObject>\n\n<!-- Component node -->\n<svg:foreignObject\n  *ngIf=\"nodeModel.isComponentType\"\n  class=\"selectable\"\n  [attr.width]=\"nodeModel.size().width\"\n  [attr.height]=\"nodeModel.size().height\"\n  (pointerStart)=\"pullNode()\"\n>\n  <div\n    #htmlWrapper\n    class=\"wrapper\"\n    [style.width]=\"styleWidth()\"\n    [style.height]=\"styleHeight()\"\n  >\n    <ng-container\n      [ngComponentOutlet]=\"$any(nodeModel.node.type)\"\n      [ngComponentOutletInputs]=\"nodeModel.componentTypeInputs()\"\n      [ngComponentOutletInjector]=\"injector\"\n    />\n  </div>\n</svg:foreignObject>\n\n<!-- Default group node -->\n<svg:rect\n  *ngIf=\"nodeModel.node.type === 'default-group'\"\n  [resizable]=\"nodeModel.resizable()\"\n  [gap]=\"3\"\n  [resizerColor]=\"nodeModel.color()\"\n  class=\"default-group-node\"\n  rx=\"5\"\n  ry=\"5\"\n  [class.default-group-node_selected]=\"nodeModel.selected()\"\n  [attr.width]=\"nodeModel.size().width\"\n  [attr.height]=\"nodeModel.size().height\"\n  [style.stroke]=\"nodeModel.color()\"\n  [style.fill]=\"nodeModel.color()\"\n  (pointerStart)=\"pullNode(); selectNode()\"\n/>\n\n<!-- Template group node  -->\n<svg:g\n  *ngIf=\"nodeModel.node.type === 'template-group' && groupNodeTemplate\"\n  class=\"selectable\"\n  (pointerStart)=\"pullNode()\"\n>\n  <ng-container\n    [ngTemplateOutlet]=\"groupNodeTemplate\"\n    [ngTemplateOutletContext]=\"{ $implicit: { node: nodeModel.node, selected: nodeModel.selected, width: nodeModel.width, height: nodeModel.height } }\"\n    [ngTemplateOutletInjector]=\"injector\"\n  />\n</svg:g>\n\n<!-- Resizer -->\n<ng-container *ngIf=\"nodeModel.resizerTemplate() as template\">\n  <ng-container *ngIf=\"nodeModel.resizable()\">\n    <ng-template [ngTemplateOutlet]=\"template\" />\n  </ng-container>\n</ng-container>\n\n<!-- Handles -->\n<ng-container *ngFor=\"let handle of nodeModel.handles()\">\n  <svg:circle\n    *ngIf=\"!handle.template\"\n    class=\"default-handle\"\n    [attr.cx]=\"handle.offset().x\"\n    [attr.cy]=\"handle.offset().y\"\n    [attr.stroke-width]=\"handle.strokeWidth\"\n    r=\"5\"\n    (pointerStart)=\"startConnection($event, handle)\"\n    (pointerEnd)=\"endConnection(handle)\"\n  />\n\n  <svg:g\n    *ngIf=\"handle.template\"\n    [handleSizeController]=\"handle\"\n    (pointerStart)=\"startConnection($event, handle)\"\n    (pointerEnd)=\"endConnection(handle)\"\n  >\n    <ng-container *ngTemplateOutlet=\"handle.template; context: handle.templateContext\" />\n  </svg:g>\n\n  <svg:circle\n    *ngIf=\"showMagnet()\"\n    class=\"magnet\"\n    [attr.r]=\"nodeModel.magnetRadius\"\n    [attr.cx]=\"handle.offset().x\"\n    [attr.cy]=\"handle.offset().y\"\n    (pointerEnd)=\"endConnection(handle); resetValidateConnection(handle)\"\n    (pointerOver)=\"validateConnection(handle)\"\n    (pointerOut)=\"resetValidateConnection(handle)\"\n  />\n</ng-container>\n\n<!-- Toolbar -->\n<svg:foreignObject\n  *ngIf=\"toolbar() as toolbar\"\n  [attr.width]=\"toolbar.size().width\"\n  [attr.height]=\"toolbar.size().height\"\n  [attr.transform]=\"toolbar.transform()\"\n>\n  <ng-container [ngTemplateOutlet]=\"toolbar.template()\" />\n</svg:foreignObject>\n"]}
|
|
117
|
+
args: [{ standalone: true, selector: 'g[node]', changeDetection: ChangeDetectionStrategy.OnPush, providers: [HandleService, NodeAccessorService], host: {
|
|
118
|
+
class: 'vflow-node',
|
|
119
|
+
}, imports: [
|
|
120
|
+
PointerDirective,
|
|
121
|
+
DefaultNodeComponent,
|
|
122
|
+
HandleComponent,
|
|
123
|
+
NgTemplateOutlet,
|
|
124
|
+
NgComponentOutlet,
|
|
125
|
+
ResizableComponent,
|
|
126
|
+
HandleSizeControllerDirective,
|
|
127
|
+
], template: "<!-- Default node -->\n@if (nodeModel().node.type === \"default\") {\n @defer {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"nodeModel().size().width\"\n [attr.height]=\"nodeModel().size().height\"\n (pointerStart)=\"pullNode(); selectNode()\"\n >\n <default-node\n #htmlWrapper\n [selected]=\"nodeModel().selected()\"\n [style.width]=\"styleWidth()\"\n [style.height]=\"styleHeight()\"\n [style.max-width]=\"styleWidth()\"\n [style.max-height]=\"styleHeight()\"\n >\n <div [outerHTML]=\"nodeModel().text()\"></div>\n\n <handle type=\"source\" position=\"right\"/>\n <handle type=\"target\" position=\"left\" />\n </default-node>\n </svg:foreignObject>\n }\n}\n\n<!-- Template node -->\n@if (nodeModel().node.type === \"html-template\" && nodeTemplate()) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"nodeModel().size().width\"\n [attr.height]=\"nodeModel().size().height\"\n (pointerStart)=\"pullNode()\"\n >\n <div\n #htmlWrapper\n class=\"wrapper\"\n [style.width]=\"styleWidth()\"\n [style.height]=\"styleHeight()\"\n >\n <ng-container\n [ngTemplateOutlet]=\"nodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"{\n $implicit: { node: nodeModel().node, selected: nodeModel().selected },\n }\"\n [ngTemplateOutletInjector]=\"injector\"\n />\n </div>\n </svg:foreignObject>\n}\n\n<!-- Component node -->\n@if (nodeModel().isComponentType) {\n <svg:foreignObject\n class=\"selectable\"\n [attr.width]=\"nodeModel().size().width\"\n [attr.height]=\"nodeModel().size().height\"\n (pointerStart)=\"pullNode()\"\n >\n <div\n #htmlWrapper\n class=\"wrapper\"\n [style.width]=\"styleWidth()\"\n [style.height]=\"styleHeight()\"\n >\n <ng-container\n [ngComponentOutlet]=\"$any(nodeModel().node.type)\"\n [ngComponentOutletInputs]=\"nodeModel().componentTypeInputs\"\n [ngComponentOutletInjector]=\"injector\"\n />\n </div>\n </svg:foreignObject>\n}\n\n<!-- Default group node -->\n@if (nodeModel().node.type === \"default-group\") {\n @defer {\n <svg:rect\n [resizable]=\"nodeModel().resizable()\"\n [gap]=\"3\"\n [resizerColor]=\"nodeModel().color()\"\n class=\"default-group-node\"\n rx=\"5\"\n ry=\"5\"\n [class.default-group-node_selected]=\"nodeModel().selected()\"\n [attr.width]=\"nodeModel().size().width\"\n [attr.height]=\"nodeModel().size().height\"\n [style.stroke]=\"nodeModel().color()\"\n [style.fill]=\"nodeModel().color()\"\n (pointerStart)=\"pullNode(); selectNode()\"\n />\n }\n}\n\n<!-- Template group node -->\n@if (nodeModel().node.type === \"template-group\" && groupNodeTemplate()) {\n <svg:g class=\"selectable\" (pointerStart)=\"pullNode()\">\n <ng-container\n [ngTemplateOutlet]=\"groupNodeTemplate() ?? null\"\n [ngTemplateOutletContext]=\"{\n $implicit: {\n node: nodeModel().node,\n selected: nodeModel().selected,\n width: nodeModel().width,\n height: nodeModel().height,\n },\n }\"\n [ngTemplateOutletInjector]=\"injector\"\n />\n </svg:g>\n}\n\n<!-- Resizer -->\n@if (nodeModel().resizerTemplate(); as template) {\n @if (nodeModel().resizable()) {\n <ng-template [ngTemplateOutlet]=\"template\" />\n }\n}\n\n<!-- Handles -->\n@for (handle of nodeModel().handles(); track handle) {\n @if (!handle.template) {\n <svg:circle\n class=\"default-handle\"\n [attr.cx]=\"handle.offset().x\"\n [attr.cy]=\"handle.offset().y\"\n [attr.stroke-width]=\"handle.strokeWidth\"\n r=\"5\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection(handle)\"\n />\n }\n\n @if (handle.template) {\n <svg:g\n [handleSizeController]=\"handle\"\n (pointerStart)=\"startConnection($event, handle)\"\n (pointerEnd)=\"endConnection(handle)\"\n >\n <ng-container\n *ngTemplateOutlet=\"handle.template; context: handle.templateContext\"\n />\n </svg:g>\n }\n\n @if (showMagnet()) {\n <svg:circle\n class=\"magnet\"\n [attr.r]=\"nodeModel().magnetRadius\"\n [attr.cx]=\"handle.offset().x\"\n [attr.cy]=\"handle.offset().y\"\n (pointerEnd)=\"endConnection(handle); resetValidateConnection(handle)\"\n (pointerOver)=\"validateConnection(handle)\"\n (pointerOut)=\"resetValidateConnection(handle)\"\n />\n }\n}\n\n<!-- Toolbar -->\n@if (toolbar(); as toolbar) {\n <svg:foreignObject\n [attr.width]=\"toolbar.size().width\"\n [attr.height]=\"toolbar.size().height\"\n [attr.transform]=\"toolbar.transform()\"\n >\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"] }]
|
|
128
|
+
}], propDecorators: { ngOnInit: [], ngAfterViewInit: [] } });
|
|
129
|
+
//# 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  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  Injector,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  TemplateRef,\n  computed,\n  effect,\n  inject,\n  input,\n  viewChild,\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\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 { resizable } from '../../utils/resizable';\nimport { filter, map, startWith, switchMap, tap } from 'rxjs';\nimport {\n  InjectionContext,\n  WithInjector,\n} from '../../decorators/run-in-injection-context.decorator';\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// 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';\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  ],\n})\nexport class NodeComponent\n  implements OnInit, AfterViewInit, OnDestroy, WithInjector {\n  public 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 connectionController = inject(ConnectionControllerDirective);\n  private nodeAccessor = inject(NodeAccessorService);\n  private overlaysService = inject(OverlaysService);\n  private zone = inject(NgZone);\n\n  public nodeModel = input.required<NodeModel>();\n\n  public nodeTemplate = input<TemplateRef<any>>();\n\n  public groupNodeTemplate = input<TemplateRef<any>>();\n\n  public htmlWrapperRef = viewChild.required<ElementRef<HTMLDivElement>>('htmlWrapper');\n\n  protected showMagnet = computed(\n    () =>\n      this.flowStatusService.status().state === 'connection-start' ||\n      this.flowStatusService.status().state === 'connection-validation',\n  );\n\n  protected styleWidth = computed(() => `${this.nodeModel().size().width}px`);\n  protected styleHeight = computed(() => `${this.nodeModel().size().height}px`);\n\n  protected toolbar = computed(() =>\n    this.overlaysService.nodeToolbars().get(this.nodeModel()),\n  );\n\n  @InjectionContext\n  public ngOnInit() {\n    this.nodeAccessor.model.set(this.nodeModel());\n\n    this.handleService.node.set(this.nodeModel());\n\n    effect(() => {\n      if (this.nodeModel().draggable()) {\n        this.draggableService.enable(\n          this.hostRef.nativeElement,\n          this.nodeModel(),\n        );\n      } else {\n        this.draggableService.disable(this.hostRef.nativeElement);\n      }\n    });\n\n    this.nodeModel()\n      .handles$.pipe(\n        switchMap((handles) =>\n          resizable(\n            handles.map((h) => h.parentReference!),\n            this.zone,\n          ).pipe(map(() => handles)),\n        ),\n        tap((handles) => {\n          // TODO (performance) inspect how to avoid calls of this when flow initially rendered\n          handles.forEach((h) => h.updateParent());\n        }),\n        takeUntilDestroyed(),\n      )\n      .subscribe();\n  }\n\n  @InjectionContext\n  public ngAfterViewInit(): void {\n    this.nodeModel().linkDefaultNodeSizeWithModelSize();\n\n    if (\n      this.nodeModel().node.type === 'html-template' ||\n      this.nodeModel().isComponentType\n    ) {\n      resizable([this.htmlWrapperRef().nativeElement], this.zone)\n        .pipe(\n          startWith(null),\n          tap(() =>\n            this.nodeModel()\n              .handles()\n              .forEach((h) => h.updateParent()),\n          ),\n          filter(() => !this.nodeModel().resizing()),\n          tap(() => {\n            const width = this.htmlWrapperRef().nativeElement.clientWidth;\n            const height = this.htmlWrapperRef().nativeElement.clientHeight;\n\n            this.nodeModel().size.set({ width, height });\n          }),\n          takeUntilDestroyed(),\n        )\n        .subscribe();\n    }\n  }\n\n  public ngOnDestroy(): void {\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(handle: HandleModel) {\n    this.connectionController.endConnection(handle);\n  }\n\n  protected pullNode() {\n    this.nodeRenderingService.pullNode(this.nodeModel());\n  }\n\n  protected selectNode() {\n    if (this.flowSettingsService.entitiesSelectable()) {\n      this.selectionService.select(this.nodeModel());\n    }\n  }\n}\n","<!-- Default node -->\n@if (nodeModel().node.type === \"default\") {\n  @defer {\n    <svg:foreignObject\n      class=\"selectable\"\n      [attr.width]=\"nodeModel().size().width\"\n      [attr.height]=\"nodeModel().size().height\"\n      (pointerStart)=\"pullNode(); selectNode()\"\n    >\n      <default-node\n        #htmlWrapper\n        [selected]=\"nodeModel().selected()\"\n        [style.width]=\"styleWidth()\"\n        [style.height]=\"styleHeight()\"\n        [style.max-width]=\"styleWidth()\"\n        [style.max-height]=\"styleHeight()\"\n      >\n        <div [outerHTML]=\"nodeModel().text()\"></div>\n\n        <handle type=\"source\" position=\"right\"/>\n        <handle type=\"target\" position=\"left\" />\n      </default-node>\n    </svg:foreignObject>\n  }\n}\n\n<!-- Template node -->\n@if (nodeModel().node.type === \"html-template\" && nodeTemplate()) {\n  <svg:foreignObject\n    class=\"selectable\"\n    [attr.width]=\"nodeModel().size().width\"\n    [attr.height]=\"nodeModel().size().height\"\n    (pointerStart)=\"pullNode()\"\n  >\n    <div\n      #htmlWrapper\n      class=\"wrapper\"\n      [style.width]=\"styleWidth()\"\n      [style.height]=\"styleHeight()\"\n    >\n      <ng-container\n        [ngTemplateOutlet]=\"nodeTemplate() ?? null\"\n        [ngTemplateOutletContext]=\"{\n          $implicit: { node: nodeModel().node, selected: nodeModel().selected },\n        }\"\n        [ngTemplateOutletInjector]=\"injector\"\n      />\n    </div>\n  </svg:foreignObject>\n}\n\n<!-- Component node -->\n@if (nodeModel().isComponentType) {\n  <svg:foreignObject\n    class=\"selectable\"\n    [attr.width]=\"nodeModel().size().width\"\n    [attr.height]=\"nodeModel().size().height\"\n    (pointerStart)=\"pullNode()\"\n  >\n    <div\n      #htmlWrapper\n      class=\"wrapper\"\n      [style.width]=\"styleWidth()\"\n      [style.height]=\"styleHeight()\"\n    >\n      <ng-container\n        [ngComponentOutlet]=\"$any(nodeModel().node.type)\"\n        [ngComponentOutletInputs]=\"nodeModel().componentTypeInputs\"\n        [ngComponentOutletInjector]=\"injector\"\n      />\n    </div>\n  </svg:foreignObject>\n}\n\n<!-- Default group node -->\n@if (nodeModel().node.type === \"default-group\") {\n  @defer {\n    <svg:rect\n      [resizable]=\"nodeModel().resizable()\"\n      [gap]=\"3\"\n      [resizerColor]=\"nodeModel().color()\"\n      class=\"default-group-node\"\n      rx=\"5\"\n      ry=\"5\"\n      [class.default-group-node_selected]=\"nodeModel().selected()\"\n      [attr.width]=\"nodeModel().size().width\"\n      [attr.height]=\"nodeModel().size().height\"\n      [style.stroke]=\"nodeModel().color()\"\n      [style.fill]=\"nodeModel().color()\"\n      (pointerStart)=\"pullNode(); selectNode()\"\n    />\n  }\n}\n\n<!-- Template group node  -->\n@if (nodeModel().node.type === \"template-group\" && groupNodeTemplate()) {\n  <svg:g class=\"selectable\" (pointerStart)=\"pullNode()\">\n    <ng-container\n      [ngTemplateOutlet]=\"groupNodeTemplate() ?? null\"\n      [ngTemplateOutletContext]=\"{\n        $implicit: {\n          node: nodeModel().node,\n          selected: nodeModel().selected,\n          width: nodeModel().width,\n          height: nodeModel().height,\n        },\n      }\"\n      [ngTemplateOutletInjector]=\"injector\"\n    />\n  </svg:g>\n}\n\n<!-- Resizer -->\n@if (nodeModel().resizerTemplate(); as template) {\n  @if (nodeModel().resizable()) {\n    <ng-template [ngTemplateOutlet]=\"template\" />\n  }\n}\n\n<!-- Handles -->\n@for (handle of nodeModel().handles(); track handle) {\n  @if (!handle.template) {\n    <svg:circle\n      class=\"default-handle\"\n      [attr.cx]=\"handle.offset().x\"\n      [attr.cy]=\"handle.offset().y\"\n      [attr.stroke-width]=\"handle.strokeWidth\"\n      r=\"5\"\n      (pointerStart)=\"startConnection($event, handle)\"\n      (pointerEnd)=\"endConnection(handle)\"\n    />\n  }\n\n  @if (handle.template) {\n    <svg:g\n      [handleSizeController]=\"handle\"\n      (pointerStart)=\"startConnection($event, handle)\"\n      (pointerEnd)=\"endConnection(handle)\"\n    >\n      <ng-container\n        *ngTemplateOutlet=\"handle.template; context: handle.templateContext\"\n      />\n    </svg:g>\n  }\n\n  @if (showMagnet()) {\n    <svg:circle\n      class=\"magnet\"\n      [attr.r]=\"nodeModel().magnetRadius\"\n      [attr.cx]=\"handle.offset().x\"\n      [attr.cy]=\"handle.offset().y\"\n      (pointerEnd)=\"endConnection(handle); resetValidateConnection(handle)\"\n      (pointerOver)=\"validateConnection(handle)\"\n      (pointerOut)=\"resetValidateConnection(handle)\"\n    />\n  }\n}\n\n<!-- Toolbar -->\n@if (toolbar(); as toolbar) {\n  <svg:foreignObject\n    [attr.width]=\"toolbar.size().width\"\n    [attr.height]=\"toolbar.size().height\"\n    [attr.transform]=\"toolbar.transform()\"\n  >\n    <ng-container [ngTemplateOutlet]=\"toolbar.template()\" />\n  </svg:foreignObject>\n}\n"]}
|