ngx-vflow 0.1.7 → 0.1.9
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +68 -2
- package/esm2022/lib/vflow/directives/changes-controller.directive.mjs +7 -1
- package/esm2022/lib/vflow/directives/connection-controller.directive.mjs +9 -1
- package/esm2022/lib/vflow/interfaces/node.interface.mjs +1 -1
- package/fesm2022/ngx-vflow.mjs +81 -1
- package/fesm2022/ngx-vflow.mjs.map +1 -1
- package/lib/vflow/components/vflow/vflow.component.d.ts +75 -13
- package/lib/vflow/directives/changes-controller.directive.d.ts +6 -0
- package/lib/vflow/directives/connection-controller.directive.d.ts +8 -0
- package/lib/vflow/interfaces/node.interface.d.ts +3 -4
- package/package.json +1 -1
|
@@ -43,19 +43,46 @@ export class VflowComponent {
|
|
|
43
43
|
this.nodesChangeService = inject(NodesChangeService);
|
|
44
44
|
this.edgesChangeService = inject(EdgeChangesService);
|
|
45
45
|
this.injector = inject(Injector);
|
|
46
|
+
/**
|
|
47
|
+
* Minimum zoom value
|
|
48
|
+
*/
|
|
46
49
|
this.minZoom = 0.5;
|
|
50
|
+
/**
|
|
51
|
+
* Maximum zoom value
|
|
52
|
+
*/
|
|
47
53
|
this.maxZoom = 3;
|
|
54
|
+
/**
|
|
55
|
+
* Background color for flow
|
|
56
|
+
*/
|
|
48
57
|
this.background = '#FFFFFF';
|
|
49
58
|
// #endregion
|
|
50
59
|
// #region SIGNAL_API
|
|
60
|
+
/**
|
|
61
|
+
* Signal for reading viewport change
|
|
62
|
+
*/
|
|
51
63
|
this.viewport = this.viewportService.readableViewport.asReadonly();
|
|
64
|
+
/**
|
|
65
|
+
* Signal for reading nodes change
|
|
66
|
+
*/
|
|
52
67
|
this.nodesChange = toSignal(this.nodesChangeService.changes$, { initialValue: [] });
|
|
68
|
+
/**
|
|
69
|
+
* Signal to reading edges change
|
|
70
|
+
*/
|
|
53
71
|
this.edgesChange = toSignal(this.edgesChangeService.changes$, { initialValue: [] });
|
|
54
72
|
// #endregion
|
|
55
73
|
// #region RX_API
|
|
56
|
-
|
|
74
|
+
/**
|
|
75
|
+
* Observable with viewport change
|
|
76
|
+
*/
|
|
77
|
+
this.viewportChange$ = toObservable(this.viewportService.readableViewport)
|
|
57
78
|
.pipe(skip(1)); // skip default value that set by signal
|
|
79
|
+
/**
|
|
80
|
+
* Observable with nodes change
|
|
81
|
+
*/
|
|
58
82
|
this.nodesChange$ = this.nodesChangeService.changes$;
|
|
83
|
+
/**
|
|
84
|
+
* Observable with nodes change
|
|
85
|
+
*/
|
|
59
86
|
this.edgesChange$ = this.edgesChangeService.changes$;
|
|
60
87
|
// #endregion
|
|
61
88
|
// TODO: probably better to make it injectable
|
|
@@ -74,13 +101,29 @@ export class VflowComponent {
|
|
|
74
101
|
set view(view) {
|
|
75
102
|
this.flowModel.view.set(view);
|
|
76
103
|
}
|
|
104
|
+
/**
|
|
105
|
+
* Object that controls flow direction.
|
|
106
|
+
*
|
|
107
|
+
* For example, if you want to archieve right to left direction
|
|
108
|
+
* then you need to pass these positions { source: 'left', target: 'right' }
|
|
109
|
+
*
|
|
110
|
+
* ! Be carefult using this field, it may depricate in future releases !
|
|
111
|
+
*/
|
|
77
112
|
set handlePositions(handlePositions) {
|
|
78
113
|
this.flowModel.handlePositions.set(handlePositions);
|
|
79
114
|
}
|
|
115
|
+
/**
|
|
116
|
+
* Settings for connection (it renders when user tries to create edge between nodes)
|
|
117
|
+
*
|
|
118
|
+
* You need to pass `ConnectionSettings` in this input.
|
|
119
|
+
*/
|
|
80
120
|
set connection(connection) { this.flowEntitiesService.connection.set(connection); }
|
|
81
121
|
get connection() { return this.flowEntitiesService.connection(); }
|
|
82
122
|
// #endregion
|
|
83
123
|
// #region MAIN_INPUTS
|
|
124
|
+
/**
|
|
125
|
+
* Nodes to render
|
|
126
|
+
*/
|
|
84
127
|
set nodes(newNodes) {
|
|
85
128
|
const newModels = runInInjectionContext(this.injector, () => ReferenceKeeper.nodes(newNodes, this.flowEntitiesService.nodes()));
|
|
86
129
|
// TODO better to solve this by DI
|
|
@@ -90,6 +133,9 @@ export class VflowComponent {
|
|
|
90
133
|
this.flowEntitiesService.nodes.set(newModels);
|
|
91
134
|
}
|
|
92
135
|
get nodeModels() { return this.flowEntitiesService.nodes(); }
|
|
136
|
+
/**
|
|
137
|
+
* Edges to render
|
|
138
|
+
*/
|
|
93
139
|
set edges(newEdges) {
|
|
94
140
|
const newModels = ReferenceKeeper.edges(newEdges, this.flowEntitiesService.edges());
|
|
95
141
|
// quick and dirty binding nodes to edges
|
|
@@ -98,15 +144,35 @@ export class VflowComponent {
|
|
|
98
144
|
}
|
|
99
145
|
get edgeModels() { return this.flowEntitiesService.validEdges(); }
|
|
100
146
|
// #region METHODS_API
|
|
147
|
+
/**
|
|
148
|
+
* Change viewport to specified state
|
|
149
|
+
*
|
|
150
|
+
* @param viewport viewport state
|
|
151
|
+
*/
|
|
101
152
|
viewportTo(viewport) {
|
|
102
153
|
this.viewportService.writableViewport.set({ changeType: 'absolute', state: viewport });
|
|
103
154
|
}
|
|
155
|
+
/**
|
|
156
|
+
* Change zoom
|
|
157
|
+
*
|
|
158
|
+
* @param zoom zoom value
|
|
159
|
+
*/
|
|
104
160
|
zoomTo(zoom) {
|
|
105
161
|
this.viewportService.writableViewport.set({ changeType: 'absolute', state: { zoom } });
|
|
106
162
|
}
|
|
163
|
+
/**
|
|
164
|
+
* Move to specified coordinate
|
|
165
|
+
*
|
|
166
|
+
* @param point point where to move
|
|
167
|
+
*/
|
|
107
168
|
panTo(point) {
|
|
108
169
|
this.viewportService.writableViewport.set({ changeType: 'absolute', state: point });
|
|
109
170
|
}
|
|
171
|
+
/**
|
|
172
|
+
* Get node by id
|
|
173
|
+
*
|
|
174
|
+
* @param id node id
|
|
175
|
+
*/
|
|
110
176
|
getNode(id) {
|
|
111
177
|
return this.flowEntitiesService.getNode(id)?.node;
|
|
112
178
|
}
|
|
@@ -180,4 +246,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
180
246
|
function bindFlowToNodes(flow, nodes) {
|
|
181
247
|
nodes.forEach(n => n.bindFlow(flow));
|
|
182
248
|
}
|
|
183
|
-
//# 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  protected viewportService = inject(ViewportService)\n  protected flowEntitiesService = inject(FlowEntitiesService)\n  protected nodesChangeService = inject(NodesChangeService)\n  protected edgesChangeService = inject(EdgeChangesService)\n  protected 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  @Input()\n  public minZoom = 0.5\n\n  @Input()\n  public maxZoom = 3\n\n  @Input()\n  public set handlePositions(handlePositions: HandlePositions) {\n    this.flowModel.handlePositions.set(handlePositions)\n  }\n\n  @Input()\n  public background: string = '#FFFFFF'\n\n  @Input({ transform: (settings: ConnectionSettings) => new ConnectionModel(settings) })\n  public set connection(connection: ConnectionModel) { this.flowEntitiesService.connection.set(connection) }\n  public get connection() { return this.flowEntitiesService.connection() }\n  // #endregion\n\n  // #region MAIN_INPUTS\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  public get nodeModels() { return this.flowEntitiesService.nodes() }\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  public 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  public readonly viewport = this.viewportService.readableViewport.asReadonly()\n\n  public readonly nodesChange =\n    toSignal(this.nodesChangeService.changes$, { initialValue: [] as NodeChange[] })\n\n  public readonly edgesChange =\n    toSignal(this.edgesChangeService.changes$, { initialValue: [] as EdgeChange[] })\n  // #endregion\n\n  // #region RX_API\n  public readonly viewportChanges$ = toObservable(this.viewportService.readableViewport)\n    .pipe(skip(1)) // skip default value that set by signal\n\n  public readonly nodesChange$ = this.nodesChangeService.changes$\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  public viewportTo(viewport: ViewportState): void {\n    this.viewportService.writableViewport.set({ changeType: 'absolute', state: viewport })\n  }\n\n  public zoomTo(zoom: number): void {\n    this.viewportService.writableViewport.set({ changeType: 'absolute', state: { zoom } })\n  }\n\n  public panTo(point: Point): void {\n    this.viewportService.writableViewport.set({ changeType: 'absolute', state: point })\n  }\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"]}
|
|
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"]}
|
|
@@ -8,7 +8,13 @@ export class ChangesControllerDirective {
|
|
|
8
8
|
constructor() {
|
|
9
9
|
this.nodesChangeService = inject(NodesChangeService);
|
|
10
10
|
this.edgesChangeService = inject(EdgeChangesService);
|
|
11
|
+
/**
|
|
12
|
+
* Watch nodes change
|
|
13
|
+
*/
|
|
11
14
|
this.onNodesChange = new EventEmitter();
|
|
15
|
+
/**
|
|
16
|
+
* Watch nodes change
|
|
17
|
+
*/
|
|
12
18
|
this.onEdgesChange = new EventEmitter();
|
|
13
19
|
this.nodesChangeProxySubscription = this.nodesChangeService.changes$
|
|
14
20
|
.pipe(tap((changes) => this.onNodesChange.emit(changes)), takeUntilDestroyed())
|
|
@@ -31,4 +37,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
31
37
|
}], onEdgesChange: [{
|
|
32
38
|
type: Output
|
|
33
39
|
}] } });
|
|
34
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
40
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhbmdlcy1jb250cm9sbGVyLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9kaXJlY3RpdmVzL2NoYW5nZXMtY29udHJvbGxlci5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxZQUFZLEVBQUUsTUFBTSxFQUFVLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUVoRixPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUN0RSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUN0RSxPQUFPLEVBQUUsR0FBRyxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBQzNCLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLDRCQUE0QixDQUFDOztBQU9oRSxNQUFNLE9BQU8sMEJBQTBCO0lBSnZDO1FBS1ksdUJBQWtCLEdBQUcsTUFBTSxDQUFDLGtCQUFrQixDQUFDLENBQUE7UUFDL0MsdUJBQWtCLEdBQUcsTUFBTSxDQUFDLGtCQUFrQixDQUFDLENBQUE7UUFFekQ7O1dBRUc7UUFFSSxrQkFBYSxHQUFHLElBQUksWUFBWSxFQUFnQixDQUFBO1FBRXZEOztXQUVHO1FBRUksa0JBQWEsR0FBRyxJQUFJLFlBQVksRUFBZ0IsQ0FBQTtRQUU3QyxpQ0FBNEIsR0FBRyxJQUFJLENBQUMsa0JBQWtCLENBQUMsUUFBUTthQUN0RSxJQUFJLENBQ0gsR0FBRyxDQUFDLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQyxFQUNsRCxrQkFBa0IsRUFBRSxDQUNyQjthQUNBLFNBQVMsRUFBRSxDQUFBO1FBRUosaUNBQTRCLEdBQUcsSUFBSSxDQUFDLGtCQUFrQixDQUFDLFFBQVE7YUFDdEUsSUFBSSxDQUNILEdBQUcsQ0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUMsRUFDbEQsa0JBQWtCLEVBQUUsQ0FDckI7YUFDQSxTQUFTLEVBQUUsQ0FBQTtLQUNmOytHQTdCWSwwQkFBMEI7bUdBQTFCLDBCQUEwQjs7NEZBQTFCLDBCQUEwQjtrQkFKdEMsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUscUJBQXFCO29CQUMvQixVQUFVLEVBQUUsSUFBSTtpQkFDakI7OEJBU1EsYUFBYTtzQkFEbkIsTUFBTTtnQkFPQSxhQUFhO3NCQURuQixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGlyZWN0aXZlLCBFdmVudEVtaXR0ZXIsIE91dHB1dCwgZWZmZWN0LCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IE5vZGVDaGFuZ2UgfSBmcm9tICcuLi90eXBlcy9ub2RlLWNoYW5nZS50eXBlJztcbmltcG9ydCB7IEVkZ2VDaGFuZ2VzU2VydmljZSB9IGZyb20gJy4uL3NlcnZpY2VzL2VkZ2UtY2hhbmdlcy5zZXJ2aWNlJztcbmltcG9ydCB7IE5vZGVzQ2hhbmdlU2VydmljZSB9IGZyb20gJy4uL3NlcnZpY2VzL25vZGUtY2hhbmdlcy5zZXJ2aWNlJztcbmltcG9ydCB7IHRhcCB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgdGFrZVVudGlsRGVzdHJveWVkIH0gZnJvbSAnQGFuZ3VsYXIvY29yZS9yeGpzLWludGVyb3AnO1xuaW1wb3J0IHsgRWRnZUNoYW5nZSB9IGZyb20gJy4uL3R5cGVzL2VkZ2UtY2hhbmdlLnR5cGUnO1xuXG5ARGlyZWN0aXZlKHtcbiAgc2VsZWN0b3I6ICdbY2hhbmdlc0NvbnRyb2xsZXJdJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZVxufSlcbmV4cG9ydCBjbGFzcyBDaGFuZ2VzQ29udHJvbGxlckRpcmVjdGl2ZSB7XG4gIHByb3RlY3RlZCBub2Rlc0NoYW5nZVNlcnZpY2UgPSBpbmplY3QoTm9kZXNDaGFuZ2VTZXJ2aWNlKVxuICBwcm90ZWN0ZWQgZWRnZXNDaGFuZ2VTZXJ2aWNlID0gaW5qZWN0KEVkZ2VDaGFuZ2VzU2VydmljZSlcblxuICAvKipcbiAgICogV2F0Y2ggbm9kZXMgY2hhbmdlXG4gICAqL1xuICBAT3V0cHV0KClcbiAgcHVibGljIG9uTm9kZXNDaGFuZ2UgPSBuZXcgRXZlbnRFbWl0dGVyPE5vZGVDaGFuZ2VbXT4oKVxuXG4gIC8qKlxuICAgKiBXYXRjaCBub2RlcyBjaGFuZ2VcbiAgICovXG4gIEBPdXRwdXQoKVxuICBwdWJsaWMgb25FZGdlc0NoYW5nZSA9IG5ldyBFdmVudEVtaXR0ZXI8RWRnZUNoYW5nZVtdPigpXG5cbiAgcHJvdGVjdGVkIG5vZGVzQ2hhbmdlUHJveHlTdWJzY3JpcHRpb24gPSB0aGlzLm5vZGVzQ2hhbmdlU2VydmljZS5jaGFuZ2VzJFxuICAgIC5waXBlKFxuICAgICAgdGFwKChjaGFuZ2VzKSA9PiB0aGlzLm9uTm9kZXNDaGFuZ2UuZW1pdChjaGFuZ2VzKSksXG4gICAgICB0YWtlVW50aWxEZXN0cm95ZWQoKVxuICAgIClcbiAgICAuc3Vic2NyaWJlKClcblxuICBwcm90ZWN0ZWQgZWRnZXNDaGFuZ2VQcm94eVN1YnNjcmlwdGlvbiA9IHRoaXMuZWRnZXNDaGFuZ2VTZXJ2aWNlLmNoYW5nZXMkXG4gICAgLnBpcGUoXG4gICAgICB0YXAoKGNoYW5nZXMpID0+IHRoaXMub25FZGdlc0NoYW5nZS5lbWl0KGNoYW5nZXMpKSxcbiAgICAgIHRha2VVbnRpbERlc3Ryb3llZCgpXG4gICAgKVxuICAgIC5zdWJzY3JpYmUoKVxufVxuIl19
|
|
@@ -4,6 +4,14 @@ import { FlowEntitiesService } from '../services/flow-entities.service';
|
|
|
4
4
|
import * as i0 from "@angular/core";
|
|
5
5
|
export class ConnectionControllerDirective {
|
|
6
6
|
constructor() {
|
|
7
|
+
/**
|
|
8
|
+
* This event fires when user tries to create new Edge.
|
|
9
|
+
*
|
|
10
|
+
* `Connection` is an entity that contains data about source and target nodes.
|
|
11
|
+
*
|
|
12
|
+
* Also it's important to note, that this event only fires when connection is valid by validator function in `ConnectionSettings`,
|
|
13
|
+
* by default without passing the validator every connection concidered valid.
|
|
14
|
+
*/
|
|
7
15
|
this.onConnect = new EventEmitter();
|
|
8
16
|
this.statusService = inject(FlowStatusService);
|
|
9
17
|
this.flowEntitiesService = inject(FlowEntitiesService);
|
|
@@ -33,4 +41,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
33
41
|
}], propDecorators: { onConnect: [{
|
|
34
42
|
type: Output
|
|
35
43
|
}] } });
|
|
36
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
44
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29ubmVjdGlvbi1jb250cm9sbGVyLmRpcmVjdGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9kaXJlY3RpdmVzL2Nvbm5lY3Rpb24tY29udHJvbGxlci5kaXJlY3RpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxZQUFZLEVBQUUsTUFBTSxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFaEYsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0saUNBQWlDLENBQUM7QUFFcEUsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sbUNBQW1DLENBQUM7O0FBTXhFLE1BQU0sT0FBTyw2QkFBNkI7SUFKMUM7UUFLRTs7Ozs7OztXQU9HO1FBRUksY0FBUyxHQUFHLElBQUksWUFBWSxFQUFjLENBQUE7UUFFekMsa0JBQWEsR0FBRyxNQUFNLENBQUMsaUJBQWlCLENBQUMsQ0FBQTtRQUN6Qyx3QkFBbUIsR0FBRyxNQUFNLENBQUMsbUJBQW1CLENBQUMsQ0FBQTtRQUUvQyxrQkFBYSxHQUFHLE1BQU0sQ0FBQyxHQUFHLEVBQUU7WUFDcEMsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxNQUFNLEVBQUUsQ0FBQTtZQUUxQyxJQUFJLE1BQU0sQ0FBQyxLQUFLLEtBQUssZ0JBQWdCLEVBQUU7Z0JBQ3JDLE1BQU0sV0FBVyxHQUFHLE1BQU0sQ0FBQyxPQUFPLENBQUMsVUFBVSxDQUFBO2dCQUM3QyxNQUFNLFdBQVcsR0FBRyxNQUFNLENBQUMsT0FBTyxDQUFDLFVBQVUsQ0FBQTtnQkFFN0MsTUFBTSxNQUFNLEdBQUcsV0FBVyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUE7Z0JBQ2xDLE1BQU0sTUFBTSxHQUFHLFdBQVcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFBO2dCQUVsQyxNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsbUJBQW1CLENBQUMsVUFBVSxFQUFFLENBQUE7Z0JBRXhELElBQUksVUFBVSxDQUFDLFNBQVMsQ0FBQyxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUsQ0FBQyxFQUFFO29CQUM1QyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUsQ0FBQyxDQUFBO2lCQUN4QzthQUNGO1FBQ0gsQ0FBQyxFQUFFLEVBQUUsaUJBQWlCLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQTtLQUNoQzsrR0FoQ1ksNkJBQTZCO21HQUE3Qiw2QkFBNkI7OzRGQUE3Qiw2QkFBNkI7a0JBSnpDLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLHdCQUF3QjtvQkFDbEMsVUFBVSxFQUFFLElBQUk7aUJBQ2pCOzhCQVdRLFNBQVM7c0JBRGYsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgRXZlbnRFbWl0dGVyLCBPdXRwdXQsIGVmZmVjdCwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb25uZWN0aW9uIH0gZnJvbSAnLi4vaW50ZXJmYWNlcy9jb25uZWN0aW9uLmludGVyZmFjZSc7XG5pbXBvcnQgeyBGbG93U3RhdHVzU2VydmljZSB9IGZyb20gJy4uL3NlcnZpY2VzL2Zsb3ctc3RhdHVzLnNlcnZpY2UnO1xuXG5pbXBvcnQgeyBGbG93RW50aXRpZXNTZXJ2aWNlIH0gZnJvbSAnLi4vc2VydmljZXMvZmxvdy1lbnRpdGllcy5zZXJ2aWNlJztcblxuQERpcmVjdGl2ZSh7XG4gIHNlbGVjdG9yOiAnW2Nvbm5lY3Rpb25Db250cm9sbGVyXScsXG4gIHN0YW5kYWxvbmU6IHRydWVcbn0pXG5leHBvcnQgY2xhc3MgQ29ubmVjdGlvbkNvbnRyb2xsZXJEaXJlY3RpdmUge1xuICAvKipcbiAgICogVGhpcyBldmVudCBmaXJlcyB3aGVuIHVzZXIgdHJpZXMgdG8gY3JlYXRlIG5ldyBFZGdlLlxuICAgKlxuICAgKiBgQ29ubmVjdGlvbmAgaXMgYW4gZW50aXR5IHRoYXQgY29udGFpbnMgZGF0YSBhYm91dCBzb3VyY2UgYW5kIHRhcmdldCBub2Rlcy5cbiAgICpcbiAgICogQWxzbyBpdCdzIGltcG9ydGFudCB0byBub3RlLCB0aGF0IHRoaXMgZXZlbnQgb25seSBmaXJlcyB3aGVuIGNvbm5lY3Rpb24gaXMgdmFsaWQgYnkgdmFsaWRhdG9yIGZ1bmN0aW9uIGluIGBDb25uZWN0aW9uU2V0dGluZ3NgLFxuICAgKiBieSBkZWZhdWx0IHdpdGhvdXQgcGFzc2luZyB0aGUgdmFsaWRhdG9yIGV2ZXJ5IGNvbm5lY3Rpb24gY29uY2lkZXJlZCB2YWxpZC5cbiAgICovXG4gIEBPdXRwdXQoKVxuICBwdWJsaWMgb25Db25uZWN0ID0gbmV3IEV2ZW50RW1pdHRlcjxDb25uZWN0aW9uPigpXG5cbiAgcHJpdmF0ZSBzdGF0dXNTZXJ2aWNlID0gaW5qZWN0KEZsb3dTdGF0dXNTZXJ2aWNlKVxuICBwcml2YXRlIGZsb3dFbnRpdGllc1NlcnZpY2UgPSBpbmplY3QoRmxvd0VudGl0aWVzU2VydmljZSlcblxuICBwcm90ZWN0ZWQgY29ubmVjdEVmZmVjdCA9IGVmZmVjdCgoKSA9PiB7XG4gICAgY29uc3Qgc3RhdHVzID0gdGhpcy5zdGF0dXNTZXJ2aWNlLnN0YXR1cygpXG5cbiAgICBpZiAoc3RhdHVzLnN0YXRlID09PSAnY29ubmVjdGlvbi1lbmQnKSB7XG4gICAgICBjb25zdCBzb3VyY2VNb2RlbCA9IHN0YXR1cy5wYXlsb2FkLnNvdXJjZU5vZGVcbiAgICAgIGNvbnN0IHRhcmdldE1vZGVsID0gc3RhdHVzLnBheWxvYWQudGFyZ2V0Tm9kZVxuXG4gICAgICBjb25zdCBzb3VyY2UgPSBzb3VyY2VNb2RlbC5ub2RlLmlkXG4gICAgICBjb25zdCB0YXJnZXQgPSB0YXJnZXRNb2RlbC5ub2RlLmlkXG5cbiAgICAgIGNvbnN0IGNvbm5lY3Rpb24gPSB0aGlzLmZsb3dFbnRpdGllc1NlcnZpY2UuY29ubmVjdGlvbigpXG5cbiAgICAgIGlmIChjb25uZWN0aW9uLnZhbGlkYXRvcih7IHNvdXJjZSwgdGFyZ2V0IH0pKSB7XG4gICAgICAgIHRoaXMub25Db25uZWN0LmVtaXQoeyBzb3VyY2UsIHRhcmdldCB9KVxuICAgICAgfVxuICAgIH1cbiAgfSwgeyBhbGxvd1NpZ25hbFdyaXRlczogdHJ1ZSB9KVxufVxuIl19
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibm9kZS5pbnRlcmZhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZ3gtdmZsb3ctbGliL3NyYy9saWIvdmZsb3cvaW50ZXJmYWNlcy9ub2RlLmludGVyZmFjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUG9pbnQgfSBmcm9tIFwiLi9wb2ludC5pbnRlcmZhY2VcIlxuXG5leHBvcnQgdHlwZSBOb2RlPFQgPSB1bmtub3duPiA9IFNoYXJlZE5vZGUgJiAoXG4gIERlZmF1bHROb2RlIHxcbiAgSHRtbFRlbXBsYXRlTm9kZTxUPlxuKVxuZXhwb3J0IGludGVyZmFjZSBTaGFyZWROb2RlIHtcbiAgaWQ6IHN0cmluZ1xuICBwb2ludDogUG9pbnRcbiAgZHJhZ2dhYmxlPzogYm9vbGVhblxufVxuXG5leHBvcnQgaW50ZXJmYWNlIERlZmF1bHROb2RlIHtcbiAgdHlwZTogJ2RlZmF1bHQnXG4gIHRleHQ/OiBzdHJpbmdcbn1cblxuZXhwb3J0IGludGVyZmFjZSBIdG1sVGVtcGxhdGVOb2RlPFQgPSB1bmtub3duPiB7XG4gIHR5cGU6ICdodG1sLXRlbXBsYXRlJ1xuICBkYXRhPzogVFxufVxuIl19
|
package/fesm2022/ngx-vflow.mjs
CHANGED
|
@@ -357,6 +357,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
357
357
|
|
|
358
358
|
class ConnectionControllerDirective {
|
|
359
359
|
constructor() {
|
|
360
|
+
/**
|
|
361
|
+
* This event fires when user tries to create new Edge.
|
|
362
|
+
*
|
|
363
|
+
* `Connection` is an entity that contains data about source and target nodes.
|
|
364
|
+
*
|
|
365
|
+
* Also it's important to note, that this event only fires when connection is valid by validator function in `ConnectionSettings`,
|
|
366
|
+
* by default without passing the validator every connection concidered valid.
|
|
367
|
+
*/
|
|
360
368
|
this.onConnect = new EventEmitter();
|
|
361
369
|
this.statusService = inject(FlowStatusService);
|
|
362
370
|
this.flowEntitiesService = inject(FlowEntitiesService);
|
|
@@ -703,7 +711,13 @@ class ChangesControllerDirective {
|
|
|
703
711
|
constructor() {
|
|
704
712
|
this.nodesChangeService = inject(NodesChangeService);
|
|
705
713
|
this.edgesChangeService = inject(EdgeChangesService);
|
|
714
|
+
/**
|
|
715
|
+
* Watch nodes change
|
|
716
|
+
*/
|
|
706
717
|
this.onNodesChange = new EventEmitter();
|
|
718
|
+
/**
|
|
719
|
+
* Watch nodes change
|
|
720
|
+
*/
|
|
707
721
|
this.onEdgesChange = new EventEmitter();
|
|
708
722
|
this.nodesChangeProxySubscription = this.nodesChangeService.changes$
|
|
709
723
|
.pipe(tap((changes) => this.onNodesChange.emit(changes)), takeUntilDestroyed())
|
|
@@ -1130,19 +1144,46 @@ class VflowComponent {
|
|
|
1130
1144
|
this.nodesChangeService = inject(NodesChangeService);
|
|
1131
1145
|
this.edgesChangeService = inject(EdgeChangesService);
|
|
1132
1146
|
this.injector = inject(Injector);
|
|
1147
|
+
/**
|
|
1148
|
+
* Minimum zoom value
|
|
1149
|
+
*/
|
|
1133
1150
|
this.minZoom = 0.5;
|
|
1151
|
+
/**
|
|
1152
|
+
* Maximum zoom value
|
|
1153
|
+
*/
|
|
1134
1154
|
this.maxZoom = 3;
|
|
1155
|
+
/**
|
|
1156
|
+
* Background color for flow
|
|
1157
|
+
*/
|
|
1135
1158
|
this.background = '#FFFFFF';
|
|
1136
1159
|
// #endregion
|
|
1137
1160
|
// #region SIGNAL_API
|
|
1161
|
+
/**
|
|
1162
|
+
* Signal for reading viewport change
|
|
1163
|
+
*/
|
|
1138
1164
|
this.viewport = this.viewportService.readableViewport.asReadonly();
|
|
1165
|
+
/**
|
|
1166
|
+
* Signal for reading nodes change
|
|
1167
|
+
*/
|
|
1139
1168
|
this.nodesChange = toSignal(this.nodesChangeService.changes$, { initialValue: [] });
|
|
1169
|
+
/**
|
|
1170
|
+
* Signal to reading edges change
|
|
1171
|
+
*/
|
|
1140
1172
|
this.edgesChange = toSignal(this.edgesChangeService.changes$, { initialValue: [] });
|
|
1141
1173
|
// #endregion
|
|
1142
1174
|
// #region RX_API
|
|
1143
|
-
|
|
1175
|
+
/**
|
|
1176
|
+
* Observable with viewport change
|
|
1177
|
+
*/
|
|
1178
|
+
this.viewportChange$ = toObservable(this.viewportService.readableViewport)
|
|
1144
1179
|
.pipe(skip(1)); // skip default value that set by signal
|
|
1180
|
+
/**
|
|
1181
|
+
* Observable with nodes change
|
|
1182
|
+
*/
|
|
1145
1183
|
this.nodesChange$ = this.nodesChangeService.changes$;
|
|
1184
|
+
/**
|
|
1185
|
+
* Observable with nodes change
|
|
1186
|
+
*/
|
|
1146
1187
|
this.edgesChange$ = this.edgesChangeService.changes$;
|
|
1147
1188
|
// #endregion
|
|
1148
1189
|
// TODO: probably better to make it injectable
|
|
@@ -1161,13 +1202,29 @@ class VflowComponent {
|
|
|
1161
1202
|
set view(view) {
|
|
1162
1203
|
this.flowModel.view.set(view);
|
|
1163
1204
|
}
|
|
1205
|
+
/**
|
|
1206
|
+
* Object that controls flow direction.
|
|
1207
|
+
*
|
|
1208
|
+
* For example, if you want to archieve right to left direction
|
|
1209
|
+
* then you need to pass these positions { source: 'left', target: 'right' }
|
|
1210
|
+
*
|
|
1211
|
+
* ! Be carefult using this field, it may depricate in future releases !
|
|
1212
|
+
*/
|
|
1164
1213
|
set handlePositions(handlePositions) {
|
|
1165
1214
|
this.flowModel.handlePositions.set(handlePositions);
|
|
1166
1215
|
}
|
|
1216
|
+
/**
|
|
1217
|
+
* Settings for connection (it renders when user tries to create edge between nodes)
|
|
1218
|
+
*
|
|
1219
|
+
* You need to pass `ConnectionSettings` in this input.
|
|
1220
|
+
*/
|
|
1167
1221
|
set connection(connection) { this.flowEntitiesService.connection.set(connection); }
|
|
1168
1222
|
get connection() { return this.flowEntitiesService.connection(); }
|
|
1169
1223
|
// #endregion
|
|
1170
1224
|
// #region MAIN_INPUTS
|
|
1225
|
+
/**
|
|
1226
|
+
* Nodes to render
|
|
1227
|
+
*/
|
|
1171
1228
|
set nodes(newNodes) {
|
|
1172
1229
|
const newModels = runInInjectionContext(this.injector, () => ReferenceKeeper.nodes(newNodes, this.flowEntitiesService.nodes()));
|
|
1173
1230
|
// TODO better to solve this by DI
|
|
@@ -1177,6 +1234,9 @@ class VflowComponent {
|
|
|
1177
1234
|
this.flowEntitiesService.nodes.set(newModels);
|
|
1178
1235
|
}
|
|
1179
1236
|
get nodeModels() { return this.flowEntitiesService.nodes(); }
|
|
1237
|
+
/**
|
|
1238
|
+
* Edges to render
|
|
1239
|
+
*/
|
|
1180
1240
|
set edges(newEdges) {
|
|
1181
1241
|
const newModels = ReferenceKeeper.edges(newEdges, this.flowEntitiesService.edges());
|
|
1182
1242
|
// quick and dirty binding nodes to edges
|
|
@@ -1185,15 +1245,35 @@ class VflowComponent {
|
|
|
1185
1245
|
}
|
|
1186
1246
|
get edgeModels() { return this.flowEntitiesService.validEdges(); }
|
|
1187
1247
|
// #region METHODS_API
|
|
1248
|
+
/**
|
|
1249
|
+
* Change viewport to specified state
|
|
1250
|
+
*
|
|
1251
|
+
* @param viewport viewport state
|
|
1252
|
+
*/
|
|
1188
1253
|
viewportTo(viewport) {
|
|
1189
1254
|
this.viewportService.writableViewport.set({ changeType: 'absolute', state: viewport });
|
|
1190
1255
|
}
|
|
1256
|
+
/**
|
|
1257
|
+
* Change zoom
|
|
1258
|
+
*
|
|
1259
|
+
* @param zoom zoom value
|
|
1260
|
+
*/
|
|
1191
1261
|
zoomTo(zoom) {
|
|
1192
1262
|
this.viewportService.writableViewport.set({ changeType: 'absolute', state: { zoom } });
|
|
1193
1263
|
}
|
|
1264
|
+
/**
|
|
1265
|
+
* Move to specified coordinate
|
|
1266
|
+
*
|
|
1267
|
+
* @param point point where to move
|
|
1268
|
+
*/
|
|
1194
1269
|
panTo(point) {
|
|
1195
1270
|
this.viewportService.writableViewport.set({ changeType: 'absolute', state: point });
|
|
1196
1271
|
}
|
|
1272
|
+
/**
|
|
1273
|
+
* Get node by id
|
|
1274
|
+
*
|
|
1275
|
+
* @param id node id
|
|
1276
|
+
*/
|
|
1197
1277
|
getNode(id) {
|
|
1198
1278
|
return this.flowEntitiesService.getNode(id)?.node;
|
|
1199
1279
|
}
|