mirror of https://github.com/langgenius/dify.git
223 lines
6.7 KiB
TypeScript
223 lines
6.7 KiB
TypeScript
import type {
|
|
EdgeMouseHandler,
|
|
OnEdgesChange,
|
|
} from 'reactflow'
|
|
import { produce } from 'immer'
|
|
import { useCallback } from 'react'
|
|
import {
|
|
useStoreApi,
|
|
} from 'reactflow'
|
|
import { useWorkflowStore } from '../store'
|
|
import {
|
|
applyConnectedHandleNodeData,
|
|
buildContextMenuEdges,
|
|
clearEdgeMenuIfNeeded,
|
|
clearNodeSelectionState,
|
|
updateEdgeHoverState,
|
|
updateEdgeSelectionState,
|
|
} from './use-edges-interactions.helpers'
|
|
import { useNodesSyncDraft } from './use-nodes-sync-draft'
|
|
import { useNodesReadOnly } from './use-workflow'
|
|
import { useWorkflowHistory, WorkflowHistoryEvent } from './use-workflow-history'
|
|
|
|
export const useEdgesInteractions = () => {
|
|
const store = useStoreApi()
|
|
const workflowStore = useWorkflowStore()
|
|
const { handleSyncWorkflowDraft } = useNodesSyncDraft()
|
|
const { getNodesReadOnly } = useNodesReadOnly()
|
|
const { saveStateToHistory } = useWorkflowHistory()
|
|
|
|
const deleteEdgeById = useCallback((edgeId: string) => {
|
|
const {
|
|
getNodes,
|
|
setNodes,
|
|
edges,
|
|
setEdges,
|
|
} = store.getState()
|
|
const currentEdgeIndex = edges.findIndex(edge => edge.id === edgeId)
|
|
|
|
if (currentEdgeIndex < 0)
|
|
return
|
|
const currentEdge = edges[currentEdgeIndex]
|
|
const nodes = getNodes()
|
|
const newNodes = applyConnectedHandleNodeData(nodes, [{ type: 'remove', edge: currentEdge }])
|
|
setNodes(newNodes)
|
|
const newEdges = produce(edges, (draft) => {
|
|
draft.splice(currentEdgeIndex, 1)
|
|
})
|
|
setEdges(newEdges)
|
|
if (clearEdgeMenuIfNeeded({ edgeMenu: workflowStore.getState().edgeMenu, edgeIds: [currentEdge.id] }))
|
|
workflowStore.setState({ edgeMenu: undefined })
|
|
handleSyncWorkflowDraft()
|
|
saveStateToHistory(WorkflowHistoryEvent.EdgeDelete)
|
|
}, [store, workflowStore, handleSyncWorkflowDraft, saveStateToHistory])
|
|
|
|
const handleEdgeEnter = useCallback<EdgeMouseHandler>((_, edge) => {
|
|
if (getNodesReadOnly())
|
|
return
|
|
|
|
const {
|
|
edges,
|
|
setEdges,
|
|
} = store.getState()
|
|
setEdges(updateEdgeHoverState(edges, edge.id, true))
|
|
}, [store, getNodesReadOnly])
|
|
|
|
const handleEdgeLeave = useCallback<EdgeMouseHandler>((_, edge) => {
|
|
if (getNodesReadOnly())
|
|
return
|
|
|
|
const {
|
|
edges,
|
|
setEdges,
|
|
} = store.getState()
|
|
setEdges(updateEdgeHoverState(edges, edge.id, false))
|
|
}, [store, getNodesReadOnly])
|
|
|
|
const handleEdgeDeleteByDeleteBranch = useCallback((nodeId: string, branchId: string) => {
|
|
if (getNodesReadOnly())
|
|
return
|
|
|
|
const {
|
|
getNodes,
|
|
setNodes,
|
|
edges,
|
|
setEdges,
|
|
} = store.getState()
|
|
const edgeWillBeDeleted = edges.filter(edge => edge.source === nodeId && edge.sourceHandle === branchId)
|
|
|
|
if (!edgeWillBeDeleted.length)
|
|
return
|
|
|
|
const nodes = getNodes()
|
|
const newNodes = applyConnectedHandleNodeData(
|
|
nodes,
|
|
edgeWillBeDeleted.map(edge => ({ type: 'remove' as const, edge })),
|
|
)
|
|
setNodes(newNodes)
|
|
const newEdges = produce(edges, (draft) => {
|
|
return draft.filter(edge => !edgeWillBeDeleted.find(e => e.id === edge.id))
|
|
})
|
|
setEdges(newEdges)
|
|
if (clearEdgeMenuIfNeeded({
|
|
edgeMenu: workflowStore.getState().edgeMenu,
|
|
edgeIds: edgeWillBeDeleted.map(edge => edge.id),
|
|
})) {
|
|
workflowStore.setState({ edgeMenu: undefined })
|
|
}
|
|
handleSyncWorkflowDraft()
|
|
saveStateToHistory(WorkflowHistoryEvent.EdgeDeleteByDeleteBranch)
|
|
}, [getNodesReadOnly, store, workflowStore, handleSyncWorkflowDraft, saveStateToHistory])
|
|
|
|
const handleEdgeDelete = useCallback(() => {
|
|
if (getNodesReadOnly())
|
|
return
|
|
const { edges } = store.getState()
|
|
const currentEdge = edges.find(edge => edge.selected)
|
|
|
|
if (!currentEdge)
|
|
return
|
|
|
|
deleteEdgeById(currentEdge.id)
|
|
}, [deleteEdgeById, getNodesReadOnly, store])
|
|
|
|
const handleEdgeDeleteById = useCallback((edgeId: string) => {
|
|
if (getNodesReadOnly())
|
|
return
|
|
|
|
deleteEdgeById(edgeId)
|
|
}, [deleteEdgeById, getNodesReadOnly])
|
|
|
|
const handleEdgesChange = useCallback<OnEdgesChange>((changes) => {
|
|
if (getNodesReadOnly())
|
|
return
|
|
|
|
const {
|
|
edges,
|
|
setEdges,
|
|
} = store.getState()
|
|
setEdges(updateEdgeSelectionState(edges, changes))
|
|
}, [store, getNodesReadOnly])
|
|
|
|
const handleEdgeSourceHandleChange = useCallback((nodeId: string, oldHandleId: string, newHandleId: string) => {
|
|
if (getNodesReadOnly())
|
|
return
|
|
|
|
const { getNodes, setNodes, edges, setEdges } = store.getState()
|
|
const nodes = getNodes()
|
|
|
|
// Find edges connected to the old handle
|
|
const affectedEdges = edges.filter(
|
|
edge => edge.source === nodeId && edge.sourceHandle === oldHandleId,
|
|
)
|
|
|
|
if (affectedEdges.length === 0)
|
|
return
|
|
|
|
// Update node metadata: remove old handle, add new handle
|
|
const newNodes = applyConnectedHandleNodeData(nodes, [
|
|
...affectedEdges.map(edge => ({ type: 'remove' as const, edge })),
|
|
...affectedEdges.map(edge => ({
|
|
type: 'add' as const,
|
|
edge: { ...edge, sourceHandle: newHandleId },
|
|
})),
|
|
])
|
|
setNodes(newNodes)
|
|
|
|
// Update edges to use new sourceHandle and regenerate edge IDs
|
|
const newEdges = produce(edges, (draft) => {
|
|
draft.forEach((edge) => {
|
|
if (edge.source === nodeId && edge.sourceHandle === oldHandleId) {
|
|
edge.sourceHandle = newHandleId
|
|
edge.id = `${edge.source}-${newHandleId}-${edge.target}-${edge.targetHandle}`
|
|
}
|
|
})
|
|
})
|
|
setEdges(newEdges)
|
|
if (clearEdgeMenuIfNeeded({
|
|
edgeMenu: workflowStore.getState().edgeMenu,
|
|
edgeIds: affectedEdges.map(edge => edge.id),
|
|
})) {
|
|
workflowStore.setState({ edgeMenu: undefined })
|
|
}
|
|
handleSyncWorkflowDraft()
|
|
saveStateToHistory(WorkflowHistoryEvent.EdgeSourceHandleChange)
|
|
}, [getNodesReadOnly, store, workflowStore, handleSyncWorkflowDraft, saveStateToHistory])
|
|
|
|
const handleEdgeContextMenu = useCallback<EdgeMouseHandler>((e, edge) => {
|
|
if (getNodesReadOnly())
|
|
return
|
|
|
|
e.preventDefault()
|
|
|
|
const { getNodes, setNodes, edges, setEdges } = store.getState()
|
|
setEdges(buildContextMenuEdges(edges, edge.id))
|
|
const nodes = getNodes()
|
|
if (nodes.some(node => node.data.selected || node.selected || node.data._isBundled)) {
|
|
setNodes(clearNodeSelectionState(nodes))
|
|
}
|
|
|
|
workflowStore.setState({
|
|
nodeMenu: undefined,
|
|
panelMenu: undefined,
|
|
selectionMenu: undefined,
|
|
edgeMenu: {
|
|
clientX: e.clientX,
|
|
clientY: e.clientY,
|
|
edgeId: edge.id,
|
|
},
|
|
})
|
|
}, [store, workflowStore, getNodesReadOnly])
|
|
|
|
return {
|
|
handleEdgeEnter,
|
|
handleEdgeLeave,
|
|
handleEdgeDeleteByDeleteBranch,
|
|
handleEdgeDelete,
|
|
handleEdgeDeleteById,
|
|
handleEdgesChange,
|
|
handleEdgeSourceHandleChange,
|
|
handleEdgeContextMenu,
|
|
}
|
|
}
|