ngx-vflow 0.13.0-0 → 0.13.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 (26) hide show
  1. package/esm2022/lib/vflow/components/node/node.component.mjs +3 -3
  2. package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +13 -4
  3. package/esm2022/lib/vflow/directives/root-pointer.directive.mjs +1 -1
  4. package/esm2022/lib/vflow/directives/selectable.directive.mjs +5 -2
  5. package/esm2022/lib/vflow/public-components/resizable/resizable.component.mjs +124 -180
  6. package/esm2022/lib/vflow/services/draggable.service.mjs +40 -19
  7. package/esm2022/lib/vflow/services/keyboard.service.mjs +45 -0
  8. package/esm2022/lib/vflow/services/node-changes.service.mjs +6 -7
  9. package/esm2022/lib/vflow/services/selection.service.mjs +12 -5
  10. package/esm2022/lib/vflow/types/keyboard-action.type.mjs +2 -0
  11. package/esm2022/lib/vflow/utils/get-os.mjs +24 -0
  12. package/esm2022/public-api.mjs +2 -1
  13. package/fesm2022/ngx-vflow.mjs +257 -214
  14. package/fesm2022/ngx-vflow.mjs.map +1 -1
  15. package/lib/vflow/components/vflow/vflow.component.d.ts +4 -1
  16. package/lib/vflow/directives/root-pointer.directive.d.ts +10 -16
  17. package/lib/vflow/directives/space-point-context.directive.d.ts +1 -15
  18. package/lib/vflow/models/edge.model.d.ts +17 -1
  19. package/lib/vflow/public-components/resizable/resizable.component.d.ts +6 -11
  20. package/lib/vflow/services/draggable.service.d.ts +2 -0
  21. package/lib/vflow/services/keyboard.service.d.ts +11 -0
  22. package/lib/vflow/services/selection.service.d.ts +1 -0
  23. package/lib/vflow/types/keyboard-action.type.d.ts +2 -0
  24. package/lib/vflow/utils/get-os.d.ts +1 -0
  25. package/package.json +1 -1
  26. package/public-api.d.ts +1 -0
@@ -8,6 +8,7 @@ import { round } from '../../utils/round';
8
8
  import { Microtask } from '../../decorators/microtask.decorator';
9
9
  import { getNodesBounds } from '../../utils/nodes';
10
10
  import { NodeAccessorService } from '../../services/node-accessor.service';
11
+ import { SpacePointContextDirective } from '../../directives/space-point-context.directive';
11
12
  import * as i0 from "@angular/core";
12
13
  import * as i1 from "../../directives/pointer.directive";
