import React, { useState, useRef, useEffect } from 'react'; import { useNavigate, useParams, useSearchParams } from 'react-router-dom'; import { ArrowLeft, Play, Save, Rocket, Plus, Bot, UserCheck, Wrench, Ban, Zap, X, Copy, MousePointer2 } from 'lucide-react'; import { Button, Input, Badge } from '../components/UI'; import { mockAssistants, mockKnowledgeBases, mockWorkflows } from '../services/mockData'; import { WorkflowNode, WorkflowEdge, Workflow } from '../types'; import { DebugDrawer } from './Assistants'; export const WorkflowEditorPage: React.FC = () => { const navigate = useNavigate(); const { id } = useParams(); const [searchParams] = useSearchParams(); // Template data for new workflows const templateName = searchParams.get('name'); const templateType = searchParams.get('template'); // Find initial workflow or create a new one const existingWf = mockWorkflows.find(w => w.id === id); const [name, setName] = useState(templateName || existingWf?.name || '新工作流'); const [nodes, setNodes] = useState(() => { if (existingWf) return existingWf.nodes; if (templateType === 'lead') { return [ { name: 'introduction', type: 'conversation', isStart: true, metadata: { position: { x: 100, y: 100 } }, prompt: "You are Morgan from GrowthPartners. Start with: 'Hello, this is Morgan from GrowthPartners. We help businesses improve their operational efficiency through custom software solutions. Do you have a few minutes to chat about how we might be able to help your business?'", messagePlan: { firstMessage: "Hello, this is Morgan from GrowthPartners. Do you have a few minutes to chat?" } }, { name: 'need_discovery', type: 'conversation', metadata: { position: { x: 450, y: 250 } }, prompt: "Conduct need discovery by asking about business challenges...", variableExtractionPlan: { output: [ { title: 'industry', type: 'string', description: 'user industry' }, { title: 'pain_points', type: 'string', description: 'main challenges' } ] } }, { name: 'hangup_node', type: 'end', metadata: { position: { x: 450, y: 550 } }, tool: { type: 'endCall', function: { name: 'hangup', parameters: {} }, messages: [{ type: 'request-start', content: 'Thank you for your time!', blocking: true }] } } ]; } return [ { name: 'start_node', type: 'conversation', isStart: true, metadata: { position: { x: 200, y: 200 } }, prompt: '欢迎对话系统...', messagePlan: { firstMessage: '你好!' } } ]; }); const [edges, setEdges] = useState(existingWf?.edges || []); const [selectedNodeName, setSelectedNodeName] = useState(null); const [isAddMenuOpen, setIsAddMenuOpen] = useState(false); const [isDebugOpen, setIsDebugOpen] = useState(false); const [zoom, setZoom] = useState(1); const [panOffset, setPanOffset] = useState({ x: 0, y: 0 }); const [draggingNodeName, setDraggingNodeName] = useState(null); const [isPanning, setIsPanning] = useState(false); const dragOffset = useRef({ x: 0, y: 0 }); const panStart = useRef({ x: 0, y: 0 }); const selectedNode = nodes.find(n => n.name === selectedNodeName); // Scroll Zoom handler const handleWheel = (e: React.WheelEvent) => { e.preventDefault(); const zoomSpeed = 0.001; const newZoom = Math.min(Math.max(zoom - e.deltaY * zoomSpeed, 0.2), 3); setZoom(newZoom); }; // Middle mouse click handler for panning const handleMouseDown = (e: React.MouseEvent) => { // Button 1 is middle mouse if (e.button === 1) { e.preventDefault(); setIsPanning(true); panStart.current = { x: e.clientX - panOffset.x, y: e.clientY - panOffset.y }; } }; const handleNodeMouseDown = (e: React.MouseEvent, nodeName: string) => { if (e.button !== 0) return; // Only left click for nodes e.stopPropagation(); setSelectedNodeName(nodeName); setDraggingNodeName(nodeName); const node = nodes.find(n => n.name === nodeName); if (node) { dragOffset.current = { x: e.clientX - node.metadata.position.x * zoom, y: e.clientY - node.metadata.position.y * zoom }; } }; useEffect(() => { const handleMouseMove = (e: MouseEvent) => { if (draggingNodeName) { setNodes(prev => prev.map(n => n.name === draggingNodeName ? { ...n, metadata: { ...n.metadata, position: { x: (e.clientX - dragOffset.current.x) / zoom, y: (e.clientY - dragOffset.current.y) / zoom } } } : n )); } else if (isPanning) { setPanOffset({ x: e.clientX - panStart.current.x, y: e.clientY - panStart.current.y }); } }; const handleMouseUp = (e: MouseEvent) => { setDraggingNodeName(null); setIsPanning(false); }; window.addEventListener('mousemove', handleMouseMove); window.addEventListener('mouseup', handleMouseUp); return () => { window.removeEventListener('mousemove', handleMouseMove); window.removeEventListener('mouseup', handleMouseUp); }; }, [draggingNodeName, isPanning, zoom]); const addNode = (type: WorkflowNode['type']) => { const newNode: WorkflowNode = { name: `${type}_${Date.now()}`, type, metadata: { position: { x: (300 - panOffset.x) / zoom, y: (300 - panOffset.y) / zoom } }, prompt: type === 'conversation' ? '输入该节点的 Prompt...' : '', tool: type === 'end' ? { type: 'endCall', function: { name: 'hangup', parameters: {} } } : undefined }; setNodes([...nodes, newNode]); setIsAddMenuOpen(false); }; const updateNodeData = (field: string, value: any) => { if (!selectedNodeName) return; setNodes(prev => prev.map(n => n.name === selectedNodeName ? { ...n, [field]: value } : n)); }; const handleSave = () => { const now = new Date().toISOString().replace('T', ' ').substring(0, 16); const updatedWorkflow: Workflow = { id: id || `wf_${Date.now()}`, name, nodeCount: nodes.length, createdAt: existingWf?.createdAt || now, updatedAt: now, nodes, edges, }; if (id) { const idx = mockWorkflows.findIndex(w => w.id === id); if (idx !== -1) mockWorkflows[idx] = updatedWorkflow; } else { mockWorkflows.push(updatedWorkflow); } alert('保存成功!工作流已同步至列表。'); navigate('/workflows'); }; return (
{/* Editor Header */}
setName(e.target.value)} className="bg-transparent border-transparent focus:bg-white/5 font-bold text-sm h-8 w-48 shadow-none text-white" />
{/* Canvas Area */}
{/* Floating Controls */}
{isAddMenuOpen && (
)}
{Math.round(zoom * 100)}% | POS: {Math.round(panOffset.x)}, {Math.round(panOffset.y)}
{/* Scalable Container */}
{nodes.map(node => (
handleNodeMouseDown(e, node.name)} style={{ left: node.metadata.position.x, top: node.metadata.position.y }} className={`absolute w-56 p-4 rounded-xl border bg-card/70 backdrop-blur-sm cursor-grab active:cursor-grabbing group transition-shadow ${selectedNodeName === node.name ? 'border-primary shadow-[0_0_30px_rgba(6,182,212,0.3)]' : 'border-white/10 hover:border-white/30'}`} >
{node.name}
{node.isStart && START}
{node.prompt || (node.tool ? `Action: ${node.tool.type}` : '待配置逻辑...')}
{/* Port simulation */}
))}
{/* Bottom Right: Minimap */}
Minimap
{nodes.map((n, i) => { // Simple scaling for minimap const mx = (n.metadata.position.x / 10) % 100; const my = (n.metadata.position.y / 10) % 100; return (
); })}
{/* Viewport visualization */}
{/* Navigator Guide */}
鼠标中键拖拽画布 | 滚轮缩放
{/* Right Settings Panel */} {selectedNode && (

节点设置

updateNodeData('name', e.target.value)} className="h-8 text-xs font-mono text-white bg-white/10" />
{selectedNode.type.toUpperCase()}
{selectedNode.type === 'conversation' && ( <>