Files
FiDA_Front/src/components/Canvas/FlowCanvas/flow-canvas.vue

157 lines
4.0 KiB
Vue
Raw Normal View History

2026-02-26 11:45:32 +08:00
<template>
<div class="flow-canvas">
<VueFlow
ref="vueFlow"
:nodes="nodes"
:edges="edges"
:min-zoom="0.1"
:max-zoom="10"
:nodes-draggable="true"
@nodes-initialized="layoutGraph('LR')"
@node-drag-stop="(e) => eventManager.handleNodeDragStop(e)"
@viewport-change="(e) => eventManager.handleViewportChange(e)"
>
2026-02-27 14:58:36 +08:00
<template v-for="v in nodeTypes" :key="v" #[`node-${v}`]="nodeProps">
<node :type="v" :stateManager="stateManager" :node="nodeProps">
2026-02-27 11:43:27 +08:00
<component
2026-03-02 13:51:14 +08:00
:is="components[nodeProps.data.component]"
2026-02-27 11:43:27 +08:00
:node="nodeProps"
2026-02-27 14:58:36 +08:00
:data="nodeProps.data.data"
2026-02-27 11:43:27 +08:00
v-bind="nodeProps.data"
/>
2026-02-26 11:45:32 +08:00
</node>
</template>
</VueFlow>
</div>
2026-03-02 13:51:14 +08:00
<header-tools @export="exportFlow" @import="importFlow" />
2026-02-26 11:45:32 +08:00
<zoom
:zoom="stateManager.zoom.value"
:step="0.1"
@add="(e) => flowManager.setZoom(e)"
@sub="(e) => flowManager.setZoom(e)"
/>
</template>
<script setup lang="ts">
import { VueFlow, useVueFlow } from '@vue-flow/core'
2026-03-02 13:51:14 +08:00
import { computed, ref, markRaw, onMounted, reactive, nextTick, provide } from 'vue'
2026-02-26 11:45:32 +08:00
import { useLayout } from '@/utils/treeDiagram'
2026-03-02 13:51:14 +08:00
import { NODE_TYPE, NODE_COMPONENT } from './tools/index.d'
2026-02-26 11:45:32 +08:00
// 组件
import headerTools from './components/header-tools.vue'
import zoom from '../components/zoom.vue'
2026-03-02 13:51:14 +08:00
// 节点
2026-02-26 11:45:32 +08:00
import node from './components/node.vue'
2026-03-02 13:51:14 +08:00
import resultImage from './components/result/result-image.vue'
import card from './components/cards/index.vue'
const components = {
[NODE_COMPONENT.RESULT_IMAGE]: resultImage,
[NODE_COMPONENT.CARD]: card
}
2026-02-26 11:45:32 +08:00
// 管理器
import { StateManager } from './manager/StateManager'
import { EventManager } from './manager/EventManager'
import { FlowManager } from './manager/FlowManager'
2026-02-27 09:56:49 +08:00
import { NodeManager } from './manager/NodeManager'
2026-02-26 11:45:32 +08:00
2026-02-27 16:47:02 +08:00
const props = defineProps({
config: {
type: Object,
default: () => ({})
}
})
2026-02-26 11:45:32 +08:00
const vueFlow = ref<any>()
2026-02-27 14:58:36 +08:00
const nodeTypes = ref([NODE_TYPE.INPUT, NODE_TYPE.SECONDARY, NODE_TYPE.OUTPUT])
2026-02-26 11:45:32 +08:00
2026-02-27 09:56:49 +08:00
// 状态管理器
2026-02-26 11:45:32 +08:00
const stateManager = new StateManager({ vueFlow })
2026-02-27 11:43:27 +08:00
const nodes = computed(() => stateManager.nodes_.value)
2026-02-26 11:45:32 +08:00
const edges = computed(() => stateManager.edges.value)
2026-02-27 09:56:49 +08:00
// 事件管理器
const eventManager = new EventManager({ stateManager, vueFlow })
// 流程管理器
const flowManager = new FlowManager({ stateManager, vueFlow })
// 节点管理器
const nodeManager = new NodeManager({ stateManager, vueFlow })
stateManager.setManager({
eventManager,
flowManager,
nodeManager
2026-02-26 11:45:32 +08:00
})
2026-02-27 09:56:49 +08:00
2026-02-27 11:43:27 +08:00
provide('stateManager', stateManager)
provide('eventManager', eventManager)
provide('flowManager', flowManager)
provide('nodeManager', nodeManager)
provide('nodeManager', nodeManager)
2026-02-26 11:45:32 +08:00
const { fitView } = useVueFlow()
const { layout } = useLayout()
const index = ref(0)
async function layoutGraph(direction) {
2026-03-02 13:51:14 +08:00
// if (index.value > 0) return
2026-02-26 11:45:32 +08:00
index.value++
setTimeout(() => {
stateManager.nodes.value = layout(
stateManager.nodes.value,
stateManager.edges.value,
direction
)
nextTick(() => {
2026-02-27 11:43:27 +08:00
fitView({ maxZoom: 1 })
2026-02-26 11:45:32 +08:00
})
}, 0)
}
2026-02-27 16:04:55 +08:00
const exportFlow = () => {
// flowManager.exportFlow()
2026-03-02 13:51:14 +08:00
const str = JSON.stringify(stateManager.nodes.value)
const json = JSON.parse(str)
localStorage.setItem('flow_json', str)
}
const importFlow = async () => {
try {
stateManager.nodes.value = []
await nextTick()
const json = JSON.parse(localStorage.getItem('flow_json') || '[]')
stateManager.nodes.value = json
setTimeout(() => {
nextTick(() => {
fitView({ maxZoom: 1 })
})
}, 0)
} catch (error) {
console.log(error)
}
2026-02-27 16:04:55 +08:00
}
2026-02-26 11:45:32 +08:00
onMounted(() => {
2026-02-27 14:58:36 +08:00
// window['vueFlow'] = vueFlow
// window['nodes'] = nodes
2026-02-27 16:47:02 +08:00
console.log(props.config)
2026-02-27 14:58:36 +08:00
nodeManager.createResultNode({
data: {
isHeader: false,
data: {
2026-02-27 16:47:02 +08:00
url: props.config.url
2026-02-27 14:58:36 +08:00
}
}
})
2026-02-26 11:45:32 +08:00
})
</script>
<style lang="less">
@import '@vue-flow/core/dist/style.css';
@import '@vue-flow/core/dist/theme-default.css';
</style>
<style lang="less" scoped>
.flow-canvas {
width: 100%;
height: 100%;
> .vue-flow {
width: 100%;
height: 100%;
}
}
</style>