13
14
  export class ResizableComponent {
@@ -15,6 +16,7 @@ export class ResizableComponent {
15
16
  this.nodeAccessor = inject(NodeAccessorService);
16
17
  this.rootPointer = inject(RootPointerDirective);
17
18
  this.viewportService = inject(ViewportService);
19
+ this.spacePointContext = inject(SpacePointContextDirective);
18
20
  this.hostRef = inject(ElementRef);
19
21
  this.resizerColor = '#2e414c';
20
22
  this.gap = 1.5;
@@ -26,7 +28,7 @@ export class ResizableComponent {
26
28
  this.minHeight = 0;
27
29
  // TODO: allow reszie beside the flow
28
30
  this.resizeOnGlobalMouseMove = this.rootPointer.pointerMovement$
29
- .pipe(filter(() => this.resizeSide !== null), tap((event) => this.resize(event)), takeUntilDestroyed())
31
+ .pipe(filter(() => this.resizeSide !== null), filter((event) => event.movementX !== 0 || event.movementY !== 0), tap((event) => this.resize(event)), takeUntilDestroyed())
30
32
  .subscribe();
31
33
  this.endResizeOnGlobalMouseUp = this.rootPointer.documentPointerEnd$
32
34
  .pipe(tap(() => this.endResize()), takeUntilDestroyed())
@@ -55,193 +57,55 @@ export class ResizableComponent {
55
57
  this.resizeSide = side;
56
58
  this.model.resizing.set(true);
57
59
  }
58
- resize({ movementX, movementY }) {
59
- const offsetX = round(movementX / this.zoom());
60
- const offsetY = round(movementY / this.zoom());
61
- switch (this.resizeSide) {
62
- case 'left':
63
- let x = this.model.point().x + offsetX;
64
- x = Math.max(x, this.getMinX());
65
- x = Math.min(x, this.getMaxX());
66
- // TODO this fixes increasing width when current node hits the parent
67
- if (x === this.getMinX() || x === this.getMaxX()) {
68
- return;
69
- }
70
- this.model.setPoint({ x, y: this.model.point().y }, false);
71
- this.model.size.update(({ height, width }) => {
72
- width -= offsetX;
73
- width = Math.max(width, this.minWidth);
74
- width = Math.min(width, this.getMaxWidth());
75
- return { height, width: width };
76
- });
77
- return;
78
- case 'right':
79
- this.model.size.update(({ height, width }) => {
80
- width += offsetX;
81
- width = Math.max(width, this.minWidth);
82
- width = Math.min(width, this.getMaxWidth());
83
- const bounds = getNodesBounds(this.model.children());
84
- width = Math.max(width, bounds.x + bounds.width);
85
- return { height, width };
86
- });
87
- return;
88
- case 'top':
89
- let y = this.model.point().y + offsetY;
90
- y = Math.max(y, this.getMinY());
91
- y = Math.min(y, this.getMaxY());
92
- if (y === this.getMinY() || y === this.getMaxY()) {
93
- return;
94
- }
95
- this.model.setPoint({ x: this.model.point().x, y }, false);
96
- this.model.size.update(({ height, width }) => {
97
- height -= offsetY;
98
- height = Math.max(height, this.minHeight);
99
- height = Math.min(height, this.getMaxHeight());
100
- return { width, height };
101
- });
102
- return;
103
- case 'bottom':
104
- this.model.size.update(({ height, width }) => {
105
- height += offsetY;
106
- height = Math.max(height, this.minHeight);
107
- height = Math.min(height, this.getMaxHeight());
108
- const bounds = getNodesBounds(this.model.children());
109
- height = Math.max(height, bounds.y + bounds.height);
110
- return { width, height };
111
- });
112
- return;
113
- case 'top-left': {
114
- let x = this.model.point().x + offsetX;
115
- x = Math.max(x, this.getMinX());
116
- x = Math.min(x, this.getMaxX());
117
- let y = this.model.point().y + offsetY;
118
- y = Math.max(y, this.getMinY());
119
- y = Math.min(y, this.getMaxY());
120
- if (x === this.getMinX() || y === this.getMinY() ||
121
- x === this.getMaxX() || y === this.getMaxY()) {
122
- return;
123
- }
124
- this.model.setPoint({ x, y }, false);
125
- this.model.size.update(({ height, width }) => {
126
- width -= offsetX;
127
- width = Math.max(width, this.minWidth);
128
- width = Math.min(width, this.getMaxWidth());
129
- height -= offsetY;
130
- height = Math.max(height, this.minHeight);
131
- height = Math.min(height, this.getMaxHeight());
132
- return { height, width };
133
- });
134
- return;
135
- }
136
- case 'top-right': {
137
- let y = this.model.point().y + offsetY;
138
- y = Math.max(y, this.getMinY());
139
- y = Math.min(y, this.getMaxY());
140
- if (y === this.getMinX() || y === this.getMaxY()) {
141
- return;
142
- }
143
- this.model.setPoint({ x: this.model.point().x, y }, false);
144
- this.model.size.update(({ height, width }) => {
145
- const bounds = getNodesBounds(this.model.children());
146
- width += offsetX;
147
- width = Math.max(width, this.minWidth);
148
- width = Math.min(width, this.getMaxWidth());
149
- width = Math.max(width, bounds.x + bounds.width);
150
- height -= offsetY;
151
- height = Math.max(height, this.minHeight);
152
- height = Math.min(height, this.getMaxHeight());
153
- return { height, width };
154
- });
155
- return;
156
- }
157
- case 'bottom-left': {
158
- let x = this.model.point().x + offsetX;
159
- x = Math.max(x, this.getMinX());
160
- x = Math.min(x, this.getMaxX());
161
- if (x === this.getMinX() || x === this.getMaxX()) {
162
- return;
163
- }
164
- this.model.setPoint({ x, y: this.model.point().y }, false);
165
- this.model.size.update(({ height, width }) => {
166
- width -= offsetX;
167
- width = Math.max(width, this.minWidth);
168
- width = Math.min(width, this.getMaxWidth());
169
- height += offsetY;
170
- height = Math.max(height, this.minHeight);
171
- height = Math.min(height, this.getMaxHeight());
172
- const bounds = getNodesBounds(this.model.children());
173
- height = Math.max(height, bounds.y + bounds.height);
174
- return { height, width };
175
- });
176
- return;
177
- }
178
- case 'bottom-right': {
179
- this.model.size.update(({ height, width }) => {
180
- const bounds = getNodesBounds(this.model.children());
181
- width += offsetX;
182
- width = Math.max(width, this.minWidth);
183
- width = Math.min(width, this.getMaxWidth());
184
- width = Math.max(width, bounds.x + bounds.width);
185
- height += offsetY;
186
- height = Math.max(height, this.minHeight);
187
- height = Math.min(height, this.getMaxHeight());
188
- height = Math.max(height, bounds.y + bounds.height);
189
- return { height, width };
190
- });
191
- }
192
- }
60
+ resize(event) {
61
+ if (!this.resizeSide)
62
+ return;
63
+ if (this.isResizeConstrained(event))
64
+ return;
65
+ const offset = calcOffset(event.movementX, event.movementY, this.zoom());
66
+ const { x, y, width, height } = constrainRect(applyResize(this.resizeSide, this.model, offset), this.model, this.resizeSide, this.minWidth, this.minHeight);
67
+ this.model.setPoint({ x, y }, false);
68
+ this.model.size.set({ width, height });
193
69
  }
194
70
  endResize() {
195
71
  this.resizeSide = null;
196
72
  this.model.resizing.set(false);
197
73
  }
198
- getMaxWidth() {
199
- const parent = this.model.parent();
200
- if (parent) {
201
- return parent.size().width - this.model.point().x;
202
- }
203
- return Infinity;
204
- }
205
- getMaxHeight() {
206
- const parent = this.model.parent();
207
- if (parent) {
208
- return parent.size().height - this.model.point().y;
209
- }
210
- return Infinity;
211
- }
212
- getMinX() {
213
- const parent = this.model.parent();
214
- if (parent) {
215
- return 0;
74
+ isResizeConstrained({ x, y, movementX, movementY }) {
75
+ const flowPoint = this.spacePointContext.documentPointToFlowPoint({ x, y });
76
+ if (this.resizeSide?.includes('right')) {
77
+ if (movementX > 0 && flowPoint.x < (this.model.point().x + this.model.size().width)) {
78
+ return true;
79
+ }
80
+ if (movementX < 0 && flowPoint.x > this.model.point().x + this.model.size().width) {
81
+ return true;
82
+ }
216
83
  }
217
- return -Infinity;
218
- }
219
- getMinY() {
220
- const parent = this.model.parent();
221
- if (parent) {
222
- return 0;
84
+ if (this.resizeSide?.includes('left')) {
85
+ if (movementX < 0 && flowPoint.x > this.model.point().x) {
86
+ return true;
87
+ }
88
+ if (movementX > 0 && flowPoint.x < this.model.point().x) {
89
+ return true;
90
+ }
223
91
  }
224
- return -Infinity;
225
- }
226
- getMaxX() {
227
- const x = this.model.point().x;
228
- const width = this.model.size().width;
229
- const children = this.model.children();
230
- if (children) {
231
- const bounds = getNodesBounds(children);
232
- return x + (bounds.x + bounds.width) >= x + width ? x : (width - this.minWidth) + x;
92
+ if (this.resizeSide?.includes('bottom')) {
93
+ if (movementY > 0 && flowPoint.y < (this.model.point().y + this.model.size().height)) {
94
+ return true;
95
+ }
96
+ if (movementY < 0 && flowPoint.y > this.model.point().y + this.model.size().height) {
97
+ return true;
98
+ }
233
99
  }
234
- return (width - this.minWidth) + x;
235
- }
236
- getMaxY() {
237
- const y = this.model.point().y;
238
- const height = this.model.size().height;
239
- const children = this.model.children();
240
- if (children) {
241
- const bounds = getNodesBounds(children);
242
- return y + (bounds.y + bounds.height) >= y + height ? y : (height - this.minHeight) + y;
100
+ if (this.resizeSide?.includes('top')) {
101
+ if (movementY < 0 && flowPoint.y > this.model.point().y) {
102
+ return true;
103
+ }
104
+ if (movementY > 0 && flowPoint.y < this.model.point().y) {
105
+ return true;
106
+ }
243
107
  }
244
- return (height - this.minHeight) + y;
108
+ return false;
245
109
  }
246
110
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: ResizableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
247
111
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: ResizableComponent, selector: "[resizable]", inputs: { resizable: "resizable", resizerColor: "resizerColor", gap: "gap" }, viewQueries: [{ propertyName: "resizer", first: true, predicate: ["resizer"], descendants: true, static: true }], ngImport: i0, template: "<ng-template #resizer>\n <svg:g>\n <!-- top line -->\n <svg:line\n class=\"top\"\n [attr.x1]=\"lineGap\"\n [attr.y1]=\"-gap\"\n [attr.x2]=\"model.size().width - lineGap\"\n [attr.y2]=\"-gap\"\n [attr.stroke]=\"resizerColor\"\n stroke-width=\"2\"\n (pointerStart)=\"startResize('top', $event)\"\n />\n <!-- Left line -->\n <svg:line\n class=\"left\"\n [attr.x1]=\"-gap\"\n [attr.y1]=\"lineGap\"\n [attr.x2]=\"-gap\"\n [attr.y2]=\"model.size().height - lineGap\"\n [attr.stroke]=\"resizerColor\"\n stroke-width=\"2\"\n (pointerStart)=\"startResize('left', $event)\"\n />\n <!-- Bottom line -->\n <svg:line\n class=\"bottom\"\n [attr.x1]=\"lineGap\"\n [attr.y1]=\"model.size().height + gap\"\n [attr.x2]=\"model.size().width - lineGap\"\n [attr.y2]=\"model.size().height + gap\"\n [attr.stroke]=\"resizerColor\"\n stroke-width=\"2\"\n (pointerStart)=\"startResize('bottom', $event)\"\n />\n <!-- Right line -->\n <svg:line\n class=\"right\"\n [attr.x1]=\"model.size().width + gap\"\n [attr.y1]=\"lineGap\"\n [attr.x2]=\"model.size().width + gap\"\n [attr.y2]=\"model.size().height - lineGap\"\n [attr.stroke]=\"resizerColor\"\n stroke-width=\"2\"\n (pointerStart)=\"startResize('right', $event)\"\n />\n\n <!-- Top Left -->\n <svg:rect\n class=\"top-left\"\n [attr.x]=\"-(handleSize / 2) - gap\"\n [attr.y]=\"-(handleSize / 2) - gap\"\n [attr.width]=\"handleSize\"\n [attr.height]=\"handleSize\"\n [attr.fill]=\"resizerColor\"\n (pointerStart)=\"startResize('top-left', $event)\"\n />\n\n <!-- Top right -->\n <svg:rect\n class=\"top-right\"\n [attr.x]=\"model.size().width - (handleSize / 2) + gap\"\n [attr.y]=\"-(handleSize / 2) - gap\"\n [attr.width]=\"handleSize\"\n [attr.height]=\"handleSize\"\n [attr.fill]=\"resizerColor\"\n (pointerStart)=\"startResize('top-right', $event)\"\n />\n\n <!-- Bottom left -->\n <svg:rect\n class=\"bottom-left\"\n [attr.x]=\"-(handleSize / 2) - gap\"\n [attr.y]=\"model.size().height - (handleSize / 2) + gap\"\n [attr.width]=\"handleSize\"\n [attr.height]=\"handleSize\"\n [attr.fill]=\"resizerColor\"\n (pointerStart)=\"startResize('bottom-left', $event)\"\n />\n\n <!-- Bottom right -->\n <svg:rect\n class=\"bottom-right\"\n [attr.x]=\"model.size().width - (handleSize / 2) + gap\"\n [attr.y]=\"model.size().height - (handleSize / 2) + gap\"\n [attr.width]=\"handleSize\"\n [attr.height]=\"handleSize\"\n [attr.fill]=\"resizerColor\"\n (pointerStart)=\"startResize('bottom-right', $event)\"\n />\n </svg:g>\n</ng-template>\n\n<ng-content />\n", styles: [".top{cursor:n-resize}.left{cursor:w-resize}.right{cursor:e-resize}.bottom{cursor:s-resize}.top-left{cursor:nw-resize}.top-right{cursor:ne-resize}.bottom-left{cursor:sw-resize}.bottom-right{cursor:se-resize}\n"], dependencies: [{ kind: "directive", type: i1.PointerDirective, selector: "[pointerStart], [pointerEnd], [pointerOver], [pointerOut]", outputs: ["pointerOver", "pointerOut", "pointerStart", "pointerEnd"] }] }); }
@@ -262,4 +126,84 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
262
126
  type: ViewChild,
263
127
  args: ['resizer', { static: true }]
264
128
  }], ngAfterViewInit: [] } });
265
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resizable.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/resizable/resizable.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/resizable/resizable.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, computed, ElementRef, inject, Input, OnInit, TemplateRef, ViewChild } from '@angular/core';\nimport { RootPointerDirective } from '../../directives/root-pointer.directive';\nimport { filter, tap } from 'rxjs';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { ViewportService } from '../../services/viewport.service';\nimport { round } from '../../utils/round';\nimport { Microtask } from '../../decorators/microtask.decorator';\nimport { getNodesBounds } from '../../utils/nodes';\nimport { NodeAccessorService } from '../../services/node-accessor.service';\n\ntype Side = 'top' | 'right' | 'bottom' | 'left' | 'top-right' | 'top-left' | 'bottom-right' | 'bottom-left';\n\n@Component({\n  selector: '[resizable]',\n  templateUrl: './resizable.component.html',\n  styleUrls: ['./resizable.component.scss']\n})\nexport class ResizableComponent implements OnInit, AfterViewInit {\n  private nodeAccessor = inject(NodeAccessorService)\n  private rootPointer = inject(RootPointerDirective)\n  private viewportService = inject(ViewportService)\n  private hostRef = inject<ElementRef<Element>>(ElementRef)\n\n  @Input()\n  public set resizable(value: boolean | '') {\n    if (typeof value === 'boolean') {\n      this.model.resizable.set(value)\n    } else {\n      this.model.resizable.set(true)\n    }\n  }\n\n  @Input()\n  public resizerColor = '#2e414c'\n\n  @Input()\n  public gap = 1.5;\n\n  @ViewChild('resizer', { static: true })\n  private resizer!: TemplateRef<unknown>\n\n  protected get model() {\n    return this.nodeAccessor.model()!\n  }\n\n  protected lineGap = 3\n  protected handleSize = 6\n\n  private resizeSide: Side | null = null\n\n  private zoom = computed(() => this.viewportService.readableViewport().zoom ?? 0)\n\n  private minWidth = 0\n  private minHeight = 0\n\n  // TODO: allow reszie beside the flow\n  protected resizeOnGlobalMouseMove = this.rootPointer.pointerMovement$\n    .pipe(\n      filter(() => this.resizeSide !== null),\n      tap((event) => this.resize(event)),\n      takeUntilDestroyed()\n    )\n    .subscribe()\n\n  protected endResizeOnGlobalMouseUp = this.rootPointer.documentPointerEnd$\n    .pipe(\n      tap(() => this.endResize()),\n      takeUntilDestroyed()\n    )\n    .subscribe()\n\n  public ngOnInit(): void {\n    this.model.resizerTemplate.set(this.resizer)\n  }\n\n  @Microtask\n  public ngAfterViewInit() {\n    this.minWidth = +getComputedStyle(this.hostRef.nativeElement).minWidth.replace('px', '') || 0\n    this.minHeight = +getComputedStyle(this.hostRef.nativeElement).minHeight.replace('px', '') || 0\n  }\n\n  protected startResize(side: Side, event: Event) {\n    event.stopPropagation()\n\n    this.resizeSide = side\n    this.model.resizing.set(true);\n  }\n\n  protected resize({ movementX, movementY }: { movementX: number, movementY: number }) {\n    const offsetX = round(movementX / this.zoom())\n    const offsetY = round(movementY / this.zoom())\n\n    switch (this.resizeSide) {\n      case 'left':\n        let x = this.model.point().x + offsetX\n        x = Math.max(x, this.getMinX())\n        x = Math.min(x, this.getMaxX())\n\n        // TODO this fixes increasing width when current node hits the parent\n        if (x === this.getMinX() || x === this.getMaxX()) {\n          return\n        }\n\n        this.model.setPoint({ x, y: this.model.point().y }, false)\n\n        this.model.size.update(({ height, width }) => {\n          width -= offsetX\n          width = Math.max(width, this.minWidth)\n          width = Math.min(width, this.getMaxWidth())\n\n          return { height, width: width }\n        })\n\n        return\n      case 'right':\n        this.model.size.update(({ height, width }) => {\n          width += offsetX\n          width = Math.max(width, this.minWidth)\n          width = Math.min(width, this.getMaxWidth())\n\n          const bounds = getNodesBounds(this.model.children())\n          width = Math.max(width, bounds.x + bounds.width)\n\n          return { height, width }\n        })\n\n        return\n      case 'top':\n        let y = this.model.point().y + offsetY\n        y = Math.max(y, this.getMinY())\n        y = Math.min(y, this.getMaxY())\n\n        if (y === this.getMinY() || y === this.getMaxY()) {\n          return\n        }\n\n        this.model.setPoint({ x: this.model.point().x, y }, false)\n\n        this.model.size.update(({ height, width }) => {\n          height -= offsetY\n          height = Math.max(height, this.minHeight)\n          height = Math.min(height, this.getMaxHeight())\n\n          return { width, height }\n        })\n\n        return\n      case 'bottom':\n        this.model.size.update(({ height, width }) => {\n          height += offsetY\n          height = Math.max(height, this.minHeight)\n          height = Math.min(height, this.getMaxHeight())\n\n          const bounds = getNodesBounds(this.model.children())\n          height = Math.max(height, bounds.y + bounds.height)\n\n          return { width, height }\n        })\n\n        return\n\n      case 'top-left': {\n        let x = this.model.point().x + offsetX\n        x = Math.max(x, this.getMinX())\n        x = Math.min(x, this.getMaxX())\n\n        let y = this.model.point().y + offsetY\n        y = Math.max(y, this.getMinY())\n        y = Math.min(y, this.getMaxY())\n\n        if (\n          x === this.getMinX() || y === this.getMinY() ||\n          x === this.getMaxX() || y === this.getMaxY()\n        ) {\n          return\n        }\n\n        this.model.setPoint({ x, y }, false)\n\n        this.model.size.update(({ height, width }) => {\n          width -= offsetX\n          width = Math.max(width, this.minWidth)\n          width = Math.min(width, this.getMaxWidth())\n\n          height -= offsetY\n          height = Math.max(height, this.minHeight)\n          height = Math.min(height, this.getMaxHeight())\n\n          return { height, width }\n        })\n\n        return\n      }\n\n      case 'top-right': {\n        let y = this.model.point().y + offsetY\n        y = Math.max(y, this.getMinY())\n        y = Math.min(y, this.getMaxY())\n\n        if (y === this.getMinX() || y === this.getMaxY()) {\n          return\n        }\n\n        this.model.setPoint({ x: this.model.point().x, y }, false)\n\n        this.model.size.update(({ height, width }) => {\n          const bounds = getNodesBounds(this.model.children())\n\n          width += offsetX\n          width = Math.max(width, this.minWidth)\n          width = Math.min(width, this.getMaxWidth())\n          width = Math.max(width, bounds.x + bounds.width)\n\n\n          height -= offsetY\n          height = Math.max(height, this.minHeight)\n          height = Math.min(height, this.getMaxHeight())\n\n          return { height, width }\n        })\n\n        return\n      }\n\n      case 'bottom-left': {\n        let x = this.model.point().x + offsetX\n        x = Math.max(x, this.getMinX())\n        x = Math.min(x, this.getMaxX())\n\n        if (x === this.getMinX() || x === this.getMaxX()) {\n          return\n        }\n\n        this.model.setPoint({ x, y: this.model.point().y }, false)\n\n        this.model.size.update(({ height, width }) => {\n          width -= offsetX\n          width = Math.max(width, this.minWidth)\n          width = Math.min(width, this.getMaxWidth())\n\n          height += offsetY\n          height = Math.max(height, this.minHeight)\n          height = Math.min(height, this.getMaxHeight())\n\n          const bounds = getNodesBounds(this.model.children())\n          height = Math.max(height, bounds.y + bounds.height)\n\n          return { height, width }\n        })\n\n        return\n      }\n\n      case 'bottom-right': {\n        this.model.size.update(({ height, width }) => {\n          const bounds = getNodesBounds(this.model.children())\n\n          width += offsetX\n          width = Math.max(width, this.minWidth)\n          width = Math.min(width, this.getMaxWidth())\n          width = Math.max(width, bounds.x + bounds.width)\n\n\n          height += offsetY\n          height = Math.max(height, this.minHeight)\n          height = Math.min(height, this.getMaxHeight())\n          height = Math.max(height, bounds.y + bounds.height)\n\n          return { height, width }\n        })\n      }\n    }\n  }\n\n  protected endResize() {\n    this.resizeSide = null\n    this.model.resizing.set(false)\n  }\n\n  private getMaxWidth() {\n    const parent = this.model.parent()\n\n    if (parent) {\n      return parent.size().width - this.model.point().x\n    }\n\n    return Infinity\n  }\n\n  private getMaxHeight() {\n    const parent = this.model.parent()\n\n    if (parent) {\n      return parent.size().height - this.model.point().y\n    }\n\n    return Infinity\n  }\n\n  private getMinX() {\n    const parent = this.model.parent()\n\n    if (parent) {\n      return 0\n    }\n\n    return -Infinity\n  }\n\n  private getMinY() {\n    const parent = this.model.parent()\n\n    if (parent) {\n      return 0\n    }\n\n    return -Infinity\n  }\n\n  private getMaxX() {\n    const x = this.model.point().x\n    const width = this.model.size().width\n    const children = this.model.children()\n\n    if (children) {\n      const bounds = getNodesBounds(children)\n\n      return x + (bounds.x + bounds.width) >= x + width ? x : (width - this.minWidth) + x\n    }\n\n    return (width - this.minWidth) + x\n  }\n\n  private getMaxY() {\n    const y = this.model.point().y\n    const height = this.model.size().height\n    const children = this.model.children()\n\n    if (children) {\n      const bounds = getNodesBounds(children)\n\n      return y + (bounds.y + bounds.height) >= y + height ? y : (height - this.minHeight) + y\n    }\n\n    return (height - this.minHeight) + y\n  }\n}\n","<ng-template #resizer>\n  <svg:g>\n    <!-- top line -->\n    <svg:line\n      class=\"top\"\n      [attr.x1]=\"lineGap\"\n      [attr.y1]=\"-gap\"\n      [attr.x2]=\"model.size().width - lineGap\"\n      [attr.y2]=\"-gap\"\n      [attr.stroke]=\"resizerColor\"\n      stroke-width=\"2\"\n      (pointerStart)=\"startResize('top', $event)\"\n    />\n    <!-- Left line -->\n    <svg:line\n      class=\"left\"\n      [attr.x1]=\"-gap\"\n      [attr.y1]=\"lineGap\"\n      [attr.x2]=\"-gap\"\n      [attr.y2]=\"model.size().height - lineGap\"\n      [attr.stroke]=\"resizerColor\"\n      stroke-width=\"2\"\n      (pointerStart)=\"startResize('left', $event)\"\n    />\n    <!-- Bottom line -->\n    <svg:line\n      class=\"bottom\"\n      [attr.x1]=\"lineGap\"\n      [attr.y1]=\"model.size().height + gap\"\n      [attr.x2]=\"model.size().width - lineGap\"\n      [attr.y2]=\"model.size().height + gap\"\n      [attr.stroke]=\"resizerColor\"\n      stroke-width=\"2\"\n      (pointerStart)=\"startResize('bottom', $event)\"\n    />\n    <!-- Right line -->\n    <svg:line\n      class=\"right\"\n      [attr.x1]=\"model.size().width + gap\"\n      [attr.y1]=\"lineGap\"\n      [attr.x2]=\"model.size().width + gap\"\n      [attr.y2]=\"model.size().height - lineGap\"\n      [attr.stroke]=\"resizerColor\"\n      stroke-width=\"2\"\n      (pointerStart)=\"startResize('right', $event)\"\n    />\n\n    <!-- Top Left -->\n    <svg:rect\n      class=\"top-left\"\n      [attr.x]=\"-(handleSize / 2) - gap\"\n      [attr.y]=\"-(handleSize / 2) - gap\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor\"\n      (pointerStart)=\"startResize('top-left', $event)\"\n    />\n\n    <!-- Top right -->\n    <svg:rect\n      class=\"top-right\"\n      [attr.x]=\"model.size().width - (handleSize / 2) + gap\"\n      [attr.y]=\"-(handleSize / 2) - gap\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor\"\n      (pointerStart)=\"startResize('top-right', $event)\"\n    />\n\n    <!-- Bottom left -->\n    <svg:rect\n      class=\"bottom-left\"\n      [attr.x]=\"-(handleSize / 2) - gap\"\n      [attr.y]=\"model.size().height - (handleSize / 2) + gap\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor\"\n      (pointerStart)=\"startResize('bottom-left', $event)\"\n    />\n\n    <!-- Bottom right -->\n    <svg:rect\n      class=\"bottom-right\"\n      [attr.x]=\"model.size().width - (handleSize / 2) + gap\"\n      [attr.y]=\"model.size().height - (handleSize / 2) + gap\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor\"\n      (pointerStart)=\"startResize('bottom-right', $event)\"\n    />\n  </svg:g>\n</ng-template>\n\n<ng-content />\n"]}
129
+ function calcOffset(movementX, movementY, zoom) {
130
+ return {
131
+ offsetX: round(movementX / zoom),
132
+ offsetY: round(movementY / zoom)
133
+ };
134
+ }
135
+ function applyResize(side, model, offset) {
136
+ const { offsetX, offsetY } = offset;
137
+ const { x, y } = model.point();
138
+ const { width, height } = model.size();
139
+ // Handle each case of resizing (top, bottom, left, right, corners)
140
+ switch (side) {
141
+ case 'left':
142
+ return { x: x + offsetX, y, width: width - offsetX, height };
143
+ case 'right':
144
+ return { x, y, width: width + offsetX, height };
145
+ case 'top':
146
+ return { x, y: y + offsetY, width, height: height - offsetY };
147
+ case 'bottom':
148
+ return { x, y, width, height: height + offsetY };
149
+ case 'top-left':
150
+ return { x: x + offsetX, y: y + offsetY, width: width - offsetX, height: height - offsetY };
151
+ case 'top-right':
152
+ return { x, y: y + offsetY, width: width + offsetX, height: height - offsetY };
153
+ case 'bottom-left':
154
+ return { x: x + offsetX, y, width: width - offsetX, height: height + offsetY };
155
+ case 'bottom-right':
156
+ return { x, y, width: width + offsetX, height: height + offsetY };
157
+ }
158
+ }
159
+ function constrainRect(rect, model, side, minWidth, minHeight) {
160
+ let { x, y, width, height } = rect;
161
+ // 1. Prevent negative dimensions
162
+ width = Math.max(width, 0);
163
+ height = Math.max(height, 0);
164
+ // 2. Apply minimum size constraints
165
+ width = Math.max(minWidth, width);
166
+ height = Math.max(minHeight, height);
167
+ // Apply left/top constraints based on minimum size
168
+ x = Math.min(x, model.point().x + model.size().width - minWidth);
169
+ y = Math.min(y, model.point().y + model.size().height - minHeight);
170
+ const parent = model.parent();
171
+ // 3. Apply maximum size constraints based on parent size (if exists)
172
+ if (parent) {
173
+ x = Math.max(x, 0); // Left boundary of the parent
174
+ y = Math.max(y, 0); // Top boundary of the parent
175
+ if (x === 0) {
176
+ width = model.point().x + model.size().width;
177
+ }
178
+ if (y === 0) {
179
+ height = model.point().y + model.size().height;
180
+ }
181
+ width = Math.min(width, parent.size().width - model.point().x);
182
+ height = Math.min(height, parent.size().height - model.point().y);
183
+ }
184
+ const bounds = getNodesBounds(model.children());
185
+ // 4. Apply child node constraints (if children exist)
186
+ if (bounds) {
187
+ if (side.includes('left')) {
188
+ x = Math.min(x, (model.point().x + model.size().width) - (bounds.x + bounds.width));
189
+ width = Math.max(width, bounds.x + bounds.width);
190
+ }
191
+ if (side.includes('right')) {
192
+ width = Math.max(width, bounds.x + bounds.width);
193
+ }
194
+ if (side.includes('bottom')) {
195
+ height = Math.max(height, bounds.y + bounds.height);
196
+ }
197
+ if (side.includes('top')) {
198
+ y = Math.min(y, (model.point().y + model.size().height) - (bounds.y + bounds.height));
199
+ height = Math.max(height, bounds.y + bounds.height);
200
+ }
201
+ }
202
+ return {
203
+ x,
204
+ y,
205
+ width,
206
+ height
207
+ };
208
+ }
209
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resizable.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/resizable/resizable.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/resizable/resizable.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, computed, ElementRef, inject, Input, OnInit, TemplateRef, ViewChild } from '@angular/core';\nimport { RootPointerDirective } from '../../directives/root-pointer.directive';\nimport { filter, tap } from 'rxjs';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { ViewportService } from '../../services/viewport.service';\nimport { round } from '../../utils/round';\nimport { Microtask } from '../../decorators/microtask.decorator';\nimport { getNodesBounds } from '../../utils/nodes';\nimport { NodeAccessorService } from '../../services/node-accessor.service';\nimport { NodeModel } from '../../models/node.model';\nimport { Rect } from '../../interfaces/rect';\nimport { PointerEvent } from '../../directives/root-pointer.directive';\nimport { SpacePointContextDirective } from '../../directives/space-point-context.directive';\n\ntype Side = 'top' | 'right' | 'bottom' | 'left' | 'top-right' | 'top-left' | 'bottom-right' | 'bottom-left';\n\n@Component({\n  selector: '[resizable]',\n  templateUrl: './resizable.component.html',\n  styleUrls: ['./resizable.component.scss']\n})\nexport class ResizableComponent implements OnInit, AfterViewInit {\n  private nodeAccessor = inject(NodeAccessorService)\n  private rootPointer = inject(RootPointerDirective)\n  private viewportService = inject(ViewportService)\n  private spacePointContext = inject(SpacePointContextDirective)\n  private hostRef = inject<ElementRef<Element>>(ElementRef)\n\n  @Input()\n  public set resizable(value: boolean | '') {\n    if (typeof value === 'boolean') {\n      this.model.resizable.set(value)\n    } else {\n      this.model.resizable.set(true)\n    }\n  }\n\n  @Input()\n  public resizerColor = '#2e414c'\n\n  @Input()\n  public gap = 1.5;\n\n  @ViewChild('resizer', { static: true })\n  private resizer!: TemplateRef<unknown>\n\n  protected get model() {\n    return this.nodeAccessor.model()!\n  }\n\n  protected lineGap = 3\n  protected handleSize = 6\n\n  private resizeSide: Side | null = null\n\n  private zoom = computed(() => this.viewportService.readableViewport().zoom ?? 0)\n\n  private minWidth = 0\n  private minHeight = 0\n\n  // TODO: allow reszie beside the flow\n  protected resizeOnGlobalMouseMove = this.rootPointer.pointerMovement$\n    .pipe(\n      filter(() => this.resizeSide !== null),\n      filter((event) => event.movementX !== 0 || event.movementY !== 0),\n      tap((event) => this.resize(event)),\n      takeUntilDestroyed()\n    )\n    .subscribe()\n\n  protected endResizeOnGlobalMouseUp = this.rootPointer.documentPointerEnd$\n    .pipe(\n      tap(() => this.endResize()),\n      takeUntilDestroyed()\n    )\n    .subscribe()\n\n  public ngOnInit(): void {\n    this.model.resizerTemplate.set(this.resizer)\n  }\n\n  @Microtask\n  public ngAfterViewInit() {\n    this.minWidth = +getComputedStyle(this.hostRef.nativeElement).minWidth.replace('px', '') || 0\n    this.minHeight = +getComputedStyle(this.hostRef.nativeElement).minHeight.replace('px', '') || 0\n  }\n\n  protected startResize(side: Side, event: Event) {\n    event.stopPropagation()\n    this.resizeSide = side\n    this.model.resizing.set(true)\n  }\n\n  protected resize(event: PointerEvent) {\n    if (!this.resizeSide) return;\n    if (this.isResizeConstrained(event)) return;\n\n    const offset = calcOffset(event.movementX, event.movementY, this.zoom())\n    const { x, y, width, height } = constrainRect(\n      applyResize(this.resizeSide, this.model, offset),\n      this.model,\n      this.resizeSide,\n      this.minWidth,\n      this.minHeight\n    )\n\n    this.model.setPoint({ x, y }, false)\n    this.model.size.set({ width, height })\n  }\n\n  protected endResize() {\n    this.resizeSide = null\n    this.model.resizing.set(false)\n  }\n\n  private isResizeConstrained({ x, y, movementX, movementY }: PointerEvent): boolean {\n    const flowPoint = this.spacePointContext.documentPointToFlowPoint({ x, y })\n\n    if (this.resizeSide?.includes('right')) {\n      if (movementX > 0 && flowPoint.x < (this.model.point().x + this.model.size().width)) {\n        return true\n      }\n\n      if (movementX < 0 && flowPoint.x > this.model.point().x + this.model.size().width) {\n        return true\n      }\n    }\n\n    if (this.resizeSide?.includes('left')) {\n      if (movementX < 0 && flowPoint.x > this.model.point().x) {\n        return true\n      }\n\n      if (movementX > 0 && flowPoint.x < this.model.point().x) {\n        return true\n      }\n    }\n\n    if (this.resizeSide?.includes('bottom')) {\n      if (movementY > 0 && flowPoint.y < (this.model.point().y + this.model.size().height)) {\n        return true\n      }\n\n      if (movementY < 0 && flowPoint.y > this.model.point().y + this.model.size().height) {\n        return true\n      }\n    }\n\n    if (this.resizeSide?.includes('top')) {\n      if (movementY < 0 && flowPoint.y > this.model.point().y) {\n        return true\n      }\n\n      if (movementY > 0 && flowPoint.y < this.model.point().y) {\n        return true\n      }\n    }\n\n    return false\n  }\n}\n\nfunction calcOffset(movementX: number, movementY: number, zoom: number) {\n  return {\n    offsetX: round(movementX / zoom),\n    offsetY: round(movementY / zoom)\n  };\n}\n\nfunction applyResize(\n  side: Side,\n  model: NodeModel,\n  offset: { offsetX: number, offsetY: number }\n): Rect {\n  const { offsetX, offsetY } = offset\n  const { x, y } = model.point()\n  const { width, height } = model.size()\n\n  // Handle each case of resizing (top, bottom, left, right, corners)\n  switch (side) {\n    case 'left':\n      return { x: x + offsetX, y, width: width - offsetX, height }\n    case 'right':\n      return { x, y, width: width + offsetX, height }\n    case 'top':\n      return { x, y: y + offsetY, width, height: height - offsetY }\n    case 'bottom':\n      return { x, y, width, height: height + offsetY }\n    case 'top-left':\n      return { x: x + offsetX, y: y + offsetY, width: width - offsetX, height: height - offsetY }\n    case 'top-right':\n      return { x, y: y + offsetY, width: width + offsetX, height: height - offsetY }\n    case 'bottom-left':\n      return { x: x + offsetX, y, width: width - offsetX, height: height + offsetY }\n    case 'bottom-right':\n      return { x, y, width: width + offsetX, height: height + offsetY }\n  }\n}\n\nfunction constrainRect(\n  rect: Rect,\n  model: NodeModel,\n  side: Side,\n  minWidth: number,\n  minHeight: number\n) {\n  let { x, y, width, height } = rect\n\n  // 1. Prevent negative dimensions\n  width = Math.max(width, 0)\n  height = Math.max(height, 0)\n\n  // 2. Apply minimum size constraints\n  width = Math.max(minWidth, width)\n  height = Math.max(minHeight, height)\n\n  // Apply left/top constraints based on minimum size\n  x = Math.min(x, model.point().x + model.size().width - minWidth)\n  y = Math.min(y, model.point().y + model.size().height - minHeight)\n\n  const parent = model.parent()\n  // 3. Apply maximum size constraints based on parent size (if exists)\n  if (parent) {\n    x = Math.max(x, 0); // Left boundary of the parent\n    y = Math.max(y, 0); // Top boundary of the parent\n\n    if (x === 0) {\n      width = model.point().x + model.size().width\n    }\n\n    if (y === 0) {\n      height = model.point().y + model.size().height\n    }\n\n    width = Math.min(width, parent.size().width - model.point().x)\n    height = Math.min(height, parent.size().height - model.point().y)\n  }\n\n  const bounds = getNodesBounds(model.children())\n  // 4. Apply child node constraints (if children exist)\n  if (bounds) {\n    if (side.includes('left')) {\n      x = Math.min(x, (model.point().x + model.size().width) - (bounds.x + bounds.width))\n      width = Math.max(width, bounds.x + bounds.width)\n    }\n\n    if (side.includes('right')) {\n      width = Math.max(width, bounds.x + bounds.width)\n    }\n\n    if (side.includes('bottom')) {\n      height = Math.max(height, bounds.y + bounds.height)\n    }\n\n    if (side.includes('top')) {\n      y = Math.min(y, (model.point().y + model.size().height) - (bounds.y + bounds.height))\n      height = Math.max(height, bounds.y + bounds.height)\n    }\n  }\n\n  return {\n    x,\n    y,\n    width,\n    height\n  };\n}\n","<ng-template #resizer>\n  <svg:g>\n    <!-- top line -->\n    <svg:line\n      class=\"top\"\n      [attr.x1]=\"lineGap\"\n      [attr.y1]=\"-gap\"\n      [attr.x2]=\"model.size().width - lineGap\"\n      [attr.y2]=\"-gap\"\n      [attr.stroke]=\"resizerColor\"\n      stroke-width=\"2\"\n      (pointerStart)=\"startResize('top', $event)\"\n    />\n    <!-- Left line -->\n    <svg:line\n      class=\"left\"\n      [attr.x1]=\"-gap\"\n      [attr.y1]=\"lineGap\"\n      [attr.x2]=\"-gap\"\n      [attr.y2]=\"model.size().height - lineGap\"\n      [attr.stroke]=\"resizerColor\"\n      stroke-width=\"2\"\n      (pointerStart)=\"startResize('left', $event)\"\n    />\n    <!-- Bottom line -->\n    <svg:line\n      class=\"bottom\"\n      [attr.x1]=\"lineGap\"\n      [attr.y1]=\"model.size().height + gap\"\n      [attr.x2]=\"model.size().width - lineGap\"\n      [attr.y2]=\"model.size().height + gap\"\n      [attr.stroke]=\"resizerColor\"\n      stroke-width=\"2\"\n      (pointerStart)=\"startResize('bottom', $event)\"\n    />\n    <!-- Right line -->\n    <svg:line\n      class=\"right\"\n      [attr.x1]=\"model.size().width + gap\"\n      [attr.y1]=\"lineGap\"\n      [attr.x2]=\"model.size().width + gap\"\n      [attr.y2]=\"model.size().height - lineGap\"\n      [attr.stroke]=\"resizerColor\"\n      stroke-width=\"2\"\n      (pointerStart)=\"startResize('right', $event)\"\n    />\n\n    <!-- Top Left -->\n    <svg:rect\n      class=\"top-left\"\n      [attr.x]=\"-(handleSize / 2) - gap\"\n      [attr.y]=\"-(handleSize / 2) - gap\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor\"\n      (pointerStart)=\"startResize('top-left', $event)\"\n    />\n\n    <!-- Top right -->\n    <svg:rect\n      class=\"top-right\"\n      [attr.x]=\"model.size().width - (handleSize / 2) + gap\"\n      [attr.y]=\"-(handleSize / 2) - gap\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor\"\n      (pointerStart)=\"startResize('top-right', $event)\"\n    />\n\n    <!-- Bottom left -->\n    <svg:rect\n      class=\"bottom-left\"\n      [attr.x]=\"-(handleSize / 2) - gap\"\n      [attr.y]=\"model.size().height - (handleSize / 2) + gap\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor\"\n      (pointerStart)=\"startResize('bottom-left', $event)\"\n    />\n\n    <!-- Bottom right -->\n    <svg:rect\n      class=\"bottom-right\"\n      [attr.x]=\"model.size().width - (handleSize / 2) + gap\"\n      [attr.y]=\"model.size().height - (handleSize / 2) + gap\"\n      [attr.width]=\"handleSize\"\n      [attr.height]=\"handleSize\"\n      [attr.fill]=\"resizerColor\"\n      (pointerStart)=\"startResize('bottom-right', $event)\"\n    />\n  </svg:g>\n</ng-template>\n\n<ng-content />\n"]}
@@ -1,9 +1,13 @@
1
- import { Injectable } from '@angular/core';
1
+ import { Injectable, inject } from '@angular/core';
2
2
  import { select } from 'd3-selection';
3
3
  import { drag } from 'd3-drag';
4
4
  import { round } from '../utils/round';
5
+ import { FlowEntitiesService } from './flow-entities.service';
5
6
  import * as i0 from "@angular/core";
6
7
  export class DraggableService {
8
+ constructor() {
9
+ this.entitiesService = inject(FlowEntitiesService);
10
+ }
7
11
  /**
8
12
  * Enable draggable behavior for element.
9
13
  *
@@ -39,27 +43,24 @@ export class DraggableService {
39
43
  * @returns
40
44
  */
41
45
  getDragBehavior(model) {
42
- let deltaX;
43
- let deltaY;
46
+ let dragNodes = [];
47
+ let initialPositions = [];
44
48
  return drag()
45
49
  .on('start', (event) => {
46
- deltaX = model.point().x - event.x;
47
- deltaY = model.point().y - event.y;
50
+ dragNodes = this.getDragNodes(model);
51
+ initialPositions = dragNodes.map(node => ({
52
+ x: node.point().x - event.x,
53
+ y: node.point().y - event.y
54
+ }));
48
55
  })
49
56
  .on('drag', (event) => {
50
- let point = {
51
- x: round(event.x + deltaX),
52
- y: round(event.y + deltaY)
53
- };
54
- const parent = model.parent();
55
- // keep node in bounds of parent
56
- if (parent) {
57
- point.x = Math.min(parent.size().width - model.size().width, point.x);
58
- point.x = Math.max(0, point.x);
59
- point.y = Math.min(parent.size().height - model.size().height, point.y);
60
- point.y = Math.max(0, point.y);
61
- }
62
- model.setPoint(point, true);
57
+ dragNodes.forEach((model, index) => {
58
+ let point = {
59
+ x: round(event.x + initialPositions[index].x),
60
+ y: round(event.y + initialPositions[index].y)
61
+ };
62
+ moveNode(model, point);
63
+ });
63
64
  });
64
65
  }
65
66
  /**
@@ -72,10 +73,30 @@ export class DraggableService {
72
73
  event.sourceEvent.stopPropagation();
73
74
  });
74
75
  }
76
+ getDragNodes(model) {
77
+ return model.selected()
78
+ ? this.entitiesService
79
+ .nodes()
80
+ // selected draggable nodes (with current node)
81
+ .filter(node => node.selected() && node.draggable())
82
+ // we only can move current node if it's not selected
83
+ : [model];
84
+ }
75
85
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DraggableService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
76
86
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DraggableService }); }
77
87
  }
78
88
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: DraggableService, decorators: [{
79
89
  type: Injectable
80
90
  }] });
81
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZHJhZ2dhYmxlLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZ3gtdmZsb3ctbGliL3NyYy9saWIvdmZsb3cvc2VydmljZXMvZHJhZ2dhYmxlLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBa0IsTUFBTSxlQUFlLENBQUM7QUFDM0QsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGNBQWMsQ0FBQztBQUN0QyxPQUFPLEVBQWUsSUFBSSxFQUFFLE1BQU0sU0FBUyxDQUFDO0FBRTVDLE9BQU8sRUFBRSxLQUFLLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQzs7QUFLdkMsTUFBTSxPQUFPLGdCQUFnQjtJQUMzQjs7Ozs7T0FLRztJQUNJLE1BQU0sQ0FBQyxPQUFnQixFQUFFLEtBQWdCO1FBQzlDLE1BQU0sU0FBUyxHQUFHLE1BQU0sQ0FBQyxPQUFPLENBQUMsQ0FBQTtRQUVqQyxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxlQUFlLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQTtJQUM3QyxDQUFDO0lBRUQ7Ozs7O09BS0c7SUFDSSxPQUFPLENBQUMsT0FBZ0I7UUFDN0IsTUFBTSxTQUFTLEdBQUcsTUFBTSxDQUFDLE9BQU8sQ0FBQyxDQUFBO1FBRWpDLFNBQVMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLHFCQUFxQixFQUFFLENBQUMsQ0FBQTtJQUM5QyxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNJLE9BQU8sQ0FBQyxPQUFnQjtRQUM3QixNQUFNLENBQUMsT0FBTyxDQUFDLENBQUMsRUFBRSxDQUFDLE9BQU8sRUFBRSxJQUFJLENBQUMsQ0FBQTtJQUNuQyxDQUFDO0lBRUQ7Ozs7O09BS0c7SUFDSyxlQUFlLENBQUMsS0FBZ0I7UUFDdEMsSUFBSSxNQUFjLENBQUE7UUFDbEIsSUFBSSxNQUFjLENBQUE7UUFFbEIsT0FBTyxJQUFJLEVBQUU7YUFDVixFQUFFLENBQUMsT0FBTyxFQUFFLENBQUMsS0FBZ0IsRUFBRSxFQUFFO1lBQ2hDLE1BQU0sR0FBRyxLQUFLLENBQUMsS0FBSyxFQUFFLENBQUMsQ0FBQyxHQUFHLEtBQUssQ0FBQyxDQUFDLENBQUE7WUFDbEMsTUFBTSxHQUFHLEtBQUssQ0FBQyxLQUFLLEVBQUUsQ0FBQyxDQUFDLEdBQUcsS0FBSyxDQUFDLENBQUMsQ0FBQTtRQUNwQyxDQUFDLENBQUM7YUFFRCxFQUFFLENBQUMsTUFBTSxFQUFFLENBQUMsS0FBZ0IsRUFBRSxFQUFFO1lBQy9CLElBQUksS0FBSyxHQUFHO2dCQUNWLENBQUMsRUFBRSxLQUFLLENBQUMsS0FBSyxDQUFDLENBQUMsR0FBRyxNQUFNLENBQUM7Z0JBQzFCLENBQUMsRUFBRSxLQUFLLENBQUMsS0FBSyxDQUFDLENBQUMsR0FBRyxNQUFNLENBQUM7YUFDM0IsQ0FBQTtZQUVELE1BQU0sTUFBTSxHQUFHLEtBQUssQ0FBQyxNQUFNLEVBQUUsQ0FBQTtZQUM3QixnQ0FBZ0M7WUFDaEMsSUFBSSxNQUFNLEVBQUU7Z0JBQ1YsS0FBSyxDQUFDLENBQUMsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDLElBQUksRUFBRSxDQUFDLEtBQUssRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUE7Z0JBQ3JFLEtBQUssQ0FBQyxDQUFDLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFBO2dCQUU5QixLQUFLLENBQUMsQ0FBQyxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRSxDQUFDLE1BQU0sR0FBRyxLQUFLLENBQUMsSUFBSSxFQUFFLENBQUMsTUFBTSxFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQTtnQkFDdkUsS0FBSyxDQUFDLENBQUMsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUE7YUFDL0I7WUFFRCxLQUFLLENBQUMsUUFBUSxDQUFDLEtBQUssRUFBRSxJQUFJLENBQUMsQ0FBQTtRQUM3QixDQUFDLENBQUMsQ0FBQTtJQUNOLENBQUM7SUFFRDs7O09BR0c7SUFDSyxxQkFBcUI7UUFDM0IsT0FBTyxJQUFJLEVBQUU7YUFDVixFQUFFLENBQUMsTUFBTSxFQUFFLENBQUMsS0FBZ0IsRUFBRSxFQUFFO1lBQzlCLEtBQUssQ0FBQyxXQUFxQixDQUFDLGVBQWUsRUFBRSxDQUFBO1FBQ2hELENBQUMsQ0FBQyxDQUFBO0lBQ04sQ0FBQzsrR0EvRVUsZ0JBQWdCO21IQUFoQixnQkFBZ0I7OzRGQUFoQixnQkFBZ0I7a0JBRDVCLFVBQVUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlLCBlZmZlY3QsIGluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgc2VsZWN0IH0gZnJvbSAnZDMtc2VsZWN0aW9uJztcbmltcG9ydCB7IEQzRHJhZ0V2ZW50LCBkcmFnIH0gZnJvbSAnZDMtZHJhZyc7XG5pbXBvcnQgeyBOb2RlTW9kZWwgfSBmcm9tICcuLi9tb2RlbHMvbm9kZS5tb2RlbCc7XG5pbXBvcnQgeyByb3VuZCB9IGZyb20gJy4uL3V0aWxzL3JvdW5kJztcblxudHlwZSBEcmFnRXZlbnQgPSBEM0RyYWdFdmVudDxFbGVtZW50LCB1bmtub3duLCB1bmtub3duPlxuXG5ASW5qZWN0YWJsZSgpXG5leHBvcnQgY2xhc3MgRHJhZ2dhYmxlU2VydmljZSB7XG4gIC8qKlxuICAgKiBFbmFibGUgZHJhZ2dhYmxlIGJlaGF2aW9yIGZvciBlbGVtZW50LlxuICAgKlxuICAgKiBAcGFyYW0gZWxlbWVudCB0YXJnZXQgZWxlbWVudCBmb3IgdG9nZ2xpbmcgZHJhZ2dhYmxlXG4gICAqIEBwYXJhbSBtb2RlbCBtb2RlbCB3aXRoIGRhdGEgZm9yIHRoaXMgZWxlbWVudFxuICAgKi9cbiAgcHVibGljIGVuYWJsZShlbGVtZW50OiBFbGVtZW50LCBtb2RlbDogTm9kZU1vZGVsKSB7XG4gICAgY29uc3QgZDNFbGVtZW50ID0gc2VsZWN0KGVsZW1lbnQpXG5cbiAgICBkM0VsZW1lbnQuY2FsbCh0aGlzLmdldERyYWdCZWhhdmlvcihtb2RlbCkpXG4gIH1cblxuICAvKipcbiAgICogRGlzYWJsZSBkcmFnZ2FibGUgYmVoYXZpb3IgZm9yIGVsZW1lbnQuXG4gICAqXG4gICAqIEBwYXJhbSBlbGVtZW50IHRhcmdldCBlbGVtZW50IGZvciB0b2dnbGluZyBkcmFnZ2FibGVcbiAgICogQHBhcmFtIG1vZGVsIG1vZGVsIHdpdGggZGF0YSBmb3IgdGhpcyBlbGVtZW50XG4gICAqL1xuICBwdWJsaWMgZGlzYWJsZShlbGVtZW50OiBFbGVtZW50KSB7XG4gICAgY29uc3QgZDNFbGVtZW50ID0gc2VsZWN0KGVsZW1lbnQpXG5cbiAgICBkM0VsZW1lbnQuY2FsbCh0aGlzLmdldElnbm9yZURyYWdCZWhhdmlvcigpKVxuICB9XG5cbiAgLyoqXG4gICAqIFRPRE86IG5vdCBzaHVyZSBpZiB0aGlzIHdvcmssIG5lZWQgdG8gY2hlY2tcbiAgICpcbiAgICogQHBhcmFtIGVsZW1lbnRcbiAgICovXG4gIHB1YmxpYyBkZXN0cm95KGVsZW1lbnQ6IEVsZW1lbnQpIHtcbiAgICBzZWxlY3QoZWxlbWVudCkub24oJy5kcmFnJywgbnVsbClcbiAgfVxuXG4gIC8qKlxuICAgKiBOb2RlIGRyYWcgYmVoYXZpb3IuIFVwZGF0ZWQgbm9kZSdzIGNvb3JkaW5hdGUgYWNjb3JkaW5nIHRvIGRyYWdnaW5nXG4gICAqXG4gICAqIEBwYXJhbSBtb2RlbFxuICAgKiBAcmV0dXJuc1xuICAgKi9cbiAgcHJpdmF0ZSBnZXREcmFnQmVoYXZpb3IobW9kZWw6IE5vZGVNb2RlbCkge1xuICAgIGxldCBkZWx0YVg6IG51bWJlclxuICAgIGxldCBkZWx0YVk6IG51bWJlclxuXG4gICAgcmV0dXJuIGRyYWcoKVxuICAgICAgLm9uKCdzdGFydCcsIChldmVudDogRHJhZ0V2ZW50KSA9PiB7XG4gICAgICAgIGRlbHRhWCA9IG1vZGVsLnBvaW50KCkueCAtIGV2ZW50LnhcbiAgICAgICAgZGVsdGFZID0gbW9kZWwucG9pbnQoKS55IC0gZXZlbnQueVxuICAgICAgfSlcblxuICAgICAgLm9uKCdkcmFnJywgKGV2ZW50OiBEcmFnRXZlbnQpID0+IHtcbiAgICAgICAgbGV0IHBvaW50ID0ge1xuICAgICAgICAgIHg6IHJvdW5kKGV2ZW50LnggKyBkZWx0YVgpLFxuICAgICAgICAgIHk6IHJvdW5kKGV2ZW50LnkgKyBkZWx0YVkpXG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBwYXJlbnQgPSBtb2RlbC5wYXJlbnQoKVxuICAgICAgICAvLyBrZWVwIG5vZGUgaW4gYm91bmRzIG9mIHBhcmVudFxuICAgICAgICBpZiAocGFyZW50KSB7XG4gICAgICAgICAgcG9pbnQueCA9IE1hdGgubWluKHBhcmVudC5zaXplKCkud2lkdGggLSBtb2RlbC5zaXplKCkud2lkdGgsIHBvaW50LngpXG4gICAgICAgICAgcG9pbnQueCA9IE1hdGgubWF4KDAsIHBvaW50LngpXG5cbiAgICAgICAgICBwb2ludC55ID0gTWF0aC5taW4ocGFyZW50LnNpemUoKS5oZWlnaHQgLSBtb2RlbC5zaXplKCkuaGVpZ2h0LCBwb2ludC55KVxuICAgICAgICAgIHBvaW50LnkgPSBNYXRoLm1heCgwLCBwb2ludC55KVxuICAgICAgICB9XG5cbiAgICAgICAgbW9kZWwuc2V0UG9pbnQocG9pbnQsIHRydWUpXG4gICAgICB9KVxuICB9XG5cbiAgLyoqXG4gICAqIFNwZWNpZnkgaWdub3JpbmcgZHJhZyBiZWhhdmlvci4gSXQncyByZXNwb25zaWJsZSBmb3Igbm90IG1vdmluZyB0aGUgbWFwIHdoZW4gdXNlciB0cmllcyB0byBkcmFnIG5vZGVcbiAgICogd2l0aCBkaXNhYmxlZCBkcmFnIGJlaGF2aW9yXG4gICAqL1xuICBwcml2YXRlIGdldElnbm9yZURyYWdCZWhhdmlvcigpIHtcbiAgICByZXR1cm4gZHJhZygpXG4gICAgICAub24oJ2RyYWcnLCAoZXZlbnQ6IERyYWdFdmVudCkgPT4ge1xuICAgICAgICAoZXZlbnQuc291cmNlRXZlbnQgYXMgRXZlbnQpLnN0b3BQcm9wYWdhdGlvbigpXG4gICAgICB9KVxuICB9XG59XG4iXX0=
91
+ function moveNode(model, point) {
92
+ const parent = model.parent();
93
+ // keep node in bounds of parent
94
+ if (parent) {
95
+ point.x = Math.min(parent.size().width - model.size().width, point.x);
96
+ point.x = Math.max(0, point.x);
97
+ point.y = Math.min(parent.size().height - model.size().height, point.y);
98
+ point.y = Math.max(0, point.y);
99
+ }
100
+ model.setPoint(point, true);
101
+ }
102
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"draggable.service.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/services/draggable.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, effect, inject } from '@angular/core';\nimport { select } from 'd3-selection';\nimport { D3DragEvent, drag } from 'd3-drag';\nimport { NodeModel } from '../models/node.model';\nimport { round } from '../utils/round';\nimport { FlowEntitiesService } from './flow-entities.service';\nimport { Point } from '../interfaces/point.interface';\n\ntype DragEvent = D3DragEvent<Element, unknown, unknown>\n\n@Injectable()\nexport class DraggableService {\n  private entitiesService = inject(FlowEntitiesService);\n\n  /**\n   * Enable draggable behavior for element.\n   *\n   * @param element target element for toggling draggable\n   * @param model model with data for this element\n   */\n  public enable(element: Element, model: NodeModel) {\n    const d3Element = select(element)\n\n    d3Element.call(this.getDragBehavior(model))\n  }\n\n  /**\n   * Disable draggable behavior for element.\n   *\n   * @param element target element for toggling draggable\n   * @param model model with data for this element\n   */\n  public disable(element: Element) {\n    const d3Element = select(element)\n\n    d3Element.call(this.getIgnoreDragBehavior())\n  }\n\n  /**\n   * TODO: not shure if this work, need to check\n   *\n   * @param element\n   */\n  public destroy(element: Element) {\n    select(element).on('.drag', null)\n  }\n\n  /**\n   * Node drag behavior. Updated node's coordinate according to dragging\n   *\n   * @param model\n   * @returns\n   */\n  private getDragBehavior(model: NodeModel) {\n    let dragNodes: NodeModel[] = []\n    let initialPositions: Point[] = []\n\n    return drag()\n      .on('start', (event: DragEvent) => {\n        dragNodes = this.getDragNodes(model)\n\n        initialPositions = dragNodes.map(node => ({\n          x: node.point().x - event.x,\n          y: node.point().y - event.y\n        }))\n      })\n\n      .on('drag', (event: DragEvent) => {\n        dragNodes.forEach((model, index) => {\n          let point = {\n            x: round(event.x + initialPositions[index].x),\n            y: round(event.y + initialPositions[index].y)\n          }\n\n          moveNode(model, point)\n        })\n      })\n  }\n\n  /**\n   * Specify ignoring drag behavior. It's responsible for not moving the map when user tries to drag node\n   * with disabled drag behavior\n   */\n  private getIgnoreDragBehavior() {\n    return drag()\n      .on('drag', (event: DragEvent) => {\n        (event.sourceEvent as Event).stopPropagation()\n      })\n  }\n\n  private getDragNodes(model: NodeModel) {\n    return model.selected()\n      ? this.entitiesService\n        .nodes()\n        // selected draggable nodes (with current node)\n        .filter(node => node.selected() && node.draggable())\n      // we only can move current node if it's not selected\n      : [model]\n  }\n}\n\nfunction moveNode(model: NodeModel, point: Point) {\n  const parent = model.parent()\n  // keep node in bounds of parent\n  if (parent) {\n    point.x = Math.min(parent.size().width - model.size().width, point.x)\n    point.x = Math.max(0, point.x)\n\n    point.y = Math.min(parent.size().height - model.size().height, point.y)\n    point.y = Math.max(0, point.y)\n  }\n\n  model.setPoint(point, true)\n}\n"]}