ngx-vflow 0.3.0 → 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (56) hide show
  1. package/esm2022/lib/vflow/components/edge/edge.component.mjs +18 -6
  2. package/esm2022/lib/vflow/components/handle/handle.component.mjs +7 -7
  3. package/esm2022/lib/vflow/components/node/node.component.mjs +23 -8
  4. package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +63 -22
  5. package/esm2022/lib/vflow/decorators/run-in-injection-context.decorator.mjs +6 -14
  6. package/esm2022/lib/vflow/directives/changes-controller.directive.mjs +115 -14
  7. package/esm2022/lib/vflow/directives/map-context.directive.mjs +31 -4
  8. package/esm2022/lib/vflow/directives/pointer.directive.mjs +84 -0
  9. package/esm2022/lib/vflow/directives/root-pointer.directive.mjs +42 -0
  10. package/esm2022/lib/vflow/directives/root-svg-context.directive.mjs +5 -2
  11. package/esm2022/lib/vflow/directives/selectable.directive.mjs +39 -0
  12. package/esm2022/lib/vflow/directives/space-point-context.directive.mjs +10 -6
  13. package/esm2022/lib/vflow/interfaces/flow-entity.interface.mjs +2 -0
  14. package/esm2022/lib/vflow/interfaces/template-context.interface.mjs +1 -1
  15. package/esm2022/lib/vflow/models/edge.model.mjs +3 -1
  16. package/esm2022/lib/vflow/models/node.model.mjs +9 -12
  17. package/esm2022/lib/vflow/services/edge-changes.service.mjs +6 -2
  18. package/esm2022/lib/vflow/services/flow-entities.service.mjs +5 -1
  19. package/esm2022/lib/vflow/services/flow-settings.service.mjs +25 -0
  20. package/esm2022/lib/vflow/services/node-changes.service.mjs +7 -3
  21. package/esm2022/lib/vflow/services/node-rendering.service.mjs +22 -0
  22. package/esm2022/lib/vflow/services/selection.service.mjs +45 -0
  23. package/esm2022/lib/vflow/types/edge-change.type.mjs +1 -1
  24. package/esm2022/lib/vflow/types/node-change.type.mjs +1 -1
  25. package/esm2022/lib/vflow/vflow.module.mjs +15 -4
  26. package/esm2022/public-api.mjs +2 -1
  27. package/fesm2022/ngx-vflow.mjs +582 -161
  28. package/fesm2022/ngx-vflow.mjs.map +1 -1
  29. package/lib/vflow/components/edge/edge.component.d.ts +5 -2
  30. package/lib/vflow/components/handle/handle.component.d.ts +4 -3
  31. package/lib/vflow/components/node/node.component.d.ts +10 -4
  32. package/lib/vflow/components/vflow/vflow.component.d.ts +12 -6
  33. package/lib/vflow/decorators/run-in-injection-context.decorator.d.ts +2 -5
  34. package/lib/vflow/directives/changes-controller.directive.d.ts +35 -9
  35. package/lib/vflow/directives/map-context.directive.d.ts +5 -0
  36. package/lib/vflow/directives/pointer.directive.d.ts +21 -0
  37. package/lib/vflow/directives/root-pointer.directive.d.ts +40 -0
  38. package/lib/vflow/directives/selectable.directive.d.ts +11 -0
  39. package/lib/vflow/directives/space-point-context.directive.d.ts +13 -3
  40. package/lib/vflow/interfaces/flow-entity.interface.d.ts +4 -0
  41. package/lib/vflow/interfaces/template-context.interface.d.ts +1 -0
  42. package/lib/vflow/models/edge.model.d.ts +8 -20
  43. package/lib/vflow/models/node.model.d.ts +6 -9
  44. package/lib/vflow/services/edge-changes.service.d.ts +5 -0
  45. package/lib/vflow/services/flow-entities.service.d.ts +5 -2
  46. package/lib/vflow/services/flow-settings.service.d.ts +20 -0
  47. package/lib/vflow/services/node-changes.service.d.ts +5 -0
  48. package/lib/vflow/services/node-rendering.service.d.ts +9 -0
  49. package/lib/vflow/services/selection.service.d.ts +19 -0
  50. package/lib/vflow/types/edge-change.type.d.ts +5 -1
  51. package/lib/vflow/types/node-change.type.d.ts +5 -1
  52. package/lib/vflow/vflow.module.d.ts +6 -3
  53. package/package.json +3 -3
  54. package/public-api.d.ts +1 -0
  55. package/esm2022/lib/vflow/models/flow.model.mjs +0 -18
  56. package/lib/vflow/models/flow.model.d.ts +0 -16
@@ -1,11 +1,10 @@
1
- import { ChangeDetectionStrategy, Component, ContentChild, Injector, Input, ViewChild, inject, runInInjectionContext } from '@angular/core';
1
+ import { ChangeDetectionStrategy, Component, ContentChild, Injector, Input, ViewChild, computed, inject, runInInjectionContext } from '@angular/core';
2
2
  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
6
  import { ConnectionTemplateDirective, EdgeLabelHtmlTemplateDirective, EdgeTemplateDirective, NodeHtmlTemplateDirective } from '../../directives/template.directive';
7
7
  import { addNodesToEdges } from '../../utils/add-nodes-to-edges';
8
- import { FlowModel } from '../../models/flow.model';
9
8
  import { skip } from 'rxjs';
10
9
  import { FlowStatusService } from '../../services/flow-status.service';
11
10
  import { ConnectionControllerDirective } from '../../directives/connection-controller.directive';
@@ -15,6 +14,9 @@ import { ReferenceKeeper } from '../../utils/reference-keeper';
15
14
  import { NodesChangeService } from '../../services/node-changes.service';
16
15
  import { EdgeChangesService } from '../../services/edge-changes.service';
17
16
  import { ChangesControllerDirective } from '../../directives/changes-controller.directive';
17
+ import { NodeRenderingService } from '../../services/node-rendering.service';
18
+ import { SelectionService } from '../../services/selection.service';
19
+ import { FlowSettingsService } from '../../services/flow-settings.service';
18
20
  import * as i0 from "@angular/core";
19
21
  import * as i1 from "../../directives/connection-controller.directive";
20
22
  import * as i2 from "../../directives/changes-controller.directive";
@@ -27,13 +29,41 @@ import * as i8 from "../../directives/space-point-context.directive";
27
29
  import * as i9 from "../../directives/map-context.directive";
28
30
  import * as i10 from "../../directives/reference.directive";
29
31
  import * as i11 from "../../directives/root-svg-context.directive";
32
+ import * as i12 from "../../directives/root-pointer.directive";
30
33
  const connectionControllerHostDirective = {
31
34
  directive: ConnectionControllerDirective,
32
35
  outputs: ['onConnect']
33
36
  };
34
37
  const changesControllerHostDirective = {
35
38
  directive: ChangesControllerDirective,
36
- outputs: ['onNodesChange', 'onEdgesChange']
39
+ outputs: [
40
+ 'onNodesChange',
41
+ 'onNodesChange.position',
42
+ 'onNodesChange.position.single',
43
+ 'onNodesChange.position.many',
44
+ 'onNodesChange.add',
45
+ 'onNodesChange.add.single',
46
+ 'onNodesChange.add.many',
47
+ 'onNodesChange.remove',
48
+ 'onNodesChange.remove.single',
49
+ 'onNodesChange.remove.many',
50
+ 'onNodesChange.select',
51
+ 'onNodesChange.select.single',
52
+ 'onNodesChange.select.many',
53
+ 'onEdgesChange',
54
+ 'onEdgesChange.detached',
55
+ 'onEdgesChange.detached.single',
56
+ 'onEdgesChange.detached.many',
57
+ 'onEdgesChange.add',
58
+ 'onEdgesChange.add.single',
59
+ 'onEdgesChange.add.many',
60
+ 'onEdgesChange.remove',
61
+ 'onEdgesChange.remove.single',
62
+ 'onEdgesChange.remove.many',
63
+ 'onEdgesChange.select',
64
+ 'onEdgesChange.select.single',
65
+ 'onEdgesChange.select.many',
66
+ ]
37
67
  };
