ngx-vflow 0.13.0-0 → 0.14.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 (42) hide show
  1. package/esm2022/lib/vflow/components/background/background.component.mjs +7 -16
  2. package/esm2022/lib/vflow/components/default-node/default-node.component.mjs +18 -0
  3. package/esm2022/lib/vflow/components/node/node.component.mjs +8 -7
  4. package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +23 -10
  5. package/esm2022/lib/vflow/directives/root-pointer.directive.mjs +1 -1
  6. package/esm2022/lib/vflow/directives/selectable.directive.mjs +5 -2
  7. package/esm2022/lib/vflow/models/minimap.model.mjs +7 -0
  8. package/esm2022/lib/vflow/public-components/minimap/minimap.component.mjs +119 -0
  9. package/esm2022/lib/vflow/public-components/resizable/resizable.component.mjs +124 -180
  10. package/esm2022/lib/vflow/services/draggable.service.mjs +40 -19
  11. package/esm2022/lib/vflow/services/flow-entities.service.mjs +6 -5
  12. package/esm2022/lib/vflow/services/flow-settings.service.mjs +2 -1
  13. package/esm2022/lib/vflow/services/keyboard.service.mjs +45 -0
  14. package/esm2022/lib/vflow/services/node-changes.service.mjs +6 -7
  15. package/esm2022/lib/vflow/services/selection.service.mjs +12 -5
  16. package/esm2022/lib/vflow/types/keyboard-action.type.mjs +2 -0
  17. package/esm2022/lib/vflow/utils/get-os.mjs +24 -0
  18. package/esm2022/lib/vflow/utils/transform-background.mjs +6 -0
  19. package/esm2022/lib/vflow/vflow.module.mjs +11 -3
  20. package/esm2022/public-api.mjs +3 -1
  21. package/fesm2022/ngx-vflow.mjs +423 -242
  22. package/fesm2022/ngx-vflow.mjs.map +1 -1
  23. package/lib/vflow/components/background/background.component.d.ts +3 -6
  24. package/lib/vflow/components/default-node/default-node.component.d.ts +6 -0
  25. package/lib/vflow/components/vflow/vflow.component.d.ts +6 -2
  26. package/lib/vflow/directives/root-pointer.directive.d.ts +10 -16
  27. package/lib/vflow/directives/space-point-context.directive.d.ts +1 -15
  28. package/lib/vflow/models/edge.model.d.ts +17 -1
  29. package/lib/vflow/models/minimap.model.d.ts +4 -0
  30. package/lib/vflow/public-components/minimap/minimap.component.d.ts +49 -0
  31. package/lib/vflow/public-components/resizable/resizable.component.d.ts +6 -11
  32. package/lib/vflow/services/draggable.service.d.ts +2 -0
  33. package/lib/vflow/services/flow-entities.service.d.ts +7 -5
  34. package/lib/vflow/services/flow-settings.service.d.ts +2 -0
  35. package/lib/vflow/services/keyboard.service.d.ts +11 -0
  36. package/lib/vflow/services/selection.service.d.ts +1 -0
  37. package/lib/vflow/types/keyboard-action.type.d.ts +2 -0
  38. package/lib/vflow/utils/get-os.d.ts +1 -0
  39. package/lib/vflow/utils/transform-background.d.ts +2 -0
  40. package/lib/vflow/vflow.module.d.ts +21 -19
  41. package/package.json +1 -1
  42. package/public-api.d.ts +2 -0
@@ -56,4 +56,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
56
56
  type: Directive,
57
57
  args: [{ selector: 'svg[rootPointer]' }]
58
58
  }] });
