ngx-vflow 0.1.13 → 0.2.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 +1 -1
- package/esm2022/lib/vflow/components/connection/connection.component.mjs +27 -7
- package/esm2022/lib/vflow/components/handle/handle.component.mjs +49 -0
- package/esm2022/lib/vflow/components/node/node.component.mjs +26 -60
- package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +5 -8
- package/esm2022/lib/vflow/directives/connection-controller.directive.mjs +5 -3
- package/esm2022/lib/vflow/interfaces/connection.interface.mjs +1 -1
- package/esm2022/lib/vflow/models/edge.model.mjs +32 -5
- package/esm2022/lib/vflow/models/flow.model.mjs +3 -1
- package/esm2022/lib/vflow/models/handle.model.mjs +47 -0
- package/esm2022/lib/vflow/models/node.model.mjs +21 -60
- package/esm2022/lib/vflow/services/flow-status.service.mjs +7 -7
- package/esm2022/lib/vflow/services/handle.service.mjs +19 -0
- package/esm2022/lib/vflow/types/handle-type.type.mjs +2 -0
- package/esm2022/lib/vflow/vflow.module.mjs +7 -2
- package/esm2022/public-api.mjs +2 -1
- package/fesm2022/ngx-vflow.mjs +229 -143
- package/fesm2022/ngx-vflow.mjs.map +1 -1
- package/lib/vflow/components/handle/handle.component.d.ts +23 -0
- package/lib/vflow/components/node/node.component.d.ts +8 -25
- package/lib/vflow/components/vflow/vflow.component.d.ts +3 -4
- package/lib/vflow/interfaces/connection.interface.d.ts +2 -0
- package/lib/vflow/models/flow.model.d.ts +2 -0
- package/lib/vflow/models/handle.model.d.ts +24 -0
- package/lib/vflow/models/node.model.d.ts +4 -40
- package/lib/vflow/services/flow-status.service.d.ts +9 -3
- package/lib/vflow/services/handle.service.d.ts +22 -0
- package/lib/vflow/types/handle-type.type.d.ts +1 -0
- package/lib/vflow/vflow.module.d.ts +9 -8
- package/package.json +2 -2
- package/public-api.d.ts +1 -0
package/README.md
CHANGED
|
@@ -58,7 +58,7 @@ export class DefaultEdgesDemoComponent {
|
|
|
58
58
|
|
|
59
59
|
The code above renders to this:
|
|
60
60
|
|
|
61
|
-
<img width="
|
|
61
|
+
<img width="398" alt="image" src="https://github.com/artem-mangilev/ngx-vflow/assets/53087914/2a1b778a-2bfa-4176-9d50-065fdb1f1dec">
|
|
62
62
|
|
|
63
63
|
For more complex example you may see the documentation: https://www.ngx-vflow.org/
|
|
64
64
|
|
|
@@ -13,23 +13,31 @@ export class ConnectionComponent {
|
|
|
13
13
|
this.path = computed(() => {
|
|
14
14
|
const status = this.flowStatusService.status();
|
|
15
15
|
if (status.state === 'connection-start') {
|
|
16
|
-
const
|
|
17
|
-
const sourcePoint =
|
|
16
|
+
const sourceHandle = status.payload.sourceHandle;
|
|
17
|
+
const sourcePoint = sourceHandle.pointAbsolute();
|
|
18
|
+
const sourcePosition = sourceHandle.rawHandle.position;
|
|
18
19
|
const targetPoint = this.spacePointContext.svgCurrentSpacePoint();
|
|
20
|
+
const targetPosition = getOppositePostion(sourceHandle.rawHandle.position);
|
|
19
21
|
switch (this.model.curve) {
|
|
20
22
|
case 'straight': return straightPath(sourcePoint, targetPoint).path;
|
|
21
|
-
case 'bezier': return bezierPath(sourcePoint, targetPoint,
|
|
23
|
+
case 'bezier': return bezierPath(sourcePoint, targetPoint, sourcePosition, targetPosition).path;
|
|
22
24
|
}
|
|
23
25
|
}
|
|
24
26
|
if (status.state === 'connection-validation') {
|
|
25
|
-
const
|
|
27
|
+
const sourceHandle = status.payload.sourceHandle;
|
|
28
|
+
const sourcePoint = sourceHandle.pointAbsolute();
|
|
29
|
+
const sourcePosition = sourceHandle.rawHandle.position;
|
|
30
|
+
const targetHandle = status.payload.targetHandle;
|
|
26
31
|
// ignore magnet if validation failed
|
|
27
32
|
const targetPoint = status.payload.valid
|
|
28
|
-
?
|
|
33
|
+
? targetHandle.pointAbsolute()
|
|
29
34
|
: this.spacePointContext.svgCurrentSpacePoint();
|
|
35
|
+
const targetPosition = status.payload.valid
|
|
36
|
+
? targetHandle.rawHandle.position
|
|
37
|
+
: getOppositePostion(sourceHandle.rawHandle.position);
|
|
30
38
|
switch (this.model.curve) {
|
|
31
39
|
case 'straight': return straightPath(sourcePoint, targetPoint).path;
|
|
32
|
-
case 'bezier': return bezierPath(sourcePoint, targetPoint,
|
|
40
|
+
case 'bezier': return bezierPath(sourcePoint, targetPoint, sourcePosition, targetPosition).path;
|
|
33
41
|
}
|
|
34
42
|
}
|
|
35
43
|
return null;
|
|
@@ -97,4 +105,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
97
105
|
}], template: [{
|
|
98
106
|
type: Input
|
|
99
107
|
}] } });
|
|
100
|
-
|
|
108
|
+
function getOppositePostion(position) {
|
|
109
|
+
switch (position) {
|
|
110
|
+
case 'top':
|
|
111
|
+
return 'bottom';
|
|
112
|
+
case 'bottom':
|
|
113
|
+
return 'top';
|
|
114
|
+
case 'left':
|
|
115
|
+
return 'right';
|
|
116
|
+
case 'right':
|
|
117
|
+
return 'left';
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"connection.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/connection/connection.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, Input, TemplateRef, computed, inject } from '@angular/core';\nimport { FlowStatusService } from '../../services/flow-status.service';\nimport { straightPath } from '../../math/edge-path/straigh-path';\nimport { SpacePointContextDirective } from '../../directives/space-point-context.directive';\nimport { ConnectionModel } from '../../models/connection.model';\nimport { bezierPath } from '../../math/edge-path/bezier-path';\nimport { hashCode } from '../../utils/hash';\nimport { Position } from '../../types/position.type';\n\n@Component({\n  selector: 'g[connection]',\n  template: `\n    <ng-container *ngIf=\"model.type === 'default'\">\n      <svg:path\n        *ngIf=\"path() as path\"\n        [attr.d]=\"path\"\n        [attr.marker-end]=\"markerUrl()\"\n        [attr.stroke]=\"defaultColor\"\n        fill=\"none\"\n        stroke-width=\"2\"\n      />\n    </ng-container>\n\n    <ng-container *ngIf=\"model.type === 'template' && template\">\n      <ng-container *ngTemplateOutlet=\"template; context: getContext()\" />\n    </ng-container>\n  `,\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ConnectionComponent {\n  @Input({ required: true })\n  public model!: ConnectionModel\n\n  @Input()\n  public template?: TemplateRef<any>\n\n  private flowStatusService = inject(FlowStatusService)\n  private spacePointContext = inject(SpacePointContextDirective)\n\n  protected path = computed(() => {\n    const status = this.flowStatusService.status()\n\n    if (status.state === 'connection-start') {\n      const sourceHandle = status.payload.sourceHandle\n      const sourcePoint = sourceHandle.pointAbsolute()\n      const sourcePosition = sourceHandle.rawHandle.position\n\n      const targetPoint = this.spacePointContext.svgCurrentSpacePoint()\n      const targetPosition = getOppositePostion(sourceHandle.rawHandle.position)\n\n      switch (this.model.curve) {\n        case 'straight': return straightPath(sourcePoint, targetPoint).path\n        case 'bezier': return bezierPath(\n          sourcePoint, targetPoint,\n          sourcePosition, targetPosition\n        ).path\n      }\n    }\n\n    if (status.state === 'connection-validation') {\n      const sourceHandle = status.payload.sourceHandle\n      const sourcePoint = sourceHandle.pointAbsolute()\n      const sourcePosition = sourceHandle.rawHandle.position\n\n      const targetHandle = status.payload.targetHandle\n      // ignore magnet if validation failed\n      const targetPoint = status.payload.valid\n        ? targetHandle.pointAbsolute()\n        : this.spacePointContext.svgCurrentSpacePoint()\n      const targetPosition = status.payload.valid\n        ? targetHandle.rawHandle.position\n        : getOppositePostion(sourceHandle.rawHandle.position)\n\n      switch (this.model.curve) {\n        case 'straight': return straightPath(sourcePoint, targetPoint).path\n        case 'bezier': return bezierPath(\n          sourcePoint, targetPoint,\n          sourcePosition, targetPosition\n        ).path\n      }\n    }\n\n    return null\n  })\n\n  protected markerUrl = computed(() => {\n    const marker = this.model.connection.marker\n\n    if (marker) {\n      return `url(#${hashCode(JSON.stringify(marker))})`\n    }\n\n    return ''\n  })\n\n  protected readonly defaultColor = 'rgb(177, 177, 183)'\n\n  protected getContext() {\n    return {\n      $implicit: {\n        path: this.path,\n        marker: this.markerUrl\n      }\n    }\n  }\n}\n\nfunction getOppositePostion(position: Position): Position {\n  switch (position) {\n    case 'top':\n      return 'bottom'\n    case 'bottom':\n      return 'top'\n    case 'left':\n      return 'right'\n    case 'right':\n      return 'left'\n  }\n}\n"]}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { Component, ElementRef, Input, inject, signal } from '@angular/core';
|
|
2
|
+
import { HandleService } from '../../services/handle.service';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class HandleComponent {
|
|
5
|
+
constructor() {
|
|
6
|
+
this.handleService = inject(HandleService);
|
|
7
|
+
this.element = inject(ElementRef).nativeElement;
|
|
8
|
+
}
|
|
9
|
+
ngOnInit() {
|
|
10
|
+
queueMicrotask(() => {
|
|
11
|
+
const rect = this.parentRect();
|
|
12
|
+
this.handleService.createHandle({
|
|
13
|
+
position: this.position,
|
|
14
|
+
type: this.type,
|
|
15
|
+
id: this.id,
|
|
16
|
+
parentPosition: signal({ x: rect.x, y: rect.y }),
|
|
17
|
+
parentSize: signal({ width: rect.width, height: rect.height })
|
|
18
|
+
});
|
|
19
|
+
});
|
|
20
|
+
}
|
|
21
|
+
parentRect() {
|
|
22
|
+
// we assume there is only one foreignObject that wraps node
|
|
23
|
+
const fo = this.element.closest('foreignObject');
|
|
24
|
+
const parent = this.element.parentElement;
|
|
25
|
+
const foRect = fo.getBoundingClientRect();
|
|
26
|
+
const parentRect = parent.getBoundingClientRect();
|
|
27
|
+
return {
|
|
28
|
+
x: parentRect.left - foRect.left,
|
|
29
|
+
y: parentRect.top - foRect.top,
|
|
30
|
+
width: parentRect.width,
|
|
31
|
+
height: parentRect.height
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: HandleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
35
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: HandleComponent, selector: "handle", inputs: { position: "position", type: "type", id: "id" }, ngImport: i0, template: "" }); }
|
|
36
|
+
}
|
|
37
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: HandleComponent, decorators: [{
|
|
38
|
+
type: Component,
|
|
39
|
+
args: [{ selector: 'handle', template: "" }]
|
|
40
|
+
}], propDecorators: { position: [{
|
|
41
|
+
type: Input,
|
|
42
|
+
args: [{ required: true }]
|
|
43
|
+
}], type: [{
|
|
44
|
+
type: Input,
|
|
45
|
+
args: [{ required: true }]
|
|
46
|
+
}], id: [{
|
|
47
|
+
type: Input
|
|
48
|
+
}] } });
|
|
49
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGFuZGxlLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9jb21wb25lbnRzL2hhbmRsZS9oYW5kbGUuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LXZmbG93LWxpYi9zcmMvbGliL3ZmbG93L2NvbXBvbmVudHMvaGFuZGxlL2hhbmRsZS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQW1DLFNBQVMsRUFBRSxVQUFVLEVBQUUsS0FBSyxFQUFVLE1BQU0sRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFdEgsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLCtCQUErQixDQUFDOztBQU05RCxNQUFNLE9BQU8sZUFBZTtJQUo1QjtRQUtVLGtCQUFhLEdBQUcsTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFBO1FBQ3JDLFlBQU8sR0FBRyxNQUFNLENBQTBCLFVBQVUsQ0FBQyxDQUFDLGFBQWEsQ0FBQTtLQWtENUU7SUE5QlEsUUFBUTtRQUNiLGNBQWMsQ0FBQyxHQUFHLEVBQUU7WUFDbEIsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFBO1lBRTlCLElBQUksQ0FBQyxhQUFhLENBQUMsWUFBWSxDQUFDO2dCQUM5QixRQUFRLEVBQUUsSUFBSSxDQUFDLFFBQVE7Z0JBQ3ZCLElBQUksRUFBRSxJQUFJLENBQUMsSUFBSTtnQkFDZixFQUFFLEVBQUUsSUFBSSxDQUFDLEVBQUU7Z0JBQ1gsY0FBYyxFQUFFLE1BQU0sQ0FBQyxFQUFFLENBQUMsRUFBRSxJQUFJLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxJQUFJLENBQUMsQ0FBQyxFQUFFLENBQUM7Z0JBQ2hELFVBQVUsRUFBRSxNQUFNLENBQUMsRUFBRSxLQUFLLEVBQUUsSUFBSSxDQUFDLEtBQUssRUFBRSxNQUFNLEVBQUUsSUFBSSxDQUFDLE1BQU0sRUFBRSxDQUFDO2FBQy9ELENBQUMsQ0FBQTtRQUNKLENBQUMsQ0FBQyxDQUFBO0lBQ0osQ0FBQztJQUdPLFVBQVU7UUFDaEIsNERBQTREO1FBQzVELE1BQU0sRUFBRSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxDQUFDLGVBQWUsQ0FBQyxDQUFBO1FBQ2hELE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsYUFBYyxDQUFBO1FBRTFDLE1BQU0sTUFBTSxHQUFHLEVBQUcsQ0FBQyxxQkFBcUIsRUFBRSxDQUFBO1FBQzFDLE1BQU0sVUFBVSxHQUFHLE1BQU0sQ0FBQyxxQkFBcUIsRUFBRSxDQUFBO1FBRWpELE9BQU87WUFDTCxDQUFDLEVBQUUsVUFBVSxDQUFDLElBQUksR0FBRyxNQUFNLENBQUMsSUFBSTtZQUNoQyxDQUFDLEVBQUUsVUFBVSxDQUFDLEdBQUcsR0FBRyxNQUFNLENBQUMsR0FBRztZQUM5QixLQUFLLEVBQUUsVUFBVSxDQUFDLEtBQUs7WUFDdkIsTUFBTSxFQUFFLFVBQVUsQ0FBQyxNQUFNO1NBQzFCLENBQUE7SUFDSCxDQUFDOytHQW5EVSxlQUFlO21HQUFmLGVBQWUsd0dDUjVCLEVBQUE7OzRGRFFhLGVBQWU7a0JBSjNCLFNBQVM7K0JBQ0UsUUFBUTs4QkFXWCxRQUFRO3NCQURkLEtBQUs7dUJBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFO2dCQU9sQixJQUFJO3NCQURWLEtBQUs7dUJBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFO2dCQU9sQixFQUFFO3NCQURSLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBZnRlckNvbnRlbnRJbml0LCBBZnRlclZpZXdJbml0LCBDb21wb25lbnQsIEVsZW1lbnRSZWYsIElucHV0LCBPbkluaXQsIGluamVjdCwgc2lnbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBQb3NpdGlvbiB9IGZyb20gJy4uLy4uL3R5cGVzL3Bvc2l0aW9uLnR5cGUnO1xuaW1wb3J0IHsgSGFuZGxlU2VydmljZSB9IGZyb20gJy4uLy4uL3NlcnZpY2VzL2hhbmRsZS5zZXJ2aWNlJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnaGFuZGxlJyxcbiAgdGVtcGxhdGVVcmw6ICcuL2hhbmRsZS5jb21wb25lbnQuaHRtbCdcbn0pXG5leHBvcnQgY2xhc3MgSGFuZGxlQ29tcG9uZW50IGltcGxlbWVudHMgT25Jbml0IHtcbiAgcHJpdmF0ZSBoYW5kbGVTZXJ2aWNlID0gaW5qZWN0KEhhbmRsZVNlcnZpY2UpXG4gIHByaXZhdGUgZWxlbWVudCA9IGluamVjdDxFbGVtZW50UmVmPEhUTUxFbGVtZW50Pj4oRWxlbWVudFJlZikubmF0aXZlRWxlbWVudFxuXG4gIC8qKlxuICAgKiBBdCB3aGF0IHNpZGUgb2Ygbm9kZSB0aGlzIGNvbXBvbmVudCBzaG91bGQgYmUgcGxhY2VkXG4gICAqL1xuICBASW5wdXQoeyByZXF1aXJlZDogdHJ1ZSB9KVxuICBwdWJsaWMgcG9zaXRpb24hOiBQb3NpdGlvblxuXG4gIC8qKlxuICAgKiBTb3VyY2Ugb3IgdGFyZ2V0XG4gICAqL1xuICBASW5wdXQoeyByZXF1aXJlZDogdHJ1ZSB9KVxuICBwdWJsaWMgdHlwZSE6ICdzb3VyY2UnIHwgJ3RhcmdldCdcblxuICAvKipcbiAgICogU2hvdWxkIGJlIHVzZWQgaWYgbm9kZSBoYXMgbW9yZSB0aGFuIG9uZSBzb3VyY2UvdGFyZ2V0XG4gICAqL1xuICBASW5wdXQoKVxuICBwdWJsaWMgaWQ/OiBzdHJpbmdcblxuICBwdWJsaWMgbmdPbkluaXQoKSB7XG4gICAgcXVldWVNaWNyb3Rhc2soKCkgPT4ge1xuICAgICAgY29uc3QgcmVjdCA9IHRoaXMucGFyZW50UmVjdCgpXG5cbiAgICAgIHRoaXMuaGFuZGxlU2VydmljZS5jcmVhdGVIYW5kbGUoe1xuICAgICAgICBwb3NpdGlvbjogdGhpcy5wb3NpdGlvbixcbiAgICAgICAgdHlwZTogdGhpcy50eXBlLFxuICAgICAgICBpZDogdGhpcy5pZCxcbiAgICAgICAgcGFyZW50UG9zaXRpb246IHNpZ25hbCh7IHg6IHJlY3QueCwgeTogcmVjdC55IH0pLFxuICAgICAgICBwYXJlbnRTaXplOiBzaWduYWwoeyB3aWR0aDogcmVjdC53aWR0aCwgaGVpZ2h0OiByZWN0LmhlaWdodCB9KVxuICAgICAgfSlcbiAgICB9KVxuICB9XG5cblxuICBwcml2YXRlIHBhcmVudFJlY3QoKSB7XG4gICAgLy8gd2UgYXNzdW1lIHRoZXJlIGlzIG9ubHkgb25lIGZvcmVpZ25PYmplY3QgdGhhdCB3cmFwcyBub2RlXG4gICAgY29uc3QgZm8gPSB0aGlzLmVsZW1lbnQuY2xvc2VzdCgnZm9yZWlnbk9iamVjdCcpXG4gICAgY29uc3QgcGFyZW50ID0gdGhpcy5lbGVtZW50LnBhcmVudEVsZW1lbnQhXG5cbiAgICBjb25zdCBmb1JlY3QgPSBmbyEuZ2V0Qm91bmRpbmdDbGllbnRSZWN0KClcbiAgICBjb25zdCBwYXJlbnRSZWN0ID0gcGFyZW50LmdldEJvdW5kaW5nQ2xpZW50UmVjdCgpXG5cbiAgICByZXR1cm4ge1xuICAgICAgeDogcGFyZW50UmVjdC5sZWZ0IC0gZm9SZWN0LmxlZnQsXG4gICAgICB5OiBwYXJlbnRSZWN0LnRvcCAtIGZvUmVjdC50b3AsXG4gICAgICB3aWR0aDogcGFyZW50UmVjdC53aWR0aCxcbiAgICAgIGhlaWdodDogcGFyZW50UmVjdC5oZWlnaHRcbiAgICB9XG4gIH1cbn1cbiIsIiJdfQ==
|
|
@@ -1,24 +1,27 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, ElementRef, Input, ViewChild, computed, inject, signal } from '@angular/core';
|
|
1
|
+
import { ChangeDetectionStrategy, Component, ElementRef, Injector, Input, ViewChild, computed, effect, inject, runInInjectionContext, signal } from '@angular/core';
|
|
2
2
|
import { DraggableService } from '../../services/draggable.service';
|
|
3
3
|
import { FlowStatusService, batchStatusChanges } from '../../services/flow-status.service';
|
|
4
4
|
import { FlowEntitiesService } from '../../services/flow-entities.service';
|
|
5
|
+
import { HandleService } from '../../services/handle.service';
|
|
5
6
|
import * as i0 from "@angular/core";
|
|
6
7
|
import * as i1 from "@angular/common";
|
|
7
8
|
export class NodeComponent {
|
|
8
9
|
constructor() {
|
|
10
|
+
this.handleService = inject(HandleService);
|
|
11
|
+
this.injector = inject(Injector);
|
|
9
12
|
this.draggableService = inject(DraggableService);
|
|
10
13
|
this.flowStatusService = inject(FlowStatusService);
|
|
11
14
|
this.flowEntitiesService = inject(FlowEntitiesService);
|
|
12
15
|
this.hostRef = inject(ElementRef);
|
|
13
16
|
this.showMagnet = computed(() => this.flowStatusService.status().state === 'connection-start' ||
|
|
14
17
|
this.flowStatusService.status().state === 'connection-validation');
|
|
15
|
-
this.defaultHandleStrokeWidth = 2;
|
|
16
18
|
this.sourceHanldeState = signal('idle');
|
|
17
19
|
this.targetHandleState = signal('idle');
|
|
18
|
-
this.sourceHanldeStateReadonly = this.sourceHanldeState.asReadonly();
|
|
19
|
-
this.targetHanldeStateReadonly = this.targetHandleState.asReadonly();
|
|
20
20
|
}
|
|
21
21
|
ngOnInit() {
|
|
22
|
+
runInInjectionContext(this.injector, () => {
|
|
23
|
+
effect(() => this.nodeModel.rawHandles.set(this.handleService.handles()), { allowSignalWrites: true });
|
|
24
|
+
});
|
|
22
25
|
this.draggableService.toggleDraggable(this.hostRef.nativeElement, this.nodeModel);
|
|
23
26
|
}
|
|
24
27
|
ngAfterViewInit() {
|
|
@@ -33,26 +36,26 @@ export class NodeComponent {
|
|
|
33
36
|
const height = this.htmlWrapperRef.nativeElement.clientHeight;
|
|
34
37
|
this.nodeModel.size.set({ width, height });
|
|
35
38
|
}
|
|
36
|
-
this.setSourceHandleSize();
|
|
37
|
-
this.setTargetHandleSize();
|
|
38
39
|
});
|
|
39
40
|
}
|
|
40
41
|
ngOnDestroy() {
|
|
41
42
|
this.draggableService.destroy(this.hostRef.nativeElement);
|
|
42
43
|
}
|
|
43
|
-
startConnection(event) {
|
|
44
|
+
startConnection(event, handle) {
|
|
44
45
|
// ignore drag by stopping propagation
|
|
45
46
|
event.stopPropagation();
|
|
46
|
-
this.flowStatusService.setConnectionStartStatus(this.nodeModel);
|
|
47
|
+
this.flowStatusService.setConnectionStartStatus(this.nodeModel, handle);
|
|
47
48
|
}
|
|
48
49
|
endConnection() {
|
|
49
50
|
const status = this.flowStatusService.status();
|
|
50
51
|
if (status.state === 'connection-validation') {
|
|
51
52
|
const sourceNode = status.payload.sourceNode;
|
|
52
53
|
const targetNode = this.nodeModel;
|
|
54
|
+
const sourceHandle = status.payload.sourceHandle;
|
|
55
|
+
const targetHandle = status.payload.targetHandle;
|
|
53
56
|
batchStatusChanges(
|
|
54
57
|
// call to create connection
|
|
55
|
-
() => this.flowStatusService.setConnectionEndStatus(sourceNode, targetNode),
|
|
58
|
+
() => this.flowStatusService.setConnectionEndStatus(sourceNode, targetNode, sourceHandle, targetHandle),
|
|
56
59
|
// when connection created, we need go back to idle status
|
|
57
60
|
() => this.flowStatusService.setIdleStatus());
|
|
58
61
|
}
|
|
@@ -60,16 +63,22 @@ export class NodeComponent {
|
|
|
60
63
|
/**
|
|
61
64
|
* TODO srp
|
|
62
65
|
*/
|
|
63
|
-
validateTargetHandle() {
|
|
66
|
+
validateTargetHandle(targetHandle) {
|
|
64
67
|
const status = this.flowStatusService.status();
|
|
65
68
|
if (status.state === 'connection-start') {
|
|
66
69
|
const sourceNode = status.payload.sourceNode;
|
|
67
|
-
const
|
|
70
|
+
const sourceHandle = status.payload.sourceHandle;
|
|
68
71
|
const source = sourceNode.node.id;
|
|
72
|
+
const targetNode = this.nodeModel;
|
|
69
73
|
const target = targetNode.node.id;
|
|
70
|
-
const valid = this.flowEntitiesService.connection().validator({
|
|
74
|
+
const valid = this.flowEntitiesService.connection().validator({
|
|
75
|
+
source,
|
|
76
|
+
target,
|
|
77
|
+
sourceHandle: sourceHandle.rawHandle.id,
|
|
78
|
+
targetHandle: targetHandle.rawHandle.id
|
|
79
|
+
});
|
|
71
80
|
this.targetHandleState.set(valid ? 'valid' : 'invalid');
|
|
72
|
-
this.flowStatusService.setConnectionValidationStatus(sourceNode, targetNode,
|
|
81
|
+
this.flowStatusService.setConnectionValidationStatus(valid, sourceNode, targetNode, sourceHandle, targetHandle);
|
|
73
82
|
}
|
|
74
83
|
}
|
|
75
84
|
/**
|
|
@@ -80,67 +89,24 @@ export class NodeComponent {
|
|
|
80
89
|
// drop back to start status
|
|
81
90
|
const status = this.flowStatusService.status();
|
|
82
91
|
if (status.state === 'connection-validation') {
|
|
83
|
-
this.flowStatusService.setConnectionStartStatus(status.payload.sourceNode);
|
|
92
|
+
this.flowStatusService.setConnectionStartStatus(status.payload.sourceNode, status.payload.sourceHandle);
|
|
84
93
|
}
|
|
85
94
|
}
|
|
86
|
-
getHandleContext(type) {
|
|
87
|
-
if (type === 'source') {
|
|
88
|
-
return {
|
|
89
|
-
$implicit: {
|
|
90
|
-
point: this.nodeModel.sourceOffset,
|
|
91
|
-
alignedPoint: this.nodeModel.sourceOffsetAligned,
|
|
92
|
-
state: this.sourceHanldeStateReadonly
|
|
93
|
-
}
|
|
94
|
-
};
|
|
95
|
-
}
|
|
96
|
-
return {
|
|
97
|
-
$implicit: {
|
|
98
|
-
point: this.nodeModel.targetOffset,
|
|
99
|
-
alignedPoint: this.nodeModel.targetOffsetAligned,
|
|
100
|
-
state: this.targetHanldeStateReadonly
|
|
101
|
-
}
|
|
102
|
-
};
|
|
103
|
-
}
|
|
104
|
-
setSourceHandleSize() {
|
|
105
|
-
// if handle template provided, we don't know its stroke so it's 0
|
|
106
|
-
const strokeWidth = this.handleTemplate ? 0 : (2 * this.defaultHandleStrokeWidth);
|
|
107
|
-
const sourceBox = this.sourceHandleRef.nativeElement.getBBox({ stroke: true });
|
|
108
|
-
this.nodeModel.sourceHandleSize.set({
|
|
109
|
-
width: sourceBox.width + strokeWidth,
|
|
110
|
-
height: sourceBox.height + strokeWidth
|
|
111
|
-
});
|
|
112
|
-
}
|
|
113
|
-
setTargetHandleSize() {
|
|
114
|
-
const strokeWidth = this.handleTemplate ? 0 : (2 * this.defaultHandleStrokeWidth);
|
|
115
|
-
const targetBox = this.targetHandleRef.nativeElement.getBBox({ stroke: true });
|
|
116
|
-
this.nodeModel.targetHandleSize.set({
|
|
117
|
-
width: targetBox.width + strokeWidth,
|
|
118
|
-
height: targetBox.height + strokeWidth
|
|
119
|
-
});
|
|
120
|
-
}
|
|
121
95
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NodeComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
122
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: NodeComponent, selector: "g[node]", inputs: { nodeModel: "nodeModel", nodeHtmlTemplate: "nodeHtmlTemplate",
|
|
96
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: NodeComponent, selector: "g[node]", inputs: { nodeModel: "nodeModel", nodeHtmlTemplate: "nodeHtmlTemplate" }, providers: [HandleService], viewQueries: [{ propertyName: "nodeContentRef", first: true, predicate: ["nodeContent"], descendants: true }, { propertyName: "htmlWrapperRef", first: true, predicate: ["htmlWrapper"], descendants: true }], ngImport: i0, template: "<svg:foreignObject\n *ngIf=\"nodeModel.node.type === 'default'\"\n #nodeContent\n width=\"100\"\n height=\"50\"\n>\n <div class=\"default-node\" [innerHTML]=\"nodeModel.node.text ?? ''\"></div>\n</svg:foreignObject>\n\n<svg:foreignObject\n *ngIf=\"nodeModel.node.type === 'html-template' && nodeHtmlTemplate\"\n [attr.width]=\"nodeModel.size().width\"\n [attr.height]=\"nodeModel.size().height\"\n>\n <div #htmlWrapper class=\"wrapper\">\n <ng-container\n [ngTemplateOutlet]=\"nodeHtmlTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: { node: nodeModel.node } }\"\n [ngTemplateOutletInjector]=\"injector\"\n />\n </div>\n</svg:foreignObject>\n\n<ng-container *ngFor=\"let handle of nodeModel.handles()\">\n <svg:circle\n #sourceHandle\n [attr.cx]=\"handle.offset().x\"\n [attr.cy]=\"handle.offset().y\"\n [attr.stroke-width]=\"handle.strokeWidth\"\n r=\"5\"\n stroke=\"white\"\n fill=\"black\"\n (mousedown)=\"handle.rawHandle.type === 'source' ? startConnection($event, handle) : null\"\n (mouseup)=\"handle.rawHandle.type === 'target' ? endConnection() : null\"\n />\n\n <svg:circle\n *ngIf=\"handle.rawHandle.type === 'target' && showMagnet()\"\n class=\"magnet\"\n [attr.r]=\"nodeModel.magnetRadius\"\n [attr.cx]=\"handle.offset().x\"\n [attr.cy]=\"handle.offset().y\"\n (mouseup)=\"endConnection(); resetValidateTargetHandle()\"\n (mouseover)=\"validateTargetHandle(handle)\"\n (mouseout)=\"resetValidateTargetHandle()\"\n />\n</ng-container>\n\n", styles: [".wrapper{width:max-content}.magnet{opacity:0}.default-node{max-width:100px;max-height:100px;width:100px;height:50px;border:2px solid black;border-radius:5px;display:flex;align-items:center;justify-content:center;color:#000;background-color:#fff}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
123
97
|
}
|
|
124
98
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NodeComponent, decorators: [{
|
|
125
99
|
type: Component,
|
|
126
|
-
args: [{ selector: 'g[node]', changeDetection: ChangeDetectionStrategy.OnPush, template: "<svg:foreignObject\n *ngIf=\"nodeModel.node.type === 'default'\"\n #nodeContent\n width=\"100\"\n height=\"50\"\n>\n <div class=\"default-node\" [innerHTML]=\"nodeModel.node.text ?? ''\"></div>\n</svg:foreignObject>\n\n<svg:foreignObject\n *ngIf=\"nodeModel.node.type === 'html-template' && nodeHtmlTemplate\"\n [attr.width]=\"nodeModel.size().width\"\n [attr.height]=\"nodeModel.size().height\"\n>\n <div #htmlWrapper class=\"wrapper\">\n <ng-container\n
|
|
100
|
+
args: [{ selector: 'g[node]', changeDetection: ChangeDetectionStrategy.OnPush, providers: [HandleService], template: "<svg:foreignObject\n *ngIf=\"nodeModel.node.type === 'default'\"\n #nodeContent\n width=\"100\"\n height=\"50\"\n>\n <div class=\"default-node\" [innerHTML]=\"nodeModel.node.text ?? ''\"></div>\n</svg:foreignObject>\n\n<svg:foreignObject\n *ngIf=\"nodeModel.node.type === 'html-template' && nodeHtmlTemplate\"\n [attr.width]=\"nodeModel.size().width\"\n [attr.height]=\"nodeModel.size().height\"\n>\n <div #htmlWrapper class=\"wrapper\">\n <ng-container\n [ngTemplateOutlet]=\"nodeHtmlTemplate\"\n [ngTemplateOutletContext]=\"{ $implicit: { node: nodeModel.node } }\"\n [ngTemplateOutletInjector]=\"injector\"\n />\n </div>\n</svg:foreignObject>\n\n<ng-container *ngFor=\"let handle of nodeModel.handles()\">\n <svg:circle\n #sourceHandle\n [attr.cx]=\"handle.offset().x\"\n [attr.cy]=\"handle.offset().y\"\n [attr.stroke-width]=\"handle.strokeWidth\"\n r=\"5\"\n stroke=\"white\"\n fill=\"black\"\n (mousedown)=\"handle.rawHandle.type === 'source' ? startConnection($event, handle) : null\"\n (mouseup)=\"handle.rawHandle.type === 'target' ? endConnection() : null\"\n />\n\n <svg:circle\n *ngIf=\"handle.rawHandle.type === 'target' && showMagnet()\"\n class=\"magnet\"\n [attr.r]=\"nodeModel.magnetRadius\"\n [attr.cx]=\"handle.offset().x\"\n [attr.cy]=\"handle.offset().y\"\n (mouseup)=\"endConnection(); resetValidateTargetHandle()\"\n (mouseover)=\"validateTargetHandle(handle)\"\n (mouseout)=\"resetValidateTargetHandle()\"\n />\n</ng-container>\n\n", styles: [".wrapper{width:max-content}.magnet{opacity:0}.default-node{max-width:100px;max-height:100px;width:100px;height:50px;border:2px solid black;border-radius:5px;display:flex;align-items:center;justify-content:center;color:#000;background-color:#fff}\n"] }]
|
|
127
101
|
}], propDecorators: { nodeModel: [{
|
|
128
102
|
type: Input
|
|
129
103
|
}], nodeHtmlTemplate: [{
|
|
130
104
|
type: Input
|
|
131
|
-
}], handleTemplate: [{
|
|
132
|
-
type: Input
|
|
133
105
|
}], nodeContentRef: [{
|
|
134
106
|
type: ViewChild,
|
|
135
107
|
args: ['nodeContent']
|
|
136
108
|
}], htmlWrapperRef: [{
|
|
137
109
|
type: ViewChild,
|
|
138
110
|
args: ['htmlWrapper']
|
|
139
|
-
}], sourceHandleRef: [{
|
|
140
|
-
type: ViewChild,
|
|
141
|
-
args: ['sourceHandle']
|
|
142
|
-
}], targetHandleRef: [{
|
|
143
|
-
type: ViewChild,
|
|
144
|
-
args: ['targetHandle']
|
|
145
111
|
}] } });
|
|
146
|
-
//# 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, HostBinding, Input, NgZone, OnDestroy, OnInit, TemplateRef, ViewChild, computed, inject, signal } from '@angular/core';\nimport { DraggableService } from '../../services/draggable.service';\nimport { NodeModel } from '../../models/node.model';\nimport { FlowStatusService, batchStatusChanges } from '../../services/flow-status.service';\nimport { FlowEntitiesService } from '../../services/flow-entities.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})\nexport class NodeComponent implements OnInit, AfterViewInit, OnDestroy {\n  @Input()\n  public nodeModel!: NodeModel\n\n  @Input()\n  public nodeHtmlTemplate?: TemplateRef<any>\n\n  @Input()\n  public handleTemplate?: TemplateRef<any>\n\n  @ViewChild('nodeContent')\n  public nodeContentRef!: ElementRef<SVGGraphicsElement>\n\n  @ViewChild('htmlWrapper')\n  public htmlWrapperRef!: ElementRef<HTMLDivElement>\n\n  @ViewChild('sourceHandle')\n  public sourceHandleRef!: ElementRef<SVGGElement | SVGCircleElement>\n\n  @ViewChild('targetHandle')\n  public targetHandleRef!: ElementRef<SVGGElement | SVGCircleElement>\n\n  private draggableService = inject(DraggableService)\n  private flowStatusService = inject(FlowStatusService)\n  private flowEntitiesService = inject(FlowEntitiesService)\n  private hostRef = inject<ElementRef<SVGElement>>(ElementRef)\n\n  protected showMagnet = computed(() =>\n    this.flowStatusService.status().state === 'connection-start' ||\n    this.flowStatusService.status().state === 'connection-validation'\n  )\n\n  protected readonly defaultHandleStrokeWidth = 2;\n\n  private sourceHanldeState = signal<HandleState>('idle')\n  private targetHandleState = signal<HandleState>('idle')\n\n  private sourceHanldeStateReadonly = this.sourceHanldeState.asReadonly()\n  private targetHanldeStateReadonly = this.targetHandleState.asReadonly()\n\n  public ngOnInit() {\n    this.draggableService.toggleDraggable(this.hostRef.nativeElement, this.nodeModel)\n  }\n\n  public ngAfterViewInit(): void {\n    // TODO remove microtask\n    queueMicrotask(() => {\n      if (this.nodeModel.node.type === 'default') {\n        const { width, height } = this.nodeContentRef.nativeElement.getBBox()\n        this.nodeModel.size.set({ width, height })\n      }\n\n      if (this.nodeModel.node.type === 'html-template') {\n        const width = this.htmlWrapperRef.nativeElement.clientWidth\n        const height = this.htmlWrapperRef.nativeElement.clientHeight\n\n        this.nodeModel.size.set({ width, height })\n      }\n\n      this.setSourceHandleSize()\n      this.setTargetHandleSize()\n    })\n  }\n\n  public ngOnDestroy(): void {\n    this.draggableService.destroy(this.hostRef.nativeElement)\n  }\n\n  protected startConnection(event: MouseEvent) {\n    // ignore drag by stopping propagation\n    event.stopPropagation()\n\n    this.flowStatusService.setConnectionStartStatus(this.nodeModel)\n  }\n\n  protected endConnection() {\n    const status = this.flowStatusService.status()\n\n    if (status.state === 'connection-validation') {\n      const sourceNode = status.payload.sourceNode\n      const targetNode = this.nodeModel\n\n      batchStatusChanges(\n        // call to create connection\n        () => this.flowStatusService.setConnectionEndStatus(sourceNode, targetNode),\n        // when connection created, we need go back to idle status\n        () => this.flowStatusService.setIdleStatus()\n      )\n    }\n  }\n\n  /**\n   * TODO srp\n   */\n  protected validateTargetHandle() {\n    const status = this.flowStatusService.status()\n\n    if (status.state === 'connection-start') {\n      const sourceNode = status.payload.sourceNode\n      const targetNode = this.nodeModel\n\n      const source = sourceNode.node.id\n      const target = targetNode.node.id\n\n      const valid = this.flowEntitiesService.connection().validator({ source, target })\n      this.targetHandleState.set(valid ? 'valid' : 'invalid')\n\n      this.flowStatusService.setConnectionValidationStatus(sourceNode, targetNode, valid)\n    }\n  }\n\n  /**\n   * TODO srp\n   */\n  protected resetValidateTargetHandle() {\n    this.targetHandleState.set('idle')\n\n    // drop back to start status\n    const status = this.flowStatusService.status()\n    if (status.state === 'connection-validation') {\n      this.flowStatusService.setConnectionStartStatus(status.payload.sourceNode)\n    }\n  }\n\n  protected getHandleContext(type: 'source' | 'target') {\n    if (type === 'source') {\n      return {\n        $implicit: {\n          point: this.nodeModel.sourceOffset,\n          alignedPoint: this.nodeModel.sourceOffsetAligned,\n          state: this.sourceHanldeStateReadonly\n        }\n      }\n    }\n\n    return {\n      $implicit: {\n        point: this.nodeModel.targetOffset,\n        alignedPoint: this.nodeModel.targetOffsetAligned,\n        state: this.targetHanldeStateReadonly\n      }\n    }\n  }\n\n  private setSourceHandleSize() {\n    // if handle template provided, we don't know its stroke so it's 0\n    const strokeWidth = this.handleTemplate ? 0 : (2 * this.defaultHandleStrokeWidth);\n\n    const sourceBox = this.sourceHandleRef.nativeElement.getBBox({ stroke: true })\n    this.nodeModel.sourceHandleSize.set({\n      width: sourceBox.width + strokeWidth,\n      height: sourceBox.height + strokeWidth\n    })\n  }\n\n  private setTargetHandleSize() {\n    const strokeWidth = this.handleTemplate ? 0 : (2 * this.defaultHandleStrokeWidth);\n\n    const targetBox = this.targetHandleRef.nativeElement.getBBox({ stroke: true })\n    this.nodeModel.targetHandleSize.set({\n      width: targetBox.width + strokeWidth,\n      height: targetBox.height + strokeWidth\n    })\n  }\n}\n","<svg:foreignObject\n  *ngIf=\"nodeModel.node.type === 'default'\"\n  #nodeContent\n  width=\"100\"\n  height=\"50\"\n>\n  <div class=\"default-node\" [innerHTML]=\"nodeModel.node.text ?? ''\"></div>\n</svg:foreignObject>\n\n<svg:foreignObject\n  *ngIf=\"nodeModel.node.type === 'html-template' && nodeHtmlTemplate\"\n  [attr.width]=\"nodeModel.size().width\"\n  [attr.height]=\"nodeModel.size().height\"\n>\n  <div #htmlWrapper class=\"wrapper\">\n    <ng-container\n      *ngTemplateOutlet=\"nodeHtmlTemplate; context: { $implicit: { node: nodeModel.node }  }\"\n    />\n  </div>\n</svg:foreignObject>\n\n<ng-container *ngIf=\"handleTemplate\">\n  <svg:g #sourceHandle (mousedown)=\"startConnection($event)\">\n    <ng-container *ngTemplateOutlet=\"handleTemplate; context: getHandleContext('source')\" />\n  </svg:g>\n\n  <svg:g #targetHandle>\n    <ng-container *ngTemplateOutlet=\"handleTemplate; context: getHandleContext('target')\" />\n  </svg:g>\n</ng-container>\n\n<ng-container *ngIf=\"!handleTemplate\">\n  <svg:circle\n    #sourceHandle\n    [attr.cx]=\"nodeModel.sourceOffset().x\"\n    [attr.cy]=\"nodeModel.sourceOffset().y\"\n    [attr.stroke-width]=\"defaultHandleStrokeWidth\"\n    r=\"5\"\n    stroke=\"white\"\n    fill=\"black\"\n    (mousedown)=\"startConnection($event)\"\n  />\n\n  <svg:circle\n    #targetHandle\n    [attr.cx]=\"nodeModel.targetOffset().x\"\n    [attr.cy]=\"nodeModel.targetOffset().y\"\n    [attr.stroke-width]=\"defaultHandleStrokeWidth\"\n    r=\"5\"\n    stroke=\"white\"\n    fill=\"black\"\n    (mouseup)=\"endConnection()\"\n  />\n</ng-container>\n\n<svg:circle\n  *ngIf=\"showMagnet()\"\n  class=\"magnet\"\n  [attr.r]=\"nodeModel.magnetRadius\"\n  [attr.cx]=\"nodeModel.targetOffset().x\"\n  [attr.cy]=\"nodeModel.targetOffset().y\"\n  (mouseup)=\"endConnection(); resetValidateTargetHandle()\"\n  (mouseover)=\"validateTargetHandle()\"\n  (mouseout)=\"resetValidateTargetHandle()\"\n/>\n\n\n"]}
|
|
112
|
+
//# 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, HostBinding, Injector, Input, NgZone, OnDestroy, OnInit, TemplateRef, ViewChild, computed, effect, inject, runInInjectionContext, signal } from '@angular/core';\nimport { DraggableService } from '../../services/draggable.service';\nimport { NodeModel } from '../../models/node.model';\nimport { FlowStatusService, batchStatusChanges } from '../../services/flow-status.service';\nimport { FlowEntitiesService } from '../../services/flow-entities.service';\nimport { HandleService } from '../../services/handle.service';\nimport { HandleModel } from '../../models/handle.model';\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]\n})\nexport class NodeComponent implements OnInit, AfterViewInit, OnDestroy {\n  protected handleService = inject(HandleService)\n  protected injector = inject(Injector)\n\n  @Input()\n  public nodeModel!: NodeModel\n\n  @Input()\n  public nodeHtmlTemplate?: TemplateRef<any>\n\n  @ViewChild('nodeContent')\n  public nodeContentRef!: ElementRef<SVGGraphicsElement>\n\n  @ViewChild('htmlWrapper')\n  public htmlWrapperRef!: ElementRef<HTMLDivElement>\n\n  private draggableService = inject(DraggableService)\n  private flowStatusService = inject(FlowStatusService)\n  private flowEntitiesService = inject(FlowEntitiesService)\n  private hostRef = inject<ElementRef<SVGElement>>(ElementRef)\n\n  protected showMagnet = computed(() =>\n    this.flowStatusService.status().state === 'connection-start' ||\n    this.flowStatusService.status().state === 'connection-validation'\n  )\n\n  private sourceHanldeState = signal<HandleState>('idle')\n  private targetHandleState = signal<HandleState>('idle')\n\n  public ngOnInit() {\n    runInInjectionContext(this.injector, () => {\n      effect(\n        () => this.nodeModel.rawHandles.set(this.handleService.handles()),\n        { allowSignalWrites: true }\n      )\n    })\n\n    this.draggableService.toggleDraggable(this.hostRef.nativeElement, this.nodeModel)\n  }\n\n  public ngAfterViewInit(): void {\n    // TODO remove microtask\n    queueMicrotask(() => {\n      if (this.nodeModel.node.type === 'default') {\n        const { width, height } = this.nodeContentRef.nativeElement.getBBox()\n        this.nodeModel.size.set({ width, height })\n      }\n\n      if (this.nodeModel.node.type === 'html-template') {\n        const width = this.htmlWrapperRef.nativeElement.clientWidth\n        const height = this.htmlWrapperRef.nativeElement.clientHeight\n\n        this.nodeModel.size.set({ width, height })\n      }\n    })\n  }\n\n  public ngOnDestroy(): void {\n    this.draggableService.destroy(this.hostRef.nativeElement)\n  }\n\n  protected startConnection(event: MouseEvent, handle: HandleModel) {\n    // ignore drag by stopping propagation\n    event.stopPropagation()\n\n    this.flowStatusService.setConnectionStartStatus(this.nodeModel, handle)\n  }\n\n  protected endConnection() {\n    const status = this.flowStatusService.status()\n\n    if (status.state === 'connection-validation') {\n      const sourceNode = status.payload.sourceNode\n      const targetNode = this.nodeModel\n      const sourceHandle = status.payload.sourceHandle\n      const targetHandle = status.payload.targetHandle\n\n      batchStatusChanges(\n        // call to create connection\n        () => this.flowStatusService.setConnectionEndStatus(sourceNode, targetNode, sourceHandle, targetHandle),\n        // when connection created, we need go back to idle status\n        () => this.flowStatusService.setIdleStatus()\n      )\n    }\n  }\n\n  /**\n   * TODO srp\n   */\n  protected validateTargetHandle(targetHandle: HandleModel) {\n    const status = this.flowStatusService.status()\n\n    if (status.state === 'connection-start') {\n      const sourceNode = status.payload.sourceNode\n      const sourceHandle = status.payload.sourceHandle\n      const source = sourceNode.node.id\n\n      const targetNode = this.nodeModel\n      const target = targetNode.node.id\n\n      const valid = this.flowEntitiesService.connection().validator({\n        source,\n        target,\n        sourceHandle: sourceHandle.rawHandle.id,\n        targetHandle: targetHandle.rawHandle.id\n      })\n      this.targetHandleState.set(valid ? 'valid' : 'invalid')\n\n      this.flowStatusService.setConnectionValidationStatus(valid, sourceNode, targetNode, sourceHandle, targetHandle)\n    }\n  }\n\n  /**\n   * TODO srp\n   */\n  protected resetValidateTargetHandle() {\n    this.targetHandleState.set('idle')\n\n    // drop back to start status\n    const status = this.flowStatusService.status()\n    if (status.state === 'connection-validation') {\n      this.flowStatusService.setConnectionStartStatus(status.payload.sourceNode, status.payload.sourceHandle)\n    }\n  }\n\n  // protected getHandleContext(type: 'source' | 'target') {\n  //   if (type === 'source') {\n  //     return {\n  //       $implicit: {\n  //         point: this.nodeModel.sourceOffset,\n  //         alignedPoint: this.nodeModel.sourceOffsetAligned,\n  //         state: this.sourceHanldeStateReadonly\n  //       }\n  //     }\n  //   }\n\n  //   return {\n  //     $implicit: {\n  //       point: this.nodeModel.targetOffset,\n  //       alignedPoint: this.nodeModel.targetOffsetAligned,\n  //       state: this.targetHanldeStateReadonly\n  //     }\n  //   }\n  // }\n}\n","<svg:foreignObject\n  *ngIf=\"nodeModel.node.type === 'default'\"\n  #nodeContent\n  width=\"100\"\n  height=\"50\"\n>\n  <div class=\"default-node\" [innerHTML]=\"nodeModel.node.text ?? ''\"></div>\n</svg:foreignObject>\n\n<svg:foreignObject\n  *ngIf=\"nodeModel.node.type === 'html-template' && nodeHtmlTemplate\"\n  [attr.width]=\"nodeModel.size().width\"\n  [attr.height]=\"nodeModel.size().height\"\n>\n  <div #htmlWrapper class=\"wrapper\">\n    <ng-container\n      [ngTemplateOutlet]=\"nodeHtmlTemplate\"\n      [ngTemplateOutletContext]=\"{ $implicit: { node: nodeModel.node } }\"\n      [ngTemplateOutletInjector]=\"injector\"\n    />\n  </div>\n</svg:foreignObject>\n\n<ng-container *ngFor=\"let handle of nodeModel.handles()\">\n  <svg:circle\n    #sourceHandle\n    [attr.cx]=\"handle.offset().x\"\n    [attr.cy]=\"handle.offset().y\"\n    [attr.stroke-width]=\"handle.strokeWidth\"\n    r=\"5\"\n    stroke=\"white\"\n    fill=\"black\"\n    (mousedown)=\"handle.rawHandle.type === 'source' ? startConnection($event, handle) : null\"\n    (mouseup)=\"handle.rawHandle.type === 'target' ? endConnection() : null\"\n  />\n\n  <svg:circle\n    *ngIf=\"handle.rawHandle.type === 'target' && showMagnet()\"\n    class=\"magnet\"\n    [attr.r]=\"nodeModel.magnetRadius\"\n    [attr.cx]=\"handle.offset().x\"\n    [attr.cy]=\"handle.offset().y\"\n    (mouseup)=\"endConnection(); resetValidateTargetHandle()\"\n    (mouseover)=\"validateTargetHandle(handle)\"\n    (mouseout)=\"resetValidateTargetHandle()\"\n  />\n</ng-container>\n\n"]}
|
|
@@ -3,7 +3,7 @@ import { MapContextDirective } from '../../directives/map-context.directive';
|
|
|
3
3
|
import { DraggableService } from '../../services/draggable.service';
|
|
4
4
|
import { ViewportService } from '../../services/viewport.service';
|
|
5
5
|
import { toObservable, toSignal } from '@angular/core/rxjs-interop';
|
|
6
|
-
import { ConnectionTemplateDirective, EdgeLabelHtmlTemplateDirective, EdgeTemplateDirective,
|
|
6
|
+
import { ConnectionTemplateDirective, EdgeLabelHtmlTemplateDirective, EdgeTemplateDirective, NodeHtmlTemplateDirective } from '../../directives/template.directive';
|
|
7
7
|
import { addNodesToEdges } from '../../utils/add-nodes-to-edges';
|
|
8
8
|
import { FlowModel } from '../../models/flow.model';
|
|
9
9
|
import { skip } from 'rxjs';
|
|
@@ -107,7 +107,7 @@ export class VflowComponent {
|
|
|
107
107
|
* For example, if you want to archieve right to left direction
|
|
108
108
|
* then you need to pass these positions { source: 'left', target: 'right' }
|
|
109
109
|
*
|
|
110
|
-
*
|
|
110
|
+
* @deprecated
|
|
111
111
|
*/
|
|
112
112
|
set handlePositions(handlePositions) {
|
|
113
113
|
this.flowModel.handlePositions.set(handlePositions);
|
|
@@ -191,7 +191,7 @@ export class VflowComponent {
|
|
|
191
191
|
FlowEntitiesService,
|
|
192
192
|
NodesChangeService,
|
|
193
193
|
EdgeChangesService
|
|
194
|
-
], queries: [{ propertyName: "nodeHtmlDirective", first: true, predicate: NodeHtmlTemplateDirective, descendants: true }, { propertyName: "edgeTemplateDirective", first: true, predicate: EdgeTemplateDirective, descendants: true }, { propertyName: "edgeLabelHtmlDirective", first: true, predicate: EdgeLabelHtmlTemplateDirective, descendants: true }, { propertyName: "connectionTemplateDirective", first: true, predicate: ConnectionTemplateDirective, descendants: true }
|
|
194
|
+
], queries: [{ propertyName: "nodeHtmlDirective", first: true, predicate: NodeHtmlTemplateDirective, descendants: true }, { propertyName: "edgeTemplateDirective", first: true, predicate: EdgeTemplateDirective, descendants: true }, { propertyName: "edgeLabelHtmlDirective", first: true, predicate: EdgeLabelHtmlTemplateDirective, descendants: true }, { propertyName: "connectionTemplateDirective", first: true, predicate: ConnectionTemplateDirective, descendants: true }], viewQueries: [{ propertyName: "mapContext", first: true, predicate: MapContextDirective, descendants: true }], hostDirectives: [{ directive: i1.ConnectionControllerDirective, outputs: ["onConnect", "onConnect"] }, { directive: i2.ChangesControllerDirective, outputs: ["onNodesChange", "onNodesChange", "onEdgesChange", "onEdgesChange"] }], ngImport: i0, template: "<svg:svg\n rootSvgRef\n rootSvgContext\n class=\"root-svg\"\n #flow\n [style.backgroundColor]=\"background\"\n [attr.width]=\"flowModel.flowWidth()\"\n [attr.height]=\"flowModel.flowHeight()\"\n>\n <defs [markers]=\"markers()\" flowDefs />\n\n <svg:g\n mapContext\n spacePointContext\n [minZoom]=\"minZoom\"\n [maxZoom]=\"maxZoom\"\n >\n <!-- Connection -->\n <svg:g\n connection\n [model]=\"connection\"\n [template]=\"connectionTemplateDirective?.templateRef\"\n />\n\n <!-- Edges -->\n <svg:g\n *ngFor=\"let model of edgeModels; trackBy: trackEdges\"\n edge\n [model]=\"model\"\n [edgeTemplate]=\"edgeTemplateDirective?.templateRef\"\n [edgeLabelHtmlTemplate]=\"edgeLabelHtmlDirective?.templateRef\"\n />\n\n <!-- Nodes -->\n <svg:g\n *ngFor=\"let model of nodeModels; trackBy: trackNodes\"\n node\n [nodeModel]=\"model\"\n [nodeHtmlTemplate]=\"nodeHtmlDirective?.templateRef\"\n [attr.transform]=\"model.pointTransform()\"\n />\n </svg:g>\n\n</svg:svg>\n", styles: [":host{display:block;width:100%;height:100%;-webkit-user-select:none;user-select:none}:host ::ng-deep *{box-sizing:border-box}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i4.NodeComponent, selector: "g[node]", inputs: ["nodeModel", "nodeHtmlTemplate"] }, { kind: "component", type: i5.EdgeComponent, selector: "g[edge]", inputs: ["model", "edgeTemplate", "edgeLabelHtmlTemplate"] }, { kind: "component", type: i6.ConnectionComponent, selector: "g[connection]", inputs: ["model", "template"] }, { kind: "component", type: i7.DefsComponent, selector: "defs[flowDefs]", inputs: ["markers"] }, { kind: "directive", type: i8.SpacePointContextDirective, selector: "g[spacePointContext]" }, { kind: "directive", type: i9.MapContextDirective, selector: "g[mapContext]", inputs: ["minZoom", "maxZoom"] }, { kind: "directive", type: i10.RootSvgReferenceDirective, selector: "svg[rootSvgRef]" }, { kind: "directive", type: i11.RootSvgContextDirective, selector: "svg[rootSvgContext]" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
195
195
|
}
|
|
196
196
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: VflowComponent, decorators: [{
|
|
197
197
|
type: Component,
|
|
@@ -205,7 +205,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
205
205
|
], hostDirectives: [
|
|
206
206
|
connectionControllerHostDirective,
|
|
207
207
|
changesControllerHostDirective
|
|
208
|
-
], template: "<svg:svg\n rootSvgRef\n rootSvgContext\n class=\"root-svg\"\n #flow\n [style.backgroundColor]=\"background\"\n [attr.width]=\"flowModel.flowWidth()\"\n [attr.height]=\"flowModel.flowHeight()\"\n>\n <defs [markers]=\"markers()\" flowDefs />\n\n <svg:g\n mapContext\n spacePointContext\n [minZoom]=\"minZoom\"\n [maxZoom]=\"maxZoom\"\n >\n <!-- Connection -->\n <svg:g\n connection\n [model]=\"connection\"\n [template]=\"connectionTemplateDirective?.templateRef\"\n />\n\n <!-- Edges -->\n <svg:g\n *ngFor=\"let model of edgeModels; trackBy: trackEdges\"\n edge\n [model]=\"model\"\n [edgeTemplate]=\"edgeTemplateDirective?.templateRef\"\n [edgeLabelHtmlTemplate]=\"edgeLabelHtmlDirective?.templateRef\"\n />\n\n <!-- Nodes -->\n <svg:g\n *ngFor=\"let model of nodeModels; trackBy: trackNodes\"\n node\n [nodeModel]=\"model\"\n [nodeHtmlTemplate]=\"nodeHtmlDirective?.templateRef\"\n [
|
|
208
|
+
], template: "<svg:svg\n rootSvgRef\n rootSvgContext\n class=\"root-svg\"\n #flow\n [style.backgroundColor]=\"background\"\n [attr.width]=\"flowModel.flowWidth()\"\n [attr.height]=\"flowModel.flowHeight()\"\n>\n <defs [markers]=\"markers()\" flowDefs />\n\n <svg:g\n mapContext\n spacePointContext\n [minZoom]=\"minZoom\"\n [maxZoom]=\"maxZoom\"\n >\n <!-- Connection -->\n <svg:g\n connection\n [model]=\"connection\"\n [template]=\"connectionTemplateDirective?.templateRef\"\n />\n\n <!-- Edges -->\n <svg:g\n *ngFor=\"let model of edgeModels; trackBy: trackEdges\"\n edge\n [model]=\"model\"\n [edgeTemplate]=\"edgeTemplateDirective?.templateRef\"\n [edgeLabelHtmlTemplate]=\"edgeLabelHtmlDirective?.templateRef\"\n />\n\n <!-- Nodes -->\n <svg:g\n *ngFor=\"let model of nodeModels; trackBy: trackNodes\"\n node\n [nodeModel]=\"model\"\n [nodeHtmlTemplate]=\"nodeHtmlDirective?.templateRef\"\n [attr.transform]=\"model.pointTransform()\"\n />\n </svg:g>\n\n</svg:svg>\n", styles: [":host{display:block;width:100%;height:100%;-webkit-user-select:none;user-select:none}:host ::ng-deep *{box-sizing:border-box}\n"] }]
|
|
209
209
|
}], propDecorators: { view: [{
|
|
210
210
|
type: Input
|
|
211
211
|
}], minZoom: [{
|
|
@@ -236,9 +236,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
236
236
|
}], connectionTemplateDirective: [{
|
|
237
237
|
type: ContentChild,
|
|
238
238
|
args: [ConnectionTemplateDirective]
|
|
239
|
-
}], handleTemplateDirective: [{
|
|
240
|
-
type: ContentChild,
|
|
241
|
-
args: [HandleTemplateDirective]
|
|
242
239
|
}], mapContext: [{
|
|
243
240
|
type: ViewChild,
|
|
244
241
|
args: [MapContextDirective]
|
|
@@ -246,4 +243,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
246
243
|
function bindFlowToNodes(flow, nodes) {
|
|
247
244
|
nodes.forEach(n => n.bindFlow(flow));
|
|
248
245
|
}
|
|
249
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vflow.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/vflow/vflow.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/vflow/vflow.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterContentInit, ChangeDetectionStrategy, Component, ContentChild, EventEmitter, Injector, Input, OnChanges, Output, Signal, SimpleChanges, ViewChild, computed, effect, inject, runInInjectionContext } from '@angular/core';\nimport { Node } from '../../interfaces/node.interface';\nimport { MapContextDirective } from '../../directives/map-context.directive';\nimport { DraggableService } from '../../services/draggable.service';\nimport { NodeModel } from '../../models/node.model';\nimport { ViewportService } from '../../services/viewport.service';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { Edge } from '../../interfaces/edge.interface';\nimport { EdgeModel } from '../../models/edge.model';\nimport { ConnectionTemplateDirective, EdgeLabelHtmlTemplateDirective, EdgeTemplateDirective, HandleTemplateDirective, NodeHtmlTemplateDirective } from '../../directives/template.directive';\nimport { HandlePositions } from '../../interfaces/handle-positions.interface';\nimport { addNodesToEdges } from '../../utils/add-nodes-to-edges';\nimport { FlowModel } from '../../models/flow.model';\nimport { skip } from 'rxjs';\nimport { Point } from '../../interfaces/point.interface';\nimport { ViewportState } from '../../interfaces/viewport.interface';\nimport { FlowStatusService } from '../../services/flow-status.service';\nimport { ConnectionControllerDirective } from '../../directives/connection-controller.directive';\nimport { FlowEntitiesService } from '../../services/flow-entities.service';\nimport { ConnectionSettings } from '../../interfaces/connection-settings.interface';\nimport { ConnectionModel } from '../../models/connection.model';\nimport { ReferenceKeeper } from '../../utils/reference-keeper';\nimport { NodesChangeService } from '../../services/node-changes.service';\nimport { EdgeChangesService } from '../../services/edge-changes.service';\nimport { NodeChange } from '../../types/node-change.type';\nimport { ChangesControllerDirective } from '../../directives/changes-controller.directive';\nimport { EdgeChange } from '../../types/edge-change.type';\n\nconst connectionControllerHostDirective = {\n  directive: ConnectionControllerDirective,\n  outputs: ['onConnect']\n}\n\nconst changesControllerHostDirective = {\n  directive: ChangesControllerDirective,\n  outputs: ['onNodesChange', 'onEdgesChange']\n}\n\n@Component({\n  selector: 'vflow',\n  templateUrl: './vflow.component.html',\n  styleUrls: ['./vflow.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    DraggableService,\n    ViewportService,\n    FlowStatusService,\n    FlowEntitiesService,\n    NodesChangeService,\n    EdgeChangesService\n  ],\n  hostDirectives: [\n    connectionControllerHostDirective,\n    changesControllerHostDirective\n  ]\n})\nexport class VflowComponent {\n  // #region DI\n  private viewportService = inject(ViewportService)\n  private flowEntitiesService = inject(FlowEntitiesService)\n  private nodesChangeService = inject(NodesChangeService)\n  private edgesChangeService = inject(EdgeChangesService)\n  private injector = inject(Injector)\n  // #endregion\n\n  // #region SETTINGS\n\n  /**\n   * Size for flow view\n   *\n   * accepts\n   * - absolute size in format [width, height] or\n   * - 'auto' to compute size based on parent element size\n   */\n  @Input()\n  public set view(view: [number, number] | 'auto') {\n    this.flowModel.view.set(view)\n  }\n\n  /**\n   * Minimum zoom value\n   */\n  @Input()\n  public minZoom = 0.5\n\n  /**\n   * Maximum zoom value\n   */\n  @Input()\n  public maxZoom = 3\n\n  /**\n   * Object that controls flow direction.\n   *\n   * For example, if you want to archieve right to left direction\n   * then you need to pass these positions { source: 'left', target: 'right' }\n   *\n   * ! Be carefult using this field, it may depricate in future releases !\n   */\n  @Input()\n  public set handlePositions(handlePositions: HandlePositions) {\n    this.flowModel.handlePositions.set(handlePositions)\n  }\n\n  /**\n   * Background color for flow\n   */\n  @Input()\n  public background: string = '#FFFFFF'\n\n  /**\n   * Settings for connection (it renders when user tries to create edge between nodes)\n   *\n   * You need to pass `ConnectionSettings` in this input.\n   */\n  @Input({ transform: (settings: ConnectionSettings) => new ConnectionModel(settings) })\n  public set connection(connection: ConnectionModel) { this.flowEntitiesService.connection.set(connection) }\n\n  public get connection() { return this.flowEntitiesService.connection() }\n  // #endregion\n\n  // #region MAIN_INPUTS\n  /**\n   * Nodes to render\n   */\n  @Input({ required: true })\n  public set nodes(newNodes: Node[]) {\n    const newModels = runInInjectionContext(this.injector,\n      () => ReferenceKeeper.nodes(newNodes, this.flowEntitiesService.nodes())\n    )\n\n    // TODO better to solve this by DI\n    bindFlowToNodes(this.flowModel, newModels)\n\n    // quick and dirty binding nodes to edges\n    addNodesToEdges(newModels, this.flowEntitiesService.edges())\n\n    this.flowEntitiesService.nodes.set(newModels)\n  }\n\n  protected get nodeModels() { return this.flowEntitiesService.nodes() }\n\n  /**\n   * Edges to render\n   */\n  @Input()\n  public set edges(newEdges: Edge[]) {\n    const newModels = ReferenceKeeper.edges(newEdges, this.flowEntitiesService.edges())\n\n    // quick and dirty binding nodes to edges\n    addNodesToEdges(this.nodeModels, newModels)\n\n    this.flowEntitiesService.edges.set(newModels)\n  }\n\n  protected get edgeModels() { return this.flowEntitiesService.validEdges() }\n  // #endregion\n\n  // #region TEMPLATES\n  @ContentChild(NodeHtmlTemplateDirective)\n  protected nodeHtmlDirective?: NodeHtmlTemplateDirective\n\n  @ContentChild(EdgeTemplateDirective)\n  protected edgeTemplateDirective?: EdgeTemplateDirective\n\n  @ContentChild(EdgeLabelHtmlTemplateDirective)\n  protected edgeLabelHtmlDirective?: EdgeLabelHtmlTemplateDirective\n\n  @ContentChild(ConnectionTemplateDirective)\n  protected connectionTemplateDirective?: ConnectionTemplateDirective\n\n  @ContentChild(HandleTemplateDirective)\n  protected handleTemplateDirective?: HandleTemplateDirective\n  // #endregion\n\n  // #region DIRECTIVES\n  @ViewChild(MapContextDirective)\n  protected mapContext!: MapContextDirective\n  // #endregion\n\n  // #region SIGNAL_API\n  /**\n   * Signal for reading viewport change\n   */\n  public readonly viewport = this.viewportService.readableViewport.asReadonly()\n\n  /**\n   * Signal for reading nodes change\n   */\n  public readonly nodesChange =\n    toSignal(this.nodesChangeService.changes$, { initialValue: [] as NodeChange[] })\n\n  /**\n   * Signal to reading edges change\n   */\n  public readonly edgesChange =\n    toSignal(this.edgesChangeService.changes$, { initialValue: [] as EdgeChange[] })\n  // #endregion\n\n  // #region RX_API\n  /**\n   * Observable with viewport change\n   */\n  public readonly viewportChange$ = toObservable(this.viewportService.readableViewport)\n    .pipe(skip(1)) // skip default value that set by signal\n\n  /**\n   * Observable with nodes change\n   */\n  public readonly nodesChange$ = this.nodesChangeService.changes$\n\n  /**\n   * Observable with nodes change\n   */\n  public readonly edgesChange$ = this.edgesChangeService.changes$\n  // #endregion\n\n  // TODO: probably better to make it injectable\n  protected flowModel = new FlowModel()\n\n  protected markers = this.flowEntitiesService.markers\n\n  // #region METHODS_API\n  /**\n   * Change viewport to specified state\n   *\n   * @param viewport viewport state\n   */\n  public viewportTo(viewport: ViewportState): void {\n    this.viewportService.writableViewport.set({ changeType: 'absolute', state: viewport })\n  }\n\n  /**\n   * Change zoom\n   *\n   * @param zoom zoom value\n   */\n  public zoomTo(zoom: number): void {\n    this.viewportService.writableViewport.set({ changeType: 'absolute', state: { zoom } })\n  }\n\n  /**\n   * Move to specified coordinate\n   *\n   * @param point point where to move\n   */\n  public panTo(point: Point): void {\n    this.viewportService.writableViewport.set({ changeType: 'absolute', state: point })\n  }\n\n  /**\n   * Get node by id\n   *\n   * @param id node id\n   */\n  public getNode<T = unknown>(id: string): Node<T> | undefined {\n    return this.flowEntitiesService.getNode<T>(id)?.node\n  }\n  // #endregion\n\n  protected trackNodes(idx: number, { node }: NodeModel) {\n    return node.id\n  }\n\n  protected trackEdges(idx: number, { edge }: EdgeModel) {\n    return edge.id\n  }\n}\n\nfunction bindFlowToNodes(flow: FlowModel, nodes: NodeModel[]) {\n  nodes.forEach(n => n.bindFlow(flow))\n}\n\n","<svg:svg\n  rootSvgRef\n  rootSvgContext\n  class=\"root-svg\"\n  #flow\n  [style.backgroundColor]=\"background\"\n  [attr.width]=\"flowModel.flowWidth()\"\n  [attr.height]=\"flowModel.flowHeight()\"\n>\n  <defs [markers]=\"markers()\" flowDefs />\n\n  <svg:g\n    mapContext\n    spacePointContext\n    [minZoom]=\"minZoom\"\n    [maxZoom]=\"maxZoom\"\n  >\n    <!-- Connection -->\n    <svg:g\n      connection\n      [model]=\"connection\"\n      [template]=\"connectionTemplateDirective?.templateRef\"\n    />\n\n    <!-- Edges  -->\n    <svg:g\n      *ngFor=\"let model of edgeModels; trackBy: trackEdges\"\n      edge\n      [model]=\"model\"\n      [edgeTemplate]=\"edgeTemplateDirective?.templateRef\"\n      [edgeLabelHtmlTemplate]=\"edgeLabelHtmlDirective?.templateRef\"\n    />\n\n    <!-- Nodes -->\n    <svg:g\n      *ngFor=\"let model of nodeModels; trackBy: trackNodes\"\n      node\n      [nodeModel]=\"model\"\n      [nodeHtmlTemplate]=\"nodeHtmlDirective?.templateRef\"\n      [handleTemplate]=\"handleTemplateDirective?.templateRef\"\n      [attr.transform]=\"model.pointTransform()\"\n    />\n  </svg:g>\n\n</svg:svg>\n"]}
|
|
246
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vflow.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/vflow/vflow.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/vflow/vflow.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterContentInit, ChangeDetectionStrategy, Component, ContentChild, EventEmitter, Injector, Input, OnChanges, Output, Signal, SimpleChanges, ViewChild, computed, effect, inject, runInInjectionContext } from '@angular/core';\nimport { Node } from '../../interfaces/node.interface';\nimport { MapContextDirective } from '../../directives/map-context.directive';\nimport { DraggableService } from '../../services/draggable.service';\nimport { NodeModel } from '../../models/node.model';\nimport { ViewportService } from '../../services/viewport.service';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { Edge } from '../../interfaces/edge.interface';\nimport { EdgeModel } from '../../models/edge.model';\nimport { ConnectionTemplateDirective, EdgeLabelHtmlTemplateDirective, EdgeTemplateDirective, NodeHtmlTemplateDirective } from '../../directives/template.directive';\nimport { HandlePositions } from '../../interfaces/handle-positions.interface';\nimport { addNodesToEdges } from '../../utils/add-nodes-to-edges';\nimport { FlowModel } from '../../models/flow.model';\nimport { skip } from 'rxjs';\nimport { Point } from '../../interfaces/point.interface';\nimport { ViewportState } from '../../interfaces/viewport.interface';\nimport { FlowStatusService } from '../../services/flow-status.service';\nimport { ConnectionControllerDirective } from '../../directives/connection-controller.directive';\nimport { FlowEntitiesService } from '../../services/flow-entities.service';\nimport { ConnectionSettings } from '../../interfaces/connection-settings.interface';\nimport { ConnectionModel } from '../../models/connection.model';\nimport { ReferenceKeeper } from '../../utils/reference-keeper';\nimport { NodesChangeService } from '../../services/node-changes.service';\nimport { EdgeChangesService } from '../../services/edge-changes.service';\nimport { NodeChange } from '../../types/node-change.type';\nimport { ChangesControllerDirective } from '../../directives/changes-controller.directive';\nimport { EdgeChange } from '../../types/edge-change.type';\n\nconst connectionControllerHostDirective = {\n  directive: ConnectionControllerDirective,\n  outputs: ['onConnect']\n}\n\nconst changesControllerHostDirective = {\n  directive: ChangesControllerDirective,\n  outputs: ['onNodesChange', 'onEdgesChange']\n}\n\n@Component({\n  selector: 'vflow',\n  templateUrl: './vflow.component.html',\n  styleUrls: ['./vflow.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    DraggableService,\n    ViewportService,\n    FlowStatusService,\n    FlowEntitiesService,\n    NodesChangeService,\n    EdgeChangesService\n  ],\n  hostDirectives: [\n    connectionControllerHostDirective,\n    changesControllerHostDirective\n  ]\n})\nexport class VflowComponent {\n  // #region DI\n  private viewportService = inject(ViewportService)\n  private flowEntitiesService = inject(FlowEntitiesService)\n  private nodesChangeService = inject(NodesChangeService)\n  private edgesChangeService = inject(EdgeChangesService)\n  private injector = inject(Injector)\n  // #endregion\n\n  // #region SETTINGS\n\n  /**\n   * Size for flow view\n   *\n   * accepts\n   * - absolute size in format [width, height] or\n   * - 'auto' to compute size based on parent element size\n   */\n  @Input()\n  public set view(view: [number, number] | 'auto') {\n    this.flowModel.view.set(view)\n  }\n\n  /**\n   * Minimum zoom value\n   */\n  @Input()\n  public minZoom = 0.5\n\n  /**\n   * Maximum zoom value\n   */\n  @Input()\n  public maxZoom = 3\n\n  /**\n   * Object that controls flow direction.\n   *\n   * For example, if you want to archieve right to left direction\n   * then you need to pass these positions { source: 'left', target: 'right' }\n   *\n   * @deprecated\n   */\n  @Input()\n  public set handlePositions(handlePositions: HandlePositions) {\n    this.flowModel.handlePositions.set(handlePositions)\n  }\n\n  /**\n   * Background color for flow\n   */\n  @Input()\n  public background: string = '#FFFFFF'\n\n  /**\n   * Settings for connection (it renders when user tries to create edge between nodes)\n   *\n   * You need to pass `ConnectionSettings` in this input.\n   */\n  @Input({ transform: (settings: ConnectionSettings) => new ConnectionModel(settings) })\n  public set connection(connection: ConnectionModel) { this.flowEntitiesService.connection.set(connection) }\n\n  public get connection() { return this.flowEntitiesService.connection() }\n  // #endregion\n\n  // #region MAIN_INPUTS\n  /**\n   * Nodes to render\n   */\n  @Input({ required: true })\n  public set nodes(newNodes: Node[]) {\n    const newModels = runInInjectionContext(this.injector,\n      () => ReferenceKeeper.nodes(newNodes, this.flowEntitiesService.nodes())\n    )\n\n    // TODO better to solve this by DI\n    bindFlowToNodes(this.flowModel, newModels)\n\n    // quick and dirty binding nodes to edges\n    addNodesToEdges(newModels, this.flowEntitiesService.edges())\n\n    this.flowEntitiesService.nodes.set(newModels)\n  }\n\n  protected get nodeModels() { return this.flowEntitiesService.nodes() }\n\n  /**\n   * Edges to render\n   */\n  @Input()\n  public set edges(newEdges: Edge[]) {\n    const newModels = ReferenceKeeper.edges(newEdges, this.flowEntitiesService.edges())\n\n    // quick and dirty binding nodes to edges\n    addNodesToEdges(this.nodeModels, newModels)\n\n    this.flowEntitiesService.edges.set(newModels)\n  }\n\n  protected get edgeModels() { return this.flowEntitiesService.validEdges() }\n  // #endregion\n\n  // #region TEMPLATES\n  @ContentChild(NodeHtmlTemplateDirective)\n  protected nodeHtmlDirective?: NodeHtmlTemplateDirective\n\n  @ContentChild(EdgeTemplateDirective)\n  protected edgeTemplateDirective?: EdgeTemplateDirective\n\n  @ContentChild(EdgeLabelHtmlTemplateDirective)\n  protected edgeLabelHtmlDirective?: EdgeLabelHtmlTemplateDirective\n\n  @ContentChild(ConnectionTemplateDirective)\n  protected connectionTemplateDirective?: ConnectionTemplateDirective\n  // #endregion\n\n  // #region DIRECTIVES\n  @ViewChild(MapContextDirective)\n  protected mapContext!: MapContextDirective\n  // #endregion\n\n  // #region SIGNAL_API\n  /**\n   * Signal for reading viewport change\n   */\n  public readonly viewport = this.viewportService.readableViewport.asReadonly()\n\n  /**\n   * Signal for reading nodes change\n   */\n  public readonly nodesChange =\n    toSignal(this.nodesChangeService.changes$, { initialValue: [] as NodeChange[] })\n\n  /**\n   * Signal to reading edges change\n   */\n  public readonly edgesChange =\n    toSignal(this.edgesChangeService.changes$, { initialValue: [] as EdgeChange[] })\n  // #endregion\n\n  // #region RX_API\n  /**\n   * Observable with viewport change\n   */\n  public readonly viewportChange$ = toObservable(this.viewportService.readableViewport)\n    .pipe(skip(1)) // skip default value that set by signal\n\n  /**\n   * Observable with nodes change\n   */\n  public readonly nodesChange$ = this.nodesChangeService.changes$\n\n  /**\n   * Observable with nodes change\n   */\n  public readonly edgesChange$ = this.edgesChangeService.changes$\n  // #endregion\n\n  // TODO: probably better to make it injectable\n  protected flowModel = new FlowModel()\n\n  protected markers = this.flowEntitiesService.markers\n\n  // #region METHODS_API\n  /**\n   * Change viewport to specified state\n   *\n   * @param viewport viewport state\n   */\n  public viewportTo(viewport: ViewportState): void {\n    this.viewportService.writableViewport.set({ changeType: 'absolute', state: viewport })\n  }\n\n  /**\n   * Change zoom\n   *\n   * @param zoom zoom value\n   */\n  public zoomTo(zoom: number): void {\n    this.viewportService.writableViewport.set({ changeType: 'absolute', state: { zoom } })\n  }\n\n  /**\n   * Move to specified coordinate\n   *\n   * @param point point where to move\n   */\n  public panTo(point: Point): void {\n    this.viewportService.writableViewport.set({ changeType: 'absolute', state: point })\n  }\n\n  /**\n   * Get node by id\n   *\n   * @param id node id\n   */\n  public getNode<T = unknown>(id: string): Node<T> | undefined {\n    return this.flowEntitiesService.getNode<T>(id)?.node\n  }\n  // #endregion\n\n  protected trackNodes(idx: number, { node }: NodeModel) {\n    return node.id\n  }\n\n  protected trackEdges(idx: number, { edge }: EdgeModel) {\n    return edge.id\n  }\n}\n\nfunction bindFlowToNodes(flow: FlowModel, nodes: NodeModel[]) {\n  nodes.forEach(n => n.bindFlow(flow))\n}\n\n","<svg:svg\n  rootSvgRef\n  rootSvgContext\n  class=\"root-svg\"\n  #flow\n  [style.backgroundColor]=\"background\"\n  [attr.width]=\"flowModel.flowWidth()\"\n  [attr.height]=\"flowModel.flowHeight()\"\n>\n  <defs [markers]=\"markers()\" flowDefs />\n\n  <svg:g\n    mapContext\n    spacePointContext\n    [minZoom]=\"minZoom\"\n    [maxZoom]=\"maxZoom\"\n  >\n    <!-- Connection -->\n    <svg:g\n      connection\n      [model]=\"connection\"\n      [template]=\"connectionTemplateDirective?.templateRef\"\n    />\n\n    <!-- Edges  -->\n    <svg:g\n      *ngFor=\"let model of edgeModels; trackBy: trackEdges\"\n      edge\n      [model]=\"model\"\n      [edgeTemplate]=\"edgeTemplateDirective?.templateRef\"\n      [edgeLabelHtmlTemplate]=\"edgeLabelHtmlDirective?.templateRef\"\n    />\n\n    <!-- Nodes -->\n    <svg:g\n      *ngFor=\"let model of nodeModels; trackBy: trackNodes\"\n      node\n      [nodeModel]=\"model\"\n      [nodeHtmlTemplate]=\"nodeHtmlDirective?.templateRef\"\n      [attr.transform]=\"model.pointTransform()\"\n    />\n  </svg:g>\n\n</svg:svg>\n"]}
|