| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  | import { useCallback } from 'react' | 
					
						
							|  |  |  | import produce from 'immer' | 
					
						
							|  |  |  | import { useStoreApi } from 'reactflow' | 
					
						
							|  |  |  | import type { | 
					
						
							|  |  |  |   BlockEnum, | 
					
						
							|  |  |  |   Node, | 
					
						
							|  |  |  | } from '../../types' | 
					
						
							| 
									
										
										
										
											2025-04-01 16:52:07 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   generateNewNode, | 
					
						
							|  |  |  |   getNodeCustomTypeByNodeDataType, | 
					
						
							|  |  |  | } from '../../utils' | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   LOOP_PADDING, | 
					
						
							|  |  |  | } from '../../constants' | 
					
						
							|  |  |  | import { CUSTOM_LOOP_START_NODE } from '../loop-start/constants' | 
					
						
							| 
									
										
										
										
											2025-04-24 16:29:58 +08:00
										 |  |  | import { useNodesMetaData } from '@/app/components/workflow/hooks' | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | export const useNodeLoopInteractions = () => { | 
					
						
							|  |  |  |   const store = useStoreApi() | 
					
						
							| 
									
										
										
										
											2025-04-24 16:29:58 +08:00
										 |  |  |   const { nodesMap: nodesMetaDataMap } = useNodesMetaData() | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const handleNodeLoopRerender = useCallback((nodeId: string) => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       getNodes, | 
					
						
							|  |  |  |       setNodes, | 
					
						
							|  |  |  |     } = store.getState() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const nodes = getNodes() | 
					
						
							|  |  |  |     const currentNode = nodes.find(n => n.id === nodeId)! | 
					
						
							|  |  |  |     const childrenNodes = nodes.filter(n => n.parentId === nodeId) | 
					
						
							|  |  |  |     let rightNode: Node | 
					
						
							|  |  |  |     let bottomNode: Node | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     childrenNodes.forEach((n) => { | 
					
						
							|  |  |  |       if (rightNode) { | 
					
						
							|  |  |  |         if (n.position.x + n.width! > rightNode.position.x + rightNode.width!) | 
					
						
							|  |  |  |           rightNode = n | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         rightNode = n | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (bottomNode) { | 
					
						
							|  |  |  |         if (n.position.y + n.height! > bottomNode.position.y + bottomNode.height!) | 
					
						
							|  |  |  |           bottomNode = n | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         bottomNode = n | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const widthShouldExtend = rightNode! && currentNode.width! < rightNode.position.x + rightNode.width! | 
					
						
							|  |  |  |     const heightShouldExtend = bottomNode! && currentNode.height! < bottomNode.position.y + bottomNode.height! | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (widthShouldExtend || heightShouldExtend) { | 
					
						
							|  |  |  |       const newNodes = produce(nodes, (draft) => { | 
					
						
							|  |  |  |         draft.forEach((n) => { | 
					
						
							|  |  |  |           if (n.id === nodeId) { | 
					
						
							|  |  |  |             if (widthShouldExtend) { | 
					
						
							|  |  |  |               n.data.width = rightNode.position.x + rightNode.width! + LOOP_PADDING.right | 
					
						
							|  |  |  |               n.width = rightNode.position.x + rightNode.width! + LOOP_PADDING.right | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (heightShouldExtend) { | 
					
						
							|  |  |  |               n.data.height = bottomNode.position.y + bottomNode.height! + LOOP_PADDING.bottom | 
					
						
							|  |  |  |               n.height = bottomNode.position.y + bottomNode.height! + LOOP_PADDING.bottom | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       setNodes(newNodes) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [store]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleNodeLoopChildDrag = useCallback((node: Node) => { | 
					
						
							|  |  |  |     const { getNodes } = store.getState() | 
					
						
							|  |  |  |     const nodes = getNodes() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const restrictPosition: { x?: number; y?: number } = { x: undefined, y: undefined } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (node.data.isInLoop) { | 
					
						
							|  |  |  |       const parentNode = nodes.find(n => n.id === node.parentId) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (parentNode) { | 
					
						
							|  |  |  |         if (node.position.y < LOOP_PADDING.top) | 
					
						
							|  |  |  |           restrictPosition.y = LOOP_PADDING.top | 
					
						
							|  |  |  |         if (node.position.x < LOOP_PADDING.left) | 
					
						
							|  |  |  |           restrictPosition.x = LOOP_PADDING.left | 
					
						
							|  |  |  |         if (node.position.x + node.width! > parentNode!.width! - LOOP_PADDING.right) | 
					
						
							|  |  |  |           restrictPosition.x = parentNode!.width! - LOOP_PADDING.right - node.width! | 
					
						
							|  |  |  |         if (node.position.y + node.height! > parentNode!.height! - LOOP_PADDING.bottom) | 
					
						
							|  |  |  |           restrictPosition.y = parentNode!.height! - LOOP_PADDING.bottom - node.height! | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |       restrictPosition, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [store]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleNodeLoopChildSizeChange = useCallback((nodeId: string) => { | 
					
						
							|  |  |  |     const { getNodes } = store.getState() | 
					
						
							|  |  |  |     const nodes = getNodes() | 
					
						
							|  |  |  |     const currentNode = nodes.find(n => n.id === nodeId)! | 
					
						
							|  |  |  |     const parentId = currentNode.parentId | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (parentId) | 
					
						
							|  |  |  |       handleNodeLoopRerender(parentId) | 
					
						
							|  |  |  |   }, [store, handleNodeLoopRerender]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleNodeLoopChildrenCopy = useCallback((nodeId: string, newNodeId: string) => { | 
					
						
							|  |  |  |     const { getNodes } = store.getState() | 
					
						
							|  |  |  |     const nodes = getNodes() | 
					
						
							|  |  |  |     const childrenNodes = nodes.filter(n => n.parentId === nodeId && n.type !== CUSTOM_LOOP_START_NODE) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return childrenNodes.map((child, index) => { | 
					
						
							|  |  |  |       const childNodeType = child.data.type as BlockEnum | 
					
						
							| 
									
										
										
										
											2025-04-24 16:29:58 +08:00
										 |  |  |       const { | 
					
						
							|  |  |  |         defaultValue, | 
					
						
							|  |  |  |         title, | 
					
						
							|  |  |  |       } = nodesMetaDataMap![childNodeType] | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  |       const nodesWithSameType = nodes.filter(node => node.data.type === childNodeType) | 
					
						
							|  |  |  |       const { newNode } = generateNewNode({ | 
					
						
							| 
									
										
										
										
											2025-04-01 16:52:07 +08:00
										 |  |  |         type: getNodeCustomTypeByNodeDataType(childNodeType), | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  |         data: { | 
					
						
							| 
									
										
										
										
											2025-04-24 16:29:58 +08:00
										 |  |  |           ...defaultValue, | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  |           ...child.data, | 
					
						
							|  |  |  |           selected: false, | 
					
						
							|  |  |  |           _isBundled: false, | 
					
						
							|  |  |  |           _connectedSourceHandleIds: [], | 
					
						
							|  |  |  |           _connectedTargetHandleIds: [], | 
					
						
							| 
									
										
										
										
											2025-04-24 16:29:58 +08:00
										 |  |  |           title: nodesWithSameType.length > 0 ? `${title} ${nodesWithSameType.length + 1}` : title, | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  |           loop_id: newNodeId, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         position: child.position, | 
					
						
							|  |  |  |         positionAbsolute: child.positionAbsolute, | 
					
						
							|  |  |  |         parentId: newNodeId, | 
					
						
							|  |  |  |         extent: child.extent, | 
					
						
							|  |  |  |         zIndex: child.zIndex, | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       newNode.id = `${newNodeId}${newNode.id + index}` | 
					
						
							|  |  |  |       return newNode | 
					
						
							|  |  |  |     }) | 
					
						
							| 
									
										
										
										
											2025-04-24 16:29:58 +08:00
										 |  |  |   }, [store, nodesMetaDataMap]) | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     handleNodeLoopRerender, | 
					
						
							|  |  |  |     handleNodeLoopChildDrag, | 
					
						
							|  |  |  |     handleNodeLoopChildSizeChange, | 
					
						
							|  |  |  |     handleNodeLoopChildrenCopy, | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |