@mooncompany/uplink-chat 0.5.0

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 @mooncompany/uplink-chat might be problematic. Click here for more details.

Files changed (158) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +185 -0
  3. package/bin/uplink.js +279 -0
  4. package/middleware/error-handler.js +69 -0
  5. package/package.json +93 -0
  6. package/public/css/agents.36b98c0f.css +1469 -0
  7. package/public/css/agents.css +1469 -0
  8. package/public/css/app.a6a7f8f5.css +2731 -0
  9. package/public/css/app.css +2731 -0
  10. package/public/css/artifacts.css +444 -0
  11. package/public/css/commands.css +55 -0
  12. package/public/css/connection.css +131 -0
  13. package/public/css/dashboard.css +233 -0
  14. package/public/css/developer.css +328 -0
  15. package/public/css/files.css +123 -0
  16. package/public/css/markdown.css +156 -0
  17. package/public/css/message-actions.css +278 -0
  18. package/public/css/mobile.css +614 -0
  19. package/public/css/panels-unified.css +483 -0
  20. package/public/css/premium.css +415 -0
  21. package/public/css/realtime.css +189 -0
  22. package/public/css/satellites.css +401 -0
  23. package/public/css/shortcuts.css +185 -0
  24. package/public/css/split-view.4def0262.css +673 -0
  25. package/public/css/split-view.css +673 -0
  26. package/public/css/theme-generator.css +391 -0
  27. package/public/css/themes.css +387 -0
  28. package/public/css/timestamps.css +54 -0
  29. package/public/css/variables.css +78 -0
  30. package/public/dist/bundle.b55050c4.js +15757 -0
  31. package/public/favicon.svg +24 -0
  32. package/public/img/agents/ada.png +0 -0
  33. package/public/img/agents/clarice.png +0 -0
  34. package/public/img/agents/dennis-nedry.png +0 -0
  35. package/public/img/agents/elliot-alderson.png +0 -0
  36. package/public/img/agents/main.png +0 -0
  37. package/public/img/agents/scotty.png +0 -0
  38. package/public/img/agents/top-flight-security.png +0 -0
  39. package/public/index.html +1083 -0
  40. package/public/js/agents-data.js +234 -0
  41. package/public/js/agents-ui.js +72 -0
  42. package/public/js/agents.js +1525 -0
  43. package/public/js/app.js +79 -0
  44. package/public/js/appearance-settings.js +111 -0
  45. package/public/js/artifacts.js +432 -0
  46. package/public/js/audio-queue.js +168 -0
  47. package/public/js/bootstrap.js +54 -0
  48. package/public/js/chat.js +1211 -0
  49. package/public/js/commands.js +581 -0
  50. package/public/js/connection-api.js +121 -0
  51. package/public/js/connection.js +1231 -0
  52. package/public/js/context-tracker.js +271 -0
  53. package/public/js/core.js +172 -0
  54. package/public/js/dashboard.js +452 -0
  55. package/public/js/developer.js +432 -0
  56. package/public/js/encryption.js +124 -0
  57. package/public/js/errors.js +122 -0
  58. package/public/js/event-bus.js +77 -0
  59. package/public/js/fetch-utils.js +171 -0
  60. package/public/js/file-handler.js +229 -0
  61. package/public/js/files.js +352 -0
  62. package/public/js/gateway-chat.js +538 -0
  63. package/public/js/logger.js +112 -0
  64. package/public/js/markdown.js +190 -0
  65. package/public/js/message-actions.js +431 -0
  66. package/public/js/message-renderer.js +288 -0
  67. package/public/js/missed-messages.js +235 -0
  68. package/public/js/mobile-debug.js +95 -0
  69. package/public/js/notifications.js +367 -0
  70. package/public/js/offline-queue.js +178 -0
  71. package/public/js/onboarding.js +543 -0
  72. package/public/js/panels.js +156 -0
  73. package/public/js/premium.js +412 -0
  74. package/public/js/realtime-voice.js +844 -0
  75. package/public/js/satellite-sync.js +256 -0
  76. package/public/js/satellite-ui.js +175 -0
  77. package/public/js/satellites.js +1516 -0
  78. package/public/js/settings.js +1087 -0
  79. package/public/js/shortcuts.js +381 -0
  80. package/public/js/split-chat.js +1234 -0
  81. package/public/js/split-resize.js +211 -0
  82. package/public/js/splitview.js +340 -0
  83. package/public/js/storage.js +408 -0
  84. package/public/js/streaming-handler.js +324 -0
  85. package/public/js/stt-settings.js +316 -0
  86. package/public/js/theme-generator.js +661 -0
  87. package/public/js/themes.js +164 -0
  88. package/public/js/timestamps.js +198 -0
  89. package/public/js/tts-settings.js +575 -0
  90. package/public/js/ui.js +267 -0
  91. package/public/js/update-notifier.js +143 -0
  92. package/public/js/utils/constants.js +165 -0
  93. package/public/js/utils/sanitize.js +93 -0
  94. package/public/js/utils/sse-parser.js +195 -0
  95. package/public/js/voice.js +883 -0
  96. package/public/manifest.json +58 -0
  97. package/public/moon_texture.jpg +0 -0
  98. package/public/sw.js +221 -0
  99. package/public/three.min.js +6 -0
  100. package/server/channel.js +529 -0
  101. package/server/chat.js +270 -0
  102. package/server/config-store.js +362 -0
  103. package/server/config.js +159 -0
  104. package/server/context.js +131 -0
  105. package/server/gateway-commands.js +211 -0
  106. package/server/gateway-proxy.js +318 -0
  107. package/server/index.js +22 -0
  108. package/server/logger.js +89 -0
  109. package/server/middleware/auth.js +188 -0
  110. package/server/middleware.js +218 -0
  111. package/server/openclaw-discover.js +308 -0
  112. package/server/premium/index.js +156 -0
  113. package/server/premium/license.js +140 -0
  114. package/server/realtime/bridge.js +837 -0
  115. package/server/realtime/index.js +349 -0
  116. package/server/realtime/tts-stream.js +446 -0
  117. package/server/routes/agents.js +564 -0
  118. package/server/routes/artifacts.js +174 -0
  119. package/server/routes/chat.js +311 -0
  120. package/server/routes/config-settings.js +345 -0
  121. package/server/routes/config.js +603 -0
  122. package/server/routes/files.js +307 -0
  123. package/server/routes/index.js +18 -0
  124. package/server/routes/media.js +451 -0
  125. package/server/routes/missed-messages.js +107 -0
  126. package/server/routes/premium.js +75 -0
  127. package/server/routes/push.js +156 -0
  128. package/server/routes/satellite.js +406 -0
  129. package/server/routes/status.js +251 -0
  130. package/server/routes/stt.js +35 -0
  131. package/server/routes/voice.js +260 -0
  132. package/server/routes/webhooks.js +203 -0
  133. package/server/routes.js +206 -0
  134. package/server/runtime-config.js +336 -0
  135. package/server/share.js +305 -0
  136. package/server/stt/faster-whisper.js +72 -0
  137. package/server/stt/groq.js +51 -0
  138. package/server/stt/index.js +196 -0
  139. package/server/stt/openai.js +49 -0
  140. package/server/sync.js +244 -0
  141. package/server/tailscale-https.js +175 -0
  142. package/server/tts.js +646 -0
  143. package/server/update-checker.js +172 -0
  144. package/server/utils/filename.js +129 -0
  145. package/server/utils.js +147 -0
  146. package/server/watchdog.js +318 -0
  147. package/server/websocket/broadcast.js +359 -0
  148. package/server/websocket/connections.js +339 -0
  149. package/server/websocket/index.js +215 -0
  150. package/server/websocket/routing.js +277 -0
  151. package/server/websocket/sync.js +102 -0
  152. package/server.js +404 -0
  153. package/utils/detect-tool-usage.js +93 -0
  154. package/utils/errors.js +158 -0
  155. package/utils/html-escape.js +84 -0
  156. package/utils/id-sanitize.js +94 -0
  157. package/utils/response.js +130 -0
  158. package/utils/with-retry.js +105 -0
