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,501 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Unit tests for NodeDetailWindow component.
|
|
3
|
+
*
|
|
4
|
+
* Tests verify window opening, dragging, resizing, closing, multiple window support,
|
|
5
|
+
* and state persistence for detail windows.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
9
|
+
import { render, screen, fireEvent } from '@testing-library/react';
|
|
10
|
+
import { useUIStore } from '../../store/uiStore';
|
|
11
|
+
|
|
12
|
+
// Mock component - will be replaced by actual implementation
|
|
13
|
+
const MockNodeDetailWindow = ({ nodeId }: { nodeId: string }) => {
|
|
14
|
+
const window = useUIStore((state) => state.detailWindows.get(nodeId));
|
|
15
|
+
const updateDetailWindow = useUIStore((state) => state.updateDetailWindow);
|
|
16
|
+
const closeDetailWindow = useUIStore((state) => state.closeDetailWindow);
|
|
17
|
+
|
|
18
|
+
if (!window) return null;
|
|
19
|
+
|
|
20
|
+
const handleDragStart = (e: React.MouseEvent) => {
|
|
21
|
+
const startX = e.clientX - window.position.x;
|
|
22
|
+
const startY = e.clientY - window.position.y;
|
|
23
|
+
|
|
24
|
+
const handleDrag = (moveEvent: MouseEvent) => {
|
|
25
|
+
updateDetailWindow(nodeId, {
|
|
26
|
+
position: {
|
|
27
|
+
x: moveEvent.clientX - startX,
|
|
28
|
+
y: moveEvent.clientY - startY,
|
|
29
|
+
},
|
|
30
|
+
});
|
|
31
|
+
};
|
|
32
|
+
|
|
33
|
+
const handleDragEnd = () => {
|
|
34
|
+
document.removeEventListener('mousemove', handleDrag);
|
|
35
|
+
document.removeEventListener('mouseup', handleDragEnd);
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
document.addEventListener('mousemove', handleDrag);
|
|
39
|
+
document.addEventListener('mouseup', handleDragEnd);
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
const handleResize = (e: React.MouseEvent) => {
|
|
43
|
+
e.stopPropagation();
|
|
44
|
+
const startWidth = window.size.width;
|
|
45
|
+
const startHeight = window.size.height;
|
|
46
|
+
const startX = e.clientX;
|
|
47
|
+
const startY = e.clientY;
|
|
48
|
+
|
|
49
|
+
const handleResizeMove = (moveEvent: MouseEvent) => {
|
|
50
|
+
const deltaX = moveEvent.clientX - startX;
|
|
51
|
+
const deltaY = moveEvent.clientY - startY;
|
|
52
|
+
|
|
53
|
+
updateDetailWindow(nodeId, {
|
|
54
|
+
size: {
|
|
55
|
+
width: Math.max(400, startWidth + deltaX),
|
|
56
|
+
height: Math.max(300, startHeight + deltaY),
|
|
57
|
+
},
|
|
58
|
+
});
|
|
59
|
+
};
|
|
60
|
+
|
|
61
|
+
const handleResizeEnd = () => {
|
|
62
|
+
document.removeEventListener('mousemove', handleResizeMove);
|
|
63
|
+
document.removeEventListener('mouseup', handleResizeEnd);
|
|
64
|
+
};
|
|
65
|
+
|
|
66
|
+
document.addEventListener('mousemove', handleResizeMove);
|
|
67
|
+
document.addEventListener('mouseup', handleResizeEnd);
|
|
68
|
+
};
|
|
69
|
+
|
|
70
|
+
return (
|
|
71
|
+
<div
|
|
72
|
+
data-testid={`detail-window-${nodeId}`}
|
|
73
|
+
style={{
|
|
74
|
+
position: 'absolute',
|
|
75
|
+
left: window.position.x,
|
|
76
|
+
top: window.position.y,
|
|
77
|
+
width: window.size.width,
|
|
78
|
+
height: window.size.height,
|
|
79
|
+
}}
|
|
80
|
+
>
|
|
81
|
+
<div
|
|
82
|
+
data-testid={`window-header-${nodeId}`}
|
|
83
|
+
onMouseDown={handleDragStart}
|
|
84
|
+
style={{ cursor: 'move', padding: '8px', background: '#333' }}
|
|
85
|
+
>
|
|
86
|
+
<span>Node: {nodeId}</span>
|
|
87
|
+
<button
|
|
88
|
+
data-testid={`close-button-${nodeId}`}
|
|
89
|
+
onClick={() => closeDetailWindow(nodeId)}
|
|
90
|
+
>
|
|
91
|
+
Close
|
|
92
|
+
</button>
|
|
93
|
+
</div>
|
|
94
|
+
<div data-testid={`window-content-${nodeId}`}>Content</div>
|
|
95
|
+
<div
|
|
96
|
+
data-testid={`resize-handle-${nodeId}`}
|
|
97
|
+
onMouseDown={handleResize}
|
|
98
|
+
style={{
|
|
99
|
+
position: 'absolute',
|
|
100
|
+
bottom: 0,
|
|
101
|
+
right: 0,
|
|
102
|
+
width: '10px',
|
|
103
|
+
height: '10px',
|
|
104
|
+
cursor: 'nwse-resize',
|
|
105
|
+
}}
|
|
106
|
+
/>
|
|
107
|
+
</div>
|
|
108
|
+
);
|
|
109
|
+
};
|
|
110
|
+
|
|
111
|
+
describe('NodeDetailWindow', () => {
|
|
112
|
+
beforeEach(() => {
|
|
113
|
+
// Reset UI store before each test
|
|
114
|
+
useUIStore.setState({
|
|
115
|
+
mode: 'agent',
|
|
116
|
+
selectedNodeIds: new Set(),
|
|
117
|
+
detailWindows: new Map(),
|
|
118
|
+
defaultTab: 'liveOutput',
|
|
119
|
+
layoutDirection: 'TB',
|
|
120
|
+
autoLayoutEnabled: true,
|
|
121
|
+
});
|
|
122
|
+
});
|
|
123
|
+
|
|
124
|
+
afterEach(() => {
|
|
125
|
+
// Clean up any open windows
|
|
126
|
+
useUIStore.setState({ detailWindows: new Map() });
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
describe('Window Opening', () => {
|
|
130
|
+
it('should open window when openDetailWindow is called', () => {
|
|
131
|
+
const { rerender } = render(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
132
|
+
|
|
133
|
+
// Window should not exist initially
|
|
134
|
+
expect(screen.queryByTestId('detail-window-agent-1')).not.toBeInTheDocument();
|
|
135
|
+
|
|
136
|
+
// Open window
|
|
137
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
138
|
+
rerender(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
139
|
+
|
|
140
|
+
// Window should now be visible
|
|
141
|
+
expect(screen.getByTestId('detail-window-agent-1')).toBeInTheDocument();
|
|
142
|
+
expect(screen.getByText('Node: agent-1')).toBeInTheDocument();
|
|
143
|
+
});
|
|
144
|
+
|
|
145
|
+
it('should open window with default position and size', () => {
|
|
146
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
147
|
+
render(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
148
|
+
|
|
149
|
+
const window = useUIStore.getState().detailWindows.get('agent-1');
|
|
150
|
+
expect(window).toBeDefined();
|
|
151
|
+
expect(window?.position).toEqual({ x: 100, y: 100 });
|
|
152
|
+
expect(window?.size).toEqual({ width: 600, height: 400 });
|
|
153
|
+
});
|
|
154
|
+
|
|
155
|
+
it('should open window with staggered position for multiple windows', () => {
|
|
156
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
157
|
+
useUIStore.getState().openDetailWindow('agent-2');
|
|
158
|
+
useUIStore.getState().openDetailWindow('agent-3');
|
|
159
|
+
|
|
160
|
+
const window1 = useUIStore.getState().detailWindows.get('agent-1');
|
|
161
|
+
const window2 = useUIStore.getState().detailWindows.get('agent-2');
|
|
162
|
+
const window3 = useUIStore.getState().detailWindows.get('agent-3');
|
|
163
|
+
|
|
164
|
+
expect(window1?.position).toEqual({ x: 100, y: 100 });
|
|
165
|
+
expect(window2?.position).toEqual({ x: 120, y: 120 });
|
|
166
|
+
expect(window3?.position).toEqual({ x: 140, y: 140 });
|
|
167
|
+
});
|
|
168
|
+
|
|
169
|
+
it('should use defaultTab preference when opening window', () => {
|
|
170
|
+
useUIStore.setState({ defaultTab: 'messageHistory' });
|
|
171
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
172
|
+
|
|
173
|
+
const window = useUIStore.getState().detailWindows.get('agent-1');
|
|
174
|
+
expect(window?.activeTab).toBe('messageHistory');
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
it('should not create duplicate window if already open', () => {
|
|
178
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
179
|
+
const window1 = useUIStore.getState().detailWindows.get('agent-1');
|
|
180
|
+
|
|
181
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
182
|
+
const window2 = useUIStore.getState().detailWindows.get('agent-1');
|
|
183
|
+
|
|
184
|
+
expect(window1).toBe(window2);
|
|
185
|
+
expect(useUIStore.getState().detailWindows.size).toBe(1);
|
|
186
|
+
});
|
|
187
|
+
});
|
|
188
|
+
|
|
189
|
+
describe('Window Dragging', () => {
|
|
190
|
+
it('should update window position when dragged', () => {
|
|
191
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
192
|
+
render(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
193
|
+
|
|
194
|
+
const header = screen.getByTestId('window-header-agent-1');
|
|
195
|
+
const initialWindow = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
196
|
+
|
|
197
|
+
// Simulate drag
|
|
198
|
+
fireEvent.mouseDown(header, { clientX: 150, clientY: 150 });
|
|
199
|
+
fireEvent.mouseMove(document, { clientX: 250, clientY: 250 });
|
|
200
|
+
fireEvent.mouseUp(document);
|
|
201
|
+
|
|
202
|
+
const updatedWindow = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
203
|
+
expect(updatedWindow.position.x).toBeGreaterThan(initialWindow.position.x);
|
|
204
|
+
expect(updatedWindow.position.y).toBeGreaterThan(initialWindow.position.y);
|
|
205
|
+
});
|
|
206
|
+
|
|
207
|
+
it('should allow dragging window to different positions', () => {
|
|
208
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
209
|
+
render(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
210
|
+
|
|
211
|
+
const header = screen.getByTestId('window-header-agent-1');
|
|
212
|
+
|
|
213
|
+
// First drag
|
|
214
|
+
fireEvent.mouseDown(header, { clientX: 100, clientY: 100 });
|
|
215
|
+
fireEvent.mouseMove(document, { clientX: 200, clientY: 150 });
|
|
216
|
+
fireEvent.mouseUp(document);
|
|
217
|
+
|
|
218
|
+
const position1 = useUIStore.getState().detailWindows.get('agent-1')!.position;
|
|
219
|
+
|
|
220
|
+
// Second drag
|
|
221
|
+
fireEvent.mouseDown(header, { clientX: position1.x + 50, clientY: position1.y + 50 });
|
|
222
|
+
fireEvent.mouseMove(document, { clientX: position1.x + 150, clientY: position1.y + 100 });
|
|
223
|
+
fireEvent.mouseUp(document);
|
|
224
|
+
|
|
225
|
+
const position2 = useUIStore.getState().detailWindows.get('agent-1')!.position;
|
|
226
|
+
|
|
227
|
+
expect(position2.x).toBeGreaterThan(position1.x);
|
|
228
|
+
expect(position2.y).toBeGreaterThan(position1.y);
|
|
229
|
+
});
|
|
230
|
+
|
|
231
|
+
it('should have drag cursor on header', () => {
|
|
232
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
233
|
+
render(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
234
|
+
|
|
235
|
+
const header = screen.getByTestId('window-header-agent-1');
|
|
236
|
+
expect(header).toHaveStyle({ cursor: 'move' });
|
|
237
|
+
});
|
|
238
|
+
});
|
|
239
|
+
|
|
240
|
+
describe('Window Resizing', () => {
|
|
241
|
+
it('should update window size when resized', () => {
|
|
242
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
243
|
+
render(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
244
|
+
|
|
245
|
+
const resizeHandle = screen.getByTestId('resize-handle-agent-1');
|
|
246
|
+
const initialWindow = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
247
|
+
|
|
248
|
+
// Simulate resize
|
|
249
|
+
fireEvent.mouseDown(resizeHandle, { clientX: 700, clientY: 500 });
|
|
250
|
+
fireEvent.mouseMove(document, { clientX: 800, clientY: 600 });
|
|
251
|
+
fireEvent.mouseUp(document);
|
|
252
|
+
|
|
253
|
+
const updatedWindow = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
254
|
+
expect(updatedWindow.size.width).toBeGreaterThan(initialWindow.size.width);
|
|
255
|
+
expect(updatedWindow.size.height).toBeGreaterThan(initialWindow.size.height);
|
|
256
|
+
});
|
|
257
|
+
|
|
258
|
+
it('should enforce minimum window size', () => {
|
|
259
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
260
|
+
render(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
261
|
+
|
|
262
|
+
const resizeHandle = screen.getByTestId('resize-handle-agent-1');
|
|
263
|
+
|
|
264
|
+
// Try to resize smaller than minimum (400x300)
|
|
265
|
+
fireEvent.mouseDown(resizeHandle, { clientX: 700, clientY: 500 });
|
|
266
|
+
fireEvent.mouseMove(document, { clientX: 200, clientY: 200 }); // Very small
|
|
267
|
+
fireEvent.mouseUp(document);
|
|
268
|
+
|
|
269
|
+
const window = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
270
|
+
expect(window.size.width).toBeGreaterThanOrEqual(400);
|
|
271
|
+
expect(window.size.height).toBeGreaterThanOrEqual(300);
|
|
272
|
+
});
|
|
273
|
+
|
|
274
|
+
it('should have resize cursor on resize handle', () => {
|
|
275
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
276
|
+
render(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
277
|
+
|
|
278
|
+
const resizeHandle = screen.getByTestId('resize-handle-agent-1');
|
|
279
|
+
expect(resizeHandle).toHaveStyle({ cursor: 'nwse-resize' });
|
|
280
|
+
});
|
|
281
|
+
});
|
|
282
|
+
|
|
283
|
+
describe('Window Closing', () => {
|
|
284
|
+
it('should close window when close button is clicked', () => {
|
|
285
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
286
|
+
const { rerender } = render(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
287
|
+
|
|
288
|
+
expect(screen.getByTestId('detail-window-agent-1')).toBeInTheDocument();
|
|
289
|
+
|
|
290
|
+
// Click close button
|
|
291
|
+
const closeButton = screen.getByTestId('close-button-agent-1');
|
|
292
|
+
fireEvent.click(closeButton);
|
|
293
|
+
|
|
294
|
+
rerender(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
295
|
+
|
|
296
|
+
expect(screen.queryByTestId('detail-window-agent-1')).not.toBeInTheDocument();
|
|
297
|
+
expect(useUIStore.getState().detailWindows.size).toBe(0);
|
|
298
|
+
});
|
|
299
|
+
|
|
300
|
+
it('should remove window from detailWindows map when closed', () => {
|
|
301
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
302
|
+
expect(useUIStore.getState().detailWindows.has('agent-1')).toBe(true);
|
|
303
|
+
|
|
304
|
+
useUIStore.getState().closeDetailWindow('agent-1');
|
|
305
|
+
expect(useUIStore.getState().detailWindows.has('agent-1')).toBe(false);
|
|
306
|
+
});
|
|
307
|
+
|
|
308
|
+
it('should handle closing non-existent window gracefully', () => {
|
|
309
|
+
expect(() => {
|
|
310
|
+
useUIStore.getState().closeDetailWindow('non-existent');
|
|
311
|
+
}).not.toThrow();
|
|
312
|
+
});
|
|
313
|
+
});
|
|
314
|
+
|
|
315
|
+
describe('Multiple Windows', () => {
|
|
316
|
+
it('should support multiple windows open simultaneously', () => {
|
|
317
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
318
|
+
useUIStore.getState().openDetailWindow('agent-2');
|
|
319
|
+
useUIStore.getState().openDetailWindow('agent-3');
|
|
320
|
+
|
|
321
|
+
render(
|
|
322
|
+
<>
|
|
323
|
+
<MockNodeDetailWindow nodeId="agent-1" />
|
|
324
|
+
<MockNodeDetailWindow nodeId="agent-2" />
|
|
325
|
+
<MockNodeDetailWindow nodeId="agent-3" />
|
|
326
|
+
</>
|
|
327
|
+
);
|
|
328
|
+
|
|
329
|
+
expect(screen.getByTestId('detail-window-agent-1')).toBeInTheDocument();
|
|
330
|
+
expect(screen.getByTestId('detail-window-agent-2')).toBeInTheDocument();
|
|
331
|
+
expect(screen.getByTestId('detail-window-agent-3')).toBeInTheDocument();
|
|
332
|
+
expect(useUIStore.getState().detailWindows.size).toBe(3);
|
|
333
|
+
});
|
|
334
|
+
|
|
335
|
+
it('should maintain independent state for each window', () => {
|
|
336
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
337
|
+
useUIStore.getState().openDetailWindow('agent-2');
|
|
338
|
+
|
|
339
|
+
const window1 = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
340
|
+
const window2 = useUIStore.getState().detailWindows.get('agent-2')!;
|
|
341
|
+
|
|
342
|
+
expect(window1.nodeId).toBe('agent-1');
|
|
343
|
+
expect(window2.nodeId).toBe('agent-2');
|
|
344
|
+
expect(window1.position).not.toEqual(window2.position);
|
|
345
|
+
});
|
|
346
|
+
|
|
347
|
+
it('should allow closing one window without affecting others', () => {
|
|
348
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
349
|
+
useUIStore.getState().openDetailWindow('agent-2');
|
|
350
|
+
useUIStore.getState().openDetailWindow('agent-3');
|
|
351
|
+
|
|
352
|
+
render(
|
|
353
|
+
<>
|
|
354
|
+
<MockNodeDetailWindow nodeId="agent-1" />
|
|
355
|
+
<MockNodeDetailWindow nodeId="agent-2" />
|
|
356
|
+
<MockNodeDetailWindow nodeId="agent-3" />
|
|
357
|
+
</>
|
|
358
|
+
);
|
|
359
|
+
|
|
360
|
+
// Close middle window
|
|
361
|
+
fireEvent.click(screen.getByTestId('close-button-agent-2'));
|
|
362
|
+
|
|
363
|
+
expect(screen.queryByTestId('detail-window-agent-2')).not.toBeInTheDocument();
|
|
364
|
+
expect(screen.getByTestId('detail-window-agent-1')).toBeInTheDocument();
|
|
365
|
+
expect(screen.getByTestId('detail-window-agent-3')).toBeInTheDocument();
|
|
366
|
+
expect(useUIStore.getState().detailWindows.size).toBe(2);
|
|
367
|
+
});
|
|
368
|
+
|
|
369
|
+
it('should key windows by nodeId', () => {
|
|
370
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
371
|
+
useUIStore.getState().openDetailWindow('message-123');
|
|
372
|
+
|
|
373
|
+
expect(useUIStore.getState().detailWindows.has('agent-1')).toBe(true);
|
|
374
|
+
expect(useUIStore.getState().detailWindows.has('message-123')).toBe(true);
|
|
375
|
+
});
|
|
376
|
+
});
|
|
377
|
+
|
|
378
|
+
describe('State Persistence', () => {
|
|
379
|
+
it('should persist window position updates', () => {
|
|
380
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
381
|
+
|
|
382
|
+
// Update position
|
|
383
|
+
useUIStore.getState().updateDetailWindow('agent-1', {
|
|
384
|
+
position: { x: 300, y: 400 },
|
|
385
|
+
});
|
|
386
|
+
|
|
387
|
+
const window = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
388
|
+
expect(window.position).toEqual({ x: 300, y: 400 });
|
|
389
|
+
});
|
|
390
|
+
|
|
391
|
+
it('should persist window size updates', () => {
|
|
392
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
393
|
+
|
|
394
|
+
// Update size
|
|
395
|
+
useUIStore.getState().updateDetailWindow('agent-1', {
|
|
396
|
+
size: { width: 800, height: 600 },
|
|
397
|
+
});
|
|
398
|
+
|
|
399
|
+
const window = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
400
|
+
expect(window.size).toEqual({ width: 800, height: 600 });
|
|
401
|
+
});
|
|
402
|
+
|
|
403
|
+
it('should persist active tab changes', () => {
|
|
404
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
405
|
+
|
|
406
|
+
// Change active tab
|
|
407
|
+
useUIStore.getState().updateDetailWindow('agent-1', {
|
|
408
|
+
activeTab: 'messageHistory',
|
|
409
|
+
});
|
|
410
|
+
|
|
411
|
+
const window = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
412
|
+
expect(window.activeTab).toBe('messageHistory');
|
|
413
|
+
});
|
|
414
|
+
|
|
415
|
+
it('should preserve window state during drag operations', () => {
|
|
416
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
417
|
+
render(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
418
|
+
|
|
419
|
+
const header = screen.getByTestId('window-header-agent-1');
|
|
420
|
+
|
|
421
|
+
// Perform drag
|
|
422
|
+
fireEvent.mouseDown(header, { clientX: 150, clientY: 150 });
|
|
423
|
+
fireEvent.mouseMove(document, { clientX: 350, clientY: 350 });
|
|
424
|
+
fireEvent.mouseUp(document);
|
|
425
|
+
|
|
426
|
+
const window = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
427
|
+
expect(window.nodeId).toBe('agent-1');
|
|
428
|
+
expect(window.size).toEqual({ width: 600, height: 400 }); // Size unchanged
|
|
429
|
+
expect(window.activeTab).toBe('liveOutput'); // Tab unchanged
|
|
430
|
+
});
|
|
431
|
+
|
|
432
|
+
it('should preserve window state during resize operations', () => {
|
|
433
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
434
|
+
render(<MockNodeDetailWindow nodeId="agent-1" />);
|
|
435
|
+
|
|
436
|
+
const resizeHandle = screen.getByTestId('resize-handle-agent-1');
|
|
437
|
+
|
|
438
|
+
// Perform resize
|
|
439
|
+
fireEvent.mouseDown(resizeHandle, { clientX: 700, clientY: 500 });
|
|
440
|
+
fireEvent.mouseMove(document, { clientX: 900, clientY: 700 });
|
|
441
|
+
fireEvent.mouseUp(document);
|
|
442
|
+
|
|
443
|
+
const window = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
444
|
+
expect(window.nodeId).toBe('agent-1');
|
|
445
|
+
expect(window.position).toEqual({ x: 100, y: 100 }); // Position unchanged
|
|
446
|
+
expect(window.activeTab).toBe('liveOutput'); // Tab unchanged
|
|
447
|
+
});
|
|
448
|
+
});
|
|
449
|
+
|
|
450
|
+
describe('Edge Cases', () => {
|
|
451
|
+
it('should handle updateDetailWindow for non-existent window', () => {
|
|
452
|
+
expect(() => {
|
|
453
|
+
useUIStore.getState().updateDetailWindow('non-existent', {
|
|
454
|
+
position: { x: 100, y: 100 },
|
|
455
|
+
});
|
|
456
|
+
}).not.toThrow();
|
|
457
|
+
|
|
458
|
+
expect(useUIStore.getState().detailWindows.has('non-existent')).toBe(false);
|
|
459
|
+
});
|
|
460
|
+
|
|
461
|
+
it('should handle rapid open/close cycles', () => {
|
|
462
|
+
for (let i = 0; i < 10; i++) {
|
|
463
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
464
|
+
expect(useUIStore.getState().detailWindows.has('agent-1')).toBe(true);
|
|
465
|
+
|
|
466
|
+
useUIStore.getState().closeDetailWindow('agent-1');
|
|
467
|
+
expect(useUIStore.getState().detailWindows.has('agent-1')).toBe(false);
|
|
468
|
+
}
|
|
469
|
+
});
|
|
470
|
+
|
|
471
|
+
it('should handle many simultaneous windows', () => {
|
|
472
|
+
const nodeIds = Array.from({ length: 20 }, (_, i) => `agent-${i}`);
|
|
473
|
+
|
|
474
|
+
nodeIds.forEach((nodeId) => {
|
|
475
|
+
useUIStore.getState().openDetailWindow(nodeId);
|
|
476
|
+
});
|
|
477
|
+
|
|
478
|
+
expect(useUIStore.getState().detailWindows.size).toBe(20);
|
|
479
|
+
|
|
480
|
+
nodeIds.forEach((nodeId) => {
|
|
481
|
+
expect(useUIStore.getState().detailWindows.has(nodeId)).toBe(true);
|
|
482
|
+
});
|
|
483
|
+
});
|
|
484
|
+
|
|
485
|
+
it('should handle partial window updates', () => {
|
|
486
|
+
useUIStore.getState().openDetailWindow('agent-1');
|
|
487
|
+
|
|
488
|
+
const original = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
489
|
+
|
|
490
|
+
// Update only position
|
|
491
|
+
useUIStore.getState().updateDetailWindow('agent-1', {
|
|
492
|
+
position: { x: 500, y: 600 },
|
|
493
|
+
});
|
|
494
|
+
|
|
495
|
+
const updated = useUIStore.getState().detailWindows.get('agent-1')!;
|
|
496
|
+
expect(updated.position).toEqual({ x: 500, y: 600 });
|
|
497
|
+
expect(updated.size).toEqual(original.size);
|
|
498
|
+
expect(updated.activeTab).toEqual(original.activeTab);
|
|
499
|
+
});
|
|
500
|
+
});
|
|
501
|
+
});
|