| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   memo, | 
					
						
							|  |  |  |   useCallback, | 
					
						
							|  |  |  |   useState, | 
					
						
							|  |  |  | } from 'react' | 
					
						
							|  |  |  | import { useContext } from 'use-context-selector' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   useStoreApi, | 
					
						
							|  |  |  | } from 'reactflow' | 
					
						
							|  |  |  | import { RiBookOpenLine, RiCloseLine } from '@remixicon/react' | 
					
						
							|  |  |  | import { useTranslation } from 'react-i18next' | 
					
						
							|  |  |  | import { useStore } from '@/app/components/workflow/store' | 
					
						
							|  |  |  | import ActionButton, { ActionButtonState } from '@/app/components/base/action-button' | 
					
						
							|  |  |  | import { BubbleX, LongArrowLeft, LongArrowRight } from '@/app/components/base/icons/src/vender/line/others' | 
					
						
							|  |  |  | import BlockIcon from '@/app/components/workflow/block-icon' | 
					
						
							|  |  |  | import VariableModalTrigger from '@/app/components/workflow/panel/chat-variable-panel/components/variable-modal-trigger' | 
					
						
							|  |  |  | import VariableItem from '@/app/components/workflow/panel/chat-variable-panel/components/variable-item' | 
					
						
							|  |  |  | import RemoveEffectVarConfirm from '@/app/components/workflow/nodes/_base/components/remove-effect-var-confirm' | 
					
						
							|  |  |  | import type { | 
					
						
							|  |  |  |   ConversationVariable, | 
					
						
							|  |  |  | } from '@/app/components/workflow/types' | 
					
						
							|  |  |  | import { findUsedVarNodes, updateNodeVars } from '@/app/components/workflow/nodes/_base/components/variable/utils' | 
					
						
							|  |  |  | import { useNodesSyncDraft } from '@/app/components/workflow/hooks/use-nodes-sync-draft' | 
					
						
							|  |  |  | import { BlockEnum } from '@/app/components/workflow/types' | 
					
						
							|  |  |  | import I18n from '@/context/i18n' | 
					
						
							|  |  |  | import { LanguagesSupported } from '@/i18n/language' | 
					
						
							|  |  |  | import cn from '@/utils/classnames' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ChatVariablePanel = () => { | 
					
						
							|  |  |  |   const { t } = useTranslation() | 
					
						
							|  |  |  |   const { locale } = useContext(I18n) | 
					
						
							|  |  |  |   const store = useStoreApi() | 
					
						
							|  |  |  |   const setShowChatVariablePanel = useStore(s => s.setShowChatVariablePanel) | 
					
						
							|  |  |  |   const varList = useStore(s => s.conversationVariables) as ConversationVariable[] | 
					
						
							|  |  |  |   const updateChatVarList = useStore(s => s.setConversationVariables) | 
					
						
							|  |  |  |   const { doSyncWorkflowDraft } = useNodesSyncDraft() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const [showTip, setShowTip] = useState(true) | 
					
						
							|  |  |  |   const [showVariableModal, setShowVariableModal] = useState(false) | 
					
						
							|  |  |  |   const [currentVar, setCurrentVar] = useState<ConversationVariable>() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const [showRemoveVarConfirm, setShowRemoveConfirm] = useState(false) | 
					
						
							|  |  |  |   const [cacheForDelete, setCacheForDelete] = useState<ConversationVariable>() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const getEffectedNodes = useCallback((chatVar: ConversationVariable) => { | 
					
						
							|  |  |  |     const { getNodes } = store.getState() | 
					
						
							|  |  |  |     const allNodes = getNodes() | 
					
						
							|  |  |  |     return findUsedVarNodes( | 
					
						
							|  |  |  |       ['conversation', chatVar.name], | 
					
						
							|  |  |  |       allNodes, | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |   }, [store]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const removeUsedVarInNodes = useCallback((chatVar: ConversationVariable) => { | 
					
						
							|  |  |  |     const { getNodes, setNodes } = store.getState() | 
					
						
							|  |  |  |     const effectedNodes = getEffectedNodes(chatVar) | 
					
						
							|  |  |  |     const newNodes = getNodes().map((node) => { | 
					
						
							|  |  |  |       if (effectedNodes.find(n => n.id === node.id)) | 
					
						
							|  |  |  |         return updateNodeVars(node, ['conversation', chatVar.name], []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return node | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     setNodes(newNodes) | 
					
						
							|  |  |  |   }, [getEffectedNodes, store]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleEdit = (chatVar: ConversationVariable) => { | 
					
						
							|  |  |  |     setCurrentVar(chatVar) | 
					
						
							|  |  |  |     setShowVariableModal(true) | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleDelete = useCallback((chatVar: ConversationVariable) => { | 
					
						
							|  |  |  |     removeUsedVarInNodes(chatVar) | 
					
						
							|  |  |  |     updateChatVarList(varList.filter(v => v.id !== chatVar.id)) | 
					
						
							|  |  |  |     setCacheForDelete(undefined) | 
					
						
							|  |  |  |     setShowRemoveConfirm(false) | 
					
						
							|  |  |  |     doSyncWorkflowDraft() | 
					
						
							|  |  |  |   }, [doSyncWorkflowDraft, removeUsedVarInNodes, updateChatVarList, varList]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const deleteCheck = useCallback((chatVar: ConversationVariable) => { | 
					
						
							|  |  |  |     const effectedNodes = getEffectedNodes(chatVar) | 
					
						
							|  |  |  |     if (effectedNodes.length > 0) { | 
					
						
							|  |  |  |       setCacheForDelete(chatVar) | 
					
						
							|  |  |  |       setShowRemoveConfirm(true) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       handleDelete(chatVar) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [getEffectedNodes, handleDelete]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleSave = useCallback(async (chatVar: ConversationVariable) => { | 
					
						
							|  |  |  |     // add chatVar
 | 
					
						
							|  |  |  |     if (!currentVar) { | 
					
						
							|  |  |  |       const newList = [chatVar, ...varList] | 
					
						
							|  |  |  |       updateChatVarList(newList) | 
					
						
							|  |  |  |       doSyncWorkflowDraft() | 
					
						
							|  |  |  |       return | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // edit chatVar
 | 
					
						
							|  |  |  |     const newList = varList.map(v => v.id === currentVar.id ? chatVar : v) | 
					
						
							|  |  |  |     updateChatVarList(newList) | 
					
						
							|  |  |  |     // side effects of rename env
 | 
					
						
							|  |  |  |     if (currentVar.name !== chatVar.name) { | 
					
						
							|  |  |  |       const { getNodes, setNodes } = store.getState() | 
					
						
							|  |  |  |       const effectedNodes = getEffectedNodes(currentVar) | 
					
						
							|  |  |  |       const newNodes = getNodes().map((node) => { | 
					
						
							|  |  |  |         if (effectedNodes.find(n => n.id === node.id)) | 
					
						
							|  |  |  |           return updateNodeVars(node, ['conversation', currentVar.name], ['conversation', chatVar.name]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return node | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       setNodes(newNodes) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     doSyncWorkflowDraft() | 
					
						
							|  |  |  |   }, [currentVar, doSyncWorkflowDraft, getEffectedNodes, store, updateChatVarList, varList]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <div | 
					
						
							|  |  |  |       className={cn( | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |         'relative flex h-full w-[420px] flex-col rounded-l-2xl border border-components-panel-border bg-components-panel-bg-alt', | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |       )} | 
					
						
							|  |  |  |     > | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |       <div className='system-xl-semibold flex shrink-0 items-center justify-between p-4 pb-0 text-text-primary'> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |         {t('workflow.chatVariable.panelTitle')} | 
					
						
							|  |  |  |         <div className='flex items-center gap-1'> | 
					
						
							|  |  |  |           <ActionButton state={showTip ? ActionButtonState.Active : undefined} onClick={() => setShowTip(!showTip)}> | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |             <RiBookOpenLine className='h-4 w-4' /> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |           </ActionButton> | 
					
						
							|  |  |  |           <div | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |             className='flex h-6 w-6 cursor-pointer items-center justify-center' | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |             onClick={() => setShowChatVariablePanel(false)} | 
					
						
							|  |  |  |           > | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |             <RiCloseLine className='h-4 w-4 text-text-tertiary' /> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |           </div> | 
					
						
							|  |  |  |         </div> | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |       {showTip && ( | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |         <div className='shrink-0 px-3 pb-2 pt-2.5'> | 
					
						
							|  |  |  |           <div className='radius-2xl relative bg-background-section-burn p-3'> | 
					
						
							|  |  |  |             <div className='system-2xs-medium-uppercase inline-block rounded-[5px] border border-divider-deep px-[5px] py-[3px] text-text-tertiary'>TIPS</div> | 
					
						
							|  |  |  |             <div className='system-sm-regular mb-4 mt-1 text-text-secondary'> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |               {t('workflow.chatVariable.panelDescription')} | 
					
						
							| 
									
										
										
										
											2024-10-23 15:47:11 +08:00
										 |  |  |               <a target='_blank' rel='noopener noreferrer' className='text-text-accent' href={locale !== LanguagesSupported[1] ? 'https://docs.dify.ai/guides/workflow/variables#conversation-variables' : `https://docs.dify.ai/${locale.toLowerCase()}/guides/workflow/variables#hui-hua-bian-liang`}>{t('workflow.chatVariable.docLink')}</a> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |             </div> | 
					
						
							|  |  |  |             <div className='flex items-center gap-2'> | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |               <div className='radius-lg flex flex-col border border-workflow-block-border bg-workflow-block-bg p-3 pb-4 shadow-md'> | 
					
						
							|  |  |  |                 <BubbleX className='mb-1 h-4 w-4 shrink-0 text-util-colors-teal-teal-700' /> | 
					
						
							|  |  |  |                 <div className='system-xs-semibold text-text-secondary'>conversation_var</div> | 
					
						
							|  |  |  |                 <div className='system-2xs-regular text-text-tertiary'>String</div> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |               </div> | 
					
						
							|  |  |  |               <div className='grow'> | 
					
						
							|  |  |  |                 <div className='mb-2 flex items-center gap-2 py-1'> | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |                   <div className='flex h-3 w-16 shrink-0 items-center gap-1 px-1'> | 
					
						
							|  |  |  |                     <LongArrowLeft className='h-2 grow text-text-quaternary' /> | 
					
						
							|  |  |  |                     <div className='system-2xs-medium shrink-0 text-text-tertiary'>WRITE</div> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |                   </div> | 
					
						
							|  |  |  |                   <BlockIcon className='shrink-0' type={BlockEnum.Assigner} /> | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |                   <div className='system-xs-semibold grow truncate text-text-secondary'>{t('workflow.blocks.assigner')}</div> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |                 </div> | 
					
						
							|  |  |  |                 <div className='flex items-center gap-2 py-1'> | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |                   <div className='flex h-3 w-16 shrink-0 items-center gap-1 px-1'> | 
					
						
							|  |  |  |                     <div className='system-2xs-medium shrink-0 text-text-tertiary'>READ</div> | 
					
						
							|  |  |  |                     <LongArrowRight className='h-2 grow text-text-quaternary' /> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |                   </div> | 
					
						
							|  |  |  |                   <BlockIcon className='shrink-0' type={BlockEnum.LLM} /> | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |                   <div className='system-xs-semibold grow truncate text-text-secondary'>{t('workflow.blocks.llm')}</div> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |                 </div> | 
					
						
							|  |  |  |               </div> | 
					
						
							|  |  |  |             </div> | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |             <div className='absolute right-[38px] top-[-4px] z-10 h-3 w-3 rotate-45 bg-background-section-burn'/> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |           </div> | 
					
						
							|  |  |  |         </div> | 
					
						
							|  |  |  |       )} | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |       <div className='shrink-0 px-4 pb-3 pt-2'> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |         <VariableModalTrigger | 
					
						
							|  |  |  |           open={showVariableModal} | 
					
						
							|  |  |  |           setOpen={setShowVariableModal} | 
					
						
							|  |  |  |           showTip={showTip} | 
					
						
							|  |  |  |           chatVar={currentVar} | 
					
						
							|  |  |  |           onSave={handleSave} | 
					
						
							|  |  |  |           onClose={() => setCurrentVar(undefined)} | 
					
						
							|  |  |  |         /> | 
					
						
							|  |  |  |       </div> | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |       <div className='grow overflow-y-auto rounded-b-2xl px-4'> | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |         {varList.map(chatVar => ( | 
					
						
							|  |  |  |           <VariableItem | 
					
						
							|  |  |  |             key={chatVar.id} | 
					
						
							|  |  |  |             item={chatVar} | 
					
						
							|  |  |  |             onEdit={handleEdit} | 
					
						
							|  |  |  |             onDelete={deleteCheck} | 
					
						
							|  |  |  |           /> | 
					
						
							|  |  |  |         ))} | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |       <RemoveEffectVarConfirm | 
					
						
							|  |  |  |         isShow={showRemoveVarConfirm} | 
					
						
							|  |  |  |         onCancel={() => setShowRemoveConfirm(false)} | 
					
						
							|  |  |  |         onConfirm={() => cacheForDelete && handleDelete(cacheForDelete)} | 
					
						
							|  |  |  |       /> | 
					
						
							|  |  |  |     </div> | 
					
						
							|  |  |  |   ) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export default memo(ChatVariablePanel) |