@geenius/ai-workflow 0.1.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 (140) hide show
  1. package/.changeset/config.json +11 -0
  2. package/.github/CODEOWNERS +1 -0
  3. package/.github/ISSUE_TEMPLATE/bug_report.md +16 -0
  4. package/.github/ISSUE_TEMPLATE/feature_request.md +11 -0
  5. package/.github/PULL_REQUEST_TEMPLATE.md +10 -0
  6. package/.github/dependabot.yml +11 -0
  7. package/.github/workflows/ci.yml +23 -0
  8. package/.github/workflows/release.yml +29 -0
  9. package/.nvmrc +1 -0
  10. package/.project/ACCOUNT.yaml +4 -0
  11. package/.project/IDEAS.yaml +7 -0
  12. package/.project/PROJECT.yaml +11 -0
  13. package/.project/ROADMAP.yaml +15 -0
  14. package/CHANGELOG.md +11 -0
  15. package/CODE_OF_CONDUCT.md +16 -0
  16. package/CONTRIBUTING.md +26 -0
  17. package/LICENSE +21 -0
  18. package/README.md +1 -0
  19. package/SECURITY.md +15 -0
  20. package/SUPPORT.md +8 -0
  21. package/package.json +74 -0
  22. package/packages/convex/README.md +1 -0
  23. package/packages/convex/package.json +12 -0
  24. package/packages/convex/src/convex.config.ts +3 -0
  25. package/packages/convex/src/index.ts +3 -0
  26. package/packages/convex/src/mutations.ts +36 -0
  27. package/packages/convex/src/queries.ts +19 -0
  28. package/packages/convex/src/schema.ts +24 -0
  29. package/packages/convex/tsconfig.json +25 -0
  30. package/packages/react/README.md +1 -0
  31. package/packages/react/package.json +46 -0
  32. package/packages/react/src/components/ApprovalModal.tsx +47 -0
  33. package/packages/react/src/components/StepConfigPanel.tsx +67 -0
  34. package/packages/react/src/components/StepConnector.tsx +47 -0
  35. package/packages/react/src/components/StepNode.tsx +38 -0
  36. package/packages/react/src/components/StepPalette.tsx +48 -0
  37. package/packages/react/src/components/WorkflowCanvas.tsx +42 -0
  38. package/packages/react/src/components/WorkflowRunPanel.tsx +64 -0
  39. package/packages/react/src/components/WorkflowToolbar.tsx +43 -0
  40. package/packages/react/src/components/index.ts +9 -0
  41. package/packages/react/src/hooks/index.ts +10 -0
  42. package/packages/react/src/hooks/useApprovalGate.ts +59 -0
  43. package/packages/react/src/hooks/useWorkflow.ts +39 -0
  44. package/packages/react/src/hooks/useWorkflowBuilder.ts +121 -0
  45. package/packages/react/src/hooks/useWorkflowRun.ts +75 -0
  46. package/packages/react/src/hooks/useWorkflowStep.ts +52 -0
  47. package/packages/react/src/hooks/useWorkflowTemplates.ts +54 -0
  48. package/packages/react/src/index.ts +16 -0
  49. package/packages/react/src/pages/WorkflowBuilderPage.tsx +81 -0
  50. package/packages/react/src/pages/WorkflowRunsPage.tsx +59 -0
  51. package/packages/react/src/pages/index.ts +3 -0
  52. package/packages/react/tsconfig.json +1 -0
  53. package/packages/react/tsup.config.ts +7 -0
  54. package/packages/react-css/README.md +1 -0
  55. package/packages/react-css/package.json +44 -0
  56. package/packages/react-css/src/components/ApprovalModal.tsx +6 -0
  57. package/packages/react-css/src/components/StepConfigPanel.tsx +7 -0
  58. package/packages/react-css/src/components/StepConnector.tsx +6 -0
  59. package/packages/react-css/src/components/StepNode.tsx +7 -0
  60. package/packages/react-css/src/components/StepPalette.tsx +6 -0
  61. package/packages/react-css/src/components/WorkflowCanvas.tsx +6 -0
  62. package/packages/react-css/src/components/WorkflowRunPanel.tsx +9 -0
  63. package/packages/react-css/src/components/WorkflowToolbar.tsx +4 -0
  64. package/packages/react-css/src/components/index.ts +9 -0
  65. package/packages/react-css/src/hooks/index.ts +3 -0
  66. package/packages/react-css/src/hooks/useWorkflow.ts +39 -0
  67. package/packages/react-css/src/hooks/useWorkflowBuilder.ts +121 -0
  68. package/packages/react-css/src/index.ts +7 -0
  69. package/packages/react-css/src/pages/WorkflowBuilderPage.tsx +16 -0
  70. package/packages/react-css/src/pages/WorkflowRunsPage.tsx +6 -0
  71. package/packages/react-css/src/pages/index.ts +3 -0
  72. package/packages/react-css/src/styles.css +945 -0
  73. package/packages/react-css/tsconfig.json +26 -0
  74. package/packages/react-css/tsup.config.ts +2 -0
  75. package/packages/shared/README.md +1 -0
  76. package/packages/shared/package.json +56 -0
  77. package/packages/shared/src/__tests__/ai-workflow.test.ts +217 -0
  78. package/packages/shared/src/config.ts +49 -0
  79. package/packages/shared/src/convex/index.ts +2 -0
  80. package/packages/shared/src/convex/schemas.ts +42 -0
  81. package/packages/shared/src/engine.test.ts +1 -0
  82. package/packages/shared/src/engine.ts +295 -0
  83. package/packages/shared/src/index.ts +43 -0
  84. package/packages/shared/src/steps.ts +68 -0
  85. package/packages/shared/src/templates.ts +172 -0
  86. package/packages/shared/src/types.ts +237 -0
  87. package/packages/shared/src/utils/cost.ts +79 -0
  88. package/packages/shared/src/utils/dag.ts +133 -0
  89. package/packages/shared/src/utils/index.ts +5 -0
  90. package/packages/shared/src/utils/interpolation.ts +53 -0
  91. package/packages/shared/src/validators.ts +215 -0
  92. package/packages/shared/tsconfig.json +1 -0
  93. package/packages/shared/tsup.config.ts +5 -0
  94. package/packages/shared/vitest.config.ts +4 -0
  95. package/packages/solidjs/README.md +1 -0
  96. package/packages/solidjs/package.json +45 -0
  97. package/packages/solidjs/src/components/ApprovalModal.tsx +18 -0
  98. package/packages/solidjs/src/components/StepConfigPanel.tsx +14 -0
  99. package/packages/solidjs/src/components/StepConnector.tsx +11 -0
  100. package/packages/solidjs/src/components/StepNode.tsx +12 -0
  101. package/packages/solidjs/src/components/StepPalette.tsx +22 -0
  102. package/packages/solidjs/src/components/WorkflowCanvas.tsx +23 -0
  103. package/packages/solidjs/src/components/WorkflowRunPanel.tsx +18 -0
  104. package/packages/solidjs/src/components/WorkflowToolbar.tsx +13 -0
  105. package/packages/solidjs/src/components/index.ts +9 -0
  106. package/packages/solidjs/src/index.ts +7 -0
  107. package/packages/solidjs/src/pages/WorkflowBuilderPage.tsx +37 -0
  108. package/packages/solidjs/src/pages/WorkflowRunsPage.tsx +20 -0
  109. package/packages/solidjs/src/pages/index.ts +3 -0
  110. package/packages/solidjs/src/primitives/createApprovalGate.ts +29 -0
  111. package/packages/solidjs/src/primitives/createWorkflow.ts +28 -0
  112. package/packages/solidjs/src/primitives/createWorkflowBuilder.ts +56 -0
  113. package/packages/solidjs/src/primitives/createWorkflowRun.ts +32 -0
  114. package/packages/solidjs/src/primitives/createWorkflowStep.ts +23 -0
  115. package/packages/solidjs/src/primitives/createWorkflowTemplates.ts +28 -0
  116. package/packages/solidjs/src/primitives/index.ts +8 -0
  117. package/packages/solidjs/tsconfig.json +1 -0
  118. package/packages/solidjs/tsup.config.ts +7 -0
  119. package/packages/solidjs-css/README.md +1 -0
  120. package/packages/solidjs-css/package.json +43 -0
  121. package/packages/solidjs-css/src/components/ApprovalModal.tsx +6 -0
  122. package/packages/solidjs-css/src/components/StepConfigPanel.tsx +7 -0
  123. package/packages/solidjs-css/src/components/StepConnector.tsx +6 -0
  124. package/packages/solidjs-css/src/components/StepNode.tsx +7 -0
  125. package/packages/solidjs-css/src/components/StepPalette.tsx +7 -0
  126. package/packages/solidjs-css/src/components/WorkflowCanvas.tsx +7 -0
  127. package/packages/solidjs-css/src/components/WorkflowRunPanel.tsx +8 -0
  128. package/packages/solidjs-css/src/components/WorkflowToolbar.tsx +5 -0
  129. package/packages/solidjs-css/src/components/index.ts +9 -0
  130. package/packages/solidjs-css/src/index.ts +7 -0
  131. package/packages/solidjs-css/src/pages/WorkflowBuilderPage.tsx +2 -0
  132. package/packages/solidjs-css/src/pages/WorkflowRunsPage.tsx +7 -0
  133. package/packages/solidjs-css/src/pages/index.ts +3 -0
  134. package/packages/solidjs-css/src/primitives/createWorkflow.ts +28 -0
  135. package/packages/solidjs-css/src/primitives/createWorkflowBuilder.ts +56 -0
  136. package/packages/solidjs-css/src/primitives/index.ts +1 -0
  137. package/packages/solidjs-css/src/styles.css +945 -0
  138. package/packages/solidjs-css/tsconfig.json +27 -0
  139. package/packages/solidjs-css/tsup.config.ts +2 -0
  140. package/pnpm-workspace.yaml +2 -0
