@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,444 @@
1
+ /* ============================================
2
+ ARTIFACTS PANEL
3
+ Document viewer for agent-generated artifacts
4
+ ============================================ */
5
+
6
+ .artifacts-panel {
7
+ position: fixed;
8
+ top: 0;
9
+ right: 0;
10
+ width: 400px;
11
+ height: 100vh;
12
+ background: var(--bg);
13
+ border-left: 1px solid var(--border);
14
+ transform: translateX(100%);
15
+ transition: transform 0.3s ease;
16
+ z-index: 1000;
17
+ display: flex;
18
+ flex-direction: column;
19
+ box-shadow: -2px 0 12px rgba(0, 0, 0, 0.1);
20
+ }
21
+
22
+ .artifacts-panel.visible {
23
+ transform: translateX(0);
24
+ }
25
+
26
+ /* Header */
27
+ .artifacts-panel-header {
28
+ display: flex;
29
+ align-items: center;
30
+ gap: 8px;
31
+ padding: 16px;
32
+ border-bottom: 1px solid var(--border);
33
+ background: var(--bg-secondary);
34
+ }
35
+
36
+ .artifacts-panel-title {
37
+ flex: 1;
38
+ font-size: 14px;
39
+ font-weight: 600;
40
+ letter-spacing: 0.05em;
41
+ color: var(--text-secondary);
42
+ }
43
+
44
+ .artifacts-panel-refresh {
45
+ background: none;
46
+ border: none;
47
+ color: var(--text-secondary);
48
+ cursor: pointer;
49
+ padding: 6px;
50
+ border-radius: 4px;
51
+ display: flex;
52
+ align-items: center;
53
+ justify-content: center;
54
+ transition: background 0.2s, color 0.2s;
55
+ }
56
+
57
+ .artifacts-panel-refresh:hover {
58
+ background: var(--hover);
59
+ color: var(--text);
60
+ }
61
+
62
+ .artifacts-panel-close {
63
+ background: none;
64
+ border: none;
65
+ color: var(--text-secondary);
66
+ cursor: pointer;
67
+ font-size: 24px;
68
+ line-height: 1;
69
+ padding: 0;
70
+ width: 32px;
71
+ height: 32px;
72
+ display: flex;
73
+ align-items: center;
74
+ justify-content: center;
75
+ border-radius: 4px;
76
+ transition: background 0.2s, color 0.2s;
77
+ }
78
+
79
+ .artifacts-panel-close:hover {
80
+ background: var(--hover);
81
+ color: var(--text);
82
+ }
83
+
84
+ /* Content */
85
+ .artifacts-panel-content {
86
+ flex: 1;
87
+ overflow-y: auto;
88
+ padding: 16px;
89
+ display: flex;
90
+ flex-direction: column;
91
+ gap: 12px;
92
+ }
93
+
94
+ /* Search */
95
+ .artifacts-search {
96
+ width: 100%;
97
+ padding: 10px 12px;
98
+ border: 1px solid var(--border);
99
+ border-radius: 8px;
100
+ background: var(--bg-secondary);
101
+ color: var(--text);
102
+ font-size: 14px;
103
+ font-family: inherit;
104
+ transition: border-color 0.2s;
105
+ }
106
+
107
+ .artifacts-search:focus {
108
+ outline: none;
109
+ border-color: var(--accent);
110
+ }
111
+
112
+ .artifacts-search::placeholder {
113
+ color: var(--text-secondary);
114
+ opacity: 0.7;
115
+ }
116
+
117
+ /* List */
118
+ .artifacts-list {
119
+ display: flex;
120
+ flex-direction: column;
121
+ gap: 8px;
122
+ }
123
+
124
+ .artifact-item {
125
+ display: flex;
126
+ align-items: center;
127
+ gap: 12px;
128
+ padding: 12px;
129
+ border: 1px solid var(--border);
130
+ border-radius: 8px;
131
+ cursor: pointer;
132
+ transition: background 0.2s, border-color 0.2s, transform 0.1s;
133
+ }
134
+
135
+ .artifact-item:hover {
136
+ background: var(--hover);
137
+ border-color: var(--accent);
138
+ transform: translateY(-1px);
139
+ }
140
+
141
+ .artifact-item:active {
142
+ transform: translateY(0);
143
+ }
144
+
145
+ .artifact-icon {
146
+ font-size: 24px;
147
+ line-height: 1;
148
+ flex-shrink: 0;
149
+ }
150
+
151
+ .artifact-info {
152
+ flex: 1;
153
+ min-width: 0;
154
+ }
155
+
156
+ .artifact-name {
157
+ font-size: 14px;
158
+ font-weight: 500;
159
+ color: var(--text);
160
+ overflow: hidden;
161
+ text-overflow: ellipsis;
162
+ white-space: nowrap;
163
+ }
164
+
165
+ .artifact-meta {
166
+ font-size: 12px;
167
+ color: var(--text-secondary);
168
+ margin-top: 2px;
169
+ }
170
+
171
+ /* Empty state */
172
+ .artifacts-empty {
173
+ display: flex;
174
+ flex-direction: column;
175
+ align-items: center;
176
+ justify-content: center;
177
+ padding: 48px 24px;
178
+ text-align: center;
179
+ color: var(--text-secondary);
180
+ }
181
+
182
+ .artifacts-empty svg {
183
+ margin-bottom: 16px;
184
+ opacity: 0.5;
185
+ }
186
+
187
+ .artifacts-empty p {
188
+ font-size: 16px;
189
+ font-weight: 500;
190
+ margin: 0 0 8px;
191
+ }
192
+
193
+ .artifacts-empty small {
194
+ font-size: 13px;
195
+ opacity: 0.7;
196
+ }
197
+
198
+ /* Error state */
199
+ .artifacts-error {
200
+ padding: 24px;
201
+ text-align: center;
202
+ color: var(--error);
203
+ background: var(--error-bg, rgba(239, 68, 68, 0.1));
204
+ border-radius: 8px;
205
+ border: 1px solid var(--error);
206
+ }
207
+
208
+ /* Reader */
209
+ .artifacts-reader {
210
+ display: flex;
211
+ flex-direction: column;
212
+ gap: 16px;
213
+ }
214
+
215
+ .artifacts-reader-toolbar {
216
+ display: flex;
217
+ align-items: center;
218
+ justify-content: space-between;
219
+ gap: 8px;
220
+ }
221
+
222
+ .artifacts-reader-back,
223
+ .artifacts-reader-download {
224
+ align-self: flex-start;
225
+ display: flex;
226
+ align-items: center;
227
+ gap: 6px;
228
+ background: none;
229
+ border: 1px solid var(--border);
230
+ color: var(--text-secondary);
231
+ cursor: pointer;
232
+ padding: 8px 12px;
233
+ border-radius: 6px;
234
+ font-size: 14px;
235
+ font-family: inherit;
236
+ transition: background 0.2s, color 0.2s, border-color 0.2s;
237
+ }
238
+
239
+ .artifacts-reader-back:hover,
240
+ .artifacts-reader-download:hover {
241
+ background: var(--hover);
242
+ color: var(--text);
243
+ border-color: var(--accent);
244
+ }
245
+
246
+ .artifacts-reader-title {
247
+ font-size: 18px;
248
+ font-weight: 600;
249
+ color: var(--text);
250
+ margin: 0;
251
+ word-break: break-word;
252
+ }
253
+
254
+ .artifacts-reader-content {
255
+ font-size: 14px;
256
+ line-height: 1.6;
257
+ color: var(--text);
258
+ overflow-wrap: break-word;
259
+ }
260
+
261
+ /* Reader content styling (markdown) */
262
+ .artifacts-reader-content h1,
263
+ .artifacts-reader-content h2,
264
+ .artifacts-reader-content h3,
265
+ .artifacts-reader-content h4 {
266
+ margin-top: 24px;
267
+ margin-bottom: 12px;
268
+ font-weight: 600;
269
+ line-height: 1.3;
270
+ }
271
+
272
+ .artifacts-reader-content h1 {
273
+ font-size: 24px;
274
+ }
275
+
276
+ .artifacts-reader-content h2 {
277
+ font-size: 20px;
278
+ }
279
+
280
+ .artifacts-reader-content h3 {
281
+ font-size: 18px;
282
+ }
283
+
284
+ .artifacts-reader-content h4 {
285
+ font-size: 16px;
286
+ }
287
+
288
+ .artifacts-reader-content p {
289
+ margin: 0 0 12px;
290
+ }
291
+
292
+ .artifacts-reader-content ul,
293
+ .artifacts-reader-content ol {
294
+ margin: 0 0 12px;
295
+ padding-left: 24px;
296
+ }
297
+
298
+ .artifacts-reader-content li {
299
+ margin-bottom: 6px;
300
+ }
301
+
302
+ .artifacts-reader-content code {
303
+ background: var(--bg-secondary);
304
+ border: 1px solid var(--border);
305
+ padding: 2px 6px;
306
+ border-radius: 4px;
307
+ font-family: 'Space Mono', monospace;
308
+ font-size: 0.9em;
309
+ }
310
+
311
+ .artifacts-reader-content pre {
312
+ background: var(--bg-secondary);
313
+ border: 1px solid var(--border);
314
+ padding: 12px;
315
+ border-radius: 8px;
316
+ overflow-x: auto;
317
+ margin: 0 0 12px;
318
+ }
319
+
320
+ .artifacts-reader-content pre code {
321
+ background: none;
322
+ border: none;
323
+ padding: 0;
324
+ }
325
+
326
+ .artifacts-reader-content blockquote {
327
+ border-left: 3px solid var(--accent);
328
+ padding-left: 16px;
329
+ margin: 0 0 12px;
330
+ color: var(--text-secondary);
331
+ font-style: italic;
332
+ }
333
+
334
+ .artifacts-reader-content table {
335
+ width: 100%;
336
+ border-collapse: collapse;
337
+ margin: 0 0 12px;
338
+ font-size: 13px;
339
+ }
340
+
341
+ .artifacts-reader-content table th,
342
+ .artifacts-reader-content table td {
343
+ border: 1px solid var(--border);
344
+ padding: 8px;
345
+ text-align: left;
346
+ }
347
+
348
+ .artifacts-reader-content table th {
349
+ background: var(--bg-secondary);
350
+ font-weight: 600;
351
+ }
352
+
353
+ .artifacts-reader-content hr {
354
+ border: none;
355
+ border-top: 1px solid var(--border);
356
+ margin: 24px 0;
357
+ }
358
+
359
+ /* HTML Preview/Source Toggle */
360
+ .artifacts-html-toggle {
361
+ display: flex;
362
+ gap: 4px;
363
+ margin-bottom: 12px;
364
+ padding: 3px;
365
+ background: var(--bg-secondary);
366
+ border-radius: 6px;
367
+ width: fit-content;
368
+ }
369
+
370
+ .artifacts-toggle-btn {
371
+ padding: 6px 14px;
372
+ border: none;
373
+ background: transparent;
374
+ color: var(--text-secondary);
375
+ font-size: 13px;
376
+ font-weight: 500;
377
+ border-radius: 4px;
378
+ cursor: pointer;
379
+ transition: all 0.15s ease;
380
+ }
381
+
382
+ .artifacts-toggle-btn:hover {
383
+ color: var(--text);
384
+ }
385
+
386
+ .artifacts-toggle-btn.active {
387
+ background: var(--accent, #785aff);
388
+ color: #fff;
389
+ }
390
+
391
+ .artifacts-html-preview {
392
+ overflow-x: auto;
393
+ -webkit-overflow-scrolling: touch;
394
+ }
395
+
396
+ .artifacts-iframe {
397
+ width: 1200px;
398
+ min-height: 500px;
399
+ border: 1px solid var(--border);
400
+ border-radius: 6px;
401
+ background: #fff;
402
+ }
403
+
404
+ .artifacts-html-source pre {
405
+ margin: 0;
406
+ white-space: pre-wrap;
407
+ word-break: break-word;
408
+ font-size: 12px;
409
+ line-height: 1.6;
410
+ }
411
+
412
+ /* Deep links in chat messages */
413
+ .artifact-link {
414
+ display: inline-flex;
415
+ align-items: center;
416
+ gap: 4px;
417
+ padding: 2px 8px;
418
+ background: var(--bg-secondary, rgba(120, 90, 255, 0.1));
419
+ border: 1px solid var(--border, rgba(120, 90, 255, 0.2));
420
+ border-radius: 4px;
421
+ color: var(--accent, #785aff);
422
+ text-decoration: none;
423
+ font-size: 13px;
424
+ font-weight: 500;
425
+ cursor: pointer;
426
+ transition: all 0.15s ease;
427
+ }
428
+
429
+ .artifact-link:hover {
430
+ background: var(--accent, #785aff);
431
+ color: #fff;
432
+ text-decoration: none;
433
+ }
434
+
435
+ /* Mobile */
436
+ @media (max-width: 768px) {
437
+ .artifacts-panel {
438
+ width: 100%;
439
+ }
440
+
441
+ .artifacts-iframe {
442
+ min-height: 400px;
443
+ }
444
+ }
@@ -0,0 +1,55 @@
1
+ /* ============================================
2
+ SLASH COMMANDS STYLES
3
+ ============================================ */
4
+
5
+ .command-autocomplete {
6
+ position: absolute;
7
+ bottom: 100%;
8
+ left: var(--space-4);
9
+ right: var(--space-4);
10
+ background: var(--bg-secondary);
11
+ border: 1px solid var(--border);
12
+ border-radius: var(--radius-lg);
13
+ margin-bottom: var(--space-2);
14
+ overflow: hidden;
15
+ box-shadow: 0 -4px 20px rgba(0, 0, 0, 0.3);
16
+ z-index: var(--z-dropdown);
17
+ }
18
+
19
+ .command-option {
20
+ display: flex;
21
+ align-items: center;
22
+ justify-content: space-between;
23
+ padding: 12px 16px;
24
+ min-height: 48px; /* Touch target minimum */
25
+ cursor: pointer;
26
+ transition: background 0.15s;
27
+ border-bottom: 1px solid var(--border);
28
+ }
29
+
30
+ .command-option:last-child {
31
+ border-bottom: none;
32
+ }
33
+
34
+ .command-option:hover,
35
+ .command-option.selected {
36
+ background: rgba(0, 240, 255, 0.1);
37
+ }
38
+
39
+ .command-name {
40
+ font-family: 'Space Mono', monospace;
41
+ font-size: var(--text-sm);
42
+ color: var(--accent);
43
+ font-weight: 500;
44
+ }
45
+
46
+ .command-desc {
47
+ font-size: var(--text-xs);
48
+ color: var(--text-muted);
49
+ }
50
+
51
+ /* System messages with markdown */
52
+ .message.system strong {
53
+ color: var(--accent);
54
+ font-weight: 600;
55
+ }
@@ -0,0 +1,131 @@
1
+ /* ============================================
2
+ CONNECTION STATUS STYLES
3
+ ============================================ */
4
+
5
+ .status-dot {
6
+ width: var(--space-2);
7
+ height: var(--space-2);
8
+ border-radius: var(--radius-full);
9
+ display: inline-block;
10
+ margin-right: var(--space-2);
11
+ transition: all 0.3s;
12
+ background: var(--neutral); /* Default: neutral gray until JS sets state */
13
+ }
14
+
15
+ .status-dot.connected {
16
+ background: var(--success);
17
+ box-shadow: 0 0 var(--space-2) var(--success);
18
+ }
19
+
20
+ .status-dot.disconnected {
21
+ background: var(--warning);
22
+ animation: pulse-yellow 2s infinite;
23
+ }
24
+
25
+ .status-dot.connecting,
26
+ .status-dot.reconnecting {
27
+ background: var(--info);
28
+ animation: pulse-blue 1s infinite;
29
+ }
30
+
31
+ .status-dot.offline {
32
+ background: var(--neutral);
33
+ }
34
+
35
+ .status-dot.error,
36
+ .status-dot.failed {
37
+ background: var(--error);
38
+ box-shadow: 0 0 var(--space-2) var(--error);
39
+ }
40
+
41
+ @keyframes pulse-yellow {
42
+ 0%, 100% { opacity: 1; }
43
+ 50% { opacity: 0.5; }
44
+ }
45
+
46
+ @keyframes pulse-blue {
47
+ 0%, 100% { transform: scale(1); }
48
+ 50% { transform: scale(1.2); }
49
+ }
50
+
51
+ /* Status text */
52
+ #status {
53
+ cursor: pointer;
54
+ user-select: none;
55
+ }
56
+
57
+ #status:hover {
58
+ opacity: 0.8;
59
+ }
60
+
61
+ /* Connection mode badge */
62
+ .connection-mode-badge {
63
+ font-size: var(--text-xs);
64
+ opacity: 0.7;
65
+ }
66
+
67
+ .connection-mode-badge .mode-dot {
68
+ width: 6px;
69
+ height: 6px;
70
+ border-radius: var(--radius-full);
71
+ display: inline-block;
72
+ margin-right: var(--space-1);
73
+ }
74
+
75
+ .connection-mode-badge.direct .mode-dot {
76
+ background: var(--success);
77
+ }
78
+
79
+ .connection-mode-badge.proxied .mode-dot {
80
+ background: var(--info);
81
+ }
82
+
83
+ /* ============================================
84
+ CONTEXT TRACKER
85
+ ============================================ */
86
+
87
+ .context-badge {
88
+ display: none; /* Hidden until data arrives */
89
+ align-items: center;
90
+ gap: 6px;
91
+ font-family: 'Space Mono', monospace;
92
+ font-size: 10px;
93
+ letter-spacing: 0.5px;
94
+ color: var(--text-dim);
95
+ cursor: default;
96
+ user-select: none;
97
+ padding: 2px 0;
98
+ }
99
+
100
+ .context-bar {
101
+ width: 60px;
102
+ height: 4px;
103
+ background: rgba(255, 255, 255, 0.08);
104
+ border-radius: var(--radius-full);
105
+ overflow: hidden;
106
+ flex-shrink: 0;
107
+ }
108
+
109
+ .context-bar-fill {
110
+ height: 100%;
111
+ border-radius: var(--radius-full);
112
+ background: var(--success);
113
+ transition: width 0.6s ease, background 0.6s ease;
114
+ min-width: 1px;
115
+ }
116
+
117
+ .context-text {
118
+ white-space: nowrap;
119
+ opacity: 0.8;
120
+ }
121
+
122
+ /* Accessibility: Respect reduced motion preference (M-09) */
123
+ @media (prefers-reduced-motion: reduce) {
124
+ *,
125
+ *::before,
126
+ *::after {
127
+ animation-duration: 0.01ms !important;
128
+ animation-iteration-count: 1 !important;
129
+ transition-duration: 0.01ms !important;
130
+ }
131
+ }