machinaos 0.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.
Files changed (288) hide show
  1. package/.env.template +71 -0
  2. package/LICENSE +21 -0
  3. package/README.md +87 -0
  4. package/bin/cli.js +159 -0
  5. package/client/.dockerignore +45 -0
  6. package/client/Dockerfile +68 -0
  7. package/client/eslint.config.js +29 -0
  8. package/client/index.html +13 -0
  9. package/client/nginx.conf +66 -0
  10. package/client/package.json +48 -0
  11. package/client/src/App.tsx +27 -0
  12. package/client/src/Dashboard.tsx +1173 -0
  13. package/client/src/ParameterPanel.tsx +301 -0
  14. package/client/src/components/AIAgentNode.tsx +321 -0
  15. package/client/src/components/APIKeyValidator.tsx +118 -0
  16. package/client/src/components/ClaudeChatModelNode.tsx +18 -0
  17. package/client/src/components/ConditionalEdge.tsx +189 -0
  18. package/client/src/components/CredentialsModal.tsx +306 -0
  19. package/client/src/components/EdgeConditionEditor.tsx +443 -0
  20. package/client/src/components/GeminiChatModelNode.tsx +18 -0
  21. package/client/src/components/GenericNode.tsx +357 -0
  22. package/client/src/components/LocationParameterPanel.tsx +154 -0
  23. package/client/src/components/ModelNode.tsx +286 -0
  24. package/client/src/components/OpenAIChatModelNode.tsx +18 -0
  25. package/client/src/components/OutputPanel.tsx +471 -0
  26. package/client/src/components/ParameterRenderer.tsx +1874 -0
  27. package/client/src/components/SkillEditorModal.tsx +417 -0
  28. package/client/src/components/SquareNode.tsx +797 -0
  29. package/client/src/components/StartNode.tsx +250 -0
  30. package/client/src/components/ToolkitNode.tsx +365 -0
  31. package/client/src/components/TriggerNode.tsx +463 -0
  32. package/client/src/components/auth/LoginPage.tsx +247 -0
  33. package/client/src/components/auth/ProtectedRoute.tsx +59 -0
  34. package/client/src/components/base/BaseChatModelNode.tsx +271 -0
  35. package/client/src/components/icons/AIProviderIcons.tsx +50 -0
  36. package/client/src/components/maps/GoogleMapsPicker.tsx +137 -0
  37. package/client/src/components/maps/MapsPreviewPanel.tsx +110 -0
  38. package/client/src/components/maps/index.ts +26 -0
  39. package/client/src/components/parameterPanel/InputSection.tsx +1094 -0
  40. package/client/src/components/parameterPanel/LocationPanelLayout.tsx +65 -0
  41. package/client/src/components/parameterPanel/MapsSection.tsx +92 -0
  42. package/client/src/components/parameterPanel/MiddleSection.tsx +571 -0
  43. package/client/src/components/parameterPanel/OutputSection.tsx +81 -0
  44. package/client/src/components/parameterPanel/ParameterPanelLayout.tsx +82 -0
  45. package/client/src/components/parameterPanel/ToolSchemaEditor.tsx +436 -0
  46. package/client/src/components/parameterPanel/index.ts +42 -0
  47. package/client/src/components/shared/DataPanel.tsx +142 -0
  48. package/client/src/components/shared/JSONTreeRenderer.tsx +106 -0
  49. package/client/src/components/ui/AIResultModal.tsx +204 -0
  50. package/client/src/components/ui/AndroidSettingsPanel.tsx +401 -0
  51. package/client/src/components/ui/CodeEditor.tsx +81 -0
  52. package/client/src/components/ui/CollapsibleSection.tsx +88 -0
  53. package/client/src/components/ui/ComponentItem.tsx +154 -0
  54. package/client/src/components/ui/ComponentPalette.tsx +321 -0
  55. package/client/src/components/ui/ConsolePanel.tsx +1074 -0
  56. package/client/src/components/ui/ErrorBoundary.tsx +196 -0
  57. package/client/src/components/ui/InputNodesPanel.tsx +204 -0
  58. package/client/src/components/ui/MapSelector.tsx +314 -0
  59. package/client/src/components/ui/Modal.tsx +149 -0
  60. package/client/src/components/ui/NodeContextMenu.tsx +192 -0
  61. package/client/src/components/ui/NodeOutputPanel.tsx +1150 -0
  62. package/client/src/components/ui/OutputDisplayPanel.tsx +381 -0
  63. package/client/src/components/ui/SettingsPanel.tsx +243 -0
  64. package/client/src/components/ui/TopToolbar.tsx +736 -0
  65. package/client/src/components/ui/WhatsAppSettingsPanel.tsx +345 -0
  66. package/client/src/components/ui/WorkflowSidebar.tsx +294 -0
  67. package/client/src/config/antdTheme.ts +186 -0
  68. package/client/src/config/api.ts +54 -0
  69. package/client/src/contexts/AuthContext.tsx +221 -0
  70. package/client/src/contexts/ThemeContext.tsx +42 -0
  71. package/client/src/contexts/WebSocketContext.tsx +1971 -0
  72. package/client/src/factories/baseChatModelFactory.ts +256 -0
  73. package/client/src/hooks/useAndroidOperations.ts +164 -0
  74. package/client/src/hooks/useApiKeyValidation.ts +107 -0
  75. package/client/src/hooks/useApiKeys.ts +238 -0
  76. package/client/src/hooks/useAppTheme.ts +17 -0
  77. package/client/src/hooks/useComponentPalette.ts +51 -0
  78. package/client/src/hooks/useCopyPaste.ts +155 -0
  79. package/client/src/hooks/useDragAndDrop.ts +124 -0
  80. package/client/src/hooks/useDragVariable.ts +88 -0
  81. package/client/src/hooks/useExecution.ts +313 -0
  82. package/client/src/hooks/useParameterPanel.ts +176 -0
  83. package/client/src/hooks/useReactFlowNodes.ts +189 -0
  84. package/client/src/hooks/useToolSchema.ts +209 -0
  85. package/client/src/hooks/useWhatsApp.ts +196 -0
  86. package/client/src/hooks/useWorkflowManagement.ts +46 -0
  87. package/client/src/index.css +315 -0
  88. package/client/src/main.tsx +19 -0
  89. package/client/src/nodeDefinitions/aiAgentNodes.ts +336 -0
  90. package/client/src/nodeDefinitions/aiModelNodes.ts +340 -0
  91. package/client/src/nodeDefinitions/androidDeviceNodes.ts +140 -0
  92. package/client/src/nodeDefinitions/androidServiceNodes.ts +383 -0
  93. package/client/src/nodeDefinitions/chatNodes.ts +135 -0
  94. package/client/src/nodeDefinitions/codeNodes.ts +54 -0
  95. package/client/src/nodeDefinitions/documentNodes.ts +379 -0
  96. package/client/src/nodeDefinitions/index.ts +15 -0
  97. package/client/src/nodeDefinitions/locationNodes.ts +463 -0
  98. package/client/src/nodeDefinitions/schedulerNodes.ts +220 -0
  99. package/client/src/nodeDefinitions/skillNodes.ts +211 -0
  100. package/client/src/nodeDefinitions/toolNodes.ts +198 -0
  101. package/client/src/nodeDefinitions/utilityNodes.ts +284 -0
  102. package/client/src/nodeDefinitions/whatsappNodes.ts +865 -0
  103. package/client/src/nodeDefinitions/workflowNodes.ts +41 -0
  104. package/client/src/nodeDefinitions.ts +104 -0
  105. package/client/src/schemas/workflowSchema.ts +264 -0
  106. package/client/src/services/dynamicParameterService.ts +96 -0
  107. package/client/src/services/execution/aiAgentExecutionService.ts +35 -0
  108. package/client/src/services/executionService.ts +232 -0
  109. package/client/src/services/workflowApi.ts +91 -0
  110. package/client/src/store/useAppStore.ts +582 -0
  111. package/client/src/styles/theme.ts +508 -0
  112. package/client/src/styles/zIndex.ts +17 -0
  113. package/client/src/types/ComponentTypes.ts +39 -0
  114. package/client/src/types/EdgeCondition.ts +231 -0
  115. package/client/src/types/INodeProperties.ts +288 -0
  116. package/client/src/types/NodeTypes.ts +28 -0
  117. package/client/src/utils/formatters.ts +33 -0
  118. package/client/src/utils/googleMapsLoader.ts +140 -0
  119. package/client/src/utils/locationUtils.ts +85 -0
  120. package/client/src/utils/nodeUtils.ts +31 -0
  121. package/client/src/utils/workflow.ts +30 -0
  122. package/client/src/utils/workflowExport.ts +120 -0
  123. package/client/src/vite-env.d.ts +12 -0
  124. package/client/tailwind.config.js +60 -0
  125. package/client/tsconfig.json +25 -0
  126. package/client/tsconfig.node.json +11 -0
  127. package/client/vite.config.js +35 -0
  128. package/docker-compose.prod.yml +107 -0
  129. package/docker-compose.yml +104 -0
  130. package/docs-MachinaOs/README.md +85 -0
  131. package/docs-MachinaOs/deployment/docker.mdx +228 -0
  132. package/docs-MachinaOs/deployment/production.mdx +345 -0
  133. package/docs-MachinaOs/docs.json +75 -0
  134. package/docs-MachinaOs/faq.mdx +309 -0
  135. package/docs-MachinaOs/favicon.svg +5 -0
  136. package/docs-MachinaOs/installation.mdx +160 -0
  137. package/docs-MachinaOs/introduction.mdx +114 -0
  138. package/docs-MachinaOs/logo/dark.svg +6 -0
  139. package/docs-MachinaOs/logo/light.svg +6 -0
  140. package/docs-MachinaOs/nodes/ai-agent.mdx +216 -0
  141. package/docs-MachinaOs/nodes/ai-models.mdx +240 -0
  142. package/docs-MachinaOs/nodes/android.mdx +411 -0
  143. package/docs-MachinaOs/nodes/overview.mdx +181 -0
  144. package/docs-MachinaOs/nodes/schedulers.mdx +316 -0
  145. package/docs-MachinaOs/nodes/webhooks.mdx +330 -0
  146. package/docs-MachinaOs/nodes/whatsapp.mdx +305 -0
  147. package/docs-MachinaOs/quickstart.mdx +119 -0
  148. package/docs-MachinaOs/tutorials/ai-agent-workflow.mdx +177 -0
  149. package/docs-MachinaOs/tutorials/android-automation.mdx +242 -0
  150. package/docs-MachinaOs/tutorials/first-workflow.mdx +134 -0
  151. package/docs-MachinaOs/tutorials/whatsapp-automation.mdx +185 -0
  152. package/nul +0 -0
  153. package/package.json +70 -0
  154. package/scripts/build.js +158 -0
  155. package/scripts/check-ports.ps1 +33 -0
  156. package/scripts/clean.js +40 -0
  157. package/scripts/docker.js +93 -0
  158. package/scripts/kill-port.ps1 +154 -0
  159. package/scripts/start.js +210 -0
  160. package/scripts/stop.js +325 -0
  161. package/server/.dockerignore +44 -0
  162. package/server/Dockerfile +45 -0
  163. package/server/constants.py +249 -0
  164. package/server/core/__init__.py +1 -0
  165. package/server/core/cache.py +461 -0
  166. package/server/core/config.py +128 -0
  167. package/server/core/container.py +99 -0
  168. package/server/core/database.py +1211 -0
  169. package/server/core/logging.py +314 -0
  170. package/server/main.py +289 -0
  171. package/server/middleware/__init__.py +5 -0
  172. package/server/middleware/auth.py +89 -0
  173. package/server/models/__init__.py +1 -0
  174. package/server/models/auth.py +52 -0
  175. package/server/models/cache.py +24 -0
  176. package/server/models/database.py +211 -0
  177. package/server/models/nodes.py +455 -0
  178. package/server/package.json +9 -0
  179. package/server/pyproject.toml +72 -0
  180. package/server/requirements.txt +83 -0
  181. package/server/routers/__init__.py +1 -0
  182. package/server/routers/android.py +294 -0
  183. package/server/routers/auth.py +203 -0
  184. package/server/routers/database.py +151 -0
  185. package/server/routers/maps.py +142 -0
  186. package/server/routers/nodejs_compat.py +289 -0
  187. package/server/routers/webhook.py +90 -0
  188. package/server/routers/websocket.py +2127 -0
  189. package/server/routers/whatsapp.py +761 -0
  190. package/server/routers/workflow.py +200 -0
  191. package/server/services/__init__.py +1 -0
  192. package/server/services/ai.py +2415 -0
  193. package/server/services/android/__init__.py +27 -0
  194. package/server/services/android/broadcaster.py +114 -0
  195. package/server/services/android/client.py +608 -0
  196. package/server/services/android/manager.py +78 -0
  197. package/server/services/android/protocol.py +165 -0
  198. package/server/services/android_service.py +588 -0
  199. package/server/services/auth.py +131 -0
  200. package/server/services/chat_client.py +160 -0
  201. package/server/services/deployment/__init__.py +12 -0
  202. package/server/services/deployment/manager.py +706 -0
  203. package/server/services/deployment/state.py +47 -0
  204. package/server/services/deployment/triggers.py +275 -0
  205. package/server/services/event_waiter.py +785 -0
  206. package/server/services/execution/__init__.py +77 -0
  207. package/server/services/execution/cache.py +769 -0
  208. package/server/services/execution/conditions.py +373 -0
  209. package/server/services/execution/dlq.py +132 -0
  210. package/server/services/execution/executor.py +1351 -0
  211. package/server/services/execution/models.py +531 -0
  212. package/server/services/execution/recovery.py +235 -0
  213. package/server/services/handlers/__init__.py +126 -0
  214. package/server/services/handlers/ai.py +355 -0
  215. package/server/services/handlers/android.py +260 -0
  216. package/server/services/handlers/code.py +278 -0
  217. package/server/services/handlers/document.py +598 -0
  218. package/server/services/handlers/http.py +193 -0
  219. package/server/services/handlers/polyglot.py +105 -0
  220. package/server/services/handlers/tools.py +845 -0
  221. package/server/services/handlers/triggers.py +107 -0
  222. package/server/services/handlers/utility.py +822 -0
  223. package/server/services/handlers/whatsapp.py +476 -0
  224. package/server/services/maps.py +289 -0
  225. package/server/services/memory_store.py +103 -0
  226. package/server/services/node_executor.py +375 -0
  227. package/server/services/parameter_resolver.py +218 -0
  228. package/server/services/polyglot_client.py +169 -0
  229. package/server/services/scheduler.py +155 -0
  230. package/server/services/skill_loader.py +417 -0
  231. package/server/services/status_broadcaster.py +826 -0
  232. package/server/services/temporal/__init__.py +23 -0
  233. package/server/services/temporal/activities.py +344 -0
  234. package/server/services/temporal/client.py +76 -0
  235. package/server/services/temporal/executor.py +147 -0
  236. package/server/services/temporal/worker.py +251 -0
  237. package/server/services/temporal/workflow.py +355 -0
  238. package/server/services/temporal/ws_client.py +236 -0
  239. package/server/services/text.py +111 -0
  240. package/server/services/user_auth.py +172 -0
  241. package/server/services/websocket_client.py +29 -0
  242. package/server/services/workflow.py +597 -0
  243. package/server/skills/android-skill/SKILL.md +82 -0
  244. package/server/skills/assistant-personality/SKILL.md +45 -0
  245. package/server/skills/code-skill/SKILL.md +140 -0
  246. package/server/skills/http-skill/SKILL.md +161 -0
  247. package/server/skills/maps-skill/SKILL.md +170 -0
  248. package/server/skills/memory-skill/SKILL.md +154 -0
  249. package/server/skills/scheduler-skill/SKILL.md +84 -0
  250. package/server/skills/whatsapp-skill/SKILL.md +283 -0
  251. package/server/uv.lock +2916 -0
  252. package/server/whatsapp-rpc/.dockerignore +30 -0
  253. package/server/whatsapp-rpc/Dockerfile +44 -0
  254. package/server/whatsapp-rpc/Dockerfile.web +17 -0
  255. package/server/whatsapp-rpc/README.md +139 -0
  256. package/server/whatsapp-rpc/cli.js +95 -0
  257. package/server/whatsapp-rpc/configs/config.yaml +7 -0
  258. package/server/whatsapp-rpc/docker-compose.yml +35 -0
  259. package/server/whatsapp-rpc/docs/API.md +410 -0
  260. package/server/whatsapp-rpc/go.mod +67 -0
  261. package/server/whatsapp-rpc/go.sum +203 -0
  262. package/server/whatsapp-rpc/package.json +30 -0
  263. package/server/whatsapp-rpc/schema.json +1294 -0
  264. package/server/whatsapp-rpc/scripts/clean.cjs +66 -0
  265. package/server/whatsapp-rpc/scripts/cli.js +162 -0
  266. package/server/whatsapp-rpc/src/go/cmd/server/main.go +91 -0
  267. package/server/whatsapp-rpc/src/go/config/config.go +49 -0
  268. package/server/whatsapp-rpc/src/go/rpc/rpc.go +446 -0
  269. package/server/whatsapp-rpc/src/go/rpc/server.go +112 -0
  270. package/server/whatsapp-rpc/src/go/whatsapp/history.go +166 -0
  271. package/server/whatsapp-rpc/src/go/whatsapp/messages.go +390 -0
  272. package/server/whatsapp-rpc/src/go/whatsapp/service.go +2130 -0
  273. package/server/whatsapp-rpc/src/go/whatsapp/types.go +261 -0
  274. package/server/whatsapp-rpc/src/python/pyproject.toml +15 -0
  275. package/server/whatsapp-rpc/src/python/whatsapp_rpc/__init__.py +4 -0
  276. package/server/whatsapp-rpc/src/python/whatsapp_rpc/client.py +427 -0
  277. package/server/whatsapp-rpc/web/app.py +609 -0
  278. package/server/whatsapp-rpc/web/requirements.txt +6 -0
  279. package/server/whatsapp-rpc/web/rpc_client.py +427 -0
  280. package/server/whatsapp-rpc/web/static/openapi.yaml +59 -0
  281. package/server/whatsapp-rpc/web/templates/base.html +150 -0
  282. package/server/whatsapp-rpc/web/templates/contacts.html +240 -0
  283. package/server/whatsapp-rpc/web/templates/dashboard.html +320 -0
  284. package/server/whatsapp-rpc/web/templates/groups.html +328 -0
  285. package/server/whatsapp-rpc/web/templates/messages.html +465 -0
  286. package/server/whatsapp-rpc/web/templates/messaging.html +681 -0
  287. package/server/whatsapp-rpc/web/templates/send.html +259 -0
  288. package/server/whatsapp-rpc/web/templates/settings.html +459 -0
