token-studio 4.8.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.
Files changed (139) hide show
  1. package/.nvmrc +1 -0
  2. package/CHANGELOG.md +89 -0
  3. package/Dockerfile +17 -0
  4. package/LICENSE +22 -0
  5. package/NOTICE.md +21 -0
  6. package/PRIVACY.md +68 -0
  7. package/README.en.md +220 -0
  8. package/README.md +220 -0
  9. package/config/collectors.json +54 -0
  10. package/data/.gitkeep +1 -0
  11. package/docker-compose.yml +17 -0
  12. package/docs/assets/.gitkeep +1 -0
  13. package/docs/assets/token-studio-v44-dashboard.png +0 -0
  14. package/docs/assets/token-studio-v44-live.png +0 -0
  15. package/docs/assets/token-studio-v44-review-mobile.png +0 -0
  16. package/docs/assets/token-studio-v44-review.png +0 -0
  17. package/docs/assets/token-studio-v45-dashboard.png +0 -0
  18. package/docs/assets/token-studio-v45-live.png +0 -0
  19. package/docs/assets/token-studio-v45-review-mobile.png +0 -0
  20. package/docs/assets/token-studio-v45-review.png +0 -0
  21. package/docs/blog-case-study.md +34 -0
  22. package/docs/collector-support-matrix.md +65 -0
  23. package/docs/competitive-notes.md +87 -0
  24. package/docs/demo-data/README.md +12 -0
  25. package/docs/demo-data/token-studio-v2-demo.json +146 -0
  26. package/docs/demo-flow.md +39 -0
  27. package/docs/first-run.md +95 -0
  28. package/docs/local-collectors.md +49 -0
  29. package/docs/public-launch-checklist.md +45 -0
  30. package/docs/resume-bullets.md +7 -0
  31. package/docs/statusline.md +52 -0
  32. package/index.html +16 -0
  33. package/package.json +36 -0
  34. package/render.yaml +17 -0
  35. package/src/auto-attribution.mjs +396 -0
  36. package/src/ccusage-bridge.mjs +74 -0
  37. package/src/ccusage-import.mjs +415 -0
  38. package/src/cli.mjs +643 -0
  39. package/src/client/dashboard/App.jsx +1734 -0
  40. package/src/client/dashboard/annotation-presets.js +138 -0
  41. package/src/client/dashboard/attribution.js +328 -0
  42. package/src/client/dashboard/components-charts.jsx +622 -0
  43. package/src/client/dashboard/components-tables.jsx +1531 -0
  44. package/src/client/dashboard/components-top.jsx +307 -0
  45. package/src/client/dashboard/import-budget.js +41 -0
  46. package/src/client/dashboard/model-usage.js +108 -0
  47. package/src/client/dashboard/onboarding.js +80 -0
  48. package/src/client/dashboard/styles.css +2606 -0
  49. package/src/client/live/LiveApp.jsx +226 -0
  50. package/src/client/live/styles.css +446 -0
  51. package/src/client/main.jsx +20 -0
  52. package/src/client/review/ReviewApp.jsx +507 -0
  53. package/src/client/review/closure-progress.js +165 -0
  54. package/src/client/review/markdown-report.js +401 -0
  55. package/src/client/review/model-strategy.js +273 -0
  56. package/src/client/review/roi-advisor.js +255 -0
  57. package/src/client/review/roi-evidence.js +78 -0
  58. package/src/client/review/savings-simulator.js +252 -0
  59. package/src/client/review/sections-1.jsx +277 -0
  60. package/src/client/review/sections-2.jsx +927 -0
  61. package/src/client/review/styles.css +2321 -0
  62. package/src/client/review/utils.js +345 -0
  63. package/src/client/shared/utils.js +236 -0
  64. package/src/closure-check.mjs +537 -0
  65. package/src/closure-import.mjs +646 -0
  66. package/src/collect.mjs +247 -0
  67. package/src/collector-config.mjs +82 -0
  68. package/src/collector-registry.mjs +333 -0
  69. package/src/collectors/claude-code.mjs +355 -0
  70. package/src/collectors/codex.mjs +418 -0
  71. package/src/collectors/copilot.mjs +19 -0
  72. package/src/collectors/cursor.mjs +23 -0
  73. package/src/collectors/gemini.mjs +530 -0
  74. package/src/collectors/goose.mjs +15 -0
  75. package/src/collectors/hermes.mjs +206 -0
  76. package/src/collectors/kimi.mjs +15 -0
  77. package/src/collectors/openclaw.mjs +400 -0
  78. package/src/collectors/opencode.mjs +349 -0
  79. package/src/collectors/qwen.mjs +15 -0
  80. package/src/collectors/structured-usage.mjs +437 -0
  81. package/src/collectors/utils.mjs +93 -0
  82. package/src/db.mjs +1397 -0
  83. package/src/demo-seed.mjs +39 -0
  84. package/src/dev.mjs +43 -0
  85. package/src/live.mjs +428 -0
  86. package/src/model-policy.mjs +147 -0
  87. package/src/pricing.mjs +434 -0
  88. package/src/privacy-check.mjs +126 -0
  89. package/src/server.mjs +1240 -0
  90. package/src/source-health.mjs +195 -0
  91. package/src/statusline.mjs +156 -0
  92. package/src/terminal-report.mjs +245 -0
  93. package/src/update-pricing.mjs +8 -0
  94. package/test/annotation-presets.test.mjs +137 -0
  95. package/test/api-annotations.test.mjs +202 -0
  96. package/test/api-auto-attribution.test.mjs +169 -0
  97. package/test/api-source-health.test.mjs +109 -0
  98. package/test/api-v2.test.mjs +278 -0
  99. package/test/api-v43.test.mjs +151 -0
  100. package/test/api-v44.test.mjs +128 -0
  101. package/test/attribution-summary.test.mjs +164 -0
  102. package/test/auto-attribution.test.mjs +116 -0
  103. package/test/ccusage-bridge.test.mjs +36 -0
  104. package/test/ccusage-import.test.mjs +93 -0
  105. package/test/cli-v43.test.mjs +64 -0
  106. package/test/cli-v45.test.mjs +34 -0
  107. package/test/cli-v46.test.mjs +129 -0
  108. package/test/cli-v47.test.mjs +98 -0
  109. package/test/closure-check.test.mjs +202 -0
  110. package/test/closure-import.test.mjs +263 -0
  111. package/test/collector-config.test.mjs +25 -0
  112. package/test/collector-registry.test.mjs +56 -0
  113. package/test/csv.test.mjs +19 -0
  114. package/test/db-annotations.test.mjs +186 -0
  115. package/test/db-v2.test.mjs +200 -0
  116. package/test/db-v4.test.mjs +178 -0
  117. package/test/experimental-collectors.test.mjs +103 -0
  118. package/test/fixtures/collectors/copilot/usage.jsonl +2 -0
  119. package/test/fixtures/collectors/cursor/usage.jsonl +2 -0
  120. package/test/fixtures/collectors/goose/usage.jsonl +2 -0
  121. package/test/fixtures/collectors/kimi/usage.jsonl +2 -0
  122. package/test/fixtures/collectors/qwen/usage.jsonl +2 -0
  123. package/test/import-budget.test.mjs +40 -0
  124. package/test/live.test.mjs +256 -0
  125. package/test/markdown-report.test.mjs +193 -0
  126. package/test/model-policy.test.mjs +34 -0
  127. package/test/model-strategy.test.mjs +116 -0
  128. package/test/model-usage.test.mjs +99 -0
  129. package/test/official-pricing.test.mjs +70 -0
  130. package/test/onboarding.test.mjs +55 -0
  131. package/test/privacy-check.test.mjs +33 -0
  132. package/test/review-closure-progress.test.mjs +99 -0
  133. package/test/roi-advisor.test.mjs +188 -0
  134. package/test/roi-evidence.test.mjs +48 -0
  135. package/test/roi-summary.test.mjs +101 -0
  136. package/test/savings-simulator.test.mjs +141 -0
  137. package/test/source-health.test.mjs +62 -0
  138. package/test/statusline.test.mjs +148 -0
  139. package/vite.config.js +23 -0
