mirror of
				https://github.com/langgenius/dify.git
				synced 2025-10-25 16:08:45 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			333 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			333 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import type {
 | |
|   FC,
 | |
|   ReactElement,
 | |
| } from 'react'
 | |
| import {
 | |
|   cloneElement,
 | |
|   memo,
 | |
|   useEffect,
 | |
|   useMemo,
 | |
|   useRef,
 | |
| } from 'react'
 | |
| import {
 | |
|   RiAlertFill,
 | |
|   RiCheckboxCircleFill,
 | |
|   RiErrorWarningFill,
 | |
|   RiLoader2Line,
 | |
| } from '@remixicon/react'
 | |
| import { useTranslation } from 'react-i18next'
 | |
| import type { NodeProps } from '../../types'
 | |
| import {
 | |
|   BlockEnum,
 | |
|   NodeRunningStatus,
 | |
| } from '../../types'
 | |
| import {
 | |
|   useNodesReadOnly,
 | |
|   useToolIcon,
 | |
| } from '../../hooks'
 | |
| import {
 | |
|   hasErrorHandleNode,
 | |
|   hasRetryNode,
 | |
| } from '../../utils'
 | |
| import { useNodeIterationInteractions } from '../iteration/use-interactions'
 | |
| import { useNodeLoopInteractions } from '../loop/use-interactions'
 | |
| import type { IterationNodeType } from '../iteration/types'
 | |
| import {
 | |
|   NodeSourceHandle,
 | |
|   NodeTargetHandle,
 | |
| } from './components/node-handle'
 | |
| import NodeResizer from './components/node-resizer'
 | |
| import NodeControl from './components/node-control'
 | |
| import ErrorHandleOnNode from './components/error-handle/error-handle-on-node'
 | |
| import RetryOnNode from './components/retry/retry-on-node'
 | |
| import AddVariablePopupWithPosition from './components/add-variable-popup-with-position'
 | |
| import cn from '@/utils/classnames'
 | |
| import BlockIcon from '@/app/components/workflow/block-icon'
 | |
| import Tooltip from '@/app/components/base/tooltip'
 | |
| 
 | |
| type BaseNodeProps = {
 | |
|   children: ReactElement
 | |
| } & NodeProps
 | |
| 
 | |