@@ -0,0 +1,443 @@
1
+ /**
2
+ * EdgeConditionEditor - Modal for configuring edge conditions
3
+ *
4
+ * Allows users to set up runtime conditional branching on workflow edges.
5
+ * Conditions are evaluated against the source node's output at runtime.
6
+ */
7
+ import React, { useState, useEffect, useCallback } from 'react';
8
+ import * as Dialog from '@radix-ui/react-dialog';
9
+ import { useAppTheme } from '../hooks/useAppTheme';
10
+ import {
11
+ EdgeCondition,
12
+ ConditionOperator,
13
+ OPERATORS,
14
+ getOperatorsByCategory,
15
+ formatConditionLabel,
16
+ validateCondition,
17
+ } from '../types/EdgeCondition';
18
+
19
+ interface EdgeConditionEditorProps {
20
+ isOpen: boolean;
21
+ onClose: () => void;
22
+ /** Current condition (if editing existing) */
23
+ condition?: EdgeCondition;
24
+ /** Current edge label */
25
+ label?: string;
26
+ /** Callback when condition is saved */
27
+ onSave: (condition: EdgeCondition | undefined, label: string | undefined) => void;
28
+ /** Available fields from source node output (for autocomplete) */
29
+ availableFields?: string[];
30
+ /** Source node name for display */
31
+ sourceNodeName?: string;
32
+ }
33
+
34
+ const EdgeConditionEditor: React.FC<EdgeConditionEditorProps> = ({
35
+ isOpen,
36
+ onClose,
37
+ condition,
38
+ label,
39
+ onSave,
40
+ availableFields = [],
41
+ sourceNodeName,
42
+ }) => {
43
+ const theme = useAppTheme();
44
+
45
+ // Form state
46
+ const [field, setField] = useState(condition?.field || '');
47
+ const [operator, setOperator] = useState<ConditionOperator>(condition?.operator || 'eq');
48
+ const [value, setValue] = useState<string>(
49
+ condition?.value !== undefined ? String(condition.value) : ''
50
+ );
51
+ const [edgeLabel, setEdgeLabel] = useState(label || '');
52
+ const [error, setError] = useState<string | null>(null);
53
+
54
+ // Reset form when modal opens with new condition
55
+ useEffect(() => {
56
+ if (isOpen) {
57
+ setField(condition?.field || '');
58
+ setOperator(condition?.operator || 'eq');
59
+ setValue(condition?.value !== undefined ? String(condition.value) : '');
60
+ setEdgeLabel(label || '');
61
+ setError(null);
62
+ }
63
+ }, [isOpen, condition, label]);
64
+
65
+ const operatorMeta = OPERATORS[operator];
66
+ const operatorsByCategory = getOperatorsByCategory();
67
+
68
+ const handleSave = useCallback(() => {
69
+ // If no field specified, remove condition
70
+ if (!field.trim()) {
71
+ onSave(undefined, edgeLabel.trim() || undefined);
72
+ onClose();
73
+ return;
74
+ }
75
+
76
+ const newCondition: EdgeCondition = {
77
+ field: field.trim(),
78
+ operator,
79
+ value: operatorMeta.requiresValue ? parseValue(value, operatorMeta.valueType) : undefined,
80
+ };
81
+
82
+ const validation = validateCondition(newCondition);
83
+ if (!validation.valid) {
84
+ setError(validation.error || 'Invalid condition');
85
+ return;
86
+ }
87
+
88
+ onSave(newCondition, edgeLabel.trim() || formatConditionLabel(newCondition));
89
+ onClose();
90
+ }, [field, operator, value, edgeLabel, operatorMeta, onSave, onClose]);
91
+
92
+ const handleRemoveCondition = useCallback(() => {
93
+ onSave(undefined, undefined);
94
+ onClose();
95
+ }, [onSave, onClose]);
96
+
97
+ // Parse value based on expected type
98
+ const parseValue = (val: string, type?: string): any => {
99
+ if (type === 'number') {
100
+ const num = parseFloat(val);
101
+ return isNaN(num) ? val : num;
102
+ }
103
+ if (type === 'boolean') {
104
+ return val.toLowerCase() === 'true';
105
+ }
106
+ if (type === 'array') {
107
+ try {
108
+ return JSON.parse(val);
109
+ } catch {
110
+ return val.split(',').map(s => s.trim());
111
+ }
112
+ }
113
+ return val;
114
+ };
115
+
116
+ const inputStyle: React.CSSProperties = {
117
+ width: '100%',
118
+ padding: `${theme.spacing.sm} ${theme.spacing.md}`,
119
+ fontSize: theme.fontSize.base,
120
+ border: `1px solid ${theme.colors.border}`,
121
+ borderRadius: theme.borderRadius.md,
122
+ backgroundColor: theme.colors.background,
123
+ color: theme.colors.text,
124
+ fontFamily: 'system-ui, sans-serif',
125
+ outline: 'none',
126
+ };
127
+
128
+ const selectStyle: React.CSSProperties = {
129
+ ...inputStyle,
130
+ cursor: 'pointer',
131
+ };
132
+
133
+ const labelStyle: React.CSSProperties = {
134
+ display: 'block',
135
+ marginBottom: theme.spacing.xs,
136
+ fontSize: theme.fontSize.sm,
137
+ fontWeight: theme.fontWeight.medium,
138
+ color: theme.colors.textSecondary,
139
+ };
140
+
141
+ const sectionStyle: React.CSSProperties = {
142
+ marginBottom: theme.spacing.lg,
143
+ };
144
+
145
+ const buttonBaseStyle: React.CSSProperties = {
146
+ padding: `${theme.spacing.sm} ${theme.spacing.lg}`,
147
+ borderRadius: theme.borderRadius.md,
148
+ fontSize: theme.fontSize.sm,
149
+ fontWeight: theme.fontWeight.medium,
150
+ cursor: 'pointer',
151
+ border: 'none',
152
+ transition: theme.transitions.fast,
153
+ };
154
+
155
+ return (
156
+ <Dialog.Root open={isOpen}>
157
+ <Dialog.Portal>
158
+ <Dialog.Overlay
159
+ style={{
160
+ position: 'fixed',
161
+ inset: 0,
162
+ backgroundColor: 'rgba(0, 0, 0, 0.5)',
163
+ backdropFilter: 'blur(2px)',
164
+ zIndex: 1100,
165
+ }}
166
+ />
167
+ <Dialog.Content
168
+ style={{
169
+ position: 'fixed',
170
+ top: '50%',
171
+ left: '50%',
172
+ transform: 'translate(-50%, -50%)',
173
+ backgroundColor: theme.colors.background,
174
+ borderRadius: theme.borderRadius.lg,
175
+ boxShadow: '0 25px 50px -12px rgba(0, 0, 0, 0.25)',
176
+ width: '420px',
177
+ maxHeight: '85vh',
178
+ display: 'flex',
179
+ flexDirection: 'column',
180
+ fontFamily: 'system-ui, sans-serif',
181
+ overflow: 'hidden',
182
+ zIndex: 1101,
183
+ }}
184
+ >
185
+ {/* Header */}
186
+ <div
187
+ style={{
188
+ padding: `${theme.spacing.lg} ${theme.spacing.xl}`,
189
+ borderBottom: `1px solid ${theme.colors.border}`,
190
+ backgroundColor: theme.colors.backgroundPanel,
191
+ display: 'flex',
192
+ alignItems: 'center',
193
+ justifyContent: 'space-between',
194
+ }}
195
+ >
196
+ <Dialog.Title
197
+ style={{
198
+ margin: 0,
199
+ fontSize: theme.fontSize.lg,
200
+ fontWeight: theme.fontWeight.semibold,
201
+ color: theme.colors.text,
202
+ display: 'flex',
203
+ alignItems: 'center',
204
+ gap: theme.spacing.sm,
205
+ }}
206
+ >
207
+ <svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke={theme.dracula.cyan} strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
208
+ <path d="M16 3h5v5"/>
209
+ <path d="M21 3l-7 7"/>
210
+ <path d="M8 21H3v-5"/>
211
+ <path d="M3 21l7-7"/>
212
+ </svg>
213
+ Edge Condition
214
+ </Dialog.Title>
215
+ <Dialog.Close
216
+ onClick={onClose}
217
+ style={{
218
+ background: 'transparent',
219
+ border: 'none',
220
+ fontSize: '20px',
221
+ cursor: 'pointer',
222
+ color: theme.colors.textSecondary,
223
+ padding: theme.spacing.xs,
224
+ borderRadius: theme.borderRadius.sm,
225
+ display: 'flex',
226
+ alignItems: 'center',
227
+ justifyContent: 'center',
228
+ width: '32px',
229
+ height: '32px',
230
+ }}
231
+ >
232
+ <svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
233
+ <line x1="18" y1="6" x2="6" y2="18"/>
234
+ <line x1="6" y1="6" x2="18" y2="18"/>
235
+ </svg>
236
+ </Dialog.Close>
237
+ </div>
238
+
239
+ <Dialog.Description style={{ display: 'none' }}>
240
+ Configure condition for edge branching
241
+ </Dialog.Description>
242
+
243
+ {/* Content */}
244
+ <div style={{ padding: theme.spacing.xl, overflowY: 'auto' }}>
245
+ {/* Source node info */}
246
+ {sourceNodeName && (
247
+ <div
248
+ style={{
249
+ marginBottom: theme.spacing.lg,
250
+ padding: theme.spacing.md,
251
+ backgroundColor: theme.colors.backgroundPanel,
252
+ borderRadius: theme.borderRadius.md,
253
+ fontSize: theme.fontSize.sm,
254
+ color: theme.colors.textSecondary,
255
+ }}
256
+ >
257
+ Condition evaluates output from: <strong style={{ color: theme.colors.text }}>{sourceNodeName}</strong>
258
+ </div>
259
+ )}
260
+
261
+ {/* Field input */}
262
+ <div style={sectionStyle}>
263
+ <label style={labelStyle}>Field Path</label>
264
+ <input
265
+ type="text"
266
+ value={field}
267
+ onChange={(e) => setField(e.target.value)}
268
+ placeholder="e.g., result.status, items.0.name"
269
+ style={inputStyle}
270
+ list="available-fields"
271
+ />
272
+ {availableFields.length > 0 && (
273
+ <datalist id="available-fields">
274
+ {availableFields.map((f) => (
275
+ <option key={f} value={f} />
276
+ ))}
277
+ </datalist>
278
+ )}
279
+ <div style={{ marginTop: theme.spacing.xs, fontSize: theme.fontSize.xs, color: theme.colors.textMuted }}>
280
+ Use dot notation for nested fields (e.g., result.data.value)
281
+ </div>
282
+ </div>
283
+
284
+ {/* Operator select */}
285
+ <div style={sectionStyle}>
286
+ <label style={labelStyle}>Operator</label>
287
+ <select
288
+ value={operator}
289
+ onChange={(e) => setOperator(e.target.value as ConditionOperator)}
290
+ style={selectStyle}
291
+ >
292
+ {Object.entries(operatorsByCategory).map(([category, ops]) => (
293
+ <optgroup key={category} label={category}>
294
+ {ops.map((op) => (
295
+ <option key={op} value={op}>
296
+ {OPERATORS[op].label}
297
+ </option>
298
+ ))}
299
+ </optgroup>
300
+ ))}
301
+ </select>
302
+ <div style={{ marginTop: theme.spacing.xs, fontSize: theme.fontSize.xs, color: theme.colors.textMuted }}>
303
+ {operatorMeta.description}
304
+ </div>
305
+ </div>
306
+
307
+ {/* Value input (conditional) */}
308
+ {operatorMeta.requiresValue && (
309
+ <div style={sectionStyle}>
310
+ <label style={labelStyle}>
311
+ Value
312
+ {operatorMeta.valueType === 'array' && (
313
+ <span style={{ fontWeight: 'normal', marginLeft: theme.spacing.sm }}>
314
+ (comma-separated or JSON array)
315
+ </span>
316
+ )}
317
+ </label>
318
+ <input
319
+ type={operatorMeta.valueType === 'number' ? 'number' : 'text'}
320
+ value={value}
321
+ onChange={(e) => setValue(e.target.value)}
322
+ placeholder={
323
+ operatorMeta.valueType === 'array'
324
+ ? 'e.g., value1, value2 or ["a", "b"]'
325
+ : 'Enter comparison value'
326
+ }
327
+ style={inputStyle}
328
+ />
329
+ </div>
330
+ )}
331
+
332
+ {/* Edge label */}
333
+ <div style={sectionStyle}>
334
+ <label style={labelStyle}>Edge Label (optional)</label>
335
+ <input
336
+ type="text"
337
+ value={edgeLabel}
338
+ onChange={(e) => setEdgeLabel(e.target.value)}
339
+ placeholder="Auto-generated from condition"
340
+ style={inputStyle}
341
+ />
342
+ <div style={{ marginTop: theme.spacing.xs, fontSize: theme.fontSize.xs, color: theme.colors.textMuted }}>
343
+ Displayed on the workflow canvas
344
+ </div>
345
+ </div>
346
+
347
+ {/* Error message */}
348
+ {error && (
349
+ <div
350
+ style={{
351
+ padding: theme.spacing.md,
352
+ backgroundColor: `${theme.colors.error}20`,
353
+ border: `1px solid ${theme.colors.error}40`,
354
+ borderRadius: theme.borderRadius.md,
355
+ color: theme.colors.error,
356
+ fontSize: theme.fontSize.sm,
357
+ marginBottom: theme.spacing.lg,
358
+ }}
359
+ >
360
+ {error}
361
+ </div>
362
+ )}
363
+
364
+ {/* Preview */}
365
+ {field && (
366
+ <div
367
+ style={{
368
+ padding: theme.spacing.md,
369
+ backgroundColor: theme.colors.backgroundPanel,
370
+ borderRadius: theme.borderRadius.md,
371
+ marginBottom: theme.spacing.lg,
372
+ }}
373
+ >
374
+ <div style={{ fontSize: theme.fontSize.xs, color: theme.colors.textMuted, marginBottom: theme.spacing.xs }}>
375
+ Preview:
376
+ </div>
377
+ <div style={{ fontSize: theme.fontSize.sm, color: theme.dracula.cyan, fontFamily: 'monospace' }}>
378
+ if output.{field} {OPERATORS[operator].label.toLowerCase()}
379
+ {operatorMeta.requiresValue && value && ` "${value}"`}
380
+ </div>
381
+ </div>
382
+ )}
383
+ </div>
384
+
385
+ {/* Footer */}
386
+ <div
387
+ style={{
388
+ padding: theme.spacing.lg,
389
+ borderTop: `1px solid ${theme.colors.border}`,
390
+ backgroundColor: theme.colors.backgroundPanel,
391
+ display: 'flex',
392
+ justifyContent: 'space-between',
393
+ alignItems: 'center',
394
+ }}
395
+ >
396
+ {/* Remove condition button (left side) */}
397
+ {condition && (
398
+ <button
399
+ onClick={handleRemoveCondition}
400
+ style={{
401
+ ...buttonBaseStyle,
402
+ backgroundColor: `${theme.colors.error}20`,
403
+ color: theme.colors.error,
404
+ border: `1px solid ${theme.colors.error}40`,
405
+ }}
406
+ >
407
+ Remove Condition
408
+ </button>
409
+ )}
410
+ {!condition && <div />}
411
+
412
+ {/* Save/Cancel buttons (right side) */}
413
+ <div style={{ display: 'flex', gap: theme.spacing.sm }}>
414
+ <button
415
+ onClick={onClose}
416
+ style={{
417
+ ...buttonBaseStyle,
418
+ backgroundColor: 'transparent',
419
+ color: theme.colors.textSecondary,
420
+ border: `1px solid ${theme.colors.border}`,
421
+ }}
422
+ >
423
+ Cancel
424
+ </button>
425
+ <button
426
+ onClick={handleSave}
427
+ style={{
428
+ ...buttonBaseStyle,
429
+ backgroundColor: theme.dracula.cyan,
430
+ color: theme.dracula.background,
431
+ }}
432
+ >
433
+ Save Condition
434
+ </button>
435
+ </div>
436
+ </div>
437
+ </Dialog.Content>
438
+ </Dialog.Portal>
439
+ </Dialog.Root>
440
+ );
441
+ };
442
+
443
+ export default EdgeConditionEditor;
@@ -0,0 +1,18 @@
1
+ import React from 'react';
2
+ import { NodeProps } from 'reactflow';
3
+ import { NodeData } from '../types/NodeTypes';
4
+ import BaseChatModelNode from './base/BaseChatModelNode';
5
+
6
+ const GeminiChatModelNode: React.FC<NodeProps<NodeData>> = (props) => {
7
+ return (
8
+ <BaseChatModelNode
9
+ {...props}
10
+ providerId="gemini"
11
+ displayName="Gemini"
12
+ icon="⭐"
13
+ color="#4285F4"
14
+ />
15
+ );
16
+ };
17
+
18
+ export default GeminiChatModelNode;