@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.
Files changed (120) hide show
  1. package/LICENSE +661 -661
  2. package/NOTICE +16 -16
  3. package/README.md +110 -93
  4. package/dist/components/ui/command.d.ts +2 -2
  5. package/dist/components/ui/input.d.ts +1 -1
  6. package/dist/components/ui/resizable.d.ts +1 -1
  7. package/dist/components/ui/textarea.d.ts +1 -1
  8. package/dist/graph/BaseNode.js +10 -10
  9. package/dist/graph/reactFlowRegistry.d.ts.map +1 -1
  10. package/dist/lib/utils.d.ts +3 -0
  11. package/dist/lib/utils.d.ts.map +1 -0
  12. package/dist/lib/utils.js +6 -0
  13. package/dist/lib/utils.js.map +1 -0
  14. package/dist/toolbars/CanvasTabsToolbar.d.ts +11 -0
  15. package/dist/toolbars/CanvasTabsToolbar.d.ts.map +1 -0
  16. package/dist/toolbars/CanvasTabsToolbar.js +101 -0
  17. package/dist/toolbars/CanvasTabsToolbar.js.map +1 -0
  18. package/package.json +2 -2
  19. package/src/BuilderLayout.tsx +345 -345
  20. package/src/Canvas.tsx +261 -261
  21. package/src/CanvasEditor.tsx +142 -142
  22. package/src/CanvasTabsToolbar.tsx +176 -176
  23. package/src/RightConfigPanel.tsx +266 -266
  24. package/src/WorkflowBuilder.tsx +412 -412
  25. package/src/cn.ts +6 -6
  26. package/src/components/ui/add-button.tsx +39 -39
  27. package/src/components/ui/alert-dialog.tsx +141 -141
  28. package/src/components/ui/alert.tsx +59 -59
  29. package/src/components/ui/badge.tsx +36 -36
  30. package/src/components/ui/button.tsx +85 -85
  31. package/src/components/ui/card.tsx +79 -79
  32. package/src/components/ui/checkbox.tsx +28 -28
  33. package/src/components/ui/collapsible.tsx +9 -9
  34. package/src/components/ui/command.tsx +153 -153
  35. package/src/components/ui/delete-button.tsx +23 -23
  36. package/src/components/ui/dialog.tsx +125 -125
  37. package/src/components/ui/dropdown-menu.tsx +198 -198
  38. package/src/components/ui/input.tsx +55 -55
  39. package/src/components/ui/label.tsx +24 -24
  40. package/src/components/ui/readonly-banner.tsx +15 -15
  41. package/src/components/ui/resizable.tsx +43 -43
  42. package/src/components/ui/scroll-area.tsx +102 -102
  43. package/src/components/ui/select.tsx +160 -160
  44. package/src/components/ui/separator.tsx +29 -29
  45. package/src/components/ui/switch.tsx +27 -27
  46. package/src/components/ui/textarea.tsx +51 -51
  47. package/src/components/ui/toast.tsx +127 -127
  48. package/src/components/ui/toaster.tsx +33 -33
  49. package/src/components/ui/toggle-group.tsx +59 -59
  50. package/src/components/ui/toggle.tsx +43 -43
  51. package/src/components/ui/tooltip.tsx +32 -32
  52. package/src/dialogs/NodePickerDialog.tsx +84 -84
  53. package/src/dialogs/ValidationDialog.tsx +184 -184
  54. package/src/graph/BaseNode.tsx +557 -557
  55. package/src/graph/CustomEdge.tsx +185 -185
  56. package/src/graph/CustomNode.tsx +16 -16
  57. package/src/graph/FunctionCallNode.tsx +30 -30
  58. package/src/graph/PortHandle.tsx +189 -189
  59. package/src/graph/reactFlowRegistry.ts +26 -26
  60. package/src/hooks/use-toast.ts +125 -125
  61. package/src/hooks/useAvailableVariables.ts +20 -20
  62. package/src/hooks/useCanvasHistory.ts +22 -22
  63. package/src/hooks/useCanvasTabs.ts +168 -168
  64. package/src/hooks/useFunctionDiagnosticsSync.ts +40 -40
  65. package/src/hooks/useFunctionRegistry.ts +26 -26
  66. package/src/hooks/useFunctions.ts +44 -44
  67. package/src/hooks/useGraph.ts +161 -161
  68. package/src/hooks/useNodeDefinitions.ts +82 -82
  69. package/src/hooks/useParamErrors.ts +26 -26
  70. package/src/hooks/useResolvedTheme.ts +30 -30
  71. package/src/hooks/useResourceDiagnosticsSync.ts +58 -58
  72. package/src/hooks/useSuppressThemeTransition.ts +79 -79
  73. package/src/hooks/useWorkflowSerialization.ts +127 -127
  74. package/src/i18n/index.ts +53 -53
  75. package/src/i18n/locales/de.json +501 -501
  76. package/src/i18n/locales/en.json +557 -557
  77. package/src/index.ts +27 -27
  78. package/src/inputs/ExpressionInput.tsx +297 -297
  79. package/src/inputs/ParameterEditor.tsx +515 -515
  80. package/src/inputs/PortSection.tsx +144 -144
  81. package/src/panels/BuilderSidebar.tsx +301 -301
  82. package/src/panels/ChannelConfigPanel.tsx +49 -49
  83. package/src/panels/ChannelsPanel.tsx +28 -28
  84. package/src/panels/DebugConsolePanel.tsx +73 -73
  85. package/src/panels/DebugContextPanel.tsx +77 -77
  86. package/src/panels/DebugExternalIOPanel.tsx +180 -180
  87. package/src/panels/DiagnosticsPanel.tsx +170 -170
  88. package/src/panels/EdgeConfigPanel.tsx +104 -104
  89. package/src/panels/FunctionConfigPanel.tsx +179 -179
  90. package/src/panels/FunctionListPanel.tsx +45 -45
  91. package/src/panels/MemoryConfigPanel.tsx +55 -55
  92. package/src/panels/MemoryPanel.tsx +40 -40
  93. package/src/panels/ModelConfigPanel.tsx +41 -41
  94. package/src/panels/ModelsPanel.tsx +36 -36
  95. package/src/panels/NodeConfigPanel.tsx +630 -630
  96. package/src/panels/NodeLibrary.tsx +288 -288
  97. package/src/panels/ResourceConfigPanel.tsx +132 -132
  98. package/src/panels/ResourceListPanel.tsx +113 -113
  99. package/src/panels/VariableConfigPanel.tsx +161 -161
  100. package/src/panels/VariablesPanel.tsx +145 -145
  101. package/src/stores/canvasStore.test.ts +44 -44
  102. package/src/stores/canvasStore.ts +245 -245
  103. package/src/stores/debugStore.ts +74 -74
  104. package/src/stores/diagnosticsStore.ts +130 -130
  105. package/src/stores/editorStore.ts +202 -202
  106. package/src/styles/index.css +526 -526
  107. package/src/utils/categoryConstants.ts +26 -26
  108. package/src/utils/channelOperations.ts +86 -86
  109. package/src/utils/connectionRules.ts +137 -137
  110. package/src/utils/functionOperations.ts +179 -179
  111. package/src/utils/graphOperations.ts +550 -550
  112. package/src/utils/history.ts +207 -207
  113. package/src/utils/memoryOperations.ts +57 -57
  114. package/src/utils/migrateFunctionNodes.ts +107 -107
  115. package/src/utils/modelOperations.ts +55 -55
  116. package/src/utils/paramDisplay.ts +71 -71
  117. package/src/utils/resourceHelpers.ts +32 -32
  118. package/src/utils/translation.ts +28 -28
  119. package/src/utils/variableOperations.ts +75 -75
  120. 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
+ };