38
68
  export class VflowComponent {
39
69
  constructor() {
@@ -42,6 +72,8 @@ export class VflowComponent {
42
72
  this.flowEntitiesService = inject(FlowEntitiesService);
43
73
  this.nodesChangeService = inject(NodesChangeService);
44
74
  this.edgesChangeService = inject(EdgeChangesService);
75
+ this.nodeRenderingService = inject(NodeRenderingService);
76
+ this.flowSettingsService = inject(FlowSettingsService);
45
77
  this.injector = inject(Injector);
46
78
  /**
47
79
  * Minimum zoom value
@@ -55,6 +87,8 @@ export class VflowComponent {
55
87
  * Background color for flow
56
88
  */
57
89
  this.background = '#FFFFFF';
90
+ this.nodeModels = computed(() => this.nodeRenderingService.nodes());
91
+ this.edgeModels = computed(() => this.flowEntitiesService.validEdges());
58
92
  // #endregion
59
93
  // #region SIGNAL_API
60
94
  /**
@@ -81,12 +115,12 @@ export class VflowComponent {
81
115
  */
82
116
  this.nodesChange$ = this.nodesChangeService.changes$;
83
117
  /**
84
- * Observable with nodes change
118
+ * Observable with edges change
85
119
  */
86
120
  this.edgesChange$ = this.edgesChangeService.changes$;
87
121
  // #endregion
88
- // TODO: probably better to make it injectable
89
- this.flowModel = new FlowModel();
122
+ this.flowWidth = this.flowSettingsService.flowWidth;
123
+ this.flowHeight = this.flowSettingsService.flowHeight;
90
124
  this.markers = this.flowEntitiesService.markers;
91
125
  }
92
126
  // #endregion
@@ -99,7 +133,7 @@ export class VflowComponent {
99
133
  * - 'auto' to compute size based on parent element size
100
134
  */
101
135
  set view(view) {
102
- this.flowModel.view.set(view);
136
+ this.flowSettingsService.view.set(view);
103
137
  }
104
138
  /**
105
139
  * Object that controls flow direction.
@@ -110,7 +144,13 @@ export class VflowComponent {
110
144
  * @deprecated
111
145
  */
112
146
  set handlePositions(handlePositions) {
113
- this.flowModel.handlePositions.set(handlePositions);
147
+ this.flowSettingsService.handlePositions.set(handlePositions);
148
+ }
149
+ /**
150
+ * Global rule if you can or can't select entities
151
+ */
152
+ set entitiesSelectable(value) {
153
+ this.flowSettingsService.entitiesSelectable.set(value);
114
154
  }
115
155
  /**
116
156
  * Settings for connection (it renders when user tries to create edge between nodes)
@@ -126,23 +166,19 @@ export class VflowComponent {
126
166
  */
127
167
  set nodes(newNodes) {
128
168
  const newModels = runInInjectionContext(this.injector, () => ReferenceKeeper.nodes(newNodes, this.flowEntitiesService.nodes()));
129
- // TODO better to solve this by DI
130
- bindFlowToNodes(this.flowModel, newModels);
131
169
  // quick and dirty binding nodes to edges
132
170
  addNodesToEdges(newModels, this.flowEntitiesService.edges());
133
171
  this.flowEntitiesService.nodes.set(newModels);
134
172
  }
135
- get nodeModels() { return this.flowEntitiesService.nodes(); }
136
173
  /**
137
174
  * Edges to render
138
175
  */
139
176
  set edges(newEdges) {
140
177
  const newModels = runInInjectionContext(this.injector, () => ReferenceKeeper.edges(newEdges, this.flowEntitiesService.edges()));
141
178
  // quick and dirty binding nodes to edges
142
- addNodesToEdges(this.nodeModels, newModels);
179
+ addNodesToEdges(this.nodeModels(), newModels);
143
180
  this.flowEntitiesService.edges.set(newModels);
144
181
  }
145
- get edgeModels() { return this.flowEntitiesService.validEdges(); }
146
182
  // #region METHODS_API
147
183
  /**
148
184
  * Change viewport to specified state
@@ -190,14 +226,17 @@ export class VflowComponent {
190
226
  return edge;
191
227
  }
192
228
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: VflowComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
193
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "16.1.0", version: "16.2.12", type: VflowComponent, selector: "vflow", inputs: { view: "view", minZoom: "minZoom", maxZoom: "maxZoom", handlePositions: "handlePositions", background: "background", connection: ["connection", "connection", (settings) => new ConnectionModel(settings)], nodes: "nodes", edges: "edges" }, providers: [
229
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "16.1.0", version: "16.2.12", type: VflowComponent, selector: "vflow", inputs: { view: "view", minZoom: "minZoom", maxZoom: "maxZoom", handlePositions: "handlePositions", background: "background", entitiesSelectable: "entitiesSelectable", connection: ["connection", "connection", (settings) => new ConnectionModel(settings)], nodes: "nodes", edges: "edges" }, providers: [
194
230
  DraggableService,
195
231
  ViewportService,
196
232
  FlowStatusService,
197
233
  FlowEntitiesService,
198
234
  NodesChangeService,
199
- EdgeChangesService
200
- ], 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 }); }
235
+ EdgeChangesService,
236
+ NodeRenderingService,
237
+ SelectionService,
238
+ FlowSettingsService
239
+ ], 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", "onNodesChange.position", "onNodesChange.position", "onNodesChange.position.single", "onNodesChange.position.single", "onNodesChange.position.many", "onNodesChange.position.many", "onNodesChange.add", "onNodesChange.add", "onNodesChange.add.single", "onNodesChange.add.single", "onNodesChange.add.many", "onNodesChange.add.many", "onNodesChange.remove", "onNodesChange.remove", "onNodesChange.remove.single", "onNodesChange.remove.single", "onNodesChange.remove.many", "onNodesChange.remove.many", "onNodesChange.select", "onNodesChange.select", "onNodesChange.select.single", "onNodesChange.select.single", "onNodesChange.select.many", "onNodesChange.select.many", "onEdgesChange", "onEdgesChange", "onEdgesChange.detached", "onEdgesChange.detached", "onEdgesChange.detached.single", "onEdgesChange.detached.single", "onEdgesChange.detached.many", "onEdgesChange.detached.many", "onEdgesChange.add", "onEdgesChange.add", "onEdgesChange.add.single", "onEdgesChange.add.single", "onEdgesChange.add.many", "onEdgesChange.add.many", "onEdgesChange.remove", "onEdgesChange.remove", "onEdgesChange.remove.single", "onEdgesChange.remove.single", "onEdgesChange.remove.many", "onEdgesChange.remove.many", "onEdgesChange.select", "onEdgesChange.select", "onEdgesChange.select.single", "onEdgesChange.select.single", "onEdgesChange.select.many", "onEdgesChange.select.many"] }], ngImport: i0, template: "<svg:svg\n rootSvgRef\n rootSvgContext\n rootPointer\n class=\"root-svg\"\n #flow\n [style.backgroundColor]=\"background\"\n [attr.width]=\"flowWidth()\"\n [attr.height]=\"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]" }, { kind: "directive", type: i12.RootPointerDirective, selector: "svg[rootPointer]" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
201
240
  }
202
241
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: VflowComponent, decorators: [{
203
242
  type: Component,
@@ -207,11 +246,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
207
246
  FlowStatusService,
208
247
  FlowEntitiesService,
209
248
  NodesChangeService,
210
- EdgeChangesService
249
+ EdgeChangesService,
250
+ NodeRenderingService,
251
+ SelectionService,
252
+ FlowSettingsService
211
253
  ], hostDirectives: [
212
254
  connectionControllerHostDirective,
213
255
  changesControllerHostDirective
214
- ], 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"] }]
256
+ ], template: "<svg:svg\n rootSvgRef\n rootSvgContext\n rootPointer\n class=\"root-svg\"\n #flow\n [style.backgroundColor]=\"background\"\n [attr.width]=\"flowWidth()\"\n [attr.height]=\"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"] }]
215
257
  }], propDecorators: { view: [{
216
258
  type: Input
217
259
  }], minZoom: [{
@@ -222,6 +264,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
222
264
  type: Input
223
265
  }], background: [{
224
266
  type: Input
267
+ }], entitiesSelectable: [{
268
+ type: Input
225
269
  }], connection: [{
226
270
  type: Input,
227
271
  args: [{ transform: (settings) => new ConnectionModel(settings) }]
@@ -246,7 +290,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
246
290
  type: ViewChild,
247
291
  args: [MapContextDirective]
248
292
  }] } });
249
- function bindFlowToNodes(flow, nodes) {
250
- nodes.forEach(n => n.bindFlow(flow));
251
- }
252
- //# 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":"AAAA,OAAO,EAAoB,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAgB,QAAQ,EAAE,KAAK,EAA4C,SAAS,EAAoB,MAAM,EAAE,qBAAqB,EAAE,MAAM,eAAe,CAAC;AAExO,OAAO,EAAE,mBAAmB,EAAE,MAAM,wCAAwC,CAAC;AAC7E,OAAO,EAAE,gBAAgB,EAAE,MAAM,kCAAkC,CAAC;AAEpE,OAAO,EAAE,eAAe,EAAE,MAAM,iCAAiC,CAAC;AAClE,OAAO,EAAE,YAAY,EAAE,QAAQ,EAAE,MAAM,4BAA4B,CAAC;AAGpE,OAAO,EAAE,2BAA2B,EAAE,8BAA8B,EAAE,qBAAqB,EAAE,yBAAyB,EAAE,MAAM,qCAAqC,CAAC;AAEpK,OAAO,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAC;AACjE,OAAO,EAAE,SAAS,EAAE,MAAM,yBAAyB,CAAC;AACpD,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAG5B,OAAO,EAAE,iBAAiB,EAAE,MAAM,oCAAoC,CAAC;AACvE,OAAO,EAAE,6BAA6B,EAAE,MAAM,kDAAkD,CAAC;AACjG,OAAO,EAAE,mBAAmB,EAAE,MAAM,sCAAsC,CAAC;AAE3E,OAAO,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAC;AAChE,OAAO,EAAE,eAAe,EAAE,MAAM,8BAA8B,CAAC;AAC/D,OAAO,EAAE,kBAAkB,EAAE,MAAM,qCAAqC,CAAC;AACzE,OAAO,EAAE,kBAAkB,EAAE,MAAM,qCAAqC,CAAC;AAEzE,OAAO,EAAE,0BAA0B,EAAE,MAAM,+CAA+C,CAAC;;;;;;;;;;;;;AAG3F,MAAM,iCAAiC,GAAG;IACxC,SAAS,EAAE,6BAA6B;IACxC,OAAO,EAAE,CAAC,WAAW,CAAC;CACvB,CAAA;AAED,MAAM,8BAA8B,GAAG;IACrC,SAAS,EAAE,0BAA0B;IACrC,OAAO,EAAE,CAAC,eAAe,EAAE,eAAe,CAAC;CAC5C,CAAA;AAoBD,MAAM,OAAO,cAAc;IAlB3B;QAmBE,aAAa;QACL,oBAAe,GAAG,MAAM,CAAC,eAAe,CAAC,CAAA;QACzC,wBAAmB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAA;QACjD,uBAAkB,GAAG,MAAM,CAAC,kBAAkB,CAAC,CAAA;QAC/C,uBAAkB,GAAG,MAAM,CAAC,kBAAkB,CAAC,CAAA;QAC/C,aAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAA;QAiBnC;;WAEG;QAEI,YAAO,GAAG,GAAG,CAAA;QAEpB;;WAEG;QAEI,YAAO,GAAG,CAAC,CAAA;QAelB;;WAEG;QAEI,eAAU,GAAW,SAAS,CAAA;QAqErC,aAAa;QAEb,qBAAqB;QACrB;;WAEG;QACa,aAAQ,GAAG,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,UAAU,EAAE,CAAA;QAE7E;;WAEG;QACa,gBAAW,GACzB,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,EAAE,YAAY,EAAE,EAAkB,EAAE,CAAC,CAAA;QAElF;;WAEG;QACa,gBAAW,GACzB,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,EAAE,YAAY,EAAE,EAAkB,EAAE,CAAC,CAAA;QAClF,aAAa;QAEb,iBAAiB;QACjB;;WAEG;QACa,oBAAe,GAAG,YAAY,CAAC,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC;aAClF,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAA,CAAC,wCAAwC;QAEzD;;WAEG;QACa,iBAAY,GAAG,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAA;QAE/D;;WAEG;QACa,iBAAY,GAAG,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAA;QAC/D,aAAa;QAEb,8CAA8C;QACpC,cAAS,GAAG,IAAI,SAAS,EAAE,CAAA;QAE3B,YAAO,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAA;KAsDrD;IAlNC,aAAa;IAEb,mBAAmB;IAEnB;;;;;;OAMG;IACH,IACW,IAAI,CAAC,IAA+B;QAC7C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAA;IAC/B,CAAC;IAcD;;;;;;;OAOG;IACH,IACW,eAAe,CAAC,eAAgC;QACzD,IAAI,CAAC,SAAS,CAAC,eAAe,CAAC,GAAG,CAAC,eAAe,CAAC,CAAA;IACrD,CAAC;IAQD;;;;OAIG;IACH,IACW,UAAU,CAAC,UAA2B,IAAI,IAAI,CAAC,mBAAmB,CAAC,UAAU,CAAC,GAAG,CAAC,UAAU,CAAC,CAAA,CAAC,CAAC;IAE1G,IAAW,UAAU,KAAK,OAAO,IAAI,CAAC,mBAAmB,CAAC,UAAU,EAAE,CAAA,CAAC,CAAC;IACxE,aAAa;IAEb,sBAAsB;IACtB;;OAEG;IACH,IACW,KAAK,CAAC,QAAgB;QAC/B,MAAM,SAAS,GAAG,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EACnD,GAAG,EAAE,CAAC,eAAe,CAAC,KAAK,CAAC,QAAQ,EAAE,IAAI,CAAC,mBAAmB,CAAC,KAAK,EAAE,CAAC,CACxE,CAAA;QAED,kCAAkC;QAClC,eAAe,CAAC,IAAI,CAAC,SAAS,EAAE,SAAS,CAAC,CAAA;QAE1C,yCAAyC;QACzC,eAAe,CAAC,SAAS,EAAE,IAAI,CAAC,mBAAmB,CAAC,KAAK,EAAE,CAAC,CAAA;QAE5D,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,CAAA;IAC/C,CAAC;IAED,IAAc,UAAU,KAAK,OAAO,IAAI,CAAC,mBAAmB,CAAC,KAAK,EAAE,CAAA,CAAC,CAAC;IAEtE;;OAEG;IACH,IACW,KAAK,CAAC,QAAgB;QAC/B,MAAM,SAAS,GAAG,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EACnD,GAAG,EAAE,CAAC,eAAe,CAAC,KAAK,CAAC,QAAQ,EAAE,IAAI,CAAC,mBAAmB,CAAC,KAAK,EAAE,CAAC,CACxE,CAAA;QAED,yCAAyC;QACzC,eAAe,CAAC,IAAI,CAAC,UAAU,EAAE,SAAS,CAAC,CAAA;QAE3C,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,CAAA;IAC/C,CAAC;IAED,IAAc,UAAU,KAAK,OAAO,IAAI,CAAC,mBAAmB,CAAC,UAAU,EAAE,CAAA,CAAC,CAAC;IAgE3E,sBAAsB;IACtB;;;;OAIG;IACI,UAAU,CAAC,QAAuB;QACvC,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,GAAG,CAAC,EAAE,UAAU,EAAE,UAAU,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC,CAAA;IACxF,CAAC;IAED;;;;OAIG;IACI,MAAM,CAAC,IAAY;QACxB,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,GAAG,CAAC,EAAE,UAAU,EAAE,UAAU,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,EAAE,CAAC,CAAA;IACxF,CAAC;IAED;;;;OAIG;IACI,KAAK,CAAC,KAAY;QACvB,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,GAAG,CAAC,EAAE,UAAU,EAAE,UAAU,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,CAAA;IACrF,CAAC;IAED;;;;OAIG;IACI,OAAO,CAAc,EAAU;QACpC,OAAO,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAI,EAAE,CAAC,EAAE,IAAI,CAAA;IACtD,CAAC;IAED;;OAEG;IACI,gBAAgB;QACrB,OAAO,IAAI,CAAC,mBAAmB,CAAC,gBAAgB,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAA;IACrE,CAAC;IACD,aAAa;IAEH,UAAU,CAAC,GAAW,EAAE,EAAE,IAAI,EAAa;QACnD,OAAO,IAAI,CAAA;IACb,CAAC;IAES,UAAU,CAAC,GAAW,EAAE,EAAE,IAAI,EAAa;QACnD,OAAO,IAAI,CAAA;IACb,CAAC;+GAxNU,cAAc;mGAAd,cAAc,4LA2DL,CAAC,QAA4B,EAAE,EAAE,CAAC,IAAI,eAAe,CAAC,QAAQ,CAAC,gDAxExE;YACT,gBAAgB;YAChB,eAAe;YACf,iBAAiB;YACjB,mBAAmB;YACnB,kBAAkB;YAClB,kBAAkB;SACnB,yEA+Ga,yBAAyB,wFAGzB,qBAAqB,yFAGrB,8BAA8B,8FAG9B,2BAA2B,4FAK9B,mBAAmB,qRC/KhC,ojCA4CA;;4FDYa,cAAc;kBAlB1B,SAAS;+BACE,OAAO,mBAGA,uBAAuB,CAAC,MAAM,aACpC;wBACT,gBAAgB;wBAChB,eAAe;wBACf,iBAAiB;wBACjB,mBAAmB;wBACnB,kBAAkB;wBAClB,kBAAkB;qBACnB,kBACe;wBACd,iCAAiC;wBACjC,8BAA8B;qBAC/B;8BAqBU,IAAI;sBADd,KAAK;gBASC,OAAO;sBADb,KAAK;gBAOC,OAAO;sBADb,KAAK;gBAYK,eAAe;sBADzB,KAAK;gBASC,UAAU;sBADhB,KAAK;gBASK,UAAU;sBADpB,KAAK;uBAAC,EAAE,SAAS,EAAE,CAAC,QAA4B,EAAE,EAAE,CAAC,IAAI,eAAe,CAAC,QAAQ,CAAC,EAAE;gBAW1E,KAAK;sBADf,KAAK;uBAAC,EAAE,QAAQ,EAAE,IAAI,EAAE;gBAqBd,KAAK;sBADf,KAAK;gBAiBI,iBAAiB;sBAD1B,YAAY;uBAAC,yBAAyB;gBAI7B,qBAAqB;sBAD9B,YAAY;uBAAC,qBAAqB;gBAIzB,sBAAsB;sBAD/B,YAAY;uBAAC,8BAA8B;gBAIlC,2BAA2B;sBADpC,YAAY;uBAAC,2BAA2B;gBAM/B,UAAU;sBADnB,SAAS;uBAAC,mBAAmB;;AAoGhC,SAAS,eAAe,CAAC,IAAe,EAAE,KAAkB;IAC1D,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAA;AACtC,CAAC","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 = runInInjectionContext(this.injector,\n      () => ReferenceKeeper.edges(newEdges, this.flowEntitiesService.edges())\n    )\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\n  /**\n   * Sync method to get detached edges\n   */\n  public getDetachedEdges(): Edge[] {\n    return this.flowEntitiesService.getDetachedEdges().map(e => e.edge)\n  }\n  // #endregion\n\n  protected trackNodes(idx: number, { node }: NodeModel) {\n    return node\n  }\n\n  protected trackEdges(idx: number, { edge }: EdgeModel) {\n    return edge\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"]}
293
+ //# 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":"AAAA,OAAO,EAAoB,uBAAuB,EAAE,SAAS,EAAE,YAAY,EAAgB,QAAQ,EAAE,KAAK,EAA4C,SAAS,EAAE,QAAQ,EAAU,MAAM,EAAE,qBAAqB,EAAE,MAAM,eAAe,CAAC;AAExO,OAAO,EAAE,mBAAmB,EAAE,MAAM,wCAAwC,CAAC;AAC7E,OAAO,EAAE,gBAAgB,EAAE,MAAM,kCAAkC,CAAC;AAEpE,OAAO,EAAE,eAAe,EAAE,MAAM,iCAAiC,CAAC;AAClE,OAAO,EAAE,YAAY,EAAE,QAAQ,EAAE,MAAM,4BAA4B,CAAC;AAGpE,OAAO,EAAE,2BAA2B,EAAE,8BAA8B,EAAE,qBAAqB,EAAE,yBAAyB,EAAE,MAAM,qCAAqC,CAAC;AAEpK,OAAO,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAC;AACjE,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAG5B,OAAO,EAAE,iBAAiB,EAAE,MAAM,oCAAoC,CAAC;AACvE,OAAO,EAAE,6BAA6B,EAAE,MAAM,kDAAkD,CAAC;AACjG,OAAO,EAAE,mBAAmB,EAAE,MAAM,sCAAsC,CAAC;AAE3E,OAAO,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAC;AAChE,OAAO,EAAE,eAAe,EAAE,MAAM,8BAA8B,CAAC;AAC/D,OAAO,EAAE,kBAAkB,EAAE,MAAM,qCAAqC,CAAC;AACzE,OAAO,EAAE,kBAAkB,EAAE,MAAM,qCAAqC,CAAC;AAEzE,OAAO,EAAE,0BAA0B,EAAE,MAAM,+CAA+C,CAAC;AAE3F,OAAO,EAAE,oBAAoB,EAAE,MAAM,uCAAuC,CAAC;AAC7E,OAAO,EAAE,gBAAgB,EAAE,MAAM,kCAAkC,CAAC;AACpE,OAAO,EAAE,mBAAmB,EAAE,MAAM,sCAAsC,CAAC;;;;;;;;;;;;;;AAE3E,MAAM,iCAAiC,GAAG;IACxC,SAAS,EAAE,6BAA6B;IACxC,OAAO,EAAE,CAAC,WAAW,CAAC;CACvB,CAAA;AAED,MAAM,8BAA8B,GAAG;IACrC,SAAS,EAAE,0BAA0B;IACrC,OAAO,EAAE;QACP,eAAe;QACf,wBAAwB;QACxB,+BAA+B;QAC/B,6BAA6B;QAC7B,mBAAmB;QACnB,0BAA0B;QAC1B,wBAAwB;QACxB,sBAAsB;QACtB,6BAA6B;QAC7B,2BAA2B;QAC3B,sBAAsB;QACtB,6BAA6B;QAC7B,2BAA2B;QAC3B,eAAe;QACf,wBAAwB;QACxB,+BAA+B;QAC/B,6BAA6B;QAC7B,mBAAmB;QACnB,0BAA0B;QAC1B,wBAAwB;QACxB,sBAAsB;QACtB,6BAA6B;QAC7B,2BAA2B;QAC3B,sBAAsB;QACtB,6BAA6B;QAC7B,2BAA2B;KAC5B;CACF,CAAA;AAuBD,MAAM,OAAO,cAAc;IArB3B;QAsBE,aAAa;QACL,oBAAe,GAAG,MAAM,CAAC,eAAe,CAAC,CAAA;QACzC,wBAAmB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAA;QACjD,uBAAkB,GAAG,MAAM,CAAC,kBAAkB,CAAC,CAAA;QAC/C,uBAAkB,GAAG,MAAM,CAAC,kBAAkB,CAAC,CAAA;QAC/C,yBAAoB,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAA;QACnD,wBAAmB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAA;QACjD,aAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAA;QAiBnC;;WAEG;QAEI,YAAO,GAAG,GAAG,CAAA;QAEpB;;WAEG;QAEI,YAAO,GAAG,CAAC,CAAA;QAelB;;WAEG;QAEI,eAAU,GAAW,SAAS,CAAA;QAqC3B,eAAU,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,KAAK,EAAE,CAAC,CAAA;QAiB9D,eAAU,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,mBAAmB,CAAC,UAAU,EAAE,CAAC,CAAA;QAoB5E,aAAa;QAEb,qBAAqB;QACrB;;WAEG;QACa,aAAQ,GAAG,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,UAAU,EAAE,CAAA;QAE7E;;WAEG;QACa,gBAAW,GACzB,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,EAAE,YAAY,EAAE,EAAkB,EAAE,CAAC,CAAA;QAElF;;WAEG;QACa,gBAAW,GACzB,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,EAAE,YAAY,EAAE,EAAkB,EAAE,CAAC,CAAA;QAClF,aAAa;QAEb,iBAAiB;QACjB;;WAEG;QACa,oBAAe,GAAG,YAAY,CAAC,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC;aAClF,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAA,CAAC,wCAAwC;QAEzD;;WAEG;QACa,iBAAY,GAAG,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAA;QAE/D;;WAEG;QACa,iBAAY,GAAG,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAA;QAC/D,aAAa;QAEH,cAAS,GAAG,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAA;QAC9C,eAAU,GAAG,IAAI,CAAC,mBAAmB,CAAC,UAAU,CAAA;QAEhD,YAAO,GAAG,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAA;KAsDrD;IAvNC,aAAa;IAEb,mBAAmB;IAEnB;;;;;;OAMG;IACH,IACW,IAAI,CAAC,IAA+B;QAC7C,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAA;IACzC,CAAC;IAcD;;;;;;;OAOG;IACH,IACW,eAAe,CAAC,eAAgC;QACzD,IAAI,CAAC,mBAAmB,CAAC,eAAe,CAAC,GAAG,CAAC,eAAe,CAAC,CAAA;IAC/D,CAAC;IAQD;;OAEG;IACH,IACW,kBAAkB,CAAC,KAAc;QAC1C,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,CAAC,GAAG,CAAC,KAAK,CAAC,CAAA;IACxD,CAAC;IAED;;;;OAIG;IACH,IACW,UAAU,CAAC,UAA2B,IAAI,IAAI,CAAC,mBAAmB,CAAC,UAAU,CAAC,GAAG,CAAC,UAAU,CAAC,CAAA,CAAC,CAAC;IAE1G,IAAW,UAAU,KAAK,OAAO,IAAI,CAAC,mBAAmB,CAAC,UAAU,EAAE,CAAA,CAAC,CAAC;IACxE,aAAa;IAEb,sBAAsB;IACtB;;OAEG;IACH,IACW,KAAK,CAAC,QAAgB;QAC/B,MAAM,SAAS,GAAG,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EACnD,GAAG,EAAE,CAAC,eAAe,CAAC,KAAK,CAAC,QAAQ,EAAE,IAAI,CAAC,mBAAmB,CAAC,KAAK,EAAE,CAAC,CACxE,CAAA;QAED,yCAAyC;QACzC,eAAe,CAAC,SAAS,EAAE,IAAI,CAAC,mBAAmB,CAAC,KAAK,EAAE,CAAC,CAAA;QAE5D,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,CAAA;IAC/C,CAAC;IAID;;OAEG;IACH,IACW,KAAK,CAAC,QAAgB;QAC/B,MAAM,SAAS,GAAG,qBAAqB,CAAC,IAAI,CAAC,QAAQ,EACnD,GAAG,EAAE,CAAC,eAAe,CAAC,KAAK,CAAC,QAAQ,EAAE,IAAI,CAAC,mBAAmB,CAAC,KAAK,EAAE,CAAC,CACxE,CAAA;QAED,yCAAyC;QACzC,eAAe,CAAC,IAAI,CAAC,UAAU,EAAE,EAAE,SAAS,CAAC,CAAA;QAE7C,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,CAAA;IAC/C,CAAC;IAkED,sBAAsB;IACtB;;;;OAIG;IACI,UAAU,CAAC,QAAuB;QACvC,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,GAAG,CAAC,EAAE,UAAU,EAAE,UAAU,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC,CAAA;IACxF,CAAC;IAED;;;;OAIG;IACI,MAAM,CAAC,IAAY;QACxB,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,GAAG,CAAC,EAAE,UAAU,EAAE,UAAU,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,EAAE,CAAC,CAAA;IACxF,CAAC;IAED;;;;OAIG;IACI,KAAK,CAAC,KAAY;QACvB,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,GAAG,CAAC,EAAE,UAAU,EAAE,UAAU,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,CAAA;IACrF,CAAC;IAED;;;;OAIG;IACI,OAAO,CAAc,EAAU;QACpC,OAAO,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAI,EAAE,CAAC,EAAE,IAAI,CAAA;IACtD,CAAC;IAED;;OAEG;IACI,gBAAgB;QACrB,OAAO,IAAI,CAAC,mBAAmB,CAAC,gBAAgB,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAA;IACrE,CAAC;IACD,aAAa;IAEH,UAAU,CAAC,GAAW,EAAE,EAAE,IAAI,EAAa;QACnD,OAAO,IAAI,CAAA;IACb,CAAC;IAES,UAAU,CAAC,GAAW,EAAE,EAAE,IAAI,EAAa;QACnD,OAAO,IAAI,CAAA;IACb,CAAC;+GA/NU,cAAc;mGAAd,cAAc,sOAqEL,CAAC,QAA4B,EAAE,EAAE,CAAC,IAAI,eAAe,CAAC,QAAQ,CAAC,gDArFxE;YACT,gBAAgB;YAChB,eAAe;YACf,iBAAiB;YACjB,mBAAmB;YACnB,kBAAkB;YAClB,kBAAkB;YAClB,oBAAoB;YACpB,gBAAgB;YAChB,mBAAmB;SACpB,yEAsHa,yBAAyB,wFAGzB,qBAAqB,yFAGrB,8BAA8B,8FAG9B,2BAA2B,4FAK9B,mBAAmB,ykDCtNhC,mjCA6CA;;4FD2Ca,cAAc;kBArB1B,SAAS;+BACE,OAAO,mBAGA,uBAAuB,CAAC,MAAM,aACpC;wBACT,gBAAgB;wBAChB,eAAe;wBACf,iBAAiB;wBACjB,mBAAmB;wBACnB,kBAAkB;wBAClB,kBAAkB;wBAClB,oBAAoB;wBACpB,gBAAgB;wBAChB,mBAAmB;qBACpB,kBACe;wBACd,iCAAiC;wBACjC,8BAA8B;qBAC/B;8BAuBU,IAAI;sBADd,KAAK;gBASC,OAAO;sBADb,KAAK;gBAOC,OAAO;sBADb,KAAK;gBAYK,eAAe;sBADzB,KAAK;gBASC,UAAU;sBADhB,KAAK;gBAOK,kBAAkB;sBAD5B,KAAK;gBAWK,UAAU;sBADpB,KAAK;uBAAC,EAAE,SAAS,EAAE,CAAC,QAA4B,EAAE,EAAE,CAAC,IAAI,eAAe,CAAC,QAAQ,CAAC,EAAE;gBAW1E,KAAK;sBADf,KAAK;uBAAC,EAAE,QAAQ,EAAE,IAAI,EAAE;gBAkBd,KAAK;sBADf,KAAK;gBAiBI,iBAAiB;sBAD1B,YAAY;uBAAC,yBAAyB;gBAI7B,qBAAqB;sBAD9B,YAAY;uBAAC,qBAAqB;gBAIzB,sBAAsB;sBAD/B,YAAY;uBAAC,8BAA8B;gBAIlC,2BAA2B;sBADpC,YAAY;uBAAC,2BAA2B;gBAM/B,UAAU;sBADnB,SAAS;uBAAC,mBAAmB","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 { 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';\nimport { NodeRenderingService } from '../../services/node-rendering.service';\nimport { SelectionService } from '../../services/selection.service';\nimport { FlowSettingsService } from '../../services/flow-settings.service';\n\nconst connectionControllerHostDirective = {\n  directive: ConnectionControllerDirective,\n  outputs: ['onConnect']\n}\n\nconst changesControllerHostDirective = {\n  directive: ChangesControllerDirective,\n  outputs: [\n    'onNodesChange',\n    'onNodesChange.position',\n    'onNodesChange.position.single',\n    'onNodesChange.position.many',\n    'onNodesChange.add',\n    'onNodesChange.add.single',\n    'onNodesChange.add.many',\n    'onNodesChange.remove',\n    'onNodesChange.remove.single',\n    'onNodesChange.remove.many',\n    'onNodesChange.select',\n    'onNodesChange.select.single',\n    'onNodesChange.select.many',\n    'onEdgesChange',\n    'onEdgesChange.detached',\n    'onEdgesChange.detached.single',\n    'onEdgesChange.detached.many',\n    'onEdgesChange.add',\n    'onEdgesChange.add.single',\n    'onEdgesChange.add.many',\n    'onEdgesChange.remove',\n    'onEdgesChange.remove.single',\n    'onEdgesChange.remove.many',\n    'onEdgesChange.select',\n    'onEdgesChange.select.single',\n    'onEdgesChange.select.many',\n  ]\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    NodeRenderingService,\n    SelectionService,\n    FlowSettingsService\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 nodeRenderingService = inject(NodeRenderingService)\n  private flowSettingsService = inject(FlowSettingsService)\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.flowSettingsService.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.flowSettingsService.handlePositions.set(handlePositions)\n  }\n\n  /**\n   * Background color for flow\n   */\n  @Input()\n  public background: string = '#FFFFFF'\n\n  /**\n   * Global rule if you can or can't select entities\n   */\n  @Input()\n  public set entitiesSelectable(value: boolean) {\n    this.flowSettingsService.entitiesSelectable.set(value)\n  }\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    // quick and dirty binding nodes to edges\n    addNodesToEdges(newModels, this.flowEntitiesService.edges())\n\n    this.flowEntitiesService.nodes.set(newModels)\n  }\n\n  protected nodeModels = computed(() => this.nodeRenderingService.nodes())\n\n  /**\n   * Edges to render\n   */\n  @Input()\n  public set edges(newEdges: Edge[]) {\n    const newModels = runInInjectionContext(this.injector,\n      () => ReferenceKeeper.edges(newEdges, this.flowEntitiesService.edges())\n    )\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 edgeModels = computed(() => 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 edges change\n   */\n  public readonly edgesChange$ = this.edgesChangeService.changes$\n  // #endregion\n\n  protected flowWidth = this.flowSettingsService.flowWidth\n  protected flowHeight = this.flowSettingsService.flowHeight\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\n  /**\n   * Sync method to get detached edges\n   */\n  public getDetachedEdges(): Edge[] {\n    return this.flowEntitiesService.getDetachedEdges().map(e => e.edge)\n  }\n  // #endregion\n\n  protected trackNodes(idx: number, { node }: NodeModel) {\n    return node\n  }\n\n  protected trackEdges(idx: number, { edge }: EdgeModel) {\n    return edge\n  }\n}\n\n\n","<svg:svg\n  rootSvgRef\n  rootSvgContext\n  rootPointer\n  class=\"root-svg\"\n  #flow\n  [style.backgroundColor]=\"background\"\n  [attr.width]=\"flowWidth()\"\n  [attr.height]=\"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"]}
@@ -1,9 +1,8 @@
1
- import { Directive, Injector, inject, runInInjectionContext } from "@angular/core";
2
- import * as i0 from "@angular/core";
1
+ import { runInInjectionContext } from "@angular/core";
3
2
  export function InjectionContext(target, key, descriptor) {
4
3
  const originalMethod = descriptor.value;
5
4
  descriptor.value = function (...args) {
6
- if (this instanceof WithInjectorDirective) {
5
+ if (implementsWithInjector(this)) {
7
6
  return runInInjectionContext(this.injector, () => originalMethod.apply(this, args));
8
7
  }
9
8
  else {
@@ -13,14 +12,7 @@ export function InjectionContext(target, key, descriptor) {
13
12
  // Return the modified descriptor
14
13
  return descriptor;
15
14
  }
16
- export class WithInjectorDirective {
17
- constructor() {
18
- this.injector = inject(Injector);
19
- }
20
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: WithInjectorDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
21
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: WithInjectorDirective, ngImport: i0 }); }
22
- }
23
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: WithInjectorDirective, decorators: [{
24
- type: Directive
25
- }] });
26
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnVuLWluLWluamVjdGlvbi1jb250ZXh0LmRlY29yYXRvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9kZWNvcmF0b3JzL3J1bi1pbi1pbmplY3Rpb24tY29udGV4dC5kZWNvcmF0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxRQUFRLEVBQUUsTUFBTSxFQUFFLHFCQUFxQixFQUFFLE1BQU0sZUFBZSxDQUFDOztBQUVuRixNQUFNLFVBQVUsZ0JBQWdCLENBQUMsTUFBVyxFQUFFLEdBQVcsRUFBRSxVQUE4QjtJQUN2RixNQUFNLGNBQWMsR0FBRyxVQUFVLENBQUMsS0FBSyxDQUFDO0lBRXhDLFVBQVUsQ0FBQyxLQUFLLEdBQUcsVUFBVSxHQUFHLElBQVc7UUFDekMsSUFBSSxJQUFJLFlBQVkscUJBQXFCLEVBQUU7WUFDekMsT0FBTyxxQkFBcUIsQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLEdBQUcsRUFBRSxDQUFDLGNBQWMsQ0FBQyxLQUFLLENBQUMsSUFBSSxFQUFFLElBQUksQ0FBQyxDQUFDLENBQUE7U0FDcEY7YUFBTTtZQUNMLE1BQU0sSUFBSSxLQUFLLENBQUMsK0VBQStFLENBQUMsQ0FBQTtTQUNqRztJQUNILENBQUMsQ0FBQztJQUVGLGlDQUFpQztJQUNqQyxPQUFPLFVBQVUsQ0FBQztBQUNwQixDQUFDO0FBR0QsTUFBTSxPQUFnQixxQkFBcUI7SUFEM0M7UUFFa0IsYUFBUSxHQUFHLE1BQU0sQ0FBQyxRQUFRLENBQUMsQ0FBQTtLQUM1QzsrR0FGcUIscUJBQXFCO21HQUFyQixxQkFBcUI7OzRGQUFyQixxQkFBcUI7a0JBRDFDLFNBQVMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXJlY3RpdmUsIEluamVjdG9yLCBpbmplY3QsIHJ1bkluSW5qZWN0aW9uQ29udGV4dCB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBJbmplY3Rpb25Db250ZXh0KHRhcmdldDogYW55LCBrZXk6IHN0cmluZywgZGVzY3JpcHRvcjogUHJvcGVydHlEZXNjcmlwdG9yKSB7XG4gIGNvbnN0IG9yaWdpbmFsTWV0aG9kID0gZGVzY3JpcHRvci52YWx1ZTtcblxuICBkZXNjcmlwdG9yLnZhbHVlID0gZnVuY3Rpb24gKC4uLmFyZ3M6IGFueVtdKSB7XG4gICAgaWYgKHRoaXMgaW5zdGFuY2VvZiBXaXRoSW5qZWN0b3JEaXJlY3RpdmUpIHtcbiAgICAgIHJldHVybiBydW5JbkluamVjdGlvbkNvbnRleHQodGhpcy5pbmplY3RvciwgKCkgPT4gb3JpZ2luYWxNZXRob2QuYXBwbHkodGhpcywgYXJncykpXG4gICAgfSBlbHNlIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignQ2xhc3MgdGhhdCBjb250YWlucyBkZWNvcmF0ZWQgbWV0aG9kIG11c3QgZXh0ZW5kcyBXaXRoSW5qZWN0b3JEaXJlY3RpdmUgY2xhc3MnKVxuICAgIH1cbiAgfTtcblxuICAvLyBSZXR1cm4gdGhlIG1vZGlmaWVkIGRlc2NyaXB0b3JcbiAgcmV0dXJuIGRlc2NyaXB0b3I7XG59XG5cbkBEaXJlY3RpdmUoKVxuZXhwb3J0IGFic3RyYWN0IGNsYXNzIFdpdGhJbmplY3RvckRpcmVjdGl2ZSB7XG4gIHB1YmxpYyByZWFkb25seSBpbmplY3RvciA9IGluamVjdChJbmplY3Rvcilcbn1cbiJdfQ==
15
+ const implementsWithInjector = (instance) => {
16
+ return 'injector' in instance && 'get' in instance.injector;
17
+ };
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnVuLWluLWluamVjdGlvbi1jb250ZXh0LmRlY29yYXRvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9kZWNvcmF0b3JzL3J1bi1pbi1pbmplY3Rpb24tY29udGV4dC5kZWNvcmF0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUErQixxQkFBcUIsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUVuRixNQUFNLFVBQVUsZ0JBQWdCLENBQUMsTUFBVyxFQUFFLEdBQVcsRUFBRSxVQUE4QjtJQUN2RixNQUFNLGNBQWMsR0FBRyxVQUFVLENBQUMsS0FBSyxDQUFDO0lBRXhDLFVBQVUsQ0FBQyxLQUFLLEdBQUcsVUFBVSxHQUFHLElBQVc7UUFDekMsSUFBSSxzQkFBc0IsQ0FBQyxJQUFJLENBQUMsRUFBRTtZQUNoQyxPQUFPLHFCQUFxQixDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsR0FBRyxFQUFFLENBQUMsY0FBYyxDQUFDLEtBQUssQ0FBQyxJQUFJLEVBQUUsSUFBSSxDQUFDLENBQUMsQ0FBQTtTQUNwRjthQUFNO1lBQ0wsTUFBTSxJQUFJLEtBQUssQ0FBQywrRUFBK0UsQ0FBQyxDQUFBO1NBQ2pHO0lBQ0gsQ0FBQyxDQUFDO0lBRUYsaUNBQWlDO0lBQ2pDLE9BQU8sVUFBVSxDQUFDO0FBQ3BCLENBQUM7QUFNRCxNQUFNLHNCQUFzQixHQUFHLENBQUMsUUFBWSxFQUE0QixFQUFFO0lBQ3hFLE9BQU8sVUFBVSxJQUFJLFFBQVEsSUFBSSxLQUFLLElBQUssUUFBUSxDQUFDLFFBQXFCLENBQUM7QUFDNUUsQ0FBQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlLCBJbmplY3RvciwgaW5qZWN0LCBydW5JbkluamVjdGlvbkNvbnRleHQgfSBmcm9tIFwiQGFuZ3VsYXIvY29yZVwiO1xuXG5leHBvcnQgZnVuY3Rpb24gSW5qZWN0aW9uQ29udGV4dCh0YXJnZXQ6IGFueSwga2V5OiBzdHJpbmcsIGRlc2NyaXB0b3I6IFByb3BlcnR5RGVzY3JpcHRvcikge1xuICBjb25zdCBvcmlnaW5hbE1ldGhvZCA9IGRlc2NyaXB0b3IudmFsdWU7XG5cbiAgZGVzY3JpcHRvci52YWx1ZSA9IGZ1bmN0aW9uICguLi5hcmdzOiBhbnlbXSkge1xuICAgIGlmIChpbXBsZW1lbnRzV2l0aEluamVjdG9yKHRoaXMpKSB7XG4gICAgICByZXR1cm4gcnVuSW5JbmplY3Rpb25Db250ZXh0KHRoaXMuaW5qZWN0b3IsICgpID0+IG9yaWdpbmFsTWV0aG9kLmFwcGx5KHRoaXMsIGFyZ3MpKVxuICAgIH0gZWxzZSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ0NsYXNzIHRoYXQgY29udGFpbnMgZGVjb3JhdGVkIG1ldGhvZCBtdXN0IGV4dGVuZHMgV2l0aEluamVjdG9yRGlyZWN0aXZlIGNsYXNzJylcbiAgICB9XG4gIH07XG5cbiAgLy8gUmV0dXJuIHRoZSBtb2RpZmllZCBkZXNjcmlwdG9yXG4gIHJldHVybiBkZXNjcmlwdG9yO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFdpdGhJbmplY3RvciB7XG4gIGluamVjdG9yOiBJbmplY3RvclxufVxuXG5jb25zdCBpbXBsZW1lbnRzV2l0aEluamVjdG9yID0gKGluc3RhbmNlOiB7fSk6IGluc3RhbmNlIGlzIFdpdGhJbmplY3RvciA9PiB7XG4gIHJldHVybiAnaW5qZWN0b3InIGluIGluc3RhbmNlICYmICdnZXQnIGluIChpbnN0YW5jZS5pbmplY3RvciBhcyBJbmplY3Rvcik7XG59XG4iXX0=
@@ -1,8 +1,7 @@
1
- import { Directive, EventEmitter, Output, inject } from '@angular/core';
1
+ import { Directive, Output, inject } from '@angular/core';
2
2
  import { EdgeChangesService } from '../services/edge-changes.service';
3
3
  import { NodesChangeService } from '../services/node-changes.service';
4
- import { tap } from 'rxjs';
5
- import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
4
+ import { filter, map } from 'rxjs';
6
5
  import * as i0 from "@angular/core";
7
6
  export class ChangesControllerDirective {
8
7
  constructor() {
@@ -11,20 +10,50 @@ export class ChangesControllerDirective {
11
10
  /**
12
11
  * Watch nodes change
13
12
  */
14
- this.onNodesChange = new EventEmitter();
13
+ this.onNodesChange = this.nodesChangeService.changes$;
14
+ this.onNodesChangePosition = this.nodeChangesOfType('position');
15
+ this.onNodesChangePositionSignle = this.singleChange(this.nodeChangesOfType('position'));
16
+ this.onNodesChangePositionMany = this.manyChanges(this.nodeChangesOfType('position'));
17
+ this.onNodesChangeAdd = this.nodeChangesOfType('add');
18
+ this.onNodesChangeAddSingle = this.singleChange(this.nodeChangesOfType('add'));
19
+ this.onNodesChangeAddMany = this.manyChanges(this.nodeChangesOfType('add'));
20
+ this.onNodesChangeRemove = this.nodeChangesOfType('remove');
21
+ this.onNodesChangeRemoveSingle = this.singleChange(this.nodeChangesOfType('remove'));
22
+ this.onNodesChangeRemoveMany = this.manyChanges(this.nodeChangesOfType('remove'));
23
+ this.onNodesChangeSelect = this.nodeChangesOfType('select');
24
+ this.onNodesChangeSelectSingle = this.singleChange(this.nodeChangesOfType('select'));
25
+ this.onNodesChangeSelectMany = this.manyChanges(this.nodeChangesOfType('select'));
15
26
  /**
16
- * Watch nodes change
27
+ * Watch edges change
17
28
  */
18
- this.onEdgesChange = new EventEmitter();
19
- this.nodesChangeProxySubscription = this.nodesChangeService.changes$
20
- .pipe(tap((changes) => this.onNodesChange.emit(changes)), takeUntilDestroyed())
21
- .subscribe();
22
- this.edgesChangeProxySubscription = this.edgesChangeService.changes$
23
- .pipe(tap((changes) => this.onEdgesChange.emit(changes)), takeUntilDestroyed())
24
- .subscribe();
29
+ this.onEdgesChange = this.edgesChangeService.changes$;
30
+ this.onNodesChangeDetached = this.edgeChangesOfType('detached');
31
+ this.onNodesChangeDetachedSingle = this.singleChange(this.edgeChangesOfType('detached'));
32
+ this.onNodesChangeDetachedMany = this.manyChanges(this.edgeChangesOfType('detached'));
33
+ this.onEdgesChangeAdd = this.edgeChangesOfType('add');
34
+ this.onEdgeChangeAddSingle = this.singleChange(this.edgeChangesOfType('add'));
35
+ this.onEdgeChangeAddMany = this.manyChanges(this.edgeChangesOfType('add'));
36
+ this.onEdgeChangeRemove = this.edgeChangesOfType('remove');
37
+ this.onEdgeChangeRemoveSingle = this.singleChange(this.edgeChangesOfType('remove'));
38
+ this.onEdgeChangeRemoveMany = this.manyChanges(this.edgeChangesOfType('remove'));
39
+ this.onEdgeChangeSelect = this.edgeChangesOfType('select');
40
+ this.onEdgeChangeSelectSingle = this.singleChange(this.edgeChangesOfType('select'));
41
+ this.onEdgeChangeSelectMany = this.manyChanges(this.edgeChangesOfType('select'));
42
+ }
43
+ nodeChangesOfType(type) {
44
+ return this.nodesChangeService.changes$.pipe(map(changes => changes.filter((c) => c.type === type)), filter(changes => !!changes.length));
45
+ }
46
+ edgeChangesOfType(type) {
47
+ return this.edgesChangeService.changes$.pipe(map(changes => changes.filter((c) => c.type === type)), filter(changes => !!changes.length));
48
+ }
49
+ singleChange(changes$) {
50
+ return changes$.pipe(filter(changes => changes.length === 1), map(([first]) => first));
51
+ }
52
+ manyChanges(changes$) {
53
+ return changes$.pipe(filter(changes => changes.length > 1));
25
54
  }
26
55
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ChangesControllerDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
27
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: ChangesControllerDirective, isStandalone: true, selector: "[changesController]", outputs: { onNodesChange: "onNodesChange", onEdgesChange: "onEdgesChange" }, ngImport: i0 }); }
56
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: ChangesControllerDirective, isStandalone: true, selector: "[changesController]", outputs: { onNodesChange: "onNodesChange", onNodesChangePosition: "onNodesChange.position", onNodesChangePositionSignle: "onNodesChange.position.single", onNodesChangePositionMany: "onNodesChange.position.many", onNodesChangeAdd: "onNodesChange.add", onNodesChangeAddSingle: "onNodesChange.add.single", onNodesChangeAddMany: "onNodesChange.add.many", onNodesChangeRemove: "onNodesChange.remove", onNodesChangeRemoveSingle: "onNodesChange.remove.single", onNodesChangeRemoveMany: "onNodesChange.remove.many", onNodesChangeSelect: "onNodesChange.select", onNodesChangeSelectSingle: "onNodesChange.select.single", onNodesChangeSelectMany: "onNodesChange.select.many", onEdgesChange: "onEdgesChange", onNodesChangeDetached: "onEdgesChange.detached", onNodesChangeDetachedSingle: "onEdgesChange.detached.single", onNodesChangeDetachedMany: "onEdgesChange.detached.many", onEdgesChangeAdd: "onEdgesChange.add", onEdgeChangeAddSingle: "onEdgesChange.add.single", onEdgeChangeAddMany: "onEdgesChange.add.many", onEdgeChangeRemove: "onEdgesChange.remove", onEdgeChangeRemoveSingle: "onEdgesChange.remove.single", onEdgeChangeRemoveMany: "onEdgesChange.remove.many", onEdgeChangeSelect: "onEdgesChange.select", onEdgeChangeSelectSingle: "onEdgesChange.select.single", onEdgeChangeSelectMany: "onEdgesChange.select.many" }, ngImport: i0 }); }
28
57
  }
