@foresthubai/workflow-builder 0.3.0 → 0.4.0
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/LICENSE +661 -661
- package/NOTICE +16 -16
- package/README.md +110 -93
- package/dist/components/ui/command.d.ts +2 -2
- package/dist/components/ui/input.d.ts +1 -1
- package/dist/components/ui/resizable.d.ts +1 -1
- package/dist/components/ui/textarea.d.ts +1 -1
- package/dist/graph/BaseNode.js +10 -10
- package/dist/graph/reactFlowRegistry.d.ts.map +1 -1
- package/dist/lib/utils.d.ts +3 -0
- package/dist/lib/utils.d.ts.map +1 -0
- package/dist/lib/utils.js +6 -0
- package/dist/lib/utils.js.map +1 -0
- package/dist/toolbars/CanvasTabsToolbar.d.ts +11 -0
- package/dist/toolbars/CanvasTabsToolbar.d.ts.map +1 -0
- package/dist/toolbars/CanvasTabsToolbar.js +101 -0
- package/dist/toolbars/CanvasTabsToolbar.js.map +1 -0
- package/package.json +2 -2
- package/src/BuilderLayout.tsx +345 -345
- package/src/Canvas.tsx +261 -261
- package/src/CanvasEditor.tsx +142 -142
- package/src/CanvasTabsToolbar.tsx +176 -176
- package/src/RightConfigPanel.tsx +266 -266
- package/src/WorkflowBuilder.tsx +412 -412
- package/src/cn.ts +6 -6
- package/src/components/ui/add-button.tsx +39 -39
- package/src/components/ui/alert-dialog.tsx +141 -141
- package/src/components/ui/alert.tsx +59 -59
- package/src/components/ui/badge.tsx +36 -36
- package/src/components/ui/button.tsx +85 -85
- package/src/components/ui/card.tsx +79 -79
- package/src/components/ui/checkbox.tsx +28 -28
- package/src/components/ui/collapsible.tsx +9 -9
- package/src/components/ui/command.tsx +153 -153
- package/src/components/ui/delete-button.tsx +23 -23
- package/src/components/ui/dialog.tsx +125 -125
- package/src/components/ui/dropdown-menu.tsx +198 -198
- package/src/components/ui/input.tsx +55 -55
- package/src/components/ui/label.tsx +24 -24
- package/src/components/ui/readonly-banner.tsx +15 -15
- package/src/components/ui/resizable.tsx +43 -43
- package/src/components/ui/scroll-area.tsx +102 -102
- package/src/components/ui/select.tsx +160 -160
- package/src/components/ui/separator.tsx +29 -29
- package/src/components/ui/switch.tsx +27 -27
- package/src/components/ui/textarea.tsx +51 -51
- package/src/components/ui/toast.tsx +127 -127
- package/src/components/ui/toaster.tsx +33 -33
- package/src/components/ui/toggle-group.tsx +59 -59
- package/src/components/ui/toggle.tsx +43 -43
- package/src/components/ui/tooltip.tsx +32 -32
- package/src/dialogs/NodePickerDialog.tsx +84 -84
- package/src/dialogs/ValidationDialog.tsx +184 -184
- package/src/graph/BaseNode.tsx +557 -557
- package/src/graph/CustomEdge.tsx +185 -185
- package/src/graph/CustomNode.tsx +16 -16
- package/src/graph/FunctionCallNode.tsx +30 -30
- package/src/graph/PortHandle.tsx +189 -189
- package/src/graph/reactFlowRegistry.ts +26 -26
- package/src/hooks/use-toast.ts +125 -125
- package/src/hooks/useAvailableVariables.ts +20 -20
- package/src/hooks/useCanvasHistory.ts +22 -22
- package/src/hooks/useCanvasTabs.ts +168 -168
- package/src/hooks/useFunctionDiagnosticsSync.ts +40 -40
- package/src/hooks/useFunctionRegistry.ts +26 -26
- package/src/hooks/useFunctions.ts +44 -44
- package/src/hooks/useGraph.ts +161 -161
- package/src/hooks/useNodeDefinitions.ts +82 -82
- package/src/hooks/useParamErrors.ts +26 -26
- package/src/hooks/useResolvedTheme.ts +30 -30
- package/src/hooks/useResourceDiagnosticsSync.ts +58 -58
- package/src/hooks/useSuppressThemeTransition.ts +79 -79
- package/src/hooks/useWorkflowSerialization.ts +127 -127
- package/src/i18n/index.ts +53 -53
- package/src/i18n/locales/de.json +501 -501
- package/src/i18n/locales/en.json +557 -557
- package/src/index.ts +27 -27
- package/src/inputs/ExpressionInput.tsx +297 -297
- package/src/inputs/ParameterEditor.tsx +515 -515
- package/src/inputs/PortSection.tsx +144 -144
- package/src/panels/BuilderSidebar.tsx +301 -301
- package/src/panels/ChannelConfigPanel.tsx +49 -49
- package/src/panels/ChannelsPanel.tsx +28 -28
- package/src/panels/DebugConsolePanel.tsx +73 -73
- package/src/panels/DebugContextPanel.tsx +77 -77
- package/src/panels/DebugExternalIOPanel.tsx +180 -180
- package/src/panels/DiagnosticsPanel.tsx +170 -170
- package/src/panels/EdgeConfigPanel.tsx +104 -104
- package/src/panels/FunctionConfigPanel.tsx +179 -179
- package/src/panels/FunctionListPanel.tsx +45 -45
- package/src/panels/MemoryConfigPanel.tsx +55 -55
- package/src/panels/MemoryPanel.tsx +40 -40
- package/src/panels/ModelConfigPanel.tsx +41 -41
- package/src/panels/ModelsPanel.tsx +36 -36
- package/src/panels/NodeConfigPanel.tsx +630 -630
- package/src/panels/NodeLibrary.tsx +288 -288
- package/src/panels/ResourceConfigPanel.tsx +132 -132
- package/src/panels/ResourceListPanel.tsx +113 -113
- package/src/panels/VariableConfigPanel.tsx +161 -161
- package/src/panels/VariablesPanel.tsx +145 -145
- package/src/stores/canvasStore.test.ts +44 -44
- package/src/stores/canvasStore.ts +245 -245
- package/src/stores/debugStore.ts +74 -74
- package/src/stores/diagnosticsStore.ts +130 -130
- package/src/stores/editorStore.ts +202 -202
- package/src/styles/index.css +526 -526
- package/src/utils/categoryConstants.ts +26 -26
- package/src/utils/channelOperations.ts +86 -86
- package/src/utils/connectionRules.ts +137 -137
- package/src/utils/functionOperations.ts +179 -179
- package/src/utils/graphOperations.ts +550 -550
- package/src/utils/history.ts +207 -207
- package/src/utils/memoryOperations.ts +57 -57
- package/src/utils/migrateFunctionNodes.ts +107 -107
- package/src/utils/modelOperations.ts +55 -55
- package/src/utils/paramDisplay.ts +71 -71
- package/src/utils/resourceHelpers.ts +32 -32
- package/src/utils/translation.ts +28 -28
- package/src/utils/variableOperations.ts +75 -75
- package/tailwind-preset.ts +166 -166
|
@@ -1,301 +1,301 @@
|
|
|
1
|
-
import { Button } from "../components/ui/button";
|
|
2
|
-
import { ScrollArea } from "../components/ui/scroll-area";
|
|
3
|
-
import { Tooltip, TooltipContent, TooltipTrigger } from "../components/ui/tooltip";
|
|
4
|
-
import { cn } from "../cn";
|
|
5
|
-
import { NodeCategory, NodeDefinition } from "@foresthubai/workflow-core/node";
|
|
6
|
-
import { useMemo } from "react";
|
|
7
|
-
import { Blocks, BrainCircuit, Braces, Bug, Cpu, Database, TriangleAlert, Variable, X } from "lucide-react";
|
|
8
|
-
import { useTranslation } from "react-i18next";
|
|
9
|
-
import { DiagnosticsPanel } from "./DiagnosticsPanel";
|
|
10
|
-
import { FunctionListPanel } from "./FunctionListPanel";
|
|
11
|
-
import NodeLibrary from "./NodeLibrary";
|
|
12
|
-
import { ChannelsPanel } from "./ChannelsPanel";
|
|
13
|
-
import { MemoryPanel } from "./MemoryPanel";
|
|
14
|
-
import { ModelsPanel } from "./ModelsPanel";
|
|
15
|
-
import { VariablesPanel } from "./VariablesPanel";
|
|
16
|
-
import { useDiagnosticsStore } from "../stores/diagnosticsStore";
|
|
17
|
-
import { DebugContextPanel } from "./DebugContextPanel";
|
|
18
|
-
import type { FunctionDeclaration } from "@foresthubai/workflow-core/function";
|
|
19
|
-
|
|
20
|
-
export type SidebarTab =
|
|
21
|
-
| "nodes"
|
|
22
|
-
| "variables"
|
|
23
|
-
| "channels"
|
|
24
|
-
| "memory"
|
|
25
|
-
| "models"
|
|
26
|
-
| "functions"
|
|
27
|
-
| "diagnostics"
|
|
28
|
-
| "debug-context"
|
|
29
|
-
| null;
|
|
30
|
-
|
|
31
|
-
interface BuilderSidebarProps {
|
|
32
|
-
canvasId: string;
|
|
33
|
-
activeTab: SidebarTab;
|
|
34
|
-
onTabChange: (tab: SidebarTab) => void;
|
|
35
|
-
onAddNode: (nodeType: NodeDefinition, position?: { x: number; y: number }) => void;
|
|
36
|
-
nodeDefinitions: NodeDefinition[];
|
|
37
|
-
getAllCategories: () => NodeCategory[];
|
|
38
|
-
onSelectNode: (nodeId: string) => void;
|
|
39
|
-
onSelectEdge: (edgeId: string) => void;
|
|
40
|
-
isFunctionCanvas: boolean;
|
|
41
|
-
// Function management
|
|
42
|
-
functions: FunctionDeclaration[];
|
|
43
|
-
onOpenFunction: (functionId: string) => void;
|
|
44
|
-
onCreateFunction: () => string;
|
|
45
|
-
// Debug mode
|
|
46
|
-
isDebugMode?: boolean;
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
export const BuilderSidebar = ({
|
|
50
|
-
canvasId,
|
|
51
|
-
activeTab,
|
|
52
|
-
onTabChange,
|
|
53
|
-
onAddNode,
|
|
54
|
-
nodeDefinitions,
|
|
55
|
-
getAllCategories,
|
|
56
|
-
onSelectNode,
|
|
57
|
-
onSelectEdge,
|
|
58
|
-
isFunctionCanvas,
|
|
59
|
-
functions,
|
|
60
|
-
onOpenFunction,
|
|
61
|
-
onCreateFunction,
|
|
62
|
-
isDebugMode,
|
|
63
|
-
}: BuilderSidebarProps) => {
|
|
64
|
-
const { t } = useTranslation();
|
|
65
|
-
|
|
66
|
-
const staticTabs = useMemo(
|
|
67
|
-
() => [
|
|
68
|
-
{ id: "nodes" as const, icon: Blocks, label: t("nodeLibrary") },
|
|
69
|
-
{ id: "variables" as const, icon: Variable, label: t("variables") },
|
|
70
|
-
{ id: "channels" as const, icon: Cpu, label: t("channels") },
|
|
71
|
-
{ id: "memory" as const, icon: Database, label: t("memoryFiles", "Memory") },
|
|
72
|
-
{ id: "models" as const, icon: BrainCircuit, label: t("models", "AI Models") },
|
|
73
|
-
{ id: "functions" as const, icon: Braces, label: t("functions") },
|
|
74
|
-
{ id: "diagnostics" as const, icon: TriangleAlert, label: t("diagnostics") },
|
|
75
|
-
],
|
|
76
|
-
[t],
|
|
77
|
-
);
|
|
78
|
-
|
|
79
|
-
const debugTabs = useMemo(() => [{ id: "debug-context" as const, icon: Bug, label: t("debug.context") }], [t]);
|
|
80
|
-
|
|
81
|
-
// Current-canvas diagnostics counts for the diagnostics tab — node + edge only.
|
|
82
|
-
const totalErrors = useDiagnosticsStore((s) => {
|
|
83
|
-
let count = 0;
|
|
84
|
-
for (const diags of Object.values(s.byNodeId)) for (const d of diags) if (d.severity === "error") count++;
|
|
85
|
-
for (const diags of Object.values(s.byEdgeId)) for (const d of diags) if (d.severity === "error") count++;
|
|
86
|
-
return count;
|
|
87
|
-
});
|
|
88
|
-
const totalWarnings = useDiagnosticsStore((s) => {
|
|
89
|
-
let count = 0;
|
|
90
|
-
for (const diags of Object.values(s.byNodeId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
91
|
-
for (const diags of Object.values(s.byEdgeId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
92
|
-
return count;
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
// Channels are project-scoped and own their own tab/badge — counts here
|
|
96
|
-
// drive the "channels" tab icon color + badge (mirrors the diagnostics tab).
|
|
97
|
-
const channelErrors = useDiagnosticsStore((s) => {
|
|
98
|
-
let count = 0;
|
|
99
|
-
for (const diags of Object.values(s.byChannelId)) for (const d of diags) if (d.severity === "error") count++;
|
|
100
|
-
return count;
|
|
101
|
-
});
|
|
102
|
-
const channelWarnings = useDiagnosticsStore((s) => {
|
|
103
|
-
let count = 0;
|
|
104
|
-
for (const diags of Object.values(s.byChannelId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
105
|
-
return count;
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
// Memory primitives are project-scoped too — counts drive the "memory" tab
|
|
109
|
-
// icon color + badge (mirrors the channels tab).
|
|
110
|
-
const memoryErrors = useDiagnosticsStore((s) => {
|
|
111
|
-
let count = 0;
|
|
112
|
-
for (const diags of Object.values(s.byMemoryId)) for (const d of diags) if (d.severity === "error") count++;
|
|
113
|
-
return count;
|
|
114
|
-
});
|
|
115
|
-
const memoryWarnings = useDiagnosticsStore((s) => {
|
|
116
|
-
let count = 0;
|
|
117
|
-
for (const diags of Object.values(s.byMemoryId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
118
|
-
return count;
|
|
119
|
-
});
|
|
120
|
-
|
|
121
|
-
// Declared models are project-scoped too — counts drive the "models" tab badge.
|
|
122
|
-
const modelErrors = useDiagnosticsStore((s) => {
|
|
123
|
-
let count = 0;
|
|
124
|
-
for (const diags of Object.values(s.byModelId)) for (const d of diags) if (d.severity === "error") count++;
|
|
125
|
-
return count;
|
|
126
|
-
});
|
|
127
|
-
const modelWarnings = useDiagnosticsStore((s) => {
|
|
128
|
-
let count = 0;
|
|
129
|
-
for (const diags of Object.values(s.byModelId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
130
|
-
return count;
|
|
131
|
-
});
|
|
132
|
-
|
|
133
|
-
// Functions are project-scoped too — counts drive the "functions" tab badge.
|
|
134
|
-
const functionErrors = useDiagnosticsStore((s) => {
|
|
135
|
-
let count = 0;
|
|
136
|
-
for (const diags of Object.values(s.byFunctionId)) for (const d of diags) if (d.severity === "error") count++;
|
|
137
|
-
return count;
|
|
138
|
-
});
|
|
139
|
-
const functionWarnings = useDiagnosticsStore((s) => {
|
|
140
|
-
let count = 0;
|
|
141
|
-
for (const diags of Object.values(s.byFunctionId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
142
|
-
return count;
|
|
143
|
-
});
|
|
144
|
-
|
|
145
|
-
const tabs = useMemo(() => (isDebugMode ? debugTabs : staticTabs), [isDebugMode, debugTabs, staticTabs]);
|
|
146
|
-
|
|
147
|
-
const handleTabClick = (tabId: SidebarTab) => {
|
|
148
|
-
onTabChange(activeTab === tabId ? null : tabId);
|
|
149
|
-
};
|
|
150
|
-
|
|
151
|
-
const renderTabContent = () => {
|
|
152
|
-
switch (activeTab) {
|
|
153
|
-
case "nodes":
|
|
154
|
-
return (
|
|
155
|
-
<NodeLibrary
|
|
156
|
-
onAddNode={onAddNode}
|
|
157
|
-
nodeDefinitions={nodeDefinitions}
|
|
158
|
-
getAllCategories={getAllCategories}
|
|
159
|
-
functions={functions}
|
|
160
|
-
isFunctionCanvas={!!isFunctionCanvas}
|
|
161
|
-
/>
|
|
162
|
-
);
|
|
163
|
-
case "variables":
|
|
164
|
-
return <VariablesPanel canvasId={canvasId} onSelectNode={onSelectNode} />;
|
|
165
|
-
case "channels":
|
|
166
|
-
return <ChannelsPanel />;
|
|
167
|
-
case "memory":
|
|
168
|
-
return <MemoryPanel />;
|
|
169
|
-
case "models":
|
|
170
|
-
return <ModelsPanel />;
|
|
171
|
-
case "functions":
|
|
172
|
-
return <FunctionListPanel onOpenFunction={onOpenFunction} onCreateFunction={onCreateFunction} />;
|
|
173
|
-
case "diagnostics":
|
|
174
|
-
return <DiagnosticsPanel canvasId={canvasId} onSelectNode={onSelectNode} onSelectEdge={onSelectEdge} />;
|
|
175
|
-
case "debug-context":
|
|
176
|
-
return <DebugContextPanel />;
|
|
177
|
-
default:
|
|
178
|
-
return null;
|
|
179
|
-
}
|
|
180
|
-
};
|
|
181
|
-
|
|
182
|
-
const getTabLabel = (tabId: SidebarTab) => {
|
|
183
|
-
const tab = tabs.find((tab) => tab.id === tabId);
|
|
184
|
-
return tab?.label ?? "";
|
|
185
|
-
};
|
|
186
|
-
|
|
187
|
-
return (
|
|
188
|
-
<div className="flex h-full">
|
|
189
|
-
{/* Icon Rail - Always visible */}
|
|
190
|
-
<div className="w-14 bg-card border-r border-border/50 flex flex-col items-center py-3 gap-1 shrink-0">
|
|
191
|
-
{tabs.map((tab) => {
|
|
192
|
-
const Icon = tab.icon;
|
|
193
|
-
const isActive = activeTab === tab.id;
|
|
194
|
-
|
|
195
|
-
// Per-tab error/warning counts for icon coloring + badge.
|
|
196
|
-
// Diagnostics tab covers nodes+edges; channels tab covers channels.
|
|
197
|
-
let tabErrors = 0;
|
|
198
|
-
let tabWarnings = 0;
|
|
199
|
-
if (tab.id === "diagnostics") {
|
|
200
|
-
tabErrors = totalErrors;
|
|
201
|
-
tabWarnings = totalWarnings;
|
|
202
|
-
} else if (tab.id === "channels") {
|
|
203
|
-
tabErrors = channelErrors;
|
|
204
|
-
tabWarnings = channelWarnings;
|
|
205
|
-
} else if (tab.id === "memory") {
|
|
206
|
-
tabErrors = memoryErrors;
|
|
207
|
-
tabWarnings = memoryWarnings;
|
|
208
|
-
} else if (tab.id === "models") {
|
|
209
|
-
tabErrors = modelErrors;
|
|
210
|
-
tabWarnings = modelWarnings;
|
|
211
|
-
} else if (tab.id === "functions") {
|
|
212
|
-
tabErrors = functionErrors;
|
|
213
|
-
tabWarnings = functionWarnings;
|
|
214
|
-
}
|
|
215
|
-
const tabIssueCount = tabErrors + tabWarnings;
|
|
216
|
-
const showBadge = tabIssueCount > 0;
|
|
217
|
-
|
|
218
|
-
const iconColorClass =
|
|
219
|
-
showBadge && !isActive
|
|
220
|
-
? tabErrors > 0
|
|
221
|
-
? "text-destructive hover:text-destructive hover:bg-destructive/10"
|
|
222
|
-
: "text-warning hover:text-warning hover:bg-warning/10"
|
|
223
|
-
: undefined;
|
|
224
|
-
|
|
225
|
-
return (
|
|
226
|
-
<Tooltip key={tab.id} delayDuration={300}>
|
|
227
|
-
<TooltipTrigger asChild>
|
|
228
|
-
<Button
|
|
229
|
-
variant="ghost"
|
|
230
|
-
size="icon"
|
|
231
|
-
onClick={(e) => {
|
|
232
|
-
// Drop focus after a mouse click so a follow-up Enter doesn't
|
|
233
|
-
// re-fire (toggle) this tab. Keyboard activation (detail 0) keeps focus.
|
|
234
|
-
if (e.detail !== 0) e.currentTarget.blur();
|
|
235
|
-
handleTabClick(tab.id);
|
|
236
|
-
}}
|
|
237
|
-
className={cn(
|
|
238
|
-
"w-10 h-10 transition-all duration-200 relative",
|
|
239
|
-
isActive
|
|
240
|
-
? "bg-accent text-primary shadow-sm"
|
|
241
|
-
: (iconColorClass ?? "text-muted-foreground hover:text-foreground hover:bg-accent/50"),
|
|
242
|
-
)}
|
|
243
|
-
>
|
|
244
|
-
<Icon className="w-5 h-5" />
|
|
245
|
-
{showBadge && (
|
|
246
|
-
<span
|
|
247
|
-
className={cn(
|
|
248
|
-
"absolute -top-0.5 -right-0.5 min-w-[16px] h-4 rounded-full text-[10px] font-bold flex items-center justify-center px-1 shadow-sm",
|
|
249
|
-
tabErrors > 0
|
|
250
|
-
? "bg-destructive text-destructive-foreground"
|
|
251
|
-
: "bg-warning text-warning-foreground",
|
|
252
|
-
)}
|
|
253
|
-
>
|
|
254
|
-
{tabIssueCount}
|
|
255
|
-
</span>
|
|
256
|
-
)}
|
|
257
|
-
</Button>
|
|
258
|
-
</TooltipTrigger>
|
|
259
|
-
<TooltipContent side="right" sideOffset={8}>
|
|
260
|
-
{tab.label}
|
|
261
|
-
</TooltipContent>
|
|
262
|
-
</Tooltip>
|
|
263
|
-
);
|
|
264
|
-
})}
|
|
265
|
-
</div>
|
|
266
|
-
|
|
267
|
-
{/* Content Panel - Slides in/out */}
|
|
268
|
-
<div
|
|
269
|
-
className={cn(
|
|
270
|
-
"bg-card/95 backdrop-blur-xl border-r border-border/50 transition-all duration-300 ease-in-out overflow-hidden flex flex-col",
|
|
271
|
-
activeTab ? "w-64 opacity-100" : "w-0 opacity-0",
|
|
272
|
-
)}
|
|
273
|
-
>
|
|
274
|
-
{activeTab && (
|
|
275
|
-
<>
|
|
276
|
-
{/* Panel Header */}
|
|
277
|
-
<div className="flex items-center justify-between p-3 border-b border-border/50 shrink-0">
|
|
278
|
-
<h3 className="font-semibold text-sm text-foreground">{getTabLabel(activeTab)}</h3>
|
|
279
|
-
<Button
|
|
280
|
-
variant="ghost"
|
|
281
|
-
size="icon"
|
|
282
|
-
onClick={() => onTabChange(null)}
|
|
283
|
-
className="w-7 h-7 text-muted-foreground hover:text-foreground"
|
|
284
|
-
>
|
|
285
|
-
<X className="w-4 h-4" />
|
|
286
|
-
</Button>
|
|
287
|
-
</div>
|
|
288
|
-
|
|
289
|
-
{/* Panel Content — ScrollArea overlays the scrollbar in the panel's
|
|
290
|
-
gutter so content width stays constant whether overflow is
|
|
291
|
-
present or not. Padding lives on the inner viewport because the
|
|
292
|
-
Root must clip cleanly for the absolute-positioned scrollbar. */}
|
|
293
|
-
<ScrollArea className="flex-1" viewportClassName="p-3">
|
|
294
|
-
{renderTabContent()}
|
|
295
|
-
</ScrollArea>
|
|
296
|
-
</>
|
|
297
|
-
)}
|
|
298
|
-
</div>
|
|
299
|
-
</div>
|
|
300
|
-
);
|
|
301
|
-
};
|
|
1
|
+
import { Button } from "../components/ui/button";
|
|
2
|
+
import { ScrollArea } from "../components/ui/scroll-area";
|
|
3
|
+
import { Tooltip, TooltipContent, TooltipTrigger } from "../components/ui/tooltip";
|
|
4
|
+
import { cn } from "../cn";
|
|
5
|
+
import { NodeCategory, NodeDefinition } from "@foresthubai/workflow-core/node";
|
|
6
|
+
import { useMemo } from "react";
|
|
7
|
+
import { Blocks, BrainCircuit, Braces, Bug, Cpu, Database, TriangleAlert, Variable, X } from "lucide-react";
|
|
8
|
+
import { useTranslation } from "react-i18next";
|
|
9
|
+
import { DiagnosticsPanel } from "./DiagnosticsPanel";
|
|
10
|
+
import { FunctionListPanel } from "./FunctionListPanel";
|
|
11
|
+
import NodeLibrary from "./NodeLibrary";
|
|
12
|
+
import { ChannelsPanel } from "./ChannelsPanel";
|
|
13
|
+
import { MemoryPanel } from "./MemoryPanel";
|
|
14
|
+
import { ModelsPanel } from "./ModelsPanel";
|
|
15
|
+
import { VariablesPanel } from "./VariablesPanel";
|
|
16
|
+
import { useDiagnosticsStore } from "../stores/diagnosticsStore";
|
|
17
|
+
import { DebugContextPanel } from "./DebugContextPanel";
|
|
18
|
+
import type { FunctionDeclaration } from "@foresthubai/workflow-core/function";
|
|
19
|
+
|
|
20
|
+
export type SidebarTab =
|
|
21
|
+
| "nodes"
|
|
22
|
+
| "variables"
|
|
23
|
+
| "channels"
|
|
24
|
+
| "memory"
|
|
25
|
+
| "models"
|
|
26
|
+
| "functions"
|
|
27
|
+
| "diagnostics"
|
|
28
|
+
| "debug-context"
|
|
29
|
+
| null;
|
|
30
|
+
|
|
31
|
+
interface BuilderSidebarProps {
|
|
32
|
+
canvasId: string;
|
|
33
|
+
activeTab: SidebarTab;
|
|
34
|
+
onTabChange: (tab: SidebarTab) => void;
|
|
35
|
+
onAddNode: (nodeType: NodeDefinition, position?: { x: number; y: number }) => void;
|
|
36
|
+
nodeDefinitions: NodeDefinition[];
|
|
37
|
+
getAllCategories: () => NodeCategory[];
|
|
38
|
+
onSelectNode: (nodeId: string) => void;
|
|
39
|
+
onSelectEdge: (edgeId: string) => void;
|
|
40
|
+
isFunctionCanvas: boolean;
|
|
41
|
+
// Function management
|
|
42
|
+
functions: FunctionDeclaration[];
|
|
43
|
+
onOpenFunction: (functionId: string) => void;
|
|
44
|
+
onCreateFunction: () => string;
|
|
45
|
+
// Debug mode
|
|
46
|
+
isDebugMode?: boolean;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
export const BuilderSidebar = ({
|
|
50
|
+
canvasId,
|
|
51
|
+
activeTab,
|
|
52
|
+
onTabChange,
|
|
53
|
+
onAddNode,
|
|
54
|
+
nodeDefinitions,
|
|
55
|
+
getAllCategories,
|
|
56
|
+
onSelectNode,
|
|
57
|
+
onSelectEdge,
|
|
58
|
+
isFunctionCanvas,
|
|
59
|
+
functions,
|
|
60
|
+
onOpenFunction,
|
|
61
|
+
onCreateFunction,
|
|
62
|
+
isDebugMode,
|
|
63
|
+
}: BuilderSidebarProps) => {
|
|
64
|
+
const { t } = useTranslation();
|
|
65
|
+
|
|
66
|
+
const staticTabs = useMemo(
|
|
67
|
+
() => [
|
|
68
|
+
{ id: "nodes" as const, icon: Blocks, label: t("nodeLibrary") },
|
|
69
|
+
{ id: "variables" as const, icon: Variable, label: t("variables") },
|
|
70
|
+
{ id: "channels" as const, icon: Cpu, label: t("channels") },
|
|
71
|
+
{ id: "memory" as const, icon: Database, label: t("memoryFiles", "Memory") },
|
|
72
|
+
{ id: "models" as const, icon: BrainCircuit, label: t("models", "AI Models") },
|
|
73
|
+
{ id: "functions" as const, icon: Braces, label: t("functions") },
|
|
74
|
+
{ id: "diagnostics" as const, icon: TriangleAlert, label: t("diagnostics") },
|
|
75
|
+
],
|
|
76
|
+
[t],
|
|
77
|
+
);
|
|
78
|
+
|
|
79
|
+
const debugTabs = useMemo(() => [{ id: "debug-context" as const, icon: Bug, label: t("debug.context") }], [t]);
|
|
80
|
+
|
|
81
|
+
// Current-canvas diagnostics counts for the diagnostics tab — node + edge only.
|
|
82
|
+
const totalErrors = useDiagnosticsStore((s) => {
|
|
83
|
+
let count = 0;
|
|
84
|
+
for (const diags of Object.values(s.byNodeId)) for (const d of diags) if (d.severity === "error") count++;
|
|
85
|
+
for (const diags of Object.values(s.byEdgeId)) for (const d of diags) if (d.severity === "error") count++;
|
|
86
|
+
return count;
|
|
87
|
+
});
|
|
88
|
+
const totalWarnings = useDiagnosticsStore((s) => {
|
|
89
|
+
let count = 0;
|
|
90
|
+
for (const diags of Object.values(s.byNodeId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
91
|
+
for (const diags of Object.values(s.byEdgeId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
92
|
+
return count;
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
// Channels are project-scoped and own their own tab/badge — counts here
|
|
96
|
+
// drive the "channels" tab icon color + badge (mirrors the diagnostics tab).
|
|
97
|
+
const channelErrors = useDiagnosticsStore((s) => {
|
|
98
|
+
let count = 0;
|
|
99
|
+
for (const diags of Object.values(s.byChannelId)) for (const d of diags) if (d.severity === "error") count++;
|
|
100
|
+
return count;
|
|
101
|
+
});
|
|
102
|
+
const channelWarnings = useDiagnosticsStore((s) => {
|
|
103
|
+
let count = 0;
|
|
104
|
+
for (const diags of Object.values(s.byChannelId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
105
|
+
return count;
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
// Memory primitives are project-scoped too — counts drive the "memory" tab
|
|
109
|
+
// icon color + badge (mirrors the channels tab).
|
|
110
|
+
const memoryErrors = useDiagnosticsStore((s) => {
|
|
111
|
+
let count = 0;
|
|
112
|
+
for (const diags of Object.values(s.byMemoryId)) for (const d of diags) if (d.severity === "error") count++;
|
|
113
|
+
return count;
|
|
114
|
+
});
|
|
115
|
+
const memoryWarnings = useDiagnosticsStore((s) => {
|
|
116
|
+
let count = 0;
|
|
117
|
+
for (const diags of Object.values(s.byMemoryId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
118
|
+
return count;
|
|
119
|
+
});
|
|
120
|
+
|
|
121
|
+
// Declared models are project-scoped too — counts drive the "models" tab badge.
|
|
122
|
+
const modelErrors = useDiagnosticsStore((s) => {
|
|
123
|
+
let count = 0;
|
|
124
|
+
for (const diags of Object.values(s.byModelId)) for (const d of diags) if (d.severity === "error") count++;
|
|
125
|
+
return count;
|
|
126
|
+
});
|
|
127
|
+
const modelWarnings = useDiagnosticsStore((s) => {
|
|
128
|
+
let count = 0;
|
|
129
|
+
for (const diags of Object.values(s.byModelId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
130
|
+
return count;
|
|
131
|
+
});
|
|
132
|
+
|
|
133
|
+
// Functions are project-scoped too — counts drive the "functions" tab badge.
|
|
134
|
+
const functionErrors = useDiagnosticsStore((s) => {
|
|
135
|
+
let count = 0;
|
|
136
|
+
for (const diags of Object.values(s.byFunctionId)) for (const d of diags) if (d.severity === "error") count++;
|
|
137
|
+
return count;
|
|
138
|
+
});
|
|
139
|
+
const functionWarnings = useDiagnosticsStore((s) => {
|
|
140
|
+
let count = 0;
|
|
141
|
+
for (const diags of Object.values(s.byFunctionId)) for (const d of diags) if (d.severity === "warning") count++;
|
|
142
|
+
return count;
|
|
143
|
+
});
|
|
144
|
+
|
|
145
|
+
const tabs = useMemo(() => (isDebugMode ? debugTabs : staticTabs), [isDebugMode, debugTabs, staticTabs]);
|
|
146
|
+
|
|
147
|
+
const handleTabClick = (tabId: SidebarTab) => {
|
|
148
|
+
onTabChange(activeTab === tabId ? null : tabId);
|
|
149
|
+
};
|
|
150
|
+
|
|
151
|
+
const renderTabContent = () => {
|
|
152
|
+
switch (activeTab) {
|
|
153
|
+
case "nodes":
|
|
154
|
+
return (
|
|
155
|
+
<NodeLibrary
|
|
156
|
+
onAddNode={onAddNode}
|
|
157
|
+
nodeDefinitions={nodeDefinitions}
|
|
158
|
+
getAllCategories={getAllCategories}
|
|
159
|
+
functions={functions}
|
|
160
|
+
isFunctionCanvas={!!isFunctionCanvas}
|
|
161
|
+
/>
|
|
162
|
+
);
|
|
163
|
+
case "variables":
|
|
164
|
+
return <VariablesPanel canvasId={canvasId} onSelectNode={onSelectNode} />;
|
|
165
|
+
case "channels":
|
|
166
|
+
return <ChannelsPanel />;
|
|
167
|
+
case "memory":
|
|
168
|
+
return <MemoryPanel />;
|
|
169
|
+
case "models":
|
|
170
|
+
return <ModelsPanel />;
|
|
171
|
+
case "functions":
|
|
172
|
+
return <FunctionListPanel onOpenFunction={onOpenFunction} onCreateFunction={onCreateFunction} />;
|
|
173
|
+
case "diagnostics":
|
|
174
|
+
return <DiagnosticsPanel canvasId={canvasId} onSelectNode={onSelectNode} onSelectEdge={onSelectEdge} />;
|
|
175
|
+
case "debug-context":
|
|
176
|
+
return <DebugContextPanel />;
|
|
177
|
+
default:
|
|
178
|
+
return null;
|
|
179
|
+
}
|
|
180
|
+
};
|
|
181
|
+
|
|
182
|
+
const getTabLabel = (tabId: SidebarTab) => {
|
|
183
|
+
const tab = tabs.find((tab) => tab.id === tabId);
|
|
184
|
+
return tab?.label ?? "";
|
|
185
|
+
};
|
|
186
|
+
|
|
187
|
+
return (
|
|
188
|
+
<div className="flex h-full">
|
|
189
|
+
{/* Icon Rail - Always visible */}
|
|
190
|
+
<div className="w-14 bg-card border-r border-border/50 flex flex-col items-center py-3 gap-1 shrink-0">
|
|
191
|
+
{tabs.map((tab) => {
|
|
192
|
+
const Icon = tab.icon;
|
|
193
|
+
const isActive = activeTab === tab.id;
|
|
194
|
+
|
|
195
|
+
// Per-tab error/warning counts for icon coloring + badge.
|
|
196
|
+
// Diagnostics tab covers nodes+edges; channels tab covers channels.
|
|
197
|
+
let tabErrors = 0;
|
|
198
|
+
let tabWarnings = 0;
|
|
199
|
+
if (tab.id === "diagnostics") {
|
|
200
|
+
tabErrors = totalErrors;
|
|
201
|
+
tabWarnings = totalWarnings;
|
|
202
|
+
} else if (tab.id === "channels") {
|
|
203
|
+
tabErrors = channelErrors;
|
|
204
|
+
tabWarnings = channelWarnings;
|
|
205
|
+
} else if (tab.id === "memory") {
|
|
206
|
+
tabErrors = memoryErrors;
|
|
207
|
+
tabWarnings = memoryWarnings;
|
|
208
|
+
} else if (tab.id === "models") {
|
|
209
|
+
tabErrors = modelErrors;
|
|
210
|
+
tabWarnings = modelWarnings;
|
|
211
|
+
} else if (tab.id === "functions") {
|
|
212
|
+
tabErrors = functionErrors;
|
|
213
|
+
tabWarnings = functionWarnings;
|
|
214
|
+
}
|
|
215
|
+
const tabIssueCount = tabErrors + tabWarnings;
|
|
216
|
+
const showBadge = tabIssueCount > 0;
|
|
217
|
+
|
|
218
|
+
const iconColorClass =
|
|
219
|
+
showBadge && !isActive
|
|
220
|
+
? tabErrors > 0
|
|
221
|
+
? "text-destructive hover:text-destructive hover:bg-destructive/10"
|
|
222
|
+
: "text-warning hover:text-warning hover:bg-warning/10"
|
|
223
|
+
: undefined;
|
|
224
|
+
|
|
225
|
+
return (
|
|
226
|
+
<Tooltip key={tab.id} delayDuration={300}>
|
|
227
|
+
<TooltipTrigger asChild>
|
|
228
|
+
<Button
|
|
229
|
+
variant="ghost"
|
|
230
|
+
size="icon"
|
|
231
|
+
onClick={(e) => {
|
|
232
|
+
// Drop focus after a mouse click so a follow-up Enter doesn't
|
|
233
|
+
// re-fire (toggle) this tab. Keyboard activation (detail 0) keeps focus.
|
|
234
|
+
if (e.detail !== 0) e.currentTarget.blur();
|
|
235
|
+
handleTabClick(tab.id);
|
|
236
|
+
}}
|
|
237
|
+
className={cn(
|
|
238
|
+
"w-10 h-10 transition-all duration-200 relative",
|
|
239
|
+
isActive
|
|
240
|
+
? "bg-accent text-primary shadow-sm"
|
|
241
|
+
: (iconColorClass ?? "text-muted-foreground hover:text-foreground hover:bg-accent/50"),
|
|
242
|
+
)}
|
|
243
|
+
>
|
|
244
|
+
<Icon className="w-5 h-5" />
|
|
245
|
+
{showBadge && (
|
|
246
|
+
<span
|
|
247
|
+
className={cn(
|
|
248
|
+
"absolute -top-0.5 -right-0.5 min-w-[16px] h-4 rounded-full text-[10px] font-bold flex items-center justify-center px-1 shadow-sm",
|
|
249
|
+
tabErrors > 0
|
|
250
|
+
? "bg-destructive text-destructive-foreground"
|
|
251
|
+
: "bg-warning text-warning-foreground",
|
|
252
|
+
)}
|
|
253
|
+
>
|
|
254
|
+
{tabIssueCount}
|
|
255
|
+
</span>
|
|
256
|
+
)}
|
|
257
|
+
</Button>
|
|
258
|
+
</TooltipTrigger>
|
|
259
|
+
<TooltipContent side="right" sideOffset={8}>
|
|
260
|
+
{tab.label}
|
|
261
|
+
</TooltipContent>
|
|
262
|
+
</Tooltip>
|
|
263
|
+
);
|
|
264
|
+
})}
|
|
265
|
+
</div>
|
|
266
|
+
|
|
267
|
+
{/* Content Panel - Slides in/out */}
|
|
268
|
+
<div
|
|
269
|
+
className={cn(
|
|
270
|
+
"bg-card/95 backdrop-blur-xl border-r border-border/50 transition-all duration-300 ease-in-out overflow-hidden flex flex-col",
|
|
271
|
+
activeTab ? "w-64 opacity-100" : "w-0 opacity-0",
|
|
272
|
+
)}
|
|
273
|
+
>
|
|
274
|
+
{activeTab && (
|
|
275
|
+
<>
|
|
276
|
+
{/* Panel Header */}
|
|
277
|
+
<div className="flex items-center justify-between p-3 border-b border-border/50 shrink-0">
|
|
278
|
+
<h3 className="font-semibold text-sm text-foreground">{getTabLabel(activeTab)}</h3>
|
|
279
|
+
<Button
|
|
280
|
+
variant="ghost"
|
|
281
|
+
size="icon"
|
|
282
|
+
onClick={() => onTabChange(null)}
|
|
283
|
+
className="w-7 h-7 text-muted-foreground hover:text-foreground"
|
|
284
|
+
>
|
|
285
|
+
<X className="w-4 h-4" />
|
|
286
|
+
</Button>
|
|
287
|
+
</div>
|
|
288
|
+
|
|
289
|
+
{/* Panel Content — ScrollArea overlays the scrollbar in the panel's
|
|
290
|
+
gutter so content width stays constant whether overflow is
|
|
291
|
+
present or not. Padding lives on the inner viewport because the
|
|
292
|
+
Root must clip cleanly for the absolute-positioned scrollbar. */}
|
|
293
|
+
<ScrollArea className="flex-1" viewportClassName="p-3">
|
|
294
|
+
{renderTabContent()}
|
|
295
|
+
</ScrollArea>
|
|
296
|
+
</>
|
|
297
|
+
)}
|
|
298
|
+
</div>
|
|
299
|
+
</div>
|
|
300
|
+
);
|
|
301
|
+
};
|