ngx-vflow 1.16.0 → 1.16.2

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.
@@ -1,12 +1,12 @@
1
1
  import { Injectable, computed, inject } from '@angular/core';
2
2
  import { FlowEntitiesService } from './flow-entities.service';
3
- import { isGroupNode } from '../utils/is-group-node';
4
3
  import { FlowSettingsService } from './flow-settings.service';
5
4
  import { isRectInViewport } from '../utils/viewport';
6
5
  import { ViewportService } from './viewport.service';
7
6
  import { toObservable } from '@angular/core/rxjs-interop';
8
7
  import { asyncScheduler, debounceTime, filter, map, merge, observeOn } from 'rxjs';
9
8
  import { toLazySignal } from '../utils/signals/to-lazy-signal';
9
+ import { isGroupNode } from '../utils/is-group-node';
10
10
  import * as i0 from "@angular/core";
11
11
  export class NodeRenderingService {
12
12
  constructor() {
@@ -20,10 +20,10 @@ export class NodeRenderingService {
20
20
  return this.viewportNodesAfterInteraction().sort((aNode, bNode) => aNode.renderOrder() - bNode.renderOrder());
21
21
  });
22
22
  this.groups = computed(() => {
23
- return this.nodes().filter((n) => isGroupNode(n));
23
+ return this.nodes().filter((n) => !!n.children().length || isGroupNode(n));
24
24
  });
25
25
  this.nonGroups = computed(() => {
26
- return this.nodes().filter((n) => !isGroupNode(n));
26
+ return this.nodes().filter((n) => !this.groups().includes(n));
27
27
  });
28
28
  this.viewportNodes = computed(() => {
29
29
  const nodes = this.flowEntitiesService.nodes();
@@ -63,4 +63,4 @@ export class NodeRenderingService {
63
63
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NodeRenderingService, decorators: [{
64
64
  type: Injectable
65
65
  }] });
66
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-rendering.service.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/services/node-rendering.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, computed, inject } from '@angular/core';\nimport { FlowEntitiesService } from './flow-entities.service';\nimport { NodeModel } from '../models/node.model';\nimport { isGroupNode } from '../utils/is-group-node';\nimport { FlowSettingsService } from './flow-settings.service';\nimport { isRectInViewport } from '../utils/viewport';\nimport { ViewportService } from './viewport.service';\nimport { toObservable } from '@angular/core/rxjs-interop';\nimport { asyncScheduler, debounceTime, filter, map, merge, observeOn } from 'rxjs';\nimport { toLazySignal } from '../utils/signals/to-lazy-signal';\n\n@Injectable()\nexport class NodeRenderingService {\n  private flowEntitiesService = inject(FlowEntitiesService);\n  private flowSettingsService = inject(FlowSettingsService);\n  private viewportService = inject(ViewportService);\n\n  public readonly nodes = computed(() => {\n    if (!this.flowSettingsService.optimization().virtualization) {\n      return [...this.flowEntitiesService.nodes()].sort((aNode, bNode) => aNode.renderOrder() - bNode.renderOrder());\n    }\n\n    return this.viewportNodesAfterInteraction().sort((aNode, bNode) => aNode.renderOrder() - bNode.renderOrder());\n  });\n\n  public readonly groups = computed(() => {\n    return this.nodes().filter((n) => isGroupNode(n));\n  });\n\n  public readonly nonGroups = computed(() => {\n    return this.nodes().filter((n) => !isGroupNode(n));\n  });\n\n  public viewportNodes = computed(() => {\n    const nodes = this.flowEntitiesService.nodes();\n    const viewport = this.viewportService.readableViewport();\n    const flowWidth = this.flowSettingsService.computedFlowWidth();\n    const flowHeight = this.flowSettingsService.computedFlowHeight();\n\n    return nodes.filter((n) => {\n      const { x, y } = n.globalPoint();\n      const width = n.width();\n      const height = n.height();\n\n      return isRectInViewport({ x, y, width, height }, viewport, flowWidth, flowHeight);\n    });\n  });\n\n  private viewportNodesAfterInteraction = toLazySignal(\n    merge(\n      // TODO: maybe there is a better way wait when viewport is ready?\n      // (to correctly calculate viewport nodes on first render)\n      toObservable(this.flowEntitiesService.nodes).pipe(\n        observeOn(asyncScheduler),\n        filter((nodes) => !!nodes.length),\n      ),\n\n      this.viewportService.viewportChangeEnd$.pipe(debounceTime(300)),\n    ).pipe(\n      map(() => {\n        const viewport = this.viewportService.readableViewport();\n        const zoomThreshold = this.flowSettingsService.optimization().virtualizationZoomThreshold;\n\n        return viewport.zoom < zoomThreshold ? [] : this.viewportNodes();\n      }),\n    ),\n    {\n      initialValue: [],\n    },\n  );\n\n  private maxOrder = computed(() => {\n    return Math.max(...this.flowEntitiesService.nodes().map((n) => n.renderOrder()));\n  });\n\n  public pullNode(node: NodeModel) {\n    // pull node\n    node.renderOrder.set(this.maxOrder() + 1);\n\n    // pull children\n    node.children().forEach((n) => this.pullNode(n));\n  }\n}\n"]}
66
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"node-rendering.service.js","sourceRoot":"","sources":["../../../../../../projects/ngx-vflow-lib/src/lib/vflow/services/node-rendering.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, computed, inject } from '@angular/core';\nimport { FlowEntitiesService } from './flow-entities.service';\nimport { NodeModel } from '../models/node.model';\nimport { FlowSettingsService } from './flow-settings.service';\nimport { isRectInViewport } from '../utils/viewport';\nimport { ViewportService } from './viewport.service';\nimport { toObservable } from '@angular/core/rxjs-interop';\nimport { asyncScheduler, debounceTime, filter, map, merge, observeOn } from 'rxjs';\nimport { toLazySignal } from '../utils/signals/to-lazy-signal';\nimport { isGroupNode } from '../utils/is-group-node';\n\n@Injectable()\nexport class NodeRenderingService {\n  private flowEntitiesService = inject(FlowEntitiesService);\n  private flowSettingsService = inject(FlowSettingsService);\n  private viewportService = inject(ViewportService);\n\n  public readonly nodes = computed(() => {\n    if (!this.flowSettingsService.optimization().virtualization) {\n      return [...this.flowEntitiesService.nodes()].sort((aNode, bNode) => aNode.renderOrder() - bNode.renderOrder());\n    }\n\n    return this.viewportNodesAfterInteraction().sort((aNode, bNode) => aNode.renderOrder() - bNode.renderOrder());\n  });\n\n  public readonly groups = computed(() => {\n    return this.nodes().filter((n) => !!n.children().length || isGroupNode(n));\n  });\n\n  public readonly nonGroups = computed(() => {\n    return this.nodes().filter((n) => !this.groups().includes(n));\n  });\n\n  public viewportNodes = computed(() => {\n    const nodes = this.flowEntitiesService.nodes();\n    const viewport = this.viewportService.readableViewport();\n    const flowWidth = this.flowSettingsService.computedFlowWidth();\n    const flowHeight = this.flowSettingsService.computedFlowHeight();\n\n    return nodes.filter((n) => {\n      const { x, y } = n.globalPoint();\n      const width = n.width();\n      const height = n.height();\n\n      return isRectInViewport({ x, y, width, height }, viewport, flowWidth, flowHeight);\n    });\n  });\n\n  private viewportNodesAfterInteraction = toLazySignal(\n    merge(\n      // TODO: maybe there is a better way wait when viewport is ready?\n      // (to correctly calculate viewport nodes on first render)\n      toObservable(this.flowEntitiesService.nodes).pipe(\n        observeOn(asyncScheduler),\n        filter((nodes) => !!nodes.length),\n      ),\n\n      this.viewportService.viewportChangeEnd$.pipe(debounceTime(300)),\n    ).pipe(\n      map(() => {\n        const viewport = this.viewportService.readableViewport();\n        const zoomThreshold = this.flowSettingsService.optimization().virtualizationZoomThreshold;\n\n        return viewport.zoom < zoomThreshold ? [] : this.viewportNodes();\n      }),\n    ),\n    {\n      initialValue: [],\n    },\n  );\n\n  private maxOrder = computed(() => {\n    return Math.max(...this.flowEntitiesService.nodes().map((n) => n.renderOrder()));\n  });\n\n  public pullNode(node: NodeModel) {\n    // pull node\n    node.renderOrder.set(this.maxOrder() + 1);\n\n    // pull children\n    node.children().forEach((n) => this.pullNode(n));\n  }\n}\n"]}
@@ -1061,10 +1061,6 @@ function toSignalProperties(obj) {
1061
1061
  return newObj;
1062
1062
  }
1063
1063
 
1064
- function isGroupNode(node) {
1065
- return node.rawNode.type === 'default-group' || node.rawNode.type === 'template-group';
1066
- }
1067
-
1068
1064
  // MIT License
1069
1065
  // Copyright (c) 2023 Chau Tran
1070
1066
  // Permission is hereby granted, free of charge, to any person obtaining a copy
@@ -1123,6 +1119,10 @@ function toLazySignal(source, options) {
1123
1119
  });
1124
1120
  }
1125
1121
 
1122
+ function isGroupNode(node) {
1123
+ return node.rawNode.type === 'default-group' || node.rawNode.type === 'template-group';
1124
+ }
1125
+
1126
1126
  class NodeRenderingService {
1127
1127
  constructor() {
1128
1128
  this.flowEntitiesService = inject(FlowEntitiesService);
@@ -1135,10 +1135,10 @@ class NodeRenderingService {
1135
1135
  return this.viewportNodesAfterInteraction().sort((aNode, bNode) => aNode.renderOrder() - bNode.renderOrder());
1136
1136
  });
1137
1137
  this.groups = computed(() => {
1138
- return this.nodes().filter((n) => isGroupNode(n));
1138
+ return this.nodes().filter((n) => !!n.children().length || isGroupNode(n));
1139
1139
  });
1140
1140
  this.nonGroups = computed(() => {
1141
- return this.nodes().filter((n) => !isGroupNode(n));
1141
+ return this.nodes().filter((n) => !this.groups().includes(n));
1142
1142
  });
1143
1143
  this.viewportNodes = computed(() => {
1144
1144
  const nodes = this.flowEntitiesService.nodes();