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,251 @@
1
+ /* ========================================
2
+ Keyboard Shortcuts Dialog
3
+ Premium modal design with glassmorphism
4
+ ======================================== */
5
+
6
+ /* Overlay */
7
+ .keyboard-shortcuts-overlay {
8
+ position: fixed;
9
+ top: 0;
10
+ left: 0;
11
+ right: 0;
12
+ bottom: 0;
13
+ background: var(--color-overlay);
14
+ backdrop-filter: blur(var(--blur-sm));
15
+ -webkit-backdrop-filter: blur(var(--blur-sm));
16
+ display: flex;
17
+ align-items: center;
18
+ justify-content: center;
19
+ z-index: 1000;
20
+ animation: fadeIn var(--duration-normal) var(--ease-smooth);
21
+ }
22
+
23
+ @keyframes fadeIn {
24
+ from {
25
+ opacity: 0;
26
+ }
27
+ to {
28
+ opacity: 1;
29
+ }
30
+ }
31
+
32
+ /* Dialog Container */
33
+ .keyboard-shortcuts-dialog {
34
+ background: var(--color-bg-surface);
35
+ border: var(--border-default);
36
+ border-radius: var(--radius-lg);
37
+ box-shadow: var(--shadow-xl);
38
+ max-width: 600px;
39
+ width: 90%;
40
+ max-height: 80vh;
41
+ display: flex;
42
+ flex-direction: column;
43
+ animation: slideInUp var(--duration-normal) var(--ease-smooth);
44
+ }
45
+
46
+ @keyframes slideInUp {
47
+ from {
48
+ opacity: 0;
49
+ transform: translateY(20px) scale(0.98);
50
+ }
51
+ to {
52
+ opacity: 1;
53
+ transform: translateY(0) scale(1);
54
+ }
55
+ }
56
+
57
+ /* Header */
58
+ .keyboard-shortcuts-header {
59
+ display: flex;
60
+ justify-content: space-between;
61
+ align-items: center;
62
+ padding: var(--space-layout-sm) var(--space-layout-md);
63
+ border-bottom: var(--border-subtle);
64
+ }
65
+
66
+ .keyboard-shortcuts-title {
67
+ margin: 0;
68
+ font-size: var(--font-size-h4);
69
+ font-weight: var(--font-weight-semibold);
70
+ color: var(--color-text-primary);
71
+ letter-spacing: var(--letter-spacing-tight);
72
+ }
73
+
74
+ .keyboard-shortcuts-close {
75
+ display: inline-flex;
76
+ align-items: center;
77
+ justify-content: center;
78
+ width: 32px;
79
+ height: 32px;
80
+ padding: 0;
81
+ background: transparent;
82
+ color: var(--color-text-tertiary);
83
+ border: none;
84
+ border-radius: var(--radius-md);
85
+ cursor: pointer;
86
+ transition: var(--transition-colors), var(--transition-transform);
87
+ }
88
+
89
+ .keyboard-shortcuts-close:hover {
90
+ background: var(--color-bg-overlay);
91
+ color: var(--color-text-secondary);
92
+ }
93
+
94
+ .keyboard-shortcuts-close:active {
95
+ transform: scale(0.95);
96
+ }
97
+
98
+ .keyboard-shortcuts-close:focus-visible {
99
+ outline: none;
100
+ box-shadow: var(--shadow-glow-primary);
101
+ }
102
+
103
+ /* Content */
104
+ .keyboard-shortcuts-content {
105
+ flex: 1;
106
+ overflow-y: auto;
107
+ padding: var(--space-layout-md);
108
+ }
109
+
110
+ /* Category */
111
+ .keyboard-shortcuts-category {
112
+ margin-bottom: var(--space-layout-lg);
113
+ }
114
+
115
+ .keyboard-shortcuts-category:last-child {
116
+ margin-bottom: 0;
117
+ }
118
+
119
+ .keyboard-shortcuts-category-title {
120
+ margin: 0 0 var(--space-component-md) 0;
121
+ font-size: var(--font-size-body-sm);
122
+ font-weight: var(--font-weight-semibold);
123
+ color: var(--color-text-secondary);
124
+ text-transform: uppercase;
125
+ letter-spacing: var(--letter-spacing-wide);
126
+ }
127
+
128
+ /* Shortcut List */
129
+ .keyboard-shortcuts-list {
130
+ display: flex;
131
+ flex-direction: column;
132
+ gap: var(--gap-sm);
133
+ }
134
+
135
+ .keyboard-shortcut-item {
136
+ display: flex;
137
+ justify-content: space-between;
138
+ align-items: center;
139
+ padding: var(--space-component-sm) var(--space-component-md);
140
+ background: var(--color-bg-elevated);
141
+ border: var(--border-subtle);
142
+ border-radius: var(--radius-md);
143
+ transition: var(--transition-colors);
144
+ }
145
+
146
+ .keyboard-shortcut-item:hover {
147
+ background: var(--color-bg-overlay);
148
+ border-color: var(--color-border-strong);
149
+ }
150
+
151
+ /* Keys */
152
+ .keyboard-shortcut-keys {
153
+ display: flex;
154
+ align-items: center;
155
+ gap: var(--gap-xs);
156
+ }
157
+
158
+ .keyboard-key {
159
+ display: inline-flex;
160
+ align-items: center;
161
+ justify-content: center;
162
+ min-width: 28px;
163
+ padding: var(--spacing-1) var(--spacing-2);
164
+ background: var(--color-bg-base);
165
+ color: var(--color-text-primary);
166
+ border: var(--border-default);
167
+ border-radius: var(--radius-sm);
168
+ font-family: var(--font-family-mono);
169
+ font-size: var(--font-size-body-sm);
170
+ font-weight: var(--font-weight-medium);
171
+ box-shadow: var(--shadow-xs),
172
+ inset 0 -2px 0 0 var(--color-border-default);
173
+ line-height: 1;
174
+ white-space: nowrap;
175
+ }
176
+
177
+ .keyboard-key-separator {
178
+ color: var(--color-text-tertiary);
179
+ font-size: var(--font-size-caption);
180
+ font-weight: var(--font-weight-medium);
181
+ }
182
+
183
+ /* Description */
184
+ .keyboard-shortcut-description {
185
+ color: var(--color-text-secondary);
186
+ font-size: var(--font-size-body-sm);
187
+ text-align: right;
188
+ }
189
+
190
+ /* Footer */
191
+ .keyboard-shortcuts-footer {
192
+ padding: var(--space-component-md) var(--space-layout-md);
193
+ border-top: var(--border-subtle);
194
+ background: var(--color-bg-elevated);
195
+ }
196
+
197
+ .keyboard-shortcuts-hint {
198
+ margin: 0;
199
+ text-align: center;
200
+ color: var(--color-text-tertiary);
201
+ font-size: var(--font-size-caption);
202
+ }
203
+
204
+ /* Scrollbar */
205
+ .keyboard-shortcuts-content::-webkit-scrollbar {
206
+ width: 8px;
207
+ }
208
+
209
+ .keyboard-shortcuts-content::-webkit-scrollbar-track {
210
+ background: var(--color-bg-elevated);
211
+ }
212
+
213
+ .keyboard-shortcuts-content::-webkit-scrollbar-thumb {
214
+ background: var(--color-border-default);
215
+ border-radius: var(--radius-full);
216
+ transition: var(--transition-colors);
217
+ }
218
+
219
+ .keyboard-shortcuts-content::-webkit-scrollbar-thumb:hover {
220
+ background: var(--color-border-strong);
221
+ }
222
+
223
+ /* Responsive */
224
+ @media (max-width: 768px) {
225
+ .keyboard-shortcuts-dialog {
226
+ max-width: 95%;
227
+ max-height: 90vh;
228
+ }
229
+
230
+ .keyboard-shortcut-item {
231
+ flex-direction: column;
232
+ align-items: flex-start;
233
+ gap: var(--gap-sm);
234
+ }
235
+
236
+ .keyboard-shortcut-description {
237
+ text-align: left;
238
+ }
239
+ }
240
+
241
+ /* Accessibility - Reduced Motion */
242
+ @media (prefers-reduced-motion: reduce) {
243
+ .keyboard-shortcuts-overlay,
244
+ .keyboard-shortcuts-dialog {
245
+ animation: none;
246
+ }
247
+
248
+ .keyboard-shortcuts-close {
249
+ transition: none;
250
+ }
251
+ }
@@ -0,0 +1,151 @@
1
+ import React, { useEffect } from 'react';
2
+ import './KeyboardShortcutsDialog.css';
3
+
4
+ interface KeyboardShortcutsDialogProps {
5
+ isOpen: boolean;
6
+ onClose: () => void;
7
+ }
8
+
9
+ const isMac = typeof navigator !== 'undefined' && navigator.platform.toUpperCase().indexOf('MAC') >= 0;
10
+
11
+ interface Shortcut {
12
+ keys: string[];
13
+ description: string;
14
+ category: 'Navigation' | 'Panels' | 'General';
15
+ }
16
+
17
+ const shortcuts: Shortcut[] = [
18
+ // Navigation
19
+ {
20
+ keys: isMac ? ['⌘', 'M'] : ['Ctrl', 'M'],
21
+ description: 'Toggle Agent/Blackboard View',
22
+ category: 'Navigation',
23
+ },
24
+ {
25
+ keys: isMac ? ['⌘', 'F'] : ['Ctrl', 'F'],
26
+ description: 'Focus filter input',
27
+ category: 'Navigation',
28
+ },
29
+
30
+ // Panels
31
+ {
32
+ keys: isMac ? ['⌘', 'Shift', 'P'] : ['Ctrl', 'Shift', 'P'],
33
+ description: 'Toggle Publish Panel',
34
+ category: 'Panels',
35
+ },
36
+ {
37
+ keys: isMac ? ['⌘', 'Shift', 'D'] : ['Ctrl', 'Shift', 'D'],
38
+ description: 'Toggle Agent Details',
39
+ category: 'Panels',
40
+ },
41
+ {
42
+ keys: isMac ? ['⌘', 'Shift', 'F'] : ['Ctrl', 'Shift', 'F'],
43
+ description: 'Toggle Filters Panel',
44
+ category: 'Panels',
45
+ },
46
+ {
47
+ keys: isMac ? ['⌘', ','] : ['Ctrl', ','],
48
+ description: 'Toggle Settings Panel',
49
+ category: 'Panels',
50
+ },
51
+
52
+ // General
53
+ {
54
+ keys: ['Esc'],
55
+ description: 'Close panels and windows',
56
+ category: 'General',
57
+ },
58
+ {
59
+ keys: isMac ? ['⌘', '/'] : ['Ctrl', '/'],
60
+ description: 'Show this help dialog',
61
+ category: 'General',
62
+ },
63
+ ];
64
+
65
+ const KeyboardShortcutsDialog: React.FC<KeyboardShortcutsDialogProps> = ({ isOpen, onClose }) => {
66
+ // Handle ESC key to close dialog
67
+ useEffect(() => {
68
+ if (!isOpen) return;
69
+
70
+ const handleKeyDown = (event: KeyboardEvent) => {
71
+ if (event.key === 'Escape') {
72
+ event.preventDefault();
73
+ event.stopPropagation();
74
+ onClose();
75
+ }
76
+ };
77
+
78
+ window.addEventListener('keydown', handleKeyDown, { capture: true });
79
+ return () => {
80
+ window.removeEventListener('keydown', handleKeyDown, { capture: true });
81
+ };
82
+ }, [isOpen, onClose]);
83
+
84
+ if (!isOpen) return null;
85
+
86
+ const groupedShortcuts = shortcuts.reduce((acc, shortcut) => {
87
+ if (!acc[shortcut.category]) {
88
+ acc[shortcut.category] = [];
89
+ }
90
+ acc[shortcut.category]!.push(shortcut);
91
+ return acc;
92
+ }, {} as Record<string, Shortcut[]>);
93
+
94
+ return (
95
+ <div className="keyboard-shortcuts-overlay" onClick={onClose}>
96
+ <div className="keyboard-shortcuts-dialog" onClick={(e) => e.stopPropagation()}>
97
+ <div className="keyboard-shortcuts-header">
98
+ <h2 className="keyboard-shortcuts-title">Keyboard Shortcuts</h2>
99
+ <button
100
+ className="keyboard-shortcuts-close"
101
+ onClick={onClose}
102
+ aria-label="Close keyboard shortcuts"
103
+ >
104
+ <svg width="20" height="20" viewBox="0 0 20 20" fill="none">
105
+ <path
106
+ d="M15 5L5 15M5 5l10 10"
107
+ stroke="currentColor"
108
+ strokeWidth="2"
109
+ strokeLinecap="round"
110
+ strokeLinejoin="round"
111
+ />
112
+ </svg>
113
+ </button>
114
+ </div>
115
+
116
+ <div className="keyboard-shortcuts-content">
117
+ {Object.entries(groupedShortcuts).map(([category, categoryShortcuts]) => (
118
+ <div key={category} className="keyboard-shortcuts-category">
119
+ <h3 className="keyboard-shortcuts-category-title">{category}</h3>
120
+ <div className="keyboard-shortcuts-list">
121
+ {categoryShortcuts.map((shortcut, index) => (
122
+ <div key={index} className="keyboard-shortcut-item">
123
+ <div className="keyboard-shortcut-keys">
124
+ {shortcut.keys.map((key, keyIndex) => (
125
+ <React.Fragment key={keyIndex}>
126
+ <kbd className="keyboard-key">{key}</kbd>
127
+ {keyIndex < shortcut.keys.length - 1 && (
128
+ <span className="keyboard-key-separator">+</span>
129
+ )}
130
+ </React.Fragment>
131
+ ))}
132
+ </div>
133
+ <div className="keyboard-shortcut-description">{shortcut.description}</div>
134
+ </div>
135
+ ))}
136
+ </div>
137
+ </div>
138
+ ))}
139
+ </div>
140
+
141
+ <div className="keyboard-shortcuts-footer">
142
+ <p className="keyboard-shortcuts-hint">
143
+ Press <kbd className="keyboard-key">Esc</kbd> or click outside to close
144
+ </p>
145
+ </div>
146
+ </div>
147
+ </div>
148
+ );
149
+ };
150
+
151
+ export default KeyboardShortcutsDialog;
@@ -0,0 +1,97 @@
1
+ /* Loading Spinner Component */
2
+
3
+ .container {
4
+ display: flex;
5
+ flex-direction: column;
6
+ align-items: center;
7
+ justify-content: center;
8
+ gap: var(--spacing-4);
9
+ padding: var(--space-layout-md);
10
+ }
11
+
12
+ .spinner {
13
+ position: relative;
14
+ display: inline-block;
15
+ }
16
+
17
+ /* Size variants */
18
+ .spinner.sm {
19
+ width: 24px;
20
+ height: 24px;
21
+ }
22
+
23
+ .spinner.md {
24
+ width: 40px;
25
+ height: 40px;
26
+ }
27
+
28
+ .spinner.lg {
29
+ width: 64px;
30
+ height: 64px;
31
+ }
32
+
33
+ .ring {
34
+ position: absolute;
35
+ top: 0;
36
+ left: 0;
37
+ width: 100%;
38
+ height: 100%;
39
+ border: 3px solid transparent;
40
+ border-top-color: var(--color-primary-500);
41
+ border-radius: var(--radius-circle);
42
+ animation: spin var(--duration-slowest) cubic-bezier(0.5, 0, 0.5, 1) infinite;
43
+ }
44
+
45
+ .ring:nth-child(1) {
46
+ animation-delay: calc(var(--duration-slowest) * -0.45);
47
+ border-top-color: var(--color-primary-500);
48
+ }
49
+
50
+ .ring:nth-child(2) {
51
+ animation-delay: calc(var(--duration-slowest) * -0.3);
52
+ border-top-color: var(--color-secondary-500);
53
+ }
54
+
55
+ .ring:nth-child(3) {
56
+ animation-delay: calc(var(--duration-slowest) * -0.15);
57
+ border-top-color: var(--color-tertiary-500);
58
+ }
59
+
60
+ @keyframes spin {
61
+ 0% {
62
+ transform: rotate(0deg);
63
+ }
64
+ 100% {
65
+ transform: rotate(360deg);
66
+ }
67
+ }
68
+
69
+ .message {
70
+ font-size: var(--font-size-body-sm);
71
+ color: var(--color-text-secondary);
72
+ font-weight: var(--font-weight-medium);
73
+ margin: 0;
74
+ text-align: center;
75
+ }
76
+
77
+ /* Screen reader only class */
78
+ .sr-only {
79
+ position: absolute;
80
+ width: 1px;
81
+ height: 1px;
82
+ padding: 0;
83
+ margin: -1px;
84
+ overflow: hidden;
85
+ clip: rect(0, 0, 0, 0);
86
+ white-space: nowrap;
87
+ border-width: 0;
88
+ }
89
+
90
+ /* Respect reduced motion preference */
91
+ @media (prefers-reduced-motion: reduce) {
92
+ .ring {
93
+ animation: none;
94
+ border: 3px solid var(--color-primary-500);
95
+ opacity: 0.7;
96
+ }
97
+ }
@@ -0,0 +1,29 @@
1
+ import React from 'react';
2
+ import styles from './LoadingSpinner.module.css';
3
+
4
+ interface LoadingSpinnerProps {
5
+ size?: 'sm' | 'md' | 'lg';
6
+ message?: string;
7
+ }
8
+
9
+ /**
10
+ * Beautiful loading spinner component with optional message
11
+ */
12
+ export const LoadingSpinner: React.FC<LoadingSpinnerProps> = ({
13
+ size = 'md',
14
+ message
15
+ }) => {
16
+ return (
17
+ <div className={styles.container} role="status" aria-live="polite">
18
+ <div className={`${styles.spinner} ${styles[size]}`}>
19
+ <div className={styles.ring}></div>
20
+ <div className={styles.ring}></div>
21
+ <div className={styles.ring}></div>
22
+ </div>
23
+ {message && (
24
+ <p className={styles.message}>{message}</p>
25
+ )}
26
+ <span className="sr-only">Loading...</span>
27
+ </div>
28
+ );
29
+ };