ngx-vflow 0.14.0 → 0.15.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 (35) hide show
  1. package/esm2022/lib/vflow/components/custom-node-base/custom-node-base.component.mjs +5 -7
  2. package/esm2022/lib/vflow/components/defs/defs.component.mjs +1 -1
  3. package/esm2022/lib/vflow/components/handle/handle.component.mjs +16 -14
  4. package/esm2022/lib/vflow/components/node/node.component.mjs +8 -3
  5. package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +8 -5
  6. package/esm2022/lib/vflow/directives/drag-handle.directive.mjs +27 -0
  7. package/esm2022/lib/vflow/directives/map-context.directive.mjs +24 -17
  8. package/esm2022/lib/vflow/math/edge-path/bezier-path.mjs +14 -17
  9. package/esm2022/lib/vflow/math/edge-path/straigh-path.mjs +2 -6
  10. package/esm2022/lib/vflow/models/handle.model.mjs +3 -2
  11. package/esm2022/lib/vflow/models/node.model.mjs +2 -1
  12. package/esm2022/lib/vflow/models/toolbar.model.mjs +36 -0
  13. package/esm2022/lib/vflow/public-components/minimap/minimap.component.mjs +1 -1
  14. package/esm2022/lib/vflow/public-components/node-toolbar/node-toolbar.component.mjs +66 -0
  15. package/esm2022/lib/vflow/services/draggable.service.mjs +13 -15
  16. package/esm2022/lib/vflow/services/overlays.service.mjs +34 -0
  17. package/esm2022/lib/vflow/testing-utils/provide-custom-node-mocks.mjs +67 -0
  18. package/esm2022/lib/vflow/vflow.module.mjs +21 -9
  19. package/esm2022/public-api.mjs +5 -1
  20. package/fesm2022/ngx-vflow.mjs +308 -87
  21. package/fesm2022/ngx-vflow.mjs.map +1 -1
  22. package/lib/vflow/components/handle/handle.component.d.ts +3 -3
  23. package/lib/vflow/components/node/node.component.d.ts +2 -0
  24. package/lib/vflow/directives/drag-handle.directive.d.ts +8 -0
  25. package/lib/vflow/directives/map-context.directive.d.ts +3 -2
  26. package/lib/vflow/models/handle.model.d.ts +1 -0
  27. package/lib/vflow/models/node.model.d.ts +1 -0
  28. package/lib/vflow/models/toolbar.model.d.ts +19 -0
  29. package/lib/vflow/public-components/node-toolbar/node-toolbar.component.d.ts +22 -0
  30. package/lib/vflow/services/draggable.service.d.ts +0 -5
  31. package/lib/vflow/services/overlays.service.d.ts +11 -0
  32. package/lib/vflow/testing-utils/provide-custom-node-mocks.d.ts +2 -0
  33. package/lib/vflow/vflow.module.d.ts +14 -12
  34. package/package.json +1 -3
  35. package/public-api.d.ts +3 -0
@@ -53,30 +53,37 @@ export class MapContextDirective {
53
53
  this.viewportService.readableViewport.set(mapTransformToViewportState(transform));
54
54
  this.zoomableSelection.attr('transform', transform.toString());
55
55
  };
56
+ this.handleZoomStart = ({ transform }) => {
57
+ this.viewportForSelection = {
58
+ start: mapTransformToViewportState(transform)
59
+ };
60
+ };
61
+ this.handleZoomEnd = ({ transform, sourceEvent }) => {
62
+ this.viewportForSelection = {
63
+ ...this.viewportForSelection,
64
+ end: mapTransformToViewportState(transform),
65
+ target: evTarget(sourceEvent)
66
+ };
67
+ this.selectionService.setViewport(this.viewportForSelection);
68
+ };
69
+ this.filterCondition = (event) => {
70
+ if (event.type === 'mousedown' || event.type === 'touchstart') {
71
+ return event.target.closest('.vflow-node') === null;
72
+ }
73
+ return true;
74
+ };
56
75
  }
57
76
  ngOnInit() {
58
77
  this.zoomBehavior = zoom()
59
78
  .scaleExtent([this.flowSettingsService.minZoom(), this.flowSettingsService.maxZoom()])
60
- .on('start', (event) => this.onD3zoomStart(event))
61
- .on('zoom', (event) => this.handleZoom(event))
62
- .on('end', (event) => this.onD3zoomEnd(event));
79
+ .filter(this.filterCondition)
80
+ .on('start', this.handleZoomStart)
81
+ .on('zoom', this.handleZoom)
82
+ .on('end', this.handleZoomEnd);
63
83
  this.rootSvgSelection
64
84
  .call(this.zoomBehavior)
65
85
  .on('dblclick.zoom', null);
66
86
  }
67
- onD3zoomStart({ transform }) {
68
- this.viewportForSelection = {
69
- start: mapTransformToViewportState(transform)
70
- };
71
- }
72
- onD3zoomEnd({ transform, sourceEvent }) {
73
- this.viewportForSelection = {
74
- ...this.viewportForSelection,
75
- end: mapTransformToViewportState(transform),
76
- target: evTarget(sourceEvent)
77
- };
78
- this.selectionService.setViewport(this.viewportForSelection);
79
- }
80
87
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: MapContextDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
81
88
  static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: MapContextDirective, selector: "g[mapContext]", ngImport: i0 }); }
82
89
  }
@@ -91,4 +98,4 @@ const evTarget = (anyEvent) => {
91
98
  }
92
99
  return undefined;
93
100
  };
94
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"map-context.directive.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/directives/map-context.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, ElementRef, Input, OnInit, effect, inject, untracked } from '@angular/core';\nimport { select } from 'd3-selection';\nimport { D3ZoomEvent, ZoomBehavior, ZoomTransform, zoom, zoomIdentity } from 'd3-zoom';\nimport { ViewportService } from '../services/viewport.service';\nimport { isDefined } from '../utils/is-defined';\nimport { RootSvgReferenceDirective } from './reference.directive';\nimport { ViewportState } from '../interfaces/viewport.interface';\nimport { SelectionService, ViewportForSelection } from '../services/selection.service';\nimport { FlowSettingsService } from '../services/flow-settings.service';\n\n@Directive({ selector: 'g[mapContext]' })\nexport class MapContextDirective implements OnInit {\n  protected rootSvg = inject(RootSvgReferenceDirective).element\n  protected host = inject<ElementRef<SVGGElement>>(ElementRef).nativeElement\n  protected selectionService = inject(SelectionService)\n  protected viewportService = inject(ViewportService)\n  protected flowSettingsService = inject(FlowSettingsService);\n\n  protected rootSvgSelection = select(this.rootSvg)\n  protected zoomableSelection = select(this.host)\n\n  protected viewportForSelection: Partial<ViewportForSelection> = {}\n\n  // under the hood this effect triggers handleZoom, so error throws without this flag\n  protected manualViewportChangeEffect = effect(() => {\n    const viewport = this.viewportService.writableViewport()\n    const state = viewport.state\n\n    if (viewport.changeType === 'initial') {\n      return\n    }\n\n    // If only zoom provided\n    if (isDefined(state.zoom) && (!isDefined(state.x) && !isDefined(state.y))) {\n      this.rootSvgSelection\n        .transition().duration(viewport.duration)\n        .call(this.zoomBehavior.scaleTo, state.zoom)\n\n      return\n    }\n\n    // If only pan provided\n    if ((isDefined(state.x) && isDefined(state.y)) && !isDefined(state.zoom)) {\n      // remain same zoom value\n      const zoom = untracked(this.viewportService.readableViewport).zoom\n\n      this.rootSvgSelection\n        .transition().duration(viewport.duration)\n        .call(\n          this.zoomBehavior.transform,\n          zoomIdentity.translate(state.x, state.y).scale(zoom)\n        )\n\n      return\n    }\n\n    // If whole viewort state provided\n    if (isDefined(state.x) && isDefined(state.y) && isDefined(state.zoom)) {\n      this.rootSvgSelection\n        .transition().duration(viewport.duration)\n        .call(\n          this.zoomBehavior.transform,\n          zoomIdentity.translate(state.x, state.y).scale(state.zoom),\n        )\n\n      return\n    }\n  }, { allowSignalWrites: true })\n\n  protected zoomBehavior!: ZoomBehavior<SVGSVGElement, unknown>;\n\n  public ngOnInit(): void {\n    this.zoomBehavior = zoom<SVGSVGElement, unknown>()\n      .scaleExtent([this.flowSettingsService.minZoom(), this.flowSettingsService.maxZoom()])\n      .on('start', (event: ZoomEvent) => this.onD3zoomStart(event))\n      .on('zoom', (event: ZoomEvent) => this.handleZoom(event))\n      .on('end', (event: ZoomEvent) => this.onD3zoomEnd(event))\n\n    this.rootSvgSelection\n      .call(this.zoomBehavior)\n      .on('dblclick.zoom', null)\n  }\n\n  private handleZoom = ({ transform }: ZoomEvent) => {\n    // update public signal for user to read\n    this.viewportService.readableViewport.set(mapTransformToViewportState(transform))\n\n    this.zoomableSelection.attr('transform', transform.toString())\n  }\n\n  private onD3zoomStart({ transform }: ZoomEvent) {\n    this.viewportForSelection = {\n      start: mapTransformToViewportState(transform)\n    }\n  }\n\n  private onD3zoomEnd({ transform, sourceEvent }: ZoomEvent) {\n    this.viewportForSelection = {\n      ...this.viewportForSelection,\n      end: mapTransformToViewportState(transform),\n      target: evTarget(sourceEvent)\n    }\n\n    this.selectionService.setViewport(\n      this.viewportForSelection as ViewportForSelection\n    )\n  }\n}\n\nconst mapTransformToViewportState = (transform: ZoomTransform): ViewportState =>\n  ({ zoom: transform.k, x: transform.x, y: transform.y })\n\nconst evTarget = (anyEvent: any): Element | undefined => {\n  if (anyEvent instanceof Event && anyEvent.target instanceof Element) {\n    return anyEvent.target\n  }\n\n  return undefined;\n}\n\ndeclare module 'd3-selection' {\n  interface Selection<GElement extends BaseType, Datum, PElement extends BaseType, PDatum> {\n    transition(): Selection<GElement, Datum, PElement, PDatum>\n    duration(duration: number): Selection<GElement, Datum, PElement, PDatum>\n  }\n}\n\ntype ZoomEvent = D3ZoomEvent<SVGSVGElement, unknown>\n"]}
101
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"map-context.directive.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/directives/map-context.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, ElementRef, Input, OnInit, effect, inject, untracked } from '@angular/core';\nimport { select } from 'd3-selection';\nimport { D3ZoomEvent, ZoomBehavior, ZoomTransform, zoom, zoomIdentity } from 'd3-zoom';\nimport { ViewportService } from '../services/viewport.service';\nimport { isDefined } from '../utils/is-defined';\nimport { RootSvgReferenceDirective } from './reference.directive';\nimport { ViewportState } from '../interfaces/viewport.interface';\nimport { SelectionService, ViewportForSelection } from '../services/selection.service';\nimport { FlowSettingsService } from '../services/flow-settings.service';\n\n@Directive({ selector: 'g[mapContext]' })\nexport class MapContextDirective implements OnInit {\n  protected rootSvg = inject(RootSvgReferenceDirective).element\n  protected host = inject<ElementRef<SVGGElement>>(ElementRef).nativeElement\n  protected selectionService = inject(SelectionService)\n  protected viewportService = inject(ViewportService)\n  protected flowSettingsService = inject(FlowSettingsService);\n\n  protected rootSvgSelection = select(this.rootSvg)\n  protected zoomableSelection = select(this.host)\n\n  protected viewportForSelection: Partial<ViewportForSelection> = {}\n\n  // under the hood this effect triggers handleZoom, so error throws without this flag\n  protected manualViewportChangeEffect = effect(() => {\n    const viewport = this.viewportService.writableViewport()\n    const state = viewport.state\n\n    if (viewport.changeType === 'initial') {\n      return\n    }\n\n    // If only zoom provided\n    if (isDefined(state.zoom) && (!isDefined(state.x) && !isDefined(state.y))) {\n      this.rootSvgSelection\n        .transition().duration(viewport.duration)\n        .call(this.zoomBehavior.scaleTo, state.zoom)\n\n      return\n    }\n\n    // If only pan provided\n    if ((isDefined(state.x) && isDefined(state.y)) && !isDefined(state.zoom)) {\n      // remain same zoom value\n      const zoom = untracked(this.viewportService.readableViewport).zoom\n\n      this.rootSvgSelection\n        .transition().duration(viewport.duration)\n        .call(\n          this.zoomBehavior.transform,\n          zoomIdentity.translate(state.x, state.y).scale(zoom)\n        )\n\n      return\n    }\n\n    // If whole viewort state provided\n    if (isDefined(state.x) && isDefined(state.y) && isDefined(state.zoom)) {\n      this.rootSvgSelection\n        .transition().duration(viewport.duration)\n        .call(\n          this.zoomBehavior.transform,\n          zoomIdentity.translate(state.x, state.y).scale(state.zoom),\n        )\n\n      return\n    }\n  }, { allowSignalWrites: true })\n\n  protected zoomBehavior!: ZoomBehavior<SVGSVGElement, unknown>;\n\n  public ngOnInit(): void {\n    this.zoomBehavior = zoom<SVGSVGElement, unknown>()\n      .scaleExtent([this.flowSettingsService.minZoom(), this.flowSettingsService.maxZoom()])\n      .filter(this.filterCondition)\n      .on('start', this.handleZoomStart)\n      .on('zoom', this.handleZoom)\n      .on('end', this.handleZoomEnd)\n\n    this.rootSvgSelection\n      .call(this.zoomBehavior)\n      .on('dblclick.zoom', null)\n  }\n\n  private handleZoom = ({ transform }: ZoomEvent) => {\n    // update public signal for user to read\n    this.viewportService.readableViewport.set(mapTransformToViewportState(transform))\n\n    this.zoomableSelection.attr('transform', transform.toString())\n  }\n\n  private handleZoomStart = ({ transform }: ZoomEvent) => {\n    this.viewportForSelection = {\n      start: mapTransformToViewportState(transform)\n    }\n  }\n\n  private handleZoomEnd = ({ transform, sourceEvent }: ZoomEvent) => {\n    this.viewportForSelection = {\n      ...this.viewportForSelection,\n      end: mapTransformToViewportState(transform),\n      target: evTarget(sourceEvent)\n    }\n\n    this.selectionService.setViewport(\n      this.viewportForSelection as ViewportForSelection\n    )\n  }\n\n  private filterCondition = (event: Event) => {\n    if (event.type === 'mousedown' || event.type === 'touchstart') {\n      return (event.target as Element).closest('.vflow-node') === null\n    }\n\n    return true\n  }\n}\n\nconst mapTransformToViewportState = (transform: ZoomTransform): ViewportState =>\n  ({ zoom: transform.k, x: transform.x, y: transform.y })\n\nconst evTarget = (anyEvent: any): Element | undefined => {\n  if (anyEvent instanceof Event && anyEvent.target instanceof Element) {\n    return anyEvent.target\n  }\n\n  return undefined;\n}\n\ndeclare module 'd3-selection' {\n  interface Selection<GElement extends BaseType, Datum, PElement extends BaseType, PDatum> {\n    transition(): Selection<GElement, Datum, PElement, PDatum>\n    duration(duration: number): Selection<GElement, Datum, PElement, PDatum>\n  }\n}\n\ntype ZoomEvent = D3ZoomEvent<SVGSVGElement, unknown>\n"]}
@@ -1,13 +1,10 @@
1
- import { path as d3Path } from 'd3-path';
2
1
  import { getPointOnLineByRatio } from '../point-on-line-by-ratio';