@@ -0,0 +1,2606 @@
1
+ /* =============================================================
2
+ Token Studio ROI — refined light theme
3
+ ============================================================= */
4
+
5
+ :root {
6
+ /* Surfaces & text — warm cream (matches review page) */
7
+ --bg: oklch(0.97 0.008 80); /* warm paper */
8
+ --surface: oklch(0.995 0.005 80); /* card face — almost white, warm tint */
9
+ --surface-2: oklch(0.95 0.008 80);
10
+ --surface-3: oklch(0.92 0.010 80);
11
+ --border: oklch(0.88 0.010 60); /* warm rule */
12
+ --border-2: oklch(0.93 0.010 60);
13
+ --line: oklch(0.93 0.008 60);
14
+
15
+ --text: oklch(0.16 0.010 60); /* warm ink */
16
+ --text-2: oklch(0.32 0.010 60);
17
+ --muted: oklch(0.50 0.008 60);
18
+ --subtle: oklch(0.68 0.006 60);
19
+
20
+ /* Accents */
21
+ --c-indigo: oklch(0.55 0.16 265);
22
+ --c-blue: oklch(0.62 0.13 240);
23
+ --c-violet: oklch(0.60 0.15 295);
24
+ --c-teal: oklch(0.65 0.11 200);
25
+ --c-amber: oklch(0.72 0.14 75);
26
+ --c-green: oklch(0.65 0.12 150);
27
+ --c-rose: oklch(0.62 0.16 20);
28
+
29
+ --good: oklch(0.55 0.14 145);
30
+ --bad: oklch(0.55 0.18 25);
31
+
32
+ --radius: 12px;
33
+ --radius-sm: 8px;
34
+ --radius-lg: 16px;
35
+
36
+ --shadow-sm: 0 1px 0 oklch(0.88 0.010 60);
37
+ --shadow: 0 1px 3px rgb(60 40 20 / 0.05), 0 0 0 1px var(--border);
38
+ --shadow-lg: 0 1px 3px rgb(60 40 20 / 0.05), 0 8px 24px -8px rgb(60 40 20 / 0.08), 0 0 0 1px var(--border);
39
+
40
+ --font: 'Inter', system-ui, -apple-system, 'Segoe UI', sans-serif;
41
+ --font-mono: 'JetBrains Mono', ui-monospace, SFMono-Regular, Menlo, monospace;
42
+ }
43
+
44
+ * { box-sizing: border-box; }
45
+
46
+ html, body {
47
+ margin: 0;
48
+ padding: 0;
49
+ background: var(--bg);
50
+ color: var(--text);
51
+ font-family: var(--font);
52
+ font-size: 14px;
53
+ line-height: 1.5;
54
+ -webkit-font-smoothing: antialiased;
55
+ -moz-osx-font-smoothing: grayscale;
56
+ font-feature-settings: 'cv11', 'ss01';
57
+ }
58
+
59
+ button, input, select { font-family: inherit; color: inherit; }
60
+
61
+ /* ─── Layout ──────────────────────────────────────────────── */
62
+
63
+ .app {
64
+ max-width: 1440px;
65
+ margin: 0 auto;
66
+ padding: 24px 32px 64px;
67
+ }
68
+
69
+ /* ─── Page nav pill (shared across both pages) ───────────── */
70
+
71
+ .page-switch {
72
+ display: inline-flex;
73
+ background: var(--surface-2);
74
+ border: 1px solid var(--border);
75
+ border-radius: 999px;
76
+ padding: 3px;
77
+ gap: 2px;
78
+ }
79
+ .page-chip {
80
+ height: 26px;
81
+ padding: 0 14px;
82
+ font-size: 12.5px;
83
+ font-weight: 500;
84
+ color: var(--muted);
85
+ background: transparent;
86
+ border: 0;
87
+ border-radius: 999px;
88
+ cursor: pointer;
89
+ text-decoration: none;
90
+ display: inline-flex;
91
+ align-items: center;
92
+ transition: color 120ms ease;
93
+ letter-spacing: 0.01em;
94
+ white-space: nowrap;
95
+ }
96
+ .page-chip:hover { color: var(--text); }
97
+ .page-chip.active {
98
+ background: var(--text);
99
+ color: var(--bg);
100
+ pointer-events: none;
101
+ }
102
+
103
+ .demo-mode-badge {
104
+ height: 26px;
105
+ display: inline-flex;
106
+ align-items: center;
107
+ padding: 0 10px;
108
+ border: 1px solid color-mix(in oklab, var(--c-amber), transparent 42%);
109
+ border-radius: 999px;
110
+ background: color-mix(in oklab, var(--c-amber), transparent 88%);
111
+ color: var(--text);
112
+ font-size: 11px;
113
+ font-weight: 700;
114
+ letter-spacing: 0.08em;
115
+ text-transform: uppercase;
116
+ white-space: nowrap;
117
+ }
118
+
119
+ /* ─── Topbar ──────────────────────────────────────────────── */
120
+
121
+ .topbar {
122
+ display: flex;
123
+ align-items: center;
124
+ justify-content: space-between;
125
+ gap: 16px;
126
+ padding-bottom: 16px;
127
+ border-bottom: 1px solid var(--border-2);
128
+ margin-bottom: 20px;
129
+ }
130
+
131
+ .topbar-left {
132
+ display: flex;
133
+ align-items: center;
134
+ gap: 20px;
135
+ }
136
+
137
+ .brand {
138
+ display: flex;
139
+ align-items: center;
140
+ gap: 12px;
141
+ }
142
+
143
+ .brand-mark {
144
+ width: 36px;
145
+ height: 36px;
146
+ border-radius: 10px;
147
+ background: linear-gradient(135deg, oklch(0.30 0.04 265), oklch(0.20 0.02 265));
148
+ color: white;
149
+ display: grid;
150
+ place-items: center;
151
+ font-weight: 700;
152
+ font-size: 14px;
153
+ letter-spacing: -0.02em;
154
+ }
155
+
156
+ .brand h1 {
157
+ font-size: 16px;
158
+ font-weight: 600;
159
+ margin: 0;
160
+ letter-spacing: -0.01em;
161
+ }
162
+
163
+ .brand-sub {
164
+ font-size: 12px;
165
+ color: var(--muted);
166
+ margin: 1px 0 0;
167
+ letter-spacing: 0.01em;
168
+ }
169
+
170
+ .topbar-right {
171
+ display: flex;
172
+ align-items: center;
173
+ gap: 8px;
174
+ }
175
+
176
+ .sync-pill,
177
+ .collect-pill {
178
+ display: inline-flex;
179
+ align-items: center;
180
+ gap: 6px;
181
+ height: 30px;
182
+ padding: 0 10px;
183
+ font-size: 12px;
184
+ color: var(--muted);
185
+ background: var(--surface);
186
+ border: 1px solid var(--border);
187
+ border-radius: 8px;
188
+ white-space: nowrap;
189
+ }
190
+
191
+ .sync-dot,
192
+ .collect-dot {
193
+ width: 6px; height: 6px;
194
+ border-radius: 50%;
195
+ background: var(--good);
196
+ box-shadow: 0 0 0 3px color-mix(in oklab, var(--good), transparent 80%);
197
+ animation: pulse 2.4s infinite;
198
+ }
199
+
200
+ .collect-pill {
201
+ max-width: 160px;
202
+ }
203
+
204
+ .collect-pill span:last-child {
205
+ overflow: hidden;
206
+ text-overflow: ellipsis;
207
+ }
208
+
209
+ .collect-running .collect-dot {
210
+ background: var(--c-indigo);
211
+ box-shadow: 0 0 0 3px color-mix(in oklab, var(--c-indigo), transparent 80%);
212
+ }
213
+
214
+ .collect-ok .collect-dot {
215
+ background: var(--good);
216
+ box-shadow: 0 0 0 3px color-mix(in oklab, var(--good), transparent 80%);
217
+ }
218
+
219
+ .collect-error .collect-dot {
220
+ background: var(--bad);
221
+ box-shadow: 0 0 0 3px color-mix(in oklab, var(--bad), transparent 80%);
222
+ }
223
+
224
+ @keyframes pulse {
225
+ 0%, 100% { opacity: 1; }
226
+ 50% { opacity: 0.55; }
227
+ }
228
+
229
+ .btn {
230
+ height: 30px;
231
+ padding: 0 12px;
232
+ font-size: 12.5px;
233
+ font-weight: 500;
234
+ background: var(--surface);
235
+ border: 1px solid var(--border);
236
+ border-radius: 8px;
237
+ color: var(--text);
238
+ cursor: pointer;
239
+ display: inline-flex;
240
+ align-items: center;
241
+ gap: 6px;
242
+ transition: background 120ms ease, border-color 120ms ease;
243
+ }
244
+
245
+ .btn:hover { background: var(--surface-2); border-color: var(--border); }
246
+ .btn:active { background: var(--surface-3); }
247
+ .btn:disabled {
248
+ opacity: 0.55;
249
+ cursor: not-allowed;
250
+ }
251
+
252
+ .btn-primary {
253
+ background: var(--text);
254
+ color: var(--bg);
255
+ border-color: var(--text);
256
+ }
257
+ .btn-primary:hover { background: oklch(0.25 0.005 80); border-color: oklch(0.25 0.005 80); }
258
+
259
+ .btn-icon {
260
+ width: 30px; padding: 0;
261
+ display: grid; place-items: center;
262
+ }
263
+
264
+ .btn .icon { width: 14px; height: 14px; opacity: 0.7; }
265
+
266
+ /* ─── Filter bar ──────────────────────────────────────────── */
267
+
268
+ .filterbar {
269
+ margin: 16px 0 20px;
270
+ padding: 12px 14px;
271
+ background: var(--surface);
272
+ border: 1px solid var(--border);
273
+ border-radius: var(--radius);
274
+ display: flex;
275
+ flex-direction: column;
276
+ gap: 10px;
277
+ }
278
+
279
+ .filter-row {
280
+ width: 100%;
281
+ display: flex;
282
+ align-items: center;
283
+ flex-wrap: wrap;
284
+ gap: 12px;
285
+ }
286
+
287
+ .filter-row-secondary {
288
+ padding-top: 10px;
289
+ border-top: 1px solid var(--border-2);
290
+ }
291
+
292
+ .filter-group {
293
+ display: flex;
294
+ align-items: center;
295
+ gap: 6px;
296
+ flex-wrap: wrap;
297
+ }
298
+
299
+ .filter-group-sources {
300
+ min-width: 0;
301
+ }
302
+
303
+ .filter-label {
304
+ font-size: 11.5px;
305
+ text-transform: uppercase;
306
+ letter-spacing: 0.08em;
307
+ color: var(--subtle);
308
+ margin-right: 6px;
309
+ font-weight: 500;
310
+ }
311
+
312
+ .chip-row {
313
+ display: inline-flex;
314
+ background: var(--surface-2);
315
+ border: 1px solid var(--border);
316
+ border-radius: 8px;
317
+ padding: 2px;
318
+ gap: 2px;
319
+ }
320
+
321
+ .chip {
322
+ height: 26px;
323
+ padding: 0 10px;
324
+ font-size: 12px;
325
+ font-weight: 500;
326
+ color: var(--text-2);
327
+ background: transparent;
328
+ border: 0;
329
+ border-radius: 6px;
330
+ cursor: pointer;
331
+ display: inline-flex;
332
+ align-items: center;
333
+ gap: 5px;
334
+ transition: background 120ms ease, color 120ms ease;
335
+ }
336
+
337
+ .chip:hover { color: var(--text); }
338
+ .chip.active {
339
+ background: var(--surface);
340
+ color: var(--text);
341
+ box-shadow: 0 1px 2px rgb(0 0 0 / 0.06);
342
+ }
343
+
344
+ .pill {
345
+ height: 28px;
346
+ padding: 0 10px 0 8px;
347
+ font-size: 12px;
348
+ background: var(--surface);
349
+ border: 1px solid var(--border);
350
+ border-radius: 999px;
351
+ display: inline-flex;
352
+ align-items: center;
353
+ gap: 6px;
354
+ cursor: pointer;
355
+ color: var(--text-2);
356
+ transition: background 120ms ease, border-color 120ms ease;
357
+ }
358
+ .pill:hover { background: var(--surface-2); }
359
+ .pill.active {
360
+ background: oklch(0.97 0.02 265);
361
+ border-color: oklch(0.85 0.04 265);
362
+ color: var(--c-indigo);
363
+ }
364
+ .pill-dot {
365
+ width: 7px; height: 7px; border-radius: 50%; background: currentColor; opacity: 0.7;
366
+ }
367
+
368
+ .select-wrap {
369
+ position: relative;
370
+ display: inline-flex;
371
+ }
372
+ .select-wrap select {
373
+ appearance: none;
374
+ height: 28px;
375
+ padding: 0 28px 0 10px;
376
+ font-size: 12px;
377
+ background: var(--surface);
378
+ border: 1px solid var(--border);
379
+ border-radius: 8px;
380
+ cursor: pointer;
381
+ color: var(--text);
382
+ }
383
+ .select-wrap::after {
384
+ content: '';
385
+ position: absolute;
386
+ right: 10px;
387
+ top: 50%;
388
+ width: 6px; height: 6px;
389
+ border-right: 1.5px solid var(--muted);
390
+ border-bottom: 1.5px solid var(--muted);
391
+ transform: translateY(-70%) rotate(45deg);
392
+ pointer-events: none;
393
+ }
394
+
395
+ .toggle {
396
+ display: inline-flex;
397
+ align-items: center;
398
+ gap: 8px;
399
+ height: 28px;
400
+ padding: 0 10px;
401
+ background: var(--surface);
402
+ border: 1px solid var(--border);
403
+ border-radius: 8px;
404
+ font-size: 12px;
405
+ color: var(--text-2);
406
+ cursor: pointer;
407
+ }
408
+ .toggle.on { background: oklch(0.97 0.02 265); border-color: oklch(0.85 0.04 265); color: var(--c-indigo); }
409
+ .toggle-slot {
410
+ width: 22px; height: 12px; background: oklch(0.88 0.004 80);
411
+ border-radius: 999px; position: relative;
412
+ transition: background 160ms ease;
413
+ }
414
+ .toggle-slot::after {
415
+ content: '';
416
+ position: absolute;
417
+ width: 10px; height: 10px;
418
+ background: white;
419
+ border-radius: 50%;
420
+ top: 1px; left: 1px;
421
+ box-shadow: 0 1px 2px rgb(0 0 0 / 0.15);
422
+ transition: transform 160ms ease;
423
+ }
424
+ .toggle.on .toggle-slot { background: var(--c-indigo); }
425
+ .toggle.on .toggle-slot::after { transform: translateX(10px); }
426
+
427
+ .filter-spacer { flex: 1; }
428
+
429
+ /* ─── KPI grid ────────────────────────────────────────────── */
430
+
431
+ .kpi-row {
432
+ display: grid;
433
+ grid-template-columns: repeat(6, 1fr);
434
+ gap: 10px;
435
+ margin-bottom: 16px;
436
+ }
437
+
438
+ .kpi {
439
+ background: var(--surface);
440
+ border: 1px solid var(--border);
441
+ border-radius: var(--radius);
442
+ padding: 14px 16px 16px;
443
+ position: relative;
444
+ overflow: hidden;
445
+ }
446
+
447
+ .kpi-label {
448
+ display: flex;
449
+ align-items: center;
450
+ justify-content: space-between;
451
+ font-size: 11.5px;
452
+ color: var(--muted);
453
+ letter-spacing: 0.04em;
454
+ font-weight: 500;
455
+ margin-bottom: 8px;
456
+ }
457
+
458
+ .kpi-label .dot {
459
+ width: 6px; height: 6px;
460
+ border-radius: 50%;
461
+ background: currentColor;
462
+ }
463
+
464
+ .kpi-value {
465
+ font-size: 24px;
466
+ font-weight: 600;
467
+ letter-spacing: -0.02em;
468
+ font-variant-numeric: tabular-nums;
469
+ color: var(--text);
470
+ line-height: 1.1;
471
+ }
472
+
473
+ .kpi-sub {
474
+ display: flex;
475
+ align-items: center;
476
+ gap: 6px;
477
+ margin-top: 8px;
478
+ font-size: 11.5px;
479
+ color: var(--muted);
480
+ white-space: nowrap;
481
+ overflow: hidden;
482
+ text-overflow: ellipsis;
483
+ position: relative;
484
+ z-index: 1;
485
+ }
486
+
487
+ .kpi-sub > span:last-child {
488
+ overflow: hidden;
489
+ text-overflow: ellipsis;
490
+ min-width: 0;
491
+ }
492
+
493
+ .delta {
494
+ display: inline-flex;
495
+ align-items: center;
496
+ gap: 2px;
497
+ font-variant-numeric: tabular-nums;
498
+ font-weight: 600;
499
+ font-size: 11.5px;
500
+ padding: 1px 5px;
501
+ border-radius: 4px;
502
+ }
503
+ .delta.up { color: var(--good); background: color-mix(in oklab, var(--good), transparent 90%); }
504
+ .delta.down { color: var(--bad); background: color-mix(in oklab, var(--bad), transparent 90%); }
505
+ .delta.flat { color: var(--muted); background: var(--surface-2); }
506
+
507
+ .kpi-spark {
508
+ position: absolute;
509
+ right: 0; bottom: 0;
510
+ width: 100%;
511
+ height: 28px;
512
+ pointer-events: none;
513
+ opacity: 0.28;
514
+ z-index: 0;
515
+ mask-image: linear-gradient(to top, black 30%, transparent 100%);
516
+ -webkit-mask-image: linear-gradient(to top, black 30%, transparent 100%);
517
+ }
518
+
519
+ /* ─── Attribution overview ─────────────────────────────────── */
520
+
521
+ .attribution-overview {
522
+ margin: 12px 0 18px;
523
+ padding: 16px;
524
+ background: var(--surface);
525
+ border: 1px solid var(--border);
526
+ border-radius: var(--radius);
527
+ box-shadow: var(--shadow-sm);
528
+ }
529
+
530
+ .attribution-overview-head {
531
+ display: flex;
532
+ align-items: flex-start;
533
+ justify-content: space-between;
534
+ gap: 16px;
535
+ margin-bottom: 14px;
536
+ }
537
+
538
+ .attribution-overview-head h2 {
539
+ margin: 4px 0 0;
540
+ font-size: 16px;
541
+ line-height: 1.25;
542
+ letter-spacing: 0;
543
+ }
544
+
545
+ .attribution-overview-total {
546
+ display: grid;
547
+ justify-items: end;
548
+ gap: 3px;
549
+ color: var(--muted);
550
+ font-size: 11px;
551
+ white-space: nowrap;
552
+ }
553
+
554
+ .attribution-overview-total strong {
555
+ color: var(--text);
556
+ font-size: 18px;
557
+ font-variant-numeric: tabular-nums;
558
+ }
559
+
560
+ .attribution-overview-side {
561
+ display: flex;
562
+ align-items: center;
563
+ justify-content: flex-end;
564
+ gap: 12px;
565
+ }
566
+
567
+ .attribution-callout {
568
+ margin: -2px 0 12px;
569
+ padding: 9px 11px;
570
+ color: var(--text-2);
571
+ background: color-mix(in oklab, var(--c-amber), transparent 90%);
572
+ border: 1px solid color-mix(in oklab, var(--c-amber), transparent 72%);
573
+ border-radius: 8px;
574
+ font-size: 12px;
575
+ line-height: 1.55;
576
+ }
577
+
578
+ .attribution-card-grid {
579
+ display: grid;
580
+ grid-template-columns: repeat(5, minmax(0, 1fr));
581
+ gap: 10px;
582
+ }
583
+
584
+ .attribution-card {
585
+ min-width: 0;
586
+ padding: 12px;
587
+ border: 1px solid var(--border);
588
+ border-radius: 8px;
589
+ background: color-mix(in oklab, var(--surface), var(--bg) 35%);
590
+ }
591
+
592
+ .attribution-card-top,
593
+ .attribution-card-meta {
594
+ display: flex;
595
+ align-items: center;
596
+ justify-content: space-between;
597
+ gap: 8px;
598
+ }
599
+
600
+ .attribution-card-top span {
601
+ color: var(--text-2);
602
+ font-size: 12px;
603
+ font-weight: 600;
604
+ }
605
+
606
+ .attribution-card-top strong {
607
+ font-size: 12px;
608
+ font-variant-numeric: tabular-nums;
609
+ }
610
+
611
+ .attribution-card-value {
612
+ margin-top: 8px;
613
+ font-size: 22px;
614
+ line-height: 1.1;
615
+ font-weight: 650;
616
+ font-variant-numeric: tabular-nums;
617
+ color: var(--text);
618
+ }
619
+
620
+ .attribution-card-meta {
621
+ margin-top: 9px;
622
+ color: var(--muted);
623
+ font-size: 11px;
624
+ white-space: nowrap;
625
+ }
626
+
627
+ .attribution-meter {
628
+ height: 4px;
629
+ margin-top: 10px;
630
+ border-radius: 999px;
631
+ overflow: hidden;
632
+ background: color-mix(in oklab, var(--border), transparent 30%);
633
+ }
634
+
635
+ .attribution-meter span {
636
+ display: block;
637
+ height: 100%;
638
+ border-radius: inherit;
639
+ background: var(--accent);
640
+ }
641
+
642
+ .attribution-card-published { --accent: var(--c-violet); }
643
+ .attribution-card-completed { --accent: var(--good); }
644
+ .attribution-card-progress { --accent: var(--c-blue); }
645
+ .attribution-card-discarded { --accent: var(--bad); }
646
+ .attribution-card-unattributed { --accent: var(--c-amber); }
647
+
648
+ .attribution-card-published .attribution-card-top strong { color: var(--c-violet); }
649
+ .attribution-card-completed .attribution-card-top strong { color: var(--good); }
650
+ .attribution-card-progress .attribution-card-top strong { color: var(--c-blue); }
651
+ .attribution-card-discarded .attribution-card-top strong { color: var(--bad); }
652
+ .attribution-card-unattributed .attribution-card-top strong { color: var(--c-amber); }
653
+
654
+ /* ─── Official pricing notice ─────────────────────────────── */
655
+
656
+ .pricing-notice {
657
+ display: flex;
658
+ justify-content: space-between;
659
+ gap: 16px;
660
+ margin: 0 0 14px;
661
+ padding: 14px 16px;
662
+ background: var(--surface);
663
+ border: 1px solid var(--border);
664
+ border-radius: var(--radius);
665
+ }
666
+
667
+ .pricing-notice > div:first-child {
668
+ display: grid;
669
+ gap: 4px;
670
+ min-width: 0;
671
+ }
672
+
673
+ .pricing-notice strong {
674
+ font-size: 13px;
675
+ color: var(--text);
676
+ }
677
+
678
+ .pricing-notice span {
679
+ color: var(--muted);
680
+ font-size: 12px;
681
+ line-height: 1.55;
682
+ overflow-wrap: anywhere;
683
+ }
684
+
685
+ .pricing-unpriced {
686
+ display: flex;
687
+ align-items: center;
688
+ justify-content: flex-end;
689
+ gap: 6px;
690
+ flex-wrap: wrap;
691
+ max-width: 46%;
692
+ }
693
+
694
+ .pricing-unpriced b {
695
+ padding: 4px 7px;
696
+ border-radius: 999px;
697
+ background: color-mix(in oklab, var(--c-amber), transparent 86%);
698
+ border: 1px solid color-mix(in oklab, var(--c-amber), transparent 70%);
699
+ color: var(--text);
700
+ font-size: 11px;
701
+ font-weight: 600;
702
+ white-space: nowrap;
703
+ }
704
+
705
+ /* ─── Model overview ─────────────────────────────────────── */
706
+
707
+ .model-overview {
708
+ margin: 0 0 14px;
709
+ padding: 16px;
710
+ background: var(--surface);
711
+ border: 1px solid var(--border);
712
+ border-radius: var(--radius);
713
+ box-shadow: var(--shadow-sm);
714
+ }
715
+
716
+ .model-overview-head {
717
+ display: flex;
718
+ align-items: flex-start;
719
+ justify-content: space-between;
720
+ gap: 16px;
721
+ margin-bottom: 12px;
722
+ }
723
+
724
+ .model-overview-head h2 {
725
+ margin: 4px 0 0;
726
+ font-size: 16px;
727
+ line-height: 1.25;
728
+ letter-spacing: 0;
729
+ }
730
+
731
+ .model-overview-actions {
732
+ display: flex;
733
+ align-items: center;
734
+ justify-content: flex-end;
735
+ gap: 8px;
736
+ color: var(--muted);
737
+ font-size: 11.5px;
738
+ white-space: nowrap;
739
+ }
740
+
741
+ .model-card-grid {
742
+ display: grid;
743
+ grid-template-columns: repeat(4, minmax(0, 1fr));
744
+ gap: 10px;
745
+ }
746
+
747
+ .model-card {
748
+ min-width: 0;
749
+ text-align: left;
750
+ padding: 12px;
751
+ background: color-mix(in oklab, var(--surface), var(--bg) 35%);
752
+ border: 1px solid var(--border);
753
+ border-radius: 8px;
754
+ cursor: pointer;
755
+ display: grid;
756
+ gap: 8px;
757
+ transition: background 120ms ease, border-color 120ms ease, box-shadow 120ms ease;
758
+ }
759
+
760
+ .model-card:hover {
761
+ background: var(--surface);
762
+ border-color: color-mix(in oklab, var(--c-indigo), var(--border) 70%);
763
+ }
764
+
765
+ .model-card.active {
766
+ border-color: var(--c-indigo);
767
+ box-shadow: 0 0 0 2px color-mix(in oklab, var(--c-indigo), transparent 86%);
768
+ }
769
+
770
+ .model-card-top,
771
+ .model-card-meta,
772
+ .model-card-sub {
773
+ display: flex;
774
+ align-items: center;
775
+ justify-content: space-between;
776
+ gap: 8px;
777
+ min-width: 0;
778
+ }
779
+
780
+ .model-card-top strong {
781
+ overflow: hidden;
782
+ text-overflow: ellipsis;
783
+ white-space: nowrap;
784
+ font-size: 12.5px;
785
+ }
786
+
787
+ .model-card-top span {
788
+ flex-shrink: 0;
789
+ color: var(--muted);
790
+ font-size: 11px;
791
+ }
792
+
793
+ .model-card-value {
794
+ font-size: 21px;
795
+ line-height: 1.1;
796
+ font-weight: 650;
797
+ font-variant-numeric: tabular-nums;
798
+ }
799
+
800
+ .model-card-meta,
801
+ .model-card-sub {
802
+ color: var(--muted);
803
+ font-size: 11px;
804
+ }
805
+
806
+ .model-card-sub span:last-child {
807
+ min-width: 0;
808
+ overflow: hidden;
809
+ text-overflow: ellipsis;
810
+ white-space: nowrap;
811
+ text-align: right;
812
+ }
813
+
814
+ /* ─── ROI review ──────────────────────────────────────────── */
815
+
816
+ .roi-review {
817
+ display: grid;
818
+ grid-template-columns: minmax(0, 0.95fr) minmax(0, 1.25fr) minmax(0, 1fr);
819
+ gap: 12px;
820
+ margin: 0 0 18px;
821
+ }
822
+
823
+ .roi-panel {
824
+ min-width: 0;
825
+ background: var(--surface);
826
+ border: 1px solid var(--border);
827
+ border-radius: var(--radius);
828
+ padding: 16px;
829
+ }
830
+
831
+ .panel-header.compact {
832
+ margin-bottom: 12px;
833
+ }
834
+
835
+ .risk-list,
836
+ .project-roi-list,
837
+ .weekly-output-list {
838
+ display: grid;
839
+ gap: 10px;
840
+ }
841
+
842
+ .risk-row {
843
+ display: grid;
844
+ grid-template-columns: minmax(0, 1fr) 96px 48px;
845
+ align-items: center;
846
+ gap: 10px;
847
+ }
848
+
849
+ .risk-row strong,
850
+ .project-roi-main strong {
851
+ display: block;
852
+ overflow: hidden;
853
+ text-overflow: ellipsis;
854
+ white-space: nowrap;
855
+ }
856
+
857
+ .risk-row span,
858
+ .project-roi-main span,
859
+ .project-roi-meta,
860
+ .weekly-label {
861
+ color: var(--muted);
862
+ font-size: 11.5px;
863
+ }
864
+
865
+ .risk-row b {
866
+ font-size: 12px;
867
+ text-align: right;
868
+ font-variant-numeric: tabular-nums;
869
+ }
870
+
871
+ .risk-meter,
872
+ .project-roi-bars {
873
+ height: 7px;
874
+ overflow: hidden;
875
+ border-radius: 999px;
876
+ background: var(--surface-2);
877
+ }
878
+
879
+ .risk-meter span,
880
+ .project-roi-bars span {
881
+ display: block;
882
+ height: 100%;
883
+ }
884
+
885
+ .risk-unattributed .risk-meter span { background: var(--c-amber); }
886
+ .risk-progress .risk-meter span { background: var(--c-blue); }
887
+ .risk-discarded .risk-meter span { background: var(--bad); }
888
+
889
+ .project-roi-row {
890
+ display: grid;
891
+ gap: 7px;
892
+ padding-bottom: 10px;
893
+ border-bottom: 1px solid var(--line);
894
+ }
895
+
896
+ .project-roi-row:last-child {
897
+ padding-bottom: 0;
898
+ border-bottom: 0;
899
+ }
900
+
901
+ .project-roi-main,
902
+ .project-roi-meta,
903
+ .weekly-grid {
904
+ display: flex;
905
+ align-items: center;
906
+ justify-content: space-between;
907
+ gap: 10px;
908
+ }
909
+
910
+ .project-roi-bars {
911
+ display: flex;
912
+ }
913
+
914
+ .roi-published { background: var(--c-violet); }
915
+ .roi-completed { background: var(--good); }
916
+ .roi-discarded { background: var(--bad); }
917
+ .roi-unattributed { background: var(--c-amber); }
918
+
919
+ .weekly-grid {
920
+ display: grid;
921
+ grid-template-columns: repeat(2, minmax(0, 1fr));
922
+ gap: 8px;
923
+ }
924
+
925
+ .weekly-grid > div {
926
+ padding: 10px;
927
+ background: var(--surface-2);
928
+ border: 1px solid var(--border-2);
929
+ border-radius: 8px;
930
+ min-width: 0;
931
+ }
932
+
933
+ .weekly-grid strong {
934
+ display: block;
935
+ margin-top: 3px;
936
+ overflow: hidden;
937
+ text-overflow: ellipsis;
938
+ white-space: nowrap;
939
+ }
940
+
941
+ .weekly-output-list {
942
+ margin-top: 12px;
943
+ }
944
+
945
+ .weekly-output-list a,
946
+ .output-link {
947
+ color: var(--c-indigo);
948
+ text-decoration: none;
949
+ font-weight: 600;
950
+ overflow-wrap: anywhere;
951
+ }
952
+
953
+ .weekly-output-list a:hover,
954
+ .output-link:hover {
955
+ text-decoration: underline;
956
+ }
957
+
958
+ .compact-empty {
959
+ padding: 12px;
960
+ }
961
+
962
+ .first-run-panel {
963
+ margin: 0 0 16px;
964
+ padding: 16px;
965
+ border: 1px solid var(--border);
966
+ border-radius: 10px;
967
+ background: linear-gradient(135deg, color-mix(in oklab, var(--surface), var(--c-indigo) 6%), var(--surface));
968
+ box-shadow: 0 14px 34px -30px rgb(0 0 0 / 0.28);
969
+ }
970
+
971
+ .first-run-main {
972
+ display: grid;
973
+ grid-template-columns: minmax(0, 1fr) auto;
974
+ gap: 16px;
975
+ align-items: start;
976
+ }
977
+
978
+ .first-run-main h2 {
979
+ margin: 3px 0 6px;
980
+ font-size: 16px;
981
+ }
982
+
983
+ .first-run-main p {
984
+ margin: 0;
985
+ color: var(--muted);
986
+ font-size: 12.5px;
987
+ line-height: 1.55;
988
+ }
989
+
990
+ .first-run-actions {
991
+ display: flex;
992
+ flex-wrap: wrap;
993
+ gap: 8px;
994
+ justify-content: flex-end;
995
+ }
996
+
997
+ .first-run-actions a {
998
+ text-decoration: none;
999
+ }
1000
+
1001
+ .first-run-steps {
1002
+ display: grid;
1003
+ grid-template-columns: repeat(3, minmax(0, 1fr));
1004
+ gap: 10px;
1005
+ margin-top: 14px;
1006
+ }
1007
+
1008
+ .first-run-step {
1009
+ min-width: 0;
1010
+ padding: 12px;
1011
+ border: 1px solid var(--border-2);
1012
+ border-radius: 8px;
1013
+ background: color-mix(in oklab, var(--surface), var(--bg) 28%);
1014
+ }
1015
+
1016
+ .first-run-step.done {
1017
+ border-color: color-mix(in oklab, var(--good), transparent 72%);
1018
+ background: color-mix(in oklab, var(--good), transparent 94%);
1019
+ }
1020
+
1021
+ .first-run-step span,
1022
+ .first-run-notices span {
1023
+ color: var(--muted);
1024
+ font-size: 10.5px;
1025
+ text-transform: uppercase;
1026
+ letter-spacing: 0.08em;
1027
+ }
1028
+
1029
+ .first-run-step strong {
1030
+ display: block;
1031
+ margin-top: 4px;
1032
+ color: var(--text);
1033
+ font-size: 13px;
1034
+ }
1035
+
1036
+ .first-run-step p {
1037
+ margin: 5px 0 0;
1038
+ color: var(--muted);
1039
+ font-size: 11.5px;
1040
+ line-height: 1.45;
1041
+ }
1042
+
1043
+ .first-run-notices {
1044
+ display: grid;
1045
+ gap: 8px;
1046
+ margin-top: 10px;
1047
+ }
1048
+
1049
+ .first-run-notices button {
1050
+ display: flex;
1051
+ justify-content: space-between;
1052
+ gap: 12px;
1053
+ align-items: center;
1054
+ padding: 10px 12px;
1055
+ border: 1px solid var(--border-2);
1056
+ border-radius: 8px;
1057
+ background: var(--surface);
1058
+ color: var(--text);
1059
+ text-align: left;
1060
+ cursor: pointer;
1061
+ }
1062
+
1063
+ .first-run-notices button:hover {
1064
+ border-color: var(--c-indigo);
1065
+ background: color-mix(in oklab, var(--c-indigo), transparent 94%);
1066
+ }
1067
+
1068
+ /* ─── Source health center ───────────────────────────────── */
1069
+
1070
+ .source-health-panel {
1071
+ margin: 0 0 16px;
1072
+ padding: 16px;
1073
+ border: 1px solid var(--border);
1074
+ border-radius: var(--radius);
1075
+ background: var(--surface);
1076
+ box-shadow: var(--shadow-sm);
1077
+ }
1078
+
1079
+ .source-health-head {
1080
+ display: flex;
1081
+ justify-content: space-between;
1082
+ gap: 16px;
1083
+ align-items: flex-start;
1084
+ margin-bottom: 12px;
1085
+ }
1086
+
1087
+ .source-health-head h2 {
1088
+ margin: 3px 0 5px;
1089
+ font-size: 16px;
1090
+ line-height: 1.25;
1091
+ }
1092
+
1093
+ .source-health-head p {
1094
+ max-width: 820px;
1095
+ margin: 0;
1096
+ color: var(--muted);
1097
+ font-size: 12.5px;
1098
+ line-height: 1.5;
1099
+ }
1100
+
1101
+ .source-health-actions {
1102
+ display: flex;
1103
+ flex-wrap: wrap;
1104
+ justify-content: flex-end;
1105
+ gap: 8px;
1106
+ }
1107
+
1108
+ .source-health-stats {
1109
+ display: grid;
1110
+ grid-template-columns: repeat(4, minmax(0, 1fr));
1111
+ gap: 8px;
1112
+ margin-bottom: 10px;
1113
+ }
1114
+
1115
+ .source-health-stats div,
1116
+ .source-health-card {
1117
+ border: 1px solid var(--border-2);
1118
+ border-radius: 8px;
1119
+ background: color-mix(in oklab, var(--surface), var(--bg) 26%);
1120
+ }
1121
+
1122
+ .source-health-stats div {
1123
+ padding: 9px 10px;
1124
+ }
1125
+
1126
+ .source-health-stats span {
1127
+ display: block;
1128
+ color: var(--muted);
1129
+ font-size: 11px;
1130
+ }
1131
+
1132
+ .source-health-stats strong {
1133
+ display: block;
1134
+ margin-top: 2px;
1135
+ font-size: 18px;
1136
+ }
1137
+
1138
+ .source-health-grid {
1139
+ display: grid;
1140
+ grid-template-columns: repeat(5, minmax(0, 1fr));
1141
+ gap: 8px;
1142
+ }
1143
+
1144
+ .source-health-card {
1145
+ min-width: 0;
1146
+ padding: 10px;
1147
+ display: grid;
1148
+ gap: 8px;
1149
+ }
1150
+
1151
+ .source-health-card-top {
1152
+ display: flex;
1153
+ justify-content: space-between;
1154
+ gap: 8px;
1155
+ min-width: 0;
1156
+ }
1157
+
1158
+ .source-health-card-top strong {
1159
+ min-width: 0;
1160
+ overflow: hidden;
1161
+ text-overflow: ellipsis;
1162
+ white-space: nowrap;
1163
+ font-size: 12.5px;
1164
+ }
1165
+
1166
+ .source-health-card-top span {
1167
+ flex: 0 0 auto;
1168
+ color: var(--muted);
1169
+ font-size: 10.5px;
1170
+ text-transform: uppercase;
1171
+ letter-spacing: 0.04em;
1172
+ }
1173
+
1174
+ .source-health-card-meta {
1175
+ display: flex;
1176
+ flex-wrap: wrap;
1177
+ gap: 4px;
1178
+ }
1179
+
1180
+ .source-health-card-meta span {
1181
+ padding: 2px 6px;
1182
+ border: 1px solid var(--border-2);
1183
+ border-radius: 999px;
1184
+ color: var(--muted);
1185
+ background: var(--surface);
1186
+ font-size: 10.5px;
1187
+ }
1188
+
1189
+ .source-health-card-counts {
1190
+ display: grid;
1191
+ grid-template-columns: repeat(3, minmax(0, 1fr));
1192
+ gap: 3px 6px;
1193
+ align-items: end;
1194
+ }
1195
+
1196
+ .source-health-card-counts b,
1197
+ .source-health-card-counts span {
1198
+ min-width: 0;
1199
+ overflow: hidden;
1200
+ text-overflow: ellipsis;
1201
+ white-space: nowrap;
1202
+ }
1203
+
1204
+ .source-health-card-counts b {
1205
+ font-size: 14px;
1206
+ }
1207
+
1208
+ .source-health-card-counts span {
1209
+ color: var(--muted);
1210
+ font-size: 10.5px;
1211
+ }
1212
+
1213
+ .source-health-card code {
1214
+ min-width: 0;
1215
+ padding: 7px 8px;
1216
+ border: 1px solid var(--border-2);
1217
+ border-radius: 7px;
1218
+ background: var(--surface-2);
1219
+ color: var(--text-2);
1220
+ font-family: var(--font-mono);
1221
+ font-size: 10.5px;
1222
+ line-height: 1.35;
1223
+ overflow-wrap: anywhere;
1224
+ }
1225
+
1226
+ .source-health-card.status-import-only {
1227
+ border-color: color-mix(in oklab, var(--c-indigo), transparent 70%);
1228
+ }
1229
+
1230
+ .source-health-card.health-has-data {
1231
+ background: color-mix(in oklab, var(--good), transparent 94%);
1232
+ }
1233
+
1234
+ /* ─── Grid ────────────────────────────────────────────────── */
1235
+
1236
+ .grid {
1237
+ display: grid;
1238
+ grid-template-columns: repeat(12, 1fr);
1239
+ gap: 16px;
1240
+ }
1241
+
1242
+ .col-4 { grid-column: span 4; }
1243
+ .col-5 { grid-column: span 5; }
1244
+ .col-6 { grid-column: span 6; }
1245
+ .col-7 { grid-column: span 7; }
1246
+ .col-8 { grid-column: span 8; }
1247
+ .col-12 { grid-column: span 12; }
1248
+
1249
+ /* ─── Panel ───────────────────────────────────────────────── */
1250
+
1251
+ .panel {
1252
+ background: var(--surface);
1253
+ border: 1px solid var(--border);
1254
+ border-radius: var(--radius);
1255
+ padding: 18px 20px 20px;
1256
+ }
1257
+
1258
+ .panel-header {
1259
+ display: flex;
1260
+ align-items: flex-start;
1261
+ justify-content: space-between;
1262
+ margin-bottom: 12px;
1263
+ gap: 16px;
1264
+ }
1265
+
1266
+ .panel-title {
1267
+ font-size: 13px;
1268
+ font-weight: 600;
1269
+ margin: 0;
1270
+ color: var(--text);
1271
+ letter-spacing: -0.005em;
1272
+ }
1273
+
1274
+ .panel-sub {
1275
+ font-size: 11.5px;
1276
+ color: var(--muted);
1277
+ margin: 2px 0 0;
1278
+ }
1279
+
1280
+ .panel-actions {
1281
+ display: inline-flex;
1282
+ align-items: center;
1283
+ gap: 6px;
1284
+ }
1285
+
1286
+ .panel-tabs {
1287
+ display: inline-flex;
1288
+ background: var(--surface-2);
1289
+ border-radius: 7px;
1290
+ padding: 2px;
1291
+ gap: 2px;
1292
+ border: 1px solid var(--border);
1293
+ }
1294
+
1295
+ .tab {
1296
+ height: 24px;
1297
+ padding: 0 9px;
1298
+ font-size: 11.5px;
1299
+ font-weight: 500;
1300
+ color: var(--text-2);
1301
+ background: transparent;
1302
+ border: 0;
1303
+ border-radius: 5px;
1304
+ cursor: pointer;
1305
+ }
1306
+ .tab.active {
1307
+ background: var(--surface);
1308
+ color: var(--text);
1309
+ box-shadow: 0 1px 1px rgb(0 0 0 / 0.05);
1310
+ }
1311
+
1312
+ /* ─── Chart bodies ────────────────────────────────────────── */
1313
+
1314
+ .chart {
1315
+ width: 100%;
1316
+ height: 320px;
1317
+ }
1318
+
1319
+ .chart-tall { height: 360px; }
1320
+ .chart-sm { height: 200px; }
1321
+ .chart-md { height: 240px; }
1322
+
1323
+ /* ─── Source Donut + legend ──────────────────────────────── */
1324
+
1325
+ .donut-row {
1326
+ display: flex;
1327
+ align-items: center;
1328
+ gap: 16px;
1329
+ }
1330
+ .donut-row .chart { height: 200px; width: 200px; flex-shrink: 0; }
1331
+ .legend { flex: 1; display: flex; flex-direction: column; gap: 8px; min-width: 0; }
1332
+ .legend-item {
1333
+ display: flex; align-items: center; gap: 8px;
1334
+ padding: 6px 8px;
1335
+ border-radius: 7px;
1336
+ cursor: pointer;
1337
+ font-size: 12.5px;
1338
+ transition: background 120ms ease;
1339
+ }
1340
+ .legend-item:hover { background: var(--surface-2); }
1341
+ .legend-item.dim { opacity: 0.4; }
1342
+ .legend-swatch { width: 8px; height: 8px; border-radius: 2px; flex-shrink: 0; }
1343
+ .legend-name { flex: 1; min-width: 0; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
1344
+ .legend-val { font-variant-numeric: tabular-nums; color: var(--muted); font-size: 11.5px; }
1345
+ .legend-pct { font-variant-numeric: tabular-nums; color: var(--text); font-weight: 600; min-width: 36px; text-align: right; }
1346
+
1347
+ /* ─── Top-Models bars ────────────────────────────────────── */
1348
+
1349
+ .bars {
1350
+ display: flex;
1351
+ flex-direction: column;
1352
+ gap: 8px;
1353
+ }
1354
+ .bar-row {
1355
+ display: grid;
1356
+ grid-template-columns: 1fr auto;
1357
+ align-items: center;
1358
+ gap: 12px;
1359
+ padding: 8px 4px;
1360
+ border-radius: 6px;
1361
+ cursor: pointer;
1362
+ }
1363
+ .bar-row:hover { background: var(--surface-2); }
1364
+ .bar-label {
1365
+ display: flex; flex-direction: column; gap: 4px; min-width: 0;
1366
+ }
1367
+ .bar-label .model {
1368
+ font-size: 12.5px;
1369
+ font-weight: 500;
1370
+ color: var(--text);
1371
+ overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
1372
+ }
1373
+ .bar-label .meta {
1374
+ font-size: 11px;
1375
+ color: var(--muted);
1376
+ display: flex; gap: 8px; align-items: center;
1377
+ white-space: nowrap;
1378
+ overflow: hidden;
1379
+ }
1380
+ .bar-label .meta > * { white-space: nowrap; flex-shrink: 0; }
1381
+ .bar-track {
1382
+ margin-top: 4px;
1383
+ height: 6px;
1384
+ background: var(--surface-2);
1385
+ border-radius: 3px;
1386
+ overflow: hidden;
1387
+ }
1388
+ .bar-fill {
1389
+ height: 100%;
1390
+ border-radius: 3px;
1391
+ transition: width 400ms ease;
1392
+ }
1393
+ .bar-value {
1394
+ font-variant-numeric: tabular-nums;
1395
+ font-size: 12.5px;
1396
+ font-weight: 600;
1397
+ color: var(--text);
1398
+ text-align: right;
1399
+ white-space: nowrap;
1400
+ flex-shrink: 0;
1401
+ }
1402
+ .bar-value small {
1403
+ display: block;
1404
+ font-size: 10.5px;
1405
+ font-weight: 400;
1406
+ color: var(--muted);
1407
+ margin-top: 2px;
1408
+ white-space: nowrap;
1409
+ }
1410
+
1411
+ /* ─── Cache hit-rate gauge ───────────────────────────────── */
1412
+
1413
+ .gauge {
1414
+ display: flex;
1415
+ flex-direction: column;
1416
+ align-items: center;
1417
+ justify-content: center;
1418
+ padding: 16px 8px 8px;
1419
+ }
1420
+ .gauge-wrap { position: relative; width: 180px; height: 100px; }
1421
+ .gauge-text {
1422
+ position: absolute;
1423
+ inset: 0;
1424
+ display: grid; place-items: center;
1425
+ padding-top: 14px;
1426
+ }
1427
+ .gauge-num {
1428
+ font-size: 30px;
1429
+ font-weight: 600;
1430
+ letter-spacing: -0.02em;
1431
+ font-variant-numeric: tabular-nums;
1432
+ }
1433
+ .gauge-suffix { font-size: 14px; color: var(--muted); margin-left: 1px; }
1434
+ .gauge-meta {
1435
+ margin-top: 10px;
1436
+ display: flex; gap: 16px;
1437
+ font-size: 11.5px;
1438
+ color: var(--muted);
1439
+ }
1440
+ .gauge-meta b { color: var(--text); font-weight: 600; font-variant-numeric: tabular-nums; }
1441
+
1442
+ /* ─── Heatmap ────────────────────────────────────────────── */
1443
+
1444
+ .heatmap {
1445
+ width: 100%;
1446
+ display: grid;
1447
+ gap: 2px;
1448
+ }
1449
+ .heat-cell {
1450
+ height: 22px;
1451
+ border-radius: 3px;
1452
+ cursor: pointer;
1453
+ transition: transform 120ms ease, outline 120ms ease;
1454
+ }
1455
+ .heat-cell:hover {
1456
+ outline: 1.5px solid var(--text);
1457
+ outline-offset: 1px;
1458
+ z-index: 2;
1459
+ }
1460
+ .heat-row-label,
1461
+ .heat-col-label {
1462
+ font-size: 10.5px;
1463
+ color: var(--muted);
1464
+ font-variant-numeric: tabular-nums;
1465
+ }
1466
+ .heat-col-label { text-align: center; }
1467
+ .heat-row-label { padding-right: 6px; text-align: right; }
1468
+ .heat-scale {
1469
+ display: inline-flex; align-items: center; gap: 6px;
1470
+ font-size: 11px; color: var(--muted);
1471
+ }
1472
+ .heat-scale-bar {
1473
+ display: inline-flex; height: 8px; width: 100px; border-radius: 2px; overflow: hidden;
1474
+ }
1475
+ .heat-scale-bar span { flex: 1; }
1476
+
1477
+ /* ─── Tables ─────────────────────────────────────────────── */
1478
+
1479
+ .table-wrap {
1480
+ overflow-x: auto;
1481
+ border-radius: var(--radius-sm);
1482
+ border: 1px solid var(--border);
1483
+ }
1484
+
1485
+ table.dt {
1486
+ width: 100%;
1487
+ border-collapse: collapse;
1488
+ font-size: 12.5px;
1489
+ }
1490
+
1491
+ table.dt thead th {
1492
+ text-align: left;
1493
+ font-weight: 500;
1494
+ font-size: 11px;
1495
+ text-transform: uppercase;
1496
+ letter-spacing: 0.06em;
1497
+ color: var(--muted);
1498
+ padding: 10px 12px;
1499
+ background: var(--surface-2);
1500
+ border-bottom: 1px solid var(--border);
1501
+ cursor: pointer;
1502
+ user-select: none;
1503
+ white-space: nowrap;
1504
+ }
1505
+
1506
+ table.dt thead th .sort-ind {
1507
+ display: inline-block;
1508
+ margin-left: 4px;
1509
+ opacity: 0.4;
1510
+ font-size: 9px;
1511
+ }
1512
+ table.dt thead th.sorted .sort-ind { opacity: 1; color: var(--c-indigo); }
1513
+
1514
+ table.dt tbody td {
1515
+ padding: 10px 12px;
1516
+ border-bottom: 1px solid var(--line);
1517
+ color: var(--text);
1518
+ font-variant-numeric: tabular-nums;
1519
+ }
1520
+ table.dt tbody tr:last-child td { border-bottom: 0; }
1521
+ table.dt tbody tr {
1522
+ cursor: pointer;
1523
+ transition: background 100ms ease;
1524
+ }
1525
+ table.dt tbody tr:hover { background: var(--surface-2); }
1526
+ table.dt tbody tr.selected { background: oklch(0.97 0.02 265); }
1527
+
1528
+ .num { text-align: right; font-variant-numeric: tabular-nums; }
1529
+ .num-strong { font-weight: 600; }
1530
+ .muted { color: var(--muted); }
1531
+ .tag {
1532
+ display: inline-flex; align-items: center; gap: 4px;
1533
+ height: 20px; padding: 0 6px;
1534
+ font-size: 11px;
1535
+ border-radius: 4px;
1536
+ background: var(--surface-2);
1537
+ border: 1px solid var(--border);
1538
+ color: var(--text-2);
1539
+ }
1540
+ .tag-dot { width: 6px; height: 6px; border-radius: 50%; }
1541
+ .mono { font-family: var(--font-mono); font-size: 11.5px; }
1542
+
1543
+ .share-bar {
1544
+ height: 4px;
1545
+ width: 90px;
1546
+ background: var(--surface-2);
1547
+ border-radius: 2px;
1548
+ overflow: hidden;
1549
+ display: inline-block;
1550
+ vertical-align: middle;
1551
+ }
1552
+ .share-bar span { display: block; height: 100%; background: var(--c-indigo); border-radius: 2px; }
1553
+ .share-pct {
1554
+ display: inline-block;
1555
+ margin-left: 8px;
1556
+ font-size: 11px;
1557
+ color: var(--muted);
1558
+ vertical-align: middle;
1559
+ min-width: 36px;
1560
+ }
1561
+
1562
+ .status-badge {
1563
+ display: inline-flex; align-items: center; gap: 5px;
1564
+ font-size: 11px;
1565
+ padding: 2px 7px;
1566
+ border-radius: 4px;
1567
+ font-weight: 500;
1568
+ }
1569
+ .status-badge::before {
1570
+ content: '';
1571
+ width: 5px; height: 5px;
1572
+ border-radius: 50%;
1573
+ background: currentColor;
1574
+ }
1575
+ .status-ok { background: color-mix(in oklab, var(--good), transparent 90%); color: var(--good); }
1576
+ .status-error { background: color-mix(in oklab, var(--bad), transparent 90%); color: var(--bad); }
1577
+ .status-warn { background: color-mix(in oklab, var(--c-amber), transparent 85%); color: oklch(0.50 0.15 70); }
1578
+ .status-empty { background: color-mix(in oklab, var(--muted), transparent 85%); color: var(--muted); }
1579
+ .status-skip { background: color-mix(in oklab, var(--muted), transparent 85%); color: var(--muted); }
1580
+ .annotation-status-未标注 { background: color-mix(in oklab, var(--muted), transparent 88%); color: var(--muted); }
1581
+ .annotation-status-进行中 { background: color-mix(in oklab, var(--c-blue), transparent 88%); color: var(--c-blue); }
1582
+ .annotation-status-已完成 { background: color-mix(in oklab, var(--good), transparent 88%); color: var(--good); }
1583
+ .annotation-status-已发布 { background: color-mix(in oklab, var(--c-violet), transparent 88%); color: var(--c-violet); }
1584
+ .annotation-status-已废弃 { background: color-mix(in oklab, var(--bad), transparent 88%); color: var(--bad); }
1585
+ .value-level-未评估 { background: color-mix(in oklab, var(--muted), transparent 88%); color: var(--muted); }
1586
+ .value-level-低 { background: color-mix(in oklab, var(--bad), transparent 88%); color: var(--bad); }
1587
+ .value-level-中 { background: color-mix(in oklab, var(--c-blue), transparent 88%); color: var(--c-blue); }
1588
+ .value-level-高 { background: color-mix(in oklab, var(--good), transparent 88%); color: var(--good); }
1589
+ .value-level-关键 { background: color-mix(in oklab, var(--c-violet), transparent 88%); color: var(--c-violet); }
1590
+
1591
+ .tag-soft {
1592
+ background: color-mix(in oklab, var(--c-indigo), transparent 92%);
1593
+ color: var(--c-indigo);
1594
+ border-color: color-mix(in oklab, var(--c-indigo), transparent 78%);
1595
+ }
1596
+
1597
+ .session-project {
1598
+ display: grid;
1599
+ gap: 2px;
1600
+ min-width: 220px;
1601
+ }
1602
+ .session-project-raw {
1603
+ color: var(--muted);
1604
+ font-size: 10.5px;
1605
+ overflow: hidden;
1606
+ text-overflow: ellipsis;
1607
+ white-space: nowrap;
1608
+ max-width: 360px;
1609
+ }
1610
+ .session-project-rule {
1611
+ width: fit-content;
1612
+ color: var(--c-amber);
1613
+ font-size: 10.5px;
1614
+ border: 1px solid color-mix(in oklab, var(--c-amber), transparent 70%);
1615
+ border-radius: 999px;
1616
+ padding: 0 6px;
1617
+ background: color-mix(in oklab, var(--c-amber), transparent 92%);
1618
+ }
1619
+ .annotation-note {
1620
+ display: block;
1621
+ max-width: 150px;
1622
+ overflow: hidden;
1623
+ text-overflow: ellipsis;
1624
+ white-space: nowrap;
1625
+ color: var(--text-2);
1626
+ }
1627
+ .btn-mini {
1628
+ height: 24px;
1629
+ padding: 0 8px;
1630
+ font-size: 11.5px;
1631
+ }
1632
+ .btn-annotated {
1633
+ border-color: color-mix(in oklab, var(--c-indigo), transparent 60%);
1634
+ color: var(--c-indigo);
1635
+ background: color-mix(in oklab, var(--c-indigo), transparent 94%);
1636
+ }
1637
+
1638
+ /* ─── Drawer (drill-down) ────────────────────────────────── */
1639
+
1640
+ .drawer-backdrop {
1641
+ position: fixed; inset: 0;
1642
+ background: oklch(0.18 0.005 80 / 0.30);
1643
+ backdrop-filter: blur(2px);
1644
+ z-index: 50;
1645
+ opacity: 0;
1646
+ pointer-events: none;
1647
+ transition: opacity 180ms ease;
1648
+ }
1649
+ .drawer-backdrop.open { opacity: 1; pointer-events: auto; }
1650
+
1651
+ .drawer {
1652
+ position: fixed;
1653
+ top: 0; right: 0; bottom: 0;
1654
+ width: 560px;
1655
+ max-width: 92vw;
1656
+ background: var(--surface);
1657
+ border-left: 1px solid var(--border);
1658
+ box-shadow: -20px 0 50px -20px rgb(0 0 0 / 0.12);
1659
+ z-index: 60;
1660
+ transform: translateX(100%);
1661
+ transition: transform 220ms cubic-bezier(0.22, 1, 0.36, 1);
1662
+ display: flex;
1663
+ flex-direction: column;
1664
+ }
1665
+ .drawer.open { transform: translateX(0); }
1666
+
1667
+ .drawer-header {
1668
+ padding: 18px 22px 14px;
1669
+ border-bottom: 1px solid var(--border-2);
1670
+ }
1671
+ .drawer-header h3 {
1672
+ margin: 0;
1673
+ font-size: 15px;
1674
+ font-weight: 600;
1675
+ }
1676
+ .drawer-header .sub {
1677
+ font-size: 12px;
1678
+ color: var(--muted);
1679
+ margin-top: 2px;
1680
+ }
1681
+ .drawer-close {
1682
+ position: absolute;
1683
+ top: 18px; right: 18px;
1684
+ width: 28px; height: 28px;
1685
+ display: grid; place-items: center;
1686
+ background: var(--surface-2);
1687
+ border: 1px solid var(--border);
1688
+ border-radius: 7px;
1689
+ cursor: pointer;
1690
+ color: var(--muted);
1691
+ }
1692
+ .drawer-close:hover { color: var(--text); }
1693
+
1694
+ .drawer-body { flex: 1; overflow: auto; padding: 20px 22px 24px; }
1695
+
1696
+ .drawer-kpi-row {
1697
+ display: grid;
1698
+ grid-template-columns: repeat(3, 1fr);
1699
+ gap: 8px;
1700
+ margin-bottom: 18px;
1701
+ }
1702
+ .drawer-kpi {
1703
+ padding: 10px 12px;
1704
+ background: var(--surface-2);
1705
+ border: 1px solid var(--border-2);
1706
+ border-radius: 8px;
1707
+ }
1708
+ .drawer-kpi .l { font-size: 10.5px; text-transform: uppercase; letter-spacing: 0.06em; color: var(--muted); }
1709
+ .drawer-kpi .v { font-size: 18px; font-weight: 600; font-variant-numeric: tabular-nums; margin-top: 4px; }
1710
+
1711
+ .detail-section { margin-top: 18px; }
1712
+ .detail-section h4 {
1713
+ font-size: 11.5px;
1714
+ font-weight: 600;
1715
+ text-transform: uppercase;
1716
+ letter-spacing: 0.06em;
1717
+ color: var(--muted);
1718
+ margin: 0 0 10px;
1719
+ }
1720
+ .detail-row {
1721
+ display: flex; justify-content: space-between; align-items: center;
1722
+ padding: 8px 0;
1723
+ border-bottom: 1px dashed var(--line);
1724
+ font-size: 12.5px;
1725
+ }
1726
+ .detail-row:last-child { border-bottom: 0; }
1727
+ .detail-row .k { color: var(--muted); }
1728
+ .detail-row .v { color: var(--text); font-variant-numeric: tabular-nums; }
1729
+
1730
+ /* ─── Misc ───────────────────────────────────────────────── */
1731
+
1732
+ .search-input {
1733
+ height: 28px;
1734
+ padding: 0 10px;
1735
+ font-size: 12px;
1736
+ background: var(--surface);
1737
+ border: 1px solid var(--border);
1738
+ border-radius: 8px;
1739
+ color: var(--text);
1740
+ width: 180px;
1741
+ }
1742
+ .search-input::placeholder { color: var(--subtle); }
1743
+ .search-input:focus { outline: 2px solid color-mix(in oklab, var(--c-indigo), transparent 70%); outline-offset: -1px; border-color: var(--c-indigo); }
1744
+
1745
+ /* ─── Batch and panel messages ───────────────────────────── */
1746
+
1747
+ .panel-message,
1748
+ .batch-bar {
1749
+ margin-bottom: 12px;
1750
+ border: 1px solid var(--border);
1751
+ border-radius: 8px;
1752
+ padding: 10px 12px;
1753
+ }
1754
+
1755
+ .panel-message {
1756
+ font-size: 12px;
1757
+ }
1758
+
1759
+ .panel-message-ok {
1760
+ color: var(--good);
1761
+ background: color-mix(in oklab, var(--good), transparent 92%);
1762
+ border-color: color-mix(in oklab, var(--good), transparent 70%);
1763
+ }
1764
+
1765
+ .panel-message-error {
1766
+ color: var(--bad);
1767
+ background: color-mix(in oklab, var(--bad), transparent 92%);
1768
+ border-color: color-mix(in oklab, var(--bad), transparent 70%);
1769
+ }
1770
+
1771
+ .batch-bar {
1772
+ display: flex;
1773
+ align-items: center;
1774
+ justify-content: space-between;
1775
+ gap: 12px;
1776
+ background: var(--surface-2);
1777
+ }
1778
+
1779
+ .review-progress {
1780
+ display: grid;
1781
+ grid-template-columns: minmax(0, 1fr) auto;
1782
+ gap: 14px;
1783
+ align-items: center;
1784
+ margin-bottom: 12px;
1785
+ border: 1px solid var(--border);
1786
+ border-radius: 8px;
1787
+ padding: 12px;
1788
+ background: color-mix(in oklab, var(--surface), var(--bg) 34%);
1789
+ }
1790
+
1791
+ .review-progress-main {
1792
+ display: grid;
1793
+ gap: 8px;
1794
+ min-width: 0;
1795
+ }
1796
+
1797
+ .review-progress-main strong {
1798
+ display: block;
1799
+ font-size: 13px;
1800
+ color: var(--text);
1801
+ }
1802
+
1803
+ .review-progress-main span,
1804
+ .review-progress-side span {
1805
+ display: block;
1806
+ color: var(--muted);
1807
+ font-size: 11.5px;
1808
+ line-height: 1.45;
1809
+ }
1810
+
1811
+ .review-progress-meter {
1812
+ height: 8px;
1813
+ overflow: hidden;
1814
+ border-radius: 999px;
1815
+ background: var(--surface-2);
1816
+ border: 1px solid var(--border-2);
1817
+ }
1818
+
1819
+ .review-progress-meter span {
1820
+ display: block;
1821
+ height: 100%;
1822
+ border-radius: inherit;
1823
+ background: linear-gradient(90deg, var(--c-indigo), var(--good));
1824
+ }
1825
+
1826
+ .review-progress-side {
1827
+ display: grid;
1828
+ justify-items: end;
1829
+ gap: 5px;
1830
+ white-space: nowrap;
1831
+ }
1832
+
1833
+ .batch-actions {
1834
+ display: flex;
1835
+ flex-wrap: wrap;
1836
+ gap: 6px;
1837
+ justify-content: flex-end;
1838
+ }
1839
+
1840
+ table.dt input[type="checkbox"] {
1841
+ width: 15px;
1842
+ height: 15px;
1843
+ accent-color: var(--c-indigo);
1844
+ }
1845
+
1846
+ .auto-attribution-panel {
1847
+ margin: 16px 0;
1848
+ padding: 16px;
1849
+ border: 1px solid var(--border);
1850
+ border-radius: 8px;
1851
+ background: linear-gradient(135deg, color-mix(in oklab, var(--surface), var(--c-indigo) 7%), var(--surface));
1852
+ box-shadow: 0 12px 32px -28px rgb(0 0 0 / 0.25);
1853
+ }
1854
+
1855
+ .auto-attribution-main {
1856
+ display: grid;
1857
+ grid-template-columns: minmax(0, 1fr) auto;
1858
+ gap: 14px;
1859
+ align-items: start;
1860
+ }
1861
+
1862
+ .auto-attribution-main h2 {
1863
+ margin: 3px 0 6px;
1864
+ font-size: 16px;
1865
+ }
1866
+
1867
+ .auto-attribution-main p {
1868
+ margin: 0;
1869
+ color: var(--muted);
1870
+ font-size: 12.5px;
1871
+ line-height: 1.55;
1872
+ }
1873
+
1874
+ .auto-attribution-actions {
1875
+ display: flex;
1876
+ flex-wrap: wrap;
1877
+ gap: 8px;
1878
+ justify-content: flex-end;
1879
+ }
1880
+
1881
+ .auto-attribution-stats {
1882
+ display: grid;
1883
+ grid-template-columns: repeat(4, minmax(0, 1fr));
1884
+ gap: 10px;
1885
+ margin-top: 14px;
1886
+ }
1887
+
1888
+ .auto-attribution-stats div {
1889
+ padding: 10px 12px;
1890
+ border: 1px solid var(--border-2);
1891
+ border-radius: 8px;
1892
+ background: color-mix(in oklab, var(--surface), var(--bg) 26%);
1893
+ }
1894
+
1895
+ .auto-attribution-stats span {
1896
+ display: block;
1897
+ color: var(--muted);
1898
+ font-size: 11.5px;
1899
+ }
1900
+
1901
+ .auto-attribution-stats strong {
1902
+ display: block;
1903
+ margin-top: 3px;
1904
+ font-size: 17px;
1905
+ }
1906
+
1907
+ .auto-attribution-message {
1908
+ margin-top: 10px;
1909
+ padding: 8px 10px;
1910
+ border-radius: 8px;
1911
+ font-size: 12px;
1912
+ }
1913
+
1914
+ .auto-attribution-message-ok {
1915
+ color: var(--good);
1916
+ background: color-mix(in oklab, var(--good), transparent 90%);
1917
+ }
1918
+
1919
+ .auto-attribution-message-error {
1920
+ color: var(--bad);
1921
+ background: color-mix(in oklab, var(--bad), transparent 90%);
1922
+ }
1923
+
1924
+ .attribution-source-badge {
1925
+ display: inline-flex;
1926
+ align-items: center;
1927
+ min-height: 22px;
1928
+ padding: 2px 8px;
1929
+ border-radius: 999px;
1930
+ font-size: 11px;
1931
+ white-space: nowrap;
1932
+ border: 1px solid var(--border-2);
1933
+ background: var(--surface-2);
1934
+ color: var(--muted);
1935
+ }
1936
+
1937
+ .attribution-source-badge.manual,
1938
+ .attribution-source-badge.imported {
1939
+ color: var(--good);
1940
+ background: color-mix(in oklab, var(--good), transparent 90%);
1941
+ border-color: color-mix(in oklab, var(--good), transparent 70%);
1942
+ }
1943
+
1944
+ .attribution-source-badge.auto-high {
1945
+ color: var(--c-indigo);
1946
+ background: color-mix(in oklab, var(--c-indigo), transparent 90%);
1947
+ border-color: color-mix(in oklab, var(--c-indigo), transparent 70%);
1948
+ }
1949
+
1950
+ .attribution-source-badge.auto-low {
1951
+ color: var(--c-amber);
1952
+ background: color-mix(in oklab, var(--c-amber), transparent 88%);
1953
+ border-color: color-mix(in oklab, var(--c-amber), transparent 68%);
1954
+ }
1955
+
1956
+ /* ─── Annotation modal ───────────────────────────────────── */
1957
+
1958
+ .modal-backdrop {
1959
+ position: fixed;
1960
+ inset: 0;
1961
+ background: oklch(0.18 0.005 80 / 0.30);
1962
+ backdrop-filter: blur(2px);
1963
+ z-index: 70;
1964
+ }
1965
+
1966
+ .annotation-modal {
1967
+ position: fixed;
1968
+ top: 50%;
1969
+ left: 50%;
1970
+ transform: translate(-50%, -50%);
1971
+ width: min(560px, calc(100vw - 32px));
1972
+ max-height: calc(100vh - 48px);
1973
+ overflow: auto;
1974
+ background: var(--surface);
1975
+ border: 1px solid var(--border);
1976
+ border-radius: var(--radius);
1977
+ box-shadow: 0 18px 60px -24px rgb(0 0 0 / 0.28), 0 0 0 1px var(--border);
1978
+ z-index: 80;
1979
+ }
1980
+
1981
+ .import-budget-modal {
1982
+ width: min(820px, calc(100vw - 32px));
1983
+ }
1984
+
1985
+ .annotation-modal-header {
1986
+ position: relative;
1987
+ padding: 18px 52px 14px 18px;
1988
+ border-bottom: 1px solid var(--border-2);
1989
+ }
1990
+ .annotation-modal-header h3 {
1991
+ margin: 3px 0 0;
1992
+ font-size: 15px;
1993
+ font-weight: 600;
1994
+ overflow-wrap: anywhere;
1995
+ }
1996
+ .eyebrow {
1997
+ font-size: 11px;
1998
+ color: var(--muted);
1999
+ text-transform: uppercase;
2000
+ letter-spacing: 0.08em;
2001
+ }
2002
+ .annotation-modal-body {
2003
+ padding: 18px;
2004
+ display: grid;
2005
+ gap: 14px;
2006
+ }
2007
+
2008
+ .auto-suggestion-box {
2009
+ display: grid;
2010
+ gap: 10px;
2011
+ padding: 12px;
2012
+ border: 1px solid color-mix(in oklab, var(--c-indigo), transparent 68%);
2013
+ border-radius: 8px;
2014
+ background: color-mix(in oklab, var(--c-indigo), transparent 94%);
2015
+ }
2016
+
2017
+ .auto-suggestion-head {
2018
+ display: flex;
2019
+ justify-content: space-between;
2020
+ gap: 12px;
2021
+ align-items: start;
2022
+ }
2023
+
2024
+ .auto-suggestion-head strong {
2025
+ display: block;
2026
+ font-size: 12.5px;
2027
+ }
2028
+
2029
+ .auto-suggestion-head span {
2030
+ display: block;
2031
+ margin-top: 3px;
2032
+ color: var(--muted);
2033
+ font-size: 11.5px;
2034
+ line-height: 1.5;
2035
+ }
2036
+
2037
+ .auto-suggestion-grid {
2038
+ display: grid;
2039
+ grid-template-columns: repeat(3, minmax(0, 1fr));
2040
+ gap: 8px;
2041
+ }
2042
+
2043
+ .auto-suggestion-grid div {
2044
+ min-width: 0;
2045
+ padding: 7px 8px;
2046
+ border: 1px solid var(--border-2);
2047
+ border-radius: 8px;
2048
+ background: var(--surface);
2049
+ }
2050
+
2051
+ .auto-suggestion-grid span,
2052
+ .auto-suggestion-grid b {
2053
+ display: block;
2054
+ overflow-wrap: anywhere;
2055
+ }
2056
+
2057
+ .auto-suggestion-grid span {
2058
+ color: var(--muted);
2059
+ font-size: 10.5px;
2060
+ }
2061
+
2062
+ .auto-suggestion-grid b {
2063
+ margin-top: 2px;
2064
+ font-size: 12px;
2065
+ }
2066
+ .form-grid {
2067
+ display: grid;
2068
+ grid-template-columns: repeat(2, minmax(0, 1fr));
2069
+ gap: 12px;
2070
+ }
2071
+ .form-grid-3 {
2072
+ grid-template-columns: repeat(3, minmax(0, 1fr));
2073
+ }
2074
+ .form-field-wide {
2075
+ grid-column: 1 / -1;
2076
+ }
2077
+ .form-field {
2078
+ display: grid;
2079
+ gap: 6px;
2080
+ }
2081
+ .form-field span {
2082
+ font-size: 12px;
2083
+ color: var(--muted);
2084
+ }
2085
+ .form-field input,
2086
+ .form-field select,
2087
+ .form-field textarea {
2088
+ width: 100%;
2089
+ border: 1px solid var(--border);
2090
+ border-radius: 8px;
2091
+ background: var(--surface);
2092
+ color: var(--text);
2093
+ font-size: 13px;
2094
+ padding: 9px 10px;
2095
+ }
2096
+ .form-field textarea {
2097
+ min-height: 92px;
2098
+ resize: vertical;
2099
+ }
2100
+ .form-field textarea[readonly] {
2101
+ min-height: 58px;
2102
+ font-family: var(--font-mono);
2103
+ font-size: 12px;
2104
+ line-height: 1.45;
2105
+ color: var(--text-2);
2106
+ background: var(--surface-2);
2107
+ }
2108
+ .form-field input:focus,
2109
+ .form-field select:focus,
2110
+ .form-field textarea:focus {
2111
+ outline: 2px solid color-mix(in oklab, var(--c-indigo), transparent 72%);
2112
+ outline-offset: -1px;
2113
+ border-color: var(--c-indigo);
2114
+ }
2115
+ .preset-row {
2116
+ display: flex;
2117
+ align-items: center;
2118
+ gap: 10px;
2119
+ padding: 9px 10px;
2120
+ border: 1px solid var(--border-2);
2121
+ border-radius: 8px;
2122
+ background: var(--surface-2);
2123
+ }
2124
+ .preset-row span {
2125
+ color: var(--muted);
2126
+ font-size: 11.5px;
2127
+ line-height: 1.45;
2128
+ }
2129
+ .quick-template-picker {
2130
+ display: grid;
2131
+ gap: 9px;
2132
+ padding: 12px;
2133
+ border: 1px solid var(--border-2);
2134
+ border-radius: 10px;
2135
+ background: color-mix(in oklab, var(--surface-2), transparent 18%);
2136
+ }
2137
+ .quick-template-head {
2138
+ display: flex;
2139
+ align-items: baseline;
2140
+ justify-content: space-between;
2141
+ gap: 12px;
2142
+ }
2143
+ .quick-template-head span {
2144
+ color: var(--text);
2145
+ font-size: 12px;
2146
+ font-weight: 650;
2147
+ }
2148
+ .quick-template-head p {
2149
+ margin: 0;
2150
+ color: var(--muted);
2151
+ font-size: 11.5px;
2152
+ line-height: 1.45;
2153
+ text-align: right;
2154
+ }
2155
+ .quick-template-grid {
2156
+ display: grid;
2157
+ grid-template-columns: repeat(2, minmax(0, 1fr));
2158
+ gap: 8px;
2159
+ }
2160
+ .quick-template-button {
2161
+ display: grid;
2162
+ gap: 3px;
2163
+ min-height: 52px;
2164
+ padding: 9px 10px;
2165
+ border: 1px solid var(--border);
2166
+ border-radius: 8px;
2167
+ background: var(--surface);
2168
+ color: var(--text);
2169
+ cursor: pointer;
2170
+ text-align: left;
2171
+ }
2172
+ .quick-template-button:hover {
2173
+ border-color: var(--c-indigo);
2174
+ background: color-mix(in oklab, var(--c-indigo), transparent 94%);
2175
+ }
2176
+ .quick-template-button:disabled {
2177
+ cursor: not-allowed;
2178
+ opacity: 0.6;
2179
+ }
2180
+ .quick-template-button strong {
2181
+ font-size: 12.5px;
2182
+ line-height: 1.25;
2183
+ }
2184
+ .quick-template-button span {
2185
+ color: var(--muted);
2186
+ font-size: 11.5px;
2187
+ line-height: 1.35;
2188
+ }
2189
+ .annotation-meta {
2190
+ display: flex;
2191
+ flex-wrap: wrap;
2192
+ gap: 8px;
2193
+ color: var(--muted);
2194
+ font-size: 11.5px;
2195
+ }
2196
+ .annotation-meta span {
2197
+ background: var(--surface-2);
2198
+ border: 1px solid var(--border-2);
2199
+ border-radius: 999px;
2200
+ padding: 3px 8px;
2201
+ }
2202
+ .form-error {
2203
+ padding: 9px 10px;
2204
+ border: 1px solid color-mix(in oklab, var(--bad), transparent 60%);
2205
+ border-radius: 8px;
2206
+ color: var(--bad);
2207
+ background: color-mix(in oklab, var(--bad), transparent 92%);
2208
+ font-size: 12px;
2209
+ }
2210
+ .notice-list {
2211
+ display: grid;
2212
+ gap: 10px;
2213
+ font-size: 13px;
2214
+ color: var(--text-2);
2215
+ }
2216
+
2217
+ .notice-list > div {
2218
+ padding: 10px 12px;
2219
+ border: 1px solid var(--border-2);
2220
+ border-radius: 8px;
2221
+ background: var(--surface-2);
2222
+ }
2223
+
2224
+ .import-budget-body {
2225
+ gap: 18px;
2226
+ }
2227
+
2228
+ .import-budget-section {
2229
+ display: grid;
2230
+ gap: 12px;
2231
+ padding: 14px;
2232
+ border: 1px solid var(--border-2);
2233
+ border-radius: 10px;
2234
+ background: color-mix(in oklab, var(--surface), var(--bg) 18%);
2235
+ }
2236
+
2237
+ .import-budget-section-head {
2238
+ display: flex;
2239
+ justify-content: space-between;
2240
+ gap: 16px;
2241
+ align-items: flex-start;
2242
+ }
2243
+
2244
+ .import-budget-section-head h4 {
2245
+ margin: 0 0 4px;
2246
+ font-size: 14px;
2247
+ }
2248
+
2249
+ .import-budget-section-head p {
2250
+ margin: 0;
2251
+ color: var(--muted);
2252
+ font-size: 12px;
2253
+ line-height: 1.5;
2254
+ }
2255
+
2256
+ .import-budget-actions {
2257
+ display: flex;
2258
+ flex-wrap: wrap;
2259
+ gap: 8px;
2260
+ align-items: center;
2261
+ }
2262
+
2263
+ .import-budget-command-actions {
2264
+ align-items: end;
2265
+ justify-content: flex-start;
2266
+ padding-top: 22px;
2267
+ }
2268
+
2269
+ .budget-template-row {
2270
+ display: flex;
2271
+ flex-wrap: wrap;
2272
+ gap: 8px;
2273
+ }
2274
+
2275
+ .import-preview {
2276
+ display: grid;
2277
+ gap: 10px;
2278
+ padding: 12px;
2279
+ border: 1px solid var(--border-2);
2280
+ border-radius: 10px;
2281
+ background: var(--surface);
2282
+ }
2283
+
2284
+ .import-preview-apply {
2285
+ border-color: color-mix(in oklab, var(--good), transparent 68%);
2286
+ background: color-mix(in oklab, var(--good), transparent 94%);
2287
+ }
2288
+
2289
+ .import-preview-grid {
2290
+ display: grid;
2291
+ grid-template-columns: repeat(6, minmax(0, 1fr));
2292
+ gap: 8px;
2293
+ }
2294
+
2295
+ .import-preview-grid div {
2296
+ padding: 8px;
2297
+ border: 1px solid var(--border-2);
2298
+ border-radius: 8px;
2299
+ background: color-mix(in oklab, var(--surface-2), transparent 24%);
2300
+ }
2301
+
2302
+ .import-preview-grid span,
2303
+ .import-preview-grid strong {
2304
+ display: block;
2305
+ min-width: 0;
2306
+ overflow-wrap: anywhere;
2307
+ }
2308
+
2309
+ .import-preview-grid span {
2310
+ color: var(--muted);
2311
+ font-size: 10.5px;
2312
+ }
2313
+
2314
+ .import-preview-grid strong {
2315
+ margin-top: 3px;
2316
+ font-size: 12px;
2317
+ }
2318
+
2319
+ .import-preview p {
2320
+ margin: 0;
2321
+ color: var(--muted);
2322
+ font-size: 12px;
2323
+ }
2324
+
2325
+ .import-warning-list,
2326
+ .budget-profile-list {
2327
+ display: grid;
2328
+ gap: 8px;
2329
+ }
2330
+
2331
+ .import-warning-list div {
2332
+ padding: 9px 10px;
2333
+ border: 1px solid color-mix(in oklab, var(--c-amber), transparent 65%);
2334
+ border-radius: 8px;
2335
+ background: color-mix(in oklab, var(--c-amber), transparent 92%);
2336
+ }
2337
+
2338
+ .import-warning-list strong,
2339
+ .import-warning-list span {
2340
+ display: block;
2341
+ overflow-wrap: anywhere;
2342
+ }
2343
+
2344
+ .import-warning-list strong {
2345
+ font-size: 12px;
2346
+ color: oklch(0.48 0.13 72);
2347
+ }
2348
+
2349
+ .import-warning-list span {
2350
+ margin-top: 2px;
2351
+ color: var(--muted);
2352
+ font-size: 11.5px;
2353
+ line-height: 1.45;
2354
+ }
2355
+
2356
+ .import-budget-toggle input {
2357
+ width: 18px;
2358
+ height: 18px;
2359
+ accent-color: var(--c-indigo);
2360
+ }
2361
+
2362
+ .budget-profile-row {
2363
+ display: grid;
2364
+ grid-template-columns: minmax(0, 1fr) auto auto;
2365
+ gap: 12px;
2366
+ align-items: center;
2367
+ padding: 11px 12px;
2368
+ border: 1px solid var(--border-2);
2369
+ border-radius: 8px;
2370
+ background: var(--surface);
2371
+ }
2372
+
2373
+ .budget-profile-row.disabled {
2374
+ opacity: 0.62;
2375
+ }
2376
+
2377
+ .budget-profile-row strong,
2378
+ .budget-profile-row b,
2379
+ .budget-profile-row span {
2380
+ display: block;
2381
+ min-width: 0;
2382
+ }
2383
+
2384
+ .budget-profile-row strong,
2385
+ .budget-profile-row b {
2386
+ color: var(--text);
2387
+ font-size: 12.5px;
2388
+ }
2389
+
2390
+ .budget-profile-row span {
2391
+ margin-top: 2px;
2392
+ color: var(--muted);
2393
+ font-size: 11.5px;
2394
+ line-height: 1.45;
2395
+ }
2396
+
2397
+ .annotation-modal-actions {
2398
+ display: flex;
2399
+ align-items: center;
2400
+ gap: 8px;
2401
+ padding: 14px 18px 18px;
2402
+ border-top: 1px solid var(--border-2);
2403
+ }
2404
+ .form-spacer { flex: 1; }
2405
+
2406
+ .divider { width: 1px; height: 18px; background: var(--border); margin: 0 4px; }
2407
+
2408
+ .scroll-area { max-height: 360px; overflow-y: auto; padding-right: 4px; }
2409
+ .scroll-area::-webkit-scrollbar { width: 6px; height: 6px; }
2410
+ .scroll-area::-webkit-scrollbar-thumb { background: var(--border); border-radius: 3px; }
2411
+
2412
+ .empty {
2413
+ padding: 28px;
2414
+ text-align: center;
2415
+ color: var(--muted);
2416
+ font-size: 12.5px;
2417
+ }
2418
+
2419
+ .spin {
2420
+ animation: spin 1s linear infinite;
2421
+ }
2422
+ @keyframes spin {
2423
+ to { transform: rotate(360deg); }
2424
+ }
2425
+
2426
+ /* responsive — small fallback */
2427
+ @media (max-width: 1180px) {
2428
+ .kpi-row { grid-template-columns: repeat(3, 1fr); }
2429
+ .model-card-grid { grid-template-columns: repeat(2, minmax(0, 1fr)); }
2430
+ .attribution-card-grid { grid-template-columns: repeat(3, minmax(0, 1fr)); }
2431
+ .roi-review { grid-template-columns: 1fr; }
2432
+ .col-8 { grid-column: span 12; }
2433
+ .col-4 { grid-column: span 12; }
2434
+ .col-6 { grid-column: span 12; }
2435
+ }
2436
+
2437
+ @media (max-width: 720px) {
2438
+ .app {
2439
+ padding: 16px 12px 40px;
2440
+ }
2441
+ .topbar,
2442
+ .topbar-left,
2443
+ .topbar-right,
2444
+ .filter-row {
2445
+ align-items: stretch;
2446
+ flex-direction: column;
2447
+ }
2448
+ .page-switch,
2449
+ .panel-tabs,
2450
+ .panel-actions,
2451
+ .annotation-modal-actions {
2452
+ overflow-x: auto;
2453
+ }
2454
+ .panel-actions,
2455
+ .batch-bar,
2456
+ .review-progress,
2457
+ .pricing-notice,
2458
+ .model-overview-head,
2459
+ .model-overview-actions,
2460
+ .attribution-overview-side,
2461
+ .project-roi-main,
2462
+ .project-roi-meta {
2463
+ align-items: stretch;
2464
+ flex-direction: column;
2465
+ }
2466
+ .review-progress {
2467
+ grid-template-columns: 1fr;
2468
+ }
2469
+ .auto-attribution-main,
2470
+ .auto-attribution-stats,
2471
+ .auto-suggestion-grid {
2472
+ grid-template-columns: 1fr;
2473
+ }
2474
+ .auto-attribution-actions {
2475
+ justify-content: flex-start;
2476
+ }
2477
+ .first-run-main,
2478
+ .first-run-steps {
2479
+ grid-template-columns: 1fr;
2480
+ }
2481
+ .first-run-actions {
2482
+ justify-content: flex-start;
2483
+ }
2484
+ .source-health-head,
2485
+ .source-health-actions {
2486
+ align-items: stretch;
2487
+ flex-direction: column;
2488
+ }
2489
+ .source-health-stats,
2490
+ .source-health-grid {
2491
+ grid-template-columns: 1fr;
2492
+ }
2493
+ .first-run-notices button {
2494
+ align-items: flex-start;
2495
+ flex-direction: column;
2496
+ }
2497
+ .review-progress-side {
2498
+ justify-items: stretch;
2499
+ white-space: normal;
2500
+ }
2501
+ .pricing-unpriced {
2502
+ justify-content: flex-start;
2503
+ max-width: none;
2504
+ }
2505
+ .batch-actions {
2506
+ justify-content: flex-start;
2507
+ }
2508
+ .risk-row {
2509
+ grid-template-columns: minmax(0, 1fr) 74px 42px;
2510
+ }
2511
+ .weekly-grid {
2512
+ grid-template-columns: 1fr;
2513
+ }
2514
+ .kpi-row {
2515
+ grid-template-columns: 1fr;
2516
+ }
2517
+ .attribution-overview {
2518
+ padding: 14px;
2519
+ }
2520
+ .attribution-overview-head {
2521
+ display: grid;
2522
+ gap: 10px;
2523
+ }
2524
+ .attribution-overview-total {
2525
+ justify-items: start;
2526
+ }
2527
+ .model-card-grid {
2528
+ grid-template-columns: 1fr;
2529
+ }
2530
+ .model-card-meta,
2531
+ .model-card-sub {
2532
+ align-items: flex-start;
2533
+ flex-direction: column;
2534
+ gap: 3px;
2535
+ }
2536
+ .model-card-sub span:last-child {
2537
+ text-align: left;
2538
+ max-width: 100%;
2539
+ }
2540
+ .attribution-card-grid {
2541
+ grid-template-columns: 1fr;
2542
+ }
2543
+ .grid {
2544
+ gap: 12px;
2545
+ }
2546
+ table.dt {
2547
+ min-width: 1180px;
2548
+ }
2549
+ .drawer {
2550
+ width: 100vw;
2551
+ max-width: 100vw;
2552
+ }
2553
+ .annotation-modal {
2554
+ width: 100vw;
2555
+ min-height: 100vh;
2556
+ max-height: 100vh;
2557
+ border-radius: 0;
2558
+ }
2559
+ .import-budget-modal {
2560
+ width: 100vw;
2561
+ }
2562
+ .form-grid {
2563
+ grid-template-columns: 1fr;
2564
+ }
2565
+ .form-field-wide {
2566
+ grid-column: auto;
2567
+ }
2568
+ .import-budget-section-head,
2569
+ .budget-profile-row {
2570
+ grid-template-columns: 1fr;
2571
+ align-items: stretch;
2572
+ }
2573
+ .import-budget-section-head {
2574
+ flex-direction: column;
2575
+ }
2576
+ .import-budget-command-actions {
2577
+ padding-top: 0;
2578
+ }
2579
+ .budget-template-row {
2580
+ align-items: stretch;
2581
+ flex-direction: column;
2582
+ }
2583
+ .import-preview-grid {
2584
+ grid-template-columns: repeat(2, minmax(0, 1fr));
2585
+ }
2586
+ .preset-row {
2587
+ align-items: stretch;
2588
+ flex-direction: column;
2589
+ }
2590
+ .quick-template-head {
2591
+ align-items: flex-start;
2592
+ flex-direction: column;
2593
+ gap: 4px;
2594
+ }
2595
+ .quick-template-head p {
2596
+ text-align: left;
2597
+ }
2598
+ .quick-template-grid {
2599
+ grid-template-columns: 1fr;
2600
+ }
2601
+ .annotation-modal-actions {
2602
+ position: sticky;
2603
+ bottom: 0;
2604
+ background: var(--surface);
2605
+ }
2606
+ }