@marktoflow/gui 2.0.0-alpha.4 → 2.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +48 -170
- package/client.log +0 -0
- package/dist/client/assets/index-DQeR1ew6.css +1 -0
- package/dist/client/assets/index-LbIVPHbD.js +833 -0
- package/dist/client/assets/index-LbIVPHbD.js.map +1 -0
- package/dist/client/index.html +2 -2
- package/dist/client/marktoflow-logo.png +0 -0
- package/dist/server/{server/index.js → index.js} +53 -6
- package/dist/server/index.js.map +1 -0
- package/dist/server/routes/admin.js +95 -0
- package/dist/server/routes/admin.js.map +1 -0
- package/dist/server/{server/routes → routes}/ai.js +2 -2
- package/dist/server/{server/routes → routes}/ai.js.map +1 -1
- package/dist/server/routes/collaboration.js +104 -0
- package/dist/server/routes/collaboration.js.map +1 -0
- package/dist/server/routes/execute.js +230 -0
- package/dist/server/routes/execute.js.map +1 -0
- package/dist/server/routes/executions.js +125 -0
- package/dist/server/routes/executions.js.map +1 -0
- package/dist/server/routes/form.js +160 -0
- package/dist/server/routes/form.js.map +1 -0
- package/dist/server/routes/settings.js +90 -0
- package/dist/server/routes/settings.js.map +1 -0
- package/dist/server/routes/templates.js +106 -0
- package/dist/server/routes/templates.js.map +1 -0
- package/dist/server/routes/versions.js +101 -0
- package/dist/server/routes/versions.js.map +1 -0
- package/dist/server/{server/routes → routes}/workflows.js +37 -1
- package/dist/server/routes/workflows.js.map +1 -0
- package/dist/server/services/AIService.js +152 -0
- package/dist/server/services/AIService.js.map +1 -0
- package/dist/server/services/ExecutionManager.js +571 -0
- package/dist/server/services/ExecutionManager.js.map +1 -0
- package/dist/server/services/VersionService.js +65 -0
- package/dist/server/services/VersionService.js.map +1 -0
- package/dist/server/{server/services → services}/WorkflowService.js +166 -17
- package/dist/server/services/WorkflowService.js.map +1 -0
- package/dist/server/{server/services → services}/agents/copilot-provider.js +32 -0
- package/dist/server/services/agents/copilot-provider.js.map +1 -0
- package/dist/server/{server/websocket → websocket}/index.js +54 -0
- package/dist/server/websocket/index.js.map +1 -0
- package/dist/{server/shared → shared}/constants.js +9 -0
- package/dist/shared/constants.js.map +1 -0
- package/dist/shared/settings.js +51 -0
- package/dist/shared/settings.js.map +1 -0
- package/package.json +32 -14
- package/public/marktoflow-logo.png +0 -0
- package/scripts/flatten-dist.js +69 -0
- package/server.log +0 -0
- package/tests/integration/fixtures/test-workflow.md +6 -0
- package/.turbo/turbo-build.log +0 -26
- package/dist/client/assets/index-C90Y_aBX.js +0 -678
- package/dist/client/assets/index-C90Y_aBX.js.map +0 -1
- package/dist/client/assets/index-CRWeQ3NN.css +0 -1
- package/dist/server/server/index.js.map +0 -1
- package/dist/server/server/routes/execute.js +0 -63
- package/dist/server/server/routes/execute.js.map +0 -1
- package/dist/server/server/routes/workflows.js.map +0 -1
- package/dist/server/server/services/AIService.js +0 -69
- package/dist/server/server/services/AIService.js.map +0 -1
- package/dist/server/server/services/WorkflowService.js.map +0 -1
- package/dist/server/server/services/agents/copilot-provider.js.map +0 -1
- package/dist/server/server/websocket/index.js.map +0 -1
- package/dist/server/shared/constants.js.map +0 -1
- package/playwright.config.ts +0 -27
- package/postcss.config.js +0 -6
- package/src/client/App.tsx +0 -520
- package/src/client/components/Canvas/Canvas.tsx +0 -423
- package/src/client/components/Canvas/ExecutionOverlay.tsx +0 -847
- package/src/client/components/Canvas/ForEachNode.tsx +0 -128
- package/src/client/components/Canvas/IfElseNode.tsx +0 -126
- package/src/client/components/Canvas/NodeContextMenu.tsx +0 -188
- package/src/client/components/Canvas/OutputNode.tsx +0 -111
- package/src/client/components/Canvas/ParallelNode.tsx +0 -140
- package/src/client/components/Canvas/StepNode.tsx +0 -106
- package/src/client/components/Canvas/SubWorkflowNode.tsx +0 -141
- package/src/client/components/Canvas/SwitchNode.tsx +0 -164
- package/src/client/components/Canvas/Toolbar.tsx +0 -189
- package/src/client/components/Canvas/TransformNode.tsx +0 -185
- package/src/client/components/Canvas/TriggerNode.tsx +0 -128
- package/src/client/components/Canvas/TryCatchNode.tsx +0 -164
- package/src/client/components/Canvas/WhileNode.tsx +0 -129
- package/src/client/components/Canvas/index.ts +0 -24
- package/src/client/components/Editor/InputsEditor.tsx +0 -458
- package/src/client/components/Editor/NewStepWizard.tsx +0 -344
- package/src/client/components/Editor/StepEditor.tsx +0 -532
- package/src/client/components/Editor/YamlEditor.tsx +0 -160
- package/src/client/components/Panels/PropertiesPanel.tsx +0 -589
- package/src/client/components/Prompt/ChangePreview.tsx +0 -281
- package/src/client/components/Prompt/PromptHistoryPanel.tsx +0 -209
- package/src/client/components/Prompt/PromptInput.tsx +0 -108
- package/src/client/components/Sidebar/Sidebar.tsx +0 -343
- package/src/client/components/common/Breadcrumb.tsx +0 -40
- package/src/client/components/common/Button.tsx +0 -68
- package/src/client/components/common/ContextMenu.tsx +0 -202
- package/src/client/components/common/KeyboardShortcuts.tsx +0 -143
- package/src/client/components/common/Modal.tsx +0 -93
- package/src/client/components/common/Tabs.tsx +0 -57
- package/src/client/components/common/ThemeToggle.tsx +0 -63
- package/src/client/components/index.ts +0 -32
- package/src/client/hooks/index.ts +0 -4
- package/src/client/hooks/useAIPrompt.ts +0 -108
- package/src/client/hooks/useCanvas.ts +0 -247
- package/src/client/hooks/useWebSocket.ts +0 -164
- package/src/client/hooks/useWorkflow.ts +0 -138
- package/src/client/main.tsx +0 -10
- package/src/client/stores/canvasStore.ts +0 -348
- package/src/client/stores/editorStore.ts +0 -133
- package/src/client/stores/executionStore.ts +0 -440
- package/src/client/stores/index.ts +0 -4
- package/src/client/stores/layoutStore.ts +0 -103
- package/src/client/stores/navigationStore.ts +0 -49
- package/src/client/stores/promptStore.ts +0 -113
- package/src/client/stores/themeStore.ts +0 -75
- package/src/client/stores/workflowStore.ts +0 -177
- package/src/client/styles/globals.css +0 -346
- package/src/client/utils/cn.ts +0 -9
- package/src/client/utils/index.ts +0 -4
- package/src/client/utils/serviceIcons.tsx +0 -97
- package/src/client/utils/stepValidation.ts +0 -155
- package/src/client/utils/workflowToGraph.ts +0 -299
- package/src/server/index.ts +0 -114
- package/src/server/routes/ai.ts +0 -91
- package/src/server/routes/execute.ts +0 -71
- package/src/server/routes/tools.ts +0 -970
- package/src/server/routes/workflows.ts +0 -106
- package/src/server/services/AIService.ts +0 -105
- package/src/server/services/FileWatcher.ts +0 -69
- package/src/server/services/WorkflowService.ts +0 -441
- package/src/server/services/agents/claude-code-provider.ts +0 -320
- package/src/server/services/agents/claude-provider.ts +0 -248
- package/src/server/services/agents/codex-provider.ts +0 -398
- package/src/server/services/agents/copilot-provider.ts +0 -311
- package/src/server/services/agents/demo-provider.ts +0 -184
- package/src/server/services/agents/index.ts +0 -31
- package/src/server/services/agents/ollama-provider.ts +0 -267
- package/src/server/services/agents/prompts.ts +0 -509
- package/src/server/services/agents/registry.ts +0 -310
- package/src/server/services/agents/types.ts +0 -146
- package/src/server/websocket/index.ts +0 -104
- package/src/shared/constants.ts +0 -180
- package/src/shared/types.ts +0 -179
- package/tailwind.config.ts +0 -73
- package/tests/e2e/app.spec.ts +0 -90
- package/tests/e2e/canvas.spec.ts +0 -128
- package/tests/e2e/workflow.spec.ts +0 -185
- package/tests/integration/api.test.ts +0 -452
- package/tests/integration/testApp.ts +0 -31
- package/tests/setup.ts +0 -72
- package/tests/unit/ForEachNode.test.tsx +0 -218
- package/tests/unit/IfElseNode.test.tsx +0 -188
- package/tests/unit/ParallelNode.test.tsx +0 -264
- package/tests/unit/SwitchNode.test.tsx +0 -252
- package/tests/unit/TransformNode.test.tsx +0 -386
- package/tests/unit/TryCatchNode.test.tsx +0 -243
- package/tests/unit/WhileNode.test.tsx +0 -226
- package/tests/unit/canvasStore.test.ts +0 -502
- package/tests/unit/codexProvider.test.ts +0 -399
- package/tests/unit/components.test.tsx +0 -151
- package/tests/unit/executionStore.test.ts +0 -527
- package/tests/unit/layoutStore.test.ts +0 -194
- package/tests/unit/navigationStore.test.ts +0 -152
- package/tests/unit/serviceIcons.test.ts +0 -197
- package/tests/unit/stepValidation.test.ts +0 -226
- package/tests/unit/themeStore.test.ts +0 -141
- package/tests/unit/workflowToGraph.test.ts +0 -289
- package/tsconfig.json +0 -29
- package/tsconfig.server.json +0 -28
- package/vite.config.ts +0 -31
- package/vitest.config.ts +0 -26
- /package/dist/server/{server/routes → routes}/tools.js +0 -0
- /package/dist/server/{server/routes → routes}/tools.js.map +0 -0
- /package/dist/server/{server/services → services}/FileWatcher.js +0 -0
- /package/dist/server/{server/services → services}/FileWatcher.js.map +0 -0
- /package/dist/server/{server/services → services}/agents/claude-code-provider.js +0 -0
- /package/dist/server/{server/services → services}/agents/claude-code-provider.js.map +0 -0
- /package/dist/server/{server/services → services}/agents/claude-provider.js +0 -0
- /package/dist/server/{server/services → services}/agents/claude-provider.js.map +0 -0
- /package/dist/server/{server/services → services}/agents/codex-provider.js +0 -0
- /package/dist/server/{server/services → services}/agents/codex-provider.js.map +0 -0
- /package/dist/server/{server/services → services}/agents/demo-provider.js +0 -0
- /package/dist/server/{server/services → services}/agents/demo-provider.js.map +0 -0
- /package/dist/server/{server/services → services}/agents/index.js +0 -0
- /package/dist/server/{server/services → services}/agents/index.js.map +0 -0
- /package/dist/server/{server/services → services}/agents/ollama-provider.js +0 -0
- /package/dist/server/{server/services → services}/agents/ollama-provider.js.map +0 -0
- /package/dist/server/{server/services → services}/agents/prompts.js +0 -0
- /package/dist/server/{server/services → services}/agents/prompts.js.map +0 -0
- /package/dist/server/{server/services → services}/agents/registry.js +0 -0
- /package/dist/server/{server/services → services}/agents/registry.js.map +0 -0
- /package/dist/server/{server/services → services}/agents/types.js +0 -0
- /package/dist/server/{server/services → services}/agents/types.js.map +0 -0
- /package/dist/{server/shared → shared}/types.js +0 -0
- /package/dist/{server/shared → shared}/types.js.map +0 -0
|
@@ -1,106 +0,0 @@
|
|
|
1
|
-
import { memo } from 'react';
|
|
2
|
-
import { Handle, Position, type Node, type NodeProps } from '@xyflow/react';
|
|
3
|
-
import { Play, CheckCircle, XCircle, Clock, AlertCircle } from 'lucide-react';
|
|
4
|
-
import { getServiceIcon } from '../../utils/serviceIcons';
|
|
5
|
-
|
|
6
|
-
export interface StepNodeData extends Record<string, unknown> {
|
|
7
|
-
id: string;
|
|
8
|
-
name?: string;
|
|
9
|
-
action: string;
|
|
10
|
-
status?: 'pending' | 'running' | 'completed' | 'failed' | 'skipped';
|
|
11
|
-
retryCount?: number;
|
|
12
|
-
error?: string;
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
export type StepNodeType = Node<StepNodeData, 'step'>;
|
|
16
|
-
|
|
17
|
-
function StepNodeComponent({ data, selected }: NodeProps<StepNodeType>) {
|
|
18
|
-
const serviceName = data.action?.split('.')[0] || 'unknown';
|
|
19
|
-
const methodName = data.action?.split('.').slice(1).join('.') || data.action;
|
|
20
|
-
const ServiceIcon = getServiceIcon(serviceName);
|
|
21
|
-
|
|
22
|
-
const statusConfig: Record<
|
|
23
|
-
NonNullable<StepNodeData['status']>,
|
|
24
|
-
{ icon: typeof Clock; color: string; bgColor: string; animate?: boolean }
|
|
25
|
-
> = {
|
|
26
|
-
pending: { icon: Clock, color: 'text-gray-400', bgColor: 'bg-gray-400/10' },
|
|
27
|
-
running: {
|
|
28
|
-
icon: Play,
|
|
29
|
-
color: 'text-warning',
|
|
30
|
-
bgColor: 'bg-warning/10',
|
|
31
|
-
animate: true,
|
|
32
|
-
},
|
|
33
|
-
completed: {
|
|
34
|
-
icon: CheckCircle,
|
|
35
|
-
color: 'text-success',
|
|
36
|
-
bgColor: 'bg-success/10',
|
|
37
|
-
},
|
|
38
|
-
failed: { icon: XCircle, color: 'text-error', bgColor: 'bg-error/10' },
|
|
39
|
-
skipped: {
|
|
40
|
-
icon: AlertCircle,
|
|
41
|
-
color: 'text-gray-500',
|
|
42
|
-
bgColor: 'bg-gray-500/10',
|
|
43
|
-
},
|
|
44
|
-
};
|
|
45
|
-
|
|
46
|
-
const status = data.status || 'pending';
|
|
47
|
-
const config = statusConfig[status];
|
|
48
|
-
const StatusIcon = config.icon;
|
|
49
|
-
|
|
50
|
-
return (
|
|
51
|
-
<div
|
|
52
|
-
className={`step-node p-0 ${selected ? 'selected' : ''} ${status === 'running' ? 'running' : ''} ${status === 'completed' ? 'completed' : ''} ${status === 'failed' ? 'failed' : ''}`}
|
|
53
|
-
>
|
|
54
|
-
{/* Input handle */}
|
|
55
|
-
<Handle
|
|
56
|
-
type="target"
|
|
57
|
-
position={Position.Top}
|
|
58
|
-
className="!w-3 !h-3 !bg-primary !border-2 !border-node-bg"
|
|
59
|
-
/>
|
|
60
|
-
|
|
61
|
-
{/* Node header */}
|
|
62
|
-
<div className="flex items-center gap-3 p-3 border-b border-node-border">
|
|
63
|
-
<div className="w-8 h-8 rounded-lg bg-primary/10 flex items-center justify-center">
|
|
64
|
-
<ServiceIcon className="w-5 h-5 text-primary" />
|
|
65
|
-
</div>
|
|
66
|
-
<div className="flex-1 min-w-0">
|
|
67
|
-
<div className="text-sm font-medium text-white truncate">
|
|
68
|
-
{data.name || data.id}
|
|
69
|
-
</div>
|
|
70
|
-
<div className="text-xs text-gray-400 truncate">{serviceName}</div>
|
|
71
|
-
</div>
|
|
72
|
-
<div
|
|
73
|
-
className={`w-6 h-6 rounded-full ${config.bgColor} flex items-center justify-center`}
|
|
74
|
-
>
|
|
75
|
-
<StatusIcon
|
|
76
|
-
className={`w-4 h-4 ${config.color} ${config.animate ? 'animate-pulse' : ''}`}
|
|
77
|
-
/>
|
|
78
|
-
</div>
|
|
79
|
-
</div>
|
|
80
|
-
|
|
81
|
-
{/* Node body */}
|
|
82
|
-
<div className="p-3">
|
|
83
|
-
<div className="text-xs text-gray-300 font-mono truncate">
|
|
84
|
-
{methodName}
|
|
85
|
-
</div>
|
|
86
|
-
{data.retryCount && data.retryCount > 0 && (
|
|
87
|
-
<div className="mt-2 text-xs text-warning">
|
|
88
|
-
Retry #{data.retryCount}
|
|
89
|
-
</div>
|
|
90
|
-
)}
|
|
91
|
-
{data.error && (
|
|
92
|
-
<div className="mt-2 text-xs text-error truncate">{data.error}</div>
|
|
93
|
-
)}
|
|
94
|
-
</div>
|
|
95
|
-
|
|
96
|
-
{/* Output handle */}
|
|
97
|
-
<Handle
|
|
98
|
-
type="source"
|
|
99
|
-
position={Position.Bottom}
|
|
100
|
-
className="!w-3 !h-3 !bg-primary !border-2 !border-node-bg"
|
|
101
|
-
/>
|
|
102
|
-
</div>
|
|
103
|
-
);
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
export const StepNode = memo(StepNodeComponent);
|
|
@@ -1,141 +0,0 @@
|
|
|
1
|
-
import { memo, useState, useCallback } from 'react';
|
|
2
|
-
import { Handle, Position, type Node, type NodeProps } from '@xyflow/react';
|
|
3
|
-
import {
|
|
4
|
-
FolderOpen,
|
|
5
|
-
ChevronDown,
|
|
6
|
-
ChevronRight,
|
|
7
|
-
ExternalLink,
|
|
8
|
-
ArrowRight,
|
|
9
|
-
} from 'lucide-react';
|
|
10
|
-
import { useNavigationStore } from '../../stores/navigationStore';
|
|
11
|
-
import { useWorkflowStore } from '../../stores/workflowStore';
|
|
12
|
-
|
|
13
|
-
export interface SubWorkflowNodeData extends Record<string, unknown> {
|
|
14
|
-
id: string;
|
|
15
|
-
name?: string;
|
|
16
|
-
workflowPath: string;
|
|
17
|
-
stepCount?: number;
|
|
18
|
-
status?: 'pending' | 'running' | 'completed' | 'failed';
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
export type SubWorkflowNodeType = Node<SubWorkflowNodeData, 'subworkflow'>;
|
|
22
|
-
|
|
23
|
-
function SubWorkflowNodeComponent({
|
|
24
|
-
data,
|
|
25
|
-
selected,
|
|
26
|
-
}: NodeProps<SubWorkflowNodeType>) {
|
|
27
|
-
const [expanded, setExpanded] = useState(false);
|
|
28
|
-
const { pushWorkflow, breadcrumbs, setRootWorkflow } = useNavigationStore();
|
|
29
|
-
const { loadWorkflow, selectedWorkflow, currentWorkflow } = useWorkflowStore();
|
|
30
|
-
|
|
31
|
-
// Handle drilling down into sub-workflow
|
|
32
|
-
const handleDrillDown = useCallback(() => {
|
|
33
|
-
// If this is the first navigation, set the current workflow as root
|
|
34
|
-
if (breadcrumbs.length === 0 && selectedWorkflow && currentWorkflow) {
|
|
35
|
-
setRootWorkflow({
|
|
36
|
-
id: selectedWorkflow,
|
|
37
|
-
name: currentWorkflow.metadata?.name || 'Main Workflow',
|
|
38
|
-
path: selectedWorkflow,
|
|
39
|
-
});
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
// Push the sub-workflow onto the navigation stack
|
|
43
|
-
pushWorkflow({
|
|
44
|
-
id: data.id,
|
|
45
|
-
name: data.name || data.id,
|
|
46
|
-
path: data.workflowPath,
|
|
47
|
-
});
|
|
48
|
-
|
|
49
|
-
// Load the sub-workflow
|
|
50
|
-
loadWorkflow(data.workflowPath);
|
|
51
|
-
}, [data, pushWorkflow, breadcrumbs, setRootWorkflow, selectedWorkflow, currentWorkflow, loadWorkflow]);
|
|
52
|
-
|
|
53
|
-
const statusColors = {
|
|
54
|
-
pending: 'border-node-border',
|
|
55
|
-
running: 'border-warning',
|
|
56
|
-
completed: 'border-success',
|
|
57
|
-
failed: 'border-error',
|
|
58
|
-
};
|
|
59
|
-
|
|
60
|
-
const status = data.status || 'pending';
|
|
61
|
-
|
|
62
|
-
return (
|
|
63
|
-
<div
|
|
64
|
-
className={`step-node p-0 min-w-[250px] ${selected ? 'selected' : ''} ${statusColors[status]}`}
|
|
65
|
-
>
|
|
66
|
-
{/* Input handle */}
|
|
67
|
-
<Handle
|
|
68
|
-
type="target"
|
|
69
|
-
position={Position.Top}
|
|
70
|
-
className="!w-3 !h-3 !bg-info !border-2 !border-node-bg"
|
|
71
|
-
/>
|
|
72
|
-
|
|
73
|
-
{/* Node header */}
|
|
74
|
-
<div className="flex items-center gap-3 p-3 border-b border-node-border bg-info/5">
|
|
75
|
-
<button
|
|
76
|
-
onClick={() => setExpanded(!expanded)}
|
|
77
|
-
className="w-6 h-6 rounded flex items-center justify-center hover:bg-white/10 transition-colors"
|
|
78
|
-
>
|
|
79
|
-
{expanded ? (
|
|
80
|
-
<ChevronDown className="w-4 h-4 text-info" />
|
|
81
|
-
) : (
|
|
82
|
-
<ChevronRight className="w-4 h-4 text-info" />
|
|
83
|
-
)}
|
|
84
|
-
</button>
|
|
85
|
-
<div className="w-8 h-8 rounded-lg bg-info/10 flex items-center justify-center">
|
|
86
|
-
<FolderOpen className="w-5 h-5 text-info" />
|
|
87
|
-
</div>
|
|
88
|
-
<div className="flex-1 min-w-0">
|
|
89
|
-
<div className="text-sm font-medium text-white truncate">
|
|
90
|
-
{data.name || data.id}
|
|
91
|
-
</div>
|
|
92
|
-
<div className="text-xs text-gray-400">Sub-workflow</div>
|
|
93
|
-
</div>
|
|
94
|
-
<button
|
|
95
|
-
onClick={handleDrillDown}
|
|
96
|
-
className="w-6 h-6 rounded flex items-center justify-center hover:bg-info/20 transition-colors"
|
|
97
|
-
title="Drill into sub-workflow"
|
|
98
|
-
>
|
|
99
|
-
<ArrowRight className="w-4 h-4 text-info" />
|
|
100
|
-
</button>
|
|
101
|
-
</div>
|
|
102
|
-
|
|
103
|
-
{/* Node body */}
|
|
104
|
-
<div className="p-3">
|
|
105
|
-
<div className="text-xs text-gray-400 font-mono truncate">
|
|
106
|
-
{data.workflowPath}
|
|
107
|
-
</div>
|
|
108
|
-
{data.stepCount !== undefined && (
|
|
109
|
-
<div className="mt-2 flex items-center gap-2">
|
|
110
|
-
<span className="text-xs px-2 py-0.5 rounded-full bg-info/10 text-info">
|
|
111
|
-
{data.stepCount} steps
|
|
112
|
-
</span>
|
|
113
|
-
</div>
|
|
114
|
-
)}
|
|
115
|
-
</div>
|
|
116
|
-
|
|
117
|
-
{/* Expanded content */}
|
|
118
|
-
{expanded && (
|
|
119
|
-
<div className="border-t border-node-border p-3 bg-black/20">
|
|
120
|
-
<button
|
|
121
|
-
onClick={handleDrillDown}
|
|
122
|
-
className="w-full flex items-center justify-center gap-2 px-3 py-2 bg-info/10 hover:bg-info/20 rounded text-sm text-info transition-colors"
|
|
123
|
-
>
|
|
124
|
-
<FolderOpen className="w-4 h-4" />
|
|
125
|
-
Open Sub-workflow
|
|
126
|
-
<ArrowRight className="w-4 h-4" />
|
|
127
|
-
</button>
|
|
128
|
-
</div>
|
|
129
|
-
)}
|
|
130
|
-
|
|
131
|
-
{/* Output handle */}
|
|
132
|
-
<Handle
|
|
133
|
-
type="source"
|
|
134
|
-
position={Position.Bottom}
|
|
135
|
-
className="!w-3 !h-3 !bg-info !border-2 !border-node-bg"
|
|
136
|
-
/>
|
|
137
|
-
</div>
|
|
138
|
-
);
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
export const SubWorkflowNode = memo(SubWorkflowNodeComponent);
|
|
@@ -1,164 +0,0 @@
|
|
|
1
|
-
import { memo } from 'react';
|
|
2
|
-
import { Handle, Position, type Node, type NodeProps } from '@xyflow/react';
|
|
3
|
-
import { GitFork, CheckCircle, XCircle, Clock } from 'lucide-react';
|
|
4
|
-
|
|
5
|
-
export interface SwitchNodeData extends Record<string, unknown> {
|
|
6
|
-
id: string;
|
|
7
|
-
name?: string;
|
|
8
|
-
expression: string;
|
|
9
|
-
cases: Record<string, unknown>;
|
|
10
|
-
hasDefault?: boolean;
|
|
11
|
-
status?: 'pending' | 'running' | 'completed' | 'failed' | 'skipped';
|
|
12
|
-
activeCase?: string | null;
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
export type SwitchNodeType = Node<SwitchNodeData, 'switch'>;
|
|
16
|
-
|
|
17
|
-
function SwitchNodeComponent({ data, selected }: NodeProps<SwitchNodeType>) {
|
|
18
|
-
const statusConfig: Record<
|
|
19
|
-
NonNullable<SwitchNodeData['status']>,
|
|
20
|
-
{ icon: typeof Clock; color: string; bgColor: string; animate?: boolean }
|
|
21
|
-
> = {
|
|
22
|
-
pending: { icon: Clock, color: 'text-gray-400', bgColor: 'bg-gray-400/10' },
|
|
23
|
-
running: {
|
|
24
|
-
icon: GitFork,
|
|
25
|
-
color: 'text-purple-400',
|
|
26
|
-
bgColor: 'bg-purple-400/10',
|
|
27
|
-
animate: true,
|
|
28
|
-
},
|
|
29
|
-
completed: {
|
|
30
|
-
icon: CheckCircle,
|
|
31
|
-
color: 'text-success',
|
|
32
|
-
bgColor: 'bg-success/10',
|
|
33
|
-
},
|
|
34
|
-
failed: { icon: XCircle, color: 'text-error', bgColor: 'bg-error/10' },
|
|
35
|
-
skipped: {
|
|
36
|
-
icon: XCircle,
|
|
37
|
-
color: 'text-gray-500',
|
|
38
|
-
bgColor: 'bg-gray-500/10',
|
|
39
|
-
},
|
|
40
|
-
};
|
|
41
|
-
|
|
42
|
-
const status = data.status || 'pending';
|
|
43
|
-
const config = statusConfig[status];
|
|
44
|
-
const StatusIcon = config.icon;
|
|
45
|
-
|
|
46
|
-
const caseKeys = Object.keys(data.cases || {});
|
|
47
|
-
const displayCases = caseKeys.slice(0, 4); // Show up to 4 cases
|
|
48
|
-
const hasMore = caseKeys.length > 4;
|
|
49
|
-
|
|
50
|
-
return (
|
|
51
|
-
<div
|
|
52
|
-
className={`control-flow-node switch-node p-0 ${selected ? 'selected' : ''} ${status === 'running' ? 'running' : ''}`}
|
|
53
|
-
style={{
|
|
54
|
-
background: 'linear-gradient(135deg, #a855f7 0%, #ec4899 100%)',
|
|
55
|
-
}}
|
|
56
|
-
>
|
|
57
|
-
{/* Input handle */}
|
|
58
|
-
<Handle
|
|
59
|
-
type="target"
|
|
60
|
-
position={Position.Top}
|
|
61
|
-
className="!w-3 !h-3 !bg-primary !border-2 !border-node-bg"
|
|
62
|
-
/>
|
|
63
|
-
|
|
64
|
-
{/* Node header */}
|
|
65
|
-
<div className="flex items-center gap-3 p-3 border-b border-white/20">
|
|
66
|
-
<div className="w-8 h-8 rounded-lg bg-white/20 flex items-center justify-center">
|
|
67
|
-
<GitFork className="w-5 h-5 text-white" />
|
|
68
|
-
</div>
|
|
69
|
-
<div className="flex-1 min-w-0">
|
|
70
|
-
<div className="text-sm font-medium text-white">
|
|
71
|
-
{data.name || 'Switch'}
|
|
72
|
-
</div>
|
|
73
|
-
<div className="text-xs text-white/70">Multi-Branch Router</div>
|
|
74
|
-
</div>
|
|
75
|
-
<div
|
|
76
|
-
className={`w-6 h-6 rounded-full ${config.bgColor} flex items-center justify-center`}
|
|
77
|
-
>
|
|
78
|
-
<StatusIcon
|
|
79
|
-
className={`w-4 h-4 ${config.color} ${config.animate ? 'animate-pulse' : ''}`}
|
|
80
|
-
/>
|
|
81
|
-
</div>
|
|
82
|
-
</div>
|
|
83
|
-
|
|
84
|
-
{/* Node body */}
|
|
85
|
-
<div className="p-3 bg-white/10">
|
|
86
|
-
<div className="text-xs text-white/90 mb-3">
|
|
87
|
-
<span className="text-white/60">Expression:</span>{' '}
|
|
88
|
-
<span className="font-mono">{data.expression || 'Not set'}</span>
|
|
89
|
-
</div>
|
|
90
|
-
|
|
91
|
-
{/* Case list */}
|
|
92
|
-
<div className="space-y-2">
|
|
93
|
-
<div className="text-xs text-white/70 font-medium mb-1">Cases:</div>
|
|
94
|
-
{displayCases.map((caseKey, index) => {
|
|
95
|
-
const isActive = data.activeCase === caseKey;
|
|
96
|
-
const handlePosition = ((index + 1) / (displayCases.length + (data.hasDefault ? 2 : 1))) * 100;
|
|
97
|
-
|
|
98
|
-
return (
|
|
99
|
-
<div key={caseKey} className="relative">
|
|
100
|
-
<div
|
|
101
|
-
className={`text-xs px-2 py-1.5 rounded font-medium transition-colors ${
|
|
102
|
-
isActive
|
|
103
|
-
? 'bg-purple-500/30 text-purple-200 ring-1 ring-purple-400/50'
|
|
104
|
-
: 'bg-white/5 text-white/70'
|
|
105
|
-
}`}
|
|
106
|
-
>
|
|
107
|
-
{caseKey}
|
|
108
|
-
</div>
|
|
109
|
-
{/* Output handle for this case */}
|
|
110
|
-
<Handle
|
|
111
|
-
type="source"
|
|
112
|
-
position={Position.Bottom}
|
|
113
|
-
id={`case-${caseKey}`}
|
|
114
|
-
style={{ left: `${handlePosition}%` }}
|
|
115
|
-
className="!w-2.5 !h-2.5 !bg-purple-400 !border-2 !border-node-bg"
|
|
116
|
-
/>
|
|
117
|
-
</div>
|
|
118
|
-
);
|
|
119
|
-
})}
|
|
120
|
-
|
|
121
|
-
{hasMore && (
|
|
122
|
-
<div className="text-xs px-2 py-1 rounded bg-white/5 text-white/60 text-center">
|
|
123
|
-
+{caseKeys.length - 4} more cases
|
|
124
|
-
</div>
|
|
125
|
-
)}
|
|
126
|
-
|
|
127
|
-
{/* Default case */}
|
|
128
|
-
{data.hasDefault && (
|
|
129
|
-
<div className="relative">
|
|
130
|
-
<div
|
|
131
|
-
className={`text-xs px-2 py-1.5 rounded font-medium transition-colors ${
|
|
132
|
-
data.activeCase === 'default'
|
|
133
|
-
? 'bg-gray-500/30 text-gray-200 ring-1 ring-gray-400/50'
|
|
134
|
-
: 'bg-white/5 text-white/70'
|
|
135
|
-
}`}
|
|
136
|
-
>
|
|
137
|
-
default
|
|
138
|
-
</div>
|
|
139
|
-
{/* Output handle for default */}
|
|
140
|
-
<Handle
|
|
141
|
-
type="source"
|
|
142
|
-
position={Position.Bottom}
|
|
143
|
-
id="case-default"
|
|
144
|
-
style={{ left: '95%' }}
|
|
145
|
-
className="!w-2.5 !h-2.5 !bg-gray-400 !border-2 !border-node-bg"
|
|
146
|
-
/>
|
|
147
|
-
</div>
|
|
148
|
-
)}
|
|
149
|
-
</div>
|
|
150
|
-
|
|
151
|
-
{/* Case count */}
|
|
152
|
-
<div className="mt-3 text-xs text-white/50 flex items-center gap-2">
|
|
153
|
-
<span>ℹ️</span>
|
|
154
|
-
<span>
|
|
155
|
-
{caseKeys.length} case{caseKeys.length !== 1 ? 's' : ''}
|
|
156
|
-
{data.hasDefault ? ' + default' : ''}
|
|
157
|
-
</span>
|
|
158
|
-
</div>
|
|
159
|
-
</div>
|
|
160
|
-
</div>
|
|
161
|
-
);
|
|
162
|
-
}
|
|
163
|
-
|
|
164
|
-
export const SwitchNode = memo(SwitchNodeComponent);
|
|
@@ -1,189 +0,0 @@
|
|
|
1
|
-
import {
|
|
2
|
-
Plus,
|
|
3
|
-
Play,
|
|
4
|
-
Pause,
|
|
5
|
-
Layout,
|
|
6
|
-
ZoomIn,
|
|
7
|
-
ZoomOut,
|
|
8
|
-
Maximize,
|
|
9
|
-
Save,
|
|
10
|
-
Undo,
|
|
11
|
-
Redo,
|
|
12
|
-
Copy,
|
|
13
|
-
Trash2,
|
|
14
|
-
} from 'lucide-react';
|
|
15
|
-
import { useCanvas } from '../../hooks/useCanvas';
|
|
16
|
-
import { useEditorStore } from '../../stores/editorStore';
|
|
17
|
-
import { useReactFlow } from '@xyflow/react';
|
|
18
|
-
|
|
19
|
-
interface ToolbarProps {
|
|
20
|
-
onAddStep: () => void;
|
|
21
|
-
onExecute?: () => void;
|
|
22
|
-
onSave?: () => void;
|
|
23
|
-
isExecuting?: boolean;
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
export function Toolbar({
|
|
27
|
-
onAddStep,
|
|
28
|
-
onExecute,
|
|
29
|
-
onSave,
|
|
30
|
-
isExecuting = false,
|
|
31
|
-
}: ToolbarProps) {
|
|
32
|
-
const { autoLayout, fitView, selectedNodes, deleteSelected, duplicateSelected } =
|
|
33
|
-
useCanvas();
|
|
34
|
-
const { undo, redo, undoStack, redoStack } = useEditorStore();
|
|
35
|
-
const { zoomIn, zoomOut } = useReactFlow();
|
|
36
|
-
|
|
37
|
-
const canUndo = undoStack.length > 0;
|
|
38
|
-
const canRedo = redoStack.length > 0;
|
|
39
|
-
const hasSelection = selectedNodes.length > 0;
|
|
40
|
-
|
|
41
|
-
return (
|
|
42
|
-
<div className="absolute top-4 left-1/2 -translate-x-1/2 z-10 flex items-center gap-1 px-2 py-1.5 bg-panel-bg/95 backdrop-blur border border-node-border rounded-lg shadow-lg">
|
|
43
|
-
{/* Add Step */}
|
|
44
|
-
<ToolbarButton
|
|
45
|
-
icon={<Plus className="w-4 h-4" />}
|
|
46
|
-
label="Add Step"
|
|
47
|
-
onClick={onAddStep}
|
|
48
|
-
shortcut="N"
|
|
49
|
-
/>
|
|
50
|
-
|
|
51
|
-
<ToolbarDivider />
|
|
52
|
-
|
|
53
|
-
{/* Undo/Redo */}
|
|
54
|
-
<ToolbarButton
|
|
55
|
-
icon={<Undo className="w-4 h-4" />}
|
|
56
|
-
label="Undo"
|
|
57
|
-
onClick={() => undo()}
|
|
58
|
-
disabled={!canUndo}
|
|
59
|
-
shortcut="⌘Z"
|
|
60
|
-
/>
|
|
61
|
-
<ToolbarButton
|
|
62
|
-
icon={<Redo className="w-4 h-4" />}
|
|
63
|
-
label="Redo"
|
|
64
|
-
onClick={() => redo()}
|
|
65
|
-
disabled={!canRedo}
|
|
66
|
-
shortcut="⌘⇧Z"
|
|
67
|
-
/>
|
|
68
|
-
|
|
69
|
-
<ToolbarDivider />
|
|
70
|
-
|
|
71
|
-
{/* Selection actions */}
|
|
72
|
-
<ToolbarButton
|
|
73
|
-
icon={<Copy className="w-4 h-4" />}
|
|
74
|
-
label="Duplicate"
|
|
75
|
-
onClick={duplicateSelected}
|
|
76
|
-
disabled={!hasSelection}
|
|
77
|
-
shortcut="⌘D"
|
|
78
|
-
/>
|
|
79
|
-
<ToolbarButton
|
|
80
|
-
icon={<Trash2 className="w-4 h-4" />}
|
|
81
|
-
label="Delete"
|
|
82
|
-
onClick={deleteSelected}
|
|
83
|
-
disabled={!hasSelection}
|
|
84
|
-
shortcut="⌫"
|
|
85
|
-
/>
|
|
86
|
-
|
|
87
|
-
<ToolbarDivider />
|
|
88
|
-
|
|
89
|
-
{/* Layout & Zoom */}
|
|
90
|
-
<ToolbarButton
|
|
91
|
-
icon={<Layout className="w-4 h-4" />}
|
|
92
|
-
label="Auto Layout"
|
|
93
|
-
onClick={autoLayout}
|
|
94
|
-
shortcut="⌘L"
|
|
95
|
-
/>
|
|
96
|
-
<ToolbarButton
|
|
97
|
-
icon={<ZoomIn className="w-4 h-4" />}
|
|
98
|
-
label="Zoom In"
|
|
99
|
-
onClick={() => zoomIn()}
|
|
100
|
-
shortcut="⌘+"
|
|
101
|
-
/>
|
|
102
|
-
<ToolbarButton
|
|
103
|
-
icon={<ZoomOut className="w-4 h-4" />}
|
|
104
|
-
label="Zoom Out"
|
|
105
|
-
onClick={() => zoomOut()}
|
|
106
|
-
shortcut="⌘-"
|
|
107
|
-
/>
|
|
108
|
-
<ToolbarButton
|
|
109
|
-
icon={<Maximize className="w-4 h-4" />}
|
|
110
|
-
label="Fit View"
|
|
111
|
-
onClick={fitView}
|
|
112
|
-
shortcut="⌘0"
|
|
113
|
-
/>
|
|
114
|
-
|
|
115
|
-
<ToolbarDivider />
|
|
116
|
-
|
|
117
|
-
{/* Execute */}
|
|
118
|
-
{onExecute && (
|
|
119
|
-
<ToolbarButton
|
|
120
|
-
icon={
|
|
121
|
-
isExecuting ? (
|
|
122
|
-
<Pause className="w-4 h-4" />
|
|
123
|
-
) : (
|
|
124
|
-
<Play className="w-4 h-4" />
|
|
125
|
-
)
|
|
126
|
-
}
|
|
127
|
-
label={isExecuting ? 'Stop' : 'Execute'}
|
|
128
|
-
onClick={onExecute}
|
|
129
|
-
variant={isExecuting ? 'destructive' : 'primary'}
|
|
130
|
-
shortcut="⌘⏎"
|
|
131
|
-
/>
|
|
132
|
-
)}
|
|
133
|
-
|
|
134
|
-
{/* Save */}
|
|
135
|
-
{onSave && (
|
|
136
|
-
<ToolbarButton
|
|
137
|
-
icon={<Save className="w-4 h-4" />}
|
|
138
|
-
label="Save"
|
|
139
|
-
onClick={onSave}
|
|
140
|
-
shortcut="⌘S"
|
|
141
|
-
/>
|
|
142
|
-
)}
|
|
143
|
-
</div>
|
|
144
|
-
);
|
|
145
|
-
}
|
|
146
|
-
|
|
147
|
-
interface ToolbarButtonProps {
|
|
148
|
-
icon: React.ReactNode;
|
|
149
|
-
label: string;
|
|
150
|
-
onClick: () => void;
|
|
151
|
-
disabled?: boolean;
|
|
152
|
-
shortcut?: string;
|
|
153
|
-
variant?: 'default' | 'primary' | 'destructive';
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
function ToolbarButton({
|
|
157
|
-
icon,
|
|
158
|
-
label,
|
|
159
|
-
onClick,
|
|
160
|
-
disabled,
|
|
161
|
-
shortcut,
|
|
162
|
-
variant = 'default',
|
|
163
|
-
}: ToolbarButtonProps) {
|
|
164
|
-
const variantClasses = {
|
|
165
|
-
default: 'text-gray-300 hover:text-white hover:bg-white/10',
|
|
166
|
-
primary: 'text-primary hover:text-primary-light hover:bg-primary/10',
|
|
167
|
-
destructive: 'text-error hover:text-error hover:bg-error/10',
|
|
168
|
-
};
|
|
169
|
-
|
|
170
|
-
return (
|
|
171
|
-
<button
|
|
172
|
-
onClick={onClick}
|
|
173
|
-
disabled={disabled}
|
|
174
|
-
className={`relative group p-2 rounded transition-colors disabled:opacity-40 disabled:cursor-not-allowed ${variantClasses[variant]}`}
|
|
175
|
-
title={`${label}${shortcut ? ` (${shortcut})` : ''}`}
|
|
176
|
-
>
|
|
177
|
-
{icon}
|
|
178
|
-
{/* Tooltip */}
|
|
179
|
-
<div className="absolute top-full left-1/2 -translate-x-1/2 mt-2 px-2 py-1 bg-black/90 rounded text-xs text-white whitespace-nowrap opacity-0 group-hover:opacity-100 transition-opacity pointer-events-none">
|
|
180
|
-
{label}
|
|
181
|
-
{shortcut && <span className="ml-2 text-gray-400">{shortcut}</span>}
|
|
182
|
-
</div>
|
|
183
|
-
</button>
|
|
184
|
-
);
|
|
185
|
-
}
|
|
186
|
-
|
|
187
|
-
function ToolbarDivider() {
|
|
188
|
-
return <div className="w-px h-6 bg-node-border mx-1" />;
|
|
189
|
-
}
|