3
2
  export function bezierPath(source, target, sourcePosition, targetPosition, usingPoints = [false, false, false]) {
4
- const path = d3Path();
5
- path.moveTo(source.x, source.y);
6
3
  const distanceVector = { x: source.x - target.x, y: source.y - target.y };
7
- const firstControl = calcControlPoint(source, sourcePosition, distanceVector);
8
- const secondControl = calcControlPoint(target, targetPosition, distanceVector);
9
- path.bezierCurveTo(firstControl.x, firstControl.y, secondControl.x, secondControl.y, target.x, target.y);
10
- return getPathData(path, source, target, firstControl, secondControl, usingPoints);
4
+ const sourceControl = calcControlPoint(source, sourcePosition, distanceVector);
5
+ const targetControl = calcControlPoint(target, targetPosition, distanceVector);
6
+ const path = `M${source.x},${source.y} C${sourceControl.x},${sourceControl.y} ${targetControl.x},${targetControl.y} ${target.x},${target.y}`;
7
+ return getPathData(path, source, target, sourceControl, targetControl, usingPoints);
11
8
  }
12
9
  /**
13
10
  * Calculate control point based on provided point
@@ -47,20 +44,20 @@ function calcControlPoint(point, pointPosition, distanceVector) {
47
44
  y: point.y - factorPoint.y * controlOffset,
48
45
  };
49
46
  }
50
- function getPathData(path, source, target, firstControl, secondControl, usingPoints) {
47
+ function getPathData(path, source, target, sourceControl, targetControl, usingPoints) {
51
48
  const [start, center, end] = usingPoints;
52
49
  const nullPoint = { x: 0, y: 0 };
53
50
  return {
54
- path: path.toString(),
51
+ path,
55
52
  points: {
56
53
  start: start
57
- ? getPointOnBezier(source, target, firstControl, secondControl, 0.1)
54
+ ? getPointOnBezier(source, target, sourceControl, targetControl, 0.1)
58
55
  : nullPoint,
59
56
  center: center
60
- ? getPointOnBezier(source, target, firstControl, secondControl, 0.5)
57
+ ? getPointOnBezier(source, target, sourceControl, targetControl, 0.5)
61
58
  : nullPoint,
62
59
  end: end
63
- ? getPointOnBezier(source, target, firstControl, secondControl, 0.9)
60
+ ? getPointOnBezier(source, target, sourceControl, targetControl, 0.9)
64
61
  : nullPoint,
65
62
  },
66
63
  };
@@ -68,10 +65,10 @@ function getPathData(path, source, target, firstControl, secondControl, usingPoi
68
65
  /**
69
66
  * Get point on bezier curve by ratio
70
67
  */
