@jefuriiij/synthra 0.1.2 → 0.1.3

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.
package/dist/cli/index.js CHANGED
@@ -18,7 +18,7 @@ var init_package = __esm({
18
18
  "package.json"() {
19
19
  package_default = {
20
20
  name: "@jefuriiij/synthra",
21
- version: "0.1.2",
21
+ version: "0.1.3",
22
22
  publishConfig: {
23
23
  access: "public"
24
24
  },
@@ -438,107 +438,439 @@ var public_default = `<!doctype html>
438
438
  <head>
439
439
  <meta charset="UTF-8" />
440
440
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
441
- <title>Synthra \u2014 Token Dashboard</title>
441
+ <title>Synthra \xB7 Dashboard</title>
442
+ <link rel="preconnect" href="https://fonts.googleapis.com">
443
+ <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
444
+ <link href="https://fonts.googleapis.com/css2?family=Instrument+Serif:ital@0;1&family=Geist:wght@400;500;600;700&family=Geist+Mono:wght@400;500&display=swap" rel="stylesheet">
442
445
  <link rel="stylesheet" href="./style.css" />
443
446
  </head>
444
447
  <body>
445
- <header>
448
+
449
+ <!-- ============ Top nav ============ -->
450
+ <header class="topnav">
446
451
  <div class="brand">
447
- <h1>Synthra</h1>
448
- <span class="tag">Token Dashboard</span>
452
+ <div class="brand-mark"></div>
453
+ <div class="brand-name">Synth<em>ra</em></div>
454
+ <div class="brand-eyebrow">Dashboard</div>
455
+ </div>
456
+ <div class="top-right">
457
+ <span class="status-pill">
458
+ <span class="dot" id="dot"></span>
459
+ <span id="status">connecting\u2026</span>
460
+ </span>
449
461
  </div>
450
- <div class="meta">
451
- <span class="active-project" id="active-project">\u2026</span>
452
- <span class="dot" id="dot"></span>
453
- <span id="status">connecting\u2026</span>
462
+ <div class="topnav-right">
463
+ <span class="port-badge">port <span class="mono" id="port-num">8901</span></span>
464
+ <button class="faq-btn has-tooltip" id="faq-btn" data-tooltip="Open the FAQ \u2014 explains where every number on this dashboard comes from, how cost is calculated, and what the savings floor actually measures." aria-label="Open FAQ">?</button>
454
465
  </div>
455
466
  </header>
456
467
 
457
- <main>
458
- <section>
459
- <h2>
460
- <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M3 3v18h18"/><path d="M7 14l4-4 4 4 5-5"/></svg>
461
- Global totals
462
- <span class="muted">(all projects)</span>
463
- </h2>
464
- <div class="cards" id="cards"></div>
465
- </section>
466
-
467
- <section>
468
- <h2>
469
- <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M22 19a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h5l2 3h9a2 2 0 0 1 2 2z"/></svg>
470
- Projects
471
- </h2>
472
- <div class="projects" id="projects"></div>
473
- <p class="empty hidden" id="projects-empty">No projects registered yet. Run <code>syn .</code> in any project to add it.</p>
474
- </section>
475
-
476
- <section>
477
- <h2>
478
- <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><circle cx="12" cy="12" r="10"/><polyline points="12 6 12 12 16 14"/></svg>
479
- Recent calls
480
- <span class="muted">(across all projects)</span>
481
- </h2>
482
- <table id="turns">
483
- <thead>
484
- <tr>
485
- <th>Time</th>
486
- <th>Project</th>
487
- <th>Model</th>
488
- <th class="num"><span class="has-tooltip" data-tooltip="New (uncached) tokens you sent to Claude this turn. Usually small \u2014 most of the conversation comes from cache.">Input <span class="help-icon">i</span></span></th>
489
- <th class="num"><span class="has-tooltip" data-tooltip="Tokens Claude generated in its response. The most expensive line item \u2014 ~5\xD7 the input rate on Opus.">Output <span class="help-icon">i</span></span></th>
490
- <th class="num"><span class="has-tooltip" data-tooltip="Cache read / cache write. Reads (~10% of input rate) reuse prior context; writes (~125% of input rate) save new context for future turns.">Cache R / W <span class="help-icon">i</span></span></th>
491
- <th class="num"><span class="has-tooltip" data-tooltip="Approximate USD cost for this turn \u2014 input \xD7 rate + output \xD7 5\xD7rate + cache_read \xD7 0.1\xD7rate + cache_write \xD7 1.25\xD7rate, using the turn's model.">Cost <span class="help-icon">i</span></span></th>
492
- </tr>
493
- </thead>
494
- <tbody></tbody>
495
- </table>
496
- <p class="empty hidden" id="turns-empty">No turns logged yet. Use Claude via the IDE extension or <code>claude</code> CLI while <code>syn .</code> is running.</p>
497
- </section>
498
-
499
- <section>
500
- <h2>
501
- <svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M12 22s8-4 8-10V5l-8-3-8 3v7c0 6 8 10 8 10z"/></svg>
502
- Recent gate decisions
503
- </h2>
504
- <table id="gates">
505
- <thead>
506
- <tr>
507
- <th>Time</th>
508
- <th>Project</th>
509
- <th>Tool</th>
510
- <th><span class="has-tooltip" data-tooltip="ALLOW = Synthra let the tool call through. BLOCK = Synthra intercepted it because the graph already had high-confidence context \u2014 Claude should use graph_continue instead.">Decision <span class="help-icon">i</span></span></th>
511
- <th>Query</th>
512
- </tr>
513
- </thead>
514
- <tbody></tbody>
515
- </table>
516
- <p class="empty hidden" id="gates-empty">No gate decisions yet.</p>
517
- </section>
468
+ <!-- ============ Hero strip ============ -->
469
+ <section class="hero-strip">
470
+ <div class="date-block">
471
+ <div class="d-day" id="hero-day">\u2014</div>
472
+ <div class="d-rest">
473
+ <div id="hero-weekday">\u2014</div>
474
+ <div class="d-mute" id="hero-month">\u2014</div>
475
+ </div>
476
+ </div>
477
+
478
+ <div class="hero-spacer"></div>
479
+
480
+ <div class="active-block">
481
+ <div class="ab-label">Active project</div>
482
+ <div class="ab-value" id="active-project">\u2014</div>
483
+ </div>
484
+
485
+ </section>
486
+
487
+ <!-- ============ Main 3-column grid ============ -->
488
+ <main class="grid-main">
489
+
490
+ <!-- ===== Left ===== -->
491
+ <aside class="col-left">
492
+ <div class="panel">
493
+ <div class="p-head">Legend</div>
494
+
495
+ <div class="p-section">
496
+ <div class="ps-head">Model family</div>
497
+ <div class="check"><span class="dot-sq opus"></span><span>Opus</span></div>
498
+ <div class="check"><span class="dot-sq sonnet"></span><span>Sonnet</span></div>
499
+ <div class="check"><span class="dot-sq haiku"></span><span>Haiku</span></div>
500
+ <div class="check"><span class="dot-sq unknown"></span><span>Other</span></div>
501
+ </div>
502
+
503
+ <div class="p-section">
504
+ <div class="ps-head">Projects</div>
505
+ <div class="proj-filter" id="proj-filter"></div>
506
+ </div>
507
+ </div>
508
+
509
+ <div class="card donut-card has-tooltip" data-tooltip="Which Claude models you've been calling, weighted by turn count. Opus = slow and expensive; Sonnet = workhorse; Haiku = cheap and fast. Helps you see where your budget is actually going.">
510
+ <div class="card-head">
511
+ <div class="card-eyebrow">Model usage</div>
512
+ <div class="card-meta">by turns</div>
513
+ </div>
514
+ <div class="donut-wrap">
515
+ <svg viewBox="0 0 140 140" class="donut" id="donut-svg" aria-hidden="true">
516
+ <circle cx="70" cy="70" r="52" class="donut-track"/>
517
+ </svg>
518
+ <div class="donut-center">
519
+ <div class="donut-total" id="donut-total">0</div>
520
+ <div class="donut-total-k">turns</div>
521
+ </div>
522
+ </div>
523
+ <div class="donut-legend" id="donut-legend"></div>
524
+ </div>
525
+ </aside>
526
+
527
+ <!-- ===== Center ===== -->
528
+ <div class="col-center">
529
+
530
+ <!-- Metric strip \u2014 divider-separated, no individual card chrome -->
531
+ <div class="metric-strip">
532
+ <div class="metric-item has-tooltip" data-tooltip="Total back-and-forth exchanges with Claude across all projects. One turn = you send a message, Claude responds. Counted from the Stop hook against transcript JSONL files.">
533
+ <div class="m-label">Turns</div>
534
+ <div class="m-value" id="m-turns">0</div>
535
+ </div>
536
+ <div class="metric-item has-tooltip" data-tooltip="\u2193 Input \u2014 new, uncached tokens you sent to Claude. Usually small (a few hundred per turn) because most of the conversation history comes from prompt cache.">
537
+ <div class="m-label">\u2193 Input</div>
538
+ <div class="m-value" id="m-input">0</div>
539
+ </div>
540
+ <div class="metric-item has-tooltip" data-tooltip="\u2191 Output \u2014 tokens Claude generated in its responses. Most expensive line item per turn (~5\xD7 input rate on Opus). High output usually means long code edits.">
541
+ <div class="m-label">\u2191 Output</div>
542
+ <div class="m-value" id="m-output">0</div>
543
+ </div>
544
+ <div class="metric-item has-tooltip" data-tooltip="\u27F2 Cache read \u2014 tokens reused from the prompt cache (system prompt, conversation history, Synthra's pre-packed context). Cheap, around 10% of the input rate. The bulk of every long session.">
545
+ <div class="m-label">\u27F2 Cache R</div>
546
+ <div class="m-value" id="m-cache-r">0</div>
547
+ </div>
548
+ <div class="metric-item has-tooltip" data-tooltip="\uFF0B Cache write \u2014 tokens newly added to the prompt cache so future turns can read them cheaply. Premium-priced (~125% of input rate) but pays back across the session.">
549
+ <div class="m-label">\uFF0B Cache W</div>
550
+ <div class="m-value" id="m-cache-w">0</div>
551
+ </div>
552
+ </div>
553
+
554
+ <!-- Savings hero -->
555
+ <div class="card savings has-tooltip" data-tooltip="What Synthra has saved you, as a deliberately conservative floor estimate. Each time the gate blocks an exploratory Grep/Glob, we credit 500 tokens \xD7 $3 per million-token input rate. Real savings are usually higher because the formula ignores cache thrash and follow-up Reads that the block also prevents. The audit line below shows the exact math live.">
556
+ <div class="card-head">
557
+ <div class="card-eyebrow">Synthra savings <span class="src-badge estimated">floor</span></div>
558
+ <div class="card-meta" id="savings-pct">\u2014 off</div>
559
+ </div>
560
+ <div class="savings-body">
561
+ <div class="savings-figure">
562
+ <div class="savings-money" id="savings-money">$0.00</div>
563
+ <div class="savings-tokens"><span id="savings-tokens">0</span> tokens avoided</div>
564
+ </div>
565
+ <div class="savings-bar">
566
+ <div class="savings-actual" id="savings-actual-bar" style="width:100%"></div>
567
+ <div class="savings-saved" id="savings-saved-bar" style="width:0%"></div>
568
+ </div>
569
+ <div class="savings-legend">
570
+ <div class="sl-row"><span class="sl-dot actual"></span>You paid <b id="savings-actual-amt">$0.00</b></div>
571
+ <div class="sl-row"><span class="sl-dot saved"></span>Baseline <b id="savings-baseline-amt">$0.00</b></div>
572
+ </div>
573
+ </div>
574
+ <div class="savings-audit">
575
+ <span class="audit-formula">
576
+ <b id="audit-blocks">0</b> blocks \xD7 <b>500</b> tokens \xD7 <b>$3</b> / M input rate = <b id="audit-result" class="audit-result">$0.00</b>
577
+ </span>
578
+ </div>
579
+ </div>
580
+
581
+ <!-- Recent turns -->
582
+ <div class="card turns-card has-tooltip" data-tooltip="Every conversational turn Synthra has observed across all your projects, newest first. Each row shows when, which project, which model, and how the cost broke down between fresh input, generated output, and cache.">
583
+ <div class="card-head">
584
+ <div class="card-eyebrow">Recent turns</div>
585
+ <div class="card-meta" id="turns-count">\u2014 shown</div>
586
+ </div>
587
+ <div class="turns-scroll">
588
+ <table class="turns-table">
589
+ <thead>
590
+ <tr>
591
+ <th>Time</th>
592
+ <th>Project</th>
593
+ <th>Model</th>
594
+ <th class="num">In</th>
595
+ <th class="num">Out</th>
596
+ <th class="num">Cache R/W</th>
597
+ <th class="num">Cost</th>
598
+ </tr>
599
+ </thead>
600
+ <tbody id="turns-body"></tbody>
601
+ </table>
602
+ <p class="empty hidden" id="turns-empty">No turns logged yet. Run <code>syn .</code> in any project and chat with Claude.</p>
603
+ </div>
604
+ </div>
605
+
606
+ </div>
607
+
608
+ <!-- ===== Right ===== -->
609
+ <aside class="col-right">
610
+
611
+ <!-- Cost hero -->
612
+ <div class="card cost-hero has-tooltip" data-tooltip="Your all-time Claude spend across every project Synthra has tracked on this machine. Token counts come from Claude's transcript JSONL files; dollar amounts are computed by multiplying those counts by Anthropic's published per-model rates. See the FAQ for full rate tables.">
613
+ <div class="card-head">
614
+ <div class="card-eyebrow">Total spend \xB7 <em>all time</em></div>
615
+ </div>
616
+ <div class="big-money" id="big-cost">$0.<em>00</em></div>
617
+ <div class="cost-sub">
618
+ <div class="cs-row">
619
+ <span class="cs-k">Tokens (in+out)</span>
620
+ <span class="cs-v" id="cs-tokens">0</span>
621
+ </div>
622
+ <div class="cs-row">
623
+ <span class="cs-k">Avg / turn</span>
624
+ <span class="cs-v" id="cs-avg">$0.00</span>
625
+ </div>
626
+ </div>
627
+ </div>
628
+
629
+ <!-- The Moat -->
630
+ <div class="card moat has-tooltip" data-tooltip="Synthra's PreToolUse hook intercepts. Each block = Synthra recognized the graph already had high-confidence context for the query, so it stopped Claude from running an exploratory Grep or Glob. The list below shows the latest decisions across all projects.">
631
+ <div class="card-head">
632
+ <div class="card-eyebrow">The <em>Moat</em></div>
633
+ <div class="card-meta">PreToolUse</div>
634
+ </div>
635
+ <div class="moat-value"><span id="blocks">0</span> <em>blocks</em></div>
636
+ <div class="gate-mini" id="gate-mini"></div>
637
+ </div>
638
+
639
+ </aside>
518
640
  </main>
519
641
 
520
- <footer>
521
- <span>Synthra Token Dashboard \xB7 live polling every 2s</span>
522
- <span class="muted">Cost figures are approximate \u2014 based on published Anthropic rates.</span>
642
+ <!-- ============ Project dialog ============ -->
643
+ <div class="dialog-backdrop hidden" id="dialog-backdrop" role="dialog" aria-modal="true">
644
+ <div class="dialog">
645
+ <button class="dialog-close" id="dialog-close" aria-label="Close">\xD7</button>
646
+ <div class="dialog-eyebrow">Project \xB7 <em>details</em></div>
647
+ <div class="dialog-name" id="d-name">\u2014</div>
648
+ <div class="dialog-path" id="d-path">\u2014</div>
649
+ <div class="dialog-grid">
650
+ <div class="dg-cell">
651
+ <div class="dg-k">Total cost</div>
652
+ <div class="dg-v money" id="d-cost">$0.00</div>
653
+ </div>
654
+ <div class="dg-cell">
655
+ <div class="dg-k">Turns</div>
656
+ <div class="dg-v" id="d-turns">0</div>
657
+ </div>
658
+ <div class="dg-cell">
659
+ <div class="dg-k">\u2193 Raw input</div>
660
+ <div class="dg-v" id="d-input">0</div>
661
+ </div>
662
+ <div class="dg-cell">
663
+ <div class="dg-k">\u2191 Output</div>
664
+ <div class="dg-v" id="d-output">0</div>
665
+ </div>
666
+ <div class="dg-cell">
667
+ <div class="dg-k">\u27F2 Cache read</div>
668
+ <div class="dg-v" id="d-cache-r">0</div>
669
+ </div>
670
+ <div class="dg-cell">
671
+ <div class="dg-k">\uFF0B Cache write</div>
672
+ <div class="dg-v" id="d-cache-w">0</div>
673
+ </div>
674
+ <div class="dg-cell">
675
+ <div class="dg-k">Moat blocks</div>
676
+ <div class="dg-v" id="d-blocks">0</div>
677
+ </div>
678
+ <div class="dg-cell">
679
+ <div class="dg-k">Last active</div>
680
+ <div class="dg-v dg-v-sm" id="d-last">\u2014</div>
681
+ </div>
682
+ </div>
683
+ </div>
684
+ </div>
685
+
686
+ <!-- ============ FAQ dialog ============ -->
687
+ <div class="dialog-backdrop hidden" id="faq-backdrop" role="dialog" aria-modal="true" aria-label="Dashboard FAQ">
688
+ <div class="dialog dialog-faq">
689
+ <button class="dialog-close" id="faq-close" aria-label="Close">\xD7</button>
690
+ <div class="dialog-eyebrow">FAQ \xB7 <em>where every number comes from</em></div>
691
+ <div class="dialog-name">Understanding your dashboard</div>
692
+ <div class="dialog-path">Synthra reports what Claude actually used \u2014 read from transcripts, not estimated.</div>
693
+
694
+ <div class="faq-content">
695
+
696
+ <details open>
697
+ <summary>Where does Synthra get these numbers from?</summary>
698
+ <div class="faq-body">
699
+ <p>Synthra <strong>does not estimate</strong> token counts \u2014 it reads them directly from Claude's own log files. Every number on this dashboard is traceable:</p>
700
+ <table>
701
+ <tr><td>Turns, \u2193 In, \u2191 Out, Cache R/W</td><td>Parsed from Claude's transcript JSONLs at <code>~/.claude/projects/&lt;encoded-cwd&gt;/*.jsonl</code></td></tr>
702
+ <tr><td>Model used per turn</td><td>From each turn's <code>model</code> field in the same transcripts</td></tr>
703
+ <tr><td>Moat blocks + gate decisions</td><td>From <code>.synthra-graph/gate_log.jsonl</code> inside each project</td></tr>
704
+ <tr><td>Active project, project list</td><td>From <code>~/.synthra/projects.json</code> (built up as you run <code>syn .</code>)</td></tr>
705
+ <tr><td>Total spend (USD)</td><td>Above token counts \xD7 Anthropic's published per-model rates \u2014 see "How is cost calculated?" below</td></tr>
706
+ <tr><td>Synthra savings (USD)</td><td><strong>Estimated</strong> \u2014 see "About the Savings (floor) card" below</td></tr>
707
+ </table>
708
+ <p>The cost-calculation logic lives in <code>src/shared/pricing.ts</code>; the aggregation logic in <code>src/dashboard/delta.ts</code>. Both are linked at the bottom.</p>
709
+ </div>
710
+ </details>
711
+
712
+ <details>
713
+ <summary>What do the columns in Recent Turns mean?</summary>
714
+ <div class="faq-body">
715
+ <table>
716
+ <tr><td><code>Time</code></td><td>When this turn happened (local time)</td></tr>
717
+ <tr><td><code>Project</code></td><td>Which directory the turn ran in</td></tr>
718
+ <tr><td><code>Model</code></td><td>Claude model used \u2014 Opus, Sonnet, or Haiku (color-coded in the model pill)</td></tr>
719
+ <tr><td><code>In</code></td><td>Raw input tokens \u2014 brand-new content sent to Claude this turn, not cached</td></tr>
720
+ <tr><td><code>Out</code></td><td>Tokens Claude generated in its response</td></tr>
721
+ <tr><td><code>Cache R/W</code></td><td>Cache <strong>R</strong>ead (reused from prior turns) / Cache <strong>W</strong>rite (newly cached for future turns)</td></tr>
722
+ <tr><td><code>Cost</code></td><td>Per-turn USD estimate using Anthropic's published rates</td></tr>
723
+ </table>
724
+ <p><strong>Why is Raw Input often tiny (e.g. 6 tokens)?</strong> Because Claude Code aggressively caches the system prompt, CLAUDE.md, tool definitions, and conversation history. On each turn, only your brand-new message is "raw input" \u2014 everything else is a cheap cache read. This is normal and saves significant money.</p>
725
+ </div>
726
+ </details>
727
+
728
+ <details>
729
+ <summary>How is cost calculated?</summary>
730
+ <div class="faq-body">
731
+ <p>Each token type has a different per-million-token rate. Synthra uses these rates (defined in <code>src/shared/pricing.ts</code>):</p>
732
+ <table>
733
+ <thead><tr><td><strong>Token type</strong></td><td><strong>Haiku 4.5</strong></td><td><strong>Sonnet 4.x</strong></td><td><strong>Opus 4.x</strong></td></tr></thead>
734
+ <tr><td>Raw Input</td><td>$1.00/M</td><td>$3.00/M</td><td>$15.00/M</td></tr>
735
+ <tr><td>Cache Write</td><td>$1.25/M</td><td>$3.75/M</td><td>$18.75/M</td></tr>
736
+ <tr><td>Cache Read</td><td>$0.10/M</td><td>$0.30/M</td><td>$1.50/M</td></tr>
737
+ <tr><td>Output</td><td>$5.00/M</td><td>$15.00/M</td><td>$75.00/M</td></tr>
738
+ </table>
739
+ <p><strong>Cost</strong> = (Input \xD7 input rate) + (Output \xD7 output rate) + (Cache Read \xD7 read rate) + (Cache Write \xD7 write rate)</p>
740
+ <p>Cache reads are <strong>10\xD7 cheaper</strong> than raw input. Cache writes are <strong>25% more expensive</strong> than raw input. So Claude Code's caching strategy pays for itself quickly across a session.</p>
741
+ <div class="warning"><span class="icon">\u26A0</span>These are <strong>Anthropic API rates</strong>, not your plan billing. If you're on Claude Pro, Team, Max, or Enterprise, your actual billing is different \u2014 the costs shown are estimates of <em>API-equivalent</em> usage, useful for comparing sessions against each other. See <a href="https://www.anthropic.com/pricing" target="_blank" rel="noopener noreferrer">anthropic.com/pricing</a> for the source of these rates.</div>
742
+ </div>
743
+ </details>
744
+
745
+ <details>
746
+ <summary>What is the total context size per turn?</summary>
747
+ <div class="faq-body">
748
+ <p><code>Total context = Raw Input + Cache Read + Cache Write</code></p>
749
+ <p>Example: if Raw Input is 6, Cache Read is 60K, and Cache Write is 13K, your turn used ~73K tokens of context \u2014 but 99.99% was efficiently cached, so you only paid the cache-read rate on most of it. The Recent Turns table lets you scan this row by row.</p>
750
+ <p>Anthropic's prompt-caching mechanics are documented at <a href="https://docs.anthropic.com/en/docs/build-with-claude/prompt-caching" target="_blank" rel="noopener noreferrer">docs.anthropic.com/.../prompt-caching</a>.</p>
751
+ </div>
752
+ </details>
753
+
754
+ <details>
755
+ <summary>About the "Savings (floor)" card</summary>
756
+ <div class="faq-body">
757
+ <p>This is the only number on the dashboard that's <strong>estimated</strong> rather than measured. Here's the math:</p>
758
+ <p class="formula-box">savings = blocks \xD7 500 tokens \xD7 $3 per million input rate</p>
759
+ <p>Each time Synthra's PreToolUse hook blocks a <code>Grep</code> or <code>Glob</code> call (because the graph already has high-confidence context), we credit a deliberately conservative 500 tokens at the Sonnet input rate.</p>
760
+ <p>It under-counts on purpose because the formula ignores:</p>
761
+ <ul>
762
+ <li><strong>Cache thrash</strong> \u2014 the blocked tool result would have been written to the cache at ~125% of the input rate, which we don't count</li>
763
+ <li><strong>Cascading reads</strong> \u2014 Claude usually follows a Grep with several <code>Read</code> calls, which the block also prevents but we don't credit</li>
764
+ <li><strong>Bigger codebases</strong> \u2014 actual Grep results often exceed 500 tokens by 3\u20136\xD7 in a real repo</li>
765
+ </ul>
766
+ <p>Real savings are typically <strong>2\u20135\xD7 the floor</strong>. The audit row on the Savings card shows the formula live so you can verify the math.</p>
767
+ </div>
768
+ </details>
769
+
770
+ <details>
771
+ <summary>What is "The Moat"?</summary>
772
+ <div class="faq-body">
773
+ <p>"The Moat" is Synthra's <strong>PreToolUse hook</strong>. Every time Claude Code is about to run a <code>Grep</code> or <code>Glob</code>, the hook calls Synthra's local server first and asks: "do you already have high-confidence context for this query?"</p>
774
+ <p>If yes, Synthra returns <code>{"decision":"block"}</code> \u2014 Claude can't run the tool. Claude then has to use Synthra's graph tools (<code>graph_continue</code>, <code>graph_read</code>) instead, which return the answer without burning tokens on exploration. This is deterministic enforcement \u2014 not prose policy. Claude literally can't disobey.</p>
775
+ <p>The Moat card shows total blocks; the inline list below shows the most recent gate decisions (block vs allow) with the originating query.</p>
776
+ <p>Every decision is logged to <code>.synthra-graph/gate_log.jsonl</code> for the project.</p>
777
+ </div>
778
+ </details>
779
+
780
+ <details>
781
+ <summary>How does Synthra build the codebase graph?</summary>
782
+ <div class="faq-body">
783
+ <p>When you run <code>syn .</code> in a project, Synthra walks the file tree (respecting <code>.gitignore</code> + <code>.synthraignore</code>) and parses each file with <strong>tree-sitter</strong> WebAssembly grammars. Currently 14 languages are supported:</p>
784
+ <p>TypeScript, JavaScript, JSX/TSX, Python, Svelte, Vue, Go, Rust, Java, Kotlin, PHP, Ruby, C/C++, C#, and Dart.</p>
785
+ <p>For each file we extract: function and class definitions, exports, imports, and test-to-source links. The output is a structured graph stored at <code>.synthra-graph/info_graph.json</code> with a symbol index at <code>.synthra-graph/symbol_index.json</code>.</p>
786
+ <p>The graph is what makes pre-injection and The Moat work \u2014 both query it before Claude ever has to Grep.</p>
787
+ </div>
788
+ </details>
789
+
790
+ <details>
791
+ <summary>Where does Synthra store data on disk?</summary>
792
+ <div class="faq-body">
793
+ <p>Synthra uses two folders per project, intentionally separated:</p>
794
+ <table>
795
+ <tr><td><code>.synthra-graph/</code></td><td><strong>Gitignored</strong>. Heavy generated state \u2014 the graph, symbol index, token + gate logs, session info. Rebuilt by <code>syn scan</code>.</td></tr>
796
+ <tr><td><code>.synthra/</code></td><td><strong>Git-tracked.</strong> Decisions, context notes, branch-scoped memory \u2014 the part teammates inherit when they clone the repo.</td></tr>
797
+ </table>
798
+ <p>Plus one global file at <code>~/.synthra/projects.json</code> that tracks every project on this machine.</p>
799
+ </div>
800
+ </details>
801
+
802
+ <details>
803
+ <summary>How does branch-aware memory work?</summary>
804
+ <div class="faq-body">
805
+ <p>Inside <code>.synthra/</code>, context is partitioned by git branch:</p>
806
+ <ul>
807
+ <li><code>.synthra/context-store.json</code> \u2014 decisions on the default branch</li>
808
+ <li><code>.synthra/CONTEXT.md</code> \u2014 narrative notes on the default branch</li>
809
+ <li><code>.synthra/branches/&lt;sanitized-name&gt;/</code> \u2014 overrides on feature branches</li>
810
+ </ul>
811
+ <p>When you switch branches, Synthra's git-watcher (using <code>fs.watch</code> on <code>.git/HEAD</code>) detects the change and reloads the right context. Decisions scoped to a feature branch don't leak back to <code>main</code> until merge.</p>
812
+ </div>
813
+ </details>
814
+
815
+ <details>
816
+ <summary>How does Synthra actually reduce my Claude bill?</summary>
817
+ <div class="faq-body">
818
+ <p>Three mechanisms, in order of impact:</p>
819
+ <ul>
820
+ <li><strong>Pre-injection</strong> \u2014 at session start, Synthra packs ~4K tokens of graph context (function signatures, top inline bodies, file relationships) into Claude's prompt. Claude doesn't have to Grep / Read to discover what's in the codebase \u2014 it already knows.</li>
821
+ <li><strong>The Moat</strong> \u2014 the PreToolUse hook deterministically blocks exploratory Grep/Glob when the graph already has high-confidence context. Counts on the Moat card.</li>
822
+ <li><strong>Branch-aware memory</strong> \u2014 decisions and CONTEXT notes persist in <code>.synthra/</code>, so Claude doesn't have to be re-told what was decided last session.</li>
823
+ </ul>
824
+ <p>The dashboard shows real token counts from Claude's own logs so you can see the effect over time, not just take Synthra's word for it.</p>
825
+ </div>
826
+ </details>
827
+
828
+ <details>
829
+ <summary>Why do long conversations get expensive?</summary>
830
+ <div class="faq-body">
831
+ <p>Each turn re-sends the entire conversation history as input. Even cached, the cumulative input grows roughly <strong>quadratically</strong>:</p>
832
+ <table>
833
+ <thead><tr><td><strong>Turns</strong></td><td><strong>Per-turn input</strong></td><td><strong>Cumulative input</strong></td></tr></thead>
834
+ <tr><td>10</td><td>~2K (mostly cached)</td><td>~110K</td></tr>
835
+ <tr><td>30</td><td>~2K</td><td>~930K</td></tr>
836
+ <tr><td>50</td><td>~2K</td><td>~2.55M</td></tr>
837
+ </table>
838
+ <p>Prompt caching helps a lot (cache reads are 10\xD7 cheaper than fresh input), but context still grows. This is why Synthra's pre-injection matters: starting with the answer already in context means you reach a useful state in fewer turns.</p>
839
+ <p><strong>Tip:</strong> Use <code>/compact</code> in Claude Code or start fresh sessions when a thread feels stale.</p>
840
+ </div>
841
+ </details>
842
+
843
+ <details>
844
+ <summary>Sources &amp; references</summary>
845
+ <div class="faq-body">
846
+ <p>Synthra is open source. Every number on this dashboard can be cross-checked:</p>
847
+ <ul class="link-list">
848
+ <li><a href="https://github.com/jefuriiij/synthra" target="_blank" rel="noopener noreferrer">github.com/jefuriiij/synthra</a> \u2014 source code, issues, roadmap</li>
849
+ <li><a href="https://www.npmjs.com/package/@jefuriiij/synthra" target="_blank" rel="noopener noreferrer">npm: @jefuriiij/synthra</a> \u2014 release history, install instructions</li>
850
+ <li><a href="https://www.anthropic.com/pricing" target="_blank" rel="noopener noreferrer">anthropic.com/pricing</a> \u2014 official rate table Synthra uses</li>
851
+ <li><a href="https://docs.anthropic.com/en/docs/build-with-claude/prompt-caching" target="_blank" rel="noopener noreferrer">Anthropic docs \xB7 prompt caching</a> \u2014 explains cache read/write behavior</li>
852
+ <li><code>src/shared/pricing.ts</code> \u2014 the file in this repo holding the rate table</li>
853
+ <li><code>src/dashboard/delta.ts</code> \u2014 where dashboard aggregates are computed</li>
854
+ <li><code>src/server/routes/gate.ts</code> \u2014 the Moat implementation</li>
855
+ </ul>
856
+ </div>
857
+ </details>
858
+
859
+ </div>
860
+ </div>
861
+ </div>
862
+
863
+ <!-- ============ Footer ============ -->
864
+ <footer class="foot">
865
+ <div>Synth<em>ra</em> \xB7 v0.1.3</div>
866
+ <div>Cost figures approximate \xB7 @jefuriiij</div>
523
867
  </footer>
524
868
 
525
869
  <script>
526
- // Inline SVG icons. Each is a stroke-based icon, currentColor for the
527
- // stroke, designed to inherit color from the parent's CSS.
528
- const ICONS = {
529
- dollar: '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><line x1="12" y1="1" x2="12" y2="23"/><path d="M17 5H9.5a3.5 3.5 0 0 0 0 7h5a3.5 3.5 0 0 1 0 7H6"/></svg>',
530
- chat: '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"/></svg>',
531
- arrowDown: '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><line x1="12" y1="5" x2="12" y2="19"/><polyline points="19 12 12 19 5 12"/></svg>',
532
- arrowUp: '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><line x1="12" y1="19" x2="12" y2="5"/><polyline points="5 12 12 5 19 12"/></svg>',
533
- refresh: '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="23 4 23 10 17 10"/><polyline points="1 20 1 14 7 14"/><path d="M3.51 9a9 9 0 0 1 14.85-3.36L23 10M1 14l4.64 4.36A9 9 0 0 0 20.49 15"/></svg>',
534
- save: '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M19 21H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h11l5 5v11a2 2 0 0 1-2 2z"/><polyline points="17 21 17 13 7 13 7 21"/><polyline points="7 3 7 8 15 8"/></svg>',
535
- folder: '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M22 19a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h5l2 3h9a2 2 0 0 1 2 2z"/></svg>',
536
- shield: '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M12 22s8-4 8-10V5l-8-3-8 3v7c0 6 8 10 8 10z"/></svg>',
537
- trending: '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="23 6 13.5 15.5 8.5 10.5 1 18"/><polyline points="17 6 23 6 23 12"/></svg>',
538
- ban: '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><circle cx="12" cy="12" r="10"/><line x1="4.93" y1="4.93" x2="19.07" y2="19.07"/></svg>',
539
- };
870
+ const $ = (sel) => document.querySelector(sel);
871
+ const SAVED_RATE_PER_M = 3.00; // USD per million tokens \u2014 conservative input rate
540
872
 
541
- // Classify a model name into a family for color-coding.
873
+ // ----- model classification -----
542
874
  function modelFamily(model) {
543
875
  if (!model) return 'unknown';
544
876
  const m = model.toLowerCase();
@@ -548,863 +880,403 @@ var public_default = `<!doctype html>
548
880
  if (m.includes('haiku')) return 'haiku';
549
881
  return 'unknown';
550
882
  }
551
-
552
883
  function modelLabel(model) {
553
- if (!model || model === '<synthetic>') return model === '<synthetic>' ? 'synthetic' : 'unknown';
554
- return model;
884
+ if (!model || model === '<synthetic>') return 'synthetic';
885
+ return model.replace(/^claude-/, '');
555
886
  }
556
887
 
557
- const $ = (sel) => document.querySelector(sel);
558
- const cardsEl = $("#cards");
559
- const projectsEl = $("#projects");
560
- const turnsBody = $("#turns tbody");
561
- const gatesBody = $("#gates tbody");
562
- const turnsEmpty = $("#turns-empty");
563
- const gatesEmpty = $("#gates-empty");
564
- const projectsEmpty = $("#projects-empty");
565
- const statusEl = $("#status");
566
- const dotEl = $("#dot");
567
- const activeProjectEl = $("#active-project");
568
-
888
+ // ----- formatting -----
569
889
  function fmt(n) {
570
- if (typeof n !== "number" || !Number.isFinite(n)) return "0";
571
- if (n >= 1_000_000) return (n / 1_000_000).toFixed(1) + "M";
572
- if (n >= 1000) return (n / 1000).toFixed(1) + "K";
890
+ if (typeof n !== 'number' || !Number.isFinite(n)) return '0';
891
+ if (n >= 1_000_000) return (n / 1_000_000).toFixed(1) + '<em>M</em>';
892
+ if (n >= 1_000) return (n / 1_000).toFixed(1) + '<em>k</em>';
573
893
  return n.toLocaleString();
574
894
  }
575
-
576
- function fmtFull(n) {
577
- return (typeof n === "number" ? n : 0).toLocaleString();
895
+ function fmtPlain(n) {
896
+ if (typeof n !== 'number' || !Number.isFinite(n)) return '0';
897
+ if (n >= 1_000_000) return (n / 1_000_000).toFixed(1) + 'M';
898
+ if (n >= 1_000) return (n / 1_000).toFixed(1) + 'k';
899
+ return n.toLocaleString();
578
900
  }
579
-
580
- function fmtCost(usd) {
581
- if (typeof usd !== "number") return "~$0.00";
582
- if (usd >= 1) return "~$" + usd.toFixed(2);
583
- if (usd >= 0.01) return "~$" + usd.toFixed(3);
584
- return "~$" + usd.toFixed(4);
901
+ function fmtCostBig(usd) {
902
+ if (typeof usd !== 'number' || !Number.isFinite(usd)) usd = 0;
903
+ let s;
904
+ if (usd >= 1) s = usd.toFixed(2);
905
+ else if (usd >= 0.01) s = usd.toFixed(3);
906
+ else s = usd.toFixed(4);
907
+ const dot = s.indexOf('.');
908
+ if (dot === -1) return '$' + s;
909
+ return '$' + s.slice(0, dot) + '.<em>' + s.slice(dot + 1) + '</em>';
910
+ }
911
+ function fmtCostFlat(usd) {
912
+ if (typeof usd !== 'number' || !Number.isFinite(usd)) usd = 0;
913
+ if (usd >= 1) return '$' + usd.toFixed(2);
914
+ if (usd >= 0.01) return '$' + usd.toFixed(3);
915
+ return '$' + usd.toFixed(4);
585
916
  }
586
-
587
917
  function fmtTs(iso) {
588
918
  try {
589
919
  const d = new Date(iso);
590
920
  const today = new Date();
591
921
  const isToday = d.toDateString() === today.toDateString();
592
- if (isToday) return d.toLocaleTimeString();
593
- return d.toLocaleString();
594
- } catch {
595
- return iso;
596
- }
922
+ if (isToday) return d.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' });
923
+ return d.toLocaleDateString([], { month: 'short', day: 'numeric' });
924
+ } catch { return iso; }
597
925
  }
598
926
 
599
- // Definitions for the global-totals cards. Each: label, value-source key,
600
- // icon, optional class (accent | money), tooltip text.
601
- function cardConfigs(g) {
602
- return [
603
- {
604
- label: "Total cost",
605
- value: fmtCost(g.estimated_cost_usd),
606
- icon: ICONS.dollar,
607
- cls: "money",
608
- tooltip: "Approximate USD cost across all projects. Computed from per-model pricing (Opus, Sonnet, Haiku) applied to each turn's input/output/cache. Tilde everywhere \u2014 real cost depends on Anthropic's current rates.",
609
- },
610
- {
611
- label: "Turns",
612
- value: fmt(g.total_turns),
613
- icon: ICONS.chat,
614
- tooltip: "Total number of back-and-forth exchanges with Claude across all projects. One turn = you send a message, Claude responds.",
615
- },
616
- {
617
- label: "Input",
618
- value: fmt(g.total_input_tokens),
619
- icon: ICONS.arrowDown,
620
- tooltip: "New (uncached) tokens sent to Claude across all turns. Usually small \u2014 most of the conversation comes from cache.",
621
- },
622
- {
623
- label: "Output",
624
- value: fmt(g.total_output_tokens),
625
- icon: ICONS.arrowUp,
626
- tooltip: "Tokens Claude generated in responses. The most expensive line item per turn (~5\xD7 input rate on Opus).",
627
- },
628
- {
629
- label: "Cache read",
630
- value: fmt(g.total_cache_read),
631
- icon: ICONS.refresh,
632
- tooltip: "Tokens read from Claude's prompt cache \u2014 conversation history, system prompt, Synthra's pack. Cheap: ~10% of the input rate. The bulk of every long session.",
633
- },
634
- {
635
- label: "Cache write",
636
- value: fmt(g.total_cache_create),
637
- icon: ICONS.save,
638
- tooltip: "Tokens newly added to the prompt cache so future turns can read them cheaply. Premium-priced (~125% of input) but pays back over the rest of the session.",
639
- },
640
- {
641
- label: "Projects",
642
- value: fmt(g.project_count),
643
- icon: ICONS.folder,
644
- tooltip: "Projects that have ever run \`syn .\` on this machine. Tracked in ~/.synthra/projects.json.",
645
- },
646
- {
647
- label: "Blocked Grep / Glob",
648
- value: fmt(g.blocked_count),
649
- icon: ICONS.shield,
650
- cls: "accent",
651
- tooltip: "PreToolUse hook intercepts: Synthra blocked these Grep/Glob calls because the graph already had high-confidence context for the query. Claude pivots to graph_continue or graph_read instead.",
652
- },
653
- {
654
- label: "Tokens saved",
655
- value: fmt(g.estimated_tokens_saved),
656
- icon: ICONS.trending,
657
- cls: "accent",
658
- tooltip: "Estimated tokens avoided by blocking exploratory Grep/Glob calls. Calculated as blocks \xD7 500 \u2014 conservative; under-counts the cache thrash you also avoid.",
659
- },
660
- ];
927
+ let lastData = null;
928
+
929
+ // ----- date in hero -----
930
+ function setHeroDate() {
931
+ const now = new Date();
932
+ $('#hero-day').textContent = now.getDate();
933
+ $('#hero-weekday').textContent = now.toLocaleDateString([], { weekday: 'short' });
934
+ $('#hero-month').textContent = now.toLocaleDateString([], { month: 'long' });
661
935
  }
662
936
 
663
- function renderCards(g) {
664
- cardsEl.innerHTML = "";
665
- for (const c of cardConfigs(g)) {
666
- const el = document.createElement("div");
667
- el.className = "card" + (c.cls ? " " + c.cls : "");
668
- el.innerHTML =
669
- '<div class="card-head">' +
670
- '<div class="card-label has-tooltip" data-tooltip="' + c.tooltip.replace(/"/g, '&quot;') + '">' +
671
- c.label +
672
- ' <span class="help-icon">i</span>' +
673
- '</div>' +
674
- '<span class="card-icon">' + c.icon + '</span>' +
675
- '</div>' +
676
- '<div class="card-value">' + c.value + '</div>';
677
- cardsEl.appendChild(el);
678
- }
937
+ // ----- renderers -----
938
+ function renderSession(turns) {
939
+ const agg = turns.reduce((a, t) => {
940
+ a.turns += 1;
941
+ a.input += t.input || 0;
942
+ a.output += t.output || 0;
943
+ a.cacheR += t.cache_read || 0;
944
+ a.cacheW += t.cache_create || 0;
945
+ return a;
946
+ }, { turns: 0, input: 0, output: 0, cacheR: 0, cacheW: 0 });
947
+
948
+ $('#m-turns').innerHTML = fmt(agg.turns);
949
+ $('#m-input').innerHTML = fmt(agg.input);
950
+ $('#m-output').innerHTML = fmt(agg.output);
951
+ $('#m-cache-r').innerHTML = fmt(agg.cacheR);
952
+ $('#m-cache-w').innerHTML = fmt(agg.cacheW);
679
953
  }
680
954
 
681
- function renderProjects(projects, globalCost) {
682
- projectsEl.innerHTML = "";
683
- if (!projects.length) {
684
- projectsEmpty.classList.remove("hidden");
685
- return;
686
- }
687
- projectsEmpty.classList.add("hidden");
688
- const maxTokens = Math.max(1, ...projects.map((p) => p.total_input_tokens + p.total_output_tokens));
689
- for (const p of projects) {
690
- const total = p.total_input_tokens + p.total_output_tokens;
691
- const pct = Math.round((total / maxTokens) * 100);
692
- const sharePct = globalCost > 0 ? ((p.estimated_cost_usd / globalCost) * 100).toFixed(1) : "0.0";
693
- const row = document.createElement("div");
694
- row.className = "project-row";
695
- row.innerHTML =
696
- '<div class="project-name">' +
697
- '<strong>' + ICONS.folder + p.name + '</strong>' +
698
- '<code class="project-path">' + p.path + '</code>' +
699
- '</div>' +
700
- '<div class="project-stats">' +
701
- '<div class="stat"><span class="stat-value cost">' + fmtCost(p.estimated_cost_usd) + '</span><span class="stat-label">cost (' + sharePct + '%)</span></div>' +
702
- '<div class="stat"><span class="stat-value">' + fmt(total) + '</span><span class="stat-label">tokens</span></div>' +
703
- '<div class="stat"><span class="stat-value">' + fmt(p.total_turns) + '</span><span class="stat-label">turns</span></div>' +
704
- '<div class="stat"><span class="stat-value">' + fmt(p.blocked_count) + '</span><span class="stat-label">blocks</span></div>' +
705
- '</div>' +
706
- '<div class="bar"><div class="bar-fill" style="width:' + pct + '%"></div></div>';
707
- projectsEl.appendChild(row);
708
- }
955
+ function renderSavings(g) {
956
+ const tokensSaved = g.estimated_tokens_saved || 0;
957
+ const blocks = g.blocked_count || 0;
958
+ const savedUsd = tokensSaved * SAVED_RATE_PER_M / 1_000_000;
959
+ const actualUsd = g.estimated_cost_usd || 0;
960
+ const baselineUsd = actualUsd + savedUsd;
961
+ const savedPct = baselineUsd > 0 ? (savedUsd / baselineUsd) * 100 : 0;
962
+ const actualPct = baselineUsd > 0 ? (actualUsd / baselineUsd) * 100 : 100;
963
+
964
+ $('#savings-money').textContent = fmtCostFlat(savedUsd);
965
+ $('#savings-pct').textContent = savedPct.toFixed(1) + '% off';
966
+ $('#savings-tokens').textContent = fmtPlain(tokensSaved);
967
+ $('#savings-actual-bar').style.width = actualPct.toFixed(2) + '%';
968
+ $('#savings-saved-bar').style.width = savedPct.toFixed(2) + '%';
969
+ $('#savings-actual-amt').textContent = fmtCostFlat(actualUsd);
970
+ $('#savings-baseline-amt').textContent = fmtCostFlat(baselineUsd);
971
+
972
+ // Audit row \u2014 live formula
973
+ $('#audit-blocks').textContent = blocks.toLocaleString();
974
+ $('#audit-result').textContent = fmtCostFlat(savedUsd);
975
+ }
976
+
977
+ function renderCostHero(g) {
978
+ $('#big-cost').innerHTML = fmtCostBig(g.estimated_cost_usd);
979
+ const totalTokens = (g.total_input_tokens || 0) + (g.total_output_tokens || 0);
980
+ $('#cs-tokens').textContent = fmtPlain(totalTokens);
981
+ const avg = g.total_turns > 0 ? g.estimated_cost_usd / g.total_turns : 0;
982
+ $('#cs-avg').textContent = fmtCostFlat(avg);
983
+ }
984
+
985
+ function renderMoat(g) {
986
+ $('#blocks').textContent = fmtPlain(g.blocked_count);
709
987
  }
710
988
 
711
989
  function renderTurns(turns) {
712
- turnsBody.innerHTML = "";
990
+ const tbody = $('#turns-body');
991
+ const empty = $('#turns-empty');
992
+ tbody.innerHTML = '';
713
993
  if (!turns.length) {
714
- turnsEmpty.classList.remove("hidden");
994
+ empty.classList.remove('hidden');
995
+ $('#turns-count').textContent = '0 shown';
715
996
  return;
716
997
  }
717
- turnsEmpty.classList.add("hidden");
998
+ empty.classList.add('hidden');
999
+ $('#turns-count').textContent = turns.length + ' shown';
1000
+ const frag = document.createDocumentFragment();
718
1001
  for (const t of turns) {
719
1002
  const family = modelFamily(t.model);
720
- const tr = document.createElement("tr");
1003
+ const tr = document.createElement('tr');
721
1004
  tr.innerHTML =
722
- "<td>" + fmtTs(t.ts) + "</td>" +
723
- "<td><code>" + t.project_name + "</code></td>" +
724
- '<td><span class="model-pill ' + family + '">' + modelLabel(t.model) + "</span></td>" +
725
- '<td class="num">' + fmtFull(t.input) + "</td>" +
726
- '<td class="num">' + fmtFull(t.output) + "</td>" +
727
- '<td class="num">' + fmt(t.cache_read) + " / " + fmt(t.cache_create) + "</td>" +
728
- '<td class="num cost">' + fmtCost(t.cost_usd) + "</td>";
729
- turnsBody.appendChild(tr);
1005
+ '<td class="ts">' + fmtTs(t.ts || t.written_at) + '</td>' +
1006
+ '<td class="proj">' + (t.project_name || '\u2014') + '</td>' +
1007
+ '<td><span class="model-pill ' + family + '"><span class="sq"></span>' + modelLabel(t.model) + '</span></td>' +
1008
+ '<td class="num">' + fmtPlain(t.input || 0) + '</td>' +
1009
+ '<td class="num">' + fmtPlain(t.output || 0) + '</td>' +
1010
+ '<td class="num">' + fmtPlain(t.cache_read || 0) + ' / ' + fmtPlain(t.cache_create || 0) + '</td>' +
1011
+ '<td class="num cost">' + fmtCostFlat(t.cost_usd || 0) + '</td>';
1012
+ frag.appendChild(tr);
730
1013
  }
1014
+ tbody.appendChild(frag);
731
1015
  }
732
1016
 
733
- function renderGates(gates) {
734
- gatesBody.innerHTML = "";
1017
+ function renderGateMini(gates) {
1018
+ const el = $('#gate-mini');
1019
+ el.innerHTML = '';
735
1020
  if (!gates.length) {
736
- gatesEmpty.classList.remove("hidden");
1021
+ el.innerHTML = '<div class="empty">No gate decisions yet.</div>';
737
1022
  return;
738
1023
  }
739
- gatesEmpty.classList.add("hidden");
740
- for (const g of gates) {
741
- const tr = document.createElement("tr");
742
- const isBlock = g.decision === "block";
743
- const cls = isBlock ? "decision-block" : "decision-allow";
744
- const label = isBlock
745
- ? '<span class="' + cls + '">' + ICONS.ban + ' BLOCK</span>'
746
- : '<span class="' + cls + '">ALLOW</span>';
747
- tr.innerHTML =
748
- "<td>" + fmtTs(g.ts) + "</td>" +
749
- "<td><code>" + g.project_name + "</code></td>" +
750
- "<td><code>" + g.tool + "</code></td>" +
751
- "<td>" + label + "</td>" +
752
- "<td><code>" + (g.query || "") + "</code></td>";
753
- gatesBody.appendChild(tr);
1024
+ const frag = document.createDocumentFragment();
1025
+ for (const g of gates.slice(0, 12)) {
1026
+ const row = document.createElement('div');
1027
+ row.className = 'gate-row';
1028
+ const cls = g.decision === 'block' ? 'block' : 'allow';
1029
+ row.innerHTML =
1030
+ '<span class="g-ts">' + fmtTs(g.ts) + '</span>' +
1031
+ '<span class="g-decision ' + cls + '">' + (g.decision || '\u2014').toUpperCase() + '</span>' +
1032
+ '<span class="g-q">' + (g.query || g.tool || '\u2014') + '</span>';
1033
+ frag.appendChild(row);
754
1034
  }
1035
+ el.appendChild(frag);
755
1036
  }
756
1037
 
757
- async function tick() {
758
- try {
759
- const res = await fetch("/data");
760
- if (!res.ok) throw new Error("HTTP " + res.status);
761
- const data = await res.json();
762
- activeProjectEl.textContent = data.active.project_root;
763
- renderCards(data.global);
764
- renderProjects(data.projects, data.global.estimated_cost_usd);
765
- renderTurns(data.recent_turns);
766
- renderGates(data.recent_gates);
767
- statusEl.textContent = "live \xB7 " + new Date().toLocaleTimeString();
768
- dotEl.classList.add("live");
769
- dotEl.classList.remove("dead");
770
- } catch (e) {
771
- statusEl.textContent = "disconnected \xB7 " + e.message;
772
- dotEl.classList.add("dead");
773
- dotEl.classList.remove("live");
1038
+ function renderProjectLegend(projects) {
1039
+ const el = $('#proj-filter');
1040
+ const incoming = projects.map((p) => p.name).join('|');
1041
+ if (el.dataset.signature === incoming) return;
1042
+ el.dataset.signature = incoming;
1043
+ el.innerHTML = '';
1044
+ if (!projects.length) {
1045
+ el.innerHTML = '<div class="empty">No projects yet.</div>';
1046
+ return;
1047
+ }
1048
+ const frag = document.createDocumentFragment();
1049
+ for (const p of projects) {
1050
+ const row = document.createElement('button');
1051
+ row.className = 'check check-clickable';
1052
+ row.type = 'button';
1053
+ row.innerHTML =
1054
+ '<span class="dot-sq" style="background: var(--sky); opacity:.55"></span>' +
1055
+ '<span class="pf-name" title="' + p.path + '">' + p.name + '</span>' +
1056
+ '<span class="pf-arrow">\u203A</span>';
1057
+ row.addEventListener('click', () => openProjectDialog(p.name));
1058
+ frag.appendChild(row);
774
1059
  }
1060
+ el.appendChild(frag);
775
1061
  }
776
1062
 
777
- tick();
778
- setInterval(tick, 2000);
779
- </script>
780
- </body>
781
- </html>
782
- `;
783
-
784
- // src/dashboard/public/style.css
785
- var style_default = `/* Synthra token dashboard \u2014 palette per project brief.
786
- Base: cream-on-near-black, dark-red borders, hot-pink highlights.
787
- Plus: money green for dollar amounts, per-family model colors,
788
- subtle dot grid + top glow background, tooltip system, icons. */
789
-
790
- :root {
791
- /* Brand */
792
- --color-heading: #ECEBD8;
793
- --color-body: #EDECD9;
794
- --color-bg: #000000;
795
- --color-surface: #140009;
796
- --color-surface-raised: #1E000D;
797
- --color-border: #4D0020;
798
- --color-accent: #FF0073;
799
- --color-accent-darker: #EB006A;
800
- --color-form-bg: #2E0014;
801
- --color-muted: rgba(237, 236, 217, 0.55);
802
- --color-very-muted: rgba(237, 236, 217, 0.35);
803
- --color-block: #FF0073;
804
- --color-allow: #ECEBD8;
805
-
806
- /* Money green \u2014 used everywhere a $ amount appears */
807
- --color-money: #36E596;
808
- --color-money-darker: #00B85F;
809
- --color-money-bg: rgba(54, 229, 150, 0.08);
810
-
811
- /* Per-family model colors */
812
- --color-model-opus: #C9A2FF;
813
- --color-model-sonnet: #6BD0FF;
814
- --color-model-haiku: #7BFFC7;
815
-
816
- /* Background layering */
817
- --bg-dot-color: rgba(237, 236, 217, 0.045);
818
- --bg-glow-color: rgba(255, 0, 115, 0.07);
819
- }
820
-
821
- * {
822
- box-sizing: border-box;
823
- margin: 0;
824
- padding: 0;
825
- }
826
-
827
- html, body {
828
- background-color: var(--color-bg);
829
- color: var(--color-body);
830
- font-family:
831
- ui-sans-serif, -apple-system, BlinkMacSystemFont, "Segoe UI",
832
- system-ui, sans-serif;
833
- font-size: 14px;
834
- line-height: 1.5;
835
- min-height: 100vh;
836
- }
837
-
838
- /* Body becomes a flex column so main can grow + header/footer stay sticky
839
- at the natural top/bottom of the viewport. */
840
- body {
841
- display: flex;
842
- flex-direction: column;
843
- min-height: 100vh;
844
- }
845
-
846
- main {
847
- flex: 1;
848
- }
849
-
850
- /* Layered backdrop: dot grid + soft pink glow at the top.
851
- Both are fixed so they don't repeat as you scroll. */
852
- body {
853
- background-image:
854
- radial-gradient(ellipse 70% 40% at 50% 0%, var(--bg-glow-color), transparent 70%),
855
- radial-gradient(circle at 1px 1px, var(--bg-dot-color) 1px, transparent 0);
856
- background-size: 100% 100%, 22px 22px;
857
- background-attachment: fixed;
858
- }
859
-
860
- code, .num, table, .project-path {
861
- font-family: ui-monospace, "SF Mono", Menlo, Consolas, monospace;
862
- }
863
-
864
- header {
865
- display: flex;
866
- align-items: center;
867
- gap: 1.5rem;
868
- padding: 1.1rem 2rem;
869
- background: linear-gradient(180deg, rgba(20, 0, 9, 0.7), rgba(20, 0, 9, 0.4));
870
- backdrop-filter: blur(8px);
871
- border-bottom: 1px solid var(--color-border);
872
- position: sticky;
873
- top: 0;
874
- z-index: 5;
875
- }
876
-
877
- header .brand {
878
- display: flex;
879
- align-items: baseline;
880
- gap: 0.75rem;
881
- }
882
-
883
- header h1 {
884
- color: var(--color-heading);
885
- font-size: 1.35rem;
886
- font-weight: 700;
887
- letter-spacing: 0.02em;
888
- }
889
-
890
- header .tag {
891
- color: var(--color-muted);
892
- font-size: 0.78rem;
893
- text-transform: uppercase;
894
- letter-spacing: 0.1em;
895
- }
896
-
897
- header .meta {
898
- margin-left: auto;
899
- display: flex;
900
- align-items: center;
901
- gap: 0.75rem;
902
- font-size: 0.78rem;
903
- font-family: ui-monospace, monospace;
904
- color: var(--color-muted);
905
- }
906
-
907
- header .active-project {
908
- max-width: 480px;
909
- white-space: nowrap;
910
- overflow: hidden;
911
- text-overflow: ellipsis;
912
- }
913
-
914
- main {
915
- padding: 2rem;
916
- display: flex;
917
- flex-direction: column;
918
- gap: 2rem;
919
- max-width: 1400px;
920
- margin: 0 auto;
921
- width: 100%;
922
- }
923
-
924
- section {
925
- display: flex;
926
- flex-direction: column;
927
- gap: 0.85rem;
928
- }
929
-
930
- h2 {
931
- color: var(--color-heading);
932
- font-size: 0.85rem;
933
- font-weight: 600;
934
- letter-spacing: 0.08em;
935
- text-transform: uppercase;
936
- display: flex;
937
- align-items: center;
938
- gap: 0.5rem;
939
- }
940
-
941
- h2 svg {
942
- width: 14px;
943
- height: 14px;
944
- opacity: 0.5;
945
- }
946
-
947
- h2 .muted {
948
- color: var(--color-very-muted);
949
- font-size: 0.78rem;
950
- font-weight: 400;
951
- letter-spacing: 0.06em;
952
- text-transform: none;
953
- margin-left: 0.5rem;
954
- }
955
-
956
- /* ============================================================
957
- Cards row (Global totals)
958
- ============================================================ */
959
-
960
- .cards {
961
- display: grid;
962
- grid-template-columns: repeat(auto-fill, minmax(170px, 1fr));
963
- gap: 0.7rem;
964
- }
965
-
966
- .card {
967
- position: relative;
968
- background: var(--color-surface);
969
- border: 1px solid var(--color-border);
970
- border-radius: 8px;
971
- padding: 0.95rem 1rem 0.85rem;
972
- transition: transform 120ms ease, border-color 120ms ease;
973
- }
974
-
975
- .card:hover {
976
- transform: translateY(-1px);
977
- border-color: rgba(77, 0, 32, 0.85);
978
- }
979
-
980
- .card.accent {
981
- background: var(--color-surface-raised);
982
- border-color: var(--color-accent);
983
- }
984
-
985
- .card.money {
986
- background: linear-gradient(180deg, var(--color-money-bg), transparent 80%), var(--color-surface);
987
- border-color: var(--color-money-darker);
988
- }
989
-
990
- .card.money:hover {
991
- border-color: var(--color-money);
992
- }
993
-
994
- .card-head {
995
- display: flex;
996
- align-items: center;
997
- justify-content: space-between;
998
- gap: 0.5rem;
999
- margin-bottom: 0.4rem;
1000
- }
1001
-
1002
- .card-label {
1003
- color: var(--color-muted);
1004
- font-size: 0.66rem;
1005
- text-transform: uppercase;
1006
- letter-spacing: 0.09em;
1007
- display: flex;
1008
- align-items: center;
1009
- gap: 0.35rem;
1010
- }
1011
-
1012
- .card-icon {
1013
- width: 14px;
1014
- height: 14px;
1015
- color: var(--color-muted);
1016
- opacity: 0.65;
1017
- flex-shrink: 0;
1018
- }
1019
-
1020
- .card.accent .card-icon { color: var(--color-accent); opacity: 0.85; }
1021
- .card.money .card-icon { color: var(--color-money); opacity: 0.85; }
1022
-
1023
- .card-value {
1024
- color: var(--color-heading);
1025
- font-family: ui-monospace, monospace;
1026
- font-size: 1.5rem;
1027
- font-weight: 600;
1028
- letter-spacing: -0.01em;
1029
- }
1030
-
1031
- .card.accent .card-value { color: var(--color-accent); }
1032
- .card.money .card-value { color: var(--color-money); }
1033
-
1034
- /* ============================================================
1035
- Tooltip (data-tooltip on .has-tooltip)
1036
- ============================================================ */
1037
-
1038
- .has-tooltip {
1039
- position: relative;
1040
- cursor: help;
1041
- }
1042
-
1043
- .has-tooltip::before,
1044
- .has-tooltip::after {
1045
- position: absolute;
1046
- pointer-events: none;
1047
- opacity: 0;
1048
- transition: opacity 150ms ease, transform 150ms ease;
1049
- z-index: 100;
1050
- }
1051
-
1052
- .has-tooltip::after {
1053
- content: attr(data-tooltip);
1054
- bottom: calc(100% + 10px);
1055
- left: 50%;
1056
- transform: translate(-50%, 4px);
1057
- background: var(--color-surface-raised);
1058
- color: var(--color-body);
1059
- border: 1px solid var(--color-border);
1060
- border-radius: 6px;
1061
- padding: 0.6rem 0.75rem;
1062
- font-size: 0.74rem;
1063
- font-weight: 400;
1064
- text-transform: none;
1065
- letter-spacing: 0;
1066
- white-space: normal;
1067
- width: 260px;
1068
- text-align: left;
1069
- line-height: 1.45;
1070
- box-shadow: 0 8px 24px rgba(0, 0, 0, 0.7);
1071
- }
1072
-
1073
- .has-tooltip::before {
1074
- content: "";
1075
- bottom: calc(100% + 4px);
1076
- left: 50%;
1077
- transform: translate(-50%, 4px);
1078
- border: 6px solid transparent;
1079
- border-top-color: var(--color-border);
1080
- }
1081
-
1082
- .has-tooltip:hover::after,
1083
- .has-tooltip:hover::before {
1084
- opacity: 1;
1085
- transform: translate(-50%, 0);
1086
- }
1087
-
1088
- /* Small \u24D8 icon used to indicate tooltips */
1089
- .help-icon {
1090
- display: inline-flex;
1091
- align-items: center;
1092
- justify-content: center;
1093
- width: 13px;
1094
- height: 13px;
1095
- border-radius: 50%;
1096
- border: 1px solid var(--color-muted);
1097
- color: var(--color-muted);
1098
- font-size: 9px;
1099
- font-weight: 600;
1100
- font-family: ui-sans-serif, sans-serif;
1101
- line-height: 1;
1102
- cursor: help;
1103
- user-select: none;
1104
- transition: color 120ms, border-color 120ms;
1105
- }
1106
-
1107
- .has-tooltip:hover .help-icon {
1108
- border-color: var(--color-accent);
1109
- color: var(--color-accent);
1110
- }
1111
-
1112
- .card.money .has-tooltip:hover .help-icon { border-color: var(--color-money); color: var(--color-money); }
1113
-
1114
- /* ============================================================
1115
- Projects list
1116
- ============================================================ */
1117
-
1118
- .projects {
1119
- display: flex;
1120
- flex-direction: column;
1121
- gap: 0.6rem;
1122
- }
1123
-
1124
- .project-row {
1125
- display: grid;
1126
- grid-template-columns: minmax(220px, 1fr) auto;
1127
- grid-template-rows: auto auto;
1128
- gap: 0.6rem 1.25rem;
1129
- align-items: center;
1130
- background: var(--color-surface);
1131
- border: 1px solid var(--color-border);
1132
- border-radius: 8px;
1133
- padding: 0.9rem 1.2rem;
1134
- transition: background 120ms ease;
1135
- }
1136
-
1137
- .project-row:hover {
1138
- background: var(--color-surface-raised);
1139
- }
1140
-
1141
- .project-name {
1142
- display: flex;
1143
- flex-direction: column;
1144
- gap: 0.15rem;
1145
- overflow: hidden;
1146
- }
1147
-
1148
- .project-name strong {
1149
- color: var(--color-heading);
1150
- font-size: 0.95rem;
1151
- font-weight: 600;
1152
- display: flex;
1153
- align-items: center;
1154
- gap: 0.4rem;
1155
- }
1156
-
1157
- .project-name strong svg {
1158
- width: 13px;
1159
- height: 13px;
1160
- opacity: 0.65;
1161
- color: var(--color-muted);
1162
- flex-shrink: 0;
1163
- }
1164
-
1165
- .project-name .project-path {
1166
- color: var(--color-very-muted);
1167
- font-size: 0.72rem;
1168
- white-space: nowrap;
1169
- overflow: hidden;
1170
- text-overflow: ellipsis;
1171
- }
1172
-
1173
- .project-stats {
1174
- display: flex;
1175
- gap: 1.6rem;
1176
- justify-self: end;
1177
- text-align: right;
1178
- }
1179
-
1180
- .stat {
1181
- display: flex;
1182
- flex-direction: column;
1183
- gap: 0.15rem;
1184
- min-width: 70px;
1185
- }
1186
-
1187
- .stat-value {
1188
- color: var(--color-heading);
1189
- font-family: ui-monospace, monospace;
1190
- font-size: 0.95rem;
1191
- font-weight: 600;
1192
- }
1193
-
1194
- .stat-value.cost {
1195
- color: var(--color-money);
1196
- }
1197
-
1198
- .stat-label {
1199
- color: var(--color-muted);
1200
- font-size: 0.66rem;
1201
- text-transform: uppercase;
1202
- letter-spacing: 0.08em;
1203
- }
1204
-
1205
- .bar {
1206
- grid-column: 1 / -1;
1207
- height: 3px;
1208
- background: var(--color-form-bg);
1209
- border-radius: 2px;
1210
- overflow: hidden;
1211
- }
1212
-
1213
- .bar-fill {
1214
- height: 100%;
1215
- background: linear-gradient(90deg, var(--color-accent-darker), var(--color-accent));
1216
- border-radius: 2px;
1217
- transition: width 400ms ease;
1218
- }
1219
-
1220
- /* ============================================================
1221
- Tables
1222
- ============================================================ */
1223
-
1224
- table {
1225
- width: 100%;
1226
- border-collapse: collapse;
1227
- font-size: 0.83rem;
1228
- background: var(--color-surface);
1229
- border: 1px solid var(--color-border);
1230
- border-radius: 8px;
1231
- overflow: hidden;
1232
- }
1233
-
1234
- table thead th {
1235
- text-align: left;
1236
- color: var(--color-muted);
1237
- text-transform: uppercase;
1238
- font-size: 0.66rem;
1239
- letter-spacing: 0.08em;
1240
- font-weight: 600;
1241
- padding: 0.65rem 0.85rem;
1242
- border-bottom: 1px solid var(--color-border);
1243
- background: var(--color-surface);
1244
- }
1245
-
1246
- table thead th.num {
1247
- text-align: right;
1248
- }
1249
-
1250
- table thead th .has-tooltip {
1251
- display: inline-flex;
1252
- align-items: center;
1253
- gap: 0.3rem;
1254
- }
1255
-
1256
- table tbody td {
1257
- padding: 0.55rem 0.85rem;
1258
- border-bottom: 1px solid rgba(77, 0, 32, 0.4);
1259
- color: var(--color-body);
1260
- }
1261
-
1262
- table tbody td.num {
1263
- text-align: right;
1264
- }
1265
-
1266
- table tbody td.cost {
1267
- color: var(--color-money);
1268
- font-weight: 600;
1269
- }
1270
-
1271
- table tbody tr:last-child td {
1272
- border-bottom: none;
1273
- }
1274
-
1275
- table tbody tr:hover {
1276
- background: var(--color-surface-raised);
1277
- }
1278
-
1279
- /* ============================================================
1280
- Model pills \u2014 color-coded by family
1281
- ============================================================ */
1282
-
1283
- .model-pill {
1284
- display: inline-block;
1285
- padding: 0.1rem 0.5rem;
1286
- border-radius: 3px;
1287
- background: var(--color-form-bg);
1288
- font-family: ui-monospace, monospace;
1289
- font-size: 0.82em;
1290
- border: 1px solid transparent;
1291
- color: var(--color-body);
1292
- white-space: nowrap;
1293
- }
1294
-
1295
- .model-pill.opus {
1296
- color: var(--color-model-opus);
1297
- border-color: rgba(201, 162, 255, 0.3);
1298
- background: rgba(201, 162, 255, 0.08);
1299
- }
1063
+ // ----- Project dialog -----
1064
+ function lastActiveFor(name) {
1065
+ const turns = lastData?.recent_turns || [];
1066
+ for (const t of turns) {
1067
+ if (t.project_name === name) {
1068
+ const ts = t.ts || t.written_at;
1069
+ if (!ts) return '\u2014';
1070
+ try {
1071
+ return new Date(ts).toLocaleString([], {
1072
+ year: 'numeric', month: 'short', day: 'numeric',
1073
+ hour: '2-digit', minute: '2-digit',
1074
+ });
1075
+ } catch { return ts; }
1076
+ }
1077
+ }
1078
+ return '\u2014';
1079
+ }
1300
1080
 
1301
- .model-pill.sonnet {
1302
- color: var(--color-model-sonnet);
1303
- border-color: rgba(107, 208, 255, 0.3);
1304
- background: rgba(107, 208, 255, 0.08);
1305
- }
1081
+ function openProjectDialog(name) {
1082
+ const p = (lastData?.projects || []).find((x) => x.name === name);
1083
+ if (!p) return;
1084
+ $('#d-name').textContent = p.name;
1085
+ $('#d-path').textContent = p.path || '';
1086
+ $('#d-cost').textContent = fmtCostFlat(p.estimated_cost_usd);
1087
+ $('#d-turns').textContent = fmtPlain(p.total_turns);
1088
+ $('#d-input').textContent = fmtPlain(p.total_input_tokens);
1089
+ $('#d-output').textContent = fmtPlain(p.total_output_tokens);
1090
+ $('#d-cache-r').textContent = fmtPlain(p.total_cache_read);
1091
+ $('#d-cache-w').textContent = fmtPlain(p.total_cache_create);
1092
+ $('#d-blocks').textContent = fmtPlain(p.blocked_count);
1093
+ $('#d-last').textContent = lastActiveFor(p.name);
1094
+ $('#dialog-backdrop').classList.remove('hidden');
1095
+ }
1306
1096
 
1307
- .model-pill.haiku {
1308
- color: var(--color-model-haiku);
1309
- border-color: rgba(123, 255, 199, 0.3);
1310
- background: rgba(123, 255, 199, 0.08);
1311
- }
1097
+ function closeProjectDialog() {
1098
+ $('#dialog-backdrop').classList.add('hidden');
1099
+ }
1312
1100
 
1313
- .model-pill.unknown {
1314
- color: var(--color-muted);
1315
- font-style: italic;
1316
- border-color: rgba(237, 236, 217, 0.15);
1317
- }
1101
+ $('#dialog-close').addEventListener('click', closeProjectDialog);
1102
+ $('#dialog-backdrop').addEventListener('click', (e) => {
1103
+ if (e.target.id === 'dialog-backdrop') closeProjectDialog();
1104
+ });
1318
1105
 
1319
- /* Existing inline code in tables (project name, etc.) */
1320
- code {
1321
- color: var(--color-heading);
1322
- background: var(--color-form-bg);
1323
- padding: 0.1rem 0.4rem;
1324
- border-radius: 3px;
1325
- font-size: 0.85em;
1326
- }
1106
+ // ----- FAQ dialog -----
1107
+ const faqBackdrop = $('#faq-backdrop');
1108
+ function openFaq() { faqBackdrop.classList.remove('hidden'); }
1109
+ function closeFaq() { faqBackdrop.classList.add('hidden'); }
1110
+ $('#faq-btn').addEventListener('click', openFaq);
1111
+ $('#faq-close').addEventListener('click', closeFaq);
1112
+ faqBackdrop.addEventListener('click', (e) => {
1113
+ if (e.target.id === 'faq-backdrop') closeFaq();
1114
+ });
1327
1115
 
1328
- /* ============================================================
1329
- Decisions (allow / block)
1330
- ============================================================ */
1116
+ document.addEventListener('keydown', (e) => {
1117
+ if (e.key === 'Escape') { closeProjectDialog(); closeFaq(); }
1118
+ });
1331
1119
 
1332
- .decision-block {
1333
- color: var(--color-block);
1334
- font-weight: 700;
1335
- text-transform: uppercase;
1336
- font-size: 0.72rem;
1337
- letter-spacing: 0.06em;
1338
- display: inline-flex;
1339
- align-items: center;
1340
- gap: 0.3rem;
1341
- }
1120
+ // Reflect the actual port the dashboard is served on
1121
+ const portEl = $('#port-num');
1122
+ if (portEl && window.location.port) portEl.textContent = window.location.port;
1123
+
1124
+ // ----- Donut chart (model usage by turn count) -----
1125
+ function renderDonut(turns) {
1126
+ const counts = { opus: 0, sonnet: 0, haiku: 0, unknown: 0 };
1127
+ for (const t of turns) counts[modelFamily(t.model)] += 1;
1128
+ const total = counts.opus + counts.sonnet + counts.haiku + counts.unknown;
1129
+
1130
+ const segs = [
1131
+ { fam: 'opus', label: 'Opus', n: counts.opus, color: 'var(--c-opus)' },
1132
+ { fam: 'sonnet', label: 'Sonnet', n: counts.sonnet, color: 'var(--c-sonnet)' },
1133
+ { fam: 'haiku', label: 'Haiku', n: counts.haiku, color: 'var(--c-haiku)' },
1134
+ { fam: 'unknown', label: 'Other', n: counts.unknown, color: 'var(--c-unknown)' },
1135
+ ].filter((s) => s.n > 0);
1136
+
1137
+ const svg = $('#donut-svg');
1138
+ svg.querySelectorAll('.donut-seg').forEach((el) => el.remove());
1139
+
1140
+ const C = 2 * Math.PI * 52; // \u2248 326.7
1141
+ let offset = 0;
1142
+ const ns = 'http://www.w3.org/2000/svg';
1143
+ if (total === 0) {
1144
+ // pleasant empty state \u2014 leave just the track
1145
+ } else {
1146
+ for (const s of segs) {
1147
+ const arc = (s.n / total) * C;
1148
+ const c = document.createElementNS(ns, 'circle');
1149
+ c.setAttribute('cx', '70');
1150
+ c.setAttribute('cy', '70');
1151
+ c.setAttribute('r', '52');
1152
+ c.setAttribute('fill', 'none');
1153
+ c.setAttribute('stroke', s.color);
1154
+ c.setAttribute('stroke-width', '14');
1155
+ c.setAttribute('stroke-dasharray', arc + ' ' + C);
1156
+ c.setAttribute('stroke-dashoffset', String(-offset));
1157
+ c.setAttribute('transform', 'rotate(-90 70 70)');
1158
+ c.setAttribute('stroke-linecap', segs.length === 1 ? 'round' : 'butt');
1159
+ c.classList.add('donut-seg');
1160
+ svg.appendChild(c);
1161
+ offset += arc;
1162
+ }
1163
+ }
1342
1164
 
1343
- .decision-block svg {
1344
- width: 11px;
1345
- height: 11px;
1346
- }
1165
+ $('#donut-total').textContent = total;
1347
1166
 
1348
- .decision-allow {
1349
- color: var(--color-allow);
1350
- opacity: 0.7;
1351
- text-transform: uppercase;
1352
- font-size: 0.72rem;
1353
- letter-spacing: 0.06em;
1354
- }
1167
+ const legend = $('#donut-legend');
1168
+ legend.innerHTML = '';
1169
+ const lf = document.createDocumentFragment();
1170
+ const display = segs.length ? segs : [
1171
+ { fam: 'opus', label: 'Opus', n: 0, color: 'var(--c-opus)' },
1172
+ { fam: 'sonnet', label: 'Sonnet', n: 0, color: 'var(--c-sonnet)' },
1173
+ { fam: 'haiku', label: 'Haiku', n: 0, color: 'var(--c-haiku)' },
1174
+ ];
1175
+ for (const s of display) {
1176
+ const pct = total > 0 ? Math.round((s.n / total) * 100) : 0;
1177
+ const row = document.createElement('div');
1178
+ row.className = 'dl-row';
1179
+ row.innerHTML =
1180
+ '<span class="dl-dot" style="background:' + s.color + '"></span>' +
1181
+ '<span class="dl-name">' + s.label + '</span>' +
1182
+ '<span class="dl-pct">' + pct + '%</span>';
1183
+ lf.appendChild(row);
1184
+ }
1185
+ legend.appendChild(lf);
1186
+ }
1355
1187
 
1356
- .empty {
1357
- color: var(--color-muted);
1358
- font-style: italic;
1359
- padding: 1rem;
1360
- background: var(--color-surface);
1361
- border: 1px dashed var(--color-border);
1362
- border-radius: 8px;
1363
- }
1188
+ // ----- master render -----
1189
+ function applyData(data) {
1190
+ const turns = data.recent_turns || [];
1191
+ const gates = data.recent_gates || [];
1192
+
1193
+ renderSession(turns);
1194
+ renderSavings(data.global);
1195
+ renderCostHero(data.global);
1196
+ renderMoat(data.global);
1197
+ renderTurns(turns);
1198
+ renderGateMini(gates);
1199
+ renderDonut(turns);
1200
+ }
1364
1201
 
1365
- .hidden {
1366
- display: none;
1367
- }
1202
+ // ----- polling -----
1203
+ async function tick() {
1204
+ try {
1205
+ const res = await fetch('/data');
1206
+ if (!res.ok) throw new Error('HTTP ' + res.status);
1207
+ const data = await res.json();
1208
+ lastData = data;
1209
+ $('#active-project').textContent = data.active?.project_root || '\u2014';
1210
+ renderProjectLegend(data.projects || []);
1211
+ applyData(data);
1212
+ $('#status').textContent = 'live \xB7 ' + new Date().toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' });
1213
+ $('#dot').classList.add('live'); $('#dot').classList.remove('dead');
1214
+ } catch (e) {
1215
+ $('#status').textContent = 'offline';
1216
+ $('#dot').classList.add('dead'); $('#dot').classList.remove('live');
1217
+ }
1218
+ }
1368
1219
 
1369
- footer {
1370
- display: flex;
1371
- justify-content: space-between;
1372
- align-items: center;
1373
- padding: 0.85rem 2rem;
1374
- background: linear-gradient(0deg, rgba(20, 0, 9, 0.7), rgba(20, 0, 9, 0.4));
1375
- backdrop-filter: blur(8px);
1376
- border-top: 1px solid var(--color-border);
1377
- color: var(--color-muted);
1378
- font-size: 0.72rem;
1379
- font-family: ui-monospace, monospace;
1380
- position: sticky;
1381
- bottom: 0;
1382
- z-index: 5;
1383
- }
1220
+ // ----- Viewport-clamped tooltip -----
1221
+ const tooltipEl = document.createElement('div');
1222
+ tooltipEl.className = 'global-tooltip';
1223
+ document.body.appendChild(tooltipEl);
1224
+ let activeTooltipTarget = null;
1225
+
1226
+ function positionTooltip(target) {
1227
+ const rect = target.getBoundingClientRect();
1228
+ const ttRect = tooltipEl.getBoundingClientRect();
1229
+ const margin = 12;
1230
+ let top = rect.top - ttRect.height - 10;
1231
+ let left = rect.left + rect.width / 2 - ttRect.width / 2;
1232
+ if (left < margin) left = margin;
1233
+ if (left + ttRect.width > window.innerWidth - margin) {
1234
+ left = window.innerWidth - ttRect.width - margin;
1235
+ }
1236
+ if (top < margin) {
1237
+ top = rect.bottom + 10;
1238
+ }
1239
+ if (top + ttRect.height > window.innerHeight - margin) {
1240
+ top = window.innerHeight - ttRect.height - margin;
1241
+ }
1242
+ tooltipEl.style.top = top + 'px';
1243
+ tooltipEl.style.left = left + 'px';
1244
+ }
1384
1245
 
1385
- footer .muted {
1386
- color: var(--color-very-muted);
1387
- }
1246
+ function showTooltip(target) {
1247
+ const text = target.getAttribute('data-tooltip');
1248
+ if (!text) return;
1249
+ tooltipEl.textContent = text;
1250
+ tooltipEl.classList.add('on');
1251
+ positionTooltip(target);
1252
+ }
1388
1253
 
1389
- .dot {
1390
- width: 8px;
1391
- height: 8px;
1392
- border-radius: 50%;
1393
- background: var(--color-muted);
1394
- transition: background 200ms ease, box-shadow 200ms ease;
1395
- }
1254
+ function hideTooltip() {
1255
+ tooltipEl.classList.remove('on');
1256
+ }
1396
1257
 
1397
- .dot.live {
1398
- background: var(--color-money);
1399
- box-shadow: 0 0 8px var(--color-money-darker);
1400
- }
1258
+ document.addEventListener('mouseover', (e) => {
1259
+ const t = (e.target instanceof Element) ? e.target.closest('.has-tooltip') : null;
1260
+ if (t !== activeTooltipTarget) {
1261
+ activeTooltipTarget = t;
1262
+ if (t) showTooltip(t);
1263
+ else hideTooltip();
1264
+ }
1265
+ });
1266
+ document.addEventListener('scroll', hideTooltip, true);
1267
+ document.addEventListener('keydown', (e) => { if (e.key === 'Escape') hideTooltip(); });
1401
1268
 
1402
- .dot.dead {
1403
- background: var(--color-border);
1404
- box-shadow: none;
1405
- }
1269
+ setHeroDate();
1270
+ tick();
1271
+ setInterval(tick, 2000);
1272
+ </script>
1273
+ </body>
1274
+ </html>
1406
1275
  `;
1407
1276
 
1277
+ // src/dashboard/public/style.css
1278
+ var style_default = '/* Synthra dashboard \xB7 v0.2 \xB7 Cool Marine\n Darkened surfaces; brand blue reserved for hero elements only.\n Layout: top nav + hero strip + 3-column main, fits 1280\xD7720. */\n\n:root {\n /* Core palette */\n --ink: #04081A;\n --navy: #0A1530;\n --navy-2: #122549;\n --deep-blue: #1B3A78;\n --blue: #2C5DB8;\n --blue-bright: #5C8FE6;\n --sky: #9BC2EF;\n --mist: #D7E6F7;\n --bone: #F4F7FC;\n\n /* Text */\n --text: #ECF2FB;\n --text-dim: #A9BBD6;\n --text-mute: #6D80A0;\n\n /* Rules / dividers */\n --rule: rgba(155, 194, 239, .14);\n --rule-2: rgba(155, 194, 239, .06);\n --rule-hover: rgba(155, 194, 239, .28);\n\n /* Surfaces (darker than v0.1.2) */\n --surface-1: rgba(18, 37, 73, .14);\n --surface-2: rgba(18, 37, 73, .22);\n --surface-3: rgba(4, 8, 26, .55);\n\n /* Signal accents (OKLCH shared chroma) */\n --signal-cyan: oklch(78% 0.14 220);\n --signal-amber: oklch(78% 0.14 75);\n --signal-rose: oklch(70% 0.14 20);\n --signal-green: oklch(75% 0.14 155);\n --signal-violet: oklch(72% 0.14 285);\n\n /* Model family colors */\n --c-opus: #FF6338;\n --c-sonnet: #FFB938;\n --c-haiku: #7438FF;\n --c-unknown: #12CBF5;\n\n /* Money */\n --money: var(--signal-green);\n\n /* Type */\n --font-sans: "Geist", ui-sans-serif, system-ui, -apple-system, "Segoe UI", sans-serif;\n --font-serif: "Instrument Serif", "Times New Roman", serif;\n --font-mono: "Geist Mono", ui-monospace, "SF Mono", Menlo, Consolas, monospace;\n}\n\n/* ============================================================\n Reset + base\n ============================================================ */\n*,\n*::before,\n*::after {\n box-sizing: border-box;\n}\n\nhtml,\nbody {\n margin: 0;\n padding: 0;\n}\n\nhtml,\nbody {\n height: 100vh;\n overflow: hidden;\n}\n\nbody {\n background: var(--ink);\n color: var(--text);\n font-family: var(--font-sans);\n font-size: 13px;\n line-height: 1.5;\n -webkit-font-smoothing: antialiased;\n text-rendering: optimizeLegibility;\n display: grid;\n grid-template-rows: auto auto 1fr auto;\n position: relative;\n}\n\n/* Layered backdrop \u2014 quieter */\nbody::before,\nbody::after {\n content: "";\n position: fixed;\n inset: 0;\n pointer-events: none;\n z-index: 0;\n}\n\nbody::before {\n background-image: radial-gradient(circle, rgba(155, 194, 239, .06) 1px, transparent 1.2px);\n background-size: 22px 22px;\n}\n\nbody::after {\n background:\n radial-gradient(60% 40% at 50% 105%, rgba(44, 93, 184, .16) 0%, rgba(10, 21, 48, 0) 65%),\n radial-gradient(30% 25% at 50% 0%, rgba(92, 143, 230, .06) 0%, transparent 70%);\n}\n\nbody>* {\n position: relative;\n z-index: 1;\n}\n\nbutton {\n font: inherit;\n cursor: pointer;\n border: 0;\n background: transparent;\n color: inherit;\n}\n\na {\n color: inherit;\n text-decoration: none;\n}\n\n/* ============================================================\n Top nav\n ============================================================ */\n.topnav {\n display: grid;\n grid-template-columns: 1fr auto 1fr;\n align-items: center;\n height: 52px;\n padding: 0 24px;\n border-bottom: 1px solid var(--rule);\n background: linear-gradient(180deg, rgba(4, 8, 26, .7), rgba(4, 8, 26, .4));\n backdrop-filter: blur(10px);\n}\n\n.brand {\n display: flex;\n align-items: center;\n gap: 10px;\n}\n\n.brand-mark {\n width: 22px;\n height: 22px;\n border-radius: 7px;\n background: radial-gradient(120% 120% at 30% 30%, #6FA6E8 0%, #2C5DB8 45%, #0A1530 100%);\n box-shadow: inset 0 0 0 1px rgba(255, 255, 255, .22), 0 4px 12px -6px #2C5DB8;\n}\n\n.brand-name {\n font-size: 15px;\n font-weight: 600;\n letter-spacing: -0.01em;\n color: var(--mist);\n}\n\n.brand-name em {\n font-family: var(--font-serif);\n font-style: italic;\n font-weight: 400;\n color: var(--sky);\n letter-spacing: 0;\n}\n\n.brand-eyebrow {\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.14em;\n text-transform: uppercase;\n color: var(--text-mute);\n margin-left: 6px;\n padding-left: 10px;\n border-left: 1px solid var(--rule);\n}\n\n.top-right {\n display: flex;\n align-items: center;\n gap: 12px;\n grid-column: 2;\n justify-self: center;\n}\n\n.topnav-right {\n grid-column: 3;\n justify-self: end;\n display: flex;\n align-items: center;\n gap: 10px;\n}\n\n.port-badge {\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.12em;\n text-transform: uppercase;\n color: var(--text-mute);\n padding: 6px 10px;\n border: 1px solid var(--rule);\n border-radius: 999px;\n background: rgba(4, 8, 26, .55);\n}\n\n.port-badge .mono {\n color: var(--text-dim);\n letter-spacing: 0.04em;\n text-transform: none;\n}\n\n.faq-btn {\n width: 30px;\n height: 30px;\n border-radius: 50%;\n border: 1px solid var(--rule);\n background: rgba(4, 8, 26, .55);\n color: var(--text-dim);\n font-family: var(--font-mono);\n font-size: 13px;\n font-weight: 500;\n display: inline-flex;\n align-items: center;\n justify-content: center;\n transition: background 180ms, border-color 180ms, color 180ms, transform 180ms;\n}\n\n.faq-btn:hover {\n background: rgba(155, 194, 239, .10);\n border-color: var(--rule-hover);\n color: var(--mist);\n transform: translateY(-1px);\n}\n\n.status-pill {\n display: inline-flex;\n align-items: center;\n gap: 8px;\n padding: 6px 12px;\n border: 1px solid var(--rule);\n border-radius: 999px;\n background: rgba(4, 8, 26, .55);\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.12em;\n text-transform: uppercase;\n color: var(--text-dim);\n transition: border-color 240ms ease;\n}\n\n.status-pill:has(.dot.live) {\n border-color: rgba(155, 194, 239, .45);\n color: var(--mist);\n animation: pill-glow 2.4s ease-in-out infinite;\n}\n\n.status-pill:has(.dot.dead) {\n border-color: rgba(220, 90, 90, .40);\n color: oklch(80% 0.10 20);\n}\n\n@keyframes pill-glow {\n\n 0%,\n 100% {\n box-shadow: 0 0 14px -4px rgba(155, 194, 239, .30), inset 0 0 12px -8px rgba(155, 194, 239, .30);\n }\n\n 50% {\n box-shadow: 0 0 26px -2px rgba(155, 194, 239, .55), inset 0 0 18px -6px rgba(155, 194, 239, .45);\n }\n}\n\n.dot {\n width: 7px;\n height: 7px;\n border-radius: 2px;\n background: var(--text-mute);\n transition: background 200ms;\n}\n\n.dot.live {\n background: var(--signal-cyan);\n animation: dot-pulse 1.8s ease-in-out infinite;\n}\n\n.dot.dead {\n background: var(--signal-rose);\n box-shadow: 0 0 0 3px rgba(220, 90, 90, .10);\n}\n\n@keyframes dot-pulse {\n\n 0%,\n 100% {\n box-shadow:\n 0 0 0 3px rgba(155, 194, 239, .10),\n 0 0 6px rgba(155, 194, 239, .50);\n }\n\n 50% {\n box-shadow:\n 0 0 0 6px rgba(155, 194, 239, .05),\n 0 0 14px rgba(155, 194, 239, .90);\n }\n}\n\n/* ============================================================\n Hero strip\n ============================================================ */\n.hero-strip {\n display: flex;\n align-items: center;\n gap: 24px;\n padding: 14px 24px;\n border-bottom: 1px solid var(--rule);\n background: linear-gradient(90deg, rgba(27, 58, 120, .10) 0%, rgba(4, 8, 26, 0) 100%);\n position: relative;\n overflow: hidden;\n}\n\n.hero-spacer {\n flex: 1;\n}\n\n.date-block {\n display: flex;\n align-items: center;\n gap: 12px;\n}\n\n.d-day {\n font-family: var(--font-sans);\n font-weight: 500;\n font-size: 38px;\n line-height: 1;\n letter-spacing: -0.04em;\n color: var(--mist);\n}\n\n.d-rest {\n display: flex;\n flex-direction: column;\n gap: 2px;\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.14em;\n text-transform: uppercase;\n color: var(--text-dim);\n}\n\n.d-rest .d-mute {\n color: var(--text-mute);\n}\n\n.active-block {\n display: flex;\n flex-direction: column;\n gap: 2px;\n text-align: right;\n max-width: 360px;\n overflow: hidden;\n}\n\n.ab-label {\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.14em;\n text-transform: uppercase;\n color: var(--text-mute);\n}\n\n.ab-value {\n font-family: var(--font-mono);\n font-size: 12px;\n color: var(--mist);\n white-space: nowrap;\n overflow: hidden;\n text-overflow: ellipsis;\n max-width: 360px;\n}\n\n/* ============================================================\n Main grid\n ============================================================ */\n.grid-main {\n display: grid;\n grid-template-columns: 260px 1fr 340px;\n gap: 16px;\n padding: 16px 24px;\n min-height: 0;\n z-index: 10;\n}\n\n.col-left,\n.col-center,\n.col-right {\n display: flex;\n flex-direction: column;\n gap: 12px;\n min-height: 0;\n}\n\n/* ============================================================\n Panels / cards \u2014 darker\n ============================================================ */\n.panel,\n.card {\n position: relative;\n border: 1px solid var(--rule);\n border-radius: 14px;\n background: var(--surface-1);\n padding: 14px 16px;\n display: flex;\n flex-direction: column;\n gap: 12px;\n min-height: 0;\n transition: border-color 180ms ease, background 180ms ease;\n}\n\n.card.has-tooltip {\n cursor: help;\n}\n\n.card.has-tooltip:hover {\n border-color: var(--rule-hover);\n background: var(--surface-2);\n}\n\n.card-head {\n display: flex;\n justify-content: space-between;\n align-items: baseline;\n gap: 12px;\n}\n\n.card-eyebrow {\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.14em;\n text-transform: uppercase;\n color: var(--text-mute);\n display: inline-flex;\n align-items: center;\n gap: 6px;\n}\n\n.card-eyebrow em {\n font-family: var(--font-serif);\n font-style: italic;\n font-weight: 400;\n font-size: 12px;\n color: var(--sky);\n letter-spacing: 0;\n text-transform: none;\n}\n\n.card-meta {\n font-family: var(--font-mono);\n font-size: 9px;\n letter-spacing: 0.12em;\n text-transform: uppercase;\n color: var(--text-mute);\n}\n\n/* Legend panel */\n.panel {\n padding: 14px 14px 16px;\n gap: 14px;\n flex-shrink: 0;\n}\n\n.p-head {\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.14em;\n text-transform: uppercase;\n color: var(--text-mute);\n padding-bottom: 10px;\n border-bottom: 1px solid var(--rule-2);\n}\n\n.p-section {\n display: flex;\n flex-direction: column;\n gap: 2px;\n}\n\n.p-section+.p-section {\n padding-top: 12px;\n border-top: 1px solid var(--rule-2);\n}\n\n.ps-head {\n font-family: var(--font-mono);\n font-size: 9px;\n letter-spacing: 0.16em;\n text-transform: uppercase;\n color: var(--text-mute);\n margin-bottom: 4px;\n}\n\n.check {\n display: inline-flex;\n align-items: center;\n gap: 8px;\n padding: 3px 6px;\n font-family: var(--font-mono);\n font-size: 11px;\n color: var(--text-dim);\n letter-spacing: 0.02em;\n}\n\nbutton.check {\n border: 0;\n background: transparent;\n width: 100%;\n text-align: left;\n}\n\n.check-clickable {\n cursor: pointer;\n border-radius: 6px;\n padding: 5px 6px;\n transition: background 140ms, color 140ms, transform 140ms;\n}\n\n.check-clickable .pf-arrow {\n margin-left: auto;\n color: var(--text-mute);\n font-family: var(--font-mono);\n font-size: 12px;\n transition: color 140ms, transform 140ms;\n}\n\n.check-clickable:hover {\n background: rgba(155, 194, 239, .07);\n color: var(--mist);\n}\n\n.check-clickable:hover .pf-arrow {\n color: var(--sky);\n transform: translateX(2px);\n}\n\n.dot-sq {\n width: 8px;\n height: 8px;\n border-radius: 2px;\n background: var(--text-mute);\n flex-shrink: 0;\n}\n\n.dot-sq.opus {\n background: var(--c-opus);\n}\n\n.dot-sq.sonnet {\n background: var(--c-sonnet);\n}\n\n.dot-sq.haiku {\n background: var(--c-haiku);\n}\n\n.dot-sq.unknown {\n background: var(--c-unknown);\n}\n\n.proj-filter {\n display: flex;\n flex-direction: column;\n gap: 1px;\n max-height: 90px;\n overflow-y: auto;\n}\n\n.pf-name {\n white-space: nowrap;\n overflow: hidden;\n text-overflow: ellipsis;\n max-width: 140px;\n}\n\n/* ============================================================\n Donut card (model usage)\n ============================================================ */\n.donut-card {\n flex: 1;\n gap: 10px;\n}\n\n.donut-wrap {\n position: relative;\n width: 140px;\n height: 140px;\n margin: 4px auto 0;\n}\n\n.donut {\n width: 100%;\n height: 100%;\n display: block;\n}\n\n.donut-track {\n fill: none;\n stroke: rgba(155, 194, 239, .07);\n stroke-width: 14;\n}\n\n.donut-seg {\n transition: stroke-dashoffset 400ms ease, stroke-dasharray 400ms ease;\n}\n\n.donut-center {\n position: absolute;\n inset: 0;\n display: flex;\n flex-direction: column;\n justify-content: center;\n align-items: center;\n pointer-events: none;\n}\n\n.donut-total {\n font-family: var(--font-sans);\n font-weight: 500;\n font-size: 26px;\n letter-spacing: -0.02em;\n color: var(--mist);\n line-height: 1;\n}\n\n.donut-total-k {\n font-family: var(--font-mono);\n font-size: 9px;\n letter-spacing: 0.16em;\n text-transform: uppercase;\n color: var(--text-mute);\n margin-top: 4px;\n}\n\n.donut-legend {\n display: flex;\n flex-direction: column;\n gap: 4px;\n margin-top: 6px;\n padding-top: 10px;\n border-top: 1px solid var(--rule-2);\n}\n\n.dl-row {\n display: grid;\n grid-template-columns: auto 1fr auto;\n align-items: center;\n gap: 8px;\n font-family: var(--font-mono);\n font-size: 11px;\n color: var(--text-dim);\n}\n\n.dl-dot {\n width: 8px;\n height: 8px;\n border-radius: 2px;\n}\n\n.dl-name {\n color: var(--text-dim);\n}\n\n.dl-pct {\n color: var(--mist);\n font-weight: 500;\n}\n\n/* ============================================================\n Center column \u2014 Metric strip (no card chrome, divider-separated)\n ============================================================ */\n.metric-strip {\n display: grid;\n grid-template-columns: repeat(5, 1fr);\n border: 1px solid var(--rule);\n border-radius: 14px;\n background: var(--surface-1);\n overflow: hidden;\n flex-shrink: 0;\n}\n\n.metric-item {\n padding: 14px 18px;\n display: flex;\n flex-direction: column;\n gap: 8px;\n cursor: help;\n border-right: 1px solid var(--rule-2);\n transition: background 200ms ease;\n min-width: 0;\n}\n.metric-item:last-child { border-right: 0; }\n.metric-item:hover { background: var(--surface-2); }\n\n.m-label {\n font-family: var(--font-mono);\n font-size: 9px;\n letter-spacing: 0.14em;\n text-transform: uppercase;\n color: var(--text-mute);\n}\n\n.m-value {\n font-family: var(--font-sans);\n font-weight: 500;\n font-size: 26px;\n letter-spacing: -0.025em;\n color: var(--mist);\n line-height: 1;\n}\n\n.m-value em {\n font-family: var(--font-serif);\n font-style: italic;\n font-weight: 400;\n color: var(--sky);\n letter-spacing: -0.005em;\n}\n\n/* ============================================================\n Savings card\n ============================================================ */\n.card.savings {\n flex-shrink: 0;\n gap: 12px;\n background:\n linear-gradient(180deg, rgba(123, 255, 199, .05) 0%, rgba(4, 8, 26, .20) 50%),\n var(--surface-1);\n border-color: rgba(123, 255, 199, .18);\n}\n\n.card.savings:hover {\n border-color: rgba(123, 255, 199, .32);\n}\n\n.savings-body {\n display: grid;\n grid-template-columns: auto 1fr;\n align-items: center;\n gap: 18px;\n}\n\n.savings-figure {\n display: flex;\n flex-direction: column;\n gap: 2px;\n}\n\n.savings-money {\n font-family: var(--font-sans);\n font-weight: 500;\n font-size: 34px;\n letter-spacing: -0.035em;\n line-height: 1;\n color: var(--money);\n}\n\n.savings-tokens {\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.10em;\n text-transform: uppercase;\n color: var(--text-mute);\n margin-top: 4px;\n}\n\n.savings-bar {\n position: relative;\n height: 8px;\n border-radius: 999px;\n overflow: hidden;\n background: var(--surface-3);\n display: flex;\n}\n\n.savings-actual {\n height: 100%;\n background: rgba(215, 230, 247, .55);\n transition: width 500ms cubic-bezier(0.16, 1, 0.3, 1);\n}\n\n.savings-saved {\n height: 100%;\n background: var(--signal-green);\n transition: width 500ms cubic-bezier(0.16, 1, 0.3, 1);\n box-shadow: inset 0 1px 0 rgba(255, 255, 255, .12);\n}\n\n.savings-legend {\n grid-column: 2;\n display: flex;\n justify-content: center;\n align-items: center;\n gap: 24px;\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.08em;\n color: var(--text-mute);\n margin-top: 8px;\n}\n\n.sl-row {\n display: inline-flex;\n align-items: center;\n gap: 6px;\n}\n\n.sl-row b {\n color: var(--mist);\n font-weight: 500;\n letter-spacing: 0.04em;\n}\n\n.sl-dot {\n width: 8px;\n height: 8px;\n border-radius: 2px;\n}\n\n.sl-dot.actual {\n background: var(--mist);\n}\n\n.sl-dot.saved {\n background: var(--signal-green);\n}\n\n/* ============================================================\n Recent turns table\n ============================================================ */\n.turns-card {\n flex: 1;\n padding: 0;\n overflow: hidden;\n}\n\n.turns-card .card-head {\n padding: 14px 16px 10px;\n border-bottom: 1px solid var(--rule-2);\n}\n\n.turns-scroll {\n flex: 1;\n overflow-y: auto;\n min-height: 0;\n}\n\n.turns-table {\n width: 100%;\n border-collapse: collapse;\n}\n\n.turns-table thead th {\n position: sticky;\n top: 0;\n background: var(--ink);\n padding: 9px 16px;\n text-align: left;\n font-family: var(--font-mono);\n font-size: 9px;\n letter-spacing: 0.14em;\n text-transform: uppercase;\n color: var(--text-mute);\n font-weight: 500;\n border-bottom: 1px solid var(--rule);\n z-index: 1;\n}\n\n.turns-table thead th.num {\n text-align: right;\n}\n\n.turns-table tbody td {\n padding: 8px 16px;\n border-bottom: 1px solid var(--rule-2);\n color: var(--text-dim);\n font-size: 12px;\n}\n\n.turns-table tbody td.num {\n text-align: right;\n font-family: var(--font-mono);\n}\n\n.turns-table tbody td.cost {\n color: var(--money);\n font-family: var(--font-mono);\n font-weight: 500;\n}\n\n.turns-table tbody td.ts {\n font-family: var(--font-mono);\n font-size: 11px;\n color: var(--text-mute);\n}\n\n.turns-table tbody td.proj {\n color: var(--mist);\n}\n\n.turns-table tbody tr:hover {\n background: rgba(155, 194, 239, .03);\n}\n\n.turns-table tbody tr:last-child td {\n border-bottom: 0;\n}\n\n/* Model pills */\n.model-pill {\n display: inline-flex;\n align-items: center;\n gap: 6px;\n padding: 2px 8px;\n border-radius: 999px;\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.04em;\n border: 1px solid var(--rule);\n background: rgba(4, 8, 26, .5);\n color: var(--mist);\n}\n\n.model-pill .sq {\n width: 6px;\n height: 6px;\n border-radius: 2px;\n background: var(--text-mute);\n}\n\n.model-pill.opus {\n color: #FF8A66;\n border-color: rgba(255, 99, 56, .32);\n background: rgba(255, 99, 56, .07);\n}\n\n.model-pill.opus .sq {\n background: var(--c-opus);\n}\n\n.model-pill.sonnet {\n color: #FFC766;\n border-color: rgba(255, 185, 56, .32);\n background: rgba(255, 185, 56, .07);\n}\n\n.model-pill.sonnet .sq {\n background: var(--c-sonnet);\n}\n\n.model-pill.haiku {\n color: #A878FF;\n border-color: rgba(116, 56, 255, .42);\n background: rgba(116, 56, 255, .10);\n}\n\n.model-pill.haiku .sq {\n background: var(--c-haiku);\n}\n\n.model-pill.unknown {\n color: #5BDDF7;\n border-color: rgba(18, 203, 245, .32);\n background: rgba(18, 203, 245, .07);\n font-style: italic;\n}\n\n.model-pill.unknown .sq {\n background: var(--c-unknown);\n}\n\n/* ============================================================\n Right column \u2014 Cost hero\n ============================================================ */\n.cost-hero {\n position: relative;\n overflow: hidden;\n background:\n radial-gradient(120% 80% at 50% 110%, rgba(44, 93, 184, .18) 0%, rgba(4, 8, 26, .20) 60%),\n var(--surface-1);\n padding: 16px 18px 18px;\n gap: 10px;\n flex-shrink: 0;\n}\n\n.big-money {\n position: relative;\n font-family: var(--font-sans);\n font-weight: 500;\n font-size: 42px;\n letter-spacing: -0.035em;\n line-height: 1;\n color: var(--money);\n margin-top: 2px;\n}\n\n.big-money em {\n font-family: inherit;\n font-style: normal;\n font-weight: inherit;\n color: inherit;\n letter-spacing: inherit;\n opacity: 1;\n}\n\n.cost-sub {\n position: relative;\n display: flex;\n flex-direction: column;\n gap: 6px;\n margin-top: 4px;\n padding-top: 10px;\n border-top: 1px solid var(--rule-2);\n}\n\n.cs-row {\n display: flex;\n justify-content: space-between;\n align-items: baseline;\n font-family: var(--font-mono);\n font-size: 11px;\n}\n\n.cs-k {\n font-size: 10px;\n letter-spacing: 0.12em;\n text-transform: uppercase;\n color: var(--text-mute);\n}\n\n.cs-v {\n color: var(--mist);\n}\n\n/* ============================================================\n Moat card\n ============================================================ */\n.moat {\n flex: 1;\n gap: 8px;\n}\n\n.moat-value {\n font-family: var(--font-sans);\n font-weight: 500;\n font-size: 34px;\n letter-spacing: -0.03em;\n line-height: 1;\n color: var(--mist);\n margin-top: 2px;\n}\n\n.moat-value em {\n font-family: var(--font-serif);\n font-style: italic;\n font-weight: 400;\n font-size: 18px;\n color: var(--sky);\n letter-spacing: 0;\n margin-left: 6px;\n}\n\n.gate-mini {\n display: flex;\n flex-direction: column;\n gap: 4px;\n margin-top: 6px;\n padding-top: 10px;\n border-top: 1px solid var(--rule-2);\n overflow-y: auto;\n flex: 1;\n min-height: 0;\n}\n\n.gate-row {\n display: grid;\n grid-template-columns: auto auto 1fr;\n align-items: center;\n gap: 8px;\n font-family: var(--font-mono);\n font-size: 10px;\n color: var(--text-dim);\n padding: 3px 0;\n}\n\n.gate-row .g-ts {\n color: var(--text-mute);\n font-size: 9px;\n min-width: 38px;\n}\n\n.gate-row .g-decision {\n font-size: 9px;\n letter-spacing: 0.12em;\n text-transform: uppercase;\n padding: 2px 6px;\n border-radius: 999px;\n}\n\n.gate-row .g-decision.block {\n color: var(--signal-rose);\n background: rgba(220, 90, 90, .06);\n}\n\n.gate-row .g-decision.allow {\n color: var(--text-mute);\n background: rgba(155, 194, 239, .03);\n}\n\n.gate-row .g-q {\n color: var(--mist);\n white-space: nowrap;\n overflow: hidden;\n text-overflow: ellipsis;\n}\n\n/* ============================================================\n Tooltips\n ============================================================ */\n.has-tooltip {\n position: relative;\n}\n\n/* Global JS-positioned tooltip \u2014 viewport-clamped */\n.global-tooltip {\n position: fixed;\n top: 0;\n left: 0;\n background: linear-gradient(180deg, rgba(18, 37, 73, .98), rgba(10, 21, 48, .98));\n color: var(--mist);\n border: 1px solid var(--rule-hover);\n border-radius: 12px;\n padding: 14px 16px;\n font-family: var(--font-sans);\n font-size: 15px;\n font-weight: 400;\n text-transform: none;\n letter-spacing: 0;\n white-space: normal;\n width: 320px;\n max-width: calc(100vw - 24px);\n text-align: left;\n line-height: 1.55;\n box-shadow: 0 16px 36px rgba(0, 0, 0, .7);\n backdrop-filter: blur(10px);\n z-index: 99999;\n opacity: 0;\n pointer-events: none;\n transform: translateY(6px);\n transition: opacity 180ms ease, transform 180ms ease;\n}\n\n.global-tooltip.on {\n opacity: 1;\n transform: translateY(0);\n}\n\n/* ============================================================\n Footer\n ============================================================ */\n.foot {\n display: flex;\n justify-content: space-between;\n align-items: center;\n padding: 8px 24px;\n border-top: 1px solid var(--rule);\n background: linear-gradient(0deg, rgba(4, 8, 26, .7), rgba(4, 8, 26, .4));\n backdrop-filter: blur(10px);\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.12em;\n text-transform: uppercase;\n color: var(--text-mute);\n}\n\n.foot em {\n font-family: var(--font-serif);\n font-style: italic;\n text-transform: none;\n letter-spacing: 0;\n color: var(--sky);\n font-size: 12px;\n}\n\n.foot .mono {\n color: var(--text-dim);\n text-transform: none;\n letter-spacing: 0.04em;\n}\n\n/* ============================================================\n Empty state\n ============================================================ */\n.empty {\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.06em;\n color: var(--text-mute);\n text-align: center;\n padding: 16px 8px;\n font-style: italic;\n text-transform: none;\n}\n\n/* Scrollbar styling */\n.turns-scroll::-webkit-scrollbar,\n.proj-filter::-webkit-scrollbar,\n.gate-mini::-webkit-scrollbar {\n width: 6px;\n}\n\n.turns-scroll::-webkit-scrollbar-thumb,\n.proj-filter::-webkit-scrollbar-thumb,\n.gate-mini::-webkit-scrollbar-thumb {\n background: var(--rule);\n border-radius: 999px;\n}\n\n.turns-scroll::-webkit-scrollbar-track,\n.proj-filter::-webkit-scrollbar-track,\n.gate-mini::-webkit-scrollbar-track {\n background: transparent;\n}\n\n.hidden {\n display: none !important;\n}\n\n/* ============================================================\n Staggered cascade on first paint (one-time, MOTION 6)\n ============================================================ */\n@keyframes cascade-in {\n from { opacity: 0; transform: translateY(10px); }\n to { opacity: 1; transform: translateY(0); }\n}\n\n@media (prefers-reduced-motion: no-preference) {\n .col-left > *,\n .col-center > *,\n .col-right > * {\n opacity: 0;\n animation: cascade-in 520ms cubic-bezier(0.16, 1, 0.3, 1) forwards;\n will-change: transform, opacity;\n }\n .col-left > *:nth-child(1) { animation-delay: 0ms; }\n .col-left > *:nth-child(2) { animation-delay: 120ms; }\n .col-center > *:nth-child(1) { animation-delay: 40ms; }\n .col-center > *:nth-child(2) { animation-delay: 140ms; }\n .col-center > *:nth-child(3) { animation-delay: 240ms; }\n .col-right > *:nth-child(1) { animation-delay: 80ms; }\n .col-right > *:nth-child(2) { animation-delay: 200ms; }\n\n /* Clear will-change after animation completes */\n .col-left > *,\n .col-center > *,\n .col-right > * {\n animation-fill-mode: forwards;\n }\n}\n\n/* ============================================================\n Source / basis annotations\n ============================================================ */\n.card-source {\n font-family: var(--font-mono);\n font-size: 9px;\n letter-spacing: 0.08em;\n text-transform: lowercase;\n color: var(--text-mute);\n display: inline-flex;\n align-items: center;\n gap: 6px;\n margin-top: auto;\n padding-top: 8px;\n border-top: 1px solid var(--rule-2);\n width: 100%;\n}\n\n.src-badge {\n font-family: var(--font-mono);\n font-size: 8px;\n font-weight: 500;\n letter-spacing: 0.14em;\n text-transform: uppercase;\n padding: 2px 6px;\n border-radius: 4px;\n flex-shrink: 0;\n}\n\n.src-badge.verified {\n color: var(--signal-green);\n background: rgba(123, 255, 199, .08);\n border: 1px solid rgba(123, 255, 199, .25);\n}\n\n.src-badge.estimated,\n.src-badge.estimated.floor {\n color: var(--signal-amber);\n background: rgba(255, 185, 56, .10);\n border: 1px solid rgba(255, 185, 56, .30);\n}\n\n.src-badge.priced {\n color: var(--signal-cyan);\n background: rgba(155, 194, 239, .08);\n border: 1px solid rgba(155, 194, 239, .25);\n}\n\n/* Eyebrow that contains a badge */\n.card-eyebrow .src-badge {\n margin-left: 4px;\n}\n\n/* Savings audit row \u2014 live formula reveal */\n.savings-audit {\n margin-top: 10px;\n padding: 10px 12px;\n border: 1px dashed rgba(255, 185, 56, .25);\n border-radius: 8px;\n background: rgba(255, 185, 56, .04);\n font-family: var(--font-mono);\n font-size: 10.5px;\n letter-spacing: 0.04em;\n color: var(--text-mute);\n text-align: center;\n}\n\n.savings-audit b {\n color: var(--text-dim);\n font-weight: 500;\n}\n\n.savings-audit .audit-result {\n color: var(--money);\n}\n\n/* ============================================================\n FAQ dialog\n ============================================================ */\n.dialog.dialog-faq {\n max-width: min(80vw, 1100px);\n width: 100%;\n max-height: 86vh;\n display: flex;\n flex-direction: column;\n padding: 28px 32px 24px;\n gap: 6px;\n}\n\n.dialog.dialog-faq .dialog-path {\n margin-bottom: 4px;\n word-break: normal;\n overflow-wrap: anywhere;\n}\n\n.faq-content {\n flex: 1 1 auto;\n min-height: 0;\n overflow-y: auto;\n margin-top: 18px;\n padding-right: 8px;\n display: flex;\n flex-direction: column;\n gap: 10px;\n}\n\n.faq-content::-webkit-scrollbar {\n width: 6px;\n}\n\n.faq-content::-webkit-scrollbar-thumb {\n background: var(--rule);\n border-radius: 999px;\n}\n\n.faq-content::-webkit-scrollbar-track {\n background: transparent;\n}\n\n.faq-content details {\n border: 1px solid var(--rule);\n border-radius: 12px;\n background: var(--surface-1);\n overflow: hidden;\n transition: background 180ms, border-color 180ms;\n flex-shrink: 0;\n}\n\n.faq-content details:hover {\n border-color: rgba(155, 194, 239, .22);\n}\n\n.faq-content details[open] {\n background: var(--surface-2);\n border-color: var(--rule-hover);\n}\n\n.faq-content summary {\n cursor: pointer;\n padding: 14px 20px;\n font-family: var(--font-sans);\n font-size: 14px;\n font-weight: 500;\n color: var(--mist);\n list-style: none;\n display: flex;\n align-items: center;\n gap: 12px;\n user-select: none;\n}\n\n.faq-content summary::-webkit-details-marker {\n display: none;\n}\n\n.faq-content summary::before {\n content: "\u203A";\n display: inline-flex;\n align-items: center;\n justify-content: center;\n width: 16px;\n height: 16px;\n color: var(--text-mute);\n font-family: var(--font-mono);\n font-size: 14px;\n transition: transform 220ms ease, color 220ms ease;\n}\n\n.faq-content details[open] summary::before {\n transform: rotate(90deg);\n color: var(--sky);\n}\n\n.faq-content .faq-body {\n padding: 0 22px 20px 46px;\n color: var(--text-dim);\n font-size: 13.5px;\n line-height: 1.7;\n display: flex;\n flex-direction: column;\n gap: 10px;\n}\n\n.faq-content .faq-body p {\n margin: 0;\n}\n\n.faq-content .faq-body ul {\n margin: 0;\n padding-left: 20px;\n display: flex;\n flex-direction: column;\n gap: 6px;\n}\n\n.faq-content .faq-body li {\n margin: 0;\n}\n\n.faq-content .faq-body b,\n.faq-content .faq-body strong {\n color: var(--mist);\n font-weight: 500;\n}\n\n.faq-content .faq-body code {\n font-family: var(--font-mono);\n font-size: 12px;\n background: rgba(155, 194, 239, .08);\n padding: 2px 6px;\n border-radius: 4px;\n color: var(--mist);\n border: 1px solid rgba(155, 194, 239, .12);\n word-break: break-word;\n}\n\n.faq-content .faq-body a {\n color: var(--blue-bright);\n text-decoration: underline;\n text-decoration-color: rgba(92, 143, 230, .40);\n text-underline-offset: 3px;\n transition: color 140ms, text-decoration-color 140ms;\n}\n\n.faq-content .faq-body a:hover {\n color: var(--mist);\n text-decoration-color: var(--sky);\n}\n\n.faq-content .faq-body table {\n width: 100%;\n border-collapse: collapse;\n margin: 4px 0;\n font-size: 13px;\n table-layout: fixed;\n}\n\n.faq-content .faq-body thead td {\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.10em;\n text-transform: uppercase;\n color: var(--text-mute);\n padding-bottom: 6px;\n border-bottom: 1px solid var(--rule);\n font-weight: 500;\n}\n\n.faq-content .faq-body td {\n padding: 9px 10px;\n border-bottom: 1px solid var(--rule-2);\n vertical-align: top;\n word-break: break-word;\n}\n\n.faq-content .faq-body tr:last-child td {\n border-bottom: 0;\n}\n\n.faq-content .faq-body td:first-child {\n color: var(--text-dim);\n width: 38%;\n}\n\n.faq-content .faq-body td:first-child code {\n font-size: 11.5px;\n}\n\n.faq-content .faq-body .formula-box {\n font-family: var(--font-mono);\n font-size: 12.5px;\n background: rgba(255, 185, 56, .06);\n padding: 12px 14px;\n border-radius: 8px;\n border: 1px dashed rgba(255, 185, 56, .30);\n color: var(--mist);\n letter-spacing: 0.02em;\n}\n\n.faq-content .faq-body .link-list {\n list-style: none;\n padding-left: 0;\n}\n\n.faq-content .faq-body .link-list li {\n padding-left: 18px;\n position: relative;\n}\n\n.faq-content .faq-body .link-list li::before {\n content: "\u203A";\n position: absolute;\n left: 0;\n color: var(--sky);\n font-family: var(--font-mono);\n}\n\n.faq-content .faq-body .warning {\n margin-top: 14px;\n padding: 12px 14px;\n background: rgba(255, 185, 56, .06);\n border: 1px solid rgba(255, 185, 56, .25);\n border-left: 3px solid var(--signal-amber);\n border-radius: 8px;\n font-size: 12.5px;\n color: var(--text-dim);\n}\n\n.faq-content .faq-body .warning .icon {\n color: var(--signal-amber);\n margin-right: 8px;\n font-weight: 500;\n}\n\n/* ============================================================\n Project dialog\n ============================================================ */\n.dialog-backdrop {\n position: fixed;\n inset: 0;\n background: rgba(4, 8, 26, .78);\n backdrop-filter: blur(10px);\n z-index: 10000;\n display: flex;\n align-items: center;\n justify-content: center;\n padding: 24px;\n animation: dlg-fade 180ms ease;\n}\n\n@keyframes dlg-fade {\n from {\n opacity: 0;\n }\n\n to {\n opacity: 1;\n }\n}\n\n.dialog {\n position: relative;\n width: 100%;\n max-width: 520px;\n background:\n radial-gradient(120% 80% at 50% 0%, rgba(44, 93, 184, .22) 0%, rgba(4, 8, 26, .20) 60%),\n linear-gradient(180deg, rgba(18, 37, 73, .88) 0%, rgba(10, 21, 48, .96) 100%);\n border: 1px solid var(--rule-hover);\n border-radius: 18px;\n padding: 28px 32px 32px;\n box-shadow:\n 0 30px 80px -20px rgba(0, 0, 0, .7),\n inset 0 1px 0 rgba(255, 255, 255, .04);\n animation: dlg-rise 220ms cubic-bezier(.2, .7, .2, 1);\n}\n\n@keyframes dlg-rise {\n from {\n opacity: 0;\n transform: translateY(8px) scale(.98);\n }\n\n to {\n opacity: 1;\n transform: translateY(0) scale(1);\n }\n}\n\n.dialog-close {\n position: absolute;\n top: 14px;\n right: 14px;\n width: 30px;\n height: 30px;\n border-radius: 50%;\n color: var(--text-mute);\n font-size: 22px;\n line-height: 1;\n display: flex;\n align-items: center;\n justify-content: center;\n transition: background 180ms, color 180ms;\n}\n\n.dialog-close:hover {\n background: rgba(155, 194, 239, .10);\n color: var(--mist);\n}\n\n.dialog-eyebrow {\n font-family: var(--font-mono);\n font-size: 10px;\n letter-spacing: 0.14em;\n text-transform: uppercase;\n color: var(--text-mute);\n margin-bottom: 10px;\n}\n\n.dialog-eyebrow em {\n font-family: var(--font-serif);\n font-style: italic;\n font-weight: 400;\n font-size: 12px;\n color: var(--sky);\n letter-spacing: 0;\n text-transform: none;\n}\n\n.dialog-name {\n font-family: var(--font-sans);\n font-weight: 500;\n font-size: 28px;\n letter-spacing: -0.025em;\n color: var(--mist);\n line-height: 1.1;\n}\n\n.dialog-path {\n font-family: var(--font-mono);\n font-size: 11px;\n color: var(--text-mute);\n margin-top: 6px;\n word-break: break-all;\n}\n\n.dialog-grid {\n display: grid;\n grid-template-columns: repeat(2, 1fr);\n gap: 18px 24px;\n margin-top: 22px;\n padding-top: 20px;\n border-top: 1px solid var(--rule-2);\n}\n\n.dg-cell {\n display: flex;\n flex-direction: column;\n gap: 4px;\n}\n\n.dg-k {\n font-family: var(--font-mono);\n font-size: 9px;\n letter-spacing: 0.16em;\n text-transform: uppercase;\n color: var(--text-mute);\n}\n\n.dg-v {\n font-family: var(--font-sans);\n font-weight: 500;\n font-size: 22px;\n letter-spacing: -0.02em;\n color: var(--mist);\n line-height: 1;\n}\n\n.dg-v.money {\n color: var(--money);\n}\n\n.dg-v-sm {\n font-size: 13px;\n font-family: var(--font-mono);\n font-weight: 400;\n color: var(--text-dim);\n letter-spacing: 0;\n}';
1279
+
1408
1280
  // src/dashboard/server.ts
1409
1281
  var FALLBACK_RANGE = 9;
1410
1282
  async function startDashboard(paths, preferredPort = 8901) {