29
58
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ChangesControllerDirective, decorators: [{
30
59
  type: Directive,
@@ -34,7 +63,79 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
34
63
  }]
35
64
  }], propDecorators: { onNodesChange: [{
36
65
  type: Output
66
+ }], onNodesChangePosition: [{
67
+ type: Output,
68
+ args: ['onNodesChange.position']
69
+ }], onNodesChangePositionSignle: [{
70
+ type: Output,
71
+ args: ['onNodesChange.position.single']
72
+ }], onNodesChangePositionMany: [{
73
+ type: Output,
74
+ args: ['onNodesChange.position.many']
75
+ }], onNodesChangeAdd: [{
76
+ type: Output,
77
+ args: ['onNodesChange.add']
78
+ }], onNodesChangeAddSingle: [{
79
+ type: Output,
80
+ args: ['onNodesChange.add.single']
81
+ }], onNodesChangeAddMany: [{
82
+ type: Output,
83
+ args: ['onNodesChange.add.many']
84
+ }], onNodesChangeRemove: [{
85
+ type: Output,
86
+ args: ['onNodesChange.remove']
87
+ }], onNodesChangeRemoveSingle: [{
88
+ type: Output,
89
+ args: ['onNodesChange.remove.single']
90
+ }], onNodesChangeRemoveMany: [{
91
+ type: Output,
92
+ args: ['onNodesChange.remove.many']
93
+ }], onNodesChangeSelect: [{
94
+ type: Output,
95
+ args: ['onNodesChange.select']
96
+ }], onNodesChangeSelectSingle: [{
97
+ type: Output,
98
+ args: ['onNodesChange.select.single']
99
+ }], onNodesChangeSelectMany: [{
100
+ type: Output,
101
+ args: ['onNodesChange.select.many']
37
102
  }], onEdgesChange: [{
38
103
  type: Output
104
+ }], onNodesChangeDetached: [{
105
+ type: Output,
106
+ args: ['onEdgesChange.detached']
107
+ }], onNodesChangeDetachedSingle: [{
108
+ type: Output,
109
+ args: ['onEdgesChange.detached.single']
110
+ }], onNodesChangeDetachedMany: [{
111
+ type: Output,
112
+ args: ['onEdgesChange.detached.many']
113
+ }], onEdgesChangeAdd: [{
114
+ type: Output,
115
+ args: ['onEdgesChange.add']
116
+ }], onEdgeChangeAddSingle: [{
117
+ type: Output,
118
+ args: ['onEdgesChange.add.single']
119
+ }], onEdgeChangeAddMany: [{
120
+ type: Output,
121
+ args: ['onEdgesChange.add.many']
122
+ }], onEdgeChangeRemove: [{
123
+ type: Output,
124
+ args: ['onEdgesChange.remove']
125
+ }], onEdgeChangeRemoveSingle: [{
126
+ type: Output,
127
+ args: ['onEdgesChange.remove.single']
128
+ }], onEdgeChangeRemoveMany: [{
129
+ type: Output,
130
+ args: ['onEdgesChange.remove.many']
131
+ }], onEdgeChangeSelect: [{
132
+ type: Output,
133
+ args: ['onEdgesChange.select']
134
+ }], onEdgeChangeSelectSingle: [{
135
+ type: Output,
136
+ args: ['onEdgesChange.select.single']
137
+ }], onEdgeChangeSelectMany: [{
138
+ type: Output,
139
+ args: ['onEdgesChange.select.many']
39
140
  }] } });
