flock-core 0.5.0b50__py3-none-any.whl → 0.5.0b52__py3-none-any.whl

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.

Potentially problematic release.


This version of flock-core might be problematic. Click here for more details.

Files changed (117) hide show
  1. flock/dashboard/launcher.py +1 -1
  2. flock/frontend/README.md +678 -0
  3. flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
  4. flock/frontend/index.html +12 -0
  5. flock/frontend/package-lock.json +4347 -0
  6. flock/frontend/package.json +48 -0
  7. flock/frontend/src/App.tsx +79 -0
  8. flock/frontend/src/__tests__/e2e/critical-scenarios.test.tsx +587 -0
  9. flock/frontend/src/__tests__/integration/filtering-e2e.test.tsx +387 -0
  10. flock/frontend/src/__tests__/integration/graph-rendering.test.tsx +640 -0
  11. flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
  12. flock/frontend/src/components/common/BuildInfo.tsx +39 -0
  13. flock/frontend/src/components/common/EmptyState.module.css +115 -0
  14. flock/frontend/src/components/common/EmptyState.tsx +128 -0
  15. flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
  16. flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
  17. flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
  18. flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
  19. flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
  20. flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
  21. flock/frontend/src/components/controls/PublishControl.css +547 -0
  22. flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
  23. flock/frontend/src/components/controls/PublishControl.tsx +432 -0
  24. flock/frontend/src/components/details/DetailWindowContainer.tsx +62 -0
  25. flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
  26. flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
  27. flock/frontend/src/components/details/MessageHistoryTab.tsx +299 -0
  28. flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
  29. flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
  30. flock/frontend/src/components/details/RunStatusTab.tsx +307 -0
  31. flock/frontend/src/components/details/tabs.test.tsx +1015 -0
  32. flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
  33. flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
  34. flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
  35. flock/frontend/src/components/filters/FilterBar.module.css +29 -0
  36. flock/frontend/src/components/filters/FilterBar.test.tsx +133 -0
  37. flock/frontend/src/components/filters/FilterBar.tsx +33 -0
  38. flock/frontend/src/components/filters/FilterPills.module.css +79 -0
  39. flock/frontend/src/components/filters/FilterPills.test.tsx +173 -0
  40. flock/frontend/src/components/filters/FilterPills.tsx +67 -0
  41. flock/frontend/src/components/filters/TimeRangeFilter.module.css +91 -0
  42. flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
  43. flock/frontend/src/components/filters/TimeRangeFilter.tsx +105 -0
  44. flock/frontend/src/components/graph/AgentNode.test.tsx +75 -0
  45. flock/frontend/src/components/graph/AgentNode.tsx +322 -0
  46. flock/frontend/src/components/graph/GraphCanvas.tsx +406 -0
  47. flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
  48. flock/frontend/src/components/graph/MessageNode.test.tsx +62 -0
  49. flock/frontend/src/components/graph/MessageNode.tsx +116 -0
  50. flock/frontend/src/components/graph/MiniMap.tsx +47 -0
  51. flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
  52. flock/frontend/src/components/layout/DashboardLayout.css +407 -0
  53. flock/frontend/src/components/layout/DashboardLayout.tsx +300 -0
  54. flock/frontend/src/components/layout/Header.module.css +88 -0
  55. flock/frontend/src/components/layout/Header.tsx +52 -0
  56. flock/frontend/src/components/modules/EventLogModule.test.tsx +401 -0
  57. flock/frontend/src/components/modules/EventLogModule.tsx +396 -0
  58. flock/frontend/src/components/modules/EventLogModuleWrapper.tsx +17 -0
  59. flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
  60. flock/frontend/src/components/modules/ModuleRegistry.ts +85 -0
  61. flock/frontend/src/components/modules/ModuleWindow.tsx +155 -0
  62. flock/frontend/src/components/modules/registerModules.ts +20 -0
  63. flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
  64. flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
  65. flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
  66. flock/frontend/src/components/settings/SettingsPanel.css +327 -0
  67. flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
  68. flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
  69. flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
  70. flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
  71. flock/frontend/src/hooks/useModulePersistence.ts +154 -0
  72. flock/frontend/src/hooks/useModules.ts +139 -0
  73. flock/frontend/src/hooks/usePersistence.ts +139 -0
  74. flock/frontend/src/main.tsx +13 -0
  75. flock/frontend/src/services/api.ts +213 -0
  76. flock/frontend/src/services/indexeddb.test.ts +793 -0
  77. flock/frontend/src/services/indexeddb.ts +794 -0
  78. flock/frontend/src/services/layout.test.ts +437 -0
  79. flock/frontend/src/services/layout.ts +146 -0
  80. flock/frontend/src/services/themeApplicator.ts +140 -0
  81. flock/frontend/src/services/themeService.ts +77 -0
  82. flock/frontend/src/services/websocket.test.ts +595 -0
  83. flock/frontend/src/services/websocket.ts +685 -0
  84. flock/frontend/src/store/filterStore.test.ts +242 -0
  85. flock/frontend/src/store/filterStore.ts +103 -0
  86. flock/frontend/src/store/graphStore.test.ts +186 -0
  87. flock/frontend/src/store/graphStore.ts +414 -0
  88. flock/frontend/src/store/moduleStore.test.ts +253 -0
  89. flock/frontend/src/store/moduleStore.ts +57 -0
  90. flock/frontend/src/store/settingsStore.ts +188 -0
  91. flock/frontend/src/store/streamStore.ts +68 -0
  92. flock/frontend/src/store/uiStore.test.ts +54 -0
  93. flock/frontend/src/store/uiStore.ts +110 -0
  94. flock/frontend/src/store/wsStore.ts +34 -0
  95. flock/frontend/src/styles/index.css +15 -0
  96. flock/frontend/src/styles/scrollbar.css +47 -0
  97. flock/frontend/src/styles/variables.css +488 -0
  98. flock/frontend/src/test/setup.ts +1 -0
  99. flock/frontend/src/types/filters.ts +14 -0
  100. flock/frontend/src/types/graph.ts +55 -0
  101. flock/frontend/src/types/modules.ts +7 -0
  102. flock/frontend/src/types/theme.ts +55 -0
  103. flock/frontend/src/utils/mockData.ts +85 -0
  104. flock/frontend/src/utils/performance.ts +16 -0
  105. flock/frontend/src/utils/transforms.test.ts +860 -0
  106. flock/frontend/src/utils/transforms.ts +323 -0
  107. flock/frontend/src/vite-env.d.ts +17 -0
  108. flock/frontend/tsconfig.json +27 -0
  109. flock/frontend/tsconfig.node.json +11 -0
  110. flock/frontend/vite.config.ts +25 -0
  111. flock/frontend/vitest.config.ts +11 -0
  112. flock/helper/cli_helper.py +1 -1
  113. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/METADATA +1 -1
  114. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/RECORD +117 -7
  115. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/WHEEL +0 -0
  116. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/entry_points.txt +0 -0
  117. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,432 @@