59
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"root-pointer.directive.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/directives/root-pointer.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, ElementRef, inject } from '@angular/core';\nimport { Observable, Subject, animationFrameScheduler, fromEvent, map, merge, observeOn, share, skip, tap } from 'rxjs';\nimport { Point } from '../interfaces/point.interface';\n\n@Directive({ selector: 'svg[rootPointer]' })\nexport class RootPointerDirective {\n  private host = inject<ElementRef<SVGSVGElement>>(ElementRef).nativeElement\n\n  private initialTouch$ = new Subject<TouchEvent>()\n\n  private prevTouchEvent: TouchEvent | null = null\n\n  // TODO: do not emit if mouse not down\n  public mouseMovement$ = fromEvent<MouseEvent>(this.host, 'mousemove').pipe(\n    map(event => ({\n      x: event.clientX,\n      y: event.clientY,\n      movementX: event.movementX,\n      movementY: event.movementY,\n      target: event.target,\n      originalEvent: event\n    })),\n    observeOn(animationFrameScheduler),\n    share()\n  ) satisfies Observable<Point>;\n\n  public touchMovement$ = merge(\n    this.initialTouch$,\n    fromEvent<TouchEvent>(this.host, 'touchmove')\n  ).pipe(\n    tap((event) => event.preventDefault()),\n    map((originalEvent) => {\n      const x = originalEvent.touches[0]?.clientX ?? 0\n      const y = originalEvent.touches[0]?.clientY ?? 0\n      const movementX = this.prevTouchEvent\n        ? originalEvent.touches[0].pageX - this.prevTouchEvent.touches[0].pageX\n        : 0\n      const movementY = this.prevTouchEvent\n        ? originalEvent.touches[0].pageY - this.prevTouchEvent.touches[0].pageY\n        : 0\n\n      const target = document.elementFromPoint(x, y)\n\n      return { x, y, movementX, movementY, target, originalEvent }\n    }),\n    tap((event) => this.prevTouchEvent = event.originalEvent),\n    observeOn(animationFrameScheduler),\n    share()\n  ) satisfies Observable<Point>;\n\n  public pointerMovement$ = merge(\n    this.mouseMovement$,\n    this.touchMovement$\n  )\n\n  public touchEnd$ = fromEvent<TouchEvent>(this.host, 'touchend').pipe(\n    map((originalEvent) => {\n      const x = originalEvent.changedTouches[0]?.clientX ?? 0\n      const y = originalEvent.changedTouches[0]?.clientY ?? 0\n      const target = document.elementFromPoint(x, y)\n\n      return { x, y, target, originalEvent }\n    }),\n    tap(() => this.prevTouchEvent = null),\n    share()\n  ) satisfies Observable<Point>\n\n  public mouseUp$ = fromEvent<MouseEvent>(this.host, 'mouseup').pipe(\n    map((originalEvent) => {\n      const x = originalEvent.clientX\n      const y = originalEvent.clientY\n      const target = originalEvent.target\n\n      return { x, y, target, originalEvent }\n    }),\n    share()\n  ) satisfies Observable<Point>\n\n  public documentPointerEnd$ = merge(\n    fromEvent(document, 'mouseup'),\n    fromEvent(document, 'touchend')\n  ).pipe(\n    share()\n  )\n\n\n  /**\n   * We should know when user started a touch in order to not\n   * show old touch position when connection creation is started\n   */\n  public setInitialTouch(event: TouchEvent) {\n    this.initialTouch$.next(event)\n  }\n}\n"]}
59
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"root-pointer.directive.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/directives/root-pointer.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, ElementRef, inject } from '@angular/core';\nimport { Observable, Subject, animationFrameScheduler, fromEvent, map, merge, observeOn, share, skip, tap } from 'rxjs';\nimport { Point } from '../interfaces/point.interface';\n\nexport interface PointerEvent {\n  x: number\n  y: number\n  movementX: number\n  movementY: number\n  target: Element | null\n  originalEvent: MouseEvent | TouchEvent\n}\n\n@Directive({ selector: 'svg[rootPointer]' })\nexport class RootPointerDirective {\n  private host = inject<ElementRef<SVGSVGElement>>(ElementRef).nativeElement\n\n  private initialTouch$ = new Subject<TouchEvent>()\n\n  private prevTouchEvent: TouchEvent | null = null\n\n  // TODO: do not emit if mouse not down\n  public mouseMovement$ = fromEvent<MouseEvent>(this.host, 'mousemove').pipe(\n    map(event => ({\n      x: event.clientX,\n      y: event.clientY,\n      movementX: event.movementX,\n      movementY: event.movementY,\n      target: event.target as Element | null,\n      originalEvent: event\n    })),\n    observeOn(animationFrameScheduler),\n    share()\n  ) satisfies Observable<Point>;\n\n  public touchMovement$ = merge(\n    this.initialTouch$,\n    fromEvent<TouchEvent>(this.host, 'touchmove')\n  ).pipe(\n    tap((event) => event.preventDefault()),\n    map((originalEvent) => {\n      const x = originalEvent.touches[0]?.clientX ?? 0\n      const y = originalEvent.touches[0]?.clientY ?? 0\n      const movementX = this.prevTouchEvent\n        ? originalEvent.touches[0].pageX - this.prevTouchEvent.touches[0].pageX\n        : 0\n      const movementY = this.prevTouchEvent\n        ? originalEvent.touches[0].pageY - this.prevTouchEvent.touches[0].pageY\n        : 0\n\n      const target = document.elementFromPoint(x, y)\n\n      return { x, y, movementX, movementY, target, originalEvent }\n    }),\n    tap((event) => this.prevTouchEvent = event.originalEvent),\n    observeOn(animationFrameScheduler),\n    share()\n  ) satisfies Observable<Point>;\n\n  public pointerMovement$: Observable<PointerEvent> = merge(\n    this.mouseMovement$,\n    this.touchMovement$\n  )\n\n  public touchEnd$ = fromEvent<TouchEvent>(this.host, 'touchend').pipe(\n    map((originalEvent) => {\n      const x = originalEvent.changedTouches[0]?.clientX ?? 0\n      const y = originalEvent.changedTouches[0]?.clientY ?? 0\n      const target = document.elementFromPoint(x, y)\n\n      return { x, y, target, originalEvent }\n    }),\n    tap(() => this.prevTouchEvent = null),\n    share()\n  ) satisfies Observable<Point>\n\n  public mouseUp$ = fromEvent<MouseEvent>(this.host, 'mouseup').pipe(\n    map((originalEvent) => {\n      const x = originalEvent.clientX\n      const y = originalEvent.clientY\n      const target = originalEvent.target\n\n      return { x, y, target, originalEvent }\n    }),\n    share()\n  ) satisfies Observable<Point>\n\n  public documentPointerEnd$ = merge(\n    fromEvent(document, 'mouseup'),\n    fromEvent(document, 'touchend')\n  ).pipe(\n    share()\n  )\n\n\n  /**\n   * We should know when user started a touch in order to not\n   * show old touch position when connection creation is started\n   */\n  public setInitialTouch(event: TouchEvent) {\n    this.initialTouch$.next(event)\n  }\n}\n"]}
@@ -27,7 +27,7 @@ export class SelectableDirective {
27
27
  return null;
28
28
  }
