🎯 Use case
This file lives under “components/”, which covers shared React UI pieces. On the API surface it exposes Stats — mainly types, interfaces, or factory objects. Dependencies touch React UI, bun:bundle, asciichart, and terminal styling. It composes internal code from commands, hooks, ink, keybindings, and utils (relative imports).
Generated from folder role, exports, dependency roots, and inline comments — not hand-reviewed for every path.
🧠 Inline summary
import { c as _c } from "react/compiler-runtime"; import { feature } from 'bun:bundle'; import { plot as asciichart } from 'asciichart'; import chalk from 'chalk'; import figures from 'figures';
📤 Exports (heuristic)
Stats
📚 External import roots
Package roots from from "…" (relative paths omitted).
reactbun:bundleasciichartchalkfiguresstrip-ansi
🖥️ Source preview
import { c as _c } from "react/compiler-runtime";
import { feature } from 'bun:bundle';
import { plot as asciichart } from 'asciichart';
import chalk from 'chalk';
import figures from 'figures';
import React, { Suspense, use, useCallback, useEffect, useMemo, useState } from 'react';
import stripAnsi from 'strip-ansi';
import type { CommandResultDisplay } from '../commands.js';
import { useTerminalSize } from '../hooks/useTerminalSize.js';
import { applyColor } from '../ink/colorize.js';
import { stringWidth as getStringWidth } from '../ink/stringWidth.js';
import type { Color } from '../ink/styles.js';
// eslint-disable-next-line custom-rules/prefer-use-keybindings -- raw j/k/arrow stats navigation
import { Ansi, Box, Text, useInput } from '../ink.js';
import { useKeybinding } from '../keybindings/useKeybinding.js';
import { getGlobalConfig } from '../utils/config.js';
import { formatDuration, formatNumber } from '../utils/format.js';
import { generateHeatmap } from '../utils/heatmap.js';
import { renderModelName } from '../utils/model/model.js';
import { copyAnsiToClipboard } from '../utils/screenshotClipboard.js';
import { aggregateClaudeCodeStatsForRange, type ClaudeCodeStats, type DailyModelTokens, type StatsDateRange } from '../utils/stats.js';
import { resolveThemeSetting } from '../utils/systemTheme.js';
import { getTheme, themeColorToAnsi } from '../utils/theme.js';
import { Pane } from './design-system/Pane.js';
import { Tab, Tabs, useTabHeaderFocus } from './design-system/Tabs.js';
import { Spinner } from './Spinner.js';
function formatPeakDay(dateStr: string): string {
const date = new Date(dateStr);
return date.toLocaleDateString('en-US', {
month: 'short',
day: 'numeric'
});
}
type Props = {
onClose: (result?: string, options?: {
display?: CommandResultDisplay;
}) => void;
};
type StatsResult = {
type: 'success';
data: ClaudeCodeStats;
} | {
type: 'error';
message: string;
} | {
type: 'empty';
};
const DATE_RANGE_LABELS: Record<StatsDateRange, string> = {
'7d': 'Last 7 days',
'30d': 'Last 30 days',
all: 'All time'
};
const DATE_RANGE_ORDER: StatsDateRange[] = ['all', '7d', '30d'];
function getNextDateRange(current: StatsDateRange): StatsDateRange {
const currentIndex = DATE_RANGE_ORDER.indexOf(current);
return DATE_RANGE_ORDER[(currentIndex + 1) % DATE_RANGE_ORDER.length]!;
}
/**
* Creates a stats loading promise that never rejects.
* Always loads all-time stats for the heatmap.
*/
function createAllTimeStatsPromise(): Promise<StatsResult> {
return aggregateClaudeCodeStatsForRange('all').then((data): StatsResult => {
if (!data || data.totalSessions === 0) {
return {
type: 'empty'
};
}
return {
type: 'success',
data
};
}).catch((err): StatsResult => {
const message = err instanceof Error ? err.message : 'Failed to load stats';
return {
type: 'error',
message
};
});
}
export function Stats(t0) {
const $ = _c(4);
const {
onClose
} = t0;
let t1;
if ($[0] === Symbol.for("react.memo_cache_sentinel")) {
t1 = createAllTimeStatsPromise();
$[0] = t1;
} else {
t1 = $[0];
}
const allTimePromise = t1;
let t2;
if ($[1] === Symbol.for("react.memo_cache_sentinel")) {
t2 = <Box marginTop={1}><Spinner /><Text> Loading your Claude Code stats…</Text></Box>;
$[1] = t2;
} else {
t2 = $[1];
}
let t3;
if ($[2] !== onClose) {
t3 = <Suspense fallback={t2}><StatsContent allTimePromise={allTimePromise} onClose={onClose} /></Suspense>;
$[2] = onClose;
$[3] = t3;
} else {
t3 = $[3];
}
return t3;
}
type StatsContentProps = {
allTimePromise: Promise<StatsResult>;
onClose: Props['onClose'];
};
/**
* Inner component that uses React 19's use() to read the stats promise.
* Suspends while loading all-time stats, then handles date range changes without suspending.
*/
function StatsContent(t0) {
const $ = _c(34);
const {
allTimePromise,
onClose
} = t0;
const allTimeResult = use(allTimePromise);
const [dateRange, setDateRange] = useState("all");
let t1;
if ($[0] === Symbol.for("react.memo_cache_sentinel")) {
t1 = {};
$[0] = t1;
} else {
t1 = $[0];
}
const [statsCache, setStatsCache] = useState(t1);
const [isLoadingFiltered, setIsLoadingFiltered] = useState(false);
const [activeTab, setActiveTab] = useState("Overview");
const [copyStatus, setCopyStatus] = useState(null);
let t2;
let t3;
if ($[1] !== dateRange || $[2] !== statsCache) {
t2 = () => {
if (dateRange === "all") {
return;
}
if (statsCache[dateRange]) {
return;
}
let cancelled = false;
setIsLoadingFiltered(true);
aggregateClaudeCodeStatsForRange(dateRange).then(data => {
if (!cancelled) {
setStatsCache(prev => ({
...prev,
[dateRange]: data
}));
setIsLoadingFiltered(false);
}
}).catch(() => {
if (!cancelled) {
setIsLoadingFiltered(false);
}
});
return () => {
cancelled = true;
};
};
t3 = [dateRange, statsCache];
$[1] = dateRange;
$[2] = statsCache;
$[3] = t2;
$[4] = t3;
} else {
t2 = $[3];
t3 = $[4];
}
useEffect(t2, t3);
const displayStats = dateRange === "all" ? allTimeResult.type === "success" ? allTimeResult.data : null : statsCache[dateRange] ?? (allTimeResult.type === "success" ? allTimeResult.data : null);
const allTimeStats = allTimeResult.type === "success" ? allTimeResult.data : null;
let t4;
if ($[5] !== onClose) {
t4 = () => {
onClose("Stats dialog dismissed", {
display: "system"
});
};
$[5] = onClose;
$[6] = t4;
} else {
t4 = $[6];
}
const handleClose = t4;
let t5;
if ($[7] === Symbol.for("react.memo_cache_sentinel")) {
t5 = {
context: "Confirmation"
};
$[7] = t5;
} else {
t5 = $[7];
}
useKeybinding("confirm:no", handleClose, t5);
let t6;
if ($[8] !== activeTab || $[9] !== dateRange || $[10] !== displayStats || $[11] !== onClose) {
t6 = (input, key) => {
if (key.ctrl && (input === "c" || input === "d")) {
onClose("Stats dialog dismissed", {
display: "system"
});
}
if (key.tab) {
setActiveTab(_temp);
}
if (input === "r" && !key.ctrl && !key.meta) {
setDateRange(getNextDateRange(dateRange));
}
if (key.ctrl && input === "s" && displayStats) {
handleScreenshot(displayStats, activeTab, setCopyStatus);
}
};
$[8] = activeTab;
$[9] = dateRange;
$[10] = displayStats;
$[11] = onClose;
$[12] = t6;
} else {
t6 = $[12];
}
useInput(t6);
if (allTimeResult.type === "error") {
let t7;
if ($[13] !== allTimeResult.message) {
t7 = <Box marginTop={1}><Text color="error">Failed to load stats: {allTimeResult.message}</Text></Box>;
$[13] = allTimeResult.message;
$[14] = t7;
} else {
t7 = $[14];
}
return t7;
}
if (allTimeResult.type === "empty") {
let t7;
if ($[15] === Symbol.for("react.memo_cache_sentinel")) {
t7 = <Box marginTop={1}><Text color="warning">No stats available yet. Start using Claude Code!</Text></Box>;
$[15] = t7;
} else {
t7 = $[15];
}
return t7;
}
if (!displayStats || !allTimeStats) {
let t7;
if ($[16] === Symbol.for("react.memo_cache_sentinel")) {
t7 = <Box marginTop={1}><Spinner /><Text> Loading stats…</Text></Box>;
$[16] = t7;
} else {
t7 = $[16];
}
return t7;
}
let t7;
if ($[17] !== allTimeStats || $[18] !== dateRange || $[19] !== displayStats || $[20] !== isLoadingFiltered) {
t7 = <Tab title="Overview"><OverviewTab stats={displayStats} allTimeStats={allTimeStats} dateRange={dateRange} isLoading={isLoadingFiltered} /></Tab>;
$[17] = allTimeStats;
$[18] = dateRange;
$[19] = displayStats;
$[20] = isLoadingFiltered;
$[21] = t7;
} else {
t7 = $[21];
}
let t8;
if ($[22] !== dateRange || $[23] !== displayStats || $[24] !== isLoadingFiltered) {
t8 = <Tab title="Models"><ModelsTab stats={displayStats} dateRange={dateRange} isLoading={isLoadingFiltered} /></Tab>;
$[22] = dateRange;
$[23] = displayStats;
$[24] = isLoadingFiltered;
$[25] = t8;
} else {
t8 = $[25];
}
let t9;
if ($[26] !== t7 || $[27] !== t8) {
t9 = <Box flexDirection="row" gap={1} marginBottom={1}><Tabs title="" color="claude" defaultTab="Overview">{t7}{t8}</Tabs></Box>;
$[26] = t7;
$[27] = t8;
$[28] = t9;
} else {
t9 = $[28];
}
const t10 = copyStatus ? ` · ${copyStatus}` : "";
let t11;
if ($[29] !== t10) {
t11 = <Box paddingLeft={2}><Text dimColor={true}>Esc to cancel · r to cycle dates · ctrl+s to copy{t10}</Text></Box>;
$[29] = t10;
$[30] = t11;
} else {
t11 = $[30];
}
let t12;
if ($[31] !== t11 || $[32] !== t9) {
t12 = <Pane color="claude">{t9}{t11}</Pane>;
$[31] = t11;
$[32] = t9;
$[33] = t12;
} else {
t12 = $[33];
}
return t12;
}
function _temp(prev_0) {
return prev_0 === "Overview" ? "Models" : "Overview";
}
function DateRangeSelector(t0) {
const $ = _c(9);
const {
dateRange,
isLoading
} = t0;
let t1;
if ($[0] !== dateRange) {
t1 = DATE_RANGE_ORDER.map((range, i) => <Text key={range}>{i > 0 && <Text dimColor={true}> · </Text>}{range === dateRange ? <Text bold={true} color="claude">{DATE_RANGE_LABELS[range]}</Text> : <Text dimColor={true}>{DATE_RANGE_LABELS[range]}</Text>}</Text>);
$[0] = dateRange;
$[1] = t1;
} else {
t1 = $[1];
}
let t2;
if ($[2] !== t1) {
t2 = <Box>{t1}</Box>;
$[2] = t1;
$[3] = t2;
} else {
t2 = $[3];
}
let t3;
if ($[4] !== isLoading) {
t3 = isLoading && <Spinner />;
$[4] = isLoading;
$[5] = t3;
} else {
t3 = $[5];
}
let t4;
if ($[6] !== t2 || $[7] !== t3) {
t4 = <Box marginBottom={1} gap={1}>{t2}{t3}</Box>;
$[6] = t2;
$[7] = t3;
$[8] = t4;
} else {
t4 = $[8];
}
return t4;
}
function OverviewTab({
stats,
allTimeStats,
dateRange,
isLoading
}: {
stats: ClaudeCodeStats;
allTimeStats: ClaudeCodeStats;
dateRange: StatsDateRange;
isLoading: boolean;
}): React.ReactNode {
const {
columns: terminalWidth
} = useTerminalSize();
// Calculate favorite model and total tokens
const modelEntries = Object.entries(stats.modelUsage).sort(([, a], [, b]) => b.inputTokens + b.outputTokens - (a.inputTokens + a.outputTokens));
const favoriteModel = modelEntries[0];
const totalTokens = modelEntries.reduce((sum, [, usage]) => sum + usage.inputTokens + usage.outputTokens, 0);
// Memoize the factoid so it doesn't change when switching tabs
const factoid = useMemo(() => generateFunFactoid(stats, totalTokens), [stats, totalTokens]);
// Calculate range days based on selected date range
const rangeDays = dateRange === '7d' ? 7 : dateRange === '30d' ? 30 : stats.totalDays;
// Compute shot stats data (ant-only, gated by feature flag)
let shotStatsData: {
avgShots: string;
buckets: {
label: string;
count: number;
pct: number;
}[];
} | null = null;
if (feature('SHOT_STATS') && stats.shotDistribution) {
const dist = stats.shotDistribution;
const total = Object.values(dist).reduce((s, n) => s + n, 0);
if (total > 0) {
const totalShots = Object.entries(dist).reduce((s_0, [count, sessions]) => s_0 + parseInt(count, 10) * sessions, 0);
const bucket = (min: number, max?: number) => Object.entries(dist).filter(([k]) => {
const n_0 = parseInt(k, 10);
return n_0 >= min && (max === undefined || n_0 <= max);
}).reduce((s_1, [, v]) => s_1 + v, 0);
const pct = (n_1: number) => Math.round(n_1 / total * 100);
const b1 = bucket(1, 1);
const b2_5 = bucket(2, 5);
const b6_10 = bucket(6, 10);
const b11 = bucket(11);
shotStatsData = {
avgShots: (totalShots / total).toFixed(1),
buckets: [{
label: '1-shot',
count: b1,
pct: pct(b1)
}, {
label: '2\u20135 shot',
count: b2_5,
pct: pct(b2_5)
}, {
label: '6\u201310 shot',
count: b6_10,
pct: pct(b6_10)
}, {
label: '11+ shot',
count: b11,
pct: pct(b11)
}]
};
}
}
return <Box flexDirection="column" marginTop={1}>
{/* Activity Heatmap - always shows all-time data */}
{allTimeStats.dailyActivity.length > 0 && <Box flexDirection="column" marginBottom={1}>
<Ansi>
{generateHeatmap(allTimeStats.dailyActivity, {
terminalWidth
})}
</Ansi>
</Box>}
{/* Date range selector */}
<DateRangeSelector dateRange={dateRange} isLoading={isLoading} />
{/* Section 1: Usage */}
<Box flexDirection="row" gap={4} marginBottom={1}>
<Box flexDirection="column" width={28}>
{favoriteModel && <Text wrap="truncate">
Favorite model:{' '}
<Text color="claude" bold>
{renderModelName(favoriteModel[0])}
</Text>
</Text>}
</Box>
<Box flexDirection="column" width={28}>
<Text wrap="truncate">
Total tokens:{' '}
<Text color="claude">{formatNumber(totalTokens)}</Text>
</Text>
</Box>
</Box>
{/* Section 2: Activity - Row 1: Sessions | Longest session */}
<Box flexDirection="row" gap={4}>
<Box flexDirection="column" width={28}>
<Text wrap="truncate">
Sessions:{' '}
<Text color="claude">{formatNumber(stats.totalSessions)}</Text>
</Text>
</Box>
<Box flexDirection="column" width={28}>
{stats.longestSession && <Text wrap="truncate">
Longest session:{' '}
<Text color="claude">
{formatDuration(stats.longestSession.duration)}
</Text>
</Text>}
</Box>
</Box>
{/* Row 2: Active days | Longest streak */}
<Box flexDirection="row" gap={4}>
<Box flexDirection="column" width={28}>
<Text wrap="truncate">
Active days: <Text color="claude">{stats.activeDays}</Text>
<Text color="subtle">/{rangeDays}</Text>
</Text>
</Box>
<Box flexDirection="column" width={28}>
<Text wrap="truncate">
Longest streak:{' '}
<Text color="claude" bold>
{stats.streaks.longestStreak}
</Text>{' '}
{stats.streaks.longestStreak === 1 ? 'day' : 'days'}
</Text>
</Box>
</Box>
{/* Row 3: Most active day | Current streak */}
<Box flexDirection="row" gap={4}>
<Box flexDirection="column" width={28}>
{stats.peakActivityDay && <Text wrap="truncate">
Most active day:{' '}
<Text color="claude">{formatPeakDay(stats.peakActivityDay)}</Text>
</Text>}
</Box>
<Box flexDirection="column" width={28}>
<Text wrap="truncate">
Current streak:{' '}
<Text color="claude" bold>
{allTimeStats.streaks.currentStreak}
</Text>{' '}
{allTimeStats.streaks.currentStreak === 1 ? 'day' : 'days'}
</Text>
</Box>
</Box>
{/* Speculation time saved (ant-only) */}
{"external" === 'ant' && stats.totalSpeculationTimeSavedMs > 0 && <Box flexDirection="row" gap={4}>
<Box flexDirection="column" width={28}>
<Text wrap="truncate">
Speculation saved:{' '}
<Text color="claude">
{formatDuration(stats.totalSpeculationTimeSavedMs)}
</Text>
</Text>
</Box>
</Box>}
{/* Shot stats (ant-only) */}
{shotStatsData && <>
<Box marginTop={1}>
<Text>Shot distribution</Text>
</Box>
<Box flexDirection="row" gap={4}>
<Box flexDirection="column" width={28}>
<Text wrap="truncate">
{shotStatsData.buckets[0]!.label}:{' '}
<Text color="claude">{shotStatsData.buckets[0]!.count}</Text>
<Text color="subtle"> ({shotStatsData.buckets[0]!.pct}%)</Text>
</Text>
</Box>
<Box flexDirection="column" width={28}>
<Text wrap="truncate">
{shotStatsData.buckets[1]!.label}:{' '}
<Text color="claude">{shotStatsData.buckets[1]!.count}</Text>
<Text color="subtle"> ({shotStatsData.buckets[1]!.pct}%)</Text>
</Text>
</Box>
</Box>
<Box flexDirection="row" gap={4}>
<Box flexDirection="column" width={28}>
<Text wrap="truncate">
{shotStatsData.buckets[2]!.label}:{' '}
<Text color="claude">{shotStatsData.buckets[2]!.count}</Text>
<Text color="subtle"> ({shotStatsData.buckets[2]!.pct}%)</Text>
</Text>
</Box>
<Box flexDirection="column" width={28}>
<Text wrap="truncate">
{shotStatsData.buckets[3]!.label}:{' '}
<Text color="claude">{shotStatsData.buckets[3]!.count}</Text>
<Text color="subtle"> ({shotStatsData.buckets[3]!.pct}%)</Text>
</Text>
</Box>
</Box>
<Box flexDirection="row" gap={4}>
<Box flexDirection="column" width={28}>
<Text wrap="truncate">
Avg/session:{' '}
<Text color="claude">{shotStatsData.avgShots}</Text>
</Text>
</Box>
</Box>
</>}
{/* Fun factoid */}
{factoid && <Box marginTop={1}>
<Text color="suggestion">{factoid}</Text>
</Box>}
</Box>;
}
// Famous books and their approximate token counts (words * ~1.3)
// Sorted by tokens ascending for comparison logic
const BOOK_COMPARISONS = [{
name: 'The Little Prince',
tokens: 22000
}, {
name: 'The Old Man and the Sea',
tokens: 35000
}, {
name: 'A Christmas Carol',
tokens: 37000
}, {
name: 'Animal Farm',
tokens: 39000
}, {
name: 'Fahrenheit 451',
tokens: 60000
}, {
name: 'The Great Gatsby',
tokens: 62000
}, {
name: 'Slaughterhouse-Five',
tokens: 64000
}, {
name: 'Brave New World',
tokens: 83000
}, {
name: 'The Catcher in the Rye',
tokens: 95000
}, {
name: "Harry Potter and the Philosopher's Stone",
tokens: 103000
}, {
name: 'The Hobbit',
tokens: 123000
}, {
name: '1984',
tokens: 123000
}, {
name: 'To Kill a Mockingbird',
tokens: 130000
}, {
name: 'Pride and Prejudice',
tokens: 156000
}, {
name: 'Dune',
tokens: 244000
}, {
name: 'Moby-Dick',
tokens: 268000
}, {
name: 'Crime and Punishment',
tokens: 274000
}, {
name: 'A Game of Thrones',
tokens: 381000
}, {
name: 'Anna Karenina',
tokens: 468000
}, {
name: 'Don Quixote',
tokens: 520000
}, {
name: 'The Lord of the Rings',
tokens: 576000
}, {
name: 'The Count of Monte Cristo',
tokens: 603000
}, {
name: 'Les Misérables',
tokens: 689000
}, {
name: 'War and Peace',
tokens: 730000
}];
// Time equivalents for session durations
const TIME_COMPARISONS = [{
name: 'a TED talk',
minutes: 18
}, {
name: 'an episode of The Office',
minutes: 22
}, {
name: 'listening to Abbey Road',
minutes: 47
}, {
name: 'a yoga class',
minutes: 60
}, {
name: 'a World Cup soccer match',
minutes: 90
}, {
name: 'a half marathon (average time)',
minutes: 120
}, {
name: 'the movie Inception',
minutes: 148
}, {
name: 'watching Titanic',
minutes: 195
}, {
name: 'a transatlantic flight',
minutes: 420
}, {
name: 'a full night of sleep',
minutes: 480
}];
function generateFunFactoid(stats: ClaudeCodeStats, totalTokens: number): string {
const factoids: string[] = [];
if (totalTokens > 0) {
const matchingBooks = BOOK_COMPARISONS.filter(book => totalTokens >= book.tokens);
for (const book of matchingBooks) {
const times = totalTokens / book.tokens;
if (times >= 2) {
factoids.push(`You've used ~${Math.floor(times)}x more tokens than ${book.name}`);
} else {
factoids.push(`You've used the same number of tokens as ${book.name}`);
}
}
}
if (stats.longestSession) {
const sessionMinutes = stats.longestSession.duration / (1000 * 60);
for (const comparison of TIME_COMPARISONS) {
const ratio = sessionMinutes / comparison.minutes;
if (ratio >= 2) {
factoids.push(`Your longest session is ~${Math.floor(ratio)}x longer than ${comparison.name}`);
}
}
}
if (factoids.length === 0) {
return '';
}
const randomIndex = Math.floor(Math.random() * factoids.length);
return factoids[randomIndex]!;
}
function ModelsTab(t0) {
const $ = _c(15);
const {
stats,
dateRange,
isLoading
} = t0;
const {
headerFocused,
focusHeader
} = useTabHeaderFocus();
const [scrollOffset, setScrollOffset] = useState(0);
const {
columns: terminalWidth
} = useTerminalSize();
const modelEntries = Object.entries(stats.modelUsage).sort(_temp7);
const t1 = !headerFocused;
let t2;
if ($[0] !== t1) {
t2 = {
isActive: t1
};
$[0] = t1;
$[1] = t2;
} else {
t2 = $[1];
}
useInput((_input, key) => {
if (key.downArrow && scrollOffset < modelEntries.length - 4) {
setScrollOffset(prev => Math.min(prev + 2, modelEntries.length - 4));
}
if (key.upArrow) {
if (scrollOffset > 0) {
setScrollOffset(_temp8);
} else {
focusHeader();
}
}
}, t2);
if (modelEntries.length === 0) {
let t3;
if ($[2] === Symbol.for("react.memo_cache_sentinel")) {
t3 = <Box><Text color="subtle">No model usage data available</Text></Box>;
$[2] = t3;
} else {
t3 = $[2];
}
return t3;
}
const totalTokens = modelEntries.reduce(_temp9, 0);
const chartOutput = generateTokenChart(stats.dailyModelTokens, modelEntries.map(_temp0), terminalWidth);
const visibleModels = modelEntries.slice(scrollOffset, scrollOffset + 4);
const midpoint = Math.ceil(visibleModels.length / 2);
const leftModels = visibleModels.slice(0, midpoint);
const rightModels = visibleModels.slice(midpoint);
const canScrollUp = scrollOffset > 0;
const canScrollDown = scrollOffset < modelEntries.length - 4;
const showScrollHint = modelEntries.length > 4;
let t3;
if ($[3] !== dateRange || $[4] !== isLoading) {
t3 = <DateRangeSelector dateRange={dateRange} isLoading={isLoading} />;
$[3] = dateRange;
$[4] = isLoading;
$[5] = t3;
} else {
t3 = $[5];
}
const T0 = Box;
const t5 = "column";
const t6 = 36;
const t8 = rightModels.map(t7 => {
const [model_1, usage_1] = t7;
return <ModelEntry key={model_1} model={model_1} usage={usage_1} totalTokens={totalTokens} />;
});
let t9;
if ($[6] !== T0 || $[7] !== t8) {
t9 = <T0 flexDirection={t5} width={t6}>{t8}</T0>;
$[6] = T0;
$[7] = t8;
$[8] = t9;
} else {
t9 = $[8];
}
let t10;
if ($[9] !== canScrollDown || $[10] !== canScrollUp || $[11] !== modelEntries || $[12] !== scrollOffset || $[13] !== showScrollHint) {
t10 = showScrollHint && <Box marginTop={1}><Text color="subtle">{canScrollUp ? figures.arrowUp : " "}{" "}{canScrollDown ? figures.arrowDown : " "} {scrollOffset + 1}-{Math.min(scrollOffset + 4, modelEntries.length)} of{" "}{modelEntries.length} models (↑↓ to scroll)</Text></Box>;
$[9] = canScrollDown;
$[10] = canScrollUp;
$[11] = modelEntries;
$[12] = scrollOffset;
$[13] = showScrollHint;
$[14] = t10;
} else {
t10 = $[14];
}
return <Box flexDirection="column" marginTop={1}>{chartOutput && <Box flexDirection="column" marginBottom={1}><Text bold={true}>Tokens per Day</Text><Ansi>{chartOutput.chart}</Ansi><Text color="subtle">{chartOutput.xAxisLabels}</Text><Box>{chartOutput.legend.map(_temp1)}</Box></Box>}{t3}<Box flexDirection="row" gap={4}><Box flexDirection="column" width={36}>{leftModels.map(t4 => {
const [model_0, usage_0] = t4;
return <ModelEntry key={model_0} model={model_0} usage={usage_0} totalTokens={totalTokens} />;
})}</Box>{t9}</Box>{t10}</Box>;
}
function _temp1(item, i) {
return <Text key={item.model}>{i > 0 ? " \xB7 " : ""}<Ansi>{item.coloredBullet}</Ansi> {item.model}</Text>;
}
function _temp0(t0) {
const [model] = t0;
return model;
}
function _temp9(sum, t0) {
const [, usage] = t0;
return sum + usage.inputTokens + usage.outputTokens;
}
function _temp8(prev_0) {
return Math.max(prev_0 - 2, 0);
}
function _temp7(t0, t1) {
const [, a] = t0;
const [, b] = t1;
return b.inputTokens + b.outputTokens - (a.inputTokens + a.outputTokens);
}
type ModelEntryProps = {
model: string;
usage: {
inputTokens: number;
outputTokens: number;
cacheReadInputTokens: number;
};
totalTokens: number;
};
function ModelEntry(t0) {
const $ = _c(21);
const {
model,
usage,
totalTokens
} = t0;
const modelTokens = usage.inputTokens + usage.outputTokens;
const t1 = modelTokens / totalTokens * 100;
let t2;
if ($[0] !== t1) {
t2 = t1.toFixed(1);
$[0] = t1;
$[1] = t2;
} else {
t2 = $[1];
}
const percentage = t2;
let t3;
if ($[2] !== model) {
t3 = renderModelName(model);
$[2] = model;
$[3] = t3;
} else {
t3 = $[3];
}
let t4;
if ($[4] !== t3) {
t4 = <Text bold={true}>{t3}</Text>;
$[4] = t3;
$[5] = t4;
} else {
t4 = $[5];
}
let t5;
if ($[6] !== percentage) {
t5 = <Text color="subtle">({percentage}%)</Text>;
$[6] = percentage;
$[7] = t5;
} else {
t5 = $[7];
}
let t6;
if ($[8] !== t4 || $[9] !== t5) {
t6 = <Text>{figures.bullet} {t4}{" "}{t5}</Text>;
$[8] = t4;
$[9] = t5;
$[10] = t6;
} else {
t6 = $[10];
}
let t7;
if ($[11] !== usage.inputTokens) {
t7 = formatNumber(usage.inputTokens);
$[11] = usage.inputTokens;
$[12] = t7;
} else {
t7 = $[12];
}
let t8;
if ($[13] !== usage.outputTokens) {
t8 = formatNumber(usage.outputTokens);
$[13] = usage.outputTokens;
$[14] = t8;
} else {
t8 = $[14];
}
let t9;
if ($[15] !== t7 || $[16] !== t8) {
t9 = <Text color="subtle">{" "}In: {t7} · Out:{" "}{t8}</Text>;
$[15] = t7;
$[16] = t8;
$[17] = t9;
} else {
t9 = $[17];
}
let t10;
if ($[18] !== t6 || $[19] !== t9) {
t10 = <Box flexDirection="column">{t6}{t9}</Box>;
$[18] = t6;
$[19] = t9;
$[20] = t10;
} else {
t10 = $[20];
}
return t10;
}
type ChartLegend = {
model: string;
coloredBullet: string; // Pre-colored bullet using chalk
};
type ChartOutput = {
chart: string;
legend: ChartLegend[];
xAxisLabels: string;
};
function generateTokenChart(dailyTokens: DailyModelTokens[], models: string[], terminalWidth: number): ChartOutput | null {
if (dailyTokens.length < 2 || models.length === 0) {
return null;
}
// Y-axis labels take about 6 characters, plus some padding
// Cap at ~52 to align with heatmap width (1 year of data)
const yAxisWidth = 7;
const availableWidth = terminalWidth - yAxisWidth;
const chartWidth = Math.min(52, Math.max(20, availableWidth));
// Distribute data across the available chart width
let recentData: DailyModelTokens[];
if (dailyTokens.length >= chartWidth) {
// More data than space: take most recent N days
recentData = dailyTokens.slice(-chartWidth);
} else {
// Less data than space: expand by repeating each point
const repeatCount = Math.floor(chartWidth / dailyTokens.length);
recentData = [];
for (const day of dailyTokens) {
for (let i = 0; i < repeatCount; i++) {
recentData.push(day);
}
}
}
// Color palette for different models - use theme colors
const theme = getTheme(resolveThemeSetting(getGlobalConfig().theme));
const colors = [themeColorToAnsi(theme.suggestion), themeColorToAnsi(theme.success), themeColorToAnsi(theme.warning)];
// Prepare series data for each model
const series: number[][] = [];
const legend: ChartLegend[] = [];
// Only show top 3 models to keep chart readable
const topModels = models.slice(0, 3);
for (let i = 0; i < topModels.length; i++) {
const model = topModels[i]!;
const data = recentData.map(day => day.tokensByModel[model] || 0);
// Only include if there's actual data
if (data.some(v => v > 0)) {
series.push(data);
// Use theme colors that match the chart
const bulletColors = [theme.suggestion, theme.success, theme.warning];
legend.push({
model: renderModelName(model),
coloredBullet: applyColor(figures.bullet, bulletColors[i % bulletColors.length] as Color)
});
}
}
if (series.length === 0) {
return null;
}
const chart = asciichart(series, {
height: 8,
colors: colors.slice(0, series.length),
format: (x: number) => {
let label: string;
if (x >= 1_000_000) {
label = (x / 1_000_000).toFixed(1) + 'M';
} else if (x >= 1_000) {
label = (x / 1_000).toFixed(0) + 'k';
} else {
label = x.toFixed(0);
}
return label.padStart(6);
}
});
// Generate x-axis labels with dates
const xAxisLabels = generateXAxisLabels(recentData, recentData.length, yAxisWidth);
return {
chart,
legend,
xAxisLabels
};
}
function generateXAxisLabels(data: DailyModelTokens[], _chartWidth: number, yAxisOffset: number): string {
if (data.length === 0) return '';
// Show 3-4 date labels evenly spaced, but leave room for last label
const numLabels = Math.min(4, Math.max(2, Math.floor(data.length / 8)));
// Don't use the very last position - leave room for the label text
const usableLength = data.length - 6; // Reserve ~6 chars for last label (e.g., "Dec 7")
const step = Math.floor(usableLength / (numLabels - 1)) || 1;
const labelPositions: {
pos: number;
label: string;
}[] = [];
for (let i = 0; i < numLabels; i++) {
const idx = Math.min(i * step, data.length - 1);
const date = new Date(data[idx]!.date);
const label = date.toLocaleDateString('en-US', {
month: 'short',
day: 'numeric'
});
labelPositions.push({
pos: idx,
label
});
}
// Build the label string with proper spacing
let result = ' '.repeat(yAxisOffset);
let currentPos = 0;
for (const {
pos,
label
} of labelPositions) {
const spaces = Math.max(1, pos - currentPos);
result += ' '.repeat(spaces) + label;
currentPos = pos + label.length;
}
return result;
}
// Screenshot functionality
async function handleScreenshot(stats: ClaudeCodeStats, activeTab: 'Overview' | 'Models', setStatus: (status: string | null) => void): Promise<void> {
setStatus('copying…');
const ansiText = renderStatsToAnsi(stats, activeTab);
const result = await copyAnsiToClipboard(ansiText);
setStatus(result.success ? 'copied!' : 'copy failed');
// Clear status after 2 seconds
setTimeout(setStatus, 2000, null);
}
function renderStatsToAnsi(stats: ClaudeCodeStats, activeTab: 'Overview' | 'Models'): string {
const lines: string[] = [];
if (activeTab === 'Overview') {
lines.push(...renderOverviewToAnsi(stats));
} else {
lines.push(...renderModelsToAnsi(stats));
}
// Trim trailing empty lines
while (lines.length > 0 && stripAnsi(lines[lines.length - 1]!).trim() === '') {
lines.pop();
}
// Add "/stats" right-aligned on the last line
if (lines.length > 0) {
const lastLine = lines[lines.length - 1]!;
const lastLineLen = getStringWidth(lastLine);
// Use known content widths based on layout:
// Overview: two-column stats = COL2_START(40) + COL2_LABEL_WIDTH(18) + max_value(~12) = 70
// Models: chart width = 80
const contentWidth = activeTab === 'Overview' ? 70 : 80;
const statsLabel = '/stats';
const padding = Math.max(2, contentWidth - lastLineLen - statsLabel.length);
lines[lines.length - 1] = lastLine + ' '.repeat(padding) + chalk.gray(statsLabel);
}
return lines.join('\n');
}
function renderOverviewToAnsi(stats: ClaudeCodeStats): string[] {
const lines: string[] = [];
const theme = getTheme(resolveThemeSetting(getGlobalConfig().theme));
const h = (text: string) => applyColor(text, theme.claude as Color);
// Two-column helper with fixed spacing
// Column 1: label (18 chars) + value + padding to reach col 2
// Column 2 starts at character position 40
const COL1_LABEL_WIDTH = 18;
const COL2_START = 40;
const COL2_LABEL_WIDTH = 18;
const row = (l1: string, v1: string, l2: string, v2: string): string => {
// Build column 1: label + value
const label1 = (l1 + ':').padEnd(COL1_LABEL_WIDTH);
const col1PlainLen = label1.length + v1.length;
// Calculate spaces needed between col1 value and col2 label
const spaceBetween = Math.max(2, COL2_START - col1PlainLen);
// Build column 2: label + value
const label2 = (l2 + ':').padEnd(COL2_LABEL_WIDTH);
// Assemble with colors applied to values only
return label1 + h(v1) + ' '.repeat(spaceBetween) + label2 + h(v2);
};
// Heatmap - use fixed width for screenshot (56 = 52 weeks + 4 for day labels)
if (stats.dailyActivity.length > 0) {
lines.push(generateHeatmap(stats.dailyActivity, {
terminalWidth: 56
}));
lines.push('');
}
// Calculate values
const modelEntries = Object.entries(stats.modelUsage).sort(([, a], [, b]) => b.inputTokens + b.outputTokens - (a.inputTokens + a.outputTokens));
const favoriteModel = modelEntries[0];
const totalTokens = modelEntries.reduce((sum, [, usage]) => sum + usage.inputTokens + usage.outputTokens, 0);
// Row 1: Favorite model | Total tokens
if (favoriteModel) {
lines.push(row('Favorite model', renderModelName(favoriteModel[0]), 'Total tokens', formatNumber(totalTokens)));
}
lines.push('');
// Row 2: Sessions | Longest session
lines.push(row('Sessions', formatNumber(stats.totalSessions), 'Longest session', stats.longestSession ? formatDuration(stats.longestSession.duration) : 'N/A'));
// Row 3: Current streak | Longest streak
const currentStreakVal = `${stats.streaks.currentStreak} ${stats.streaks.currentStreak === 1 ? 'day' : 'days'}`;
const longestStreakVal = `${stats.streaks.longestStreak} ${stats.streaks.longestStreak === 1 ? 'day' : 'days'}`;
lines.push(row('Current streak', currentStreakVal, 'Longest streak', longestStreakVal));
// Row 4: Active days | Peak hour
const activeDaysVal = `${stats.activeDays}/${stats.totalDays}`;
const peakHourVal = stats.peakActivityHour !== null ? `${stats.peakActivityHour}:00-${stats.peakActivityHour + 1}:00` : 'N/A';
lines.push(row('Active days', activeDaysVal, 'Peak hour', peakHourVal));
// Speculation time saved (ant-only)
if ("external" === 'ant' && stats.totalSpeculationTimeSavedMs > 0) {
const label = 'Speculation saved:'.padEnd(COL1_LABEL_WIDTH);
lines.push(label + h(formatDuration(stats.totalSpeculationTimeSavedMs)));
}
// Shot stats (ant-only)
if (feature('SHOT_STATS') && stats.shotDistribution) {
const dist = stats.shotDistribution;
const totalWithShots = Object.values(dist).reduce((s, n) => s + n, 0);
if (totalWithShots > 0) {
const totalShots = Object.entries(dist).reduce((s, [count, sessions]) => s + parseInt(count, 10) * sessions, 0);
const avgShots = (totalShots / totalWithShots).toFixed(1);
const bucket = (min: number, max?: number) => Object.entries(dist).filter(([k]) => {
const n = parseInt(k, 10);
return n >= min && (max === undefined || n <= max);
}).reduce((s, [, v]) => s + v, 0);
const pct = (n: number) => Math.round(n / totalWithShots * 100);
const fmtBucket = (count: number, p: number) => `${count} (${p}%)`;
const b1 = bucket(1, 1);
const b2_5 = bucket(2, 5);
const b6_10 = bucket(6, 10);
const b11 = bucket(11);
lines.push('');
lines.push('Shot distribution');
lines.push(row('1-shot', fmtBucket(b1, pct(b1)), '2\u20135 shot', fmtBucket(b2_5, pct(b2_5))));
lines.push(row('6\u201310 shot', fmtBucket(b6_10, pct(b6_10)), '11+ shot', fmtBucket(b11, pct(b11))));
lines.push(`${'Avg/session:'.padEnd(COL1_LABEL_WIDTH)}${h(avgShots)}`);
}
}
lines.push('');
// Fun factoid
const factoid = generateFunFactoid(stats, totalTokens);
lines.push(h(factoid));
lines.push(chalk.gray(`Stats from the last ${stats.totalDays} days`));
return lines;
}
function renderModelsToAnsi(stats: ClaudeCodeStats): string[] {
const lines: string[] = [];
const modelEntries = Object.entries(stats.modelUsage).sort(([, a], [, b]) => b.inputTokens + b.outputTokens - (a.inputTokens + a.outputTokens));
if (modelEntries.length === 0) {
lines.push(chalk.gray('No model usage data available'));
return lines;
}
const favoriteModel = modelEntries[0];
const totalTokens = modelEntries.reduce((sum, [, usage]) => sum + usage.inputTokens + usage.outputTokens, 0);
// Generate chart if we have data - use fixed width for screenshot
const chartOutput = generateTokenChart(stats.dailyModelTokens, modelEntries.map(([model]) => model), 80 // Fixed width for screenshot
);
if (chartOutput) {
lines.push(chalk.bold('Tokens per Day'));
lines.push(chartOutput.chart);
lines.push(chalk.gray(chartOutput.xAxisLabels));
// Legend - use pre-colored bullets from chart output
const legendLine = chartOutput.legend.map(item => `${item.coloredBullet} ${item.model}`).join(' · ');
lines.push(legendLine);
lines.push('');
}
// Summary
lines.push(`${figures.star} Favorite: ${chalk.magenta.bold(renderModelName(favoriteModel?.[0] || ''))} · ${figures.circle} Total: ${chalk.magenta(formatNumber(totalTokens))} tokens`);
lines.push('');
// Model breakdown - only show top 3 for screenshot
const topModels = modelEntries.slice(0, 3);
for (const [model, usage] of topModels) {
const modelTokens = usage.inputTokens + usage.outputTokens;
const percentage = (modelTokens / totalTokens * 100).toFixed(1);
lines.push(`${figures.bullet} ${chalk.bold(renderModelName(model))} ${chalk.gray(`(${percentage}%)`)}`);
lines.push(chalk.dim(` In: ${formatNumber(usage.inputTokens)} · Out: ${formatNumber(usage.outputTokens)}`));
}
return lines;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["feature","plot","asciichart","chalk","figures","React","Suspense","use","useCallback","useEffect","useMemo","useState","stripAnsi","CommandResultDisplay","useTerminalSize","applyColor","stringWidth","getStringWidth","Color","Ansi","Box","Text","useInput","useKeybinding","getGlobalConfig","formatDuration","formatNumber","generateHeatmap","renderModelName","copyAnsiToClipboard","aggregateClaudeCodeStatsForRange","ClaudeCodeStats","DailyModelTokens","StatsDateRange","resolveThemeSetting","getTheme","themeColorToAnsi","Pane","Tab","Tabs","useTabHeaderFocus","Spinner","formatPeakDay","dateStr","date","Date","toLocaleDateString","month","day","Props","onClose","result","options","display","StatsResult","type","data","message","DATE_RANGE_LABELS","Record","all","DATE_RANGE_ORDER","getNextDateRange","current","currentIndex","indexOf","length","createAllTimeStatsPromise","Promise","then","totalSessions","catch","err","Error","Stats","t0","$","_c","t1","Symbol","for","allTimePromise","t2","t3","StatsContentProps","StatsContent","allTimeResult","dateRange","setDateRange","statsCache","setStatsCache","isLoadingFiltered","setIsLoadingFiltered","activeTab","setActiveTab","copyStatus","setCopyStatus","cancelled","prev","displayStats","allTimeStats","t4","handleClose","t5","context","t6","input","key","ctrl","tab","_temp","meta","handleScreenshot","t7","t8","t9","t10","t11","t12","prev_0","DateRangeSelector","isLoading","map","range","i","OverviewTab","stats","ReactNode","columns","terminalWidth","modelEntries","Object","entries","modelUsage","sort","a","b","inputTokens","outputTokens","favoriteModel","totalTokens","reduce","sum","usage","factoid","generateFunFactoid","rangeDays","totalDays","shotStatsData","avgShots","buckets","label","count","pct","shotDistribution","dist","total","values","s","n","totalShots","sessions","parseInt","bucket","min","max","filter","k","undefined","v","Math","round","b1","b2_5","b6_10","b11","toFixed","dailyActivity","longestSession","duration","activeDays","streaks","longestStreak","peakActivityDay","currentStreak","totalSpeculationTimeSavedMs","BOOK_COMPARISONS","name","tokens","TIME_COMPARISONS","minutes","factoids","matchingBooks","book","times","push","floor","sessionMinutes","comparison","ratio","randomIndex","random","ModelsTab","headerFocused","focusHeader","scrollOffset","setScrollOffset","_temp7","isActive","_input","downArrow","upArrow","_temp8","_temp9","chartOutput","generateTokenChart","dailyModelTokens","_temp0","visibleModels","slice","midpoint","ceil","leftModels","rightModels","canScrollUp","canScrollDown","showScrollHint","T0","model_1","usage_1","model","arrowUp","arrowDown","chart","xAxisLabels","legend","_temp1","model_0","usage_0","item","coloredBullet","ModelEntryProps","cacheReadInputTokens","ModelEntry","modelTokens","percentage","bullet","ChartLegend","ChartOutput","dailyTokens","models","yAxisWidth","availableWidth","chartWidth","recentData","repeatCount","theme","colors","suggestion","success","warning","series","topModels","tokensByModel","some","bulletColors","height","format","x","padStart","generateXAxisLabels","_chartWidth","yAxisOffset","numLabels","usableLength","step","labelPositions","pos","idx","repeat","currentPos","spaces","setStatus","status","ansiText","renderStatsToAnsi","setTimeout","lines","renderOverviewToAnsi","renderModelsToAnsi","trim","pop","lastLine","lastLineLen","contentWidth","statsLabel","padding","gray","join","h","text","claude","COL1_LABEL_WIDTH","COL2_START","COL2_LABEL_WIDTH","row","l1","v1","l2","v2","label1","padEnd","col1PlainLen","spaceBetween","label2","currentStreakVal","longestStreakVal","activeDaysVal","peakHourVal","peakActivityHour","totalWithShots","fmtBucket","p","bold","legendLine","star","magenta","circle","dim"],"sources":["Stats.tsx"],"sourcesContent":["import { feature } from 'bun:bundle'\nimport { plot as asciichart } from 'asciichart'\nimport chalk from 'chalk'\nimport figures from 'figures'\nimport React, {\n  Suspense,\n  use,\n  useCallback,\n  useEffect,\n  useMemo,\n  useState,\n} from 'react'\nimport stripAnsi from 'strip-ansi'\nimport type { CommandResultDisplay } from '../commands.js'\nimport { useTerminalSize } from '../hooks/useTerminalSize.js'\nimport { applyColor } from '../ink/colorize.js'\nimport { stringWidth as getStringWidth } from '../ink/stringWidth.js'\nimport type { Color } from '../ink/styles.js'\n// eslint-disable-next-line custom-rules/prefer-use-keybindings -- raw j/k/arrow stats navigation\nimport { Ansi, Box, Text, useInput } from '../ink.js'\nimport { useKeybinding } from '../keybindings/useKeybinding.js'\nimport { getGlobalConfig } from '../utils/config.js'\nimport { formatDuration, formatNumber } from '../utils/format.js'\nimport { generateHeatmap } from '../utils/heatmap.js'\nimport { renderModelName } from '../utils/model/model.js'\nimport { copyAnsiToClipboard } from '../utils/screenshotClipboard.js'\nimport {\n  aggregateClaudeCodeStatsForRange,\n  type ClaudeCodeStats,\n  type DailyModelTokens,\n  type StatsDateRange,\n} from '../utils/stats.js'\nimport { resolveThemeSetting } from '../utils/systemTheme.js'\nimport { getTheme, themeColorToAnsi } from '../utils/theme.js'\nimport { Pane } from './design-system/Pane.js'\nimport { Tab, Tabs, useTabHeaderFocus } from './design-system/Tabs.js'\nimport { Spinner } from './Spinner.js'\n\nfunction formatPeakDay(dateStr: string): string {\n  const date = new Date(dateStr)\n  return date.toLocaleDateString('en-US', {\n    month: 'short',\n    day: 'numeric',\n  })\n}\n\ntype Props = {\n  onClose: (\n    result?: string,\n    options?: { display?: CommandResultDisplay },\n  ) => void\n}\n\ntype StatsResult =\n  | { type: 'success'; data: ClaudeCodeStats }\n  | { type: 'error'; message: string }\n  | { type: 'empty' }\n\nconst DATE_RANGE_LABELS: Record<StatsDateRange, string> = {\n  '7d': 'Last 7 days',\n  '30d': 'Last 30 days',\n  all: 'All time',\n}\n\nconst DATE_RANGE_ORDER: StatsDateRange[] = ['all', '7d', '30d']\n\nfunction getNextDateRange(current: StatsDateRange): StatsDateRange {\n  const currentIndex = DATE_RANGE_ORDER.indexOf(current)\n  return DATE_RANGE_ORDER[(currentIndex + 1) % DATE_RANGE_ORDER.length]!\n}\n\n/**\n * Creates a stats loading promise that never rejects.\n * Always loads all-time stats for the heatmap.\n */\nfunction createAllTimeStatsPromise(): Promise<StatsResult> {\n  return aggregateClaudeCodeStatsForRange('all')\n    .then((data): StatsResult => {\n      if (!data || data.totalSessions === 0) {\n        return { type: 'empty' }\n      }\n      return { type: 'success', data }\n    })\n    .catch((err): StatsResult => {\n      const message =\n        err instanceof Error ? err.message : 'Failed to load stats'\n      return { type: 'error', message }\n    })\n}\n\nexport function Stats({ onClose }: Props): React.ReactNode {\n  // Always load all-time stats first (for heatmap)\n  const allTimePromise = useMemo(() => createAllTimeStatsPromise(), [])\n\n  return (\n    <Suspense\n      fallback={\n        <Box marginTop={1}>\n          <Spinner />\n          <Text> Loading your Claude Code stats…</Text>\n        </Box>\n      }\n    >\n      <StatsContent allTimePromise={allTimePromise} onClose={onClose} />\n    </Suspense>\n  )\n}\n\ntype StatsContentProps = {\n  allTimePromise: Promise<StatsResult>\n  onClose: Props['onClose']\n}\n\n/**\n * Inner component that uses React 19's use() to read the stats promise.\n * Suspends while loading all-time stats, then handles date range changes without suspending.\n */\nfunction StatsContent({\n  allTimePromise,\n  onClose,\n}: StatsContentProps): React.ReactNode {\n  const allTimeResult = use(allTimePromise)\n  const [dateRange, setDateRange] = useState<StatsDateRange>('all')\n  const [statsCache, setStatsCache] = useState<\n    Partial<Record<StatsDateRange, ClaudeCodeStats>>\n  >({})\n  const [isLoadingFiltered, setIsLoadingFiltered] = useState(false)\n  const [activeTab, setActiveTab] = useState<'Overview' | 'Models'>('Overview')\n  const [copyStatus, setCopyStatus] = useState<string | null>(null)\n\n  // Load filtered stats when date range changes (with caching)\n  useEffect(() => {\n    if (dateRange === 'all') {\n      return\n    }\n\n    // Already cached\n    if (statsCache[dateRange]) {\n      return\n    }\n\n    let cancelled = false\n    setIsLoadingFiltered(true)\n\n    aggregateClaudeCodeStatsForRange(dateRange)\n      .then(data => {\n        if (!cancelled) {\n          setStatsCache(prev => ({ ...prev, [dateRange]: data }))\n          setIsLoadingFiltered(false)\n        }\n      })\n      .catch(() => {\n        if (!cancelled) {\n          setIsLoadingFiltered(false)\n        }\n      })\n\n    return () => {\n      cancelled = true\n    }\n  }, [dateRange, statsCache])\n\n  // Use cached stats for current range\n  const displayStats =\n    dateRange === 'all'\n      ? allTimeResult.type === 'success'\n        ? allTimeResult.data\n        : null\n      : (statsCache[dateRange] ??\n        (allTimeResult.type === 'success' ? allTimeResult.data : null))\n\n  // All-time stats for the heatmap (always use all-time)\n  const allTimeStats =\n    allTimeResult.type === 'success' ? allTimeResult.data : null\n\n  const handleClose = useCallback(() => {\n    onClose('Stats dialog dismissed', { display: 'system' })\n  }, [onClose])\n\n  useKeybinding('confirm:no', handleClose, { context: 'Confirmation' })\n\n  useInput((input, key) => {\n    // Handle ctrl+c and ctrl+d for closing\n    if (key.ctrl && (input === 'c' || input === 'd')) {\n      onClose('Stats dialog dismissed', { display: 'system' })\n    }\n    // Track tab changes\n    if (key.tab) {\n      setActiveTab(prev => (prev === 'Overview' ? 'Models' : 'Overview'))\n    }\n    // r to cycle date range\n    if (input === 'r' && !key.ctrl && !key.meta) {\n      setDateRange(getNextDateRange(dateRange))\n    }\n    // Ctrl+S to copy screenshot to clipboard\n    if (key.ctrl && input === 's' && displayStats) {\n      void handleScreenshot(displayStats, activeTab, setCopyStatus)\n    }\n  })\n\n  if (allTimeResult.type === 'error') {\n    return (\n      <Box marginTop={1}>\n        <Text color=\"error\">Failed to load stats: {allTimeResult.message}</Text>\n      </Box>\n    )\n  }\n\n  if (allTimeResult.type === 'empty') {\n    return (\n      <Box marginTop={1}>\n        <Text color=\"warning\">\n          No stats available yet. Start using Claude Code!\n        </Text>\n      </Box>\n    )\n  }\n\n  if (!displayStats || !allTimeStats) {\n    return (\n      <Box marginTop={1}>\n        <Spinner />\n        <Text> Loading stats…</Text>\n      </Box>\n    )\n  }\n\n  return (\n    <Pane color=\"claude\">\n      <Box flexDirection=\"row\" gap={1} marginBottom={1}>\n        <Tabs title=\"\" color=\"claude\" defaultTab=\"Overview\">\n          <Tab title=\"Overview\">\n            <OverviewTab\n              stats={displayStats}\n              allTimeStats={allTimeStats}\n              dateRange={dateRange}\n              isLoading={isLoadingFiltered}\n            />\n          </Tab>\n          <Tab title=\"Models\">\n            <ModelsTab\n              stats={displayStats}\n              dateRange={dateRange}\n              isLoading={isLoadingFiltered}\n            />\n          </Tab>\n        </Tabs>\n      </Box>\n      <Box paddingLeft={2}>\n        <Text dimColor>\n          Esc to cancel · r to cycle dates · ctrl+s to copy\n          {copyStatus ? ` · ${copyStatus}` : ''}\n        </Text>\n      </Box>\n    </Pane>\n  )\n}\n\nfunction DateRangeSelector({\n  dateRange,\n  isLoading,\n}: {\n  dateRange: StatsDateRange\n  isLoading: boolean\n}): React.ReactNode {\n  return (\n    <Box marginBottom={1} gap={1}>\n      <Box>\n        {DATE_RANGE_ORDER.map((range, i) => (\n          <Text key={range}>\n            {i > 0 && <Text dimColor> · </Text>}\n            {range === dateRange ? (\n              <Text bold color=\"claude\">\n                {DATE_RANGE_LABELS[range]}\n              </Text>\n            ) : (\n              <Text dimColor>{DATE_RANGE_LABELS[range]}</Text>\n            )}\n          </Text>\n        ))}\n      </Box>\n      {isLoading && <Spinner />}\n    </Box>\n  )\n}\n\nfunction OverviewTab({\n  stats,\n  allTimeStats,\n  dateRange,\n  isLoading,\n}: {\n  stats: ClaudeCodeStats\n  allTimeStats: ClaudeCodeStats\n  dateRange: StatsDateRange\n  isLoading: boolean\n}): React.ReactNode {\n  const { columns: terminalWidth } = useTerminalSize()\n\n  // Calculate favorite model and total tokens\n  const modelEntries = Object.entries(stats.modelUsage).sort(\n    ([, a], [, b]) =>\n      b.inputTokens + b.outputTokens - (a.inputTokens + a.outputTokens),\n  )\n  const favoriteModel = modelEntries[0]\n  const totalTokens = modelEntries.reduce(\n    (sum, [, usage]) => sum + usage.inputTokens + usage.outputTokens,\n    0,\n  )\n\n  // Memoize the factoid so it doesn't change when switching tabs\n  const factoid = useMemo(\n    () => generateFunFactoid(stats, totalTokens),\n    [stats, totalTokens],\n  )\n\n  // Calculate range days based on selected date range\n  const rangeDays =\n    dateRange === '7d' ? 7 : dateRange === '30d' ? 30 : stats.totalDays\n\n  // Compute shot stats data (ant-only, gated by feature flag)\n  let shotStatsData: {\n    avgShots: string\n    buckets: { label: string; count: number; pct: number }[]\n  } | null = null\n  if (feature('SHOT_STATS') && stats.shotDistribution) {\n    const dist = stats.shotDistribution\n    const total = Object.values(dist).reduce((s, n) => s + n, 0)\n    if (total > 0) {\n      const totalShots = Object.entries(dist).reduce(\n        (s, [count, sessions]) => s + parseInt(count, 10) * sessions,\n        0,\n      )\n      const bucket = (min: number, max?: number) =>\n        Object.entries(dist)\n          .filter(([k]) => {\n            const n = parseInt(k, 10)\n            return n >= min && (max === undefined || n <= max)\n          })\n          .reduce((s, [, v]) => s + v, 0)\n      const pct = (n: number) => Math.round((n / total) * 100)\n      const b1 = bucket(1, 1)\n      const b2_5 = bucket(2, 5)\n      const b6_10 = bucket(6, 10)\n      const b11 = bucket(11)\n      shotStatsData = {\n        avgShots: (totalShots / total).toFixed(1),\n        buckets: [\n          { label: '1-shot', count: b1, pct: pct(b1) },\n          { label: '2\\u20135 shot', count: b2_5, pct: pct(b2_5) },\n          { label: '6\\u201310 shot', count: b6_10, pct: pct(b6_10) },\n          { label: '11+ shot', count: b11, pct: pct(b11) },\n        ],\n      }\n    }\n  }\n\n  return (\n    <Box flexDirection=\"column\" marginTop={1}>\n      {/* Activity Heatmap - always shows all-time data */}\n      {allTimeStats.dailyActivity.length > 0 && (\n        <Box flexDirection=\"column\" marginBottom={1}>\n          <Ansi>\n            {generateHeatmap(allTimeStats.dailyActivity, { terminalWidth })}\n          </Ansi>\n        </Box>\n      )}\n\n      {/* Date range selector */}\n      <DateRangeSelector dateRange={dateRange} isLoading={isLoading} />\n\n      {/* Section 1: Usage */}\n      <Box flexDirection=\"row\" gap={4} marginBottom={1}>\n        <Box flexDirection=\"column\" width={28}>\n          {favoriteModel && (\n            <Text wrap=\"truncate\">\n              Favorite model:{' '}\n              <Text color=\"claude\" bold>\n                {renderModelName(favoriteModel[0])}\n              </Text>\n            </Text>\n          )}\n        </Box>\n        <Box flexDirection=\"column\" width={28}>\n          <Text wrap=\"truncate\">\n            Total tokens:{' '}\n            <Text color=\"claude\">{formatNumber(totalTokens)}</Text>\n          </Text>\n        </Box>\n      </Box>\n\n      {/* Section 2: Activity - Row 1: Sessions | Longest session */}\n      <Box flexDirection=\"row\" gap={4}>\n        <Box flexDirection=\"column\" width={28}>\n          <Text wrap=\"truncate\">\n            Sessions:{' '}\n            <Text color=\"claude\">{formatNumber(stats.totalSessions)}</Text>\n          </Text>\n        </Box>\n        <Box flexDirection=\"column\" width={28}>\n          {stats.longestSession && (\n            <Text wrap=\"truncate\">\n              Longest session:{' '}\n              <Text color=\"claude\">\n                {formatDuration(stats.longestSession.duration)}\n              </Text>\n            </Text>\n          )}\n        </Box>\n      </Box>\n\n      {/* Row 2: Active days | Longest streak */}\n      <Box flexDirection=\"row\" gap={4}>\n        <Box flexDirection=\"column\" width={28}>\n          <Text wrap=\"truncate\">\n            Active days: <Text color=\"claude\">{stats.activeDays}</Text>\n            <Text color=\"subtle\">/{rangeDays}</Text>\n          </Text>\n        </Box>\n        <Box flexDirection=\"column\" width={28}>\n          <Text wrap=\"truncate\">\n            Longest streak:{' '}\n            <Text color=\"claude\" bold>\n              {stats.streaks.longestStreak}\n            </Text>{' '}\n            {stats.streaks.longestStreak === 1 ? 'day' : 'days'}\n          </Text>\n        </Box>\n      </Box>\n\n      {/* Row 3: Most active day | Current streak */}\n      <Box flexDirection=\"row\" gap={4}>\n        <Box flexDirection=\"column\" width={28}>\n          {stats.peakActivityDay && (\n            <Text wrap=\"truncate\">\n              Most active day:{' '}\n              <Text color=\"claude\">{formatPeakDay(stats.peakActivityDay)}</Text>\n            </Text>\n          )}\n        </Box>\n        <Box flexDirection=\"column\" width={28}>\n          <Text wrap=\"truncate\">\n            Current streak:{' '}\n            <Text color=\"claude\" bold>\n              {allTimeStats.streaks.currentStreak}\n            </Text>{' '}\n            {allTimeStats.streaks.currentStreak === 1 ? 'day' : 'days'}\n          </Text>\n        </Box>\n      </Box>\n\n      {/* Speculation time saved (ant-only) */}\n      {\"external\" === 'ant' &&\n        stats.totalSpeculationTimeSavedMs > 0 && (\n          <Box flexDirection=\"row\" gap={4}>\n            <Box flexDirection=\"column\" width={28}>\n              <Text wrap=\"truncate\">\n                Speculation saved:{' '}\n                <Text color=\"claude\">\n                  {formatDuration(stats.totalSpeculationTimeSavedMs)}\n                </Text>\n              </Text>\n            </Box>\n          </Box>\n        )}\n\n      {/* Shot stats (ant-only) */}\n      {shotStatsData && (\n        <>\n          <Box marginTop={1}>\n            <Text>Shot distribution</Text>\n          </Box>\n          <Box flexDirection=\"row\" gap={4}>\n            <Box flexDirection=\"column\" width={28}>\n              <Text wrap=\"truncate\">\n                {shotStatsData.buckets[0]!.label}:{' '}\n                <Text color=\"claude\">{shotStatsData.buckets[0]!.count}</Text>\n                <Text color=\"subtle\"> ({shotStatsData.buckets[0]!.pct}%)</Text>\n              </Text>\n            </Box>\n            <Box flexDirection=\"column\" width={28}>\n              <Text wrap=\"truncate\">\n                {shotStatsData.buckets[1]!.label}:{' '}\n                <Text color=\"claude\">{shotStatsData.buckets[1]!.count}</Text>\n                <Text color=\"subtle\"> ({shotStatsData.buckets[1]!.pct}%)</Text>\n              </Text>\n            </Box>\n          </Box>\n          <Box flexDirection=\"row\" gap={4}>\n            <Box flexDirection=\"column\" width={28}>\n              <Text wrap=\"truncate\">\n                {shotStatsData.buckets[2]!.label}:{' '}\n                <Text color=\"claude\">{shotStatsData.buckets[2]!.count}</Text>\n                <Text color=\"subtle\"> ({shotStatsData.buckets[2]!.pct}%)</Text>\n              </Text>\n            </Box>\n            <Box flexDirection=\"column\" width={28}>\n              <Text wrap=\"truncate\">\n                {shotStatsData.buckets[3]!.label}:{' '}\n                <Text color=\"claude\">{shotStatsData.buckets[3]!.count}</Text>\n                <Text color=\"subtle\"> ({shotStatsData.buckets[3]!.pct}%)</Text>\n              </Text>\n            </Box>\n          </Box>\n          <Box flexDirection=\"row\" gap={4}>\n            <Box flexDirection=\"column\" width={28}>\n              <Text wrap=\"truncate\">\n                Avg/session:{' '}\n                <Text color=\"claude\">{shotStatsData.avgShots}</Text>\n              </Text>\n            </Box>\n          </Box>\n        </>\n      )}\n\n      {/* Fun factoid */}\n      {factoid && (\n        <Box marginTop={1}>\n          <Text color=\"suggestion\">{factoid}</Text>\n        </Box>\n      )}\n    </Box>\n  )\n}\n\n// Famous books and their approximate token counts (words * ~1.3)\n// Sorted by tokens ascending for comparison logic\nconst BOOK_COMPARISONS = [\n  { name: 'The Little Prince', tokens: 22000 },\n  { name: 'The Old Man and the Sea', tokens: 35000 },\n  { name: 'A Christmas Carol', tokens: 37000 },\n  { name: 'Animal Farm', tokens: 39000 },\n  { name: 'Fahrenheit 451', tokens: 60000 },\n  { name: 'The Great Gatsby', tokens: 62000 },\n  { name: 'Slaughterhouse-Five', tokens: 64000 },\n  { name: 'Brave New World', tokens: 83000 },\n  { name: 'The Catcher in the Rye', tokens: 95000 },\n  { name: \"Harry Potter and the Philosopher's Stone\", tokens: 103000 },\n  { name: 'The Hobbit', tokens: 123000 },\n  { name: '1984', tokens: 123000 },\n  { name: 'To Kill a Mockingbird', tokens: 130000 },\n  { name: 'Pride and Prejudice', tokens: 156000 },\n  { name: 'Dune', tokens: 244000 },\n  { name: 'Moby-Dick', tokens: 268000 },\n  { name: 'Crime and Punishment', tokens: 274000 },\n  { name: 'A Game of Thrones', tokens: 381000 },\n  { name: 'Anna Karenina', tokens: 468000 },\n  { name: 'Don Quixote', tokens: 520000 },\n  { name: 'The Lord of the Rings', tokens: 576000 },\n  { name: 'The Count of Monte Cristo', tokens: 603000 },\n  { name: 'Les Misérables', tokens: 689000 },\n  { name: 'War and Peace', tokens: 730000 },\n]\n\n// Time equivalents for session durations\nconst TIME_COMPARISONS = [\n  { name: 'a TED talk', minutes: 18 },\n  { name: 'an episode of The Office', minutes: 22 },\n  { name: 'listening to Abbey Road', minutes: 47 },\n  { name: 'a yoga class', minutes: 60 },\n  { name: 'a World Cup soccer match', minutes: 90 },\n  { name: 'a half marathon (average time)', minutes: 120 },\n  { name: 'the movie Inception', minutes: 148 },\n  { name: 'watching Titanic', minutes: 195 },\n  { name: 'a transatlantic flight', minutes: 420 },\n  { name: 'a full night of sleep', minutes: 480 },\n]\n\nfunction generateFunFactoid(\n  stats: ClaudeCodeStats,\n  totalTokens: number,\n): string {\n  const factoids: string[] = []\n\n  if (totalTokens > 0) {\n    const matchingBooks = BOOK_COMPARISONS.filter(\n      book => totalTokens >= book.tokens,\n    )\n\n    for (const book of matchingBooks) {\n      const times = totalTokens / book.tokens\n      if (times >= 2) {\n        factoids.push(\n          `You've used ~${Math.floor(times)}x more tokens than ${book.name}`,\n        )\n      } else {\n        factoids.push(`You've used the same number of tokens as ${book.name}`)\n      }\n    }\n  }\n\n  if (stats.longestSession) {\n    const sessionMinutes = stats.longestSession.duration / (1000 * 60)\n    for (const comparison of TIME_COMPARISONS) {\n      const ratio = sessionMinutes / comparison.minutes\n      if (ratio >= 2) {\n        factoids.push(\n          `Your longest session is ~${Math.floor(ratio)}x longer than ${comparison.name}`,\n        )\n      }\n    }\n  }\n\n  if (factoids.length === 0) {\n    return ''\n  }\n  const randomIndex = Math.floor(Math.random() * factoids.length)\n  return factoids[randomIndex]!\n}\n\nfunction ModelsTab({\n  stats,\n  dateRange,\n  isLoading,\n}: {\n  stats: ClaudeCodeStats\n  dateRange: StatsDateRange\n  isLoading: boolean\n}): React.ReactNode {\n  const { headerFocused, focusHeader } = useTabHeaderFocus()\n  const [scrollOffset, setScrollOffset] = useState(0)\n  const { columns: terminalWidth } = useTerminalSize()\n  const VISIBLE_MODELS = 4 // Show 4 models at a time (2 per column)\n\n  const modelEntries = Object.entries(stats.modelUsage).sort(\n    ([, a], [, b]) =>\n      b.inputTokens + b.outputTokens - (a.inputTokens + a.outputTokens),\n  )\n\n  // Handle scrolling with arrow keys\n  useInput(\n    (_input, key) => {\n      if (\n        key.downArrow &&\n        scrollOffset < modelEntries.length - VISIBLE_MODELS\n      ) {\n        setScrollOffset(prev =>\n          Math.min(prev + 2, modelEntries.length - VISIBLE_MODELS),\n        )\n      }\n      if (key.upArrow) {\n        if (scrollOffset > 0) {\n          setScrollOffset(prev => Math.max(prev - 2, 0))\n        } else {\n          focusHeader()\n        }\n      }\n    },\n    { isActive: !headerFocused },\n  )\n\n  if (modelEntries.length === 0) {\n    return (\n      <Box>\n        <Text color=\"subtle\">No model usage data available</Text>\n      </Box>\n    )\n  }\n\n  const totalTokens = modelEntries.reduce(\n    (sum, [, usage]) => sum + usage.inputTokens + usage.outputTokens,\n    0,\n  )\n\n  // Generate token usage chart - use terminal width for responsive sizing\n  const chartOutput = generateTokenChart(\n    stats.dailyModelTokens,\n    modelEntries.map(([model]) => model),\n    terminalWidth,\n  )\n\n  // Get visible models and split into two columns\n  const visibleModels = modelEntries.slice(\n    scrollOffset,\n    scrollOffset + VISIBLE_MODELS,\n  )\n  const midpoint = Math.ceil(visibleModels.length / 2)\n  const leftModels = visibleModels.slice(0, midpoint)\n  const rightModels = visibleModels.slice(midpoint)\n\n  const canScrollUp = scrollOffset > 0\n  const canScrollDown = scrollOffset < modelEntries.length - VISIBLE_MODELS\n  const showScrollHint = modelEntries.length > VISIBLE_MODELS\n\n  return (\n    <Box flexDirection=\"column\" marginTop={1}>\n      {/* Token usage chart */}\n      {chartOutput && (\n        <Box flexDirection=\"column\" marginBottom={1}>\n          <Text bold>Tokens per Day</Text>\n          <Ansi>{chartOutput.chart}</Ansi>\n          <Text color=\"subtle\">{chartOutput.xAxisLabels}</Text>\n          <Box>\n            {chartOutput.legend.map((item, i) => (\n              <Text key={item.model}>\n                {i > 0 ? ' · ' : ''}\n                <Ansi>{item.coloredBullet}</Ansi> {item.model}\n              </Text>\n            ))}\n          </Box>\n        </Box>\n      )}\n\n      {/* Date range selector */}\n      <DateRangeSelector dateRange={dateRange} isLoading={isLoading} />\n\n      {/* Model breakdown - two columns with fixed width */}\n      <Box flexDirection=\"row\" gap={4}>\n        <Box flexDirection=\"column\" width={36}>\n          {leftModels.map(([model, usage]) => (\n            <ModelEntry\n              key={model}\n              model={model}\n              usage={usage}\n              totalTokens={totalTokens}\n            />\n          ))}\n        </Box>\n        <Box flexDirection=\"column\" width={36}>\n          {rightModels.map(([model, usage]) => (\n            <ModelEntry\n              key={model}\n              model={model}\n              usage={usage}\n              totalTokens={totalTokens}\n            />\n          ))}\n        </Box>\n      </Box>\n\n      {/* Scroll hint */}\n      {showScrollHint && (\n        <Box marginTop={1}>\n          <Text color=\"subtle\">\n            {canScrollUp ? figures.arrowUp : ' '}{' '}\n            {canScrollDown ? figures.arrowDown : ' '} {scrollOffset + 1}-\n            {Math.min(scrollOffset + VISIBLE_MODELS, modelEntries.length)} of{' '}\n            {modelEntries.length} models (↑↓ to scroll)\n          </Text>\n        </Box>\n      )}\n    </Box>\n  )\n}\n\ntype ModelEntryProps = {\n  model: string\n  usage: {\n    inputTokens: number\n    outputTokens: number\n    cacheReadInputTokens: number\n  }\n  totalTokens: number\n}\n\nfunction ModelEntry({\n  model,\n  usage,\n  totalTokens,\n}: ModelEntryProps): React.ReactNode {\n  const modelTokens = usage.inputTokens + usage.outputTokens\n  const percentage = ((modelTokens / totalTokens) * 100).toFixed(1)\n\n  return (\n    <Box flexDirection=\"column\">\n      <Text>\n        {figures.bullet} <Text bold>{renderModelName(model)}</Text>{' '}\n        <Text color=\"subtle\">({percentage}%)</Text>\n      </Text>\n      <Text color=\"subtle\">\n        {'  '}In: {formatNumber(usage.inputTokens)} · Out:{' '}\n        {formatNumber(usage.outputTokens)}\n      </Text>\n    </Box>\n  )\n}\n\ntype ChartLegend = {\n  model: string\n  coloredBullet: string // Pre-colored bullet using chalk\n}\n\ntype ChartOutput = {\n  chart: string\n  legend: ChartLegend[]\n  xAxisLabels: string\n}\n\nfunction generateTokenChart(\n  dailyTokens: DailyModelTokens[],\n  models: string[],\n  terminalWidth: number,\n): ChartOutput | null {\n  if (dailyTokens.length < 2 || models.length === 0) {\n    return null\n  }\n\n  // Y-axis labels take about 6 characters, plus some padding\n  // Cap at ~52 to align with heatmap width (1 year of data)\n  const yAxisWidth = 7\n  const availableWidth = terminalWidth - yAxisWidth\n  const chartWidth = Math.min(52, Math.max(20, availableWidth))\n\n  // Distribute data across the available chart width\n  let recentData: DailyModelTokens[]\n  if (dailyTokens.length >= chartWidth) {\n    // More data than space: take most recent N days\n    recentData = dailyTokens.slice(-chartWidth)\n  } else {\n    // Less data than space: expand by repeating each point\n    const repeatCount = Math.floor(chartWidth / dailyTokens.length)\n    recentData = []\n    for (const day of dailyTokens) {\n      for (let i = 0; i < repeatCount; i++) {\n        recentData.push(day)\n      }\n    }\n  }\n\n  // Color palette for different models - use theme colors\n  const theme = getTheme(resolveThemeSetting(getGlobalConfig().theme))\n  const colors = [\n    themeColorToAnsi(theme.suggestion),\n    themeColorToAnsi(theme.success),\n    themeColorToAnsi(theme.warning),\n  ]\n\n  // Prepare series data for each model\n  const series: number[][] = []\n  const legend: ChartLegend[] = []\n\n  // Only show top 3 models to keep chart readable\n  const topModels = models.slice(0, 3)\n\n  for (let i = 0; i < topModels.length; i++) {\n    const model = topModels[i]!\n    const data = recentData.map(day => day.tokensByModel[model] || 0)\n\n    // Only include if there's actual data\n    if (data.some(v => v > 0)) {\n      series.push(data)\n      // Use theme colors that match the chart\n      const bulletColors = [theme.suggestion, theme.success, theme.warning]\n      legend.push({\n        model: renderModelName(model),\n        coloredBullet: applyColor(\n          figures.bullet,\n          bulletColors[i % bulletColors.length] as Color,\n        ),\n      })\n    }\n  }\n\n  if (series.length === 0) {\n    return null\n  }\n\n  const chart = asciichart(series, {\n    height: 8,\n    colors: colors.slice(0, series.length),\n    format: (x: number) => {\n      let label: string\n      if (x >= 1_000_000) {\n        label = (x / 1_000_000).toFixed(1) + 'M'\n      } else if (x >= 1_000) {\n        label = (x / 1_000).toFixed(0) + 'k'\n      } else {\n        label = x.toFixed(0)\n      }\n      return label.padStart(6)\n    },\n  })\n\n  // Generate x-axis labels with dates\n  const xAxisLabels = generateXAxisLabels(\n    recentData,\n    recentData.length,\n    yAxisWidth,\n  )\n\n  return { chart, legend, xAxisLabels }\n}\n\nfunction generateXAxisLabels(\n  data: DailyModelTokens[],\n  _chartWidth: number,\n  yAxisOffset: number,\n): string {\n  if (data.length === 0) return ''\n\n  // Show 3-4 date labels evenly spaced, but leave room for last label\n  const numLabels = Math.min(4, Math.max(2, Math.floor(data.length / 8)))\n  // Don't use the very last position - leave room for the label text\n  const usableLength = data.length - 6 // Reserve ~6 chars for last label (e.g., \"Dec 7\")\n  const step = Math.floor(usableLength / (numLabels - 1)) || 1\n\n  const labelPositions: { pos: number; label: string }[] = []\n\n  for (let i = 0; i < numLabels; i++) {\n    const idx = Math.min(i * step, data.length - 1)\n    const date = new Date(data[idx]!.date)\n    const label = date.toLocaleDateString('en-US', {\n      month: 'short',\n      day: 'numeric',\n    })\n    labelPositions.push({ pos: idx, label })\n  }\n\n  // Build the label string with proper spacing\n  let result = ' '.repeat(yAxisOffset)\n  let currentPos = 0\n\n  for (const { pos, label } of labelPositions) {\n    const spaces = Math.max(1, pos - currentPos)\n    result += ' '.repeat(spaces) + label\n    currentPos = pos + label.length\n  }\n\n  return result\n}\n\n// Screenshot functionality\nasync function handleScreenshot(\n  stats: ClaudeCodeStats,\n  activeTab: 'Overview' | 'Models',\n  setStatus: (status: string | null) => void,\n): Promise<void> {\n  setStatus('copying…')\n\n  const ansiText = renderStatsToAnsi(stats, activeTab)\n  const result = await copyAnsiToClipboard(ansiText)\n\n  setStatus(result.success ? 'copied!' : 'copy failed')\n\n  // Clear status after 2 seconds\n  setTimeout(setStatus, 2000, null)\n}\n\nfunction renderStatsToAnsi(\n  stats: ClaudeCodeStats,\n  activeTab: 'Overview' | 'Models',\n): string {\n  const lines: string[] = []\n\n  if (activeTab === 'Overview') {\n    lines.push(...renderOverviewToAnsi(stats))\n  } else {\n    lines.push(...renderModelsToAnsi(stats))\n  }\n\n  // Trim trailing empty lines\n  while (\n    lines.length > 0 &&\n    stripAnsi(lines[lines.length - 1]!).trim() === ''\n  ) {\n    lines.pop()\n  }\n\n  // Add \"/stats\" right-aligned on the last line\n  if (lines.length > 0) {\n    const lastLine = lines[lines.length - 1]!\n    const lastLineLen = getStringWidth(lastLine)\n    // Use known content widths based on layout:\n    // Overview: two-column stats = COL2_START(40) + COL2_LABEL_WIDTH(18) + max_value(~12) = 70\n    // Models: chart width = 80\n    const contentWidth = activeTab === 'Overview' ? 70 : 80\n    const statsLabel = '/stats'\n    const padding = Math.max(2, contentWidth - lastLineLen - statsLabel.length)\n    lines[lines.length - 1] =\n      lastLine + ' '.repeat(padding) + chalk.gray(statsLabel)\n  }\n\n  return lines.join('\\n')\n}\n\nfunction renderOverviewToAnsi(stats: ClaudeCodeStats): string[] {\n  const lines: string[] = []\n  const theme = getTheme(resolveThemeSetting(getGlobalConfig().theme))\n  const h = (text: string) => applyColor(text, theme.claude as Color)\n\n  // Two-column helper with fixed spacing\n  // Column 1: label (18 chars) + value + padding to reach col 2\n  // Column 2 starts at character position 40\n  const COL1_LABEL_WIDTH = 18\n  const COL2_START = 40\n  const COL2_LABEL_WIDTH = 18\n\n  const row = (l1: string, v1: string, l2: string, v2: string): string => {\n    // Build column 1: label + value\n    const label1 = (l1 + ':').padEnd(COL1_LABEL_WIDTH)\n    const col1PlainLen = label1.length + v1.length\n\n    // Calculate spaces needed between col1 value and col2 label\n    const spaceBetween = Math.max(2, COL2_START - col1PlainLen)\n\n    // Build column 2: label + value\n    const label2 = (l2 + ':').padEnd(COL2_LABEL_WIDTH)\n\n    // Assemble with colors applied to values only\n    return label1 + h(v1) + ' '.repeat(spaceBetween) + label2 + h(v2)\n  }\n\n  // Heatmap - use fixed width for screenshot (56 = 52 weeks + 4 for day labels)\n  if (stats.dailyActivity.length > 0) {\n    lines.push(generateHeatmap(stats.dailyActivity, { terminalWidth: 56 }))\n    lines.push('')\n  }\n\n  // Calculate values\n  const modelEntries = Object.entries(stats.modelUsage).sort(\n    ([, a], [, b]) =>\n      b.inputTokens + b.outputTokens - (a.inputTokens + a.outputTokens),\n  )\n  const favoriteModel = modelEntries[0]\n  const totalTokens = modelEntries.reduce(\n    (sum, [, usage]) => sum + usage.inputTokens + usage.outputTokens,\n    0,\n  )\n\n  // Row 1: Favorite model | Total tokens\n  if (favoriteModel) {\n    lines.push(\n      row(\n        'Favorite model',\n        renderModelName(favoriteModel[0]),\n        'Total tokens',\n        formatNumber(totalTokens),\n      ),\n    )\n  }\n  lines.push('')\n\n  // Row 2: Sessions | Longest session\n  lines.push(\n    row(\n      'Sessions',\n      formatNumber(stats.totalSessions),\n      'Longest session',\n      stats.longestSession\n        ? formatDuration(stats.longestSession.duration)\n        : 'N/A',\n    ),\n  )\n\n  // Row 3: Current streak | Longest streak\n  const currentStreakVal = `${stats.streaks.currentStreak} ${stats.streaks.currentStreak === 1 ? 'day' : 'days'}`\n  const longestStreakVal = `${stats.streaks.longestStreak} ${stats.streaks.longestStreak === 1 ? 'day' : 'days'}`\n  lines.push(\n    row('Current streak', currentStreakVal, 'Longest streak', longestStreakVal),\n  )\n\n  // Row 4: Active days | Peak hour\n  const activeDaysVal = `${stats.activeDays}/${stats.totalDays}`\n  const peakHourVal =\n    stats.peakActivityHour !== null\n      ? `${stats.peakActivityHour}:00-${stats.peakActivityHour + 1}:00`\n      : 'N/A'\n  lines.push(row('Active days', activeDaysVal, 'Peak hour', peakHourVal))\n\n  // Speculation time saved (ant-only)\n  if (\n    \"external\" === 'ant' &&\n    stats.totalSpeculationTimeSavedMs > 0\n  ) {\n    const label = 'Speculation saved:'.padEnd(COL1_LABEL_WIDTH)\n    lines.push(label + h(formatDuration(stats.totalSpeculationTimeSavedMs)))\n  }\n\n  // Shot stats (ant-only)\n  if (feature('SHOT_STATS') && stats.shotDistribution) {\n    const dist = stats.shotDistribution\n    const totalWithShots = Object.values(dist).reduce((s, n) => s + n, 0)\n    if (totalWithShots > 0) {\n      const totalShots = Object.entries(dist).reduce(\n        (s, [count, sessions]) => s + parseInt(count, 10) * sessions,\n        0,\n      )\n      const avgShots = (totalShots / totalWithShots).toFixed(1)\n      const bucket = (min: number, max?: number) =>\n        Object.entries(dist)\n          .filter(([k]) => {\n            const n = parseInt(k, 10)\n            return n >= min && (max === undefined || n <= max)\n          })\n          .reduce((s, [, v]) => s + v, 0)\n      const pct = (n: number) => Math.round((n / totalWithShots) * 100)\n      const fmtBucket = (count: number, p: number) => `${count} (${p}%)`\n      const b1 = bucket(1, 1)\n      const b2_5 = bucket(2, 5)\n      const b6_10 = bucket(6, 10)\n      const b11 = bucket(11)\n      lines.push('')\n      lines.push('Shot distribution')\n      lines.push(\n        row(\n          '1-shot',\n          fmtBucket(b1, pct(b1)),\n          '2\\u20135 shot',\n          fmtBucket(b2_5, pct(b2_5)),\n        ),\n      )\n      lines.push(\n        row(\n          '6\\u201310 shot',\n          fmtBucket(b6_10, pct(b6_10)),\n          '11+ shot',\n          fmtBucket(b11, pct(b11)),\n        ),\n      )\n      lines.push(`${'Avg/session:'.padEnd(COL1_LABEL_WIDTH)}${h(avgShots)}`)\n    }\n  }\n\n  lines.push('')\n\n  // Fun factoid\n  const factoid = generateFunFactoid(stats, totalTokens)\n  lines.push(h(factoid))\n  lines.push(chalk.gray(`Stats from the last ${stats.totalDays} days`))\n\n  return lines\n}\n\nfunction renderModelsToAnsi(stats: ClaudeCodeStats): string[] {\n  const lines: string[] = []\n\n  const modelEntries = Object.entries(stats.modelUsage).sort(\n    ([, a], [, b]) =>\n      b.inputTokens + b.outputTokens - (a.inputTokens + a.outputTokens),\n  )\n\n  if (modelEntries.length === 0) {\n    lines.push(chalk.gray('No model usage data available'))\n    return lines\n  }\n\n  const favoriteModel = modelEntries[0]\n  const totalTokens = modelEntries.reduce(\n    (sum, [, usage]) => sum + usage.inputTokens + usage.outputTokens,\n    0,\n  )\n\n  // Generate chart if we have data - use fixed width for screenshot\n  const chartOutput = generateTokenChart(\n    stats.dailyModelTokens,\n    modelEntries.map(([model]) => model),\n    80, // Fixed width for screenshot\n  )\n\n  if (chartOutput) {\n    lines.push(chalk.bold('Tokens per Day'))\n    lines.push(chartOutput.chart)\n    lines.push(chalk.gray(chartOutput.xAxisLabels))\n    // Legend - use pre-colored bullets from chart output\n    const legendLine = chartOutput.legend\n      .map(item => `${item.coloredBullet} ${item.model}`)\n      .join(' · ')\n    lines.push(legendLine)\n    lines.push('')\n  }\n\n  // Summary\n  lines.push(\n    `${figures.star} Favorite: ${chalk.magenta.bold(renderModelName(favoriteModel?.[0] || ''))} · ${figures.circle} Total: ${chalk.magenta(formatNumber(totalTokens))} tokens`,\n  )\n  lines.push('')\n\n  // Model breakdown - only show top 3 for screenshot\n  const topModels = modelEntries.slice(0, 3)\n  for (const [model, usage] of topModels) {\n    const modelTokens = usage.inputTokens + usage.outputTokens\n    const percentage = ((modelTokens / totalTokens) * 100).toFixed(1)\n    lines.push(\n      `${figures.bullet} ${chalk.bold(renderModelName(model))} ${chalk.gray(`(${percentage}%)`)}`,\n    )\n    lines.push(\n      chalk.dim(\n        `  In: ${formatNumber(usage.inputTokens)} · Out: ${formatNumber(usage.outputTokens)}`,\n      ),\n    )\n  }\n\n  return lines\n}\n"],"mappings":"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","ignoreList":[]}