1
+ import React, { useState, useEffect } from 'react';
2
+ import { fetchArtifactTypes, publishArtifact, ArtifactType } from '../../services/api';
3
+ import { useFilterStore } from '../../store/filterStore';
4
+ import { useSettingsStore } from '../../store/settingsStore';
5
+ import './PublishControl.css';
6
+
7
+ interface ValidationErrors {
8
+ artifactType?: string;
9
+ [key: string]: string | undefined; // Allow dynamic field errors
10
+ }
11
+
12
+ interface FormData {
13
+ [key: string]: any;
14
+ }
15
+
16
+ const PublishControl: React.FC = () => {
17
+ const [artifactTypes, setArtifactTypes] = useState<ArtifactType[]>([]);
18
+ const [selectedType, setSelectedType] = useState('');
19
+ const [formData, setFormData] = useState<FormData>({});
20
+ const [loading, setLoading] = useState(false);
21
+ const [loadingTypes, setLoadingTypes] = useState(true);
22
+ const [errors, setErrors] = useState<ValidationErrors>({});
23
+ const [successMessage, setSuccessMessage] = useState('');
24
+ const [errorMessage, setErrorMessage] = useState('');
25
+ const [autoSetFilter, setAutoSetFilter] = useState(true); // Default: auto-set filter to correlation ID
26
+
27
+ const setShowControls = useSettingsStore((state) => state.setShowControls);
28
+
29
+ const handleClose = () => {
30
+ setShowControls(false);
31
+ };
32
+
33
+ // Fetch artifact types on mount
34
+ useEffect(() => {
35
+ const loadArtifactTypes = async () => {
36
+ try {
37
+ setLoadingTypes(true);
38
+ const types = await fetchArtifactTypes();
39
+ setArtifactTypes(types);
40
+ } catch (error) {
41
+ console.error('Failed to fetch artifact types:', error);
42
+ setErrorMessage('Failed to load artifact types');
43
+ } finally {
44
+ setLoadingTypes(false);
45
+ }
46
+ };
47
+
48
+ loadArtifactTypes();
49
+ }, []);
50
+
51
+ // Reset form data when artifact type changes
52
+ useEffect(() => {
53
+ if (selectedType) {
54
+ const artifactType = artifactTypes.find((t) => t.name === selectedType);
55
+ if (artifactType && artifactType.schema.properties) {
56
+ // Initialize formData with empty values based on schema
57
+ const initialData: FormData = {};
58
+ Object.keys(artifactType.schema.properties).forEach((key) => {
59
+ const prop = artifactType.schema.properties[key];
60
+ if (prop.type === 'boolean') {
61
+ initialData[key] = false;
62
+ } else if (prop.type === 'number' || prop.type === 'integer') {
63
+ initialData[key] = prop.default ?? '';
64
+ } else {
65
+ initialData[key] = prop.default ?? '';
66
+ }
67
+ });
68
+ setFormData(initialData);
69
+ }
70
+ } else {
71
+ setFormData({});
72
+ }
73
+ setErrors({});
74
+ }, [selectedType, artifactTypes]);
75
+
76
+ const validateForm = (): boolean => {
77
+ const newErrors: ValidationErrors = {};
78
+
79
+ if (!selectedType) {
80
+ newErrors.artifactType = 'Artifact type is required';
81
+ setErrors(newErrors);
82
+ return false;
83
+ }
84
+
85
+ const artifactType = artifactTypes.find((t) => t.name === selectedType);
86
+ if (!artifactType) {
87
+ newErrors.artifactType = 'Invalid artifact type';
88
+ setErrors(newErrors);
89
+ return false;
90
+ }
91
+
92
+ // Validate each field based on schema
93
+ if (artifactType.schema.properties) {
94
+ Object.entries(artifactType.schema.properties).forEach(([key, prop]: [string, any]) => {
95
+ const value = formData[key];
96
+
97
+ // Check required fields (you might need to check schema.required array)
98
+ if (value === '' || value === null || value === undefined) {
99
+ newErrors[key] = `${key} is required`;
100
+ return;
101
+ }
102
+
103
+ // Type validation
104
+ if (prop.type === 'number' || prop.type === 'integer') {
105
+ if (isNaN(Number(value))) {
106
+ newErrors[key] = `${key} must be a number`;
107
+ }
108
+ }
109
+ });
110
+ }
111
+
112
+ setErrors(newErrors);
113
+ return Object.keys(newErrors).length === 0;
114
+ };
115
+
116
+ const handleSubmit = async (e: React.FormEvent) => {
117
+ e.preventDefault();
118
+
119
+ // Clear previous messages
120
+ setSuccessMessage('');
121
+ setErrorMessage('');
122
+
123
+ // Validate form
124
+ if (!validateForm()) {
125
+ return;
126
+ }
127
+
128
+ setLoading(true);
129
+
130
+ try {
131
+ // Convert formData values to proper types
132
+ const artifactType = artifactTypes.find((t) => t.name === selectedType);
133
+ const processedData: any = {};
134
+
135
+ if (artifactType && artifactType.schema.properties) {
136
+ Object.entries(formData).forEach(([key, value]) => {
137
+ const prop = artifactType.schema.properties[key];
138
+ if (prop.type === 'number' || prop.type === 'integer') {
139
+ processedData[key] = Number(value);
140
+ } else if (prop.type === 'boolean') {
141
+ processedData[key] = Boolean(value);
142
+ } else {
143
+ processedData[key] = value;
144
+ }
145
+ });
146
+ }
147
+
148
+ const response = await publishArtifact(selectedType, processedData);
149
+
150
+ // Auto-set filter to correlation ID if checkbox is checked
151
+ if (autoSetFilter && response.correlation_id) {
152
+ useFilterStore.setState({ correlationId: response.correlation_id });
153
+ setSuccessMessage(`Successfully published artifact. Filter set to: ${response.correlation_id}`);
154
+ } else {
155
+ setSuccessMessage(`Successfully published artifact. Correlation ID: ${response.correlation_id}`);
156
+ }
157
+
158
+ // Reset form
159
+ setSelectedType('');
160
+ setFormData({});
161
+ setErrors({});
162
+
163
+ // Auto-close the panel after a brief delay to show success message
164
+ setTimeout(() => {
165
+ setShowControls(false);
166
+ }, 800);
167
+ } catch (error: any) {
168
+ // Handle errors
169
+ if (error.message.includes('Network error') || error.message.includes('Failed to connect')) {
170
+ setErrorMessage('Network error. Failed to connect to API server.');
171
+ } else {
172
+ setErrorMessage(error.message || 'Failed to publish artifact');
173
+ }
174
+ } finally {
175
+ setLoading(false);
176
+ }
177
+ };
178
+
179
+ // Helper to render form field based on schema property type
180
+ const renderField = (key: string, prop: any) => {
181
+ const value = formData[key] ?? '';
182
+ const hasError = !!errors[key];
183
+
184
+ const handleChange = (newValue: any) => {
185
+ setFormData({ ...formData, [key]: newValue });
186
+ if (errors[key]) {
187
+ const newErrors = { ...errors };
188
+ delete newErrors[key];
189
+ setErrors(newErrors);
190
+ }
191
+ };
192
+
193
+ if (prop.type === 'boolean') {
194
+ return (
195
+ <div key={key} className="publish-control__field">
196
+ <div className="publish-control__checkbox-wrapper">
197
+ <input
198
+ type="checkbox"
199
+ id={`field-${key}`}
200
+ checked={Boolean(value)}
201
+ onChange={(e) => handleChange(e.target.checked)}
202
+ disabled={loading}
203
+ className="publish-control__checkbox"
204
+ />
205
+ <label htmlFor={`field-${key}`} className="publish-control__checkbox-label">
206
+ {prop.title || key}
207
+ {prop.description && (
208
+ <span className="publish-control__field-hint"> — {prop.description}</span>
209
+ )}
210
+ </label>
211
+ </div>
212
+ {hasError && <div className="publish-control__error-text">{errors[key]}</div>}
213
+ </div>
214
+ );
215
+ }
216
+
217
+ if (prop.type === 'number' || prop.type === 'integer') {
218
+ return (
219
+ <div key={key} className="publish-control__field">
220
+ <label htmlFor={`field-${key}`} className="publish-control__label">
221
+ {prop.title || key}
222
+ {prop.description && (
223
+ <span className="publish-control__field-hint"> — {prop.description}</span>
224
+ )}
225
+ </label>
226
+ <input
227
+ type="number"
228
+ id={`field-${key}`}
229
+ value={value}
230
+ onChange={(e) => handleChange(e.target.value)}
231
+ disabled={loading}
232
+ step={prop.type === 'integer' ? 1 : 'any'}
233
+ className={`publish-control__input ${hasError ? 'publish-control__input--error' : ''}`}
234
+ />
235
+ {hasError && <div className="publish-control__error-text">{errors[key]}</div>}
236
+ </div>
237
+ );
238
+ }
239
+
240
+ if (prop.enum && Array.isArray(prop.enum)) {
241
+ return (
242
+ <div key={key} className="publish-control__field">
243
+ <label htmlFor={`field-${key}`} className="publish-control__label">
244
+ {prop.title || key}
245
+ {prop.description && (
246
+ <span className="publish-control__field-hint"> — {prop.description}</span>
247
+ )}
248
+ </label>
249
+ <select
250
+ id={`field-${key}`}
251
+ value={value}
252
+ onChange={(e) => handleChange(e.target.value)}
253
+ disabled={loading}
254
+ className={`publish-control__select ${hasError ? 'publish-control__select--error' : ''}`}
255
+ >
256
+ <option value="">Select...</option>
257
+ {prop.enum.map((option: any) => (
258
+ <option key={option} value={option}>
259
+ {option}
260
+ </option>
261
+ ))}
262
+ </select>
263
+ {hasError && <div className="publish-control__error-text">{errors[key]}</div>}
264
+ </div>
265
+ );
266
+ }
267
+
268
+ // Default: text input or textarea for strings
269
+ const isLongText = prop.maxLength > 100 || key.toLowerCase().includes('description');
270
+
271
+ if (isLongText) {
272
+ return (
273
+ <div key={key} className="publish-control__field">
274
+ <label htmlFor={`field-${key}`} className="publish-control__label">
275
+ {prop.title || key}
276
+ {prop.description && (
277
+ <span className="publish-control__field-hint"> — {prop.description}</span>
278
+ )}
279
+ </label>
280
+ <textarea
281
+ id={`field-${key}`}
282
+ value={value}
283
+ onChange={(e) => handleChange(e.target.value)}
284
+ disabled={loading}
285
+ rows={4}
286
+ className={`publish-control__textarea ${hasError ? 'publish-control__textarea--error' : ''}`}
287
+ />
288
+ {hasError && <div className="publish-control__error-text">{errors[key]}</div>}
289
+ </div>
290
+ );
291
+ }
292
+
293
+ return (
294
+ <div key={key} className="publish-control__field">
295
+ <label htmlFor={`field-${key}`} className="publish-control__label">
296
+ {prop.title || key}
297
+ {prop.description && (
298
+ <span className="publish-control__field-hint"> — {prop.description}</span>
299
+ )}
300
+ </label>
301
+ <input
302
+ type="text"
303
+ id={`field-${key}`}
304
+ value={value}
305
+ onChange={(e) => handleChange(e.target.value)}
306
+ disabled={loading}
307
+ placeholder={prop.examples?.[0] || ''}
308
+ className={`publish-control__input ${hasError ? 'publish-control__input--error' : ''}`}
309
+ />
310
+ {hasError && <div className="publish-control__error-text">{errors[key]}</div>}
311
+ </div>
312
+ );
313
+ };
314
+
315
+ return (
316
+ <div className="publish-control-panel">
317
+ <div className="publish-control-panel-inner">
318
+ {/* Header */}
319
+ <div className="publish-control-header">
320
+ <h2 className="publish-control-title">Publish Artifact</h2>
321
+ <button
322
+ onClick={handleClose}
323
+ className="publish-control-close-button"
324
+ aria-label="Close publish panel"
325
+ title="Close publish panel (Esc)"
326
+ >
327
+ <svg width="20" height="20" viewBox="0 0 20 20" fill="none">
328
+ <path
329
+ d="M15 5L5 15M5 5l10 10"
330
+ stroke="currentColor"
331
+ strokeWidth="2"
332
+ strokeLinecap="round"
333
+ strokeLinejoin="round"
334
+ />
335
+ </svg>
336
+ </button>
337
+ </div>
338
+
339
+ {/* Content */}
340
+ <div className="publish-control-content">
341
+ <form className="publish-control__form" onSubmit={handleSubmit}>
342
+ {/* Artifact Type Dropdown */}
343
+ <div className="publish-control__field">
344
+ <label htmlFor="artifact-type" className="publish-control__label">
345
+ Artifact Type
346
+ </label>
347
+ <select
348
+ id="artifact-type"
349
+ value={selectedType}
350
+ onChange={(e) => {
351
+ setSelectedType(e.target.value);
352
+ setErrors({ ...errors, artifactType: undefined });
353
+ }}
354
+ disabled={loadingTypes || loading}
355
+ className={`publish-control__select ${errors.artifactType ? 'publish-control__select--error' : ''}`}
356
+ >
357
+ <option value="">Select an artifact type...</option>
358
+ {artifactTypes.map((type) => (
359
+ <option key={type.name} value={type.name}>
360
+ {type.name}
361
+ </option>
362
+ ))}
363
+ </select>
364
+ {errors.artifactType && (
365
+ <div className="publish-control__error-text">
366
+ {errors.artifactType}
367
+ </div>
368
+ )}
369
+ </div>
370
+
371
+ {/* Dynamic Form Fields based on Schema */}
372
+ {selectedType && (() => {
373
+ const artifactType = artifactTypes.find((t) => t.name === selectedType);
374
+ if (artifactType && artifactType.schema.properties) {
375
+ return Object.entries(artifactType.schema.properties).map(([key, prop]) =>
376
+ renderField(key, prop)
377
+ );
378
+ }
379
+ return (
380
+ <div className="publish-control__field">
381
+ <p className="publish-control__hint">
382
+ Select an artifact type to see form fields
383
+ </p>
384
+ </div>
385
+ );
386
+ })()}
387
+
388
+ {/* Auto-set Filter Checkbox */}
389
+ <div className="publish-control__checkbox-wrapper">
390
+ <input
391
+ type="checkbox"
392
+ id="auto-set-filter"
393
+ checked={autoSetFilter}
394
+ onChange={(e) => setAutoSetFilter(e.target.checked)}
395
+ disabled={loading}
396
+ className="publish-control__checkbox"
397
+ />
398
+ <label htmlFor="auto-set-filter" className="publish-control__checkbox-label">
399
+ Set filter to correlation ID (show only this execution)
400
+ </label>
401
+ </div>
402
+
403
+ {/* Submit Button */}
404
+ <button
405
+ type="submit"
406
+ disabled={loading || loadingTypes}
407
+ className="publish-control__submit"
408
+ >
409
+ {loading ? 'Publishing...' : 'Publish Artifact'}
410
+ </button>
411
+
412
+ {/* Success Message */}
413
+ {successMessage && (
414
+ <div className="publish-control__message publish-control__message--success">
415
+ {successMessage}
416
+ </div>
417
+ )}
418
+
419
+ {/* Error Message */}
420
+ {errorMessage && (
421
+ <div className="publish-control__message publish-control__message--error">
422
+ {errorMessage}
423
+ </div>
424
+ )}
425
+ </form>
426
+ </div>
427
+ </div>
428
+ </div>
429
+ );
430
+ };
431
+
432
+ export default PublishControl;
@@ -0,0 +1,62 @@
1
+ import React from 'react';
2
+ import { useUIStore } from '../../store/uiStore';
3
+ import { useGraphStore } from '../../store/graphStore';
4
+ import NodeDetailWindow from './NodeDetailWindow';
5
+
6
+ /**
7
+ * Container component that renders all open detail windows.
8
+ * Manages multiple floating windows with independent drag/resize.
9
+ */
10
+ const DetailWindowContainer: React.FC = () => {
11
+ const detailWindows = useUIStore((state) => state.detailWindows);
12
+ const mode = useUIStore((state) => state.mode);
13
+ const agents = useGraphStore((state) => state.agents);
14
+ const messages = useGraphStore((state) => state.messages);
15
+
16
+ // Convert Map to array for rendering
17
+ const windowEntries = Array.from(detailWindows.entries());
18
+
19
+ return (
20
+ <div
21
+ style={{
22
+ position: 'fixed',
23
+ top: 0,
24
+ left: 0,
25
+ right: 0,
26
+ bottom: 0,
27
+ pointerEvents: 'none',
28
+ zIndex: 999,
29
+ }}
30
+ >
31
+ <div
32
+ style={{
33
+ position: 'relative',
34
+ width: '100%',
35
+ height: '100%',
36
+ pointerEvents: 'none',
37
+ }}
38
+ >
39
+ {windowEntries.map(([nodeId, _window]) => {
40
+ // Determine node type based on mode and what exists
41
+ let nodeType: 'agent' | 'message' = 'agent';
42
+
43
+ if (mode === 'agent') {
44
+ // In agent view, check if it's an agent
45
+ if (agents.has(nodeId)) {
46
+ nodeType = 'agent';
47
+ }
48
+ } else if (mode === 'blackboard') {
49
+ // In blackboard view, nodes are messages
50
+ if (messages.has(nodeId)) {
51
+ nodeType = 'message';
52
+ }
53
+ }
54
+
55
+ return <NodeDetailWindow key={nodeId} nodeId={nodeId} nodeType={nodeType} />;
56
+ })}
57
+ </div>
58
+ </div>
59
+ );
60
+ };
61
+
62
+ export default DetailWindowContainer;