@@ -0,0 +1,233 @@
1
+ /* ============================================
2
+ DASHBOARD STYLES
3
+ Usage analytics and stats
4
+ ============================================ */
5
+
6
+ .dashboard-overlay {
7
+ position: fixed;
8
+ top: 0;
9
+ left: 0;
10
+ right: 0;
11
+ bottom: 0;
12
+ background: var(--black-80);
13
+ backdrop-filter: blur(4px);
14
+ z-index: var(--z-toast);
15
+ display: none;
16
+ align-items: center;
17
+ justify-content: center;
18
+ /* Safe area support for notched devices */
19
+ padding: calc(var(--space-5) + env(safe-area-inset-top, 0px))
20
+ calc(var(--space-5) + env(safe-area-inset-right, 0px))
21
+ calc(var(--space-5) + env(safe-area-inset-bottom, 0px))
22
+ calc(var(--space-5) + env(safe-area-inset-left, 0px));
23
+ }
24
+
25
+ .dashboard-overlay.visible {
26
+ display: flex;
27
+ }
28
+
29
+ .dashboard-panel {
30
+ background: linear-gradient(180deg, rgba(10, 25, 50, 0.98) 0%, rgba(5, 15, 35, 0.98) 100%);
31
+ border: 1px solid var(--border);
32
+ border-radius: var(--radius-2xl);
33
+ width: 100%;
34
+ max-width: 500px;
35
+ max-height: 90vh;
36
+ overflow: hidden;
37
+ display: flex;
38
+ flex-direction: column;
39
+ animation: dashboardAppear 0.3s ease-out;
40
+ }
41
+
42
+ @keyframes dashboardAppear {
43
+ from {
44
+ opacity: 0;
45
+ transform: scale(0.95) translateY(20px);
46
+ }
47
+ to {
48
+ opacity: 1;
49
+ transform: scale(1) translateY(0);
50
+ }
51
+ }
52
+
53
+ .dashboard-header {
54
+ display: flex;
55
+ align-items: center;
56
+ justify-content: space-between;
57
+ padding: var(--space-4) var(--space-5);
58
+ border-bottom: 1px solid var(--border);
59
+ }
60
+
61
+ .dashboard-header h2 {
62
+ font-size: var(--text-lg);
63
+ font-weight: 600;
64
+ color: var(--text);
65
+ margin: 0;
66
+ }
67
+
68
+ .dashboard-close {
69
+ min-width: 44px;
70
+ min-height: 44px;
71
+ border: none;
72
+ border-radius: var(--radius-md);
73
+ background: var(--white-10);
74
+ color: var(--text-muted);
75
+ font-size: var(--text-xl);
76
+ cursor: pointer;
77
+ transition: all 0.2s;
78
+ display: flex;
79
+ align-items: center;
80
+ justify-content: center;
81
+ }
82
+
83
+ .dashboard-close:hover {
84
+ background: var(--error-20);
85
+ color: var(--error);
86
+ }
87
+
88
+ .dashboard-content {
89
+ flex: 1;
90
+ overflow-y: auto;
91
+ padding: var(--space-4) var(--space-5);
92
+ }
93
+
94
+ .dashboard-section {
95
+ margin-bottom: var(--space-6);
96
+ }
97
+
98
+ .dashboard-section:last-child {
99
+ margin-bottom: 0;
100
+ }
101
+
102
+ .dashboard-section h3 {
103
+ font-size: var(--text-sm);
104
+ font-weight: 500;
105
+ color: var(--text-muted);
106
+ margin: 0 0 var(--space-3) 0;
107
+ text-transform: uppercase;
108
+ letter-spacing: 0.5px;
109
+ }
110
+
111
+ /* Stat Grid */
112
+ .stat-grid {
113
+ display: grid;
114
+ grid-template-columns: repeat(2, 1fr);
115
+ gap: var(--space-3);
116
+ }
117
+
118
+ .stat-card {
119
+ background: var(--black-30);
120
+ border: 1px solid var(--border);
121
+ border-radius: var(--radius-lg);
122
+ padding: var(--space-4);
123
+ text-align: center;
124
+ }
125
+
126
+ .stat-value {
127
+ display: block;
128
+ font-size: var(--text-2xl);
129
+ font-weight: 600;
130
+ color: var(--accent);
131
+ margin-bottom: var(--space-1);
132
+ }
133
+
134
+ .stat-label {
135
+ font-size: var(--text-xs);
136
+ color: var(--text-muted);
137
+ text-transform: uppercase;
138
+ letter-spacing: 0.5px;
139
+ }
140
+
141
+ /* Charts */
142
+ .chart-container {
143
+ background: var(--black-20);
144
+ border: 1px solid var(--border);
145
+ border-radius: var(--radius-lg);
146
+ padding: var(--space-4);
147
+ }
148
+
149
+ .chart-container canvas {
150
+ width: 100%;
151
+ height: auto;
152
+ }
153
+
154
+ /* Actions */
155
+ .dashboard-actions {
156
+ display: flex;
157
+ gap: var(--space-3);
158
+ margin-top: var(--space-5);
159
+ padding-top: var(--space-4);
160
+ border-top: 1px solid var(--border);
161
+ }
162
+
163
+ .dashboard-btn {
164
+ flex: 1;
165
+ padding: var(--space-2) var(--space-4);
166
+ border: 1px solid var(--border);
167
+ border-radius: var(--radius-md);
168
+ background: var(--accent-10);
169
+ color: var(--accent);
170
+ font-size: var(--text-sm);
171
+ cursor: pointer;
172
+ transition: all 0.2s;
173
+ }
174
+
175
+ .dashboard-btn:hover {
176
+ background: var(--accent-20);
177
+ border-color: var(--accent);
178
+ }
179
+
180
+ .dashboard-btn.danger {
181
+ background: var(--error-10);
182
+ color: var(--error);
183
+ border-color: var(--error-30);
184
+ }
185
+
186
+ .dashboard-btn.danger:hover {
187
+ background: var(--error-20);
188
+ border-color: var(--error);
189
+ }
190
+
191
+ /* Scrollbar */
192
+ .dashboard-content::-webkit-scrollbar {
193
+ width: 6px;
194
+ }
195
+
196
+ .dashboard-content::-webkit-scrollbar-track {
197
+ background: transparent;
198
+ }
199
+
200
+ .dashboard-content::-webkit-scrollbar-thumb {
201
+ background: var(--border);
202
+ border-radius: 3px;
203
+ }
204
+
205
+ /* Responsive (slightly above --breakpoint-sm for dashboard) */
206
+ @media (max-width: 500px) {
207
+ .dashboard-panel {
208
+ max-height: 100vh;
209
+ border-radius: 0;
210
+ /* Safe area for full-screen panel on notched devices */
211
+ padding-top: env(safe-area-inset-top, 0px);
212
+ padding-bottom: env(safe-area-inset-bottom, 0px);
213
+ }
214
+
215
+ .stat-grid {
216
+ grid-template-columns: repeat(2, 1fr);
217
+ }
218
+
219
+ .stat-value {
220
+ font-size: var(--text-xl);
221
+ }
222
+ }
223
+
224
+ /* Accessibility: Respect reduced motion preference (M-09) */
225
+ @media (prefers-reduced-motion: reduce) {
226
+ *,
227
+ *::before,
228
+ *::after {
229
+ animation-duration: 0.01ms !important;
230
+ animation-iteration-count: 1 !important;
231
+ transition-duration: 0.01ms !important;
232
+ }
233
+ }
@@ -0,0 +1,328 @@
1
+ /* ============================================
2
+ ACTIVITY PANEL STYLES
3
+ ============================================ */
4
+
5
+ /* Floating toggle button removed - now using Activity tab in mode-tabs */
6
+
7
+ .dev-panel {
8
+ position: fixed;
9
+ bottom: 160px;
10
+ right: var(--space-4);
11
+ width: 360px;
12
+ max-width: calc(100vw - var(--space-8));
13
+ max-height: 400px;
14
+ background: var(--bg-secondary);
15
+ border: 1px solid var(--border);
16
+ border-radius: var(--radius-xl);
17
+ z-index: var(--z-modal);
18
+ display: none;
19
+ flex-direction: column;
20
+ overflow: hidden;
21
+ box-shadow: 0 10px 40px var(--black-40);
22
+ }
23
+
24
+ .dev-panel.visible {
25
+ display: flex;
26
+ }
27
+
28
+ .dev-panel-header {
29
+ display: flex;
30
+ align-items: center;
31
+ justify-content: space-between;
32
+ padding: var(--space-3) var(--space-4);
33
+ border-bottom: 1px solid var(--border);
34
+ background: var(--black-20);
35
+ }
36
+
37
+ .dev-panel-title {
38
+ font-family: 'Space Mono', monospace;
39
+ font-size: var(--text-sm);
40
+ color: var(--accent);
41
+ letter-spacing: 1px;
42
+ text-transform: uppercase;
43
+ }
44
+
45
+ .dev-panel-close {
46
+ background: none;
47
+ border: none;
48
+ color: var(--text-muted);
49
+ font-size: var(--text-2xl);
50
+ cursor: pointer;
51
+ min-width: 44px;
52
+ min-height: 44px;
53
+ display: flex;
54
+ align-items: center;
55
+ justify-content: center;
56
+ line-height: 1;
57
+ border-radius: var(--radius-md);
58
+ }
59
+
60
+ .dev-panel-close:hover {
61
+ color: var(--text);
62
+ }
63
+
64
+ .dev-panel-tabs {
65
+ display: flex;
66
+ border-bottom: 1px solid var(--border);
67
+ }
68
+
69
+ .dev-tab {
70
+ flex: 1;
71
+ padding: var(--space-2) var(--space-2);
72
+ background: none;
73
+ border: none;
74
+ color: var(--text-muted);
75
+ font-size: var(--text-xs);
76
+ cursor: pointer;
77
+ transition: all 0.15s;
78
+ border-bottom: 2px solid transparent;
79
+ }
80
+
81
+ .dev-tab:hover {
82
+ color: var(--text);
83
+ background: var(--accent-05);
84
+ }
85
+
86
+ .dev-tab.active {
87
+ color: var(--accent);
88
+ border-bottom-color: var(--accent);
89
+ }
90
+
91
+ .dev-panel-content {
92
+ flex: 1;
93
+ overflow: hidden;
94
+ }
95
+
96
+ .dev-tab-content {
97
+ display: none;
98
+ height: 100%;
99
+ padding: var(--space-4);
100
+ overflow-y: auto;
101
+ }
102
+
103
+ .dev-tab-content.active {
104
+ display: block;
105
+ }
106
+
107
+ /* Token Stats */
108
+ .token-stats {
109
+ display: grid;
110
+ grid-template-columns: 1fr 1fr;
111
+ gap: var(--space-3);
112
+ margin-bottom: var(--space-4);
113
+ }
114
+
115
+ .token-stat {
116
+ background: var(--black-20);
117
+ border: 1px solid var(--border);
118
+ border-radius: var(--radius-md);
119
+ padding: var(--space-3);
120
+ text-align: center;
121
+ }
122
+
123
+ .token-stat.total {
124
+ grid-column: span 2;
125
+ background: var(--accent-05);
126
+ border-color: var(--accent);
127
+ }
128
+
129
+ .token-label {
130
+ display: block;
131
+ font-size: var(--text-xs);
132
+ color: var(--text-muted);
133
+ text-transform: uppercase;
134
+ letter-spacing: 0.5px;
135
+ margin-bottom: var(--space-1);
136
+ }
137
+
138
+ .token-value {
139
+ font-family: 'Space Mono', monospace;
140
+ font-size: var(--text-xl);
141
+ color: var(--text);
142
+ }
143
+
144
+ .token-session {
145
+ display: flex;
146
+ justify-content: space-between;
147
+ align-items: center;
148
+ padding: var(--space-3);
149
+ background: var(--black-20);
150
+ border: 1px solid var(--border);
151
+ border-radius: var(--radius-md);
152
+ margin-bottom: var(--space-3);
153
+ }
154
+
155
+ .token-session .token-label {
156
+ margin-bottom: 0;
157
+ }
158
+
159
+ .token-session .token-value {
160
+ color: var(--accent);
161
+ }
162
+
163
+ .dev-btn {
164
+ width: 100%;
165
+ padding: var(--space-2);
166
+ background: var(--bg-input);
167
+ border: 1px solid var(--border);
168
+ border-radius: var(--radius-md);
169
+ color: var(--text-muted);
170
+ font-size: var(--text-sm);
171
+ cursor: pointer;
172
+ transition: all 0.15s;
173
+ }
174
+
175
+ .dev-btn:hover {
176
+ border-color: var(--accent);
177
+ color: var(--accent);
178
+ }
179
+
180
+ /* API Logs */
181
+ .api-log-list,
182
+ .tool-list,
183
+ .error-list {
184
+ display: flex;
185
+ flex-direction: column;
186
+ gap: var(--space-2);
187
+ }
188
+
189
+ .empty-log {
190
+ text-align: center;
191
+ color: var(--text-dim);
192
+ font-size: var(--text-sm);
193
+ padding: var(--space-5);
194
+ }
195
+
196
+ .api-log-entry,
197
+ .tool-entry,
198
+ .error-entry {
199
+ background: var(--black-20);
200
+ border: 1px solid var(--border);
201
+ border-radius: var(--radius-md);
202
+ padding: var(--space-2);
203
+ }
204
+
205
+ .api-log-header,
206
+ .tool-header,
207
+ .error-header {
208
+ display: flex;
209
+ align-items: center;
210
+ gap: var(--space-2);
211
+ flex-wrap: wrap;
212
+ }
213
+
214
+ .api-method {
215
+ font-family: 'Space Mono', monospace;
216
+ font-size: var(--text-xs);
217
+ padding: 2px var(--space-2);
218
+ border-radius: var(--radius-sm);
219
+ background: var(--bg-input);
220
+ color: var(--text-muted);
221
+ }
222
+
223
+ .api-method.POST { background: var(--success-20); color: var(--success); }
224
+ .api-method.GET { background: var(--accent-20); color: var(--accent); }
225
+ .api-method.DELETE { background: var(--error-20); color: var(--error); }
226
+
227
+ .api-url {
228
+ font-size: var(--text-sm);
229
+ color: var(--text);
230
+ flex: 1;
231
+ white-space: nowrap;
232
+ overflow: hidden;
233
+ text-overflow: ellipsis;
234
+ }
235
+
236
+ .api-status {
237
+ font-family: 'Space Mono', monospace;
238
+ font-size: var(--text-xs);
239
+ padding: 2px var(--space-2);
240
+ border-radius: var(--radius-sm);
241
+ }
242
+
243
+ .api-status.status-2 { background: var(--success-20); color: var(--success); }
244
+ .api-status.status-4,
245
+ .api-status.status-5 { background: var(--error-20); color: var(--error); }
246
+
247
+ .api-duration {
248
+ font-family: 'Space Mono', monospace;
249
+ font-size: var(--text-xs);
250
+ color: var(--text-dim);
251
+ }
252
+
253
+ .api-response,
254
+ .tool-args {
255
+ margin-top: var(--space-2);
256
+ }
257
+
258
+ .api-response summary,
259
+ .tool-args summary {
260
+ font-size: var(--text-xs);
261
+ color: var(--text-muted);
262
+ cursor: pointer;
263
+ }
264
+
265
+ .api-response pre,
266
+ .tool-args pre {
267
+ margin-top: var(--space-2);
268
+ padding: var(--space-2);
269
+ background: var(--black-30);
270
+ border-radius: var(--radius-sm);
271
+ font-size: var(--text-xs);
272
+ overflow-x: auto;
273
+ color: var(--text);
274
+ white-space: pre-wrap;
275
+ word-break: break-all;
276
+ }
277
+
278
+ /* Tool Calls */
279
+ .tool-name {
280
+ font-family: 'Space Mono', monospace;
281
+ font-size: var(--text-sm);
282
+ color: var(--accent);
283
+ }
284
+
285
+ .tool-time,
286
+ .error-time {
287
+ font-size: var(--text-xs);
288
+ color: var(--text-dim);
289
+ margin-left: auto;
290
+ }
291
+
292
+ /* Errors */
293
+ .error-entry {
294
+ border-color: var(--error-30);
295
+ }
296
+
297
+ .error-message {
298
+ font-size: var(--text-sm);
299
+ color: var(--error);
300
+ flex: 1;
301
+ }
302
+
303
+ .error-context {
304
+ font-size: var(--text-xs);
305
+ color: var(--text-dim);
306
+ margin-top: var(--space-2);
307
+ }
308
+
309
+ /* Mobile adjustments (--breakpoint-sm: 480px) */
310
+ @media (max-width: 480px) {
311
+ .dev-panel {
312
+ bottom: 140px;
313
+ left: var(--space-4);
314
+ right: var(--space-4);
315
+ width: auto;
316
+ }
317
+ }
318
+
319
+ /* Accessibility: Respect reduced motion preference (M-09) */
320
+ @media (prefers-reduced-motion: reduce) {
321
+ *,
322
+ *::before,
323
+ *::after {
324
+ animation-duration: 0.01ms !important;
325
+ animation-iteration-count: 1 !important;
326
+ transition-duration: 0.01ms !important;
327
+ }
328
+ }
@@ -0,0 +1,123 @@
1
+ /* ============================================
2
+ FILES MODULE STYLES
3
+ Drag & drop, file preview enhancements
4
+ ============================================ */
5
+
6
+ /* Drop zone overlay */
7
+ .drop-zone {
8
+ position: absolute;
9
+ top: 0;
10
+ left: 0;
11
+ right: 0;
12
+ bottom: 0;
13
+ background: rgba(0, 0, 0, 0.85);
14
+ display: none;
15
+ align-items: center;
16
+ justify-content: center;
17
+ z-index: var(--z-dropdown);
18
+ pointer-events: none;
19
+ }
20
+
21
+ .drop-zone.visible {
22
+ display: flex;
23
+ pointer-events: auto;
24
+ }
25
+
26
+ .drop-zone-content {
27
+ text-align: center;
28
+ padding: var(--space-10);
29
+ border: 2px dashed var(--accent);
30
+ border-radius: var(--radius-2xl);
31
+ background: rgba(0, 240, 255, 0.05);
32
+ animation: drop-pulse 1.5s ease-in-out infinite;
33
+ }
34
+
35
+ @keyframes drop-pulse {
36
+ 0%, 100% {
37
+ transform: scale(1);
38
+ border-color: var(--accent);
39
+ }
40
+ 50% {
41
+ transform: scale(1.02);
42
+ border-color: var(--success);
43
+ }
44
+ }
45
+
46
+ .drop-zone-icon {
47
+ font-size: 3rem;
48
+ margin-bottom: var(--space-4);
49
+ animation: bounce-soft 1s ease-in-out infinite;
50
+ }
51
+
52
+ @keyframes bounce-soft {
53
+ 0%, 100% { transform: translateY(0); }
54
+ 50% { transform: translateY(-8px); }
55
+ }
56
+
57
+ .drop-zone-text {
58
+ font-size: var(--text-lg);
59
+ color: var(--text);
60
+ margin-bottom: var(--space-2);
61
+ }
62
+
63
+ .drop-zone-hint {
64
+ font-size: var(--text-sm);
65
+ color: var(--text-muted);
66
+ }
67
+
68
+ /* Enhanced file preview */
69
+ .file-label {
70
+ display: flex;
71
+ align-items: center;
72
+ gap: var(--space-2);
73
+ padding: var(--space-3) var(--space-4);
74
+ background: var(--bg-input);
75
+ border: 1px solid var(--border);
76
+ border-radius: var(--radius-lg);
77
+ font-size: var(--text-sm);
78
+ color: var(--text);
79
+ max-width: 250px;
80
+ }
81
+
82
+ .file-label .file-size {
83
+ color: var(--text-dim);
84
+ font-size: var(--text-xs);
85
+ }
86
+
87
+ /* Code file preview - syntax hint */
88
+ .file-preview-code {
89
+ background: rgba(0, 20, 40, 0.9);
90
+ border: 1px solid var(--border);
91
+ border-radius: var(--radius-md);
92
+ padding: var(--space-3);
93
+ font-family: 'Space Mono', monospace;
94
+ font-size: var(--text-xs);
95
+ max-height: 150px;
96
+ overflow: auto;
97
+ white-space: pre-wrap;
98
+ color: var(--text);
99
+ margin-top: var(--space-2);
100
+ }
101
+
102
+ /* Text file icon colors */
103
+ .file-icon-js { color: var(--file-js); }
104
+ .file-icon-ts { color: var(--file-ts); }
105
+ .file-icon-py { color: var(--file-py); }
106
+ .file-icon-json { color: var(--file-json); }
107
+ .file-icon-html { color: var(--file-html); }
108
+ .file-icon-css { color: var(--file-css); }
109
+ .file-icon-md { color: var(--file-md); }
110
+
111
+ /* Image preview enhancements */
112
+ .image-preview img {
113
+ border: 1px solid var(--border);
114
+ box-shadow: 0 4px 12px rgba(0, 0, 0, 0.3);
115
+ }
116
+
117
+ /* Preview with code content */
118
+ .file-preview-wrapper {
119
+ display: flex;
120
+ flex-direction: column;
121
+ gap: var(--space-2);
122
+ max-width: 300px;
123
+ }