ngx-vflow 1.16.3 → 2.0.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.
- package/README.md +35 -69
- package/fesm2022/ngx-vflow-testing.mjs +77 -67
- package/fesm2022/ngx-vflow-testing.mjs.map +1 -1
- package/fesm2022/ngx-vflow.mjs +926 -559
- package/fesm2022/ngx-vflow.mjs.map +1 -1
- package/lib/vflow/components/custom-node-base/custom-node-base.component.d.ts +1 -1
- package/lib/vflow/components/vflow/vflow.component.d.ts +18 -9
- package/lib/vflow/directives/auto-pan.directive.d.ts +16 -0
- package/lib/vflow/directives/changes-controller.directive.d.ts +13 -87
- package/lib/vflow/directives/connection-controller.directive.d.ts +8 -10
- package/lib/vflow/directives/space-point-context.directive.d.ts +2 -1
- package/lib/vflow/interfaces/component-node-event.interface.d.ts +1 -2
- package/lib/vflow/interfaces/connection-events.interface.d.ts +66 -0
- package/lib/vflow/interfaces/connection-settings.interface.d.ts +7 -1
- package/lib/vflow/interfaces/connection.interface.d.ts +0 -5
- package/lib/vflow/interfaces/curve-factory.interface.d.ts +2 -2
- package/lib/vflow/interfaces/edge-label.interface.d.ts +0 -4
- package/lib/vflow/interfaces/edge.interface.d.ts +36 -8
- package/lib/vflow/interfaces/node.interface.d.ts +44 -63
- package/lib/vflow/interfaces/selection-strategy.interface.d.ts +16 -0
- package/lib/vflow/interfaces/template-context.interface.d.ts +8 -3
- package/lib/vflow/models/connection.model.d.ts +1 -1
- package/lib/vflow/models/edge.model.d.ts +20 -23
- package/lib/vflow/models/handle.model.d.ts +1 -1
- package/lib/vflow/models/node.model.d.ts +4 -7
- package/lib/vflow/public-components/custom-node/custom-node.component.d.ts +1 -1
- package/lib/vflow/public-components/custom-template-edge/custom-template-edge.component.d.ts +2 -1
- package/lib/vflow/public-components/minimap/minimap.component.d.ts +1 -1
- package/lib/vflow/services/draggable.service.d.ts +3 -0
- package/lib/vflow/services/flow-entities.service.d.ts +2 -2
- package/lib/vflow/services/flow-settings.service.d.ts +3 -0
- package/lib/vflow/services/flow-status.service.d.ts +35 -7
- package/lib/vflow/services/selection.service.d.ts +4 -1
- package/lib/vflow/strategies/default-selection.strategy.d.ts +6 -0
- package/lib/vflow/strategies/manual-selection.strategy.d.ts +5 -0
- package/lib/vflow/types/selection-mode.type.d.ts +1 -0
- package/lib/vflow/types/unwrap-signal.type.d.ts +4 -0
- package/lib/vflow/utils/adjust-direction.d.ts +2 -6
- package/lib/vflow/utils/identity-checker/reference-identity-checker.d.ts +2 -2
- package/lib/vflow/utils/is-vflow-component.d.ts +0 -2
- package/lib/vflow/utils/signals/to-lazy-signal.d.ts +4 -4
- package/package.json +3 -7
- package/public-api.d.ts +3 -2
- package/testing/component-mocks/vflow-mock.component.d.ts +8 -6
- package/testing/directive-mocks/connection-controller-mock.directive.d.ts +8 -6
- package/esm2022/lib/vflow/components/alignment-helper/alignment-helper.component.mjs +0 -103
- package/esm2022/lib/vflow/components/background/background.component.mjs +0 -121
- package/esm2022/lib/vflow/components/connection/connection.component.mjs +0 -157
- package/esm2022/lib/vflow/components/custom-node-base/custom-node-base.component.mjs +0 -57
- package/esm2022/lib/vflow/components/default-node/default-node.component.mjs +0 -16
- package/esm2022/lib/vflow/components/defs/defs.component.mjs +0 -16
- package/esm2022/lib/vflow/components/edge/edge.component.mjs +0 -54
- package/esm2022/lib/vflow/components/edge-label/edge-label.component.mjs +0 -76
- package/esm2022/lib/vflow/components/node/node.component.mjs +0 -107
- package/esm2022/lib/vflow/components/preview-flow/draw-node.mjs +0 -100
- package/esm2022/lib/vflow/components/preview-flow/preview-flow.component.mjs +0 -62
- package/esm2022/lib/vflow/components/vflow/vflow.component.mjs +0 -437
- package/esm2022/lib/vflow/constants/magic-number-to-fix-glitch-in-chrome.constant.mjs +0 -4
- package/esm2022/lib/vflow/decorators/microtask.decorator.mjs +0 -11
- package/esm2022/lib/vflow/directives/changes-controller.directive.mjs +0 -165
- package/esm2022/lib/vflow/directives/connection-controller.directive.mjs +0 -145
- package/esm2022/lib/vflow/directives/drag-handle.directive.mjs +0 -28
- package/esm2022/lib/vflow/directives/flow-size-controller.directive.mjs +0 -40
- package/esm2022/lib/vflow/directives/handle-size-controller.directive.mjs +0 -41
- package/esm2022/lib/vflow/directives/map-context.directive.mjs +0 -116
- package/esm2022/lib/vflow/directives/node-handles-controller.directive.mjs +0 -33
- package/esm2022/lib/vflow/directives/node-resize-controller.directive.mjs +0 -37
- package/esm2022/lib/vflow/directives/pointer.directive.mjs +0 -84
- package/esm2022/lib/vflow/directives/reference.directive.mjs +0 -17
- package/esm2022/lib/vflow/directives/root-pointer.directive.mjs +0 -58
- package/esm2022/lib/vflow/directives/root-svg-context.directive.mjs +0 -35
- package/esm2022/lib/vflow/directives/selectable.directive.mjs +0 -48
- package/esm2022/lib/vflow/directives/space-point-context.directive.mjs +0 -42
- package/esm2022/lib/vflow/directives/template.directive.mjs +0 -119
- package/esm2022/lib/vflow/interfaces/alignment-helper-settings.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/box.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/component-node-event.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/connection-settings.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/connection.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/connection.internal.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/contextable.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/curve-factory.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/edge-label.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/edge.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/fit-view-options.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/flow-entity.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/intersecting-nodes-options.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/marker.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/node-preview.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/node.interface.mjs +0 -53
- package/esm2022/lib/vflow/interfaces/optimization.interface.mjs +0 -7
- package/esm2022/lib/vflow/interfaces/point.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/rect.mjs +0 -10
- package/esm2022/lib/vflow/interfaces/template-context.interface.mjs +0 -2
- package/esm2022/lib/vflow/interfaces/viewport.interface.mjs +0 -2
- package/esm2022/lib/vflow/math/edge-path/bezier-path.mjs +0 -66
- package/esm2022/lib/vflow/math/edge-path/smooth-step-path.mjs +0 -226
- package/esm2022/lib/vflow/math/edge-path/straigh-path.mjs +0 -12
- package/esm2022/lib/vflow/math/point-on-line-by-ratio.mjs +0 -12
- package/esm2022/lib/vflow/models/connection.model.mjs +0 -31
- package/esm2022/lib/vflow/models/edge-label.model.mjs +0 -8
- package/esm2022/lib/vflow/models/edge.model.mjs +0 -208
- package/esm2022/lib/vflow/models/handle.model.mjs +0 -98
- package/esm2022/lib/vflow/models/minimap.model.mjs +0 -7
- package/esm2022/lib/vflow/models/node.model.mjs +0 -181
- package/esm2022/lib/vflow/models/toolbar.model.mjs +0 -36
- package/esm2022/lib/vflow/public-components/custom-dynamic-node/custom-dynamic-node.component.mjs +0 -25
- package/esm2022/lib/vflow/public-components/custom-node/custom-node.component.mjs +0 -24
- package/esm2022/lib/vflow/public-components/custom-template-edge/custom-template-edge.component.mjs +0 -29
- package/esm2022/lib/vflow/public-components/handle/handle.component.mjs +0 -53
- package/esm2022/lib/vflow/public-components/minimap/minimap.component.mjs +0 -108
- package/esm2022/lib/vflow/public-components/node-toolbar/node-toolbar.component.mjs +0 -73
- package/esm2022/lib/vflow/public-components/resizable/resizable.component.mjs +0 -272
- package/esm2022/lib/vflow/services/component-event-bus.service.mjs +0 -18
- package/esm2022/lib/vflow/services/draggable.service.mjs +0 -124
- package/esm2022/lib/vflow/services/edge-changes.service.mjs +0 -43
- package/esm2022/lib/vflow/services/edge-rendering.service.mjs +0 -40
- package/esm2022/lib/vflow/services/flow-entities.service.mjs +0 -56
- package/esm2022/lib/vflow/services/flow-rendering.service.mjs +0 -41
- package/esm2022/lib/vflow/services/flow-settings.service.mjs +0 -33
- package/esm2022/lib/vflow/services/flow-status.service.mjs +0 -49
- package/esm2022/lib/vflow/services/handle.service.mjs +0 -30
- package/esm2022/lib/vflow/services/keyboard.service.mjs +0 -47
- package/esm2022/lib/vflow/services/node-accessor.service.mjs +0 -16
- package/esm2022/lib/vflow/services/node-changes.service.mjs +0 -41
- package/esm2022/lib/vflow/services/node-rendering.service.mjs +0 -66
- package/esm2022/lib/vflow/services/overlays.service.mjs +0 -35
- package/esm2022/lib/vflow/services/preview-flow-render-strategy.service.mjs +0 -21
- package/esm2022/lib/vflow/services/selection.service.mjs +0 -54
- package/esm2022/lib/vflow/services/viewport.service.mjs +0 -64
- package/esm2022/lib/vflow/types/background.type.mjs +0 -2
- package/esm2022/lib/vflow/types/connection-mode.type.mjs +0 -2
- package/esm2022/lib/vflow/types/edge-change.type.mjs +0 -2
- package/esm2022/lib/vflow/types/handle-type.type.mjs +0 -2
- package/esm2022/lib/vflow/types/keyboard-action.type.mjs +0 -2
- package/esm2022/lib/vflow/types/node-change.type.mjs +0 -2
- package/esm2022/lib/vflow/types/position.type.mjs +0 -2
- package/esm2022/lib/vflow/types/viewport-change-type.type.mjs +0 -2
- package/esm2022/lib/vflow/utils/add-nodes-to-edges.mjs +0 -11
- package/esm2022/lib/vflow/utils/adjust-direction.mjs +0 -30
- package/esm2022/lib/vflow/utils/align-number.mjs +0 -4
- package/esm2022/lib/vflow/utils/assert-injector.mjs +0 -27
- package/esm2022/lib/vflow/utils/event.mjs +0 -4
- package/esm2022/lib/vflow/utils/get-os.mjs +0 -24
- package/esm2022/lib/vflow/utils/get-overlapping-area.mjs +0 -6
- package/esm2022/lib/vflow/utils/get-space-points.mjs +0 -25
- package/esm2022/lib/vflow/utils/hash.mjs +0 -7
- package/esm2022/lib/vflow/utils/id.mjs +0 -5
- package/esm2022/lib/vflow/utils/identity-checker/reference-identity-checker.mjs +0 -28
- package/esm2022/lib/vflow/utils/is-callable.mjs +0 -10
- package/esm2022/lib/vflow/utils/is-defined.mjs +0 -4
- package/esm2022/lib/vflow/utils/is-group-node.mjs +0 -4
- package/esm2022/lib/vflow/utils/is-vflow-component.mjs +0 -9
- package/esm2022/lib/vflow/utils/nodes.mjs +0 -60
- package/esm2022/lib/vflow/utils/resizable.mjs +0 -11
- package/esm2022/lib/vflow/utils/round.mjs +0 -2
- package/esm2022/lib/vflow/utils/signals/extended-computed.mjs +0 -15
- package/esm2022/lib/vflow/utils/signals/to-lazy-signal.mjs +0 -35
- package/esm2022/lib/vflow/utils/to-unified-node.mjs +0 -24
- package/esm2022/lib/vflow/utils/transform-background.mjs +0 -4
- package/esm2022/lib/vflow/utils/viewport.mjs +0 -51
- package/esm2022/lib/vflow/vflow.mjs +0 -29
- package/esm2022/ngx-vflow.mjs +0 -5
- package/esm2022/public-api.mjs +0 -55
- package/esm2022/testing/component-mocks/custom-template-edge-mock.component.mjs +0 -16
- package/esm2022/testing/component-mocks/handle-mock.component.mjs +0 -26
- package/esm2022/testing/component-mocks/minimap-mock.component.mjs +0 -24
- package/esm2022/testing/component-mocks/node-toolbar-mock.component.mjs +0 -23
- package/esm2022/testing/component-mocks/resizable-mock.component.mjs +0 -27
- package/esm2022/testing/component-mocks/vflow-mock.component.mjs +0 -299
- package/esm2022/testing/directive-mocks/connection-controller-mock.directive.mjs +0 -29
- package/esm2022/testing/directive-mocks/drag-handle-mock.directive.mjs +0 -11
- package/esm2022/testing/directive-mocks/selectable-mock.directive.mjs +0 -14
- package/esm2022/testing/directive-mocks/template-mock.directive.mjs +0 -101
- package/esm2022/testing/ngx-vflow-testing.mjs +0 -5
- package/esm2022/testing/provide-custom-node-mocks.mjs +0 -60
- package/esm2022/testing/public-api.mjs +0 -13
- package/esm2022/testing/types.mjs +0 -2
- package/esm2022/testing/vflow-mocks.mjs +0 -28
- package/lib/vflow/public-components/custom-dynamic-node/custom-dynamic-node.component.d.ts +0 -13
- package/lib/vflow/utils/to-unified-node.d.ts +0 -2
package/public-api.d.ts
CHANGED
|
@@ -5,7 +5,7 @@ export * from './lib/vflow/interfaces/edge.interface';
|
|
|
5
5
|
export * from './lib/vflow/interfaces/edge-label.interface';
|
|
6
6
|
export * from './lib/vflow/interfaces/connection.interface';
|
|
7
7
|
export * from './lib/vflow/interfaces/connection.interface';
|
|
8
|
-
export
|
|
8
|
+
export { ConnectionSettings, ConnectionValidatorFn } from './lib/vflow/interfaces/connection-settings.interface';
|
|
9
9
|
export * from './lib/vflow/interfaces/marker.interface';
|
|
10
10
|
export { ViewportState } from './lib/vflow/interfaces/viewport.interface';
|
|
11
11
|
export * from './lib/vflow/interfaces/component-node-event.interface';
|
|
@@ -15,16 +15,17 @@ export * from './lib/vflow/interfaces/intersecting-nodes-options.interface';
|
|
|
15
15
|
export * from './lib/vflow/interfaces/curve-factory.interface';
|
|
16
16
|
export * from './lib/vflow/interfaces/node-preview.interface';
|
|
17
17
|
export * from './lib/vflow/interfaces/alignment-helper-settings.interface';
|
|
18
|
+
export { ConnectEndEvent, ConnectStartEvent, ReconnectEndEvent, ReconnectEvent, ReconnectStartEvent, } from './lib/vflow/interfaces/connection-events.interface';
|
|
18
19
|
export * from './lib/vflow/types/node-change.type';
|
|
19
20
|
export * from './lib/vflow/types/edge-change.type';
|
|
20
21
|
export * from './lib/vflow/types/position.type';
|
|
21
22
|
export * from './lib/vflow/types/background.type';
|
|
22
23
|
export * from './lib/vflow/types/connection-mode.type';
|
|
23
24
|
export * from './lib/vflow/types/keyboard-action.type';
|
|
25
|
+
export * from './lib/vflow/types/selection-mode.type';
|
|
24
26
|
export * from './lib/vflow/components/vflow/vflow.component';
|
|
25
27
|
export * from './lib/vflow/public-components/handle/handle.component';
|
|
26
28
|
export * from './lib/vflow/public-components/custom-node/custom-node.component';
|
|
27
|
-
export * from './lib/vflow/public-components/custom-dynamic-node/custom-dynamic-node.component';
|
|
28
29
|
export * from './lib/vflow/public-components/resizable/resizable.component';
|
|
29
30
|
export * from './lib/vflow/public-components/minimap/minimap.component';
|
|
30
31
|
export * from './lib/vflow/public-components/node-toolbar/node-toolbar.component';
|
|
@@ -1,11 +1,11 @@
|
|
|
1
1
|
import { WritableSignal, OnInit } from '@angular/core';
|
|
2
|
-
import { Node,
|
|
2
|
+
import { Node, Edge, SpacePoint, Point, Background, KeyboardShortcuts, ViewportState, NodeChange, EdgeChange, FitViewOptions, VflowComponent, IntersectingNodesOptions, ɵConnectionModel as ConnectionModel, AlignmentHelperSettings, SelectionMode } from 'ngx-vflow';
|
|
3
3
|
import { ConnectionTemplateMockDirective, EdgeLabelHtmlTemplateMockDirective, EdgeTemplateMockDirective, GroupNodeTemplateMockDirective, NodeHtmlTemplateMockDirective } from '../directive-mocks/template-mock.directive';
|
|
4
4
|
import { AsInterface } from '../types';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
6
6
|
import * as i1 from "ngx-vflow";
|
|
7
7
|
export declare class VflowMockComponent implements AsInterface<VflowComponent>, OnInit {
|
|
8
|
-
readonly nodes: Node[]
|
|
8
|
+
readonly nodes: Node[];
|
|
9
9
|
readonly edges: Edge[];
|
|
10
10
|
readonly view: [
|
|
11
11
|
number,
|
|
@@ -16,6 +16,7 @@ export declare class VflowMockComponent implements AsInterface<VflowComponent>,
|
|
|
16
16
|
readonly background: Background | string;
|
|
17
17
|
readonly optimization: Required<import("ngx-vflow").Optimization>;
|
|
18
18
|
readonly entitiesSelectable = true;
|
|
19
|
+
readonly selectionMode: SelectionMode;
|
|
19
20
|
readonly keyboardShortcuts: KeyboardShortcuts;
|
|
20
21
|
readonly connection: ConnectionModel;
|
|
21
22
|
readonly snapGrid: [
|
|
@@ -24,8 +25,9 @@ export declare class VflowMockComponent implements AsInterface<VflowComponent>,
|
|
|
24
25
|
];
|
|
25
26
|
elevateNodesOnSelect: boolean;
|
|
26
27
|
elevateEdgesOnSelect: boolean;
|
|
28
|
+
autoPan: boolean;
|
|
27
29
|
alignmentHelper: import("@angular/core").InputSignal<boolean | AlignmentHelperSettings>;
|
|
28
|
-
readonly
|
|
30
|
+
readonly componentNodeEvent: import("@angular/core").OutputEmitterRef<any>;
|
|
29
31
|
protected nodeTemplateDirective: import("@angular/core").Signal<NodeHtmlTemplateMockDirective | undefined>;
|
|
30
32
|
protected groupNodeTemplateDirective: import("@angular/core").Signal<GroupNodeTemplateMockDirective | undefined>;
|
|
31
33
|
protected edgeTemplateDirective: import("@angular/core").Signal<EdgeTemplateMockDirective | undefined>;
|
|
@@ -50,12 +52,12 @@ export declare class VflowMockComponent implements AsInterface<VflowComponent>,
|
|
|
50
52
|
documentPointToFlowPoint(point: Point, options: {
|
|
51
53
|
spaces: true;
|
|
52
54
|
}): SpacePoint[];
|
|
53
|
-
getIntesectingNodes(nodeId: string, options?: IntersectingNodesOptions): Node[]
|
|
55
|
+
getIntesectingNodes(nodeId: string, options?: IntersectingNodesOptions): Node[];
|
|
54
56
|
toNodeSpace(nodeId: string, spaceNodeId: string): Point;
|
|
55
|
-
getNode<T = unknown>(id: string): Node<T> |
|
|
57
|
+
getNode<T = unknown>(id: string): Node<T> | undefined;
|
|
56
58
|
getDetachedEdges(): Edge[];
|
|
57
59
|
protected createSignal<T>(value: T): WritableSignal<T>;
|
|
58
60
|
static ɵfac: i0.ɵɵFactoryDeclaration<VflowMockComponent, never>;
|
|
59
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<VflowMockComponent, "vflow", never, { "nodes": { "alias": "nodes"; "required": true; }; "edges": { "alias": "edges"; "required": false; }; "view": { "alias": "view"; "required": false; }; "minZoom": { "alias": "minZoom"; "required": false; }; "maxZoom": { "alias": "maxZoom"; "required": false; }; "background": { "alias": "background"; "required": false; }; "optimization": { "alias": "optimization"; "required": false; }; "entitiesSelectable": { "alias": "entitiesSelectable"; "required": false; }; "keyboardShortcuts": { "alias": "keyboardShortcuts"; "required": false; }; "connection": { "alias": "connection"; "required": false; }; "snapGrid": { "alias": "snapGrid"; "required": false; }; "elevateNodesOnSelect": { "alias": "elevateNodesOnSelect"; "required": false; }; "elevateEdgesOnSelect": { "alias": "elevateEdgesOnSelect"; "required": false; }; "alignmentHelper": { "alias": "alignmentHelper"; "required": false; "isSignal": true; }; }, { "
|
|
61
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<VflowMockComponent, "vflow", never, { "nodes": { "alias": "nodes"; "required": true; }; "edges": { "alias": "edges"; "required": false; }; "view": { "alias": "view"; "required": false; }; "minZoom": { "alias": "minZoom"; "required": false; }; "maxZoom": { "alias": "maxZoom"; "required": false; }; "background": { "alias": "background"; "required": false; }; "optimization": { "alias": "optimization"; "required": false; }; "entitiesSelectable": { "alias": "entitiesSelectable"; "required": false; }; "selectionMode": { "alias": "selectionMode"; "required": false; }; "keyboardShortcuts": { "alias": "keyboardShortcuts"; "required": false; }; "connection": { "alias": "connection"; "required": false; }; "snapGrid": { "alias": "snapGrid"; "required": false; }; "elevateNodesOnSelect": { "alias": "elevateNodesOnSelect"; "required": false; }; "elevateEdgesOnSelect": { "alias": "elevateEdgesOnSelect"; "required": false; }; "autoPan": { "alias": "autoPan"; "required": false; }; "alignmentHelper": { "alias": "alignmentHelper"; "required": false; "isSignal": true; }; }, { "componentNodeEvent": "componentNodeEvent"; }, ["nodeTemplateDirective", "groupNodeTemplateDirective", "edgeTemplateDirective", "edgeLabelHtmlDirective", "connectionTemplateDirective"], ["*"], true, never>;
|
|
60
62
|
static ngAcceptInputType_connection: i1.ConnectionSettings;
|
|
61
63
|
}
|
|
@@ -1,17 +1,19 @@
|
|
|
1
|
-
import { Connection,
|
|
1
|
+
import { Connection, ReconnectEvent } from 'ngx-vflow';
|
|
2
2
|
import { AsInterface } from '../types';
|
|
3
|
-
import type { ConnectionControllerDirective, ɵHandleModel as HandleModel } from 'ngx-vflow';
|
|
3
|
+
import type { ConnectEndEvent, ConnectionControllerDirective, ConnectStartEvent, ɵHandleModel as HandleModel, ReconnectEndEvent, ReconnectStartEvent } from 'ngx-vflow';
|
|
4
4
|
import * as i0 from "@angular/core";
|
|
5
5
|
export declare class ConnectionControllerMockDirective implements AsInterface<ConnectionControllerDirective> {
|
|
6
|
+
readonly connectStart: import("@angular/core").OutputEmitterRef<ConnectStartEvent>;
|
|
6
7
|
readonly connect: import("@angular/core").OutputEmitterRef<Connection>;
|
|
7
|
-
readonly
|
|
8
|
-
readonly
|
|
9
|
-
readonly
|
|
8
|
+
readonly connectEnd: import("@angular/core").OutputEmitterRef<ConnectEndEvent>;
|
|
9
|
+
readonly reconnectStart: import("@angular/core").OutputEmitterRef<ReconnectStartEvent>;
|
|
10
|
+
readonly reconnect: import("@angular/core").OutputEmitterRef<ReconnectEvent>;
|
|
11
|
+
readonly reconnectEnd: import("@angular/core").OutputEmitterRef<ReconnectEndEvent>;
|
|
10
12
|
startConnection(handle: HandleModel): void;
|
|
11
13
|
startReconnection(handle: HandleModel): void;
|
|
12
14
|
validateConnection(handle: HandleModel): void;
|
|
13
15
|
resetValidateConnection(targetHandle: HandleModel): void;
|
|
14
16
|
endConnection(): void;
|
|
15
17
|
static ɵfac: i0.ɵɵFactoryDeclaration<ConnectionControllerMockDirective, never>;
|
|
16
|
-
static ɵdir: i0.ɵɵDirectiveDeclaration<ConnectionControllerMockDirective, "[connect], [
|
|
18
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<ConnectionControllerMockDirective, "[connectStart], [connect], [connectEnd], [reconnectStart], [reconnect], [reconnectEnd]", never, {}, { "connectStart": "connectStart"; "connect": "connect"; "connectEnd": "connectEnd"; "reconnectStart": "reconnectStart"; "reconnect": "reconnect"; "reconnectEnd": "reconnectEnd"; }, never, never, true, never>;
|
|
17
19
|
}
|
|
@@ -1,103 +0,0 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, computed, inject, input } from '@angular/core';
|
|
2
|
-
import { nodeToRect } from '../../utils/nodes';
|
|
3
|
-
import { FlowStatusService, isNodeDragEndStatus, isNodeDragStartStatus } from '../../services/flow-status.service';
|
|
4
|
-
import { rectToRectWithSides } from '../../interfaces/rect';
|
|
5
|
-
import { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';
|
|
6
|
-
import { filter, map, tap } from 'rxjs';
|
|
7
|
-
import { extendedComputed } from '../../utils/signals/extended-computed';
|
|
8
|
-
import { NodeRenderingService } from '../../services/node-rendering.service';
|
|
9
|
-
import { getSpacePoints } from '../../utils/get-space-points';
|
|
10
|
-
import * as i0 from "@angular/core";
|
|
11
|
-
export class AlignmentHelperComponent {
|
|
12
|
-
constructor() {
|
|
13
|
-
this.nodeRenderingService = inject(NodeRenderingService);
|
|
14
|
-
this.flowStatus = inject(FlowStatusService);
|
|
15
|
-
this.tolerance = input(10);
|
|
16
|
-
this.lineColor = input('#1b262c');
|
|
17
|
-
this.isNodeDragging = computed(() => isNodeDragStartStatus(this.flowStatus.status()));
|
|
18
|
-
this.intersections = extendedComputed((lastValue) => {
|
|
19
|
-
const status = this.flowStatus.status();
|
|
20
|
-
if (isNodeDragStartStatus(status)) {
|
|
21
|
-
const node = status.payload.node;
|
|
22
|
-
const d = rectToRectWithSides(nodeToRect(node));
|
|
23
|
-
const otherRects = this.nodeRenderingService
|
|
24
|
-
.viewportNodes()
|
|
25
|
-
.filter((n) => n !== node)
|
|
26
|
-
// do not check children of the dragged node
|
|
27
|
-
.filter((n) => !node.children().includes(n))
|
|
28
|
-
.map((n) => rectToRectWithSides(nodeToRect(n)));
|
|
29
|
-
const lines = [];
|
|
30
|
-
let snappedX = d.x;
|
|
31
|
-
let snappedY = d.y;
|
|
32
|
-
let closestXDiff = Infinity;
|
|
33
|
-
let closestYDiff = Infinity;
|
|
34
|
-
otherRects.forEach((o) => {
|
|
35
|
-
const dCenterX = d.left + d.width / 2;
|
|
36
|
-
const oCenterX = o.left + o.width / 2;
|
|
37
|
-
for (const [dX, oX, snapX, isCenter] of [
|
|
38
|
-
// center check
|
|
39
|
-
[dCenterX, oCenterX, oCenterX - d.width / 2, true],
|
|
40
|
-
[d.left, o.left, o.left, false],
|
|
41
|
-
[d.left, o.right, o.right, false],
|
|
42
|
-
[d.right, o.left, o.left - d.width, false],
|
|
43
|
-
[d.right, o.right, o.right - d.width, false],
|
|
44
|
-
]) {
|
|
45
|
-
const diff = Math.abs(dX - oX);
|
|
46
|
-
if (diff <= this.tolerance()) {
|
|
47
|
-
const y = Math.min(d.top, o.top);
|
|
48
|
-
const y2 = Math.max(d.bottom, o.bottom);
|
|
49
|
-
lines.push({ x: oX, y, x2: oX, y2, isCenter });
|
|
50
|
-
if (diff < closestXDiff) {
|
|
51
|
-
closestXDiff = diff;
|
|
52
|
-
snappedX = snapX;
|
|
53
|
-
}
|
|
54
|
-
if (isCenter)
|
|
55
|
-
break;
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
const dCenterY = d.top + d.height / 2;
|
|
59
|
-
const oCenterY = o.top + o.height / 2;
|
|
60
|
-
for (const [dY, oY, snapY, isCenter] of [
|
|
61
|
-
// center check
|
|
62
|
-
[dCenterY, oCenterY, oCenterY - d.height / 2, true],
|
|
63
|
-
[d.top, o.top, o.top, false],
|
|
64
|
-
[d.top, o.bottom, o.bottom, false],
|
|
65
|
-
[d.bottom, o.top, o.top - d.height, false],
|
|
66
|
-
[d.bottom, o.bottom, o.bottom - d.height, false],
|
|
67
|
-
]) {
|
|
68
|
-
const diff = Math.abs(dY - oY);
|
|
69
|
-
if (diff <= this.tolerance()) {
|
|
70
|
-
const x = Math.min(d.left, o.left);
|
|
71
|
-
const x2 = Math.max(d.right, o.right);
|
|
72
|
-
lines.push({ x, y: oY, x2, y2: oY, isCenter });
|
|
73
|
-
if (diff < closestYDiff) {
|
|
74
|
-
closestYDiff = diff;
|
|
75
|
-
snappedY = snapY;
|
|
76
|
-
}
|
|
77
|
-
if (isCenter)
|
|
78
|
-
break;
|
|
79
|
-
}
|
|
80
|
-
}
|
|
81
|
-
});
|
|
82
|
-
return { lines, snappedX, snappedY };
|
|
83
|
-
}
|
|
84
|
-
return lastValue;
|
|
85
|
-
});
|
|
86
|
-
toObservable(this.flowStatus.status)
|
|
87
|
-
.pipe(filter(isNodeDragEndStatus), map((status) => status.payload.node), map((node) => [node, this.intersections()]), tap(([node, intersections]) => {
|
|
88
|
-
if (intersections) {
|
|
89
|
-
const snapped = { x: intersections.snappedX, y: intersections.snappedY };
|
|
90
|
-
const parentIfExists = node.parent() ? [node.parent()] : [];
|
|
91
|
-
node.setPoint(getSpacePoints(snapped, parentIfExists)[0]);
|
|
92
|
-
}
|
|
93
|
-
}), takeUntilDestroyed())
|
|
94
|
-
.subscribe();
|
|
95
|
-
}
|
|
96
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AlignmentHelperComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
97
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: AlignmentHelperComponent, isStandalone: true, selector: "g[alignmentHelper]", inputs: { tolerance: { classPropertyName: "tolerance", publicName: "tolerance", isSignal: true, isRequired: false, transformFunction: null }, lineColor: { classPropertyName: "lineColor", publicName: "lineColor", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "@if (isNodeDragging()) {\n @if (intersections(); as intersections) {\n @for (intersection of intersections.lines; track $index) {\n <svg:line\n [attr.stroke]=\"lineColor()\"\n [attr.stroke-dasharray]=\"intersection.isCenter ? 4 : null\"\n [attr.x1]=\"intersection.x\"\n [attr.y1]=\"intersection.y\"\n [attr.x2]=\"intersection.x2\"\n [attr.y2]=\"intersection.y2\" />\n }\n }\n}\n", changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
98
|
-
}
|
|
99
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AlignmentHelperComponent, decorators: [{
|
|
100
|
-
type: Component,
|
|
101
|
-
args: [{ selector: 'g[alignmentHelper]', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, template: "@if (isNodeDragging()) {\n @if (intersections(); as intersections) {\n @for (intersection of intersections.lines; track $index) {\n <svg:line\n [attr.stroke]=\"lineColor()\"\n [attr.stroke-dasharray]=\"intersection.isCenter ? 4 : null\"\n [attr.x1]=\"intersection.x\"\n [attr.y1]=\"intersection.y\"\n [attr.x2]=\"intersection.x2\"\n [attr.y2]=\"intersection.y2\" />\n }\n }\n}\n" }]
|
|
102
|
-
}], ctorParameters: () => [] });
|
|
103
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alignment-helper.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/alignment-helper/alignment-helper.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/alignment-helper/alignment-helper.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, inject, input } from '@angular/core';\nimport { nodeToRect } from '../../utils/nodes';\nimport { FlowStatusService, isNodeDragEndStatus, isNodeDragStartStatus } from '../../services/flow-status.service';\nimport { rectToRectWithSides } from '../../interfaces/rect';\nimport { Box } from '../../interfaces/box';\nimport { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop';\nimport { filter, map, tap } from 'rxjs';\nimport { extendedComputed } from '../../utils/signals/extended-computed';\nimport { NodeRenderingService } from '../../services/node-rendering.service';\nimport { getSpacePoints } from '../../utils/get-space-points';\n\ninterface Intersection {\n  lines: (Box & { isCenter: boolean })[];\n  snappedX: number;\n  snappedY: number;\n}\n\n@Component({\n  selector: 'g[alignmentHelper]',\n  templateUrl: './alignment-helper.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n})\nexport class AlignmentHelperComponent {\n  private nodeRenderingService = inject(NodeRenderingService);\n  private flowStatus = inject(FlowStatusService);\n\n  readonly tolerance = input(10);\n  readonly lineColor = input('#1b262c');\n\n  protected isNodeDragging = computed(() => isNodeDragStartStatus(this.flowStatus.status()));\n\n  protected readonly intersections = extendedComputed<Intersection>((lastValue) => {\n    const status = this.flowStatus.status();\n\n    if (isNodeDragStartStatus(status)) {\n      const node = status.payload.node;\n\n      const d = rectToRectWithSides(nodeToRect(node));\n      const otherRects = this.nodeRenderingService\n        .viewportNodes()\n        .filter((n) => n !== node)\n        // do not check children of the dragged node\n        .filter((n) => !node.children().includes(n))\n        .map((n) => rectToRectWithSides(nodeToRect(n)));\n\n      const lines: Intersection['lines'] = [];\n\n      let snappedX = d.x;\n      let snappedY = d.y;\n      let closestXDiff = Infinity;\n      let closestYDiff = Infinity;\n\n      otherRects.forEach((o) => {\n        const dCenterX = d.left + d.width / 2;\n        const oCenterX = o.left + o.width / 2;\n\n        for (const [dX, oX, snapX, isCenter] of [\n          // center check\n          [dCenterX, oCenterX, oCenterX - d.width / 2, true] as const,\n          [d.left, o.left, o.left, false] as const,\n          [d.left, o.right, o.right, false] as const,\n          [d.right, o.left, o.left - d.width, false] as const,\n          [d.right, o.right, o.right - d.width, false] as const,\n        ]) {\n          const diff = Math.abs(dX - oX);\n\n          if (diff <= this.tolerance()) {\n            const y = Math.min(d.top, o.top);\n            const y2 = Math.max(d.bottom, o.bottom);\n\n            lines.push({ x: oX, y, x2: oX, y2, isCenter });\n\n            if (diff < closestXDiff) {\n              closestXDiff = diff;\n              snappedX = snapX;\n            }\n\n            if (isCenter) break;\n          }\n        }\n\n        const dCenterY = d.top + d.height / 2;\n        const oCenterY = o.top + o.height / 2;\n\n        for (const [dY, oY, snapY, isCenter] of [\n          // center check\n          [dCenterY, oCenterY, oCenterY - d.height / 2, true] as const,\n          [d.top, o.top, o.top, false] as const,\n          [d.top, o.bottom, o.bottom, false] as const,\n          [d.bottom, o.top, o.top - d.height, false] as const,\n          [d.bottom, o.bottom, o.bottom - d.height, false] as const,\n        ]) {\n          const diff = Math.abs(dY - oY);\n\n          if (diff <= this.tolerance()) {\n            const x = Math.min(d.left, o.left);\n            const x2 = Math.max(d.right, o.right);\n\n            lines.push({ x, y: oY, x2, y2: oY, isCenter });\n\n            if (diff < closestYDiff) {\n              closestYDiff = diff;\n              snappedY = snapY;\n            }\n\n            if (isCenter) break;\n          }\n        }\n      });\n\n      return { lines, snappedX, snappedY };\n    }\n\n    return lastValue;\n  });\n\n  constructor() {\n    toObservable(this.flowStatus.status)\n      .pipe(\n        filter(isNodeDragEndStatus),\n        map((status) => status.payload.node),\n        map((node) => [node, this.intersections()] as const),\n        tap(([node, intersections]) => {\n          if (intersections) {\n            const snapped = { x: intersections.snappedX, y: intersections.snappedY };\n            const parentIfExists = node.parent() ? [node.parent()!] : [];\n\n            node.setPoint(getSpacePoints(snapped, parentIfExists)[0]);\n          }\n        }),\n        takeUntilDestroyed(),\n      )\n      .subscribe();\n  }\n}\n","@if (isNodeDragging()) {\n  @if (intersections(); as intersections) {\n    @for (intersection of intersections.lines; track $index) {\n      <svg:line\n        [attr.stroke]=\"lineColor()\"\n        [attr.stroke-dasharray]=\"intersection.isCenter ? 4 : null\"\n        [attr.x1]=\"intersection.x\"\n        [attr.y1]=\"intersection.y\"\n        [attr.x2]=\"intersection.x2\"\n        [attr.y2]=\"intersection.y2\" />\n    }\n  }\n}\n"]}
|
|
@@ -1,121 +0,0 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, computed, effect, inject } from '@angular/core';
|
|
2
|
-
import { ViewportService } from '../../services/viewport.service';
|
|
3
|
-
import { RootSvgReferenceDirective } from '../../directives/reference.directive';
|
|
4
|
-
import { id } from '../../utils/id';
|
|
5
|
-
import { FlowSettingsService } from '../../services/flow-settings.service';
|
|
6
|
-
import { toObservable } from '@angular/core/rxjs-interop';
|
|
7
|
-
import { map, switchMap } from 'rxjs';
|
|
8
|
-
import { toLazySignal } from '../../utils/signals/to-lazy-signal';
|
|
9
|
-
import * as i0 from "@angular/core";
|
|
10
|
-
const defaultBg = '#fff';
|
|
11
|
-
const defaultGap = 20;
|
|
12
|
-
const defaultDotSize = 2;
|
|
13
|
-
const defaultDotColor = 'rgb(177, 177, 183)';
|
|
14
|
-
const defaultImageScale = 0.1;
|
|
15
|
-
const defaultRepeated = true;
|
|
16
|
-
export class BackgroundComponent {
|
|
17
|
-
constructor() {
|
|
18
|
-
this.viewportService = inject(ViewportService);
|
|
19
|
-
this.rootSvg = inject(RootSvgReferenceDirective).element;
|
|
20
|
-
this.settingsService = inject(FlowSettingsService);
|
|
21
|
-
this.backgroundSignal = this.settingsService.background;
|
|
22
|
-
// DOTS PATTERN
|
|
23
|
-
this.scaledGap = computed(() => {
|
|
24
|
-
const background = this.backgroundSignal();
|
|
25
|
-
if (background.type === 'dots') {
|
|
26
|
-
const zoom = this.viewportService.readableViewport().zoom;
|
|
27
|
-
return zoom * (background.gap ?? defaultGap);
|
|
28
|
-
}
|
|
29
|
-
return 0;
|
|
30
|
-
});
|
|
31
|
-
this.x = computed(() => this.viewportService.readableViewport().x % this.scaledGap());
|
|
32
|
-
this.y = computed(() => this.viewportService.readableViewport().y % this.scaledGap());
|
|
33
|
-
this.patternColor = computed(() => {
|
|
34
|
-
const bg = this.backgroundSignal();
|
|
35
|
-
if (bg.type === 'dots') {
|
|
36
|
-
return bg.color ?? defaultDotColor;
|
|
37
|
-
}
|
|
38
|
-
return defaultDotColor;
|
|
39
|
-
});
|
|
40
|
-
this.patternSize = computed(() => {
|
|
41
|
-
const background = this.backgroundSignal();
|
|
42
|
-
if (background.type === 'dots') {
|
|
43
|
-
return (this.viewportService.readableViewport().zoom * (background.size ?? defaultDotSize)) / 2;
|
|
44
|
-
}
|
|
45
|
-
return 0;
|
|
46
|
-
});
|
|
47
|
-
// IMAGE PATTERN
|
|
48
|
-
this.bgImageSrc = computed(() => {
|
|
49
|
-
const background = this.backgroundSignal();
|
|
50
|
-
return background.type === 'image' ? background.src : '';
|
|
51
|
-
});
|
|
52
|
-
this.imageSize = toLazySignal(toObservable(this.backgroundSignal).pipe(switchMap(() => createImage(this.bgImageSrc())), map((image) => ({ width: image.naturalWidth, height: image.naturalHeight }))), { initialValue: { width: 0, height: 0 } });
|
|
53
|
-
this.scaledImageWidth = computed(() => {
|
|
54
|
-
const background = this.backgroundSignal();
|
|
55
|
-
if (background.type === 'image') {
|
|
56
|
-
const zoom = background.fixed ? 1 : this.viewportService.readableViewport().zoom;
|
|
57
|
-
return this.imageSize().width * zoom * (background.scale ?? defaultImageScale);
|
|
58
|
-
}
|
|
59
|
-
return 0;
|
|
60
|
-
});
|
|
61
|
-
this.scaledImageHeight = computed(() => {
|
|
62
|
-
const background = this.backgroundSignal();
|
|
63
|
-
if (background.type === 'image') {
|
|
64
|
-
const zoom = background.fixed ? 1 : this.viewportService.readableViewport().zoom;
|
|
65
|
-
return this.imageSize().height * zoom * (background.scale ?? defaultImageScale);
|
|
66
|
-
}
|
|
67
|
-
return 0;
|
|
68
|
-
});
|
|
69
|
-
this.imageX = computed(() => {
|
|
70
|
-
const background = this.backgroundSignal();
|
|
71
|
-
if (background.type === 'image') {
|
|
72
|
-
if (!background.repeat) {
|
|
73
|
-
return background.fixed ? 0 : this.viewportService.readableViewport().x;
|
|
74
|
-
}
|
|
75
|
-
return background.fixed ? 0 : this.viewportService.readableViewport().x % this.scaledImageWidth();
|
|
76
|
-
}
|
|
77
|
-
return 0;
|
|
78
|
-
});
|
|
79
|
-
this.imageY = computed(() => {
|
|
80
|
-
const background = this.backgroundSignal();
|
|
81
|
-
if (background.type === 'image') {
|
|
82
|
-
if (!background.repeat) {
|
|
83
|
-
return background.fixed ? 0 : this.viewportService.readableViewport().y;
|
|
84
|
-
}
|
|
85
|
-
return background.fixed ? 0 : this.viewportService.readableViewport().y % this.scaledImageHeight();
|
|
86
|
-
}
|
|
87
|
-
return 0;
|
|
88
|
-
});
|
|
89
|
-
this.repeated = computed(() => {
|
|
90
|
-
const background = this.backgroundSignal();
|
|
91
|
-
return background.type === 'image' && (background.repeat ?? defaultRepeated);
|
|
92
|
-
});
|
|
93
|
-
// Without ID there will be pattern collision for several flows on the page
|
|
94
|
-
// Later pattern ID may be exposed to API
|
|
95
|
-
this.patternId = id();
|
|
96
|
-
this.patternUrl = `url(#${this.patternId})`;
|
|
97
|
-
effect(() => {
|
|
98
|
-
const background = this.backgroundSignal();
|
|
99
|
-
if (background.type === 'dots') {
|
|
100
|
-
this.rootSvg.style.backgroundColor = background.backgroundColor ?? defaultBg;
|
|
101
|
-
}
|
|
102
|
-
if (background.type === 'solid') {
|
|
103
|
-
this.rootSvg.style.backgroundColor = background.color;
|
|
104
|
-
}
|
|
105
|
-
});
|
|
106
|
-
}
|
|
107
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BackgroundComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
108
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: BackgroundComponent, isStandalone: true, selector: "g[background]", ngImport: i0, template: "@if (backgroundSignal().type === 'dots') {\n <svg:pattern\n patternUnits=\"userSpaceOnUse\"\n [attr.id]=\"patternId\"\n [attr.x]=\"x()\"\n [attr.y]=\"y()\"\n [attr.width]=\"scaledGap()\"\n [attr.height]=\"scaledGap()\">\n <svg:circle\n [attr.cx]=\"patternSize()\"\n [attr.cy]=\"patternSize()\"\n [attr.r]=\"patternSize()\"\n [attr.fill]=\"patternColor()\" />\n </svg:pattern>\n\n <svg:rect x=\"0\" y=\"0\" width=\"100%\" height=\"100%\" [attr.fill]=\"patternUrl\" />\n}\n\n@if (backgroundSignal().type === 'image') {\n @if (repeated()) {\n <svg:pattern\n patternUnits=\"userSpaceOnUse\"\n [attr.id]=\"patternId\"\n [attr.x]=\"imageX()\"\n [attr.y]=\"imageY()\"\n [attr.width]=\"scaledImageWidth()\"\n [attr.height]=\"scaledImageHeight()\">\n <svg:image [attr.href]=\"bgImageSrc()\" [attr.width]=\"scaledImageWidth()\" [attr.height]=\"scaledImageHeight()\" />\n </svg:pattern>\n\n <svg:rect x=\"0\" y=\"0\" width=\"100%\" height=\"100%\" [attr.fill]=\"patternUrl\" />\n }\n\n @if (!repeated()) {\n <svg:image\n [attr.x]=\"imageX()\"\n [attr.y]=\"imageY()\"\n [attr.width]=\"scaledImageWidth()\"\n [attr.height]=\"scaledImageHeight()\"\n [attr.href]=\"bgImageSrc()\" />\n }\n}\n", changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
109
|
-
}
|
|
110
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: BackgroundComponent, decorators: [{
|
|
111
|
-
type: Component,
|
|
112
|
-
args: [{ standalone: true, selector: 'g[background]', changeDetection: ChangeDetectionStrategy.OnPush, template: "@if (backgroundSignal().type === 'dots') {\n <svg:pattern\n patternUnits=\"userSpaceOnUse\"\n [attr.id]=\"patternId\"\n [attr.x]=\"x()\"\n [attr.y]=\"y()\"\n [attr.width]=\"scaledGap()\"\n [attr.height]=\"scaledGap()\">\n <svg:circle\n [attr.cx]=\"patternSize()\"\n [attr.cy]=\"patternSize()\"\n [attr.r]=\"patternSize()\"\n [attr.fill]=\"patternColor()\" />\n </svg:pattern>\n\n <svg:rect x=\"0\" y=\"0\" width=\"100%\" height=\"100%\" [attr.fill]=\"patternUrl\" />\n}\n\n@if (backgroundSignal().type === 'image') {\n @if (repeated()) {\n <svg:pattern\n patternUnits=\"userSpaceOnUse\"\n [attr.id]=\"patternId\"\n [attr.x]=\"imageX()\"\n [attr.y]=\"imageY()\"\n [attr.width]=\"scaledImageWidth()\"\n [attr.height]=\"scaledImageHeight()\">\n <svg:image [attr.href]=\"bgImageSrc()\" [attr.width]=\"scaledImageWidth()\" [attr.height]=\"scaledImageHeight()\" />\n </svg:pattern>\n\n <svg:rect x=\"0\" y=\"0\" width=\"100%\" height=\"100%\" [attr.fill]=\"patternUrl\" />\n }\n\n @if (!repeated()) {\n <svg:image\n [attr.x]=\"imageX()\"\n [attr.y]=\"imageY()\"\n [attr.width]=\"scaledImageWidth()\"\n [attr.height]=\"scaledImageHeight()\"\n [attr.href]=\"bgImageSrc()\" />\n }\n}\n" }]
|
|
113
|
-
}], ctorParameters: () => [] });
|
|
114
|
-
function createImage(url) {
|
|
115
|
-
const image = new Image();
|
|
116
|
-
image.src = url;
|
|
117
|
-
return new Promise((resolve) => {
|
|
118
|
-
image.onload = () => resolve(image);
|
|
119
|
-
});
|
|
120
|
-
}
|
|
121
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"background.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/background/background.component.ts","../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/background/background.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, computed, effect, inject } from '@angular/core';\nimport { ViewportService } from '../../services/viewport.service';\nimport { RootSvgReferenceDirective } from '../../directives/reference.directive';\nimport { id } from '../../utils/id';\nimport { FlowSettingsService } from '../../services/flow-settings.service';\nimport { toObservable } from '@angular/core/rxjs-interop';\nimport { map, switchMap } from 'rxjs';\nimport { toLazySignal } from '../../utils/signals/to-lazy-signal';\n\nconst defaultBg = '#fff';\nconst defaultGap = 20;\nconst defaultDotSize = 2;\nconst defaultDotColor = 'rgb(177, 177, 183)';\nconst defaultImageScale = 0.1;\nconst defaultRepeated = true;\n\n@Component({\n  standalone: true,\n  selector: 'g[background]',\n  templateUrl: './background.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class BackgroundComponent {\n  private viewportService = inject(ViewportService);\n  private rootSvg = inject(RootSvgReferenceDirective).element;\n  private settingsService = inject(FlowSettingsService);\n\n  protected backgroundSignal = this.settingsService.background;\n\n  // DOTS PATTERN\n  protected scaledGap = computed(() => {\n    const background = this.backgroundSignal();\n\n    if (background.type === 'dots') {\n      const zoom = this.viewportService.readableViewport().zoom;\n\n      return zoom * (background.gap ?? defaultGap);\n    }\n\n    return 0;\n  });\n\n  protected x = computed(() => this.viewportService.readableViewport().x % this.scaledGap());\n\n  protected y = computed(() => this.viewportService.readableViewport().y % this.scaledGap());\n\n  protected patternColor = computed(() => {\n    const bg = this.backgroundSignal();\n\n    if (bg.type === 'dots') {\n      return bg.color ?? defaultDotColor;\n    }\n\n    return defaultDotColor;\n  });\n\n  protected patternSize = computed(() => {\n    const background = this.backgroundSignal();\n\n    if (background.type === 'dots') {\n      return (this.viewportService.readableViewport().zoom * (background.size ?? defaultDotSize)) / 2;\n    }\n\n    return 0;\n  });\n\n  // IMAGE PATTERN\n  protected bgImageSrc = computed(() => {\n    const background = this.backgroundSignal();\n\n    return background.type === 'image' ? background.src : '';\n  });\n\n  protected imageSize = toLazySignal(\n    toObservable(this.backgroundSignal).pipe(\n      switchMap(() => createImage(this.bgImageSrc())),\n      map((image) => ({ width: image.naturalWidth, height: image.naturalHeight })),\n    ),\n    { initialValue: { width: 0, height: 0 } },\n  );\n\n  protected scaledImageWidth = computed(() => {\n    const background = this.backgroundSignal();\n\n    if (background.type === 'image') {\n      const zoom = background.fixed ? 1 : this.viewportService.readableViewport().zoom;\n\n      return this.imageSize().width * zoom * (background.scale ?? defaultImageScale);\n    }\n\n    return 0;\n  });\n\n  protected scaledImageHeight = computed(() => {\n    const background = this.backgroundSignal();\n\n    if (background.type === 'image') {\n      const zoom = background.fixed ? 1 : this.viewportService.readableViewport().zoom;\n\n      return this.imageSize().height * zoom * (background.scale ?? defaultImageScale);\n    }\n\n    return 0;\n  });\n\n  protected imageX = computed(() => {\n    const background = this.backgroundSignal();\n\n    if (background.type === 'image') {\n      if (!background.repeat) {\n        return background.fixed ? 0 : this.viewportService.readableViewport().x;\n      }\n\n      return background.fixed ? 0 : this.viewportService.readableViewport().x % this.scaledImageWidth();\n    }\n\n    return 0;\n  });\n\n  protected imageY = computed(() => {\n    const background = this.backgroundSignal();\n\n    if (background.type === 'image') {\n      if (!background.repeat) {\n        return background.fixed ? 0 : this.viewportService.readableViewport().y;\n      }\n\n      return background.fixed ? 0 : this.viewportService.readableViewport().y % this.scaledImageHeight();\n    }\n\n    return 0;\n  });\n\n  protected repeated = computed(() => {\n    const background = this.backgroundSignal();\n\n    return background.type === 'image' && (background.repeat ?? defaultRepeated);\n  });\n\n  // Without ID there will be pattern collision for several flows on the page\n  // Later pattern ID may be exposed to API\n  protected patternId = id();\n  protected patternUrl = `url(#${this.patternId})`;\n\n  constructor() {\n    effect(() => {\n      const background = this.backgroundSignal();\n\n      if (background.type === 'dots') {\n        this.rootSvg.style.backgroundColor = background.backgroundColor ?? defaultBg;\n      }\n\n      if (background.type === 'solid') {\n        this.rootSvg.style.backgroundColor = background.color;\n      }\n    });\n  }\n}\n\nfunction createImage(url: string) {\n  const image = new Image();\n\n  image.src = url;\n\n  return new Promise<HTMLImageElement>((resolve) => {\n    image.onload = () => resolve(image);\n  });\n}\n","@if (backgroundSignal().type === 'dots') {\n  <svg:pattern\n    patternUnits=\"userSpaceOnUse\"\n    [attr.id]=\"patternId\"\n    [attr.x]=\"x()\"\n    [attr.y]=\"y()\"\n    [attr.width]=\"scaledGap()\"\n    [attr.height]=\"scaledGap()\">\n    <svg:circle\n      [attr.cx]=\"patternSize()\"\n      [attr.cy]=\"patternSize()\"\n      [attr.r]=\"patternSize()\"\n      [attr.fill]=\"patternColor()\" />\n  </svg:pattern>\n\n  <svg:rect x=\"0\" y=\"0\" width=\"100%\" height=\"100%\" [attr.fill]=\"patternUrl\" />\n}\n\n@if (backgroundSignal().type === 'image') {\n  @if (repeated()) {\n    <svg:pattern\n      patternUnits=\"userSpaceOnUse\"\n      [attr.id]=\"patternId\"\n      [attr.x]=\"imageX()\"\n      [attr.y]=\"imageY()\"\n      [attr.width]=\"scaledImageWidth()\"\n      [attr.height]=\"scaledImageHeight()\">\n      <svg:image [attr.href]=\"bgImageSrc()\" [attr.width]=\"scaledImageWidth()\" [attr.height]=\"scaledImageHeight()\" />\n    </svg:pattern>\n\n    <svg:rect x=\"0\" y=\"0\" width=\"100%\" height=\"100%\" [attr.fill]=\"patternUrl\" />\n  }\n\n  @if (!repeated()) {\n    <svg:image\n      [attr.x]=\"imageX()\"\n      [attr.y]=\"imageY()\"\n      [attr.width]=\"scaledImageWidth()\"\n      [attr.height]=\"scaledImageHeight()\"\n      [attr.href]=\"bgImageSrc()\" />\n  }\n}\n"]}
|
|
@@ -1,157 +0,0 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, computed, inject, input } from '@angular/core';
|
|
2
|
-
import { FlowStatusService } from '../../services/flow-status.service';
|
|
3
|
-
import { straightPath } from '../../math/edge-path/straigh-path';
|
|
4
|
-
import { SpacePointContextDirective } from '../../directives/space-point-context.directive';
|
|
5
|
-
import { bezierPath } from '../../math/edge-path/bezier-path';
|
|
6
|
-
import { hashCode } from '../../utils/hash';
|
|
7
|
-
import { smoothStepPath } from '../../math/edge-path/smooth-step-path';
|
|
8
|
-
import { NgTemplateOutlet } from '@angular/common';
|
|
9
|
-
import { FlowEntitiesService } from '../../services/flow-entities.service';
|
|
10
|
-
import * as i0 from "@angular/core";
|
|
11
|
-
export class ConnectionComponent {
|
|
12
|
-
constructor() {
|
|
13
|
-
this.flowStatusService = inject(FlowStatusService);
|
|
14
|
-
this.spacePointContext = inject(SpacePointContextDirective);
|
|
15
|
-
this.flowEntitiesService = inject(FlowEntitiesService);
|
|
16
|
-
this.model = input.required();
|
|
17
|
-
this.template = input();
|
|
18
|
-
this.path = computed(() => {
|
|
19
|
-
const status = this.flowStatusService.status();
|
|
20
|
-
const curve = this.model().curve;
|
|
21
|
-
if (status.state === 'connection-start' || status.state === 'reconnection-start') {
|
|
22
|
-
const sourceHandle = status.payload.sourceHandle;
|
|
23
|
-
const sourcePoint = sourceHandle.pointAbsolute();
|
|
24
|
-
const sourcePosition = sourceHandle.rawHandle.position;
|
|
25
|
-
const targetPoint = this.spacePointContext.svgCurrentSpacePoint();
|
|
26
|
-
const targetPosition = getOppositePostion(sourceHandle.rawHandle.position);
|
|
27
|
-
const params = this.getPathFactoryParams(sourcePoint, targetPoint, sourcePosition, targetPosition);
|
|
28
|
-
switch (curve) {
|
|
29
|
-
case 'straight':
|
|
30
|
-
return straightPath(params).path;
|
|
31
|
-
case 'bezier':
|
|
32
|
-
return bezierPath(params).path;
|
|
33
|
-
case 'smooth-step':
|
|
34
|
-
return smoothStepPath(params).path;
|
|
35
|
-
case 'step':
|
|
36
|
-
return smoothStepPath(params, 0).path;
|
|
37
|
-
default:
|
|
38
|
-
return curve(params).path;
|
|
39
|
-
}
|
|
40
|
-
}
|
|
41
|
-
if (status.state === 'connection-validation' || status.state === 'reconnection-validation') {
|
|
42
|
-
const sourceHandle = status.payload.sourceHandle;
|
|
43
|
-
const sourcePoint = sourceHandle.pointAbsolute();
|
|
44
|
-
const sourcePosition = sourceHandle.rawHandle.position;
|
|
45
|
-
const targetHandle = status.payload.targetHandle;
|
|
46
|
-
// ignore magnet if validation failed
|
|
47
|
-
const targetPoint = status.payload.valid
|
|
48
|
-
? targetHandle.pointAbsolute()
|
|
49
|
-
: this.spacePointContext.svgCurrentSpacePoint();
|
|
50
|
-
const targetPosition = status.payload.valid
|
|
51
|
-
? targetHandle.rawHandle.position
|
|
52
|
-
: getOppositePostion(sourceHandle.rawHandle.position);
|
|
53
|
-
const params = this.getPathFactoryParams(sourcePoint, targetPoint, sourcePosition, targetPosition);
|
|
54
|
-
switch (curve) {
|
|
55
|
-
case 'straight':
|
|
56
|
-
return straightPath(params).path;
|
|
57
|
-
case 'bezier':
|
|
58
|
-
return bezierPath(params).path;
|
|
59
|
-
case 'smooth-step':
|
|
60
|
-
return smoothStepPath(params).path;
|
|
61
|
-
case 'step':
|
|
62
|
-
return smoothStepPath(params, 0).path;
|
|
63
|
-
default:
|
|
64
|
-
return curve(params).path;
|
|
65
|
-
}
|
|
66
|
-
}
|
|
67
|
-
return null;
|
|
68
|
-
});
|
|
69
|
-
this.markerUrl = computed(() => {
|
|
70
|
-
const marker = this.model().settings.marker;
|
|
71
|
-
if (marker) {
|
|
72
|
-
return `url(#${hashCode(JSON.stringify(marker))})`;
|
|
73
|
-
}
|
|
74
|
-
return '';
|
|
75
|
-
});
|
|
76
|
-
this.defaultColor = 'rgb(177, 177, 183)';
|
|
77
|
-
}
|
|
78
|
-
// TODO: move context to model
|
|
79
|
-
getContext() {
|
|
80
|
-
return {
|
|
81
|
-
$implicit: {
|
|
82
|
-
path: this.path,
|
|
83
|
-
marker: this.markerUrl,
|
|
84
|
-
},
|
|
85
|
-
};
|
|
86
|
-
}
|
|
87
|
-
getPathFactoryParams(sourcePoint, targetPoint, sourcePosition, targetPosition) {
|
|
88
|
-
return {
|
|
89
|
-
mode: 'connection',
|
|
90
|
-
sourcePoint,
|
|
91
|
-
targetPoint,
|
|
92
|
-
sourcePosition,
|
|
93
|
-
targetPosition,
|
|
94
|
-
allEdges: this.flowEntitiesService.rawEdges(),
|
|
95
|
-
allNodes: this.flowEntitiesService.rawNodes(),
|
|
96
|
-
};
|
|
97
|
-
}
|
|
98
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ConnectionComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
99
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: ConnectionComponent, isStandalone: true, selector: "g[connection]", inputs: { model: { classPropertyName: "model", publicName: "model", isSignal: true, isRequired: true, transformFunction: null }, template: { classPropertyName: "template", publicName: "template", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: `
|
|
100
|
-
@if (model().type === 'default') {
|
|
101
|
-
@if (path(); as path) {
|
|
102
|
-
<svg:path
|
|
103
|
-
fill="none"
|
|
104
|
-
stroke-width="2"
|
|
105
|
-
[attr.d]="path"
|
|
106
|
-
[attr.marker-end]="markerUrl()"
|
|
107
|
-
[attr.stroke]="defaultColor" />
|
|
108
|
-
}
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
@if (model().type === 'template') {
|
|
112
|
-
@if (template(); as template) {
|
|
113
|
-
<ng-container *ngTemplateOutlet="template; context: getContext()" />
|
|
114
|
-
}
|
|
115
|
-
}
|
|
116
|
-
`, isInline: true, dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
117
|
-
}
|
|
118
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ConnectionComponent, decorators: [{
|
|
119
|
-
type: Component,
|
|
120
|
-
args: [{
|
|
121
|
-
standalone: true,
|
|
122
|
-
selector: 'g[connection]',
|
|
123
|
-
template: `
|
|
124
|
-
@if (model().type === 'default') {
|
|
125
|
-
@if (path(); as path) {
|
|
126
|
-
<svg:path
|
|
127
|
-
fill="none"
|
|
128
|
-
stroke-width="2"
|
|
129
|
-
[attr.d]="path"
|
|
130
|
-
[attr.marker-end]="markerUrl()"
|
|
131
|
-
[attr.stroke]="defaultColor" />
|
|
132
|
-
}
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
@if (model().type === 'template') {
|
|
136
|
-
@if (template(); as template) {
|
|
137
|
-
<ng-container *ngTemplateOutlet="template; context: getContext()" />
|
|
138
|
-
}
|
|
139
|
-
}
|
|
140
|
-
`,
|
|
141
|
-
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
142
|
-
imports: [NgTemplateOutlet],
|
|
143
|
-
}]
|
|
144
|
-
}] });
|
|
145
|
-
function getOppositePostion(position) {
|
|
146
|
-
switch (position) {
|
|
147
|
-
case 'top':
|
|
148
|
-
return 'bottom';
|
|
149
|
-
case 'bottom':
|
|
150
|
-
return 'top';
|
|
151
|
-
case 'left':
|
|
152
|
-
return 'right';
|
|
153
|
-
case 'right':
|
|
154
|
-
return 'left';
|
|
155
|
-
}
|
|
156
|
-
}
|
|
157
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"connection.component.js","sourceRoot":"","sources":["../../../../../../../projects/ngx-vflow-lib/src/lib/vflow/components/connection/connection.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, TemplateRef, computed, inject, input } from '@angular/core';\nimport { FlowStatusService } from '../../services/flow-status.service';\nimport { straightPath } from '../../math/edge-path/straigh-path';\nimport { SpacePointContextDirective } from '../../directives/space-point-context.directive';\nimport { ConnectionModel } from '../../models/connection.model';\nimport { bezierPath } from '../../math/edge-path/bezier-path';\nimport { hashCode } from '../../utils/hash';\nimport { Position } from '../../types/position.type';\nimport { smoothStepPath } from '../../math/edge-path/smooth-step-path';\nimport { NgTemplateOutlet } from '@angular/common';\nimport { ConnectionContext } from '../../interfaces/template-context.interface';\nimport { Point } from '../../interfaces/point.interface';\nimport { CurveFactoryParams } from '../../interfaces/curve-factory.interface';\nimport { FlowEntitiesService } from '../../services/flow-entities.service';\n\n@Component({\n  standalone: true,\n  selector: 'g[connection]',\n  template: `\n    @if (model().type === 'default') {\n      @if (path(); as path) {\n        <svg:path\n          fill=\"none\"\n          stroke-width=\"2\"\n          [attr.d]=\"path\"\n          [attr.marker-end]=\"markerUrl()\"\n          [attr.stroke]=\"defaultColor\" />\n      }\n    }\n\n    @if (model().type === 'template') {\n      @if (template(); as template) {\n        <ng-container *ngTemplateOutlet=\"template; context: getContext()\" />\n      }\n    }\n  `,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [NgTemplateOutlet],\n})\nexport class ConnectionComponent {\n  private readonly flowStatusService = inject(FlowStatusService);\n  private readonly spacePointContext = inject(SpacePointContextDirective);\n  private readonly flowEntitiesService = inject(FlowEntitiesService);\n\n  public model = input.required<ConnectionModel>();\n\n  public template = input<TemplateRef<any>>();\n\n  protected path = computed(() => {\n    const status = this.flowStatusService.status();\n    const curve = this.model().curve;\n\n    if (status.state === 'connection-start' || status.state === 'reconnection-start') {\n      const sourceHandle = status.payload.sourceHandle;\n      const sourcePoint = sourceHandle.pointAbsolute();\n      const sourcePosition = sourceHandle.rawHandle.position;\n\n      const targetPoint = this.spacePointContext.svgCurrentSpacePoint();\n      const targetPosition = getOppositePostion(sourceHandle.rawHandle.position);\n\n      const params = this.getPathFactoryParams(sourcePoint, targetPoint, sourcePosition, targetPosition);\n\n      switch (curve) {\n        case 'straight':\n          return straightPath(params).path;\n        case 'bezier':\n          return bezierPath(params).path;\n        case 'smooth-step':\n          return smoothStepPath(params).path;\n        case 'step':\n          return smoothStepPath(params, 0).path;\n        default:\n          return curve(params).path;\n      }\n    }\n\n    if (status.state === 'connection-validation' || status.state === 'reconnection-validation') {\n      const sourceHandle = status.payload.sourceHandle;\n      const sourcePoint = sourceHandle.pointAbsolute();\n      const sourcePosition = sourceHandle.rawHandle.position;\n\n      const targetHandle = status.payload.targetHandle;\n      // ignore magnet if validation failed\n      const targetPoint = status.payload.valid\n        ? targetHandle.pointAbsolute()\n        : this.spacePointContext.svgCurrentSpacePoint();\n      const targetPosition = status.payload.valid\n        ? targetHandle.rawHandle.position\n        : getOppositePostion(sourceHandle.rawHandle.position);\n\n      const params = this.getPathFactoryParams(sourcePoint, targetPoint, sourcePosition, targetPosition);\n\n      switch (curve) {\n        case 'straight':\n          return straightPath(params).path;\n        case 'bezier':\n          return bezierPath(params).path;\n        case 'smooth-step':\n          return smoothStepPath(params).path;\n        case 'step':\n          return smoothStepPath(params, 0).path;\n        default:\n          return curve(params).path;\n      }\n    }\n\n    return null;\n  });\n\n  protected markerUrl = computed(() => {\n    const marker = this.model().settings.marker;\n\n    if (marker) {\n      return `url(#${hashCode(JSON.stringify(marker))})`;\n    }\n\n    return '';\n  });\n\n  protected readonly defaultColor = 'rgb(177, 177, 183)';\n\n  // TODO: move context to model\n  protected getContext(): ConnectionContext {\n    return {\n      $implicit: {\n        path: this.path,\n        marker: this.markerUrl,\n      },\n    };\n  }\n\n  private getPathFactoryParams(\n    sourcePoint: Point,\n    targetPoint: Point,\n    sourcePosition: Position,\n    targetPosition: Position,\n  ): CurveFactoryParams {\n    return {\n      mode: 'connection',\n      sourcePoint,\n      targetPoint,\n      sourcePosition,\n      targetPosition,\n      allEdges: this.flowEntitiesService.rawEdges(),\n      allNodes: this.flowEntitiesService.rawNodes(),\n    };\n  }\n}\n\nfunction getOppositePostion(position: Position): Position {\n  switch (position) {\n    case 'top':\n      return 'bottom';\n    case 'bottom':\n      return 'top';\n    case 'left':\n      return 'right';\n    case 'right':\n      return 'left';\n  }\n}\n"]}
|