import type { ConversationVariable, } from '@/app/components/workflow/types' import { RiBookOpenLine, RiCloseLine } from '@remixicon/react' import { memo, useCallback, useState, } from 'react' import { useTranslation } from 'react-i18next' 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 { webSocketClient } from '@/app/components/workflow/collaboration/core/websocket-manager' import { useCollaborativeWorkflow } from '@/app/components/workflow/hooks/use-collaborative-workflow' import RemoveEffectVarConfirm from '@/app/components/workflow/nodes/_base/components/remove-effect-var-confirm' import { findUsedVarNodes, updateNodeVars } from '@/app/components/workflow/nodes/_base/components/variable/utils' import VariableItem from '@/app/components/workflow/panel/chat-variable-panel/components/variable-item' import VariableModalTrigger from '@/app/components/workflow/panel/chat-variable-panel/components/variable-modal-trigger' import { useStore } from '@/app/components/workflow/store' import { BlockEnum } from '@/app/components/workflow/types' import { updateConversationVariables } from '@/service/workflow' import { cn } from '@/utils/classnames' import useInspectVarsCrud from '../../hooks/use-inspect-vars-crud' const ChatVariablePanel = () => { const { t } = useTranslation() const setShowChatVariablePanel = useStore(s => s.setShowChatVariablePanel) const varList = useStore(s => s.conversationVariables) as ConversationVariable[] const updateChatVarList = useStore(s => s.setConversationVariables) const appId = useStore(s => s.appId) as string const { invalidateConversationVarValues, } = useInspectVarsCrud() const [showTip, setShowTip] = useState(true) const [showVariableModal, setShowVariableModal] = useState(false) const [currentVar, setCurrentVar] = useState() const [showRemoveVarConfirm, setShowRemoveConfirm] = useState(false) const [cacheForDelete, setCacheForDelete] = useState() const collaborativeWorkflow = useCollaborativeWorkflow() const getEffectedNodes = useCallback((chatVar: ConversationVariable) => { const { nodes: allNodes } = collaborativeWorkflow.getState() return findUsedVarNodes( ['conversation', chatVar.name], allNodes, ) }, [collaborativeWorkflow]) const removeUsedVarInNodes = useCallback((chatVar: ConversationVariable) => { const { nodes, setNodes } = collaborativeWorkflow.getState() const effectedNodes = getEffectedNodes(chatVar) const newNodes = nodes.map((node) => { if (effectedNodes.find(n => n.id === node.id)) return updateNodeVars(node, ['conversation', chatVar.name], []) return node }) setNodes(newNodes) }, [getEffectedNodes, collaborativeWorkflow]) const handleEdit = (chatVar: ConversationVariable) => { setCurrentVar(chatVar) setShowVariableModal(true) } const handleDelete = useCallback(async (chatVar: ConversationVariable) => { removeUsedVarInNodes(chatVar) const newVarList = varList.filter(v => v.id !== chatVar.id) updateChatVarList(newVarList) setCacheForDelete(undefined) setShowRemoveConfirm(false) // Use new dedicated conversation variables API instead of workflow draft sync try { await updateConversationVariables({ appId, conversationVariables: newVarList, }) // Emit update event to other connected clients const socket = webSocketClient.getSocket(appId) if (socket) { socket.emit('collaboration_event', { type: 'vars_and_features_update', }) } invalidateConversationVarValues() } catch (error) { console.error('Failed to update conversation variables:', error) // Revert local state on error updateChatVarList(varList) } }, [removeUsedVarInNodes, updateChatVarList, varList, appId, invalidateConversationVarValues]) 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) => { let newList: ConversationVariable[] if (!currentVar) { // Adding new conversation variable newList = [chatVar, ...varList] updateChatVarList(newList) // Use new dedicated conversation variables API try { await updateConversationVariables({ appId, conversationVariables: newList, }) const socket = webSocketClient.getSocket(appId) if (socket) { socket.emit('collaboration_event', { type: 'vars_and_features_update', }) } invalidateConversationVarValues() } catch (error) { console.error('Failed to update conversation variables:', error) // Revert local state on error updateChatVarList(varList) } return } // Updating existing conversation variable newList = varList.map(v => v.id === currentVar.id ? chatVar : v) updateChatVarList(newList) // side effects of rename conversation variable if (currentVar.name !== chatVar.name) { const { nodes, setNodes } = collaborativeWorkflow.getState() const effectedNodes = getEffectedNodes(currentVar) const newNodes = nodes.map((node) => { if (effectedNodes.find(n => n.id === node.id)) return updateNodeVars(node, ['conversation', currentVar.name], ['conversation', chatVar.name]) return node }) setNodes(newNodes) } // Use new dedicated conversation variables API try { await updateConversationVariables({ appId, conversationVariables: newList, }) const socket = webSocketClient.getSocket(appId) if (socket) { socket.emit('collaboration_event', { type: 'vars_and_features_update', }) } invalidateConversationVarValues() } catch (error) { console.error('Failed to update conversation variables:', error) // Revert local state on error updateChatVarList(varList) } }, [currentVar, getEffectedNodes, collaborativeWorkflow, updateChatVarList, varList, appId, invalidateConversationVarValues]) return (
{t('chatVariable.panelTitle', { ns: 'workflow' })}
setShowTip(!showTip)}>
setShowChatVariablePanel(false)} >
{showTip && (
TIPS
{t('chatVariable.panelDescription', { ns: 'workflow' })}
conversation_var
String
WRITE
{t('blocks.assigner', { ns: 'workflow' })}
READ
{t('blocks.llm', { ns: 'workflow' })}
)}
setCurrentVar(undefined)} />
{varList.map(chatVar => ( ))}
setShowRemoveConfirm(false)} onConfirm={() => cacheForDelete && handleDelete(cacheForDelete)} />
) } export default memo(ChatVariablePanel)