29
29
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: SelectableDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
30
- static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: SelectableDirective, selector: "[selectable]", host: { listeners: { "mousedown": "onMousedown()" } }, ngImport: i0 }); }
30
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: SelectableDirective, selector: "[selectable]", host: { listeners: { "mousedown": "onMousedown()", "touchstart": "onMousedown()" } }, ngImport: i0 }); }
31
31
  }
32
32
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: SelectableDirective, decorators: [{
33
33
  type: Directive,
@@ -35,5 +35,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
35
35
  }], propDecorators: { onMousedown: [{
36
36
  type: HostListener,
37
37
  args: ['mousedown']
38
+ }, {
39
+ type: HostListener,
40
+ args: ['touchstart']
38
41
  }] } });
39
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VsZWN0YWJsZS5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZ3gtdmZsb3ctbGliL3NyYy9saWIvdmZsb3cvZGlyZWN0aXZlcy9zZWxlY3RhYmxlLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDaEUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sK0JBQStCLENBQUM7QUFDakUsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBR2xFLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQztBQUNsRSxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQzs7QUFHeEUsTUFBTSxPQUFPLG1CQUFtQjtJQURoQztRQUVVLHdCQUFtQixHQUFHLE1BQU0sQ0FBQyxtQkFBbUIsQ0FBQyxDQUFBO1FBQ2pELHFCQUFnQixHQUFHLE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFBO1FBQzNDLGVBQVUsR0FBRyxNQUFNLENBQUMsYUFBYSxFQUFFLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxDQUFDLENBQUE7UUFDdEQsZUFBVSxHQUFHLE1BQU0sQ0FBQyxhQUFhLEVBQUUsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQTtLQW1CL0Q7SUFoQlcsV0FBVztRQUNuQixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUE7UUFDNUIsSUFBSSxNQUFNLElBQUksSUFBSSxDQUFDLG1CQUFtQixDQUFDLGtCQUFrQixFQUFFLEVBQUU7WUFDM0QsSUFBSSxDQUFDLGdCQUFnQixDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQTtTQUNyQztJQUNILENBQUM7SUFFTyxNQUFNO1FBQ1osSUFBSSxJQUFJLENBQUMsVUFBVSxFQUFFO1lBQ25CLE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQyxTQUFTLENBQUE7U0FDakM7YUFBTSxJQUFJLElBQUksQ0FBQyxVQUFVLEVBQUU7WUFDMUIsT0FBTyxJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQTtTQUM3QjtRQUVELE9BQU8sSUFBSSxDQUFBO0lBQ2IsQ0FBQzsrR0F0QlUsbUJBQW1CO21HQUFuQixtQkFBbUI7OzRGQUFuQixtQkFBbUI7a0JBRC9CLFNBQVM7bUJBQUMsRUFBRSxRQUFRLEVBQUUsY0FBYyxFQUFFOzhCQVEzQixXQUFXO3NCQURwQixZQUFZO3VCQUFDLFdBQVciLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXJlY3RpdmUsIEhvc3RMaXN0ZW5lciwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBTZWxlY3Rpb25TZXJ2aWNlIH0gZnJvbSAnLi4vc2VydmljZXMvc2VsZWN0aW9uLnNlcnZpY2UnO1xuaW1wb3J0IHsgRWRnZUNvbXBvbmVudCB9IGZyb20gJy4uL2NvbXBvbmVudHMvZWRnZS9lZGdlLmNvbXBvbmVudCc7XG5cbmltcG9ydCB7IEZsb3dFbnRpdHkgfSBmcm9tICcuLi9pbnRlcmZhY2VzL2Zsb3ctZW50aXR5LmludGVyZmFjZSc7XG5pbXBvcnQgeyBOb2RlQ29tcG9uZW50IH0gZnJvbSAnLi4vY29tcG9uZW50cy9ub2RlL25vZGUuY29tcG9uZW50JztcbmltcG9ydCB7IEZsb3dTZXR0aW5nc1NlcnZpY2UgfSBmcm9tICcuLi9zZXJ2aWNlcy9mbG93LXNldHRpbmdzLnNlcnZpY2UnO1xuXG5ARGlyZWN0aXZlKHsgc2VsZWN0b3I6ICdbc2VsZWN0YWJsZV0nIH0pXG5leHBvcnQgY2xhc3MgU2VsZWN0YWJsZURpcmVjdGl2ZSB7XG4gIHByaXZhdGUgZmxvd1NldHRpbmdzU2VydmljZSA9IGluamVjdChGbG93U2V0dGluZ3NTZXJ2aWNlKVxuICBwcml2YXRlIHNlbGVjdGlvblNlcnZpY2UgPSBpbmplY3QoU2VsZWN0aW9uU2VydmljZSlcbiAgcHJpdmF0ZSBwYXJlbnRFZGdlID0gaW5qZWN0KEVkZ2VDb21wb25lbnQsIHsgb3B0aW9uYWw6IHRydWUgfSlcbiAgcHJpdmF0ZSBwYXJlbnROb2RlID0gaW5qZWN0KE5vZGVDb21wb25lbnQsIHsgb3B0aW9uYWw6IHRydWUgfSlcblxuICBASG9zdExpc3RlbmVyKCdtb3VzZWRvd24nKVxuICBwcm90ZWN0ZWQgb25Nb3VzZWRvd24oKSB7XG4gICAgY29uc3QgZW50aXR5ID0gdGhpcy5lbnRpdHkoKVxuICAgIGlmIChlbnRpdHkgJiYgdGhpcy5mbG93U2V0dGluZ3NTZXJ2aWNlLmVudGl0aWVzU2VsZWN0YWJsZSgpKSB7XG4gICAgICB0aGlzLnNlbGVjdGlvblNlcnZpY2Uuc2VsZWN0KGVudGl0eSlcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGVudGl0eSgpOiBGbG93RW50aXR5IHwgbnVsbCB7XG4gICAgaWYgKHRoaXMucGFyZW50Tm9kZSkge1xuICAgICAgcmV0dXJuIHRoaXMucGFyZW50Tm9kZS5ub2RlTW9kZWxcbiAgICB9IGVsc2UgaWYgKHRoaXMucGFyZW50RWRnZSkge1xuICAgICAgcmV0dXJuIHRoaXMucGFyZW50RWRnZS5tb2RlbFxuICAgIH1cblxuICAgIHJldHVybiBudWxsXG4gIH1cbn1cbiJdfQ==
42
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VsZWN0YWJsZS5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZ3gtdmZsb3ctbGliL3NyYy9saWIvdmZsb3cvZGlyZWN0aXZlcy9zZWxlY3RhYmxlLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDaEUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sK0JBQStCLENBQUM7QUFDakUsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBR2xFLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQztBQUNsRSxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQzs7QUFHeEUsTUFBTSxPQUFPLG1CQUFtQjtJQURoQztRQUVVLHdCQUFtQixHQUFHLE1BQU0sQ0FBQyxtQkFBbUIsQ0FBQyxDQUFBO1FBQ2pELHFCQUFnQixHQUFHLE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFBO1FBQzNDLGVBQVUsR0FBRyxNQUFNLENBQUMsYUFBYSxFQUFFLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxDQUFDLENBQUE7UUFDdEQsZUFBVSxHQUFHLE1BQU0sQ0FBQyxhQUFhLEVBQUUsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQTtLQW9CL0Q7SUFoQlcsV0FBVztRQUNuQixNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUE7UUFDNUIsSUFBSSxNQUFNLElBQUksSUFBSSxDQUFDLG1CQUFtQixDQUFDLGtCQUFrQixFQUFFLEVBQUU7WUFDM0QsSUFBSSxDQUFDLGdCQUFnQixDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQTtTQUNyQztJQUNILENBQUM7SUFFTyxNQUFNO1FBQ1osSUFBSSxJQUFJLENBQUMsVUFBVSxFQUFFO1lBQ25CLE9BQU8sSUFBSSxDQUFDLFVBQVUsQ0FBQyxTQUFTLENBQUE7U0FDakM7YUFBTSxJQUFJLElBQUksQ0FBQyxVQUFVLEVBQUU7WUFDMUIsT0FBTyxJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQTtTQUM3QjtRQUVELE9BQU8sSUFBSSxDQUFBO0lBQ2IsQ0FBQzsrR0F2QlUsbUJBQW1CO21HQUFuQixtQkFBbUI7OzRGQUFuQixtQkFBbUI7a0JBRC9CLFNBQVM7bUJBQUMsRUFBRSxRQUFRLEVBQUUsY0FBYyxFQUFFOzhCQVMzQixXQUFXO3NCQUZwQixZQUFZO3VCQUFDLFdBQVc7O3NCQUN4QixZQUFZO3VCQUFDLFlBQVkiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXJlY3RpdmUsIEhvc3RMaXN0ZW5lciwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBTZWxlY3Rpb25TZXJ2aWNlIH0gZnJvbSAnLi4vc2VydmljZXMvc2VsZWN0aW9uLnNlcnZpY2UnO1xuaW1wb3J0IHsgRWRnZUNvbXBvbmVudCB9IGZyb20gJy4uL2NvbXBvbmVudHMvZWRnZS9lZGdlLmNvbXBvbmVudCc7XG5cbmltcG9ydCB7IEZsb3dFbnRpdHkgfSBmcm9tICcuLi9pbnRlcmZhY2VzL2Zsb3ctZW50aXR5LmludGVyZmFjZSc7XG5pbXBvcnQgeyBOb2RlQ29tcG9uZW50IH0gZnJvbSAnLi4vY29tcG9uZW50cy9ub2RlL25vZGUuY29tcG9uZW50JztcbmltcG9ydCB7IEZsb3dTZXR0aW5nc1NlcnZpY2UgfSBmcm9tICcuLi9zZXJ2aWNlcy9mbG93LXNldHRpbmdzLnNlcnZpY2UnO1xuXG5ARGlyZWN0aXZlKHsgc2VsZWN0b3I6ICdbc2VsZWN0YWJsZV0nIH0pXG5leHBvcnQgY2xhc3MgU2VsZWN0YWJsZURpcmVjdGl2ZSB7XG4gIHByaXZhdGUgZmxvd1NldHRpbmdzU2VydmljZSA9IGluamVjdChGbG93U2V0dGluZ3NTZXJ2aWNlKVxuICBwcml2YXRlIHNlbGVjdGlvblNlcnZpY2UgPSBpbmplY3QoU2VsZWN0aW9uU2VydmljZSlcbiAgcHJpdmF0ZSBwYXJlbnRFZGdlID0gaW5qZWN0KEVkZ2VDb21wb25lbnQsIHsgb3B0aW9uYWw6IHRydWUgfSlcbiAgcHJpdmF0ZSBwYXJlbnROb2RlID0gaW5qZWN0KE5vZGVDb21wb25lbnQsIHsgb3B0aW9uYWw6IHRydWUgfSlcblxuICBASG9zdExpc3RlbmVyKCdtb3VzZWRvd24nKVxuICBASG9zdExpc3RlbmVyKCd0b3VjaHN0YXJ0JylcbiAgcHJvdGVjdGVkIG9uTW91c2Vkb3duKCkge1xuICAgIGNvbnN0IGVudGl0eSA9IHRoaXMuZW50aXR5KClcbiAgICBpZiAoZW50aXR5ICYmIHRoaXMuZmxvd1NldHRpbmdzU2VydmljZS5lbnRpdGllc1NlbGVjdGFibGUoKSkge1xuICAgICAgdGhpcy5zZWxlY3Rpb25TZXJ2aWNlLnNlbGVjdChlbnRpdHkpXG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBlbnRpdHkoKTogRmxvd0VudGl0eSB8IG51bGwge1xuICAgIGlmICh0aGlzLnBhcmVudE5vZGUpIHtcbiAgICAgIHJldHVybiB0aGlzLnBhcmVudE5vZGUubm9kZU1vZGVsXG4gICAgfSBlbHNlIGlmICh0aGlzLnBhcmVudEVkZ2UpIHtcbiAgICAgIHJldHVybiB0aGlzLnBhcmVudEVkZ2UubW9kZWxcbiAgICB9XG5cbiAgICByZXR1cm4gbnVsbFxuICB9XG59XG4iXX0=
@@ -0,0 +1,7 @@
1
+ import { signal } from "@angular/core";
2
+ export class MinimapModel {
3
+ constructor() {
4
+ this.template = signal(null);
5
+ }
6
+ }
7
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWluaW1hcC5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9tb2RlbHMvbWluaW1hcC5tb2RlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsTUFBTSxFQUFlLE1BQU0sZUFBZSxDQUFDO0FBRXBELE1BQU0sT0FBTyxZQUFZO0lBQXpCO1FBQ1MsYUFBUSxHQUFHLE1BQU0sQ0FBOEIsSUFBSSxDQUFDLENBQUE7SUFDN0QsQ0FBQztDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgc2lnbmFsLCBUZW1wbGF0ZVJlZiB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XG5cbmV4cG9ydCBjbGFzcyBNaW5pbWFwTW9kZWwge1xuICBwdWJsaWMgdGVtcGxhdGUgPSBzaWduYWw8VGVtcGxhdGVSZWY8dW5rbm93bj4gfCBudWxsPihudWxsKVxufVxuIl19
@@ -0,0 +1,119 @@
1
+ import { Component, computed, inject, Injector, Input, signal, ViewChild } from '@angular/core';
2
+ import { FlowEntitiesService } from '../../services/flow-entities.service';
3
+ import { MinimapModel } from '../../models/minimap.model';
4
+ import { FlowSettingsService } from '../../services/flow-settings.service';
5
+ import { getViewportForBounds } from '../../utils/viewport';
6
+ import { getNodesBounds } from '../../utils/nodes';
7
+ import { ViewportService } from '../../services/viewport.service';
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "@angular/common";
10
+ import * as i2 from "../../components/default-node/default-node.component";
11
+ export class MiniMapComponent {
12
+ constructor() {
13
+ this.entitiesService = inject(FlowEntitiesService);
14
+ this.flowSettingsService = inject(FlowSettingsService);
15
+ this.viewportService = inject(ViewportService);
16
+ this.injector = inject(Injector);
17
+ /**
18
+ * The color outside the viewport (invisible area)
19
+ */
20
+ this.maskColor = `rgba(215, 215, 215, 0.6)`;
21
+ /**
22
+ * The minimap stroke color
23
+ */
24
+ this.strokeColor = `rgb(200, 200, 200)`;
25
+ this.minimapOffset = 10;
26
+ this.minimapScale = computed(() => {
27
+ if (this.scaleOnHoverSignal()) {
28
+ return this.hovered() ? 0.4 : 0.2;
29
+ }
30
+ return 0.2;
31
+ });
32
+ this.viewportColor = computed(() => this.flowSettingsService.background().color ?? '#fff');
33
+ this.hovered = signal(false);
34
+ this.minimapPoint = computed(() => {
35
+ switch (this.minimapPosition()) {
36
+ case 'top-left':
37
+ return { x: this.minimapOffset, y: this.minimapOffset };
38
+ case 'top-right':
39
+ return {
40
+ x: this.flowSettingsService.computedFlowWidth() - this.minimapWidth() - this.minimapOffset,
41
+ y: this.minimapOffset
42
+ };
43
+ case 'bottom-left':
44
+ return {
45
+ x: this.minimapOffset,
46
+ y: this.flowSettingsService.computedFlowHeight() - this.minimapHeight() - this.minimapOffset
47
+ };
48
+ case 'bottom-right':
49
+ return {
50
+ x: this.flowSettingsService.computedFlowWidth() - this.minimapWidth() - this.minimapOffset,
51
+ y: this.flowSettingsService.computedFlowHeight() - this.minimapHeight() - this.minimapOffset
52
+ };
53
+ }
54
+ });
55
+ this.minimapWidth = computed(() => this.flowSettingsService.computedFlowWidth() * this.minimapScale());
56
+ this.minimapHeight = computed(() => this.flowSettingsService.computedFlowHeight() * this.minimapScale());
57
+ this.viewportTransform = computed(() => {
58
+ const viewport = this.viewportService.readableViewport();
59
+ let scale = 1 / viewport.zoom;
60
+ let x = -(viewport.x * this.minimapScale()) * scale;
61
+ x /= this.minimapScale();
62
+ let y = -(viewport.y * this.minimapScale()) * scale;
63
+ y /= this.minimapScale();
64
+ scale /= this.minimapScale();
65
+ return `translate(${x}, ${y}) scale(${scale})`;
66
+ });
67
+ this.boundsViewport = computed(() => {
68
+ const nodes = this.entitiesService.nodes();
69
+ return getViewportForBounds(getNodesBounds(nodes), this.flowSettingsService.computedFlowWidth(), this.flowSettingsService.computedFlowHeight(), -Infinity, 1.5, 0);
70
+ });
71
+ this.minimapTransform = computed(() => {
72
+ const vport = this.boundsViewport();
73
+ const x = vport.x * this.minimapScale();
74
+ const y = vport.y * this.minimapScale();
75
+ const scale = vport.zoom * this.minimapScale();
76
+ return `translate(${x} ${y}) scale(${scale})`;
77
+ });
78
+ this.minimapPosition = signal('bottom-right');
79
+ this.scaleOnHoverSignal = signal(false);
80
+ }
81
+ /**
82
+ * The corner of the flow where to render a mini-map
83
+ */
84
+ set position(value) {
85
+ this.minimapPosition.set(value);
86
+ }
87
+ /**
88
+ * Make a minimap bigger on hover
89
+ */
90
+ set scaleOnHover(value) {
91
+ this.scaleOnHoverSignal.set(value);
92
+ }
93
+ ngOnInit() {
94
+ const model = new MinimapModel();
95
+ model.template.set(this.minimap);
96
+ this.entitiesService.minimap.set(model);
97
+ }
98
+ trackNodes(idx, { node }) {
99
+ return node;
100
+ }
101
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: MiniMapComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
102
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: MiniMapComponent, selector: "mini-map", inputs: { position: "position", maskColor: "maskColor", strokeColor: "strokeColor", scaleOnHover: "scaleOnHover" }, viewQueries: [{ propertyName: "minimap", first: true, predicate: ["minimap"], descendants: true, static: true }], ngImport: i0, template: "<ng-template #minimap>\n <svg:rect\n [attr.x]=\"minimapPoint().x\"\n [attr.y]=\"minimapPoint().y\"\n [attr.width]=\"minimapWidth()\"\n [attr.height]=\"minimapHeight()\"\n [attr.stroke]=\"strokeColor\"\n fill=\"none\"\n />\n\n <svg:svg\n [attr.x]=\"minimapPoint().x\"\n [attr.y]=\"minimapPoint().y\"\n [attr.width]=\"minimapWidth()\"\n [attr.height]=\"minimapHeight()\"\n (mouseover)=\"hovered.set(true)\"\n (mouseleave)=\"hovered.set(false)\"\n >\n <svg:rect\n [attr.width]=\"minimapWidth()\"\n [attr.height]=\"minimapHeight()\"\n [attr.fill]=\"maskColor\"\n />\n\n <svg:g [attr.transform]=\"minimapTransform()\">\n <svg:rect\n [attr.fill]=\"viewportColor()\"\n [attr.transform]=\"viewportTransform()\"\n [attr.width]=\"minimapWidth()\"\n [attr.height]=\"minimapHeight()\"\n />\n\n <ng-container\n *ngFor=\"let model of entitiesService.nodes(); trackBy: trackNodes\"\n >\n <svg:foreignObject\n *ngIf=\"model.node.type === 'default' || model.node.type === 'html-template' || model.isComponentType\"\n [attr.transform]=\"model.pointTransform()\"\n [attr.width]=\"model.size().width\"\n [attr.height]=\"model.size().height\"\n >\n <default-node\n [selected]=\"model.selected()\"\n [style.width.px]=\"model.size().width\"\n [style.height.px]=\"model.size().height\"\n [style.max-width.px]=\"model.size().width\"\n [style.max-height.px]=\"model.size().height\"\n >\n <div [outerHTML]=\"model.text()\"></div>\n </default-node>\n </svg:foreignObject>\n\n <svg:rect\n *ngIf=\"model.node.type === 'default-group' || model.node.type === 'template-group'\"\n class=\"default-group-node\"\n rx=\"5\"\n ry=\"5\"\n [attr.transform]=\"model.pointTransform()\"\n [class.default-group-node_selected]=\"model.selected()\"\n [attr.width]=\"model.size().width\"\n [attr.height]=\"model.size().height\"\n [style.stroke]=\"model.color()\"\n [style.fill]=\"model.color()\"\n />\n\n </ng-container>\n </svg:g>\n </svg:svg>\n</ng-template>\n", styles: [".default-group-node{stroke-width:1.5px;fill-opacity:.05}.default-group-node_selected{stroke-width:2px}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.DefaultNodeComponent, selector: "default-node", inputs: ["selected"] }] }); }
103
+ }
104
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: MiniMapComponent, decorators: [{
105
+ type: Component,
106
+ args: [{ selector: 'mini-map', template: "<ng-template #minimap>\n <svg:rect\n [attr.x]=\"minimapPoint().x\"\n [attr.y]=\"minimapPoint().y\"\n [attr.width]=\"minimapWidth()\"\n [attr.height]=\"minimapHeight()\"\n [attr.stroke]=\"strokeColor\"\n fill=\"none\"\n />\n\n <svg:svg\n [attr.x]=\"minimapPoint().x\"\n [attr.y]=\"minimapPoint().y\"\n [attr.width]=\"minimapWidth()\"\n [attr.height]=\"minimapHeight()\"\n (mouseover)=\"hovered.set(true)\"\n (mouseleave)=\"hovered.set(false)\"\n >\n <svg:rect\n [attr.width]=\"minimapWidth()\"\n [attr.height]=\"minimapHeight()\"\n [attr.fill]=\"maskColor\"\n />\n\n <svg:g [attr.transform]=\"minimapTransform()\">\n <svg:rect\n [attr.fill]=\"viewportColor()\"\n [attr.transform]=\"viewportTransform()\"\n [attr.width]=\"minimapWidth()\"\n [attr.height]=\"minimapHeight()\"\n />\n\n <ng-container\n *ngFor=\"let model of entitiesService.nodes(); trackBy: trackNodes\"\n >\n <svg:foreignObject\n *ngIf=\"model.node.type === 'default' || model.node.type === 'html-template' || model.isComponentType\"\n [attr.transform]=\"model.pointTransform()\"\n [attr.width]=\"model.size().width\"\n [attr.height]=\"model.size().height\"\n >\n <default-node\n [selected]=\"model.selected()\"\n [style.width.px]=\"model.size().width\"\n [style.height.px]=\"model.size().height\"\n [style.max-width.px]=\"model.size().width\"\n [style.max-height.px]=\"model.size().height\"\n >\n <div [outerHTML]=\"model.text()\"></div>\n </default-node>\n </svg:foreignObject>\n\n <svg:rect\n *ngIf=\"model.node.type === 'default-group' || model.node.type === 'template-group'\"\n class=\"default-group-node\"\n rx=\"5\"\n ry=\"5\"\n [attr.transform]=\"model.pointTransform()\"\n [class.default-group-node_selected]=\"model.selected()\"\n [attr.width]=\"model.size().width\"\n [attr.height]=\"model.size().height\"\n [style.stroke]=\"model.color()\"\n [style.fill]=\"model.color()\"\n />\n\n </ng-container>\n </svg:g>\n </svg:svg>\n</ng-template>\n", styles: [".default-group-node{stroke-width:1.5px;fill-opacity:.05}.default-group-node_selected{stroke-width:2px}\n"] }]
107
+ }], propDecorators: { position: [{
108
+ type: Input
109
+ }], maskColor: [{
110
+ type: Input
111
+ }], strokeColor: [{
112
+ type: Input
113
+ }], scaleOnHover: [{
114
+ type: Input
115
+ }], minimap: [{
116
+ type: ViewChild,
117
+ args: ['minimap', { static: true }]
118
+ }] } });
119
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"minimap.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/minimap/minimap.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/public-components/minimap/minimap.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, computed, inject, Injector, Input, OnInit, signal, TemplateRef, ViewChild } from '@angular/core';\nimport { FlowEntitiesService } from '../../services/flow-entities.service';\nimport { MinimapModel } from '../../models/minimap.model';\nimport { NodeModel } from '../../models/node.model';\nimport { FlowSettingsService } from '../../services/flow-settings.service';\nimport { getViewportForBounds } from '../../utils/viewport';\nimport { getNodesBounds } from '../../utils/nodes';\nimport { ViewportService } from '../../services/viewport.service';\n\nexport type MiniMapPosition = 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right'\n\n@Component({\n  selector: 'mini-map',\n  templateUrl: './minimap.component.html',\n  styleUrls: [`./minimap.component.scss`]\n})\nexport class MiniMapComponent implements OnInit {\n  protected entitiesService = inject(FlowEntitiesService)\n  protected flowSettingsService = inject(FlowSettingsService)\n  protected viewportService = inject(ViewportService)\n  protected injector = inject(Injector)\n\n  /**\n   * The corner of the flow where to render a mini-map\n   */\n  @Input()\n  public set position(value: MiniMapPosition) {\n    this.minimapPosition.set(value)\n  }\n\n  /**\n   * The color outside the viewport (invisible area)\n   */\n  @Input()\n  public maskColor = `rgba(215, 215, 215, 0.6)`\n\n  /**\n   * The minimap stroke color\n   */\n  @Input()\n  public strokeColor = `rgb(200, 200, 200)`\n\n  /**\n   * Make a minimap bigger on hover\n   */\n  @Input()\n  public set scaleOnHover(value: boolean) {\n    this.scaleOnHoverSignal.set(value)\n  }\n\n  @ViewChild('minimap', { static: true })\n  private minimap!: TemplateRef<unknown>\n\n  private readonly minimapOffset = 10\n\n  private readonly minimapScale = computed(() => {\n    if (this.scaleOnHoverSignal()) {\n      return this.hovered() ? 0.4 : 0.2\n    }\n\n    return 0.2\n  })\n\n  protected viewportColor = computed(() => this.flowSettingsService.background().color ?? '#fff')\n\n  protected hovered = signal(false)\n\n  protected minimapPoint = computed(() => {\n    switch (this.minimapPosition()) {\n      case 'top-left':\n        return { x: this.minimapOffset, y: this.minimapOffset }\n      case 'top-right':\n        return {\n          x: this.flowSettingsService.computedFlowWidth() - this.minimapWidth() - this.minimapOffset,\n          y: this.minimapOffset\n        }\n      case 'bottom-left':\n        return {\n          x: this.minimapOffset,\n          y: this.flowSettingsService.computedFlowHeight() - this.minimapHeight() - this.minimapOffset\n        }\n      case 'bottom-right':\n        return {\n          x: this.flowSettingsService.computedFlowWidth() - this.minimapWidth() - this.minimapOffset,\n          y: this.flowSettingsService.computedFlowHeight() - this.minimapHeight() - this.minimapOffset\n        }\n    }\n  })\n\n  protected minimapWidth = computed(() =>\n    this.flowSettingsService.computedFlowWidth() * this.minimapScale()\n  )\n  protected minimapHeight = computed(() =>\n    this.flowSettingsService.computedFlowHeight() * this.minimapScale()\n  )\n\n  protected viewportTransform = computed(() => {\n    const viewport = this.viewportService.readableViewport();\n    let scale = 1 / viewport.zoom\n\n    let x = -(viewport.x * this.minimapScale()) * scale\n    x /= this.minimapScale()\n\n    let y = -(viewport.y * this.minimapScale()) * scale\n    y /= this.minimapScale()\n\n    scale /= this.minimapScale()\n\n    return `translate(${x}, ${y}) scale(${scale})`;\n  });\n\n\n  protected boundsViewport = computed(() => {\n    const nodes = this.entitiesService.nodes()\n\n    return getViewportForBounds(\n      getNodesBounds(nodes),\n      this.flowSettingsService.computedFlowWidth(),\n      this.flowSettingsService.computedFlowHeight(),\n      -Infinity,\n      1.5,\n      0\n    )\n\n  })\n\n  protected minimapTransform = computed(() => {\n    const vport = this.boundsViewport()\n\n    const x = vport.x * this.minimapScale()\n    const y = vport.y * this.minimapScale()\n    const scale = vport.zoom * this.minimapScale()\n\n    return `translate(${x} ${y}) scale(${scale})`\n  })\n\n  private minimapPosition = signal<MiniMapPosition>('bottom-right')\n\n  private scaleOnHoverSignal = signal(false)\n\n  public ngOnInit(): void {\n    const model = new MinimapModel()\n    model.template.set(this.minimap)\n\n    this.entitiesService.minimap.set(model)\n  }\n\n  protected trackNodes(idx: number, { node }: NodeModel) {\n    return node\n  }\n}\n","<ng-template #minimap>\n  <svg:rect\n    [attr.x]=\"minimapPoint().x\"\n    [attr.y]=\"minimapPoint().y\"\n    [attr.width]=\"minimapWidth()\"\n    [attr.height]=\"minimapHeight()\"\n    [attr.stroke]=\"strokeColor\"\n    fill=\"none\"\n  />\n\n  <svg:svg\n    [attr.x]=\"minimapPoint().x\"\n    [attr.y]=\"minimapPoint().y\"\n    [attr.width]=\"minimapWidth()\"\n    [attr.height]=\"minimapHeight()\"\n    (mouseover)=\"hovered.set(true)\"\n    (mouseleave)=\"hovered.set(false)\"\n  >\n    <svg:rect\n      [attr.width]=\"minimapWidth()\"\n      [attr.height]=\"minimapHeight()\"\n      [attr.fill]=\"maskColor\"\n    />\n\n    <svg:g [attr.transform]=\"minimapTransform()\">\n      <svg:rect\n        [attr.fill]=\"viewportColor()\"\n        [attr.transform]=\"viewportTransform()\"\n        [attr.width]=\"minimapWidth()\"\n        [attr.height]=\"minimapHeight()\"\n      />\n\n      <ng-container\n        *ngFor=\"let model of entitiesService.nodes(); trackBy: trackNodes\"\n      >\n        <svg:foreignObject\n          *ngIf=\"model.node.type === 'default' || model.node.type === 'html-template' || model.isComponentType\"\n          [attr.transform]=\"model.pointTransform()\"\n          [attr.width]=\"model.size().width\"\n          [attr.height]=\"model.size().height\"\n        >\n          <default-node\n            [selected]=\"model.selected()\"\n            [style.width.px]=\"model.size().width\"\n            [style.height.px]=\"model.size().height\"\n            [style.max-width.px]=\"model.size().width\"\n            [style.max-height.px]=\"model.size().height\"\n          >\n            <div [outerHTML]=\"model.text()\"></div>\n          </default-node>\n        </svg:foreignObject>\n\n        <svg:rect\n          *ngIf=\"model.node.type === 'default-group' || model.node.type === 'template-group'\"\n          class=\"default-group-node\"\n          rx=\"5\"\n          ry=\"5\"\n          [attr.transform]=\"model.pointTransform()\"\n          [class.default-group-node_selected]=\"model.selected()\"\n          [attr.width]=\"model.size().width\"\n          [attr.height]=\"model.size().height\"\n          [style.stroke]=\"model.color()\"\n          [style.fill]=\"model.color()\"\n        />\n\n      </ng-container>\n    </svg:g>\n  </svg:svg>\n</ng-template>\n"]}