flock-core 0.5.0b50__py3-none-any.whl → 0.5.0b51__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 (116) 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_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/METADATA +1 -1
  113. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/RECORD +116 -6
  114. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/WHEEL +0 -0
  115. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/entry_points.txt +0 -0
  116. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b51.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,333 @@
1
+ import { describe, it, expect, beforeEach, vi } from 'vitest';
2
+ import { moduleRegistry, ModuleDefinition } from './ModuleRegistry';
3
+
4
+ // Mock React component for testing
5
+ const MockComponent = () => null;
6
+
7
+ describe('ModuleRegistry', () => {
8
+ beforeEach(() => {
9
+ // Reset the registry before each test
10
+ // Access the private resetInstance method through the class
11
+ (moduleRegistry.constructor as any).resetInstance();
12
+
13
+ // Clear console spies
14
+ vi.clearAllMocks();
15
+ });
16
+
17
+ describe('Module Registration', () => {
18
+ it('should register a new module successfully', () => {
19
+ const consoleSpy = vi.spyOn(console, 'log');
20
+
21
+ const module: ModuleDefinition = {
22
+ id: 'test-module',
23
+ name: 'Test Module',
24
+ description: 'A test module',
25
+ component: MockComponent,
26
+ };
27
+
28
+ moduleRegistry.register(module);
29
+
30
+ const registered = moduleRegistry.get('test-module');
31
+ expect(registered).toBeDefined();
32
+ expect(registered?.id).toBe('test-module');
33
+ expect(registered?.name).toBe('Test Module');
34
+ expect(registered?.description).toBe('A test module');
35
+ expect(consoleSpy).toHaveBeenCalledWith(
36
+ 'Module "Test Module" (test-module) registered successfully.'
37
+ );
38
+ });
39
+
40
+ it('should register a module with all optional properties', () => {
41
+ const onMountMock = vi.fn();
42
+ const onUnmountMock = vi.fn();
43
+
44
+ const module: ModuleDefinition = {
45
+ id: 'full-module',
46
+ name: 'Full Module',
47
+ description: 'Module with all properties',
48
+ icon: 'icon-name',
49
+ component: MockComponent,
50
+ onMount: onMountMock,
51
+ onUnmount: onUnmountMock,
52
+ };
53
+
54
+ moduleRegistry.register(module);
55
+
56
+ const registered = moduleRegistry.get('full-module');
57
+ expect(registered).toBeDefined();
58
+ expect(registered?.icon).toBe('icon-name');
59
+ expect(registered?.onMount).toBe(onMountMock);
60
+ expect(registered?.onUnmount).toBe(onUnmountMock);
61
+ });
62
+
63
+ it('should prevent duplicate registration and warn', () => {
64
+ const consoleWarnSpy = vi.spyOn(console, 'warn');
65
+
66
+ const module: ModuleDefinition = {
67
+ id: 'duplicate-module',
68
+ name: 'Duplicate Module',
69
+ description: 'Will be registered twice',
70
+ component: MockComponent,
71
+ };
72
+
73
+ moduleRegistry.register(module);
74
+ moduleRegistry.register(module);
75
+
76
+ expect(consoleWarnSpy).toHaveBeenCalledWith(
77
+ 'Module with id "duplicate-module" is already registered. Skipping registration.'
78
+ );
79
+
80
+ // Should only have one module
81
+ const allModules = moduleRegistry.getAll();
82
+ const duplicates = allModules.filter(m => m.id === 'duplicate-module');
83
+ expect(duplicates).toHaveLength(1);
84
+ });
85
+ });
86
+
87
+ describe('Module Retrieval', () => {
88
+ it('should retrieve a module by ID', () => {
89
+ const module: ModuleDefinition = {
90
+ id: 'retrieve-test',
91
+ name: 'Retrieve Test',
92
+ description: 'Test retrieval',
93
+ component: MockComponent,
94
+ };
95
+
96
+ moduleRegistry.register(module);
97
+ const retrieved = moduleRegistry.get('retrieve-test');
98
+
99
+ expect(retrieved).toBeDefined();
100
+ expect(retrieved?.id).toBe('retrieve-test');
101
+ });
102
+
103
+ it('should return undefined for non-existent module', () => {
104
+ const result = moduleRegistry.get('non-existent-module');
105
+ expect(result).toBeUndefined();
106
+ });
107
+
108
+ it('should get all registered modules', () => {
109
+ const module1: ModuleDefinition = {
110
+ id: 'module-1',
111
+ name: 'Module 1',
112
+ description: 'First module',
113
+ component: MockComponent,
114
+ };
115
+
116
+ const module2: ModuleDefinition = {
117
+ id: 'module-2',
118
+ name: 'Module 2',
119
+ description: 'Second module',
120
+ component: MockComponent,
121
+ };
122
+
123
+ const module3: ModuleDefinition = {
124
+ id: 'module-3',
125
+ name: 'Module 3',
126
+ description: 'Third module',
127
+ component: MockComponent,
128
+ };
129
+
130
+ moduleRegistry.register(module1);
131
+ moduleRegistry.register(module2);
132
+ moduleRegistry.register(module3);
133
+
134
+ const allModules = moduleRegistry.getAll();
135
+ expect(allModules).toHaveLength(3);
136
+ expect(allModules.map(m => m.id)).toContain('module-1');
137
+ expect(allModules.map(m => m.id)).toContain('module-2');
138
+ expect(allModules.map(m => m.id)).toContain('module-3');
139
+ });
140
+
141
+ it('should return empty array when no modules registered', () => {
142
+ const allModules = moduleRegistry.getAll();
143
+ expect(allModules).toHaveLength(0);
144
+ expect(allModules).toEqual([]);
145
+ });
146
+ });
147
+
148
+ describe('Module Unregistration', () => {
149
+ it('should unregister a module successfully', () => {
150
+ const consoleSpy = vi.spyOn(console, 'log');
151
+
152
+ const module: ModuleDefinition = {
153
+ id: 'unregister-test',
154
+ name: 'Unregister Test',
155
+ description: 'Test unregistration',
156
+ component: MockComponent,
157
+ };
158
+
159
+ moduleRegistry.register(module);
160
+ expect(moduleRegistry.get('unregister-test')).toBeDefined();
161
+
162
+ moduleRegistry.unregister('unregister-test');
163
+
164
+ expect(moduleRegistry.get('unregister-test')).toBeUndefined();
165
+ expect(consoleSpy).toHaveBeenCalledWith(
166
+ 'Module "Unregister Test" (unregister-test) unregistered successfully.'
167
+ );
168
+ });
169
+
170
+ it('should handle unregistering non-existent module gracefully', () => {
171
+ const consoleSpy = vi.spyOn(console, 'log');
172
+
173
+ moduleRegistry.unregister('non-existent');
174
+
175
+ // Should not throw error and should not log anything
176
+ expect(consoleSpy).not.toHaveBeenCalled();
177
+ });
178
+
179
+ it('should remove module from getAll after unregistration', () => {
180
+ const module1: ModuleDefinition = {
181
+ id: 'module-1',
182
+ name: 'Module 1',
183
+ description: 'First module',
184
+ component: MockComponent,
185
+ };
186
+
187
+ const module2: ModuleDefinition = {
188
+ id: 'module-2',
189
+ name: 'Module 2',
190
+ description: 'Second module',
191
+ component: MockComponent,
192
+ };
193
+
194
+ moduleRegistry.register(module1);
195
+ moduleRegistry.register(module2);
196
+ expect(moduleRegistry.getAll()).toHaveLength(2);
197
+
198
+ moduleRegistry.unregister('module-1');
199
+
200
+ const remaining = moduleRegistry.getAll();
201
+ expect(remaining).toHaveLength(1);
202
+ expect(remaining[0]?.id).toBe('module-2');
203
+ });
204
+ });
205
+
206
+ describe('Singleton Pattern', () => {
207
+ it('should maintain singleton instance across method calls', () => {
208
+ const module1: ModuleDefinition = {
209
+ id: 'singleton-test-1',
210
+ name: 'Singleton Test 1',
211
+ description: 'Test singleton behavior',
212
+ component: MockComponent,
213
+ };
214
+
215
+ const module2: ModuleDefinition = {
216
+ id: 'singleton-test-2',
217
+ name: 'Singleton Test 2',
218
+ description: 'Test singleton behavior',
219
+ component: MockComponent,
220
+ };
221
+
222
+ moduleRegistry.register(module1);
223
+ moduleRegistry.register(module2);
224
+
225
+ // Should maintain state across different method calls
226
+ const retrieved1 = moduleRegistry.get('singleton-test-1');
227
+ const retrieved2 = moduleRegistry.get('singleton-test-2');
228
+ const allModules = moduleRegistry.getAll();
229
+
230
+ expect(retrieved1).toBeDefined();
231
+ expect(retrieved2).toBeDefined();
232
+ expect(allModules).toHaveLength(2);
233
+ });
234
+
235
+ it('should persist state after unregister and re-register operations', () => {
236
+ const module1: ModuleDefinition = {
237
+ id: 'persist-1',
238
+ name: 'Persist 1',
239
+ description: 'First persistent module',
240
+ component: MockComponent,
241
+ };
242
+
243
+ const module2: ModuleDefinition = {
244
+ id: 'persist-2',
245
+ name: 'Persist 2',
246
+ description: 'Second persistent module',
247
+ component: MockComponent,
248
+ };
249
+
250
+ const module3: ModuleDefinition = {
251
+ id: 'persist-3',
252
+ name: 'Persist 3',
253
+ description: 'Third persistent module',
254
+ component: MockComponent,
255
+ };
256
+
257
+ // Register three modules
258
+ moduleRegistry.register(module1);
259
+ moduleRegistry.register(module2);
260
+ moduleRegistry.register(module3);
261
+ expect(moduleRegistry.getAll()).toHaveLength(3);
262
+
263
+ // Unregister one
264
+ moduleRegistry.unregister('persist-2');
265
+ expect(moduleRegistry.getAll()).toHaveLength(2);
266
+
267
+ // State should persist - the remaining modules should still be there
268
+ expect(moduleRegistry.get('persist-1')).toBeDefined();
269
+ expect(moduleRegistry.get('persist-2')).toBeUndefined();
270
+ expect(moduleRegistry.get('persist-3')).toBeDefined();
271
+ });
272
+ });
273
+
274
+ describe('Console Logging', () => {
275
+ it('should log registration events with correct format', () => {
276
+ const consoleSpy = vi.spyOn(console, 'log');
277
+
278
+ const module: ModuleDefinition = {
279
+ id: 'log-test',
280
+ name: 'Log Test Module',
281
+ description: 'Test logging',
282
+ component: MockComponent,
283
+ };
284
+
285
+ moduleRegistry.register(module);
286
+
287
+ expect(consoleSpy).toHaveBeenCalledTimes(1);
288
+ expect(consoleSpy).toHaveBeenCalledWith(
289
+ 'Module "Log Test Module" (log-test) registered successfully.'
290
+ );
291
+ });
292
+
293
+ it('should log unregistration events with correct format', () => {
294
+ const consoleSpy = vi.spyOn(console, 'log');
295
+
296
+ const module: ModuleDefinition = {
297
+ id: 'unlog-test',
298
+ name: 'Unlog Test Module',
299
+ description: 'Test unregistration logging',
300
+ component: MockComponent,
301
+ };
302
+
303
+ moduleRegistry.register(module);
304
+ consoleSpy.mockClear(); // Clear registration log
305
+
306
+ moduleRegistry.unregister('unlog-test');
307
+
308
+ expect(consoleSpy).toHaveBeenCalledTimes(1);
309
+ expect(consoleSpy).toHaveBeenCalledWith(
310
+ 'Module "Unlog Test Module" (unlog-test) unregistered successfully.'
311
+ );
312
+ });
313
+
314
+ it('should warn when registering duplicate module', () => {
315
+ const consoleWarnSpy = vi.spyOn(console, 'warn');
316
+
317
+ const module: ModuleDefinition = {
318
+ id: 'warn-test',
319
+ name: 'Warn Test',
320
+ description: 'Test warning',
321
+ component: MockComponent,
322
+ };
323
+
324
+ moduleRegistry.register(module);
325
+ moduleRegistry.register(module);
326
+
327
+ expect(consoleWarnSpy).toHaveBeenCalledTimes(1);
328
+ expect(consoleWarnSpy).toHaveBeenCalledWith(
329
+ 'Module with id "warn-test" is already registered. Skipping registration.'
330
+ );
331
+ });
332
+ });
333
+ });
@@ -0,0 +1,85 @@
1
+ import React from 'react';
2
+ import type { Agent, Message } from '../../types/graph';
3
+ import type { TimeRange } from '../../types/filters';
4
+
5
+ export interface ModuleContext {
6
+ // Data access
7
+ agents: Map<string, Agent>;
8
+ messages: Map<string, Message>;
9
+ events: Message[];
10
+
11
+ // Filter state
12
+ filters: {
13
+ correlationId: string | null;
14
+ timeRange: TimeRange;
15
+ };
16
+
17
+ // Actions
18
+ publish: (artifact: any) => void;
19
+ invoke: (agentName: string, inputs: any[]) => void;
20
+ }
21
+
22
+ export interface ModuleDefinition {
23
+ id: string;
24
+ name: string;
25
+ description: string;
26
+ icon?: string;
27
+ component: React.ComponentType<{ context: ModuleContext }>;
28
+ onMount?: (context: ModuleContext) => void;
29
+ onUnmount?: () => void;
30
+ }
31
+
32
+ export interface ModuleRegistry {
33
+ register: (module: ModuleDefinition) => void;
34
+ unregister: (moduleId: string) => void;
35
+ getAll: () => ModuleDefinition[];
36
+ get: (moduleId: string) => ModuleDefinition | undefined;
37
+ }
38
+
39
+ class ModuleRegistryImpl implements ModuleRegistry {
40
+ private static instance: ModuleRegistryImpl;
41
+ private modules: Map<string, ModuleDefinition> = new Map();
42
+
43
+ private constructor() {}
44
+
45
+ public static getInstance(): ModuleRegistryImpl {
46
+ if (!ModuleRegistryImpl.instance) {
47
+ ModuleRegistryImpl.instance = new ModuleRegistryImpl();
48
+ }
49
+ return ModuleRegistryImpl.instance;
50
+ }
51
+
52
+ public register(module: ModuleDefinition): void {
53
+ if (this.modules.has(module.id)) {
54
+ console.warn(`Module with id "${module.id}" is already registered. Skipping registration.`);
55
+ return;
56
+ }
57
+ this.modules.set(module.id, module);
58
+ console.log(`Module "${module.name}" (${module.id}) registered successfully.`);
59
+ }
60
+
61
+ public unregister(moduleId: string): void {
62
+ if (this.modules.has(moduleId)) {
63
+ const module = this.modules.get(moduleId);
64
+ this.modules.delete(moduleId);
65
+ console.log(`Module "${module?.name}" (${moduleId}) unregistered successfully.`);
66
+ }
67
+ }
68
+
69
+ public getAll(): ModuleDefinition[] {
70
+ return Array.from(this.modules.values());
71
+ }
72
+
73
+ public get(moduleId: string): ModuleDefinition | undefined {
74
+ return this.modules.get(moduleId);
75
+ }
76
+
77
+ // For testing purposes - reset the singleton instance
78
+ public static resetInstance(): void {
79
+ if (ModuleRegistryImpl.instance) {
80
+ ModuleRegistryImpl.instance.modules.clear();
81
+ }
82
+ }
83
+ }
84
+
85
+ export const moduleRegistry = ModuleRegistryImpl.getInstance();
@@ -0,0 +1,155 @@
1
+ import React, { useCallback, memo } from 'react';
2
+ import { Rnd } from 'react-rnd';
3
+ import { useModuleStore } from '../../store/moduleStore';
4
+ import { moduleRegistry } from './ModuleRegistry';
5
+ import { useModules } from '../../hooks/useModules';
6
+
7
+ interface ModuleWindowProps {
8
+ instanceId: string;
9
+ }
10
+
11
+ const ModuleWindow: React.FC<ModuleWindowProps> = memo(({ instanceId }) => {
12
+ const instance = useModuleStore((state) => state.instances.get(instanceId));
13
+ const updateModule = useModuleStore((state) => state.updateModule);
14
+ const removeModule = useModuleStore((state) => state.removeModule);
15
+
16
+ // Get module context from useModules hook
17
+ const { context } = useModules();
18
+
19
+ const handleClose = useCallback(() => {
20
+ removeModule(instanceId);
21
+ }, [instanceId, removeModule]);
22
+
23
+ // Don't render if instance doesn't exist or is not visible
24
+ if (!instance || !instance.visible) return null;
25
+
26
+ // Get module definition from registry
27
+ const moduleDefinition = moduleRegistry.get(instance.type);
28
+ if (!moduleDefinition) {
29
+ console.error(`Module definition not found for type: ${instance.type}`);
30
+ return null;
31
+ }
32
+
33
+ const { position, size } = instance;
34
+ const ModuleComponent = moduleDefinition.component;
35
+
36
+ return (
37
+ <Rnd
38
+ position={position}
39
+ size={size}
40
+ onDragStop={(_e, d) => {
41
+ updateModule(instanceId, {
42
+ position: { x: d.x, y: d.y },
43
+ });
44
+ }}
45
+ onResizeStop={(_e, _direction, ref, _delta, position) => {
46
+ updateModule(instanceId, {
47
+ size: {
48
+ width: parseInt(ref.style.width, 10),
49
+ height: parseInt(ref.style.height, 10),
50
+ },
51
+ position,
52
+ });
53
+ }}
54
+ minWidth={1000}
55
+ minHeight={500}
56
+ bounds="parent"
57
+ dragHandleClassName="module-window-header"
58
+ style={{
59
+ zIndex: 1000,
60
+ display: 'flex',
61
+ flexDirection: 'column',
62
+ }}
63
+ >
64
+ <div
65
+ style={{
66
+ display: 'flex',
67
+ flexDirection: 'column',
68
+ width: '100%',
69
+ height: '100%',
70
+ background: 'var(--color-glass-bg)',
71
+ border: 'var(--border-width-1) solid var(--color-glass-border)',
72
+ borderRadius: 'var(--radius-xl)',
73
+ overflow: 'hidden',
74
+ boxShadow: 'var(--shadow-xl)',
75
+ backdropFilter: 'blur(var(--blur-lg))',
76
+ WebkitBackdropFilter: 'blur(var(--blur-lg))',
77
+ }}
78
+ >
79
+ {/* Header */}
80
+ <div
81
+ className="module-window-header"
82
+ style={{
83
+ display: 'flex',
84
+ alignItems: 'center',
85
+ justifyContent: 'space-between',
86
+ padding: 'var(--space-component-md) var(--space-component-lg)',
87
+ background: 'rgba(42, 42, 50, 0.5)',
88
+ borderBottom: 'var(--border-width-1) solid var(--color-border-subtle)',
89
+ cursor: 'move',
90
+ userSelect: 'none',
91
+ }}
92
+ >
93
+ <div style={{ display: 'flex', alignItems: 'center', gap: 'var(--gap-md)' }}>
94
+ {moduleDefinition.icon && (
95
+ <span style={{ fontSize: 16 }}>{moduleDefinition.icon}</span>
96
+ )}
97
+ <span
98
+ style={{
99
+ color: 'var(--color-text-primary)',
100
+ fontSize: 'var(--font-size-body-sm)',
101
+ fontWeight: 'var(--font-weight-semibold)',
102
+ fontFamily: 'var(--font-family-sans)',
103
+ }}
104
+ >
105
+ {moduleDefinition.name}
106
+ </span>
107
+ </div>
108
+ <button
109
+ onClick={handleClose}
110
+ aria-label="Close window"
111
+ style={{
112
+ background: 'transparent',
113
+ border: 'none',
114
+ color: 'var(--color-text-secondary)',
115
+ fontSize: 'var(--font-size-h3)',
116
+ cursor: 'pointer',
117
+ padding: 'var(--spacing-1) var(--spacing-2)',
118
+ lineHeight: 1,
119
+ borderRadius: 'var(--radius-md)',
120
+ transition: 'var(--transition-colors)',
121
+ display: 'flex',
122
+ alignItems: 'center',
123
+ justifyContent: 'center',
124
+ }}
125
+ onMouseEnter={(e) => {
126
+ e.currentTarget.style.color = 'var(--color-error)';
127
+ e.currentTarget.style.background = 'var(--color-error-bg)';
128
+ }}
129
+ onMouseLeave={(e) => {
130
+ e.currentTarget.style.color = 'var(--color-text-secondary)';
131
+ e.currentTarget.style.background = 'transparent';
132
+ }}
133
+ >
134
+ ×
135
+ </button>
136
+ </div>
137
+
138
+ {/* Module Content */}
139
+ <div
140
+ style={{
141
+ flex: 1,
142
+ overflow: 'hidden',
143
+ background: 'var(--color-bg-surface)',
144
+ }}
145
+ >
146
+ <ModuleComponent context={context} />
147
+ </div>
148
+ </div>
149
+ </Rnd>
150
+ );
151
+ });
152
+
153
+ ModuleWindow.displayName = 'ModuleWindow';
154
+
155
+ export default ModuleWindow;
@@ -0,0 +1,20 @@
1
+ import { moduleRegistry } from './ModuleRegistry';
2
+ import EventLogModuleWrapper from './EventLogModuleWrapper';
3
+
4
+ /**
5
+ * Register all available modules
6
+ * This should be called during application initialization
7
+ */
8
+ export function registerModules(): void {
9
+ // Register EventLog module
10
+ moduleRegistry.register({
11
+ id: 'eventLog',
12
+ name: 'Event Log',
13
+ description: 'View and filter system events',
14
+ icon: '📋',
15
+ component: EventLogModuleWrapper,
16
+ });
17
+
18
+ // Future modules can be registered here
19
+ // moduleRegistry.register({ ... });
20
+ }