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.
- flock/dashboard/launcher.py +1 -1
- flock/frontend/README.md +678 -0
- flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
- flock/frontend/index.html +12 -0
- flock/frontend/package-lock.json +4347 -0
- flock/frontend/package.json +48 -0
- flock/frontend/src/App.tsx +79 -0
- flock/frontend/src/__tests__/e2e/critical-scenarios.test.tsx +587 -0
- flock/frontend/src/__tests__/integration/filtering-e2e.test.tsx +387 -0
- flock/frontend/src/__tests__/integration/graph-rendering.test.tsx +640 -0
- flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
- flock/frontend/src/components/common/BuildInfo.tsx +39 -0
- flock/frontend/src/components/common/EmptyState.module.css +115 -0
- flock/frontend/src/components/common/EmptyState.tsx +128 -0
- flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
- flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
- flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
- flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
- flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
- flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
- flock/frontend/src/components/controls/PublishControl.css +547 -0
- flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
- flock/frontend/src/components/controls/PublishControl.tsx +432 -0
- flock/frontend/src/components/details/DetailWindowContainer.tsx +62 -0
- flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
- flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
- flock/frontend/src/components/details/MessageHistoryTab.tsx +299 -0
- flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
- flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
- flock/frontend/src/components/details/RunStatusTab.tsx +307 -0
- flock/frontend/src/components/details/tabs.test.tsx +1015 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
- flock/frontend/src/components/filters/FilterBar.module.css +29 -0
- flock/frontend/src/components/filters/FilterBar.test.tsx +133 -0
- flock/frontend/src/components/filters/FilterBar.tsx +33 -0
- flock/frontend/src/components/filters/FilterPills.module.css +79 -0
- flock/frontend/src/components/filters/FilterPills.test.tsx +173 -0
- flock/frontend/src/components/filters/FilterPills.tsx +67 -0
- flock/frontend/src/components/filters/TimeRangeFilter.module.css +91 -0
- flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
- flock/frontend/src/components/filters/TimeRangeFilter.tsx +105 -0
- flock/frontend/src/components/graph/AgentNode.test.tsx +75 -0
- flock/frontend/src/components/graph/AgentNode.tsx +322 -0
- flock/frontend/src/components/graph/GraphCanvas.tsx +406 -0
- flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
- flock/frontend/src/components/graph/MessageNode.test.tsx +62 -0
- flock/frontend/src/components/graph/MessageNode.tsx +116 -0
- flock/frontend/src/components/graph/MiniMap.tsx +47 -0
- flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
- flock/frontend/src/components/layout/DashboardLayout.css +407 -0
- flock/frontend/src/components/layout/DashboardLayout.tsx +300 -0
- flock/frontend/src/components/layout/Header.module.css +88 -0
- flock/frontend/src/components/layout/Header.tsx +52 -0
- flock/frontend/src/components/modules/EventLogModule.test.tsx +401 -0
- flock/frontend/src/components/modules/EventLogModule.tsx +396 -0
- flock/frontend/src/components/modules/EventLogModuleWrapper.tsx +17 -0
- flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
- flock/frontend/src/components/modules/ModuleRegistry.ts +85 -0
- flock/frontend/src/components/modules/ModuleWindow.tsx +155 -0
- flock/frontend/src/components/modules/registerModules.ts +20 -0
- flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
- flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
- flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
- flock/frontend/src/components/settings/SettingsPanel.css +327 -0
- flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
- flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
- flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
- flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
- flock/frontend/src/hooks/useModulePersistence.ts +154 -0
- flock/frontend/src/hooks/useModules.ts +139 -0
- flock/frontend/src/hooks/usePersistence.ts +139 -0
- flock/frontend/src/main.tsx +13 -0
- flock/frontend/src/services/api.ts +213 -0
- flock/frontend/src/services/indexeddb.test.ts +793 -0
- flock/frontend/src/services/indexeddb.ts +794 -0
- flock/frontend/src/services/layout.test.ts +437 -0
- flock/frontend/src/services/layout.ts +146 -0
- flock/frontend/src/services/themeApplicator.ts +140 -0
- flock/frontend/src/services/themeService.ts +77 -0
- flock/frontend/src/services/websocket.test.ts +595 -0
- flock/frontend/src/services/websocket.ts +685 -0
- flock/frontend/src/store/filterStore.test.ts +242 -0
- flock/frontend/src/store/filterStore.ts +103 -0
- flock/frontend/src/store/graphStore.test.ts +186 -0
- flock/frontend/src/store/graphStore.ts +414 -0
- flock/frontend/src/store/moduleStore.test.ts +253 -0
- flock/frontend/src/store/moduleStore.ts +57 -0
- flock/frontend/src/store/settingsStore.ts +188 -0
- flock/frontend/src/store/streamStore.ts +68 -0
- flock/frontend/src/store/uiStore.test.ts +54 -0
- flock/frontend/src/store/uiStore.ts +110 -0
- flock/frontend/src/store/wsStore.ts +34 -0
- flock/frontend/src/styles/index.css +15 -0
- flock/frontend/src/styles/scrollbar.css +47 -0
- flock/frontend/src/styles/variables.css +488 -0
- flock/frontend/src/test/setup.ts +1 -0
- flock/frontend/src/types/filters.ts +14 -0
- flock/frontend/src/types/graph.ts +55 -0
- flock/frontend/src/types/modules.ts +7 -0
- flock/frontend/src/types/theme.ts +55 -0
- flock/frontend/src/utils/mockData.ts +85 -0
- flock/frontend/src/utils/performance.ts +16 -0
- flock/frontend/src/utils/transforms.test.ts +860 -0
- flock/frontend/src/utils/transforms.ts +323 -0
- flock/frontend/src/vite-env.d.ts +17 -0
- flock/frontend/tsconfig.json +27 -0
- flock/frontend/tsconfig.node.json +11 -0
- flock/frontend/vite.config.ts +25 -0
- flock/frontend/vitest.config.ts +11 -0
- flock/helper/cli_helper.py +1 -1
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/METADATA +1 -1
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/RECORD +117 -7
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/WHEEL +0 -0
- {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/entry_points.txt +0 -0
- {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;
|