71
- function getPointOnBezier(sourcePoint, targetPoint, controlPoint1, controlPoint2, ratio) {
72
- const fromSourceToFirstControl = getPointOnLineByRatio(sourcePoint, controlPoint1, ratio);
73
- const fromFirstControlToSecond = getPointOnLineByRatio(controlPoint1, controlPoint2, ratio);
74
- const fromSecondControlToTarget = getPointOnLineByRatio(controlPoint2, targetPoint, ratio);
68
+ function getPointOnBezier(sourcePoint, targetPoint, sourceControl, targetControl, ratio) {
69
+ const fromSourceToFirstControl = getPointOnLineByRatio(sourcePoint, sourceControl, ratio);
70
+ const fromFirstControlToSecond = getPointOnLineByRatio(sourceControl, targetControl, ratio);
71
+ const fromSecondControlToTarget = getPointOnLineByRatio(targetControl, targetPoint, ratio);
75
72
  return getPointOnLineByRatio(getPointOnLineByRatio(fromSourceToFirstControl, fromFirstControlToSecond, ratio), getPointOnLineByRatio(fromFirstControlToSecond, fromSecondControlToTarget, ratio), ratio);
76
73
  }
77
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bezier-path.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/math/edge-path/bezier-path.ts"],"names":[],"mappings":"AAEA,OAAO,EAAQ,IAAI,IAAI,MAAM,EAAE,MAAM,SAAS,CAAC;AAG/C,OAAO,EAAE,qBAAqB,EAAE,MAAM,2BAA2B,CAAC;AAElE,MAAM,UAAU,UAAU,CACxB,MAAa,EACb,MAAa,EACb,cAAwB,EACxB,cAAwB,EACxB,cAA2B,CAAC,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC;IAEhD,MAAM,IAAI,GAAG,MAAM,EAAE,CAAC;IAEtB,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC;IAEhC,MAAM,cAAc,GAAG,EAAE,CAAC,EAAE,MAAM,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC,EAAE,CAAC,EAAE,MAAM,CAAC,CAAC,GAAG,MAAM,CAAC,CAAC,EAAE,CAAC;IAE1E,MAAM,YAAY,GAAG,gBAAgB,CAAC,MAAM,EAAE,cAAc,EAAE,cAAc,CAAC,CAAC;IAC9E,MAAM,aAAa,GAAG,gBAAgB,CAAC,MAAM,EAAE,cAAc,EAAE,cAAc,CAAC,CAAC;IAE/E,IAAI,CAAC,aAAa,CAChB,YAAY,CAAC,CAAC,EACd,YAAY,CAAC,CAAC,EACd,aAAa,CAAC,CAAC,EACf,aAAa,CAAC,CAAC,EACf,MAAM,CAAC,CAAC,EACR,MAAM,CAAC,CAAC,CACT,CAAC;IAEF,OAAO,WAAW,CAChB,IAAI,EACJ,MAAM,EACN,MAAM,EACN,YAAY,EACZ,aAAa,EACb,WAAW,CACZ,CAAC;AACJ,CAAC;AAED;;;;;;GAMG;AAEH,SAAS,gBAAgB,CACvB,KAAY,EACZ,aAAuB,EACvB,cAAqB;IAErB,MAAM,WAAW,GAAG,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC;IAEnC,QAAQ,aAAa,EAAE;QACrB,KAAK,KAAK;YACR,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC;YAClB,MAAM;QACR,KAAK,QAAQ;YACX,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;YACnB,MAAM;QACR,KAAK,OAAO;YACV,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC;YAClB,MAAM;QACR,KAAK,MAAM;YACT,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;YACnB,MAAM;KACT;IAED,qBAAqB;IACrB,MAAM,kBAAkB,GAAG;QACzB,CAAC,EAAE,cAAc,CAAC,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,CAAC;QAC7C,CAAC,EAAE,cAAc,CAAC,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC,CAAC;KAC9C,CAAC;IAEF,gDAAgD;IAChD,MAAM,SAAS,GAAG,IAAI,CAAC;IACvB,4CAA4C;IAC5C,oIAAoI;IACpI,MAAM,aAAa,GACjB,SAAS,GAAG,EAAE,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC,GAAG,kBAAkB,CAAC,CAAC,CAAC,CAAC,CAAC;IAEpF,OAAO;QACL,CAAC,EAAE,KAAK,CAAC,CAAC,GAAG,WAAW,CAAC,CAAC,GAAG,aAAa;QAC1C,CAAC,EAAE,KAAK,CAAC,CAAC,GAAG,WAAW,CAAC,CAAC,GAAG,aAAa;KAC3C,CAAC;AACJ,CAAC;AAED,SAAS,WAAW,CAClB,IAAU,EACV,MAAa,EACb,MAAa,EACb,YAAmB,EACnB,aAAoB,EACpB,WAAwB;IAExB,MAAM,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,CAAC,GAAG,WAAW,CAAC;IAEzC,MAAM,SAAS,GAAG,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC;IAEjC,OAAO;QACL,IAAI,EAAE,IAAI,CAAC,QAAQ,EAAE;QACrB,MAAM,EAAE;YACN,KAAK,EAAE,KAAK;gBACV,CAAC,CAAC,gBAAgB,CAAC,MAAM,EAAE,MAAM,EAAE,YAAY,EAAE,aAAa,EAAE,GAAG,CAAC;gBACpE,CAAC,CAAC,SAAS;YACb,MAAM,EAAE,MAAM;gBACZ,CAAC,CAAC,gBAAgB,CAAC,MAAM,EAAE,MAAM,EAAE,YAAY,EAAE,aAAa,EAAE,GAAG,CAAC;gBACpE,CAAC,CAAC,SAAS;YACb,GAAG,EAAE,GAAG;gBACN,CAAC,CAAC,gBAAgB,CAAC,MAAM,EAAE,MAAM,EAAE,YAAY,EAAE,aAAa,EAAE,GAAG,CAAC;gBACpE,CAAC,CAAC,SAAS;SACd;KACF,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,SAAS,gBAAgB,CACvB,WAAkB,EAClB,WAAkB,EAClB,aAAoB,EACpB,aAAoB,EACpB,KAAa;IAEb,MAAM,wBAAwB,GAAU,qBAAqB,CAC3D,WAAW,EACX,aAAa,EACb,KAAK,CACN,CAAC;IACF,MAAM,wBAAwB,GAAU,qBAAqB,CAC3D,aAAa,EACb,aAAa,EACb,KAAK,CACN,CAAC;IACF,MAAM,yBAAyB,GAAU,qBAAqB,CAC5D,aAAa,EACb,WAAW,EACX,KAAK,CACN,CAAC;IAEF,OAAO,qBAAqB,CAC1B,qBAAqB,CAAC,wBAAwB,EAAE,wBAAwB,EAAE,KAAK,CAAC,EAChF,qBAAqB,CAAC,wBAAwB,EAAE,yBAAyB,EAAE,KAAK,CAAC,EACjF,KAAK,CACN,CAAC;AACJ,CAAC","sourcesContent":["import { PathData } from '../../interfaces/path-data.interface';\nimport { Point } from '../../interfaces/point.interface';\nimport { Path, path as d3Path } from 'd3-path';\nimport { UsingPoints } from '../../types/using-points.type';\nimport { Position } from '../../types/position.type';\nimport { getPointOnLineByRatio } from '../point-on-line-by-ratio';\n\nexport function bezierPath(\n  source: Point,\n  target: Point,\n  sourcePosition: Position,\n  targetPosition: Position,\n  usingPoints: UsingPoints = [false, false, false]\n): PathData {\n  const path = d3Path();\n\n  path.moveTo(source.x, source.y);\n\n  const distanceVector = { x: source.x - target.x, y: source.y - target.y };\n\n  const firstControl = calcControlPoint(source, sourcePosition, distanceVector);\n  const secondControl = calcControlPoint(target, targetPosition, distanceVector);\n\n  path.bezierCurveTo(\n    firstControl.x,\n    firstControl.y,\n    secondControl.x,\n    secondControl.y,\n    target.x,\n    target.y\n  );\n\n  return getPathData(\n    path,\n    source,\n    target,\n    firstControl,\n    secondControl,\n    usingPoints\n  );\n}\n\n/**\n * Calculate control point based on provided point\n *\n * @param point relative this point control point is gonna be computed (the source or the target)\n * @param pointPosition position of {point} on block\n * @param distanceVector transmits the distance between the source and the target as x and y coordinates\n */\n\nfunction calcControlPoint(\n  point: Point,\n  pointPosition: Position,\n  distanceVector: Point\n) {\n  const factorPoint = { x: 0, y: 0 };\n\n  switch (pointPosition) {\n    case 'top':\n      factorPoint.y = 1;\n      break;\n    case 'bottom':\n      factorPoint.y = -1;\n      break;\n    case 'right':\n      factorPoint.x = 1;\n      break;\n    case 'left':\n      factorPoint.x = -1;\n      break;\n  }\n\n  // TODO: explain name\n  const fullDistanceVector = {\n    x: distanceVector.x * Math.abs(factorPoint.x),\n    y: distanceVector.y * Math.abs(factorPoint.y),\n  };\n\n  // TODO: probably need to make this configurable\n  const curvature = 0.25;\n  // thanks colleagues from react/svelte world\n  // https://github.com/xyflow/xyflow/blob/f0117939bae934447fa7f232081f937169ee23b5/packages/system/src/utils/edges/bezier-edge.ts#L56\n  const controlOffset =\n    curvature * 25 * Math.sqrt(Math.abs(fullDistanceVector.x + fullDistanceVector.y));\n\n  return {\n    x: point.x + factorPoint.x * controlOffset,\n    y: point.y - factorPoint.y * controlOffset,\n  };\n}\n\nfunction getPathData(\n  path: Path,\n  source: Point,\n  target: Point,\n  firstControl: Point,\n  secondControl: Point,\n  usingPoints: UsingPoints\n): PathData {\n  const [start, center, end] = usingPoints;\n\n  const nullPoint = { x: 0, y: 0 };\n\n  return {\n    path: path.toString(),\n    points: {\n      start: start\n        ? getPointOnBezier(source, target, firstControl, secondControl, 0.1)\n        : nullPoint,\n      center: center\n        ? getPointOnBezier(source, target, firstControl, secondControl, 0.5)\n        : nullPoint,\n      end: end\n        ? getPointOnBezier(source, target, firstControl, secondControl, 0.9)\n        : nullPoint,\n    },\n  };\n}\n\n/**\n * Get point on bezier curve by ratio\n */\nfunction getPointOnBezier(\n  sourcePoint: Point,\n  targetPoint: Point,\n  controlPoint1: Point,\n  controlPoint2: Point,\n  ratio: number\n): Point {\n  const fromSourceToFirstControl: Point = getPointOnLineByRatio(\n    sourcePoint,\n    controlPoint1,\n    ratio\n  );\n  const fromFirstControlToSecond: Point = getPointOnLineByRatio(\n    controlPoint1,\n    controlPoint2,\n    ratio\n  );\n  const fromSecondControlToTarget: Point = getPointOnLineByRatio(\n    controlPoint2,\n    targetPoint,\n    ratio\n  );\n\n  return getPointOnLineByRatio(\n    getPointOnLineByRatio(fromSourceToFirstControl, fromFirstControlToSecond, ratio),\n    getPointOnLineByRatio(fromFirstControlToSecond, fromSecondControlToTarget, ratio),\n    ratio\n  );\n}\n"]}
74
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bezier-path.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/math/edge-path/bezier-path.ts"],"names":[],"mappings":"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","sourcesContent":["import { PathData } from '../../interfaces/path-data.interface';\nimport { Point } from '../../interfaces/point.interface';\nimport { UsingPoints } from '../../types/using-points.type';\nimport { Position } from '../../types/position.type';\nimport { getPointOnLineByRatio } from '../point-on-line-by-ratio';\n\nexport function bezierPath(\n  source: Point,\n  target: Point,\n  sourcePosition: Position,\n  targetPosition: Position,\n  usingPoints: UsingPoints = [false, false, false]\n): PathData {\n  const distanceVector = { x: source.x - target.x, y: source.y - target.y };\n\n  const sourceControl = calcControlPoint(source, sourcePosition, distanceVector);\n  const targetControl = calcControlPoint(target, targetPosition, distanceVector);\n\n  const path =\n    `M${source.x},${source.y} C${sourceControl.x},${sourceControl.y} ${targetControl.x},${targetControl.y} ${target.x},${target.y}`\n\n  return getPathData(\n    path,\n    source,\n    target,\n    sourceControl,\n    targetControl,\n    usingPoints\n  );\n}\n\n/**\n * Calculate control point based on provided point\n *\n * @param point relative this point control point is gonna be computed (the source or the target)\n * @param pointPosition position of {point} on block\n * @param distanceVector transmits the distance between the source and the target as x and y coordinates\n */\n\nfunction calcControlPoint(\n  point: Point,\n  pointPosition: Position,\n  distanceVector: Point\n) {\n  const factorPoint = { x: 0, y: 0 };\n\n  switch (pointPosition) {\n    case 'top':\n      factorPoint.y = 1;\n      break;\n    case 'bottom':\n      factorPoint.y = -1;\n      break;\n    case 'right':\n      factorPoint.x = 1;\n      break;\n    case 'left':\n      factorPoint.x = -1;\n      break;\n  }\n\n  // TODO: explain name\n  const fullDistanceVector = {\n    x: distanceVector.x * Math.abs(factorPoint.x),\n    y: distanceVector.y * Math.abs(factorPoint.y),\n  };\n\n  // TODO: probably need to make this configurable\n  const curvature = 0.25;\n  // thanks colleagues from react/svelte world\n  // https://github.com/xyflow/xyflow/blob/f0117939bae934447fa7f232081f937169ee23b5/packages/system/src/utils/edges/bezier-edge.ts#L56\n  const controlOffset =\n    curvature * 25 * Math.sqrt(Math.abs(fullDistanceVector.x + fullDistanceVector.y));\n\n  return {\n    x: point.x + factorPoint.x * controlOffset,\n    y: point.y - factorPoint.y * controlOffset,\n  };\n}\n\nfunction getPathData(\n  path: string,\n  source: Point,\n  target: Point,\n  sourceControl: Point,\n  targetControl: Point,\n  usingPoints: UsingPoints\n): PathData {\n  const [start, center, end] = usingPoints;\n\n  const nullPoint = { x: 0, y: 0 };\n\n  return {\n    path,\n    points: {\n      start: start\n        ? getPointOnBezier(source, target, sourceControl, targetControl, 0.1)\n        : nullPoint,\n      center: center\n        ? getPointOnBezier(source, target, sourceControl, targetControl, 0.5)\n        : nullPoint,\n      end: end\n        ? getPointOnBezier(source, target, sourceControl, targetControl, 0.9)\n        : nullPoint,\n    },\n  };\n}\n\n/**\n * Get point on bezier curve by ratio\n */\nfunction getPointOnBezier(\n  sourcePoint: Point,\n  targetPoint: Point,\n  sourceControl: Point,\n  targetControl: Point,\n  ratio: number\n): Point {\n  const fromSourceToFirstControl: Point = getPointOnLineByRatio(\n    sourcePoint,\n    sourceControl,\n    ratio\n  );\n  const fromFirstControlToSecond: Point = getPointOnLineByRatio(\n    sourceControl,\n    targetControl,\n    ratio\n  );\n  const fromSecondControlToTarget: Point = getPointOnLineByRatio(\n    targetControl,\n    targetPoint,\n    ratio\n  );\n\n  return getPointOnLineByRatio(\n    getPointOnLineByRatio(fromSourceToFirstControl, fromFirstControlToSecond, ratio),\n    getPointOnLineByRatio(fromFirstControlToSecond, fromSecondControlToTarget, ratio),\n    ratio\n  );\n}\n"]}
@@ -1,13 +1,9 @@
1
- import { path as d3Path } from 'd3-path';
2
1
  import { getPointOnLineByRatio } from "../point-on-line-by-ratio";
3
2
  export function straightPath(source, target, usingPoints = [false, false, false]) {
4
3
  const [start, center, end] = usingPoints;
5
4
  const nullPoint = { x: 0, y: 0 };
6
- const path = d3Path();
7
- path.moveTo(source.x, source.y);
8
- path.lineTo(target.x, target.y);
9
5
  return {
10
- path: path.toString(),
6
+ path: `M ${source.x},${source.y}L ${target.x},${target.y}`,
11
7
  points: {
12
8
  start: start ? getPointOnLineByRatio(source, target, .15) : nullPoint,
13
9
  center: center ? getPointOnLineByRatio(source, target, .50) : nullPoint,
@@ -15,4 +11,4 @@ export function straightPath(source, target, usingPoints = [false, false, false]
15
11
  }
16
12
  };
17
13
  }
18
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RyYWlnaC1wYXRoLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LXZmbG93LWxpYi9zcmMvbGliL3ZmbG93L21hdGgvZWRnZS1wYXRoL3N0cmFpZ2gtcGF0aC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFFQSxPQUFPLEVBQUUsSUFBSSxJQUFJLE1BQU0sRUFBRSxNQUFNLFNBQVMsQ0FBQTtBQUV4QyxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUVsRSxNQUFNLFVBQVUsWUFBWSxDQUMxQixNQUFhLEVBQ2IsTUFBYSxFQUNiLGNBQTJCLENBQUMsS0FBSyxFQUFFLEtBQUssRUFBRSxLQUFLLENBQUM7SUFFaEQsTUFBTSxDQUFDLEtBQUssRUFBRSxNQUFNLEVBQUUsR0FBRyxDQUFDLEdBQUcsV0FBVyxDQUFBO0lBQ3hDLE1BQU0sU0FBUyxHQUFHLEVBQUUsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUE7SUFFaEMsTUFBTSxJQUFJLEdBQUcsTUFBTSxFQUFFLENBQUE7SUFFckIsSUFBSSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxFQUFFLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQTtJQUMvQixJQUFJLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxDQUFDLEVBQUUsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFBO0lBRS9CLE9BQU87UUFDTCxJQUFJLEVBQUUsSUFBSSxDQUFDLFFBQVEsRUFBRTtRQUNyQixNQUFNLEVBQUU7WUFDTixLQUFLLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FBQyxxQkFBcUIsQ0FBQyxNQUFNLEVBQUUsTUFBTSxFQUFFLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxTQUFTO1lBQ3JFLE1BQU0sRUFBRSxNQUFNLENBQUMsQ0FBQyxDQUFDLHFCQUFxQixDQUFDLE1BQU0sRUFBRSxNQUFNLEVBQUUsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLFNBQVM7WUFDdkUsR0FBRyxFQUFFLEdBQUcsQ0FBQyxDQUFDLENBQUMscUJBQXFCLENBQUMsTUFBTSxFQUFFLE1BQU0sRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsU0FBUztTQUNsRTtLQUNGLENBQUE7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUGF0aERhdGEgfSBmcm9tIFwiLi4vLi4vaW50ZXJmYWNlcy9wYXRoLWRhdGEuaW50ZXJmYWNlXCI7XG5pbXBvcnQgeyBQb2ludCB9IGZyb20gXCIuLi8uLi9pbnRlcmZhY2VzL3BvaW50LmludGVyZmFjZVwiO1xuaW1wb3J0IHsgcGF0aCBhcyBkM1BhdGggfSBmcm9tICdkMy1wYXRoJ1xuaW1wb3J0IHsgVXNpbmdQb2ludHMgfSBmcm9tIFwiLi4vLi4vdHlwZXMvdXNpbmctcG9pbnRzLnR5cGVcIjtcbmltcG9ydCB7IGdldFBvaW50T25MaW5lQnlSYXRpbyB9IGZyb20gXCIuLi9wb2ludC1vbi1saW5lLWJ5LXJhdGlvXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBzdHJhaWdodFBhdGgoXG4gIHNvdXJjZTogUG9pbnQsXG4gIHRhcmdldDogUG9pbnQsXG4gIHVzaW5nUG9pbnRzOiBVc2luZ1BvaW50cyA9IFtmYWxzZSwgZmFsc2UsIGZhbHNlXVxuKTogUGF0aERhdGEge1xuICBjb25zdCBbc3RhcnQsIGNlbnRlciwgZW5kXSA9IHVzaW5nUG9pbnRzXG4gIGNvbnN0IG51bGxQb2ludCA9IHsgeDogMCwgeTogMCB9XG5cbiAgY29uc3QgcGF0aCA9IGQzUGF0aCgpXG5cbiAgcGF0aC5tb3ZlVG8oc291cmNlLngsIHNvdXJjZS55KVxuICBwYXRoLmxpbmVUbyh0YXJnZXQueCwgdGFyZ2V0LnkpXG5cbiAgcmV0dXJuIHtcbiAgICBwYXRoOiBwYXRoLnRvU3RyaW5nKCksXG4gICAgcG9pbnRzOiB7XG4gICAgICBzdGFydDogc3RhcnQgPyBnZXRQb2ludE9uTGluZUJ5UmF0aW8oc291cmNlLCB0YXJnZXQsIC4xNSkgOiBudWxsUG9pbnQsXG4gICAgICBjZW50ZXI6IGNlbnRlciA/IGdldFBvaW50T25MaW5lQnlSYXRpbyhzb3VyY2UsIHRhcmdldCwgLjUwKSA6IG51bGxQb2ludCxcbiAgICAgIGVuZDogZW5kID8gZ2V0UG9pbnRPbkxpbmVCeVJhdGlvKHNvdXJjZSwgdGFyZ2V0LCAuODUpIDogbnVsbFBvaW50LFxuICAgIH1cbiAgfVxufVxuIl19
14
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RyYWlnaC1wYXRoLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LXZmbG93LWxpYi9zcmMvbGliL3ZmbG93L21hdGgvZWRnZS1wYXRoL3N0cmFpZ2gtcGF0aC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFHQSxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUVsRSxNQUFNLFVBQVUsWUFBWSxDQUMxQixNQUFhLEVBQ2IsTUFBYSxFQUNiLGNBQTJCLENBQUMsS0FBSyxFQUFFLEtBQUssRUFBRSxLQUFLLENBQUM7SUFFaEQsTUFBTSxDQUFDLEtBQUssRUFBRSxNQUFNLEVBQUUsR0FBRyxDQUFDLEdBQUcsV0FBVyxDQUFBO0lBQ3hDLE1BQU0sU0FBUyxHQUFHLEVBQUUsQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUE7SUFFaEMsT0FBTztRQUNMLElBQUksRUFBRSxLQUFLLE1BQU0sQ0FBQyxDQUFDLElBQUksTUFBTSxDQUFDLENBQUMsS0FBSyxNQUFNLENBQUMsQ0FBQyxJQUFJLE1BQU0sQ0FBQyxDQUFDLEVBQUU7UUFDMUQsTUFBTSxFQUFFO1lBQ04sS0FBSyxFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUMscUJBQXFCLENBQUMsTUFBTSxFQUFFLE1BQU0sRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsU0FBUztZQUNyRSxNQUFNLEVBQUUsTUFBTSxDQUFDLENBQUMsQ0FBQyxxQkFBcUIsQ0FBQyxNQUFNLEVBQUUsTUFBTSxFQUFFLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxTQUFTO1lBQ3ZFLEdBQUcsRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFDLHFCQUFxQixDQUFDLE1BQU0sRUFBRSxNQUFNLEVBQUUsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLFNBQVM7U0FDbEU7S0FDRixDQUFBO0FBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBhdGhEYXRhIH0gZnJvbSBcIi4uLy4uL2ludGVyZmFjZXMvcGF0aC1kYXRhLmludGVyZmFjZVwiO1xuaW1wb3J0IHsgUG9pbnQgfSBmcm9tIFwiLi4vLi4vaW50ZXJmYWNlcy9wb2ludC5pbnRlcmZhY2VcIjtcbmltcG9ydCB7IFVzaW5nUG9pbnRzIH0gZnJvbSBcIi4uLy4uL3R5cGVzL3VzaW5nLXBvaW50cy50eXBlXCI7XG5pbXBvcnQgeyBnZXRQb2ludE9uTGluZUJ5UmF0aW8gfSBmcm9tIFwiLi4vcG9pbnQtb24tbGluZS1ieS1yYXRpb1wiO1xuXG5leHBvcnQgZnVuY3Rpb24gc3RyYWlnaHRQYXRoKFxuICBzb3VyY2U6IFBvaW50LFxuICB0YXJnZXQ6IFBvaW50LFxuICB1c2luZ1BvaW50czogVXNpbmdQb2ludHMgPSBbZmFsc2UsIGZhbHNlLCBmYWxzZV1cbik6IFBhdGhEYXRhIHtcbiAgY29uc3QgW3N0YXJ0LCBjZW50ZXIsIGVuZF0gPSB1c2luZ1BvaW50c1xuICBjb25zdCBudWxsUG9pbnQgPSB7IHg6IDAsIHk6IDAgfVxuXG4gIHJldHVybiB7XG4gICAgcGF0aDogYE0gJHtzb3VyY2UueH0sJHtzb3VyY2UueX1MICR7dGFyZ2V0Lnh9LCR7dGFyZ2V0Lnl9YCxcbiAgICBwb2ludHM6IHtcbiAgICAgIHN0YXJ0OiBzdGFydCA/IGdldFBvaW50T25MaW5lQnlSYXRpbyhzb3VyY2UsIHRhcmdldCwgLjE1KSA6IG51bGxQb2ludCxcbiAgICAgIGNlbnRlcjogY2VudGVyID8gZ2V0UG9pbnRPbkxpbmVCeVJhdGlvKHNvdXJjZSwgdGFyZ2V0LCAuNTApIDogbnVsbFBvaW50LFxuICAgICAgZW5kOiBlbmQgPyBnZXRQb2ludE9uTGluZUJ5UmF0aW8oc291cmNlLCB0YXJnZXQsIC44NSkgOiBudWxsUG9pbnQsXG4gICAgfVxuICB9XG59XG4iXX0=
@@ -68,7 +68,8 @@ export class HandleModel {
68
68
  this.templateContext = {
69
69
  $implicit: {
70
70
  point: this.offset,
71
- state: this.state
71
+ state: this.state,
72
+ node: this.parentNode.node
72
73
  }
73
74
  };
74
75
  }
@@ -88,4 +89,4 @@ export class HandleModel {
88
89
  return { width: 0, height: 0 };
89
90
  }
90
91
  }
91
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handle.model.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/models/handle.model.ts"],"names":[],"mappings":"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","sourcesContent":["import { computed, signal } from \"@angular/core\";\nimport { NodeHandle } from \"../services/handle.service\";\nimport { NodeModel } from \"./node.model\";\nimport { Subject, map } from \"rxjs\";\nimport { toSignal } from \"@angular/core/rxjs-interop\";\n\nexport type HandleState = 'valid' | 'invalid' | 'idle'\n\nexport class HandleModel {\n  public readonly strokeWidth = 2\n\n  /**\n   * Pre-computed size for default handle, changed dynamically\n   * for custom handles\n   */\n  public size = signal({\n    width: 10 + (2 * this.strokeWidth),\n    height: 10 + (2 * this.strokeWidth)\n  })\n\n  public offset = computed(() => {\n    switch (this.rawHandle.position) {\n      case 'left': return {\n        x: 0,\n        y: this.parentPosition().y + (this.parentSize().height / 2)\n      }\n      case 'right': return {\n        x: this.parentNode.size().width,\n        y: this.parentPosition().y + (this.parentSize().height / 2)\n      }\n      case 'top': return {\n        x: this.parentPosition().x + (this.parentSize().width / 2),\n        y: 0\n      }\n      case 'bottom': return {\n        x: this.parentPosition().x + this.parentSize().width / 2,\n        y: this.parentNode.size().height\n      }\n    }\n  })\n\n  public sizeOffset = computed(() => {\n    switch (this.rawHandle.position) {\n      case 'left': return { x: -(this.size().width / 2), y: 0 }\n      case 'right': return { x: this.size().width / 2, y: 0 }\n      case 'top': return { x: 0, y: -(this.size().height / 2) }\n      case 'bottom': return { x: 0, y: this.size().height / 2 }\n    }\n  })\n\n  public pointAbsolute = computed(() => {\n    return {\n      x: this.parentNode.globalPoint().x + this.offset().x + this.sizeOffset().x,\n      y: this.parentNode.globalPoint().y + this.offset().y + this.sizeOffset().y,\n    }\n  })\n\n  public state = signal<HandleState>('idle')\n\n  private updateParentSizeAndPosition$ = new Subject<void>()\n\n  public parentSize = toSignal(\n    this.updateParentSizeAndPosition$.pipe(\n      map(() => this.getParentSize())\n    ),\n    {\n      initialValue: { width: 0, height: 0 }\n    }\n  )\n\n  public parentPosition = toSignal(\n    this.updateParentSizeAndPosition$.pipe(\n      map(() => ({\n        x: this.parentReference instanceof HTMLElement\n          ? this.parentReference.offsetLeft\n          : 0, // for now just 0 for group nodes\n        y: this.parentReference instanceof HTMLElement\n          ? this.parentReference.offsetTop\n          : 0 // for now just 0 for group nodes\n      }))\n    ),\n    {\n      initialValue: { x: 0, y: 0 }\n    }\n  )\n\n  public parentReference = this.rawHandle.parentReference!\n\n  public template = this.rawHandle.template\n\n  public templateContext = {\n    $implicit: {\n      point: this.offset,\n      state: this.state\n    }\n  }\n\n  constructor(\n    public rawHandle: NodeHandle,\n    public parentNode: NodeModel\n  ) { }\n\n  public updateParent() {\n    this.updateParentSizeAndPosition$.next()\n  }\n\n  private getParentSize(): { width: number, height: number } {\n    if (this.parentReference instanceof HTMLElement) {\n      return {\n        width: this.parentReference.offsetWidth,\n        height: this.parentReference.offsetHeight\n      }\n    } else if (this.parentReference instanceof SVGGraphicsElement) {\n      return this.parentReference.getBBox()\n    }\n\n    return { width: 0, height: 0 }\n  }\n}\n"]}
92
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handle.model.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/models/handle.model.ts"],"names":[],"mappings":"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","sourcesContent":["import { computed, signal } from \"@angular/core\";\nimport { NodeHandle } from \"../services/handle.service\";\nimport { NodeModel } from \"./node.model\";\nimport { Subject, map } from \"rxjs\";\nimport { toSignal } from \"@angular/core/rxjs-interop\";\n\nexport type HandleState = 'valid' | 'invalid' | 'idle'\n\nexport class HandleModel {\n  public readonly strokeWidth = 2\n\n  /**\n   * Pre-computed size for default handle, changed dynamically\n   * for custom handles\n   */\n  public size = signal({\n    width: 10 + (2 * this.strokeWidth),\n    height: 10 + (2 * this.strokeWidth)\n  })\n\n  public offset = computed(() => {\n    switch (this.rawHandle.position) {\n      case 'left': return {\n        x: 0,\n        y: this.parentPosition().y + (this.parentSize().height / 2)\n      }\n      case 'right': return {\n        x: this.parentNode.size().width,\n        y: this.parentPosition().y + (this.parentSize().height / 2)\n      }\n      case 'top': return {\n        x: this.parentPosition().x + (this.parentSize().width / 2),\n        y: 0\n      }\n      case 'bottom': return {\n        x: this.parentPosition().x + this.parentSize().width / 2,\n        y: this.parentNode.size().height\n      }\n    }\n  })\n\n  public sizeOffset = computed(() => {\n    switch (this.rawHandle.position) {\n      case 'left': return { x: -(this.size().width / 2), y: 0 }\n      case 'right': return { x: this.size().width / 2, y: 0 }\n      case 'top': return { x: 0, y: -(this.size().height / 2) }\n      case 'bottom': return { x: 0, y: this.size().height / 2 }\n    }\n  })\n\n  public pointAbsolute = computed(() => {\n    return {\n      x: this.parentNode.globalPoint().x + this.offset().x + this.sizeOffset().x,\n      y: this.parentNode.globalPoint().y + this.offset().y + this.sizeOffset().y,\n    }\n  })\n\n  public state = signal<HandleState>('idle')\n\n  private updateParentSizeAndPosition$ = new Subject<void>()\n\n  public parentSize = toSignal(\n    this.updateParentSizeAndPosition$.pipe(\n      map(() => this.getParentSize())\n    ),\n    {\n      initialValue: { width: 0, height: 0 }\n    }\n  )\n\n  public parentPosition = toSignal(\n    this.updateParentSizeAndPosition$.pipe(\n      map(() => ({\n        x: this.parentReference instanceof HTMLElement\n          ? this.parentReference.offsetLeft\n          : 0, // for now just 0 for group nodes\n        y: this.parentReference instanceof HTMLElement\n          ? this.parentReference.offsetTop\n          : 0 // for now just 0 for group nodes\n      }))\n    ),\n    {\n      initialValue: { x: 0, y: 0 }\n    }\n  )\n\n  public parentReference = this.rawHandle.parentReference!\n\n  public template = this.rawHandle.template\n\n  public templateContext = {\n    $implicit: {\n      point: this.offset,\n      state: this.state,\n      node: this.parentNode.node\n    }\n  }\n\n  constructor(\n    public rawHandle: NodeHandle,\n    public parentNode: NodeModel\n  ) { }\n\n  public updateParent() {\n    this.updateParentSizeAndPosition$.next()\n  }\n\n  private getParentSize(): { width: number, height: number } {\n    if (this.parentReference instanceof HTMLElement) {\n      return {\n        width: this.parentReference.offsetWidth,\n        height: this.parentReference.offsetHeight\n      }\n    } else if (this.parentReference instanceof SVGGraphicsElement) {\n      return this.parentReference.getBBox()\n    }\n\n    return { width: 0, height: 0 }\n  }\n}\n"]}
@@ -48,6 +48,7 @@ export class NodeModel {
48
48
  this.handles = signal([]);
49
49
  this.handles$ = toObservable(this.handles);
50
50
  this.draggable = signal(true);
51
+ this.dragHandlesCount = signal(0);
51
52
  // disabled for configuration for now
52
53
  this.magnetRadius = 20;
53
54
  // TODO: not sure if we need to statically store it
@@ -171,4 +172,4 @@ export class NodeModel {
171
172
  : signal({ x: this.node.point.x, y: this.node.point.y });
172
173
  }
173
174
  }
