@yh-ui/flow 0.1.21
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/LICENSE +21 -0
- package/README.md +234 -0
- package/dist/Flow.d.vue.ts +306 -0
- package/dist/Flow.vue +959 -0
- package/dist/Flow.vue.d.ts +306 -0
- package/dist/__tests__/ai-workflow.ssr.test.cjs +352 -0
- package/dist/__tests__/ai-workflow.ssr.test.d.ts +1 -0
- package/dist/__tests__/ai-workflow.ssr.test.mjs +283 -0
- package/dist/__tests__/ai-workflow.test.cjs +109 -0
- package/dist/__tests__/ai-workflow.test.d.ts +1 -0
- package/dist/__tests__/ai-workflow.test.mjs +112 -0
- package/dist/__tests__/bpmn.ssr.test.cjs +278 -0
- package/dist/__tests__/bpmn.ssr.test.d.ts +1 -0
- package/dist/__tests__/bpmn.ssr.test.mjs +237 -0
- package/dist/__tests__/bpmn.test.cjs +103 -0
- package/dist/__tests__/bpmn.test.d.ts +1 -0
- package/dist/__tests__/bpmn.test.mjs +106 -0
- package/dist/__tests__/custom-types.test.cjs +300 -0
- package/dist/__tests__/custom-types.test.d.ts +1 -0
- package/dist/__tests__/custom-types.test.mjs +248 -0
- package/dist/__tests__/edge.test.cjs +56 -0
- package/dist/__tests__/edge.test.d.ts +1 -0
- package/dist/__tests__/edge.test.mjs +69 -0
- package/dist/__tests__/event-bus.test.cjs +80 -0
- package/dist/__tests__/event-bus.test.d.ts +1 -0
- package/dist/__tests__/event-bus.test.mjs +51 -0
- package/dist/__tests__/flow.ssr.test.cjs +156 -0
- package/dist/__tests__/flow.ssr.test.d.ts +1 -0
- package/dist/__tests__/flow.ssr.test.mjs +112 -0
- package/dist/__tests__/geometry.test.cjs +191 -0
- package/dist/__tests__/geometry.test.d.ts +1 -0
- package/dist/__tests__/geometry.test.mjs +105 -0
- package/dist/__tests__/graph.test.cjs +115 -0
- package/dist/__tests__/graph.test.d.ts +1 -0
- package/dist/__tests__/graph.test.mjs +85 -0
- package/dist/__tests__/history-plugin.test.cjs +191 -0
- package/dist/__tests__/history-plugin.test.d.ts +1 -0
- package/dist/__tests__/history-plugin.test.mjs +161 -0
- package/dist/__tests__/history.test.cjs +81 -0
- package/dist/__tests__/history.test.d.ts +1 -0
- package/dist/__tests__/history.test.mjs +43 -0
- package/dist/__tests__/layout.test.cjs +213 -0
- package/dist/__tests__/layout.test.d.ts +1 -0
- package/dist/__tests__/layout.test.mjs +170 -0
- package/dist/__tests__/node-edit-panel.ssr.test.cjs +168 -0
- package/dist/__tests__/node-edit-panel.ssr.test.d.ts +1 -0
- package/dist/__tests__/node-edit-panel.ssr.test.mjs +118 -0
- package/dist/__tests__/node-group-plugin.test.cjs +235 -0
- package/dist/__tests__/node-group-plugin.test.d.ts +1 -0
- package/dist/__tests__/node-group-plugin.test.mjs +187 -0
- package/dist/__tests__/node-handles.test.cjs +340 -0
- package/dist/__tests__/node-handles.test.d.ts +1 -0
- package/dist/__tests__/node-handles.test.mjs +230 -0
- package/dist/__tests__/plugin.test.cjs +151 -0
- package/dist/__tests__/plugin.test.d.ts +1 -0
- package/dist/__tests__/plugin.test.mjs +116 -0
- package/dist/__tests__/transform.test.cjs +58 -0
- package/dist/__tests__/transform.test.d.ts +1 -0
- package/dist/__tests__/transform.test.mjs +38 -0
- package/dist/__tests__/useAlignment.test.cjs +91 -0
- package/dist/__tests__/useAlignment.test.d.ts +1 -0
- package/dist/__tests__/useAlignment.test.mjs +52 -0
- package/dist/__tests__/useEdges.test.cjs +117 -0
- package/dist/__tests__/useEdges.test.d.ts +1 -0
- package/dist/__tests__/useEdges.test.mjs +80 -0
- package/dist/__tests__/useKeyboard.test.cjs +88 -0
- package/dist/__tests__/useKeyboard.test.d.ts +1 -0
- package/dist/__tests__/useKeyboard.test.mjs +56 -0
- package/dist/__tests__/useNodes.test.cjs +150 -0
- package/dist/__tests__/useNodes.test.d.ts +1 -0
- package/dist/__tests__/useNodes.test.mjs +80 -0
- package/dist/__tests__/useSelection.test.cjs +112 -0
- package/dist/__tests__/useSelection.test.d.ts +1 -0
- package/dist/__tests__/useSelection.test.mjs +76 -0
- package/dist/__tests__/useViewport.test.cjs +171 -0
- package/dist/__tests__/useViewport.test.d.ts +1 -0
- package/dist/__tests__/useViewport.test.mjs +82 -0
- package/dist/__tests__/utils/ssr.cjs +124 -0
- package/dist/__tests__/utils/ssr.d.ts +33 -0
- package/dist/__tests__/utils/ssr.mjs +129 -0
- package/dist/__tests__/validation.test.cjs +95 -0
- package/dist/__tests__/validation.test.d.ts +1 -0
- package/dist/__tests__/validation.test.mjs +36 -0
- package/dist/components/AiNodeEditPanel.d.vue.ts +13 -0
- package/dist/components/AiNodeEditPanel.vue +413 -0
- package/dist/components/AiNodeEditPanel.vue.d.ts +13 -0
- package/dist/components/EdgeEditPanel.d.vue.ts +14 -0
- package/dist/components/EdgeEditPanel.vue +205 -0
- package/dist/components/EdgeEditPanel.vue.d.ts +14 -0
- package/dist/components/NodeEditPanel.d.vue.ts +13 -0
- package/dist/components/NodeEditPanel.vue +214 -0
- package/dist/components/NodeEditPanel.vue.d.ts +13 -0
- package/dist/components/edges/BaseEdge.d.vue.ts +23 -0
- package/dist/components/edges/BaseEdge.vue +55 -0
- package/dist/components/edges/BaseEdge.vue.d.ts +23 -0
- package/dist/components/edges/BezierEdge.d.vue.ts +22 -0
- package/dist/components/edges/BezierEdge.vue +57 -0
- package/dist/components/edges/BezierEdge.vue.d.ts +22 -0
- package/dist/components/edges/DataFlowEdge.d.vue.ts +41 -0
- package/dist/components/edges/DataFlowEdge.vue +211 -0
- package/dist/components/edges/DataFlowEdge.vue.d.ts +41 -0
- package/dist/components/edges/SmoothEdge.d.vue.ts +22 -0
- package/dist/components/edges/SmoothEdge.vue +53 -0
- package/dist/components/edges/SmoothEdge.vue.d.ts +22 -0
- package/dist/components/edges/StepEdge.d.vue.ts +22 -0
- package/dist/components/edges/StepEdge.vue +42 -0
- package/dist/components/edges/StepEdge.vue.d.ts +22 -0
- package/dist/components/edges/index.cjs +41 -0
- package/dist/components/edges/index.d.ts +5 -0
- package/dist/components/edges/index.mjs +5 -0
- package/dist/components/nodes/BaseNode.d.vue.ts +25 -0
- package/dist/components/nodes/BaseNode.vue +93 -0
- package/dist/components/nodes/BaseNode.vue.d.ts +25 -0
- package/dist/components/nodes/CustomNode.d.vue.ts +36 -0
- package/dist/components/nodes/CustomNode.vue +44 -0
- package/dist/components/nodes/CustomNode.vue.d.ts +36 -0
- package/dist/components/nodes/DatabaseNode.d.vue.ts +19 -0
- package/dist/components/nodes/DatabaseNode.vue +62 -0
- package/dist/components/nodes/DatabaseNode.vue.d.ts +19 -0
- package/dist/components/nodes/DiamondNode.d.vue.ts +19 -0
- package/dist/components/nodes/DiamondNode.vue +62 -0
- package/dist/components/nodes/DiamondNode.vue.d.ts +19 -0
- package/dist/components/nodes/GroupNode.d.vue.ts +31 -0
- package/dist/components/nodes/GroupNode.vue +48 -0
- package/dist/components/nodes/GroupNode.vue.d.ts +31 -0
- package/dist/components/nodes/InputNode.d.vue.ts +23 -0
- package/dist/components/nodes/InputNode.vue +63 -0
- package/dist/components/nodes/InputNode.vue.d.ts +23 -0
- package/dist/components/nodes/NodeResizer.d.vue.ts +27 -0
- package/dist/components/nodes/NodeResizer.vue +89 -0
- package/dist/components/nodes/NodeResizer.vue.d.ts +27 -0
- package/dist/components/nodes/NodeToolbar.d.vue.ts +32 -0
- package/dist/components/nodes/NodeToolbar.vue +101 -0
- package/dist/components/nodes/NodeToolbar.vue.d.ts +32 -0
- package/dist/components/nodes/OutputNode.d.vue.ts +21 -0
- package/dist/components/nodes/OutputNode.vue +53 -0
- package/dist/components/nodes/OutputNode.vue.d.ts +21 -0
- package/dist/components/nodes/ai-workflow/AiAgentNode.d.vue.ts +20 -0
- package/dist/components/nodes/ai-workflow/AiAgentNode.vue +59 -0
- package/dist/components/nodes/ai-workflow/AiAgentNode.vue.d.ts +20 -0
- package/dist/components/nodes/ai-workflow/AiConditionNode.d.vue.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiConditionNode.vue +65 -0
- package/dist/components/nodes/ai-workflow/AiConditionNode.vue.d.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiEndNode.d.vue.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiEndNode.vue +47 -0
- package/dist/components/nodes/ai-workflow/AiEndNode.vue.d.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiLlmNode.d.vue.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiLlmNode.vue +64 -0
- package/dist/components/nodes/ai-workflow/AiLlmNode.vue.d.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiMemoryNode.d.vue.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiMemoryNode.vue +59 -0
- package/dist/components/nodes/ai-workflow/AiMemoryNode.vue.d.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiPromptNode.d.vue.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiPromptNode.vue +61 -0
- package/dist/components/nodes/ai-workflow/AiPromptNode.vue.d.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiStartNode.d.vue.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiStartNode.vue +47 -0
- package/dist/components/nodes/ai-workflow/AiStartNode.vue.d.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiToolNode.d.vue.ts +19 -0
- package/dist/components/nodes/ai-workflow/AiToolNode.vue +59 -0
- package/dist/components/nodes/ai-workflow/AiToolNode.vue.d.ts +19 -0
- package/dist/components/nodes/ai-workflow/index.cjs +109 -0
- package/dist/components/nodes/ai-workflow/index.d.ts +23 -0
- package/dist/components/nodes/ai-workflow/index.mjs +37 -0
- package/dist/components/nodes/bpmn/BpmnEndEvent.d.vue.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnEndEvent.vue +50 -0
- package/dist/components/nodes/bpmn/BpmnEndEvent.vue.d.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnExclusiveGateway.d.vue.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnExclusiveGateway.vue +60 -0
- package/dist/components/nodes/bpmn/BpmnExclusiveGateway.vue.d.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnInclusiveGateway.d.vue.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnInclusiveGateway.vue +60 -0
- package/dist/components/nodes/bpmn/BpmnInclusiveGateway.vue.d.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnParallelGateway.d.vue.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnParallelGateway.vue +60 -0
- package/dist/components/nodes/bpmn/BpmnParallelGateway.vue.d.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnServiceTask.d.vue.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnServiceTask.vue +55 -0
- package/dist/components/nodes/bpmn/BpmnServiceTask.vue.d.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnStartEvent.d.vue.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnStartEvent.vue +50 -0
- package/dist/components/nodes/bpmn/BpmnStartEvent.vue.d.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnTask.d.vue.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnTask.vue +54 -0
- package/dist/components/nodes/bpmn/BpmnTask.vue.d.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnUserTask.d.vue.ts +19 -0
- package/dist/components/nodes/bpmn/BpmnUserTask.vue +55 -0
- package/dist/components/nodes/bpmn/BpmnUserTask.vue.d.ts +19 -0
- package/dist/components/nodes/bpmn/index.cjs +109 -0
- package/dist/components/nodes/bpmn/index.d.ts +23 -0
- package/dist/components/nodes/bpmn/index.mjs +37 -0
- package/dist/components/nodes/index.cjs +246 -0
- package/dist/components/nodes/index.d.ts +13 -0
- package/dist/components/nodes/index.mjs +44 -0
- package/dist/core/FlowContext.cjs +21 -0
- package/dist/core/FlowContext.d.ts +10 -0
- package/dist/core/FlowContext.mjs +13 -0
- package/dist/core/index.cjs +104 -0
- package/dist/core/index.d.ts +9 -0
- package/dist/core/index.mjs +9 -0
- package/dist/core/useAlignment.cjs +81 -0
- package/dist/core/useAlignment.d.ts +33 -0
- package/dist/core/useAlignment.mjs +71 -0
- package/dist/core/useEdges.cjs +132 -0
- package/dist/core/useEdges.d.ts +29 -0
- package/dist/core/useEdges.mjs +89 -0
- package/dist/core/useFlow.cjs +40 -0
- package/dist/core/useFlow.d.ts +31 -0
- package/dist/core/useFlow.mjs +32 -0
- package/dist/core/useHistory.cjs +63 -0
- package/dist/core/useHistory.d.ts +15 -0
- package/dist/core/useHistory.mjs +54 -0
- package/dist/core/useKeyboard.cjs +54 -0
- package/dist/core/useKeyboard.d.ts +18 -0
- package/dist/core/useKeyboard.mjs +45 -0
- package/dist/core/useNodeDistribution.cjs +171 -0
- package/dist/core/useNodeDistribution.d.ts +12 -0
- package/dist/core/useNodeDistribution.mjs +145 -0
- package/dist/core/useNodes.cjs +146 -0
- package/dist/core/useNodes.d.ts +26 -0
- package/dist/core/useNodes.mjs +101 -0
- package/dist/core/useSelection.cjs +83 -0
- package/dist/core/useSelection.d.ts +18 -0
- package/dist/core/useSelection.mjs +53 -0
- package/dist/core/useViewport.cjs +157 -0
- package/dist/core/useViewport.d.ts +65 -0
- package/dist/core/useViewport.mjs +125 -0
- package/dist/flow.cjs +240 -0
- package/dist/flow.d.ts +276 -0
- package/dist/flow.mjs +230 -0
- package/dist/index.cjs +378 -0
- package/dist/index.d.ts +28 -0
- package/dist/index.mjs +50 -0
- package/dist/plugins/index.cjs +96 -0
- package/dist/plugins/index.d.ts +34 -0
- package/dist/plugins/index.mjs +47 -0
- package/dist/plugins/plugin.cjs +117 -0
- package/dist/plugins/plugin.d.ts +72 -0
- package/dist/plugins/plugin.mjs +110 -0
- package/dist/plugins/plugins/controls.cjs +38 -0
- package/dist/plugins/plugins/controls.d.ts +12 -0
- package/dist/plugins/plugins/controls.mjs +28 -0
- package/dist/plugins/plugins/export.cjs +102 -0
- package/dist/plugins/plugins/export.d.ts +12 -0
- package/dist/plugins/plugins/export.mjs +89 -0
- package/dist/plugins/plugins/grid.cjs +36 -0
- package/dist/plugins/plugins/grid.d.ts +11 -0
- package/dist/plugins/plugins/grid.mjs +26 -0
- package/dist/plugins/plugins/history.cjs +140 -0
- package/dist/plugins/plugins/history.d.ts +53 -0
- package/dist/plugins/plugins/history.mjs +132 -0
- package/dist/plugins/plugins/index.cjs +104 -0
- package/dist/plugins/plugins/index.d.ts +9 -0
- package/dist/plugins/plugins/index.mjs +9 -0
- package/dist/plugins/plugins/keyboard.cjs +27 -0
- package/dist/plugins/plugins/keyboard.d.ts +10 -0
- package/dist/plugins/plugins/keyboard.mjs +18 -0
- package/dist/plugins/plugins/layout.cjs +275 -0
- package/dist/plugins/plugins/layout.d.ts +34 -0
- package/dist/plugins/plugins/layout.mjs +246 -0
- package/dist/plugins/plugins/minimap.cjs +60 -0
- package/dist/plugins/plugins/minimap.d.ts +25 -0
- package/dist/plugins/plugins/minimap.mjs +50 -0
- package/dist/plugins/plugins/node-group.cjs +209 -0
- package/dist/plugins/plugins/node-group.d.ts +26 -0
- package/dist/plugins/plugins/node-group.mjs +178 -0
- package/dist/plugins/plugins/snap.cjs +36 -0
- package/dist/plugins/plugins/snap.d.ts +12 -0
- package/dist/plugins/plugins/snap.mjs +26 -0
- package/dist/renderer/AlignmentLines.d.vue.ts +5 -0
- package/dist/renderer/AlignmentLines.vue +113 -0
- package/dist/renderer/AlignmentLines.vue.d.ts +5 -0
- package/dist/renderer/Background.d.vue.ts +7 -0
- package/dist/renderer/Background.vue +86 -0
- package/dist/renderer/Background.vue.d.ts +7 -0
- package/dist/renderer/Controls.d.vue.ts +13 -0
- package/dist/renderer/Controls.vue +82 -0
- package/dist/renderer/Controls.vue.d.ts +13 -0
- package/dist/renderer/EdgeHandlesRenderer.d.vue.ts +11 -0
- package/dist/renderer/EdgeHandlesRenderer.vue +75 -0
- package/dist/renderer/EdgeHandlesRenderer.vue.d.ts +11 -0
- package/dist/renderer/EdgeRenderer.d.vue.ts +39 -0
- package/dist/renderer/EdgeRenderer.vue +204 -0
- package/dist/renderer/EdgeRenderer.vue.d.ts +39 -0
- package/dist/renderer/FlowBackground.d.vue.ts +11 -0
- package/dist/renderer/FlowBackground.vue +82 -0
- package/dist/renderer/FlowBackground.vue.d.ts +11 -0
- package/dist/renderer/Minimap.d.vue.ts +30 -0
- package/dist/renderer/Minimap.vue +290 -0
- package/dist/renderer/Minimap.vue.d.ts +30 -0
- package/dist/renderer/NodeRenderer.d.vue.ts +56 -0
- package/dist/renderer/NodeRenderer.vue +328 -0
- package/dist/renderer/NodeRenderer.vue.d.ts +56 -0
- package/dist/renderer/SelectionBox.d.vue.ts +11 -0
- package/dist/renderer/SelectionBox.vue +28 -0
- package/dist/renderer/SelectionBox.vue.d.ts +11 -0
- package/dist/types/edge.cjs +13 -0
- package/dist/types/edge.d.ts +65 -0
- package/dist/types/edge.mjs +6 -0
- package/dist/types/events.cjs +1 -0
- package/dist/types/events.d.ts +115 -0
- package/dist/types/events.mjs +0 -0
- package/dist/types/index.cjs +1 -0
- package/dist/types/index.d.ts +366 -0
- package/dist/types/index.mjs +0 -0
- package/dist/types/node.cjs +9 -0
- package/dist/types/node.d.ts +90 -0
- package/dist/types/node.mjs +3 -0
- package/dist/types/viewport.cjs +42 -0
- package/dist/types/viewport.d.ts +62 -0
- package/dist/types/viewport.mjs +36 -0
- package/dist/utils/bpmn-engine.cjs +390 -0
- package/dist/utils/bpmn-engine.d.ts +164 -0
- package/dist/utils/bpmn-engine.mjs +378 -0
- package/dist/utils/bpmn.cjs +492 -0
- package/dist/utils/bpmn.d.ts +53 -0
- package/dist/utils/bpmn.mjs +430 -0
- package/dist/utils/collaboration.cjs +537 -0
- package/dist/utils/collaboration.d.ts +189 -0
- package/dist/utils/collaboration.mjs +521 -0
- package/dist/utils/custom-types.cjs +138 -0
- package/dist/utils/custom-types.d.ts +78 -0
- package/dist/utils/custom-types.mjs +108 -0
- package/dist/utils/edge.cjs +235 -0
- package/dist/utils/edge.d.ts +79 -0
- package/dist/utils/edge.mjs +172 -0
- package/dist/utils/event-bus.cjs +91 -0
- package/dist/utils/event-bus.d.ts +39 -0
- package/dist/utils/event-bus.mjs +82 -0
- package/dist/utils/geometry.cjs +178 -0
- package/dist/utils/geometry.d.ts +186 -0
- package/dist/utils/geometry.mjs +144 -0
- package/dist/utils/graph.cjs +158 -0
- package/dist/utils/graph.d.ts +40 -0
- package/dist/utils/graph.mjs +147 -0
- package/dist/utils/index.cjs +137 -0
- package/dist/utils/index.d.ts +12 -0
- package/dist/utils/index.mjs +12 -0
- package/dist/utils/performance.cjs +94 -0
- package/dist/utils/performance.d.ts +18 -0
- package/dist/utils/performance.mjs +82 -0
- package/dist/utils/screenshot.cjs +87 -0
- package/dist/utils/screenshot.d.ts +22 -0
- package/dist/utils/screenshot.mjs +66 -0
- package/dist/utils/theme.cjs +228 -0
- package/dist/utils/theme.d.ts +92 -0
- package/dist/utils/theme.mjs +217 -0
- package/dist/utils/transform.cjs +76 -0
- package/dist/utils/transform.d.ts +45 -0
- package/dist/utils/transform.mjs +57 -0
- package/dist/utils/validation.cjs +107 -0
- package/dist/utils/validation.d.ts +29 -0
- package/dist/utils/validation.mjs +85 -0
- package/package.json +61 -0
|
@@ -0,0 +1,275 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.createLayoutPlugin = createLayoutPlugin;
|
|
7
|
+
const defaultOptions = {
|
|
8
|
+
enabled: true,
|
|
9
|
+
type: "dagre",
|
|
10
|
+
direction: "TB",
|
|
11
|
+
nodeSpacing: 50,
|
|
12
|
+
rankSpacing: 80,
|
|
13
|
+
animate: true,
|
|
14
|
+
elkOptions: {},
|
|
15
|
+
forceOptions: {},
|
|
16
|
+
gridOptions: {
|
|
17
|
+
columns: 4
|
|
18
|
+
},
|
|
19
|
+
useWebWorker: false,
|
|
20
|
+
workerUrl: ""
|
|
21
|
+
};
|
|
22
|
+
async function applyDagreLayout(nodes, edges, options) {
|
|
23
|
+
const dagreLib = await Promise.resolve().then(() => require("dagre"));
|
|
24
|
+
const dagre = dagreLib.default || dagreLib;
|
|
25
|
+
const graphlib = dagreLib.graphlib || dagre.graphlib;
|
|
26
|
+
const g = new graphlib.Graph();
|
|
27
|
+
g.setGraph({
|
|
28
|
+
rankdir: options.direction,
|
|
29
|
+
nodesep: options.nodeSpacing,
|
|
30
|
+
ranksep: options.rankSpacing,
|
|
31
|
+
marginx: 0,
|
|
32
|
+
marginy: 0
|
|
33
|
+
});
|
|
34
|
+
g.setDefaultEdgeLabel(() => ({}));
|
|
35
|
+
nodes.forEach(node => {
|
|
36
|
+
g.setNode(node.id, {
|
|
37
|
+
width: node.width || 150,
|
|
38
|
+
height: node.height || 50
|
|
39
|
+
});
|
|
40
|
+
});
|
|
41
|
+
edges.forEach(edge => {
|
|
42
|
+
g.setEdge(edge.source, edge.target);
|
|
43
|
+
});
|
|
44
|
+
dagre.layout(g);
|
|
45
|
+
const layoutedNodes = nodes.map(node => {
|
|
46
|
+
const layoutNode = g.node(node.id);
|
|
47
|
+
if (!layoutNode) return node;
|
|
48
|
+
return {
|
|
49
|
+
...node,
|
|
50
|
+
position: {
|
|
51
|
+
x: layoutNode.x - (node.width || 150) / 2,
|
|
52
|
+
y: layoutNode.y - (node.height || 50) / 2
|
|
53
|
+
}
|
|
54
|
+
};
|
|
55
|
+
});
|
|
56
|
+
return {
|
|
57
|
+
nodes: layoutedNodes,
|
|
58
|
+
edges
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
async function applyElkLayout(nodes, edges, options) {
|
|
62
|
+
const bundledPath = "elkjs/lib/elk.bundled.js";
|
|
63
|
+
const elkPath = "elkjs";
|
|
64
|
+
let elkLib;
|
|
65
|
+
try {
|
|
66
|
+
elkLib = await Promise.resolve(`${/* @vite-ignore */
|
|
67
|
+
bundledPath}`).then(s => require(s));
|
|
68
|
+
} catch {
|
|
69
|
+
elkLib = await Promise.resolve(`${/* @vite-ignore */
|
|
70
|
+
elkPath}`).then(s => require(s));
|
|
71
|
+
}
|
|
72
|
+
const ELK = elkLib.default || elkLib;
|
|
73
|
+
const elk = new ELK();
|
|
74
|
+
const dirMap = {
|
|
75
|
+
TB: "DOWN",
|
|
76
|
+
BT: "UP",
|
|
77
|
+
LR: "RIGHT",
|
|
78
|
+
RL: "LEFT"
|
|
79
|
+
};
|
|
80
|
+
const elkOpts = options.elkOptions || {};
|
|
81
|
+
const graph = {
|
|
82
|
+
id: "root",
|
|
83
|
+
layoutOptions: {
|
|
84
|
+
"elk.algorithm": elkOpts.algorithm || "layered",
|
|
85
|
+
"elk.direction": elkOpts.direction || dirMap[options.direction] || "DOWN",
|
|
86
|
+
"elk.spacing.nodeNode": elkOpts.spacing || options.nodeSpacing,
|
|
87
|
+
"elk.edgeRouting": elkOpts.edgeRouting || "POLYLINE"
|
|
88
|
+
},
|
|
89
|
+
children: nodes.map(node => ({
|
|
90
|
+
id: node.id,
|
|
91
|
+
width: node.width || 150,
|
|
92
|
+
height: node.height || 50
|
|
93
|
+
})),
|
|
94
|
+
edges: edges.map((edge, index) => ({
|
|
95
|
+
id: edge.id || `edge-${index}`,
|
|
96
|
+
sources: [edge.source],
|
|
97
|
+
targets: [edge.target]
|
|
98
|
+
}))
|
|
99
|
+
};
|
|
100
|
+
const layoutedGraph = await elk.layout(graph);
|
|
101
|
+
const layoutedNodes = nodes.map(node => {
|
|
102
|
+
const layoutNode = layoutedGraph.children?.find(n => n.id === node.id);
|
|
103
|
+
if (!layoutNode) return node;
|
|
104
|
+
return {
|
|
105
|
+
...node,
|
|
106
|
+
position: {
|
|
107
|
+
x: layoutNode.x || 0,
|
|
108
|
+
y: layoutNode.y || 0
|
|
109
|
+
}
|
|
110
|
+
};
|
|
111
|
+
});
|
|
112
|
+
return {
|
|
113
|
+
nodes: layoutedNodes,
|
|
114
|
+
edges
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
async function applyForceLayout(nodes, edges, options, flowInstance) {
|
|
118
|
+
const d3ForcePath = "d3-force";
|
|
119
|
+
const d3ForceLib = await Promise.resolve(`${/* @vite-ignore */
|
|
120
|
+
d3ForcePath}`).then(s => require(s));
|
|
121
|
+
const d3Force = d3ForceLib.default || d3ForceLib;
|
|
122
|
+
const forceNodes = nodes.map(node => ({
|
|
123
|
+
id: node.id,
|
|
124
|
+
x: node.position.x,
|
|
125
|
+
y: node.position.y
|
|
126
|
+
}));
|
|
127
|
+
const force = d3Force.forceSimulation(forceNodes);
|
|
128
|
+
const forceOpts = options.forceOptions || {};
|
|
129
|
+
force.force("charge", d3Force.forceManyBody().strength(forceOpts.strength || -300));
|
|
130
|
+
const linkForce = d3Force.forceLink();
|
|
131
|
+
linkForce.id(d => d.id);
|
|
132
|
+
linkForce.distance(forceOpts.distance || 100);
|
|
133
|
+
force.force("link", linkForce);
|
|
134
|
+
force.force("collision", d3Force.forceCollide().radius(50));
|
|
135
|
+
const centerX = 400;
|
|
136
|
+
const centerY = 300;
|
|
137
|
+
force.force("center", d3Force.forceCenter(centerX, centerY));
|
|
138
|
+
const iterations = forceOpts.iterations || 300;
|
|
139
|
+
const ticksPerFrame = Math.max(1, Math.floor(iterations / 10));
|
|
140
|
+
return new Promise(resolve => {
|
|
141
|
+
let currentIteration = 0;
|
|
142
|
+
const step = () => {
|
|
143
|
+
const toRun = Math.min(ticksPerFrame, iterations - currentIteration);
|
|
144
|
+
for (let i = 0; i < toRun; i++) {
|
|
145
|
+
force.tick();
|
|
146
|
+
}
|
|
147
|
+
currentIteration += toRun;
|
|
148
|
+
let minX = Infinity,
|
|
149
|
+
minY = Infinity;
|
|
150
|
+
forceNodes.forEach(n => {
|
|
151
|
+
if (n.x < minX) minX = n.x;
|
|
152
|
+
if (n.y < minY) minY = n.y;
|
|
153
|
+
});
|
|
154
|
+
const offsetX = minX < 0 ? -minX + 50 : 50;
|
|
155
|
+
const offsetY = minY < 0 ? -minY + 50 : 50;
|
|
156
|
+
if (options.animate && flowInstance) {
|
|
157
|
+
forceNodes.forEach(fn => {
|
|
158
|
+
flowInstance.updateNode(fn.id, {
|
|
159
|
+
position: {
|
|
160
|
+
x: fn.x + offsetX,
|
|
161
|
+
y: fn.y + offsetY
|
|
162
|
+
}
|
|
163
|
+
});
|
|
164
|
+
});
|
|
165
|
+
}
|
|
166
|
+
if (currentIteration < iterations && force.alpha() > 5e-3) {
|
|
167
|
+
if (typeof window !== "undefined" && window.requestAnimationFrame) {
|
|
168
|
+
window.requestAnimationFrame(step);
|
|
169
|
+
} else {
|
|
170
|
+
setTimeout(step, 0);
|
|
171
|
+
}
|
|
172
|
+
} else {
|
|
173
|
+
force.stop();
|
|
174
|
+
const layoutedNodes = nodes.map(node => {
|
|
175
|
+
const forceNode = forceNodes.find(n => n.id === node.id);
|
|
176
|
+
if (!forceNode) return node;
|
|
177
|
+
return {
|
|
178
|
+
...node,
|
|
179
|
+
position: {
|
|
180
|
+
x: forceNode.x + offsetX,
|
|
181
|
+
y: forceNode.y + offsetY
|
|
182
|
+
}
|
|
183
|
+
};
|
|
184
|
+
});
|
|
185
|
+
resolve({
|
|
186
|
+
nodes: layoutedNodes,
|
|
187
|
+
edges
|
|
188
|
+
});
|
|
189
|
+
}
|
|
190
|
+
};
|
|
191
|
+
step();
|
|
192
|
+
});
|
|
193
|
+
}
|
|
194
|
+
function applyGridLayout(nodes, edges, options) {
|
|
195
|
+
const gridOpts = options.gridOptions || {};
|
|
196
|
+
const columns = gridOpts.columns || 4;
|
|
197
|
+
const startX = gridOpts.startX || 50;
|
|
198
|
+
const startY = gridOpts.startY || 50;
|
|
199
|
+
const nodeWidth = options.nodeSpacing + (nodes[0]?.width || 150);
|
|
200
|
+
const nodeHeight = options.rankSpacing + (nodes[0]?.height || 50);
|
|
201
|
+
const layoutedNodes = nodes.map((node, index) => {
|
|
202
|
+
const col = index % columns;
|
|
203
|
+
const row = Math.floor(index / columns);
|
|
204
|
+
return {
|
|
205
|
+
...node,
|
|
206
|
+
position: {
|
|
207
|
+
x: startX + col * nodeWidth,
|
|
208
|
+
y: startY + row * nodeHeight
|
|
209
|
+
}
|
|
210
|
+
};
|
|
211
|
+
});
|
|
212
|
+
return {
|
|
213
|
+
nodes: layoutedNodes,
|
|
214
|
+
edges
|
|
215
|
+
};
|
|
216
|
+
}
|
|
217
|
+
function createLayoutPlugin(options = {}) {
|
|
218
|
+
const mergedOptions = {
|
|
219
|
+
...defaultOptions,
|
|
220
|
+
...options
|
|
221
|
+
};
|
|
222
|
+
return {
|
|
223
|
+
id: "layout",
|
|
224
|
+
name: "Layout",
|
|
225
|
+
version: "1.0.0",
|
|
226
|
+
description: "Provides automatic layout algorithms for flow charts (dagre, elk, force)",
|
|
227
|
+
install(flow) {
|
|
228
|
+
if (!mergedOptions.enabled) return;
|
|
229
|
+
flow.applyLayout = async layoutOptions => {
|
|
230
|
+
const opts = {
|
|
231
|
+
...mergedOptions,
|
|
232
|
+
...layoutOptions
|
|
233
|
+
};
|
|
234
|
+
const nodes = [...flow.nodes.value];
|
|
235
|
+
const edges = [...flow.edges.value];
|
|
236
|
+
try {
|
|
237
|
+
let layouted;
|
|
238
|
+
if (opts.type === "dagre") {
|
|
239
|
+
layouted = await applyDagreLayout(nodes, edges, opts);
|
|
240
|
+
console.log("[Layout Plugin] Dagre layout applied successfully");
|
|
241
|
+
} else if (opts.type === "elk") {
|
|
242
|
+
layouted = await applyElkLayout(nodes, edges, opts);
|
|
243
|
+
console.log("[Layout Plugin] ELK layout applied successfully");
|
|
244
|
+
} else if (opts.type === "force") {
|
|
245
|
+
layouted = await applyForceLayout(nodes, edges, opts, flow);
|
|
246
|
+
console.log("[Layout Plugin] Force layout applied asynchronously");
|
|
247
|
+
} else if (opts.type === "grid") {
|
|
248
|
+
layouted = applyGridLayout(nodes, edges, opts);
|
|
249
|
+
console.log("[Layout Plugin] Grid layout applied successfully");
|
|
250
|
+
} else {
|
|
251
|
+
console.warn(`[Layout Plugin] Unknown layout type '${opts.type}'`);
|
|
252
|
+
return;
|
|
253
|
+
}
|
|
254
|
+
layouted.nodes.forEach(layoutedNode => {
|
|
255
|
+
flow.updateNode(layoutedNode.id, {
|
|
256
|
+
position: {
|
|
257
|
+
x: layoutedNode.position.x,
|
|
258
|
+
y: layoutedNode.position.y
|
|
259
|
+
}
|
|
260
|
+
});
|
|
261
|
+
});
|
|
262
|
+
if (opts.animate) {
|
|
263
|
+
setTimeout(() => {
|
|
264
|
+
flow.fitView?.({
|
|
265
|
+
padding: 50
|
|
266
|
+
});
|
|
267
|
+
}, 100);
|
|
268
|
+
}
|
|
269
|
+
} catch (error) {
|
|
270
|
+
console.error("[Layout Plugin] Layout calculation failed:", error);
|
|
271
|
+
}
|
|
272
|
+
};
|
|
273
|
+
}
|
|
274
|
+
};
|
|
275
|
+
}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import type { FlowPlugin } from '../plugin';
|
|
2
|
+
export interface LayoutOptions {
|
|
3
|
+
enabled?: boolean;
|
|
4
|
+
type?: 'dagre' | 'elk' | 'force' | 'grid';
|
|
5
|
+
direction?: 'TB' | 'BT' | 'LR' | 'RL';
|
|
6
|
+
nodeSpacing?: number;
|
|
7
|
+
rankSpacing?: number;
|
|
8
|
+
animate?: boolean;
|
|
9
|
+
/** ELK 布局专属选项 */
|
|
10
|
+
elkOptions?: {
|
|
11
|
+
algorithm?: 'layered' | 'force' | 'mrtree' | 'box' | 'disco';
|
|
12
|
+
direction?: 'DOWN' | 'RIGHT' | 'UP' | 'LEFT';
|
|
13
|
+
spacing?: number;
|
|
14
|
+
edgeRouting?: 'ORTHOGONAL' | 'POLYLINE' | 'SPLINES';
|
|
15
|
+
};
|
|
16
|
+
/** Force 布局专属选项 */
|
|
17
|
+
forceOptions?: {
|
|
18
|
+
strength?: number;
|
|
19
|
+
distance?: number;
|
|
20
|
+
theta?: number;
|
|
21
|
+
iterations?: number;
|
|
22
|
+
};
|
|
23
|
+
/** Grid 布局专属选项 */
|
|
24
|
+
gridOptions?: {
|
|
25
|
+
columns?: number;
|
|
26
|
+
startX?: number;
|
|
27
|
+
startY?: number;
|
|
28
|
+
};
|
|
29
|
+
/** 是否启用 Web Worker 进行密集计算(如用户自行提供打包的 workerUrl) */
|
|
30
|
+
useWebWorker?: boolean;
|
|
31
|
+
/** 自定义 Web Worker 的 URL,配合 useWebWorker 使用 */
|
|
32
|
+
workerUrl?: string;
|
|
33
|
+
}
|
|
34
|
+
export declare function createLayoutPlugin(options?: LayoutOptions): FlowPlugin;
|
|
@@ -0,0 +1,246 @@
|
|
|
1
|
+
const defaultOptions = {
|
|
2
|
+
enabled: true,
|
|
3
|
+
type: "dagre",
|
|
4
|
+
direction: "TB",
|
|
5
|
+
nodeSpacing: 50,
|
|
6
|
+
rankSpacing: 80,
|
|
7
|
+
animate: true,
|
|
8
|
+
elkOptions: {},
|
|
9
|
+
forceOptions: {},
|
|
10
|
+
gridOptions: { columns: 4 },
|
|
11
|
+
useWebWorker: false,
|
|
12
|
+
workerUrl: ""
|
|
13
|
+
};
|
|
14
|
+
async function applyDagreLayout(nodes, edges, options) {
|
|
15
|
+
const dagreLib = await import("dagre");
|
|
16
|
+
const dagre = dagreLib.default || dagreLib;
|
|
17
|
+
const graphlib = dagreLib.graphlib || dagre.graphlib;
|
|
18
|
+
const g = new graphlib.Graph();
|
|
19
|
+
g.setGraph({
|
|
20
|
+
rankdir: options.direction,
|
|
21
|
+
nodesep: options.nodeSpacing,
|
|
22
|
+
ranksep: options.rankSpacing,
|
|
23
|
+
marginx: 0,
|
|
24
|
+
marginy: 0
|
|
25
|
+
});
|
|
26
|
+
g.setDefaultEdgeLabel(() => ({}));
|
|
27
|
+
nodes.forEach((node) => {
|
|
28
|
+
g.setNode(node.id, {
|
|
29
|
+
width: node.width || 150,
|
|
30
|
+
height: node.height || 50
|
|
31
|
+
});
|
|
32
|
+
});
|
|
33
|
+
edges.forEach((edge) => {
|
|
34
|
+
g.setEdge(edge.source, edge.target);
|
|
35
|
+
});
|
|
36
|
+
dagre.layout(g);
|
|
37
|
+
const layoutedNodes = nodes.map((node) => {
|
|
38
|
+
const layoutNode = g.node(node.id);
|
|
39
|
+
if (!layoutNode) return node;
|
|
40
|
+
return {
|
|
41
|
+
...node,
|
|
42
|
+
position: {
|
|
43
|
+
x: layoutNode.x - (node.width || 150) / 2,
|
|
44
|
+
y: layoutNode.y - (node.height || 50) / 2
|
|
45
|
+
}
|
|
46
|
+
};
|
|
47
|
+
});
|
|
48
|
+
return { nodes: layoutedNodes, edges };
|
|
49
|
+
}
|
|
50
|
+
async function applyElkLayout(nodes, edges, options) {
|
|
51
|
+
const bundledPath = "elkjs/lib/elk.bundled.js";
|
|
52
|
+
const elkPath = "elkjs";
|
|
53
|
+
let elkLib;
|
|
54
|
+
try {
|
|
55
|
+
elkLib = await import(
|
|
56
|
+
/* @vite-ignore */
|
|
57
|
+
bundledPath
|
|
58
|
+
);
|
|
59
|
+
} catch {
|
|
60
|
+
elkLib = await import(
|
|
61
|
+
/* @vite-ignore */
|
|
62
|
+
elkPath
|
|
63
|
+
);
|
|
64
|
+
}
|
|
65
|
+
const ELK = elkLib.default || elkLib;
|
|
66
|
+
const elk = new ELK();
|
|
67
|
+
const dirMap = {
|
|
68
|
+
TB: "DOWN",
|
|
69
|
+
BT: "UP",
|
|
70
|
+
LR: "RIGHT",
|
|
71
|
+
RL: "LEFT"
|
|
72
|
+
};
|
|
73
|
+
const elkOpts = options.elkOptions || {};
|
|
74
|
+
const graph = {
|
|
75
|
+
id: "root",
|
|
76
|
+
layoutOptions: {
|
|
77
|
+
"elk.algorithm": elkOpts.algorithm || "layered",
|
|
78
|
+
"elk.direction": elkOpts.direction || dirMap[options.direction] || "DOWN",
|
|
79
|
+
"elk.spacing.nodeNode": elkOpts.spacing || options.nodeSpacing,
|
|
80
|
+
"elk.edgeRouting": elkOpts.edgeRouting || "POLYLINE"
|
|
81
|
+
},
|
|
82
|
+
children: nodes.map((node) => ({
|
|
83
|
+
id: node.id,
|
|
84
|
+
width: node.width || 150,
|
|
85
|
+
height: node.height || 50
|
|
86
|
+
})),
|
|
87
|
+
edges: edges.map((edge, index) => ({
|
|
88
|
+
id: edge.id || `edge-${index}`,
|
|
89
|
+
sources: [edge.source],
|
|
90
|
+
targets: [edge.target]
|
|
91
|
+
}))
|
|
92
|
+
};
|
|
93
|
+
const layoutedGraph = await elk.layout(graph);
|
|
94
|
+
const layoutedNodes = nodes.map((node) => {
|
|
95
|
+
const layoutNode = layoutedGraph.children?.find((n) => n.id === node.id);
|
|
96
|
+
if (!layoutNode) return node;
|
|
97
|
+
return {
|
|
98
|
+
...node,
|
|
99
|
+
position: {
|
|
100
|
+
x: layoutNode.x || 0,
|
|
101
|
+
y: layoutNode.y || 0
|
|
102
|
+
}
|
|
103
|
+
};
|
|
104
|
+
});
|
|
105
|
+
return { nodes: layoutedNodes, edges };
|
|
106
|
+
}
|
|
107
|
+
async function applyForceLayout(nodes, edges, options, flowInstance) {
|
|
108
|
+
const d3ForcePath = "d3-force";
|
|
109
|
+
const d3ForceLib = await import(
|
|
110
|
+
/* @vite-ignore */
|
|
111
|
+
d3ForcePath
|
|
112
|
+
);
|
|
113
|
+
const d3Force = d3ForceLib.default || d3ForceLib;
|
|
114
|
+
const forceNodes = nodes.map((node) => ({
|
|
115
|
+
id: node.id,
|
|
116
|
+
x: node.position.x,
|
|
117
|
+
y: node.position.y
|
|
118
|
+
}));
|
|
119
|
+
const force = d3Force.forceSimulation(forceNodes);
|
|
120
|
+
const forceOpts = options.forceOptions || {};
|
|
121
|
+
force.force("charge", d3Force.forceManyBody().strength(forceOpts.strength || -300));
|
|
122
|
+
const linkForce = d3Force.forceLink();
|
|
123
|
+
linkForce.id((d) => d.id);
|
|
124
|
+
linkForce.distance(forceOpts.distance || 100);
|
|
125
|
+
force.force("link", linkForce);
|
|
126
|
+
force.force("collision", d3Force.forceCollide().radius(50));
|
|
127
|
+
const centerX = 400;
|
|
128
|
+
const centerY = 300;
|
|
129
|
+
force.force("center", d3Force.forceCenter(centerX, centerY));
|
|
130
|
+
const iterations = forceOpts.iterations || 300;
|
|
131
|
+
const ticksPerFrame = Math.max(1, Math.floor(iterations / 10));
|
|
132
|
+
return new Promise((resolve) => {
|
|
133
|
+
let currentIteration = 0;
|
|
134
|
+
const step = () => {
|
|
135
|
+
const toRun = Math.min(ticksPerFrame, iterations - currentIteration);
|
|
136
|
+
for (let i = 0; i < toRun; i++) {
|
|
137
|
+
force.tick();
|
|
138
|
+
}
|
|
139
|
+
currentIteration += toRun;
|
|
140
|
+
let minX = Infinity, minY = Infinity;
|
|
141
|
+
forceNodes.forEach((n) => {
|
|
142
|
+
if (n.x < minX) minX = n.x;
|
|
143
|
+
if (n.y < minY) minY = n.y;
|
|
144
|
+
});
|
|
145
|
+
const offsetX = minX < 0 ? -minX + 50 : 50;
|
|
146
|
+
const offsetY = minY < 0 ? -minY + 50 : 50;
|
|
147
|
+
if (options.animate && flowInstance) {
|
|
148
|
+
forceNodes.forEach((fn) => {
|
|
149
|
+
flowInstance.updateNode(fn.id, {
|
|
150
|
+
position: { x: fn.x + offsetX, y: fn.y + offsetY }
|
|
151
|
+
});
|
|
152
|
+
});
|
|
153
|
+
}
|
|
154
|
+
if (currentIteration < iterations && force.alpha() > 5e-3) {
|
|
155
|
+
if (typeof window !== "undefined" && window.requestAnimationFrame) {
|
|
156
|
+
window.requestAnimationFrame(step);
|
|
157
|
+
} else {
|
|
158
|
+
setTimeout(step, 0);
|
|
159
|
+
}
|
|
160
|
+
} else {
|
|
161
|
+
force.stop();
|
|
162
|
+
const layoutedNodes = nodes.map((node) => {
|
|
163
|
+
const forceNode = forceNodes.find((n) => n.id === node.id);
|
|
164
|
+
if (!forceNode) return node;
|
|
165
|
+
return {
|
|
166
|
+
...node,
|
|
167
|
+
position: { x: forceNode.x + offsetX, y: forceNode.y + offsetY }
|
|
168
|
+
};
|
|
169
|
+
});
|
|
170
|
+
resolve({ nodes: layoutedNodes, edges });
|
|
171
|
+
}
|
|
172
|
+
};
|
|
173
|
+
step();
|
|
174
|
+
});
|
|
175
|
+
}
|
|
176
|
+
function applyGridLayout(nodes, edges, options) {
|
|
177
|
+
const gridOpts = options.gridOptions || {};
|
|
178
|
+
const columns = gridOpts.columns || 4;
|
|
179
|
+
const startX = gridOpts.startX || 50;
|
|
180
|
+
const startY = gridOpts.startY || 50;
|
|
181
|
+
const nodeWidth = options.nodeSpacing + (nodes[0]?.width || 150);
|
|
182
|
+
const nodeHeight = options.rankSpacing + (nodes[0]?.height || 50);
|
|
183
|
+
const layoutedNodes = nodes.map((node, index) => {
|
|
184
|
+
const col = index % columns;
|
|
185
|
+
const row = Math.floor(index / columns);
|
|
186
|
+
return {
|
|
187
|
+
...node,
|
|
188
|
+
position: {
|
|
189
|
+
x: startX + col * nodeWidth,
|
|
190
|
+
y: startY + row * nodeHeight
|
|
191
|
+
}
|
|
192
|
+
};
|
|
193
|
+
});
|
|
194
|
+
return { nodes: layoutedNodes, edges };
|
|
195
|
+
}
|
|
196
|
+
export function createLayoutPlugin(options = {}) {
|
|
197
|
+
const mergedOptions = { ...defaultOptions, ...options };
|
|
198
|
+
return {
|
|
199
|
+
id: "layout",
|
|
200
|
+
name: "Layout",
|
|
201
|
+
version: "1.0.0",
|
|
202
|
+
description: "Provides automatic layout algorithms for flow charts (dagre, elk, force)",
|
|
203
|
+
install(flow) {
|
|
204
|
+
if (!mergedOptions.enabled) return;
|
|
205
|
+
flow.applyLayout = async (layoutOptions) => {
|
|
206
|
+
const opts = { ...mergedOptions, ...layoutOptions };
|
|
207
|
+
const nodes = [...flow.nodes.value];
|
|
208
|
+
const edges = [...flow.edges.value];
|
|
209
|
+
try {
|
|
210
|
+
let layouted;
|
|
211
|
+
if (opts.type === "dagre") {
|
|
212
|
+
layouted = await applyDagreLayout(nodes, edges, opts);
|
|
213
|
+
console.log("[Layout Plugin] Dagre layout applied successfully");
|
|
214
|
+
} else if (opts.type === "elk") {
|
|
215
|
+
layouted = await applyElkLayout(nodes, edges, opts);
|
|
216
|
+
console.log("[Layout Plugin] ELK layout applied successfully");
|
|
217
|
+
} else if (opts.type === "force") {
|
|
218
|
+
layouted = await applyForceLayout(nodes, edges, opts, flow);
|
|
219
|
+
console.log("[Layout Plugin] Force layout applied asynchronously");
|
|
220
|
+
} else if (opts.type === "grid") {
|
|
221
|
+
layouted = applyGridLayout(nodes, edges, opts);
|
|
222
|
+
console.log("[Layout Plugin] Grid layout applied successfully");
|
|
223
|
+
} else {
|
|
224
|
+
console.warn(`[Layout Plugin] Unknown layout type '${opts.type}'`);
|
|
225
|
+
return;
|
|
226
|
+
}
|
|
227
|
+
layouted.nodes.forEach((layoutedNode) => {
|
|
228
|
+
flow.updateNode(layoutedNode.id, {
|
|
229
|
+
position: {
|
|
230
|
+
x: layoutedNode.position.x,
|
|
231
|
+
y: layoutedNode.position.y
|
|
232
|
+
}
|
|
233
|
+
});
|
|
234
|
+
});
|
|
235
|
+
if (opts.animate) {
|
|
236
|
+
setTimeout(() => {
|
|
237
|
+
flow.fitView?.({ padding: 50 });
|
|
238
|
+
}, 100);
|
|
239
|
+
}
|
|
240
|
+
} catch (error) {
|
|
241
|
+
console.error("[Layout Plugin] Layout calculation failed:", error);
|
|
242
|
+
}
|
|
243
|
+
};
|
|
244
|
+
}
|
|
245
|
+
};
|
|
246
|
+
}
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.createMiniMapPlugin = createMiniMapPlugin;
|
|
7
|
+
var _vue = require("vue");
|
|
8
|
+
var _Minimap = _interopRequireDefault(require("../../renderer/Minimap.vue"));
|
|
9
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
10
|
+
const defaultOptions = {
|
|
11
|
+
enabled: true,
|
|
12
|
+
nodeColor: () => "#fff",
|
|
13
|
+
nodeStrokeColor: () => "#999",
|
|
14
|
+
nodeStrokeWidth: 1,
|
|
15
|
+
maskColor: "rgba(240, 240, 240, 0.6)",
|
|
16
|
+
maskStrokeColor: "#ddd",
|
|
17
|
+
maskStrokeWidth: 1,
|
|
18
|
+
pannable: true,
|
|
19
|
+
zoomable: true,
|
|
20
|
+
position: "bottom-right",
|
|
21
|
+
width: 200,
|
|
22
|
+
height: 150,
|
|
23
|
+
interactive: false,
|
|
24
|
+
showLayoutControls: false,
|
|
25
|
+
layoutType: "none",
|
|
26
|
+
layoutDirection: "TB"
|
|
27
|
+
};
|
|
28
|
+
function createMiniMapPlugin(options = {}) {
|
|
29
|
+
const mergedOptions = {
|
|
30
|
+
...defaultOptions,
|
|
31
|
+
...options
|
|
32
|
+
};
|
|
33
|
+
return {
|
|
34
|
+
id: "minimap",
|
|
35
|
+
name: "Minimap",
|
|
36
|
+
version: "1.0.0",
|
|
37
|
+
description: "Displays a minimap for navigation",
|
|
38
|
+
component: (0, _vue.markRaw)(_Minimap.default),
|
|
39
|
+
componentProps: {
|
|
40
|
+
position: mergedOptions.position,
|
|
41
|
+
width: mergedOptions.width,
|
|
42
|
+
height: mergedOptions.height,
|
|
43
|
+
nodeColor: mergedOptions.nodeColor,
|
|
44
|
+
nodeStrokeColor: mergedOptions.nodeStrokeColor,
|
|
45
|
+
nodeStrokeWidth: mergedOptions.nodeStrokeWidth,
|
|
46
|
+
maskColor: mergedOptions.maskColor,
|
|
47
|
+
maskStrokeColor: mergedOptions.maskStrokeColor,
|
|
48
|
+
maskStrokeWidth: mergedOptions.maskStrokeWidth,
|
|
49
|
+
pannable: mergedOptions.pannable,
|
|
50
|
+
zoomable: mergedOptions.zoomable,
|
|
51
|
+
interactive: mergedOptions.interactive,
|
|
52
|
+
showLayoutControls: mergedOptions.showLayoutControls,
|
|
53
|
+
layoutType: mergedOptions.layoutType,
|
|
54
|
+
layoutDirection: mergedOptions.layoutDirection
|
|
55
|
+
},
|
|
56
|
+
install(flow) {
|
|
57
|
+
console.log("[Minimap Plugin] Installed", flow);
|
|
58
|
+
}
|
|
59
|
+
};
|
|
60
|
+
}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import type { FlowPlugin } from '../plugin';
|
|
2
|
+
import type { Node } from '../../types';
|
|
3
|
+
export interface MiniMapOptions {
|
|
4
|
+
enabled?: boolean;
|
|
5
|
+
nodeColor?: (node: Node) => string;
|
|
6
|
+
nodeStrokeColor?: (node: Node) => string;
|
|
7
|
+
nodeStrokeWidth?: number;
|
|
8
|
+
maskColor?: string;
|
|
9
|
+
maskStrokeColor?: string;
|
|
10
|
+
maskStrokeWidth?: number;
|
|
11
|
+
pannable?: boolean;
|
|
12
|
+
zoomable?: boolean;
|
|
13
|
+
position?: 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right';
|
|
14
|
+
width?: number;
|
|
15
|
+
height?: number;
|
|
16
|
+
/** Enable interactive click-to-navigate */
|
|
17
|
+
interactive?: boolean;
|
|
18
|
+
/** Show layout control buttons */
|
|
19
|
+
showLayoutControls?: boolean;
|
|
20
|
+
/** Current layout type */
|
|
21
|
+
layoutType?: 'dagre' | 'elk' | 'force' | 'grid' | 'none';
|
|
22
|
+
/** Layout direction */
|
|
23
|
+
layoutDirection?: 'TB' | 'BT' | 'LR' | 'RL';
|
|
24
|
+
}
|
|
25
|
+
export declare function createMiniMapPlugin(options?: MiniMapOptions): FlowPlugin;
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { markRaw } from "vue";
|
|
2
|
+
import Minimap from "../../renderer/Minimap.vue";
|
|
3
|
+
const defaultOptions = {
|
|
4
|
+
enabled: true,
|
|
5
|
+
nodeColor: () => "#fff",
|
|
6
|
+
nodeStrokeColor: () => "#999",
|
|
7
|
+
nodeStrokeWidth: 1,
|
|
8
|
+
maskColor: "rgba(240, 240, 240, 0.6)",
|
|
9
|
+
maskStrokeColor: "#ddd",
|
|
10
|
+
maskStrokeWidth: 1,
|
|
11
|
+
pannable: true,
|
|
12
|
+
zoomable: true,
|
|
13
|
+
position: "bottom-right",
|
|
14
|
+
width: 200,
|
|
15
|
+
height: 150,
|
|
16
|
+
interactive: false,
|
|
17
|
+
showLayoutControls: false,
|
|
18
|
+
layoutType: "none",
|
|
19
|
+
layoutDirection: "TB"
|
|
20
|
+
};
|
|
21
|
+
export function createMiniMapPlugin(options = {}) {
|
|
22
|
+
const mergedOptions = { ...defaultOptions, ...options };
|
|
23
|
+
return {
|
|
24
|
+
id: "minimap",
|
|
25
|
+
name: "Minimap",
|
|
26
|
+
version: "1.0.0",
|
|
27
|
+
description: "Displays a minimap for navigation",
|
|
28
|
+
component: markRaw(Minimap),
|
|
29
|
+
componentProps: {
|
|
30
|
+
position: mergedOptions.position,
|
|
31
|
+
width: mergedOptions.width,
|
|
32
|
+
height: mergedOptions.height,
|
|
33
|
+
nodeColor: mergedOptions.nodeColor,
|
|
34
|
+
nodeStrokeColor: mergedOptions.nodeStrokeColor,
|
|
35
|
+
nodeStrokeWidth: mergedOptions.nodeStrokeWidth,
|
|
36
|
+
maskColor: mergedOptions.maskColor,
|
|
37
|
+
maskStrokeColor: mergedOptions.maskStrokeColor,
|
|
38
|
+
maskStrokeWidth: mergedOptions.maskStrokeWidth,
|
|
39
|
+
pannable: mergedOptions.pannable,
|
|
40
|
+
zoomable: mergedOptions.zoomable,
|
|
41
|
+
interactive: mergedOptions.interactive,
|
|
42
|
+
showLayoutControls: mergedOptions.showLayoutControls,
|
|
43
|
+
layoutType: mergedOptions.layoutType,
|
|
44
|
+
layoutDirection: mergedOptions.layoutDirection
|
|
45
|
+
},
|
|
46
|
+
install(flow) {
|
|
47
|
+
console.log("[Minimap Plugin] Installed", flow);
|
|
48
|
+
}
|
|
49
|
+
};
|
|
50
|
+
}
|