40
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhbmdlcy1jb250cm9sbGVyLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9kaXJlY3RpdmVzL2NoYW5nZXMtY29udHJvbGxlci5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxZQUFZLEVBQUUsTUFBTSxFQUFVLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUVoRixPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUN0RSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUN0RSxPQUFPLEVBQUUsR0FBRyxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBQzNCLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLDRCQUE0QixDQUFDOztBQU9oRSxNQUFNLE9BQU8sMEJBQTBCO0lBSnZDO1FBS1ksdUJBQWtCLEdBQUcsTUFBTSxDQUFDLGtCQUFrQixDQUFDLENBQUE7UUFDL0MsdUJBQWtCLEdBQUcsTUFBTSxDQUFDLGtCQUFrQixDQUFDLENBQUE7UUFFekQ7O1dBRUc7UUFFSSxrQkFBYSxHQUFHLElBQUksWUFBWSxFQUFnQixDQUFBO1FBRXZEOztXQUVHO1FBRUksa0JBQWEsR0FBRyxJQUFJLFlBQVksRUFBZ0IsQ0FBQTtRQUU3QyxpQ0FBNEIsR0FBRyxJQUFJLENBQUMsa0JBQWtCLENBQUMsUUFBUTthQUN0RSxJQUFJLENBQ0gsR0FBRyxDQUFDLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQyxFQUNsRCxrQkFBa0IsRUFBRSxDQUNyQjthQUNBLFNBQVMsRUFBRSxDQUFBO1FBRUosaUNBQTRCLEdBQUcsSUFBSSxDQUFDLGtCQUFrQixDQUFDLFFBQVE7YUFDdEUsSUFBSSxDQUNILEdBQUcsQ0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUMsRUFDbEQsa0JBQWtCLEVBQUUsQ0FDckI7YUFDQSxTQUFTLEVBQUUsQ0FBQTtLQUNmOytHQTdCWSwwQkFBMEI7bUdBQTFCLDBCQUEwQjs7NEZBQTFCLDBCQUEwQjtrQkFKdEMsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUscUJBQXFCO29CQUMvQixVQUFVLEVBQUUsSUFBSTtpQkFDakI7OEJBU1EsYUFBYTtzQkFEbkIsTUFBTTtnQkFPQSxhQUFhO3NCQURuQixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlLCBFdmVudEVtaXR0ZXIsIE91dHB1dCwgZWZmZWN0LCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IE5vZGVDaGFuZ2UgfSBmcm9tICcuLi90eXBlcy9ub2RlLWNoYW5nZS50eXBlJztcbmltcG9ydCB7IEVkZ2VDaGFuZ2VzU2VydmljZSB9IGZyb20gJy4uL3NlcnZpY2VzL2VkZ2UtY2hhbmdlcy5zZXJ2aWNlJztcbmltcG9ydCB7IE5vZGVzQ2hhbmdlU2VydmljZSB9IGZyb20gJy4uL3NlcnZpY2VzL25vZGUtY2hhbmdlcy5zZXJ2aWNlJztcbmltcG9ydCB7IHRhcCB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgdGFrZVVudGlsRGVzdHJveWVkIH0gZnJvbSAnQGFuZ3VsYXIvY29yZS9yeGpzLWludGVyb3AnO1xuaW1wb3J0IHsgRWRnZUNoYW5nZSB9IGZyb20gJy4uL3R5cGVzL2VkZ2UtY2hhbmdlLnR5cGUnO1xuXG5ARGlyZWN0aXZlKHtcbiAgc2VsZWN0b3I6ICdbY2hhbmdlc0NvbnRyb2xsZXJdJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZVxufSlcbmV4cG9ydCBjbGFzcyBDaGFuZ2VzQ29udHJvbGxlckRpcmVjdGl2ZSB7XG4gIHByb3RlY3RlZCBub2Rlc0NoYW5nZVNlcnZpY2UgPSBpbmplY3QoTm9kZXNDaGFuZ2VTZXJ2aWNlKVxuICBwcm90ZWN0ZWQgZWRnZXNDaGFuZ2VTZXJ2aWNlID0gaW5qZWN0KEVkZ2VDaGFuZ2VzU2VydmljZSlcblxuICAvKipcbiAgICogV2F0Y2ggbm9kZXMgY2hhbmdlXG4gICAqL1xuICBAT3V0cHV0KClcbiAgcHVibGljIG9uTm9kZXNDaGFuZ2UgPSBuZXcgRXZlbnRFbWl0dGVyPE5vZGVDaGFuZ2VbXT4oKVxuXG4gIC8qKlxuICAgKiBXYXRjaCBub2RlcyBjaGFuZ2VcbiAgICovXG4gIEBPdXRwdXQoKVxuICBwdWJsaWMgb25FZGdlc0NoYW5nZSA9IG5ldyBFdmVudEVtaXR0ZXI8RWRnZUNoYW5nZVtdPigpXG5cbiAgcHJvdGVjdGVkIG5vZGVzQ2hhbmdlUHJveHlTdWJzY3JpcHRpb24gPSB0aGlzLm5vZGVzQ2hhbmdlU2VydmljZS5jaGFuZ2VzJFxuICAgIC5waXBlKFxuICAgICAgdGFwKChjaGFuZ2VzKSA9PiB0aGlzLm9uTm9kZXNDaGFuZ2UuZW1pdChjaGFuZ2VzKSksXG4gICAgICB0YWtlVW50aWxEZXN0cm95ZWQoKVxuICAgIClcbiAgICAuc3Vic2NyaWJlKClcblxuICBwcm90ZWN0ZWQgZWRnZXNDaGFuZ2VQcm94eVN1YnNjcmlwdGlvbiA9IHRoaXMuZWRnZXNDaGFuZ2VTZXJ2aWNlLmNoYW5nZXMkXG4gICAgLnBpcGUoXG4gICAgICB0YXAoKGNoYW5nZXMpID0+IHRoaXMub25FZGdlc0NoYW5nZS5lbWl0KGNoYW5nZXMpKSxcbiAgICAgIHRha2VVbnRpbERlc3Ryb3llZCgpXG4gICAgKVxuICAgIC5zdWJzY3JpYmUoKVxufVxuIl19
141
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"changes-controller.directive.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/directives/changes-controller.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, Output, inject } from '@angular/core';\nimport { NodeAddChange, NodeChange, NodePositionChange, NodeRemoveChange, NodeSelectedChange } from '../types/node-change.type';\nimport { EdgeChangesService } from '../services/edge-changes.service';\nimport { NodesChangeService } from '../services/node-changes.service';\nimport { Observable, filter, map } from 'rxjs';\nimport { EdgeAddChange, EdgeChange, EdgeDetachedChange, EdgeRemoveChange, EdgeSelectChange } from '../types/edge-change.type';\n\n@Directive({\n  selector: '[changesController]',\n  standalone: true\n})\nexport class ChangesControllerDirective {\n  protected nodesChangeService = inject(NodesChangeService)\n  protected edgesChangeService = inject(EdgeChangesService)\n\n  /**\n   * Watch nodes change\n   */\n  @Output()\n  public onNodesChange = this.nodesChangeService.changes$\n\n  @Output('onNodesChange.position')\n  public onNodesChangePosition = this.nodeChangesOfType('position')\n\n  @Output('onNodesChange.position.single')\n  public onNodesChangePositionSignle = this.singleChange(\n    this.nodeChangesOfType('position')\n  )\n\n  @Output('onNodesChange.position.many')\n  public onNodesChangePositionMany = this.manyChanges(\n    this.nodeChangesOfType('position')\n  )\n\n  @Output('onNodesChange.add')\n  public onNodesChangeAdd = this.nodeChangesOfType('add')\n\n  @Output('onNodesChange.add.single')\n  public onNodesChangeAddSingle = this.singleChange(\n    this.nodeChangesOfType('add')\n  )\n\n  @Output('onNodesChange.add.many')\n  public onNodesChangeAddMany = this.manyChanges(\n    this.nodeChangesOfType('add')\n  )\n\n  @Output('onNodesChange.remove')\n  public onNodesChangeRemove = this.nodeChangesOfType('remove')\n\n  @Output('onNodesChange.remove.single')\n  public onNodesChangeRemoveSingle = this.singleChange(\n    this.nodeChangesOfType('remove')\n  )\n\n  @Output('onNodesChange.remove.many')\n  public onNodesChangeRemoveMany = this.manyChanges(\n    this.nodeChangesOfType('remove')\n  )\n\n  @Output('onNodesChange.select')\n  public onNodesChangeSelect = this.nodeChangesOfType('select')\n\n  @Output('onNodesChange.select.single')\n  public onNodesChangeSelectSingle = this.singleChange(\n    this.nodeChangesOfType('select')\n  )\n\n  @Output('onNodesChange.select.many')\n  public onNodesChangeSelectMany = this.manyChanges(\n    this.nodeChangesOfType('select')\n  )\n\n  /**\n   * Watch edges change\n   */\n  @Output()\n  public onEdgesChange = this.edgesChangeService.changes$\n\n  @Output('onEdgesChange.detached')\n  public onNodesChangeDetached = this.edgeChangesOfType('detached')\n\n  @Output('onEdgesChange.detached.single')\n  public onNodesChangeDetachedSingle = this.singleChange(\n    this.edgeChangesOfType('detached')\n  )\n\n  @Output('onEdgesChange.detached.many')\n  public onNodesChangeDetachedMany = this.manyChanges(\n    this.edgeChangesOfType('detached')\n  )\n\n  @Output('onEdgesChange.add')\n  public onEdgesChangeAdd = this.edgeChangesOfType('add')\n\n  @Output('onEdgesChange.add.single')\n  public onEdgeChangeAddSingle = this.singleChange(\n    this.edgeChangesOfType('add')\n  )\n\n  @Output('onEdgesChange.add.many')\n  public onEdgeChangeAddMany = this.manyChanges(\n    this.edgeChangesOfType('add')\n  )\n\n  @Output('onEdgesChange.remove')\n  public onEdgeChangeRemove = this.edgeChangesOfType('remove')\n\n  @Output('onEdgesChange.remove.single')\n  public onEdgeChangeRemoveSingle = this.singleChange(\n    this.edgeChangesOfType('remove')\n  )\n\n  @Output('onEdgesChange.remove.many')\n  public onEdgeChangeRemoveMany = this.manyChanges(\n    this.edgeChangesOfType('remove')\n  )\n\n  @Output('onEdgesChange.select')\n  public onEdgeChangeSelect = this.edgeChangesOfType('select')\n\n  @Output('onEdgesChange.select.single')\n  public onEdgeChangeSelectSingle = this.singleChange(\n    this.edgeChangesOfType('select')\n  )\n\n  @Output('onEdgesChange.select.many')\n  public onEdgeChangeSelectMany = this.manyChanges(\n    this.edgeChangesOfType('select')\n  )\n\n  private nodeChangesOfType<T extends NodeChange['type']>(type: T) {\n    return this.nodesChangeService.changes$.pipe(\n      map(changes => changes.filter((c): c is NodeChangeMap[T] => c.type === type)),\n      filter(changes => !!changes.length)\n    )\n  }\n\n  private edgeChangesOfType<T extends EdgeChange['type']>(type: T) {\n    return this.edgesChangeService.changes$.pipe(\n      map(changes => changes.filter((c): c is EdgeChangeMap[T] => c.type === type)),\n      filter(changes => !!changes.length)\n    )\n  }\n\n  private singleChange<T>(changes$: Observable<T[]>) {\n    return changes$.pipe(\n      filter(changes => changes.length === 1),\n      map(([first]) => first)\n    )\n  }\n\n  private manyChanges<T>(changes$: Observable<T[]>) {\n    return changes$.pipe(\n      filter(changes => changes.length > 1),\n    )\n  }\n}\n\n// TODO: do not write this types manually\ntype NodeChangeMap = {\n  position: NodePositionChange,\n  add: NodeAddChange,\n  remove: NodeRemoveChange,\n  select: NodeSelectedChange\n}\n\ntype EdgeChangeMap = {\n  detached: EdgeDetachedChange,\n  add: EdgeAddChange,\n  remove: EdgeRemoveChange,\n  select: EdgeSelectChange\n}\n"]}