174
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node.model.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/models/node.model.ts"],"names":[],"mappings":"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","sourcesContent":["import { Signal, TemplateRef, computed, effect, inject, signal } from '@angular/core'\nimport { DynamicNode, Node, isDynamicNode } from '../interfaces/node.interface'\nimport { isDefined } from '../utils/is-defined'\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop'\nimport { HandleModel } from './handle.model'\nimport { FlowEntity } from '../interfaces/flow-entity.interface'\nimport { FlowSettingsService } from '../services/flow-settings.service'\nimport { animationFrameScheduler, merge, observeOn, Subject, } from 'rxjs'\nimport { Point } from '../interfaces/point.interface'\nimport { CustomNodeComponent } from '../public-components/custom-node/custom-node.component'\nimport { CustomDynamicNodeComponent } from '../public-components/custom-dynamic-node/custom-dynamic-node.component'\nimport { FlowEntitiesService } from '../services/flow-entities.service'\n\n// TODO bad naming around points\nexport class NodeModel<T = unknown> implements FlowEntity {\n  private static defaultWidth = 100\n  private static defaultHeight = 50\n  private static defaultColor = '#1b262c'\n\n  private flowSettingsService = inject(FlowSettingsService)\n  private entitiesService = inject(FlowEntitiesService);\n\n  private internalPoint = this.createInternalPointSignal()\n\n  private throttledPoint$ = toObservable(this.internalPoint).pipe(\n    observeOn(animationFrameScheduler)\n  )\n\n  private notThrottledPoint$ = new Subject<Point>()\n\n  public point = toSignal(\n    merge(this.throttledPoint$, this.notThrottledPoint$),\n    {\n      initialValue: this.internalPoint()\n    }\n  )\n\n  public point$ = this.throttledPoint$;\n\n  public size = signal({ width: 0, height: 0 })\n  public size$ = toObservable(this.size)\n\n  public width = computed(() => this.size().width)\n  public height = computed(() => this.size().height)\n\n  public renderOrder = signal(0)\n\n  public selected = signal(false)\n  public selected$ = toObservable(this.selected)\n\n  public globalPoint = computed(() => {\n    let parent = this.parent()\n    let x = this.point().x\n    let y = this.point().y\n\n    while (parent !== null) {\n      x += parent.point().x\n      y += parent.point().y\n\n      parent = parent.parent()\n    }\n\n    return { x, y }\n  })\n\n  public pointTransform = computed(() =>\n    `translate(${this.globalPoint().x}, ${this.globalPoint().y})`\n  )\n\n  // Now source and handle positions derived from parent flow\n  public sourcePosition = computed(() => this.flowSettingsService.handlePositions().source)\n  public targetPosition = computed(() => this.flowSettingsService.handlePositions().target)\n\n  public handles = signal<HandleModel[]>([])\n\n  public handles$ = toObservable(this.handles)\n\n  public draggable = signal(true)\n\n  // disabled for configuration for now\n  public readonly magnetRadius = 20\n\n  // TODO: not sure if we need to statically store it\n  public isComponentType =\n    CustomNodeComponent.isPrototypeOf(this.node.type) ||\n    CustomDynamicNodeComponent.isPrototypeOf(this.node.type)\n\n  // Default node specific thing\n  public text = this.createTextSignal()\n\n  // Component node specific thing\n  public componentTypeInputs = computed(() => {\n    return {\n      node: this.node,\n      _selected: this.selected()\n    }\n  })\n\n  public parent = computed(() =>\n    this.entitiesService.nodes().find(n => n.node.id === this.parentId()) ?? null\n  )\n\n  public children = computed(() =>\n    this.entitiesService.nodes().filter(n => n.parentId() === this.node.id)\n  )\n\n  public color = signal(NodeModel.defaultColor)\n\n  public resizable = signal(false)\n  public resizing = signal(false)\n  public resizerTemplate = signal<TemplateRef<unknown> | null>(null)\n\n  private parentId = signal<string | null>(null)\n\n  constructor(\n    public node: Node<T> | DynamicNode<T>\n  ) {\n    if (isDefined(node.draggable)) {\n      if (isDynamicNode(node)) {\n        this.draggable = node.draggable\n      } else {\n        this.draggable.set(node.draggable)\n      }\n    }\n\n    if (isDefined(node.parentId)) {\n      if (isDynamicNode(node)) {\n        this.parentId = node.parentId\n      } else {\n        this.parentId.set(node.parentId)\n      }\n    }\n\n    if (node.type === 'default-group' && node.color) {\n      if (isDynamicNode(node)) {\n        this.color = node.color\n      } else {\n        this.color.set(node.color)\n      }\n    }\n\n    if (node.type === 'default-group' && node.resizable) {\n      if (isDynamicNode(node)) {\n        this.resizable = node.resizable\n      } else {\n        this.resizable.set(node.resizable)\n      }\n    }\n  }\n\n  public setPoint(point: Point, throttle: boolean) {\n    if (throttle) {\n      this.internalPoint.set(point);\n    } else {\n      this.notThrottledPoint$.next(point)\n    }\n  }\n\n  /**\n   * TODO find the way to implement this better\n   */\n  public linkDefaultNodeSizeWithModelSize() {\n    const node = this.node\n\n    switch (node.type) {\n      case 'default':\n      case 'default-group':\n      case 'template-group': {\n        if (isDynamicNode(node)) {\n          effect(() => {\n            this.size.set({\n              width: node.width?.() ?? NodeModel.defaultWidth,\n              height: node.height?.() ?? NodeModel.defaultHeight,\n            })\n          }, { allowSignalWrites: true })\n        } else {\n          this.size.set({\n            width: node.width ?? NodeModel.defaultWidth,\n            height: node.height ?? NodeModel.defaultHeight\n          })\n        }\n      }\n    }\n\n    if (node.type === 'html-template' || this.isComponentType) {\n      if (isDynamicNode(node)) {\n        effect(() => {\n          if (node.width && node.height) {\n            this.size.set({\n              width: node.width(),\n              height: node.height(),\n            })\n          }\n        }, { allowSignalWrites: true })\n      } else {\n        if (node.width && node.height) {\n          this.size.set({ width: node.width, height: node.height })\n        }\n      }\n    }\n  }\n\n  private createTextSignal(): Signal<string> {\n    const node = this.node\n\n    if (node.type === 'default') {\n      if (isDynamicNode(node)) {\n        return node.text ?? signal('')\n      } else {\n        return signal(node.text ?? '')\n      }\n    }\n\n    return signal('')\n  }\n\n  private createInternalPointSignal() {\n    return isDynamicNode(this.node)\n      ? this.node.point\n      : signal({ x: this.node.point.x, y: this.node.point.y })\n  }\n}\n"]}
175
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node.model.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/models/node.model.ts"],"names":[],"mappings":"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","sourcesContent":["import { Signal, TemplateRef, computed, effect, inject, signal } from '@angular/core'\nimport { DynamicNode, Node, isDynamicNode } from '../interfaces/node.interface'\nimport { isDefined } from '../utils/is-defined'\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop'\nimport { HandleModel } from './handle.model'\nimport { FlowEntity } from '../interfaces/flow-entity.interface'\nimport { FlowSettingsService } from '../services/flow-settings.service'\nimport { animationFrameScheduler, merge, observeOn, Subject, } from 'rxjs'\nimport { Point } from '../interfaces/point.interface'\nimport { CustomNodeComponent } from '../public-components/custom-node/custom-node.component'\nimport { CustomDynamicNodeComponent } from '../public-components/custom-dynamic-node/custom-dynamic-node.component'\nimport { FlowEntitiesService } from '../services/flow-entities.service'\n\n// TODO bad naming around points\nexport class NodeModel<T = unknown> implements FlowEntity {\n  private static defaultWidth = 100\n  private static defaultHeight = 50\n  private static defaultColor = '#1b262c'\n\n  private flowSettingsService = inject(FlowSettingsService)\n  private entitiesService = inject(FlowEntitiesService);\n\n  private internalPoint = this.createInternalPointSignal()\n\n  private throttledPoint$ = toObservable(this.internalPoint).pipe(\n    observeOn(animationFrameScheduler)\n  )\n\n  private notThrottledPoint$ = new Subject<Point>()\n\n  public point = toSignal(\n    merge(this.throttledPoint$, this.notThrottledPoint$),\n    {\n      initialValue: this.internalPoint()\n    }\n  )\n\n  public point$ = this.throttledPoint$;\n\n  public size = signal({ width: 0, height: 0 })\n  public size$ = toObservable(this.size)\n\n  public width = computed(() => this.size().width)\n  public height = computed(() => this.size().height)\n\n  public renderOrder = signal(0)\n\n  public selected = signal(false)\n  public selected$ = toObservable(this.selected)\n\n  public globalPoint = computed(() => {\n    let parent = this.parent()\n    let x = this.point().x\n    let y = this.point().y\n\n    while (parent !== null) {\n      x += parent.point().x\n      y += parent.point().y\n\n      parent = parent.parent()\n    }\n\n    return { x, y }\n  })\n\n  public pointTransform = computed(() =>\n    `translate(${this.globalPoint().x}, ${this.globalPoint().y})`\n  )\n\n  // Now source and handle positions derived from parent flow\n  public sourcePosition = computed(() => this.flowSettingsService.handlePositions().source)\n  public targetPosition = computed(() => this.flowSettingsService.handlePositions().target)\n\n  public handles = signal<HandleModel[]>([])\n\n  public handles$ = toObservable(this.handles)\n\n  public draggable = signal(true)\n\n  public dragHandlesCount = signal(0)\n\n  // disabled for configuration for now\n  public readonly magnetRadius = 20\n\n  // TODO: not sure if we need to statically store it\n  public isComponentType =\n    CustomNodeComponent.isPrototypeOf(this.node.type) ||\n    CustomDynamicNodeComponent.isPrototypeOf(this.node.type)\n\n  // Default node specific thing\n  public text = this.createTextSignal()\n\n  // Component node specific thing\n  public componentTypeInputs = computed(() => {\n    return {\n      node: this.node,\n      _selected: this.selected()\n    }\n  })\n\n  public parent = computed(() =>\n    this.entitiesService.nodes().find(n => n.node.id === this.parentId()) ?? null\n  )\n\n  public children = computed(() =>\n    this.entitiesService.nodes().filter(n => n.parentId() === this.node.id)\n  )\n\n  public color = signal(NodeModel.defaultColor)\n\n  public resizable = signal(false)\n  public resizing = signal(false)\n  public resizerTemplate = signal<TemplateRef<unknown> | null>(null)\n\n  private parentId = signal<string | null>(null)\n\n  constructor(\n    public node: Node<T> | DynamicNode<T>\n  ) {\n    if (isDefined(node.draggable)) {\n      if (isDynamicNode(node)) {\n        this.draggable = node.draggable\n      } else {\n        this.draggable.set(node.draggable)\n      }\n    }\n\n    if (isDefined(node.parentId)) {\n      if (isDynamicNode(node)) {\n        this.parentId = node.parentId\n      } else {\n        this.parentId.set(node.parentId)\n      }\n    }\n\n    if (node.type === 'default-group' && node.color) {\n      if (isDynamicNode(node)) {\n        this.color = node.color\n      } else {\n        this.color.set(node.color)\n      }\n    }\n\n    if (node.type === 'default-group' && node.resizable) {\n      if (isDynamicNode(node)) {\n        this.resizable = node.resizable\n      } else {\n        this.resizable.set(node.resizable)\n      }\n    }\n  }\n\n  public setPoint(point: Point, throttle: boolean) {\n    if (throttle) {\n      this.internalPoint.set(point);\n    } else {\n      this.notThrottledPoint$.next(point)\n    }\n  }\n\n  /**\n   * TODO find the way to implement this better\n   */\n  public linkDefaultNodeSizeWithModelSize() {\n    const node = this.node\n\n    switch (node.type) {\n      case 'default':\n      case 'default-group':\n      case 'template-group': {\n        if (isDynamicNode(node)) {\n          effect(() => {\n            this.size.set({\n              width: node.width?.() ?? NodeModel.defaultWidth,\n              height: node.height?.() ?? NodeModel.defaultHeight,\n            })\n          }, { allowSignalWrites: true })\n        } else {\n          this.size.set({\n            width: node.width ?? NodeModel.defaultWidth,\n            height: node.height ?? NodeModel.defaultHeight\n          })\n        }\n      }\n    }\n\n    if (node.type === 'html-template' || this.isComponentType) {\n      if (isDynamicNode(node)) {\n        effect(() => {\n          if (node.width && node.height) {\n            this.size.set({\n              width: node.width(),\n              height: node.height(),\n            })\n          }\n        }, { allowSignalWrites: true })\n      } else {\n        if (node.width && node.height) {\n          this.size.set({ width: node.width, height: node.height })\n        }\n      }\n    }\n  }\n\n  private createTextSignal(): Signal<string> {\n    const node = this.node\n\n    if (node.type === 'default') {\n      if (isDynamicNode(node)) {\n        return node.text ?? signal('')\n      } else {\n        return signal(node.text ?? '')\n      }\n    }\n\n    return signal('')\n  }\n\n  private createInternalPointSignal() {\n    return isDynamicNode(this.node)\n      ? this.node.point\n      : signal({ x: this.node.point.x, y: this.node.point.y })\n  }\n}\n"]}
@@ -0,0 +1,36 @@
1
+ import { computed, signal } from "@angular/core";
2
+ export class ToolbarModel {
3
+ constructor(node) {
4
+ this.node = node;
5
+ this.position = signal('top');
6
+ this.template = signal(null);
7
+ this.offset = signal(10);
8
+ this.point = computed(() => {
9
+ switch (this.position()) {
10
+ case 'top':
11
+ return {
12
+ x: this.node.size().width / 2 - this.size().width / 2,
13
+ y: -this.size().height - this.offset()
14
+ };
15
+ case 'bottom':
16
+ return {
17
+ x: this.node.size().width / 2 - this.size().width / 2,
18
+ y: this.node.size().height + this.offset()
19
+ };
20
+ case 'left':
21
+ return {
22
+ x: -this.size().width - this.offset(),
23
+ y: this.node.size().height / 2 - this.size().height / 2
24
+ };
25
+ case 'right':
26
+ return {
27
+ x: this.node.size().width + this.offset(),
28
+ y: this.node.size().height / 2 - this.size().height / 2
29
+ };
30
+ }
31
+ });
32
+ this.transform = computed(() => `translate(${this.point().x}, ${this.point().y})`);
33
+ this.size = signal({ width: 0, height: 0 });
34
+ }
35
+ }
36
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9vbGJhci5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9tb2RlbHMvdG9vbGJhci5tb2RlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFVLE1BQU0sRUFBZSxNQUFNLGVBQWUsQ0FBQztBQUt0RSxNQUFNLE9BQU8sWUFBWTtJQW1DdkIsWUFBbUIsSUFBZTtRQUFmLFNBQUksR0FBSixJQUFJLENBQVc7UUFsQzNCLGFBQVEsR0FBRyxNQUFNLENBQVcsS0FBSyxDQUFDLENBQUE7UUFDbEMsYUFBUSxHQUFHLE1BQU0sQ0FBOEIsSUFBSSxDQUFDLENBQUE7UUFFcEQsV0FBTSxHQUFHLE1BQU0sQ0FBQyxFQUFFLENBQUMsQ0FBQTtRQUVuQixVQUFLLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRTtZQUMzQixRQUFRLElBQUksQ0FBQyxRQUFRLEVBQUUsRUFBRTtnQkFDdkIsS0FBSyxLQUFLO29CQUNSLE9BQU87d0JBQ0wsQ0FBQyxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUMsS0FBSyxHQUFHLENBQUMsR0FBRyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUMsS0FBSyxHQUFHLENBQUM7d0JBQ3JELENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQyxNQUFNLEdBQUcsSUFBSSxDQUFDLE1BQU0sRUFBRTtxQkFDdkMsQ0FBQTtnQkFDSCxLQUFLLFFBQVE7b0JBQ1gsT0FBTzt3QkFDTCxDQUFDLEVBQUUsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQyxLQUFLLEdBQUcsQ0FBQyxHQUFHLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQyxLQUFLLEdBQUcsQ0FBQzt3QkFDckQsQ0FBQyxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQyxNQUFNLEVBQUU7cUJBQzNDLENBQUE7Z0JBQ0gsS0FBSyxNQUFNO29CQUNULE9BQU87d0JBQ0wsQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLElBQUksRUFBRSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUMsTUFBTSxFQUFFO3dCQUNyQyxDQUFDLEVBQUUsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxHQUFHLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQyxNQUFNLEdBQUcsQ0FBQztxQkFDeEQsQ0FBQTtnQkFDSCxLQUFLLE9BQU87b0JBQ1YsT0FBTzt3QkFDTCxDQUFDLEVBQUUsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLE1BQU0sRUFBRTt3QkFDekMsQ0FBQyxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUMsTUFBTSxHQUFHLENBQUMsR0FBRyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUMsTUFBTSxHQUFHLENBQUM7cUJBQ3hELENBQUE7YUFDSjtRQUNILENBQUMsQ0FBQyxDQUFBO1FBRUssY0FBUyxHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxhQUFhLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQyxDQUFDLEtBQUssSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUE7UUFFN0UsU0FBSSxHQUFHLE1BQU0sQ0FBQyxFQUFFLEtBQUssRUFBRSxDQUFDLEVBQUUsTUFBTSxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUE7SUFFUCxDQUFDO0NBQ3hDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY29tcHV0ZWQsIGluamVjdCwgc2lnbmFsLCBUZW1wbGF0ZVJlZiB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XG5pbXBvcnQgeyBQb3NpdGlvbiB9IGZyb20gXCIuLi90eXBlcy9wb3NpdGlvbi50eXBlXCI7XG5pbXBvcnQgeyBOb2RlTW9kZWwgfSBmcm9tIFwiLi9ub2RlLm1vZGVsXCI7XG5pbXBvcnQgeyBWaWV3cG9ydFNlcnZpY2UgfSBmcm9tIFwiLi4vc2VydmljZXMvdmlld3BvcnQuc2VydmljZVwiO1xuXG5leHBvcnQgY2xhc3MgVG9vbGJhck1vZGVsIHtcbiAgcHVibGljIHBvc2l0aW9uID0gc2lnbmFsPFBvc2l0aW9uPigndG9wJylcbiAgcHVibGljIHRlbXBsYXRlID0gc2lnbmFsPFRlbXBsYXRlUmVmPHVua25vd24+IHwgbnVsbD4obnVsbClcblxuICBwdWJsaWMgb2Zmc2V0ID0gc2lnbmFsKDEwKVxuXG4gIHB1YmxpYyBwb2ludCA9IGNvbXB1dGVkKCgpID0+IHtcbiAgICBzd2l0Y2ggKHRoaXMucG9zaXRpb24oKSkge1xuICAgICAgY2FzZSAndG9wJzpcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICB4OiB0aGlzLm5vZGUuc2l6ZSgpLndpZHRoIC8gMiAtIHRoaXMuc2l6ZSgpLndpZHRoIC8gMixcbiAgICAgICAgICB5OiAtdGhpcy5zaXplKCkuaGVpZ2h0IC0gdGhpcy5vZmZzZXQoKVxuICAgICAgICB9XG4gICAgICBjYXNlICdib3R0b20nOlxuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIHg6IHRoaXMubm9kZS5zaXplKCkud2lkdGggLyAyIC0gdGhpcy5zaXplKCkud2lkdGggLyAyLFxuICAgICAgICAgIHk6IHRoaXMubm9kZS5zaXplKCkuaGVpZ2h0ICsgdGhpcy5vZmZzZXQoKVxuICAgICAgICB9XG4gICAgICBjYXNlICdsZWZ0JzpcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICB4OiAtdGhpcy5zaXplKCkud2lkdGggLSB0aGlzLm9mZnNldCgpLFxuICAgICAgICAgIHk6IHRoaXMubm9kZS5zaXplKCkuaGVpZ2h0IC8gMiAtIHRoaXMuc2l6ZSgpLmhlaWdodCAvIDJcbiAgICAgICAgfVxuICAgICAgY2FzZSAncmlnaHQnOlxuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIHg6IHRoaXMubm9kZS5zaXplKCkud2lkdGggKyB0aGlzLm9mZnNldCgpLFxuICAgICAgICAgIHk6IHRoaXMubm9kZS5zaXplKCkuaGVpZ2h0IC8gMiAtIHRoaXMuc2l6ZSgpLmhlaWdodCAvIDJcbiAgICAgICAgfVxuICAgIH1cbiAgfSlcblxuICBwdWJsaWMgdHJhbnNmb3JtID0gY29tcHV0ZWQoKCkgPT4gYHRyYW5zbGF0ZSgke3RoaXMucG9pbnQoKS54fSwgJHt0aGlzLnBvaW50KCkueX0pYClcblxuICBwdWJsaWMgc2l6ZSA9IHNpZ25hbCh7IHdpZHRoOiAwLCBoZWlnaHQ6IDAgfSlcblxuICBjb25zdHJ1Y3RvcihwdWJsaWMgbm9kZTogTm9kZU1vZGVsKSB7IH1cbn1cbiJdfQ==
@@ -99,7 +99,7 @@ export class MiniMapComponent {
99
99
  return node;
100
100
  }
