bingocode 1.1.162 → 1.1.164
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/package.json
CHANGED
|
@@ -14,6 +14,7 @@ import { ensureSingletonLocalServer } from '../server/ensureSingletonLocalServer
|
|
|
14
14
|
import { TopBar, BottomBar, Panel, Hint, Kbd, SecondaryMenu, StateDisplay, ScrollBar, truncate, safePadEnd } from '../manager/CliMenuUi.tsx';
|
|
15
15
|
import { WelcomeV2 } from '../components/LogoV2/WelcomeV2.tsx';
|
|
16
16
|
import { TopToolbar } from '../manager/TopToolbar.tsx';
|
|
17
|
+
import { TokenStatsPanel } from '../manager/TokenStatsPanel.tsx';
|
|
17
18
|
|
|
18
19
|
// Theme switching (Hook)
|
|
19
20
|
import { useTheme } from '../components/design-system/ThemeProvider.js';
|
|
@@ -484,7 +485,7 @@ export const CliMenuManager: React.FC = () => {
|
|
|
484
485
|
const [settingData, setSettingData] = useState<any>(null);
|
|
485
486
|
const [loadingSetting, setLoadingSetting] = useState(false);
|
|
486
487
|
const [setErr, setSetErr] = useState<string | null>(null);
|
|
487
|
-
const [settingsStage, setSettingsStage] = useState<'list' | 'langPicker'>('list');
|
|
488
|
+
const [settingsStage, setSettingsStage] = useState<'list' | 'langPicker' | 'tokenStats'>('list');
|
|
488
489
|
const [settingsCursor, setSettingsCursor] = useState(0);
|
|
489
490
|
const [autoModeEnabled, setAutoModeEnabled] = useState(false);
|
|
490
491
|
const [bypassPermsEnabled, setBypassPermsEnabled] = useState(false);
|
|
@@ -763,6 +764,7 @@ export const CliMenuManager: React.FC = () => {
|
|
|
763
764
|
// Settings: langPicker → back to list; list → back to main menu
|
|
764
765
|
if (page === 'settings') {
|
|
765
766
|
if (settingsStage === 'langPicker') { setSettingsStage('list'); return; }
|
|
767
|
+
if (settingsStage === 'tokenStats') { setSettingsStage('list'); return; }
|
|
766
768
|
}
|
|
767
769
|
setPage(null);
|
|
768
770
|
setHistoryMenuStage('list');
|
|
@@ -923,6 +925,9 @@ export const CliMenuManager: React.FC = () => {
|
|
|
923
925
|
}
|
|
924
926
|
return next;
|
|
925
927
|
});
|
|
928
|
+
} else if (settingsCursor === 3) {
|
|
929
|
+
// Row 3: open Token Stats
|
|
930
|
+
setSettingsStage('tokenStats');
|
|
926
931
|
}
|
|
927
932
|
}
|
|
928
933
|
}
|
|
@@ -1406,12 +1411,24 @@ export const CliMenuManager: React.FC = () => {
|
|
|
1406
1411
|
);
|
|
1407
1412
|
}
|
|
1408
1413
|
|
|
1414
|
+
// --- tokenStats sub-menu ---
|
|
1415
|
+
if (settingsStage === 'tokenStats') {
|
|
1416
|
+
return (
|
|
1417
|
+
<TokenStatsPanel
|
|
1418
|
+
width={VIEW_W}
|
|
1419
|
+
height={MID_H}
|
|
1420
|
+
onBack={() => setSettingsStage('list')}
|
|
1421
|
+
/>
|
|
1422
|
+
);
|
|
1423
|
+
}
|
|
1424
|
+
|
|
1409
1425
|
// --- settings list ---
|
|
1410
1426
|
type SettingRow = { key: string; label: string; value: string; interactive: boolean };
|
|
1411
1427
|
const fixedRows: SettingRow[] = [
|
|
1412
1428
|
{ key: '__lang', label: tS.langLabel, value: currentLangLabel, interactive: true },
|
|
1413
1429
|
{ key: '__autoMode', label: tS.autoModeLabel, value: autoModeEnabled ? tS.autoModeOn : tS.autoModeOff, interactive: true },
|
|
1414
1430
|
{ key: '__bypassPerms', label: tS.bypassPermsLabel, value: bypassPermsEnabled ? tS.bypassPermsOn : tS.bypassPermsOff, interactive: true },
|
|
1431
|
+
{ key: '__tokenStats', label: 'Token Stats', value: '>', interactive: true },
|
|
1415
1432
|
];
|
|
1416
1433
|
const dataEntries = settingData && typeof settingData === 'object' ? Object.entries(settingData) : [];
|
|
1417
1434
|
const dataRows: SettingRow[] = dataEntries.map(([k, v]) => ({
|
|
@@ -0,0 +1,95 @@
|
|
|
1
|
+
import React, { useState, useEffect } from 'react';
|
|
2
|
+
import { Box, Text } from 'ink';
|
|
3
|
+
import { StateDisplay, ScrollBar, safePadEnd } from '../manager/CliMenuUi.tsx';
|
|
4
|
+
import { aggregateClaudeCodeStats, type ClaudeCodeStats } from '../utils/stats.ts';
|
|
5
|
+
import { formatNumber } from '../utils/format.js';
|
|
6
|
+
|
|
7
|
+
type Props = {
|
|
8
|
+
width: number;
|
|
9
|
+
height: number;
|
|
10
|
+
onBack: () => void;
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
export const TokenStatsPanel: React.FC<Props> = ({ width, height, onBack }) => {
|
|
14
|
+
const [loading, setLoading] = useState(true);
|
|
15
|
+
const [error, setError] = useState<string | null>(null);
|
|
16
|
+
const [stats, setStats] = useState<ClaudeCodeStats | null>(null);
|
|
17
|
+
|
|
18
|
+
useEffect(() => {
|
|
19
|
+
let cancelled = false;
|
|
20
|
+
setLoading(true);
|
|
21
|
+
setError(null);
|
|
22
|
+
aggregateClaudeCodeStats()
|
|
23
|
+
.then(data => {
|
|
24
|
+
if (!cancelled) {
|
|
25
|
+
setStats(data);
|
|
26
|
+
setLoading(false);
|
|
27
|
+
}
|
|
28
|
+
})
|
|
29
|
+
.catch(err => {
|
|
30
|
+
if (!cancelled) {
|
|
31
|
+
setError(err.message || String(err));
|
|
32
|
+
setLoading(false);
|
|
33
|
+
}
|
|
34
|
+
});
|
|
35
|
+
return () => { cancelled = true; };
|
|
36
|
+
}, []);
|
|
37
|
+
|
|
38
|
+
if (loading) {
|
|
39
|
+
return <StateDisplay type="loading" message="Loading token stats..." />;
|
|
40
|
+
}
|
|
41
|
+
if (error) {
|
|
42
|
+
return <StateDisplay type="error" message={`Failed to load: ${error}`} />;
|
|
43
|
+
}
|
|
44
|
+
if (!stats || !stats.modelUsage || Object.keys(stats.modelUsage).length === 0) {
|
|
45
|
+
return <StateDisplay type="empty" message="No token data yet. Run some conversations first!" />;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
const modelEntries = Object.entries(stats.modelUsage).sort(
|
|
49
|
+
(a, b) => (b[1].inputTokens + b[1].outputTokens) - (a[1].inputTokens + a[1].outputTokens)
|
|
50
|
+
);
|
|
51
|
+
|
|
52
|
+
// Calculate column widths
|
|
53
|
+
const MODEL_COL = 14;
|
|
54
|
+
const NUM_COL = 14;
|
|
55
|
+
const COST_COL = 10;
|
|
56
|
+
const visible = height - 2; // leave room for header row + bottom hint
|
|
57
|
+
|
|
58
|
+
// Header
|
|
59
|
+
const header = [
|
|
60
|
+
safePadEnd('Model', MODEL_COL),
|
|
61
|
+
safePadEnd('Input Tokens', NUM_COL),
|
|
62
|
+
safePadEnd('Output Tokens', NUM_COL),
|
|
63
|
+
safePadEnd('Cache Read', NUM_COL),
|
|
64
|
+
safePadEnd('Cache Create', NUM_COL),
|
|
65
|
+
].join('');
|
|
66
|
+
|
|
67
|
+
const formatCount = (n: number) => formatNumber(n);
|
|
68
|
+
|
|
69
|
+
return (
|
|
70
|
+
<Box width={width} height={height} flexDirection="column">
|
|
71
|
+
<Box flexDirection="column" flexGrow={1}>
|
|
72
|
+
<Text bold>{header}</Text>
|
|
73
|
+
<Text dimColor>{'─'.repeat(width - 4)}</Text>
|
|
74
|
+
<Box flexDirection="row" flexGrow={1} position="relative">
|
|
75
|
+
<Box flexDirection="column" flexGrow={1}>
|
|
76
|
+
{modelEntries.map(([model, usage]) => {
|
|
77
|
+
const row = [
|
|
78
|
+
safePadEnd(model.length > MODEL_COL - 3 ? model.slice(0, MODEL_COL - 4) + '…' : model, MODEL_COL),
|
|
79
|
+
safePadEnd(formatCount(usage.inputTokens), NUM_COL),
|
|
80
|
+
safePadEnd(formatCount(usage.outputTokens), NUM_COL),
|
|
81
|
+
safePadEnd(formatCount(usage.cacheReadInputTokens), NUM_COL),
|
|
82
|
+
safePadEnd(formatCount(usage.cacheCreationInputTokens), NUM_COL),
|
|
83
|
+
].join('');
|
|
84
|
+
return <Text key={model}>{row}</Text>;
|
|
85
|
+
})}
|
|
86
|
+
</Box>
|
|
87
|
+
<ScrollBar total={modelEntries.length} offset={0} height={visible - 2} />
|
|
88
|
+
</Box>
|
|
89
|
+
</Box>
|
|
90
|
+
<Text dimColor>ESC to go back to settings</Text>
|
|
91
|
+
</Box>
|
|
92
|
+
);
|
|
93
|
+
};
|
|
94
|
+
|
|
95
|
+
export default TokenStatsPanel;
|