@@ -0,0 +1,945 @@
1
+ /* @geenius-ai-workflow — styles.css
2
+ * Unified BEM stylesheet for all CSS variant packages.
3
+ * Uses OKLCH color tokens consistent with the Geenius design system.
4
+ */
5
+
6
+ /* ========================================
7
+ Color Tokens (OKLCH)
8
+ ======================================== */
9
+ :root {
10
+ --ai-workflow-bg: oklch(0.10 0.01 250);
11
+ --ai-workflow-surface: oklch(0.13 0.01 250);
12
+ --ai-workflow-surface-2: oklch(0.16 0.01 250);
13
+ --ai-workflow-border: oklch(0.22 0.01 250);
14
+ --ai-workflow-text: oklch(0.95 0.01 250);
15
+ --ai-workflow-text-muted: oklch(0.65 0.01 250);
16
+ --ai-workflow-accent: oklch(0.65 0.22 265);
17
+ --ai-workflow-running: oklch(0.65 0.20 265);
18
+ --ai-workflow-success: oklch(0.72 0.20 145);
19
+ --ai-workflow-error: oklch(0.60 0.25 25);
20
+ --ai-workflow-pending: oklch(0.65 0.15 200);
21
+ --ai-workflow-skipped: oklch(0.45 0.01 250);
22
+ --ai-workflow-connector: oklch(0.35 0.02 250);
23
+ --ai-workflow-radius: 0.75rem;
24
+ --ai-workflow-radius-sm: 0.5rem;
25
+ --ai-workflow-radius-lg: 1rem;
26
+ --ai-workflow-shadow: 0 4px 24px oklch(0 0 0 / 0.4);
27
+ --ai-workflow-shadow-sm: 0 2px 8px oklch(0 0 0 / 0.2);
28
+ --ai-workflow-shadow-lg: 0 16px 48px oklch(0 0 0 / 0.5);
29
+ --ai-workflow-transition: 200ms ease;
30
+ --ai-workflow-transition-fast: 100ms ease;
31
+ --ai-workflow-transition-slow: 300ms ease;
32
+ }
33
+
34
+ /* ========================================
35
+ BLOCK: ai-workflow__canvas
36
+ ======================================== */
37
+ .ai-workflow__canvas {
38
+ position: relative;
39
+ min-height: 500px;
40
+ background: var(--ai-workflow-surface);
41
+ border-radius: var(--ai-workflow-radius);
42
+ overflow: hidden;
43
+ border: 1px solid var(--ai-workflow-border);
44
+ box-shadow: var(--ai-workflow-shadow-sm);
45
+ }
46
+
47
+ .ai-workflow__canvas--empty {
48
+ display: flex;
49
+ flex-direction: column;
50
+ align-items: center;
51
+ justify-content: center;
52
+ height: 100%;
53
+ color: var(--ai-workflow-text-muted);
54
+ }
55
+
56
+ .ai-workflow__canvas-bg {
57
+ position: absolute;
58
+ inset: 0;
59
+ background-image:
60
+ linear-gradient(var(--ai-workflow-border) 1px, transparent 1px),
61
+ linear-gradient(90deg, var(--ai-workflow-border) 1px, transparent 1px);
62
+ background-size: 20px 20px;
63
+ pointer-events: none;
64
+ }
65
+
66
+ .ai-workflow__canvas-grid {
67
+ position: relative;
68
+ z-index: 1;
69
+ height: 100%;
70
+ }
71
+
72
+ /* ========================================
73
+ BLOCK: ai-workflow__step-node
74
+ ======================================== */
75
+ .ai-workflow__step-node {
76
+ position: relative;
77
+ background: var(--ai-workflow-surface-2);
78
+ border: 2px solid var(--ai-workflow-border);
79
+ border-radius: var(--ai-workflow-radius-sm);
80
+ padding: 1rem;
81
+ min-width: 180px;
82
+ cursor: pointer;
83
+ transition: all var(--ai-workflow-transition);
84
+ box-shadow: var(--ai-workflow-shadow-sm);
85
+ }
86
+
87
+ .ai-workflow__step-node:hover {
88
+ border-color: var(--ai-workflow-accent);
89
+ background: var(--ai-workflow-surface-2);
90
+ }
91
+
92
+ .ai-workflow__step-node--selected {
93
+ border-color: var(--ai-workflow-accent);
94
+ background: oklch(0.65 0.22 265 / 0.1);
95
+ box-shadow: 0 0 0 3px oklch(0.65 0.22 265 / 0.2), var(--ai-workflow-glow);
96
+ }
97
+
98
+ .ai-workflow__step-node--pending {
99
+ border-color: var(--ai-workflow-pending);
100
+ }
101
+
102
+ .ai-workflow__step-node--running {
103
+ border-color: var(--ai-workflow-running);
104
+ animation: ai-workflow-pulse-running 1.5s ease-in-out infinite;
105
+ }
106
+
107
+ .ai-workflow__step-node--success {
108
+ border-color: var(--ai-workflow-success);
109
+ }
110
+
111
+ .ai-workflow__step-node--error {
112
+ border-color: var(--ai-workflow-error);
113
+ background: oklch(0.25 0.05 25 / 0.1);
114
+ }
115
+
116
+ .ai-workflow__step-node--skipped {
117
+ border-color: var(--ai-workflow-skipped);
118
+ opacity: 0.7;
119
+ }
120
+
121
+ @keyframes ai-workflow-pulse-running {
122
+ 0%, 100% {
123
+ box-shadow: 0 0 0 0 oklch(0.65 0.20 265 / 0.5), var(--ai-workflow-shadow-sm);
124
+ }
125
+ 50% {
126
+ box-shadow: 0 0 0 8px oklch(0.65 0.20 265 / 0), var(--ai-workflow-shadow-sm);
127
+ }
128
+ }
129
+
130
+ .ai-workflow__step-header {
131
+ display: flex;
132
+ align-items: center;
133
+ gap: 0.5rem;
134
+ margin-bottom: 0.5rem;
135
+ }
136
+
137
+ .ai-workflow__step-icon {
138
+ font-size: 1.25rem;
139
+ flex-shrink: 0;
140
+ }
141
+
142
+ .ai-workflow__step-name {
143
+ font-weight: 600;
144
+ color: var(--ai-workflow-text);
145
+ flex: 1;
146
+ overflow: hidden;
147
+ text-overflow: ellipsis;
148
+ white-space: nowrap;
149
+ }
150
+
151
+ .ai-workflow__step-type-badge {
152
+ font-size: 0.7rem;
153
+ color: var(--ai-workflow-text-muted);
154
+ text-transform: uppercase;
155
+ letter-spacing: 0.05em;
156
+ background: var(--ai-workflow-surface);
157
+ padding: 0.2rem 0.4rem;
158
+ border-radius: 2px;
159
+ }
160
+
161
+ .ai-workflow__step-optional-badge,
162
+ .ai-workflow__step-retry-badge {
163
+ display: inline-block;
164
+ font-size: 0.65rem;
165
+ padding: 0.1rem 0.4rem;
166
+ border-radius: 999px;
167
+ background: var(--ai-workflow-surface);
168
+ color: var(--ai-workflow-text-muted);
169
+ font-weight: 500;
170
+ }
171
+
172
+ .ai-workflow__step-status {
173
+ display: flex;
174
+ align-items: center;
175
+ justify-content: center;
176
+ width: 1.5rem;
177
+ height: 1.5rem;
178
+ border-radius: 50%;
179
+ font-size: 0.8rem;
180
+ }
181
+
182
+ .ai-workflow__step-status--pending {
183
+ background: var(--ai-workflow-pending);
184
+ }
185
+
186
+ .ai-workflow__step-status--running {
187
+ background: var(--ai-workflow-running);
188
+ animation: ai-workflow-spin 1s linear infinite;
189
+ }
190
+
191
+ .ai-workflow__step-status--success {
192
+ background: var(--ai-workflow-success);
193
+ }
194
+
195
+ .ai-workflow__step-status--error {
196
+ background: var(--ai-workflow-error);
197
+ }
198
+
199
+ @keyframes ai-workflow-spin {
200
+ to {
201
+ transform: rotate(360deg);
202
+ }
203
+ }
204
+
205
+ /* ========================================
206
+ BLOCK: ai-workflow__connector
207
+ ======================================== */
208
+ .ai-workflow__connector {
209
+ position: absolute;
210
+ pointer-events: none;
211
+ stroke: var(--ai-workflow-connector);
212
+ stroke-width: 2px;
213
+ fill: none;
214
+ }
215
+
216
+ .ai-workflow__connector--running {
217
+ stroke: var(--ai-workflow-running);
218
+ animation: ai-workflow-flow 2s ease-in-out infinite;
219
+ }
220
+
221
+ .ai-workflow__connector--error {
222
+ stroke: var(--ai-workflow-error);
223
+ }
224
+
225
+ .ai-workflow__connector--success {
226
+ stroke: var(--ai-workflow-success);
227
+ }
228
+
229
+ @keyframes ai-workflow-flow {
230
+ 0%, 100% {
231
+ opacity: 0.5;
232
+ }
233
+ 50% {
234
+ opacity: 1;
235
+ }
236
+ }
237
+
238
+ /* ========================================
239
+ BLOCK: ai-workflow__step-detail-panel
240
+ ======================================== */
241
+ .ai-workflow__step-detail-panel {
242
+ background: var(--ai-workflow-surface-2);
243
+ border-left: 1px solid var(--ai-workflow-border);
244
+ padding: 1.5rem;
245
+ min-width: 300px;
246
+ max-width: 400px;
247
+ overflow-y: auto;
248
+ display: flex;
249
+ flex-direction: column;
250
+ gap: 1.5rem;
251
+ }
252
+
253
+ .ai-workflow__detail-header {
254
+ display: flex;
255
+ justify-content: space-between;
256
+ align-items: flex-start;
257
+ gap: 1rem;
258
+ border-bottom: 1px solid var(--ai-workflow-border);
259
+ padding-bottom: 1rem;
260
+ }
261
+
262
+ .ai-workflow__detail-title {
263
+ font-size: 1.1rem;
264
+ font-weight: 700;
265
+ color: var(--ai-workflow-text);
266
+ margin: 0;
267
+ }
268
+
269
+ .ai-workflow__detail-close {
270
+ background: transparent;
271
+ border: none;
272
+ color: var(--ai-workflow-text-muted);
273
+ cursor: pointer;
274
+ font-size: 1.25rem;
275
+ padding: 0;
276
+ transition: color var(--ai-workflow-transition);
277
+ }
278
+
279
+ .ai-workflow__detail-close:hover {
280
+ color: var(--ai-workflow-text);
281
+ }
282
+
283
+ .ai-workflow__detail-section {
284
+ display: flex;
285
+ flex-direction: column;
286
+ gap: 0.75rem;
287
+ }
288
+
289
+ .ai-workflow__detail-label {
290
+ font-size: 0.8rem;
291
+ font-weight: 600;
292
+ text-transform: uppercase;
293
+ letter-spacing: 0.05em;
294
+ color: var(--ai-workflow-text-muted);
295
+ }
296
+
297
+ .ai-workflow__detail-value {
298
+ font-size: 0.9rem;
299
+ color: var(--ai-workflow-text);
300
+ word-break: break-word;
301
+ }
302
+
303
+ .ai-workflow__detail-code {
304
+ background: var(--ai-workflow-surface);
305
+ padding: 0.75rem;
306
+ border-radius: var(--ai-workflow-radius-sm);
307
+ border: 1px solid var(--ai-workflow-border);
308
+ font-family: 'Courier New', monospace;
309
+ font-size: 0.8rem;
310
+ color: var(--ai-workflow-accent);
311
+ overflow-x: auto;
312
+ }
313
+
314
+ /* ========================================
315
+ BLOCK: ai-workflow__run-history-table
316
+ ======================================== */
317
+ .ai-workflow__run-history-table {
318
+ width: 100%;
319
+ border-collapse: collapse;
320
+ background: var(--ai-workflow-surface);
321
+ }
322
+
323
+ .ai-workflow__history-header {
324
+ background: var(--ai-workflow-surface-2);
325
+ border-bottom: 2px solid var(--ai-workflow-border);
326
+ }
327
+
328
+ .ai-workflow__history-th {
329
+ padding: 1rem;
330
+ text-align: left;
331
+ font-weight: 600;
332
+ font-size: 0.8rem;
333
+ color: var(--ai-workflow-text-muted);
334
+ text-transform: uppercase;
335
+ letter-spacing: 0.05em;
336
+ }
337
+
338
+ .ai-workflow__history-row {
339
+ border-bottom: 1px solid var(--ai-workflow-border);
340
+ transition: background var(--ai-workflow-transition);
341
+ }
342
+
343
+ .ai-workflow__history-row:hover {
344
+ background: var(--ai-workflow-surface-2);
345
+ }
346
+
347
+ .ai-workflow__history-td {
348
+ padding: 1rem;
349
+ font-size: 0.9rem;
350
+ color: var(--ai-workflow-text);
351
+ }
352
+
353
+ .ai-workflow__history-status-badge {
354
+ display: inline-flex;
355
+ align-items: center;
356
+ gap: 0.5rem;
357
+ padding: 0.35rem 0.75rem;
358
+ border-radius: 999px;
359
+ font-size: 0.8rem;
360
+ font-weight: 600;
361
+ }
362
+
363
+ .ai-workflow__history-status-badge--pending {
364
+ background: var(--ai-workflow-pending);
365
+ color: var(--ai-workflow-text);
366
+ }
367
+
368
+ .ai-workflow__history-status-badge--running {
369
+ background: var(--ai-workflow-running);
370
+ color: var(--ai-workflow-text);
371
+ }
372
+
373
+ .ai-workflow__history-status-badge--success {
374
+ background: var(--ai-workflow-success);
375
+ color: var(--ai-workflow-text);
376
+ }
377
+
378
+ .ai-workflow__history-status-badge--error {
379
+ background: var(--ai-workflow-error);
380
+ color: white;
381
+ }
382
+
383
+ .ai-workflow__history-time {
384
+ font-size: 0.8rem;
385
+ color: var(--ai-workflow-text-muted);
386
+ }
387
+
388
+ /* ========================================
389
+ BLOCK: ai-workflow__trigger-selector
390
+ ======================================== */
391
+ .ai-workflow__trigger-selector {
392
+ display: flex;
393
+ flex-direction: column;
394
+ gap: 1rem;
395
+ padding: 1rem;
396
+ background: var(--ai-workflow-surface);
397
+ border: 1px solid var(--ai-workflow-border);
398
+ border-radius: var(--ai-workflow-radius-sm);
399
+ }
400
+
401
+ .ai-workflow__trigger-title {
402
+ font-weight: 600;
403
+ color: var(--ai-workflow-text);
404
+ font-size: 0.9rem;
405
+ }
406
+
407
+ .ai-workflow__trigger-option {
408
+ padding: 0.75rem;
409
+ background: var(--ai-workflow-surface-2);
410
+ border: 1px solid var(--ai-workflow-border);
411
+ border-radius: var(--ai-workflow-radius-sm);
412
+ cursor: pointer;
413
+ transition: all var(--ai-workflow-transition);
414
+ display: flex;
415
+ align-items: center;
416
+ gap: 0.5rem;
417
+ }
418
+
419
+ .ai-workflow__trigger-option:hover {
420
+ border-color: var(--ai-workflow-accent);
421
+ }
422
+
423
+ .ai-workflow__trigger-option--selected {
424
+ border-color: var(--ai-workflow-accent);
425
+ background: oklch(0.65 0.22 265 / 0.1);
426
+ }
427
+
428
+ .ai-workflow__trigger-radio {
429
+ width: 1rem;
430
+ height: 1rem;
431
+ border-radius: 50%;
432
+ border: 2px solid var(--ai-workflow-border);
433
+ display: flex;
434
+ align-items: center;
435
+ justify-content: center;
436
+ transition: all var(--ai-workflow-transition);
437
+ }
438
+
439
+ .ai-workflow__trigger-option--selected .ai-workflow__trigger-radio {
440
+ border-color: var(--ai-workflow-accent);
441
+ background: var(--ai-workflow-accent);
442
+ }
443
+
444
+ .ai-workflow__trigger-label {
445
+ flex: 1;
446
+ color: var(--ai-workflow-text);
447
+ font-size: 0.9rem;
448
+ font-weight: 500;
449
+ }
450
+
451
+ /* ========================================
452
+ BLOCK: ai-workflow__execution-log
453
+ ======================================== */
454
+ .ai-workflow__execution-log {
455
+ background: var(--ai-workflow-surface);
456
+ border: 1px solid var(--ai-workflow-border);
457
+ border-radius: var(--ai-workflow-radius-sm);
458
+ padding: 1rem;
459
+ font-family: 'Courier New', monospace;
460
+ font-size: 0.8rem;
461
+ max-height: 400px;
462
+ overflow-y: auto;
463
+ }
464
+
465
+ .ai-workflow__log-line {
466
+ margin: 0.5rem 0;
467
+ color: var(--ai-workflow-text-muted);
468
+ white-space: pre-wrap;
469
+ word-wrap: break-word;
470
+ }
471
+
472
+ .ai-workflow__log-line--info {
473
+ color: var(--ai-workflow-accent);
474
+ }
475
+
476
+ .ai-workflow__log-line--success {
477
+ color: var(--ai-workflow-success);
478
+ }
479
+
480
+ .ai-workflow__log-line--error {
481
+ color: var(--ai-workflow-error);
482
+ }
483
+
484
+ .ai-workflow__log-line--warning {
485
+ color: oklch(0.75 0.18 70);
486
+ }
487
+
488
+ .ai-workflow__log-timestamp {
489
+ color: var(--ai-workflow-border);
490
+ margin-right: 0.5rem;
491
+ }
492
+
493
+ /* ========================================
494
+ BLOCK: ai-workflow__progress-tracker
495
+ ======================================== */
496
+ .ai-workflow__progress-tracker {
497
+ display: flex;
498
+ align-items: center;
499
+ gap: 1rem;
500
+ padding: 1rem;
501
+ background: var(--ai-workflow-surface);
502
+ border: 1px solid var(--ai-workflow-border);
503
+ border-radius: var(--ai-workflow-radius-sm);
504
+ }
505
+
506
+ .ai-workflow__progress-bar {
507
+ flex: 1;
508
+ height: 0.5rem;
509
+ background: var(--ai-workflow-surface-2);
510
+ border-radius: 999px;
511
+ overflow: hidden;
512
+ }
513
+
514
+ .ai-workflow__progress-fill {
515
+ height: 100%;
516
+ background: linear-gradient(90deg, var(--ai-workflow-accent), var(--ai-workflow-running));
517
+ border-radius: 999px;
518
+ animation: ai-workflow-progress 2s ease-in-out;
519
+ }
520
+
521
+ .ai-workflow__progress-fill--complete {
522
+ background: var(--ai-workflow-success);
523
+ animation: none;
524
+ }
525
+
526
+ @keyframes ai-workflow-progress {
527
+ 0% {
528
+ width: 0%;
529
+ }
530
+ }
531
+
532
+ .ai-workflow__progress-text {
533
+ font-size: 0.85rem;
534
+ color: var(--ai-workflow-text);
535
+ font-weight: 600;
536
+ min-width: 60px;
537
+ text-align: right;
538
+ }
539
+
540
+ /* ========================================
541
+ BLOCK: ai-workflow__step-list
542
+ ======================================== */
543
+ .ai-workflow__step-list {
544
+ list-style: none;
545
+ padding: 0;
546
+ margin: 0;
547
+ display: flex;
548
+ flex-direction: column;
549
+ gap: 0.5rem;
550
+ }
551
+
552
+ .ai-workflow__step-list-item {
553
+ padding: 0.75rem 1rem;
554
+ background: var(--ai-workflow-surface);
555
+ border: 1px solid var(--ai-workflow-border);
556
+ border-radius: var(--ai-workflow-radius-sm);
557
+ display: flex;
558
+ align-items: center;
559
+ gap: 0.75rem;
560
+ transition: all var(--ai-workflow-transition);
561
+ cursor: pointer;
562
+ }
563
+
564
+ .ai-workflow__step-list-item:hover {
565
+ background: var(--ai-workflow-surface-2);
566
+ border-color: var(--ai-workflow-accent);
567
+ }
568
+
569
+ .ai-workflow__step-list-item--selected {
570
+ border-color: var(--ai-workflow-accent);
571
+ background: oklch(0.65 0.22 265 / 0.1);
572
+ }
573
+
574
+ .ai-workflow__step-list-number {
575
+ display: flex;
576
+ align-items: center;
577
+ justify-content: center;
578
+ width: 1.75rem;
579
+ height: 1.75rem;
580
+ border-radius: 50%;
581
+ background: var(--ai-workflow-surface-2);
582
+ font-size: 0.8rem;
583
+ font-weight: 600;
584
+ color: var(--ai-workflow-text);
585
+ flex-shrink: 0;
586
+ }
587
+
588
+ .ai-workflow__step-list-item--success .ai-workflow__step-list-number {
589
+ background: var(--ai-workflow-success);
590
+ }
591
+
592
+ .ai-workflow__step-list-item--error .ai-workflow__step-list-number {
593
+ background: var(--ai-workflow-error);
594
+ }
595
+
596
+ .ai-workflow__step-list-label {
597
+ flex: 1;
598
+ font-size: 0.9rem;
599
+ color: var(--ai-workflow-text);
600
+ font-weight: 500;
601
+ }
602
+
603
+ /* ========================================
604
+ BLOCK: ai-workflow__variable-editor
605
+ ======================================== */
606
+ .ai-workflow__variable-editor {
607
+ background: var(--ai-workflow-surface);
608
+ border: 1px solid var(--ai-workflow-border);
609
+ border-radius: var(--ai-workflow-radius-sm);
610
+ padding: 1rem;
611
+ display: flex;
612
+ flex-direction: column;
613
+ gap: 1rem;
614
+ }
615
+
616
+ .ai-workflow__variable-header {
617
+ font-weight: 600;
618
+ color: var(--ai-workflow-text);
619
+ font-size: 0.9rem;
620
+ }
621
+
622
+ .ai-workflow__variable-item {
623
+ display: flex;
624
+ gap: 0.75rem;
625
+ }
626
+
627
+ .ai-workflow__variable-name {
628
+ flex: 0 0 120px;
629
+ padding: 0.5rem;
630
+ background: var(--ai-workflow-surface-2);
631
+ border: 1px solid var(--ai-workflow-border);
632
+ border-radius: var(--ai-workflow-radius-sm);
633
+ color: var(--ai-workflow-text);
634
+ font-size: 0.85rem;
635
+ font-family: 'Courier New', monospace;
636
+ }
637
+
638
+ .ai-workflow__variable-value {
639
+ flex: 1;
640
+ padding: 0.5rem;
641
+ background: var(--ai-workflow-surface-2);
642
+ border: 1px solid var(--ai-workflow-border);
643
+ border-radius: var(--ai-workflow-radius-sm);
644
+ color: var(--ai-workflow-text);
645
+ font-size: 0.85rem;
646
+ }
647
+
648
+ .ai-workflow__variable-value:focus {
649
+ outline: none;
650
+ border-color: var(--ai-workflow-accent);
651
+ }
652
+
653
+ /* ========================================
654
+ BLOCK: ai-workflow__condition-block
655
+ ======================================== */
656
+ .ai-workflow__condition-block {
657
+ background: var(--ai-workflow-surface);
658
+ border: 1px solid var(--ai-workflow-border);
659
+ border-radius: var(--ai-workflow-radius-sm);
660
+ padding: 1rem;
661
+ display: flex;
662
+ flex-direction: column;
663
+ gap: 0.75rem;
664
+ }
665
+
666
+ .ai-workflow__condition-label {
667
+ font-weight: 600;
668
+ color: var(--ai-workflow-text);
669
+ font-size: 0.9rem;
670
+ }
671
+
672
+ .ai-workflow__condition-row {
673
+ display: flex;
674
+ gap: 0.5rem;
675
+ align-items: center;
676
+ }
677
+
678
+ .ai-workflow__condition-field {
679
+ flex: 1;
680
+ padding: 0.5rem;
681
+ background: var(--ai-workflow-surface-2);
682
+ border: 1px solid var(--ai-workflow-border);
683
+ border-radius: var(--ai-workflow-radius-sm);
684
+ color: var(--ai-workflow-text);
685
+ font-size: 0.85rem;
686
+ }
687
+
688
+ .ai-workflow__condition-field:focus {
689
+ outline: none;
690
+ border-color: var(--ai-workflow-accent);
691
+ }
692
+
693
+ .ai-workflow__condition-operator {
694
+ flex: 0 0 80px;
695
+ padding: 0.5rem;
696
+ background: var(--ai-workflow-surface-2);
697
+ border: 1px solid var(--ai-workflow-border);
698
+ border-radius: var(--ai-workflow-radius-sm);
699
+ color: var(--ai-workflow-text);
700
+ font-size: 0.85rem;
701
+ text-align: center;
702
+ }
703
+
704
+ /* ========================================
705
+ BLOCK: ai-workflow__loop-indicator
706
+ ======================================== */
707
+ .ai-workflow__loop-indicator {
708
+ display: inline-flex;
709
+ align-items: center;
710
+ gap: 0.5rem;
711
+ padding: 0.35rem 0.75rem;
712
+ background: var(--ai-workflow-accent);
713
+ color: var(--ai-workflow-text);
714
+ border-radius: 999px;
715
+ font-size: 0.8rem;
716
+ font-weight: 600;
717
+ }
718
+
719
+ .ai-workflow__loop-icon {
720
+ font-size: 0.9rem;
721
+ animation: ai-workflow-spin 1s linear infinite;
722
+ }
723
+
724
+ /* ========================================
725
+ BLOCK: ai-workflow__parallel-badge
726
+ ======================================== */
727
+ .ai-workflow__parallel-badge {
728
+ display: inline-flex;
729
+ align-items: center;
730
+ gap: 0.5rem;
731
+ padding: 0.35rem 0.75rem;
732
+ background: oklch(0.72 0.20 145);
733
+ color: var(--ai-workflow-text);
734
+ border-radius: 999px;
735
+ font-size: 0.8rem;
736
+ font-weight: 600;
737
+ }
738
+
739
+ .ai-workflow__parallel-icon {
740
+ font-size: 0.9rem;
741
+ }
742
+
743
+ /* ========================================
744
+ STATES: Skeleton Loading
745
+ ======================================== */
746
+ .ai-workflow__skeleton {
747
+ background: linear-gradient(
748
+ 90deg,
749
+ var(--ai-workflow-surface-2) 0%,
750
+ var(--ai-workflow-border) 50%,
751
+ var(--ai-workflow-surface-2) 100%
752
+ );
753
+ background-size: 200% 100%;
754
+ animation: ai-workflow-skeleton-loading 1.5s ease-in-out infinite;
755
+ border-radius: var(--ai-workflow-radius-sm);
756
+ }
757
+
758
+ .ai-workflow__skeleton--text {
759
+ height: 1rem;
760
+ margin: 0.5rem 0;
761
+ }
762
+
763
+ .ai-workflow__skeleton--node {
764
+ height: 5rem;
765
+ width: 180px;
766
+ }
767
+
768
+ .ai-workflow__skeleton--card {
769
+ height: 6rem;
770
+ }
771
+
772
+ @keyframes ai-workflow-skeleton-loading {
773
+ 0% {
774
+ background-position: 200% 0;
775
+ }
776
+ 100% {
777
+ background-position: -200% 0;
778
+ }
779
+ }
780
+
781
+ /* ========================================
782
+ STATES: Empty State
783
+ ======================================== */
784
+ .ai-workflow__empty-state {
785
+ display: flex;
786
+ flex-direction: column;
787
+ align-items: center;
788
+ justify-content: center;
789
+ gap: 1rem;
790
+ padding: 3rem 2rem;
791
+ text-align: center;
792
+ min-height: 12rem;
793
+ background: var(--ai-workflow-surface);
794
+ border: 2px dashed var(--ai-workflow-border);
795
+ border-radius: var(--ai-workflow-radius);
796
+ }
797
+
798
+ .ai-workflow__empty-icon {
799
+ font-size: 3rem;
800
+ opacity: 0.3;
801
+ }
802
+
803
+ .ai-workflow__empty-title {
804
+ margin: 0;
805
+ font-size: 1.1rem;
806
+ font-weight: 600;
807
+ color: var(--ai-workflow-text);
808
+ }
809
+
810
+ .ai-workflow__empty-description {
811
+ margin: 0;
812
+ color: var(--ai-workflow-text-muted);
813
+ font-size: 0.9rem;
814
+ max-width: 300px;
815
+ }
816
+
817
+ /* ========================================
818
+ RESPONSIVE: Tablet
819
+ ======================================== */
820
+ @media (max-width: 768px) {
821
+ .ai-workflow__step-detail-panel {
822
+ max-width: none;
823
+ min-width: 250px;
824
+ }
825
+
826
+ .ai-workflow__run-history-table {
827
+ font-size: 0.85rem;
828
+ }
829
+
830
+ .ai-workflow__history-th,
831
+ .ai-workflow__history-td {
832
+ padding: 0.75rem;
833
+ }
834
+
835
+ .ai-workflow__execution-log {
836
+ max-height: 300px;
837
+ font-size: 0.75rem;
838
+ }
839
+ }
840
+
841
+ /* ========================================
842
+ RESPONSIVE: Mobile
843
+ ======================================== */
844
+ @media (max-width: 480px) {
845
+ .ai-workflow__canvas {
846
+ min-height: 300px;
847
+ }
848
+
849
+ .ai-workflow__step-detail-panel {
850
+ min-width: 100%;
851
+ max-width: none;
852
+ border-left: none;
853
+ border-top: 1px solid var(--ai-workflow-border);
854
+ }
855
+
856
+ .ai-workflow__step-node {
857
+ min-width: 150px;
858
+ padding: 0.75rem;
859
+ }
860
+
861
+ .ai-workflow__run-history-table {
862
+ font-size: 0.75rem;
863
+ }
864
+
865
+ .ai-workflow__history-th,
866
+ .ai-workflow__history-td {
867
+ padding: 0.5rem;
868
+ }
869
+
870
+ .ai-workflow__condition-row {
871
+ flex-direction: column;
872
+ }
873
+
874
+ .ai-workflow__condition-field,
875
+ .ai-workflow__condition-operator {
876
+ width: 100%;
877
+ }
878
+
879
+ .ai-workflow__execution-log {
880
+ font-size: 0.7rem;
881
+ max-height: 250px;
882
+ }
883
+ }
884
+
885
+ /* ========================================
886
+ UTILITY: Accessibility
887
+ ======================================== */
888
+ .ai-workflow__focus-visible:focus-visible {
889
+ outline: 2px solid var(--ai-workflow-accent);
890
+ outline-offset: 2px;
891
+ }
892
+
893
+ @media (prefers-reduced-motion: reduce) {
894
+ .ai-workflow__step-node--running,
895
+ .ai-workflow__step-status--running,
896
+ .ai-workflow__loop-icon,
897
+ .ai-workflow__progress-fill,
898
+ .ai-workflow__skeleton,
899
+ .ai-workflow__connector--running {
900
+ animation: none;
901
+ transition: none;
902
+ }
903
+ }
904
+
905
+ /* ========================================
906
+ UTILITY: Dark Mode
907
+ ======================================== */
908
+ @media (prefers-color-scheme: dark) {
909
+ :root {
910
+ --ai-workflow-bg: oklch(0.08 0.01 250);
911
+ --ai-workflow-surface: oklch(0.11 0.01 250);
912
+ --ai-workflow-surface-2: oklch(0.14 0.01 250);
913
+ --ai-workflow-border: oklch(0.20 0.01 250);
914
+ --ai-workflow-text: oklch(0.97 0.01 250);
915
+ }
916
+ }
917
+
918
+ /* ========================================
919
+ UTILITY: Light Mode
920
+ ======================================== */
921
+ @media (prefers-color-scheme: light) {
922
+ :root {
923
+ --ai-workflow-bg: oklch(0.98 0.01 250);
924
+ --ai-workflow-surface: oklch(0.93 0.01 250);
925
+ --ai-workflow-surface-2: oklch(0.88 0.01 250);
926
+ --ai-workflow-border: oklch(0.75 0.01 250);
927
+ --ai-workflow-text: oklch(0.15 0.01 250);
928
+ --ai-workflow-text-muted: oklch(0.40 0.01 250);
929
+ }
930
+ }
931
+
932
+ /* ========================================
933
+ UTILITY: Print Styles
934
+ ======================================== */
935
+ @media print {
936
+ .ai-workflow__canvas {
937
+ break-inside: avoid;
938
+ page-break-inside: avoid;
939
+ }
940
+
941
+ .ai-workflow__step-node {
942
+ break-inside: avoid;
943
+ page-break-inside: avoid;
944
+ }
945
+ }