| const BaseNode: FC<BaseNodeProps> = ({
 | |
|   id,
 | |
|   data,
 | |
|   children,
 | |
| }) => {
 | |
|   const { t } = useTranslation()
 | |
|   const nodeRef = useRef<HTMLDivElement>(null)
 | |
|   const { nodesReadOnly } = useNodesReadOnly()
 | |
|   const { handleNodeIterationChildSizeChange } = useNodeIterationInteractions()
 | |
|   const { handleNodeLoopChildSizeChange } = useNodeLoopInteractions()
 | |
|   const toolIcon = useToolIcon(data)
 | |
| 
 | |
|   useEffect(() => {
 | |
|     if (nodeRef.current && data.selected && data.isInIteration) {
 | |
|       const resizeObserver = new ResizeObserver(() => {
 | |
|         handleNodeIterationChildSizeChange(id)
 | |
|       })
 | |
| 
 | |
|       resizeObserver.observe(nodeRef.current)
 | |
| 
 | |
|       return () => {
 | |
|         resizeObserver.disconnect()
 | |
|       }
 | |
|     }
 | |
|   }, [data.isInIteration, data.selected, id, handleNodeIterationChildSizeChange])
 | |
| 
 | |
|   useEffect(() => {
 | |
|     if (nodeRef.current && data.selected && data.isInLoop) {
 | |
|       const resizeObserver = new ResizeObserver(() => {
 | |
|         handleNodeLoopChildSizeChange(id)
 | |
|       })
 | |
| 
 | |
|       resizeObserver.observe(nodeRef.current)
 | |
| 
 | |
|       return () => {
 | |
|         resizeObserver.disconnect()
 | |
|       }
 | |
|     }
 | |
|   }, [data.isInLoop, data.selected, id, handleNodeLoopChildSizeChange])
 | |
| 
 | |
|   const showSelectedBorder = data.selected || data._isBundled || data._isEntering
 | |
|   const {
 | |
|     showRunningBorder,
 | |
|     showSuccessBorder,
 | |
|     showFailedBorder,
 | |
|     showExceptionBorder,
 | |
|   } = useMemo(() => {
 | |
|     return {
 | |
|       showRunningBorder: data._runningStatus === NodeRunningStatus.Running && !showSelectedBorder,
 | |
|       showSuccessBorder: data._runningStatus === NodeRunningStatus.Succeeded && !showSelectedBorder,
 | |
|       showFailedBorder: data._runningStatus === NodeRunningStatus.Failed && !showSelectedBorder,
 | |
|       showExceptionBorder: data._runningStatus === NodeRunningStatus.Exception && !showSelectedBorder,
 | |
|     }
 | |
|   }, [data._runningStatus, showSelectedBorder])
 | |
| 
 | |
|   const LoopIndex = useMemo(() => {
 | |
|     let text = ''
 | |
| 
 | |
|     if (data._runningStatus === NodeRunningStatus.Running)
 | |
|       text = t('workflow.nodes.loop.currentLoopCount', { count: data._loopIndex })
 | |
|     if (data._runningStatus === NodeRunningStatus.Succeeded || data._runningStatus === NodeRunningStatus.Failed)
 | |
|       text = t('workflow.nodes.loop.totalLoopCount', { count: data._loopIndex })
 | |
| 
 | |
|     if (text) {
 | |
|       return (
 | |
|         <div
 | |
|           className={cn(
 | |
|             'system-xs-medium mr-2 text-text-tertiary',
 | |
|             data._runningStatus === NodeRunningStatus.Running && 'text-text-accent',
 | |
|           )}
 | |
|         >
 | |
|           {text}
 | |
|         </div>
 | |
|       )
 | |
|     }
 | |
| 
 | |
|     return null
 | |
|   }, [data._loopIndex, data._runningStatus, t])
 | |
| 
 | |
|   return (
 | |
|     <div
 | |
|       className={cn(
 | |
|         'relative flex rounded-2xl border',
 | |
|         showSelectedBorder ? 'border-components-option-card-option-selected-border' : 'border-transparent',
 | |
|         !showSelectedBorder && data._inParallelHovering && 'border-workflow-block-border-highlight',
 | |
|         data._waitingRun && 'opacity-70',
 | |
|       )}
 | |
|       ref={nodeRef}
 | |
|       style={{
 | |
|         width: (data.type === BlockEnum.Iteration || data.type === BlockEnum.Loop) ? data.width : 'auto',
 | |
|         height: (data.type === BlockEnum.Iteration || data.type === BlockEnum.Loop) ? data.height : 'auto',
 | |
|       }}
 | |
|     >
 | |
|       {
 | |
|         data.type === BlockEnum.DataSource && (
 | |
|           <div className='absolute inset-[-2px] top-[-22px] z-[-1] rounded-[18px] bg-node-data-source-bg p-0.5 backdrop-blur-[6px]'>
 | |
|             <div className='system-2xs-semibold-uppercase flex h-5 items-center px-2.5 text-text-tertiary'>
 | |
|               data source
 | |
|             </div>
 | |
|           </div>
 | |
|         )
 | |
|       }
 | |
|       <div
 | |
|         className={cn(
 | |
|           'group relative pb-1 shadow-xs',
 | |
|           'rounded-[15px] border border-transparent',
 | |
|           (data.type !== BlockEnum.Iteration && data.type !== BlockEnum.Loop) && 'w-[240px] bg-workflow-block-bg',
 | |
|           (data.type === BlockEnum.Iteration || data.type === BlockEnum.Loop) && 'flex h-full w-full flex-col border-workflow-block-border bg-workflow-block-bg-transparent',
 | |
|           !data._runningStatus && 'hover:shadow-lg',
 | |
|           showRunningBorder && '!border-state-accent-solid',
 | |
|           showSuccessBorder && '!border-state-success-solid',
 | |
|           showFailedBorder && '!border-state-destructive-solid',
 | |
|           showExceptionBorder && '!border-state-warning-solid',
 | |
|           data._isBundled && '!shadow-lg',
 | |
|         )}
 | |
|       >
 | |
|         {
 | |
|           data._inParallelHovering && (
 | |
|             <div className='top system-2xs-medium-uppercase absolute -top-2.5 left-2 z-10 text-text-tertiary'>
 | |
|               {t('workflow.common.parallelRun')}
 | |
|             </div>
 | |
|           )
 | |
|         }
 | |
|         {
 | |
|           data._showAddVariablePopup && (
 | |
|             <AddVariablePopupWithPosition
 | |
|               nodeId={id}
 | |
|               nodeData={data}
 | |
|             />
 | |
|           )
 | |
|         }
 | |
|         {
 | |
|           data.type === BlockEnum.Iteration && (
 | |
|             <NodeResizer
 | |
|               nodeId={id}
 | |
|               nodeData={data}
 | |
|             />
 | |
|           )
 | |
|         }
 | |
|         {
 | |
|           data.type === BlockEnum.Loop && (
 | |
|             <NodeResizer
 | |
|               nodeId={id}
 | |
|               nodeData={data}
 | |
|             />
 | |
|           )
 | |
|         }
 | |
|         {
 | |
|           !data._isCandidate && (
 | |
|             <NodeTargetHandle
 | |
|               id={id}
 | |
|               data={data}
 | |
|               handleClassName='!top-4 !-left-[9px] !translate-y-0'
 | |
|               handleId='target'
 | |
|             />
 | |
|           )
 | |
|         }
 | |
|         {
 | |
|           data.type !== BlockEnum.IfElse && data.type !== BlockEnum.QuestionClassifier && !data._isCandidate && (
 | |
|             <NodeSourceHandle
 | |
|               id={id}
 | |
|               data={data}
 | |
|               handleClassName='!top-4 !-right-[9px] !translate-y-0'
 | |
|               handleId='source'
 | |
|             />
 | |
|           )
 | |
|         }
 | |
|         {
 | |
|           !data._runningStatus && !nodesReadOnly && !data._isCandidate && (
 | |
|             <NodeControl
 | |
|               id={id}
 | |
|               data={data}
 | |
|             />
 | |
|           )
 | |
|         }
 | |
|         <div className={cn(
 | |
|           'flex items-center rounded-t-2xl px-3 pb-2 pt-3',
 | |
|           (data.type === BlockEnum.Iteration || data.type === BlockEnum.Loop) && 'bg-transparent',
 | |
|         )}>
 | |
|           <BlockIcon
 | |
|             className='mr-2 shrink-0'
 | |
|             type={data.type}
 | |
|             size='md'
 | |
|             toolIcon={toolIcon}
 | |
|           />
 | |
|           <div
 | |
|             title={data.title}
 | |
|             className='system-sm-semibold-uppercase mr-1 flex grow items-center truncate text-text-primary'
 | |
|           >
 | |
|             <div>
 | |
|               {data.title}
 | |
|             </div>
 | |
|             {
 | |
|               data.type === BlockEnum.Iteration && (data as IterationNodeType).is_parallel && (
 | |
|                 <Tooltip popupContent={
 | |
|                   <div className='w-[180px]'>
 | |
|                     <div className='font-extrabold'>
 | |
|                       {t('workflow.nodes.iteration.parallelModeEnableTitle')}
 | |
|                     </div>
 | |
|                     {t('workflow.nodes.iteration.parallelModeEnableDesc')}
 | |
|                   </div>}
 | |
|                 >
 | |
|                   <div className='system-2xs-medium-uppercase ml-1 flex items-center justify-center rounded-[5px] border-[1px] border-text-warning px-[5px] py-[3px] text-text-warning '>
 | |
|                     {t('workflow.nodes.iteration.parallelModeUpper')}
 | |
|                   </div>
 | |
|                 </Tooltip>
 | |
|               )
 | |
|             }
 | |
|           </div>
 | |
|           {
 | |
|             data._iterationLength && data._iterationIndex && data._runningStatus === NodeRunningStatus.Running && (
 | |
|               <div className='mr-1.5 text-xs font-medium text-text-accent'>
 | |
|                 {data._iterationIndex > data._iterationLength ? data._iterationLength : data._iterationIndex}/{data._iterationLength}
 | |
|               </div>
 | |
|             )
 | |
|           }
 | |
|           {
 | |
|             data.type === BlockEnum.Loop && data._loopIndex && LoopIndex
 | |
|           }
 | |
|           {
 | |
|             (data._runningStatus === NodeRunningStatus.Running || data._singleRunningStatus === NodeRunningStatus.Running) && (
 | |
|               <RiLoader2Line className='h-3.5 w-3.5 animate-spin text-text-accent' />
 | |
|             )
 | |
|           }
 | |
|           {
 | |
|             data._runningStatus === NodeRunningStatus.Succeeded && (
 | |
|               <RiCheckboxCircleFill className='h-3.5 w-3.5 text-text-success' />
 | |
|             )
 | |
|           }
 | |
|           {
 | |
|             data._runningStatus === NodeRunningStatus.Failed && (
 | |
|               <RiErrorWarningFill className='h-3.5 w-3.5 text-text-destructive' />
 | |
|             )
 | |
|           }
 | |
|           {
 | |
|             data._runningStatus === NodeRunningStatus.Exception && (
 | |
|               <RiAlertFill className='h-3.5 w-3.5 text-text-warning-secondary' />
 | |
|             )
 | |
|           }
 | |
|         </div>
 | |
|         {
 | |
|           data.type !== BlockEnum.Iteration && data.type !== BlockEnum.Loop && (
 | |
|             cloneElement(children, { id, data })
 | |
|           )
 | |
|         }
 | |
|         {
 | |
|           (data.type === BlockEnum.Iteration || data.type === BlockEnum.Loop) && (
 | |
|             <div className='grow pb-1 pl-1 pr-1'>
 | |
|               {cloneElement(children, { id, data })}
 | |
|             </div>
 | |
|           )
 | |
|         }
 | |
|         {
 | |
|           hasRetryNode(data.type) && (
 | |
|             <RetryOnNode
 | |
|               id={id}
 | |
|               data={data}
 | |
|             />
 | |
|           )
 | |
|         }
 | |
|         {
 | |
|           hasErrorHandleNode(data.type) && (
 | |
|             <ErrorHandleOnNode
 | |
|               id={id}
 | |
|               data={data}
 | |
|             />
 | |
|           )
 | |
|         }
 | |
|         {
 | |
|           data.desc && data.type !== BlockEnum.Iteration && data.type !== BlockEnum.Loop && (
 | |
|             <div className='system-xs-regular whitespace-pre-line break-words px-3 pb-2 pt-1 text-text-tertiary'>
 | |
|               {data.desc}
 | |
|             </div>
 | |
|           )
 | |
|         }
 | |
|       </div>
 | |
|     </div>
 | |
|   )
 | |
| }
 | |
| 
 | |
| export default memo(BaseNode)
 | 