101
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"] }] }); }
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.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i2.DefaultNodeComponent, selector: "default-node", inputs: ["selected"] }] }); }
103
103
  }
104
104
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: MiniMapComponent, decorators: [{
105
105
  type: Component,
@@ -0,0 +1,66 @@
1
+ import { ChangeDetectionStrategy, Component, ElementRef, inject, Input, ViewChild } from '@angular/core';
2
+ import { Directive } from '@angular/core';
3
+ import { ToolbarModel } from '../../models/toolbar.model';
4
+ import { OverlaysService } from '../../services/overlays.service';
5
+ import { NodeAccessorService } from '../../services/node-accessor.service';
6
+ import * as i0 from "@angular/core";
7
+ export class NodeToolbarComponent {
8
+ constructor() {
9
+ this.overlaysService = inject(OverlaysService);
10
+ this.nodeService = inject(NodeAccessorService);
11
+ this.model = new ToolbarModel(this.nodeService.model());
12
+ }
13
+ set position(value) {
14
+ this.model.position.set(value);
15
+ }
16
+ ngOnInit() {
17
+ this.model.template.set(this.toolbarContentTemplate);
18
+ this.overlaysService.addToolbar(this.model);
19
+ }
20
+ ngOnDestroy() {
21
+ this.overlaysService.removeToolbar(this.model);
22
+ }
23
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NodeToolbarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
24
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: NodeToolbarComponent, selector: "node-toolbar", inputs: { position: "position" }, viewQueries: [{ propertyName: "toolbarContentTemplate", first: true, predicate: ["toolbar"], descendants: true, static: true }], ngImport: i0, template: `
25
+ <ng-template #toolbar>
26
+ <div class="wrapper" nodeToolbarWrapper [model]="model">
27
+ <ng-content />
28
+ </div>
29
+ </ng-template>
30
+ `, isInline: true, styles: [".wrapper{width:max-content}\n"], dependencies: [{ kind: "directive", type: i0.forwardRef(function () { return NodeToolbarWrapperDirective; }), selector: "[nodeToolbarWrapper]", inputs: ["model"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
31
+ }
32
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NodeToolbarComponent, decorators: [{
33
+ type: Component,
34
+ args: [{ selector: 'node-toolbar', template: `
35
+ <ng-template #toolbar>
36
+ <div class="wrapper" nodeToolbarWrapper [model]="model">
37
+ <ng-content />
38
+ </div>
39
+ </ng-template>
40
+ `, changeDetection: ChangeDetectionStrategy.OnPush, styles: [".wrapper{width:max-content}\n"] }]
41
+ }], propDecorators: { position: [{
42
+ type: Input
43
+ }], toolbarContentTemplate: [{
44
+ type: ViewChild,
45
+ args: ['toolbar', { static: true }]
46
+ }] } });
47
+ export class NodeToolbarWrapperDirective {
48
+ constructor() {
49
+ this.element = inject(ElementRef);
50
+ }
51
+ ngOnInit() {
52
+ this.model.size.set({
53
+ width: this.element.nativeElement.clientWidth,
54
+ height: this.element.nativeElement.clientHeight
55
+ });
56
+ }
57
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NodeToolbarWrapperDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
58
+ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: NodeToolbarWrapperDirective, selector: "[nodeToolbarWrapper]", inputs: { model: "model" }, ngImport: i0 }); }
59
+ }
60
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: NodeToolbarWrapperDirective, decorators: [{
61
+ type: Directive,
62
+ args: [{ selector: '[nodeToolbarWrapper]' }]
63
+ }], propDecorators: { model: [{
64
+ type: Input
65
+ }] } });
66
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibm9kZS10b29sYmFyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC12Zmxvdy1saWIvc3JjL2xpYi92Zmxvdy9wdWJsaWMtY29tcG9uZW50cy9ub2RlLXRvb2xiYXIvbm9kZS10b29sYmFyLmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQXFCLFNBQVMsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFLEtBQUssRUFBa0MsU0FBUyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzVKLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFMUMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBQzFELE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUNsRSxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxzQ0FBc0MsQ0FBQzs7QUFrQjNFLE1BQU0sT0FBTyxvQkFBb0I7SUFoQmpDO1FBaUJVLG9CQUFlLEdBQUcsTUFBTSxDQUFDLGVBQWUsQ0FBQyxDQUFDO1FBQzFDLGdCQUFXLEdBQUcsTUFBTSxDQUFDLG1CQUFtQixDQUFDLENBQUE7UUFVdkMsVUFBSyxHQUFHLElBQUksWUFBWSxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsS0FBSyxFQUFHLENBQUMsQ0FBQTtLQVc5RDtJQW5CQyxJQUNXLFFBQVEsQ0FBQyxLQUFlO1FBQ2pDLElBQUksQ0FBQyxLQUFLLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUNqQyxDQUFDO0lBT00sUUFBUTtRQUNiLElBQUksQ0FBQyxLQUFLLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsc0JBQXNCLENBQUMsQ0FBQTtRQUVwRCxJQUFJLENBQUMsZUFBZSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUE7SUFDN0MsQ0FBQztJQUVNLFdBQVc7UUFDaEIsSUFBSSxDQUFDLGVBQWUsQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFBO0lBQ2hELENBQUM7K0dBdEJVLG9CQUFvQjttR0FBcEIsb0JBQW9CLHVOQWRyQjs7Ozs7O0dBTVQsMElBa0NVLDJCQUEyQjs7NEZBMUIzQixvQkFBb0I7a0JBaEJoQyxTQUFTOytCQUNFLGNBQWMsWUFDZDs7Ozs7O0dBTVQsbUJBTWdCLHVCQUF1QixDQUFDLE1BQU07OEJBT3BDLFFBQVE7c0JBRGxCLEtBQUs7Z0JBTUMsc0JBQXNCO3NCQUQ1QixTQUFTO3VCQUFDLFNBQVMsRUFBRSxFQUFFLE1BQU0sRUFBRSxJQUFJLEVBQUU7O0FBaUJ4QyxNQUFNLE9BQU8sMkJBQTJCO0lBRHhDO1FBRVUsWUFBTyxHQUFHLE1BQU0sQ0FBMEIsVUFBVSxDQUFDLENBQUE7S0FXOUQ7SUFOUSxRQUFRO1FBQ2IsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDO1lBQ2xCLEtBQUssRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxXQUFXO1lBQzdDLE1BQU0sRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxZQUFZO1NBQ2hELENBQUMsQ0FBQTtJQUNKLENBQUM7K0dBWFUsMkJBQTJCO21HQUEzQiwyQkFBMkI7OzRGQUEzQiwyQkFBMkI7a0JBRHZDLFNBQVM7bUJBQUMsRUFBRSxRQUFRLEVBQUUsc0JBQXNCLEVBQUU7OEJBS3RDLEtBQUs7c0JBRFgsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBDaGFuZ2VEZXRlY3RvclJlZiwgQ29tcG9uZW50LCBFbGVtZW50UmVmLCBpbmplY3QsIElucHV0LCBPbkRlc3Ryb3ksIE9uSW5pdCwgVGVtcGxhdGVSZWYsIFZpZXdDaGlsZCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRGlyZWN0aXZlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBQb3NpdGlvbiB9IGZyb20gJy4uLy4uL3R5cGVzL3Bvc2l0aW9uLnR5cGUnO1xuaW1wb3J0IHsgVG9vbGJhck1vZGVsIH0gZnJvbSAnLi4vLi4vbW9kZWxzL3Rvb2xiYXIubW9kZWwnO1xuaW1wb3J0IHsgT3ZlcmxheXNTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvb3ZlcmxheXMuc2VydmljZSc7XG5pbXBvcnQgeyBOb2RlQWNjZXNzb3JTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvbm9kZS1hY2Nlc3Nvci5zZXJ2aWNlJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnbm9kZS10b29sYmFyJyxcbiAgdGVtcGxhdGU6IGBcbiAgICA8bmctdGVtcGxhdGUgI3Rvb2xiYXI+XG4gICAgICA8ZGl2IGNsYXNzPVwid3JhcHBlclwiIG5vZGVUb29sYmFyV3JhcHBlciBbbW9kZWxdPVwibW9kZWxcIj5cbiAgICAgICAgPG5nLWNvbnRlbnQgLz5cbiAgICAgIDwvZGl2PlxuICAgIDwvbmctdGVtcGxhdGU+XG4gIGAsXG4gIHN0eWxlczogW2BcbiAgICAud3JhcHBlciB7XG4gICAgICB3aWR0aDogbWF4LWNvbnRlbnQ7XG4gICAgfVxuICBgXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG59KVxuZXhwb3J0IGNsYXNzIE5vZGVUb29sYmFyQ29tcG9uZW50IGltcGxlbWVudHMgT25Jbml0LCBPbkRlc3Ryb3kge1xuICBwcml2YXRlIG92ZXJsYXlzU2VydmljZSA9IGluamVjdChPdmVybGF5c1NlcnZpY2UpO1xuICBwcml2YXRlIG5vZGVTZXJ2aWNlID0gaW5qZWN0KE5vZGVBY2Nlc3NvclNlcnZpY2UpXG5cbiAgQElucHV0KClcbiAgcHVibGljIHNldCBwb3NpdGlvbih2YWx1ZTogUG9zaXRpb24pIHtcbiAgICB0aGlzLm1vZGVsLnBvc2l0aW9uLnNldCh2YWx1ZSk7XG4gIH1cblxuICBAVmlld0NoaWxkKCd0b29sYmFyJywgeyBzdGF0aWM6IHRydWUgfSlcbiAgcHVibGljIHRvb2xiYXJDb250ZW50VGVtcGxhdGUhOiBUZW1wbGF0ZVJlZjx1bmtub3duPjtcblxuICBwcm90ZWN0ZWQgbW9kZWwgPSBuZXcgVG9vbGJhck1vZGVsKHRoaXMubm9kZVNlcnZpY2UubW9kZWwoKSEpXG5cbiAgcHVibGljIG5nT25Jbml0KCk6IHZvaWQge1xuICAgIHRoaXMubW9kZWwudGVtcGxhdGUuc2V0KHRoaXMudG9vbGJhckNvbnRlbnRUZW1wbGF0ZSlcblxuICAgIHRoaXMub3ZlcmxheXNTZXJ2aWNlLmFkZFRvb2xiYXIodGhpcy5tb2RlbClcbiAgfVxuXG4gIHB1YmxpYyBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAgICB0aGlzLm92ZXJsYXlzU2VydmljZS5yZW1vdmVUb29sYmFyKHRoaXMubW9kZWwpXG4gIH1cbn1cblxuQERpcmVjdGl2ZSh7IHNlbGVjdG9yOiAnW25vZGVUb29sYmFyV3JhcHBlcl0nIH0pXG5leHBvcnQgY2xhc3MgTm9kZVRvb2xiYXJXcmFwcGVyRGlyZWN0aXZlIGltcGxlbWVudHMgT25Jbml0IHtcbiAgcHJpdmF0ZSBlbGVtZW50ID0gaW5qZWN0PEVsZW1lbnRSZWY8SFRNTEVsZW1lbnQ+PihFbGVtZW50UmVmKVxuXG4gIEBJbnB1dCgpXG4gIHB1YmxpYyBtb2RlbCE6IFRvb2xiYXJNb2RlbFxuXG4gIHB1YmxpYyBuZ09uSW5pdCgpOiB2b2lkIHtcbiAgICB0aGlzLm1vZGVsLnNpemUuc2V0KHtcbiAgICAgIHdpZHRoOiB0aGlzLmVsZW1lbnQubmF0aXZlRWxlbWVudC5jbGllbnRXaWR0aCxcbiAgICAgIGhlaWdodDogdGhpcy5lbGVtZW50Lm5hdGl2ZUVsZW1lbnQuY2xpZW50SGVpZ2h0XG4gICAgfSlcbiAgfVxufVxuIl19