mirror of
				https://github.com/langgenius/dify.git
				synced 2025-10-25 07:58:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			161 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			161 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import type { FC } from 'react'
 | |
| import React, { useMemo } from 'react'
 | |
| import { useTranslation } from 'react-i18next'
 | |
| import { RiAddLine } from '@remixicon/react'
 | |
| import Split from '../_base/components/split'
 | |
| import ResultPanel from '../../run/result-panel'
 | |
| import InputNumberWithSlider from '../_base/components/input-number-with-slider'
 | |
| import type { LoopNodeType } from './types'
 | |
| import useConfig from './use-config'
 | |
| import ConditionWrap from './components/condition-wrap'
 | |
| import LoopVariable from './components/loop-variables'
 | |
| import type { NodePanelProps } from '@/app/components/workflow/types'
 | |
| import Field from '@/app/components/workflow/nodes/_base/components/field'
 | |
| import BeforeRunForm from '@/app/components/workflow/nodes/_base/components/before-run-form'
 | |
| import formatTracing from '@/app/components/workflow/run/utils/format-log'
 | |
| 
 | |
| import { useLogs } from '@/app/components/workflow/run/hooks'
 | |
| import { LOOP_NODE_MAX_COUNT } from '@/config'
 | |
| 
 | |
| const i18nPrefix = 'workflow.nodes.loop'
 | |
| 
 | |
| const Panel: FC<NodePanelProps<LoopNodeType>> = ({
 | |
|   id,
 | |
|   data,
 | |
| }) => {
 | |
|   const { t } = useTranslation()
 | |
| 
 | |
|   const {
 | |
|     readOnly,
 | |
|     inputs,
 | |
|     childrenNodeVars,
 | |
|     loopChildrenNodes,
 | |
|     isShowSingleRun,
 | |
|     hideSingleRun,
 | |
|     runningStatus,
 | |
|     handleRun,
 | |
|     handleStop,
 | |
|     runResult,
 | |
|     loopRunResult,
 | |
|     handleAddCondition,
 | |
|     handleUpdateCondition,
 | |
|     handleRemoveCondition,
 | |
|     handleToggleConditionLogicalOperator,
 | |
|     handleAddSubVariableCondition,
 | |
|     handleRemoveSubVariableCondition,
 | |
|     handleUpdateSubVariableCondition,
 | |
|     handleToggleSubVariableConditionLogicalOperator,
 | |
|     handleUpdateLoopCount,
 | |
|     handleAddLoopVariable,
 | |
|     handleRemoveLoopVariable,
 | |
|     handleUpdateLoopVariable,
 | |
|   } = useConfig(id, data)
 | |
| 
 | |
|   const nodeInfo = useMemo(() => {
 | |
|     const formattedNodeInfo = formatTracing(loopRunResult, t)[0]
 | |
| 
 | |
|     if (runResult && formattedNodeInfo) {
 | |
|       return {
 | |
|         ...formattedNodeInfo,
 | |
|         execution_metadata: {
 | |
|           ...runResult.execution_metadata,
 | |
|           ...formattedNodeInfo.execution_metadata,
 | |
|         },
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return formattedNodeInfo
 | |
|   }, [runResult, loopRunResult, t])
 | |
|   const logsParams = useLogs()
 | |
| 
 | |
|   return (
 | |
|     <div className='mt-2'>
 | |
|       <div>
 | |
|         <Field
 | |
|           title={<div className='pl-3'>{t('workflow.nodes.loop.loopVariables')}</div>}
 | |
|           operations={
 | |
|             <div
 | |
|               className='mr-4 flex h-5 w-5 cursor-pointer items-center justify-center'
 | |
|               onClick={handleAddLoopVariable}
 | |
|             >
 | |
|               <RiAddLine className='h-4 w-4 text-text-tertiary' />
 | |
|             </div>
 | |
|           }
 | |
|         >
 | |
|           <div className='px-4'>
 | |
|             <LoopVariable
 | |
|               variables={inputs.loop_variables}
 | |
|               nodeId={id}
 | |
|               handleRemoveLoopVariable={handleRemoveLoopVariable}
 | |
|               handleUpdateLoopVariable={handleUpdateLoopVariable}
 | |
|             />
 | |
|           </div>
 | |
|         </Field>
 | |
|         <Split className='my-2' />
 | |
|         <Field
 | |
|           title={<div className='pl-3'>{t(`${i18nPrefix}.breakCondition`)}</div>}
 | |
|           tooltip={t(`${i18nPrefix}.breakConditionTip`)}
 | |
|         >
 | |
|           <ConditionWrap
 | |
|             nodeId={id}
 | |
|             readOnly={readOnly}
 | |
|             handleAddCondition={handleAddCondition}
 | |
|             handleRemoveCondition={handleRemoveCondition}
 | |
|             handleUpdateCondition={handleUpdateCondition}
 | |
|             handleToggleConditionLogicalOperator={handleToggleConditionLogicalOperator}
 | |
|             handleAddSubVariableCondition={handleAddSubVariableCondition}
 | |
|             handleRemoveSubVariableCondition={handleRemoveSubVariableCondition}
 | |
|             handleUpdateSubVariableCondition={handleUpdateSubVariableCondition}
 | |
|             handleToggleSubVariableConditionLogicalOperator={handleToggleSubVariableConditionLogicalOperator}
 | |
|             availableNodes={loopChildrenNodes}
 | |
|             availableVars={childrenNodeVars}
 | |
|             conditions={inputs.break_conditions || []}
 | |
|             logicalOperator={inputs.logical_operator!}
 | |
|           />
 | |
|         </Field>
 | |
|         <Split className='mt-2' />
 | |
|         <div className='mt-2'>
 | |
|           <Field
 | |
|             title={<div className='pl-3'>{t(`${i18nPrefix}.loopMaxCount`)}</div>}
 | |
|           >
 | |
|             <div className='px-3 py-2'>
 | |
|               <InputNumberWithSlider
 | |
|                 min={1}
 | |
|                 max={LOOP_NODE_MAX_COUNT}
 | |
|                 value={inputs.loop_count}
 | |
|                 onChange={(val) => {
 | |
|                   const roundedVal = Math.round(val)
 | |
|                   handleUpdateLoopCount(Number.isNaN(roundedVal) ? 1 : roundedVal)
 | |
|                 }}
 | |
|               />
 | |
|             </div>
 | |
|           </Field>
 | |
|         </div>
 | |
|       </div>
 | |
|       {/* Error handling for the Loop node is currently not considered. */}
 | |
|       {/* <div className='px-4 py-2'>
 | |
|         <Field title={t(`${i18nPrefix}.errorResponseMethod`)} >
 | |
|           <Select items={responseMethod} defaultValue={inputs.error_handle_mode} onSelect={changeErrorResponseMode} allowSearch={false}>
 | |
|           </Select>
 | |
|         </Field>
 | |
|       </div> */}
 | |
|       {isShowSingleRun && (
 | |
|         <BeforeRunForm
 | |
|           nodeName={inputs.title}
 | |
|           onHide={hideSingleRun}
 | |
|           forms={[]}
 | |
|           runningStatus={runningStatus}
 | |
|           onRun={handleRun}
 | |
|           onStop={handleStop}
 | |
|           {...logsParams}
 | |
|           result={
 | |
|             <ResultPanel {...runResult} showSteps={false} nodeInfo={nodeInfo} {...logsParams} />
 | |
|           }
 | |
|         />
 | |
|       )}
 | |
|     </div>
 | |
|   )
 | |
| }
 | |
| 
 | |
| export default React.memo(Panel)
 | 
