thatgfsj-code 0.9.4 → 0.9.5
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/cmd/index.js +1 -1
- package/dist/tui/app.d.ts.map +1 -1
- package/dist/tui/app.js +46 -15
- package/dist/tui/app.js.map +1 -1
- package/dist/tui/components/Header.js +1 -1
- package/dist/tui/components/ModelSelector.d.ts +2 -2
- package/dist/tui/components/ModelSelector.d.ts.map +1 -1
- package/dist/tui/components/ModelSelector.js +53 -25
- package/dist/tui/components/ModelSelector.js.map +1 -1
- package/dist/tui/components/StatusBar.js +1 -1
- package/dist/tui/components/StatusBar.js.map +1 -1
- package/dist/tui/components/UserInput.d.ts.map +1 -1
- package/dist/tui/components/UserInput.js +28 -2
- package/dist/tui/components/UserInput.js.map +1 -1
- package/dist/tui/hooks/useCommands.d.ts +4 -0
- package/dist/tui/hooks/useCommands.d.ts.map +1 -1
- package/dist/tui/hooks/useCommands.js +58 -30
- package/dist/tui/hooks/useCommands.js.map +1 -1
- package/package.json +1 -1
- package/src/cmd/index.tsx +1 -1
- package/src/tui/app.tsx +49 -17
- package/src/tui/components/Header.tsx +1 -1
- package/src/tui/components/ModelSelector.tsx +65 -39
- package/src/tui/components/StatusBar.tsx +2 -2
- package/src/tui/components/UserInput.tsx +53 -6
- package/src/tui/hooks/useCommands.ts +61 -30
package/dist/cmd/index.js
CHANGED
|
@@ -24,7 +24,7 @@ process.on('unhandledRejection', (reason) => {
|
|
|
24
24
|
program
|
|
25
25
|
.name('gfcode')
|
|
26
26
|
.description('Thatgfsj Code - AI Coding Assistant')
|
|
27
|
-
.version('0.9.
|
|
27
|
+
.version('0.9.5')
|
|
28
28
|
.argument('[prompt]', 'Task to execute (omit to start interactive mode)')
|
|
29
29
|
.option('-m, --model <model>', 'Specify model')
|
|
30
30
|
.option('-i, --interactive', 'Force interactive mode')
|
package/dist/tui/app.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"app.d.ts","sourceRoot":"","sources":["../../src/tui/app.tsx"],"names":[],"mappings":"AAAA,6BAA6B;AAC7B,OAAO,KAAgC,MAAM,OAAO,CAAC;AAUrD,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,iBAAiB,CAAC;
|
|
1
|
+
{"version":3,"file":"app.d.ts","sourceRoot":"","sources":["../../src/tui/app.tsx"],"names":[],"mappings":"AAAA,6BAA6B;AAC7B,OAAO,KAAgC,MAAM,OAAO,CAAC;AAUrD,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,iBAAiB,CAAC;AAM3C,UAAU,KAAK;IACb,GAAG,EAAE,GAAG,CAAC;CACV;AAiBD,wBAAgB,MAAM,CAAC,EAAE,GAAG,EAAE,EAAE,KAAK,qBA6GpC"}
|
package/dist/tui/app.js
CHANGED
|
@@ -10,26 +10,54 @@ import { StatusBar } from './components/StatusBar.js';
|
|
|
10
10
|
import { ModelSelector } from './components/ModelSelector.js';
|
|
11
11
|
import { useChat } from './hooks/useChat.js';
|
|
12
12
|
import { useCommands } from './hooks/useCommands.js';
|
|
13
|
+
import { writeFileSync, readFileSync, existsSync, mkdirSync } from 'fs';
|
|
14
|
+
import { join } from 'path';
|
|
15
|
+
import { homedir } from 'os';
|
|
16
|
+
function saveModelToHistory(model) {
|
|
17
|
+
const dir = join(homedir(), '.thatgfsj');
|
|
18
|
+
const path = join(dir, 'models.json');
|
|
19
|
+
if (!existsSync(dir))
|
|
20
|
+
mkdirSync(dir, { recursive: true });
|
|
21
|
+
let history = [];
|
|
22
|
+
if (existsSync(path)) {
|
|
23
|
+
try {
|
|
24
|
+
history = JSON.parse(readFileSync(path, 'utf-8'));
|
|
25
|
+
}
|
|
26
|
+
catch { }
|
|
27
|
+
}
|
|
28
|
+
if (!history.includes(model)) {
|
|
29
|
+
history.push(model);
|
|
30
|
+
writeFileSync(path, JSON.stringify(history, null, 2));
|
|
31
|
+
}
|
|
32
|
+
}
|
|
13
33
|
export function TuiApp({ app }) {
|
|
14
34
|
const { messages, isThinking, streaming, streamingToolCalls, queuedMessage, sendMessage } = useChat(app);
|
|
15
35
|
const { handleCommand } = useCommands(app);
|
|
16
36
|
const [systemMessages, setSystemMessages] = useState([]);
|
|
17
37
|
const [showModelSelector, setShowModelSelector] = useState(false);
|
|
38
|
+
const [addModelMode, setAddModelMode] = useState(false);
|
|
18
39
|
const { stdout } = useStdout();
|
|
19
40
|
const terminalWidth = stdout?.columns || 80;
|
|
20
41
|
const onSubmit = useCallback(async (input) => {
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
42
|
+
// 添加新模型模式
|
|
43
|
+
if (addModelMode) {
|
|
44
|
+
const model = input.trim();
|
|
45
|
+
if (model) {
|
|
46
|
+
app.config.save({ model });
|
|
47
|
+
saveModelToHistory(model);
|
|
48
|
+
setSystemMessages(prev => [
|
|
49
|
+
...prev,
|
|
50
|
+
{ role: 'assistant', content: `模型已切换: ${model}` },
|
|
51
|
+
]);
|
|
52
|
+
}
|
|
53
|
+
setAddModelMode(false);
|
|
28
54
|
return;
|
|
29
55
|
}
|
|
56
|
+
if (showModelSelector)
|
|
57
|
+
return;
|
|
30
58
|
const result = handleCommand(input);
|
|
31
59
|
if (result.handled) {
|
|
32
|
-
if (input.trim().toLowerCase() === '/model') {
|
|
60
|
+
if (input.trim().toLowerCase() === '/model' || input.trim() === '/模型') {
|
|
33
61
|
setShowModelSelector(true);
|
|
34
62
|
return;
|
|
35
63
|
}
|
|
@@ -40,9 +68,8 @@ export function TuiApp({ app }) {
|
|
|
40
68
|
{ role: 'assistant', content: result.output },
|
|
41
69
|
]);
|
|
42
70
|
}
|
|
43
|
-
if (result.action === 'clear')
|
|
71
|
+
if (result.action === 'clear')
|
|
44
72
|
setSystemMessages([]);
|
|
45
|
-
}
|
|
46
73
|
if (result.action === 'reinit') {
|
|
47
74
|
const { WelcomeScreen } = await import('./welcome.js');
|
|
48
75
|
await WelcomeScreen.interactiveSetup();
|
|
@@ -51,22 +78,26 @@ export function TuiApp({ app }) {
|
|
|
51
78
|
Object.assign(app, newApp);
|
|
52
79
|
setSystemMessages(prev => [
|
|
53
80
|
...prev,
|
|
54
|
-
{ role: 'assistant', content: '
|
|
81
|
+
{ role: 'assistant', content: '配置已更新,模型: ' + app.config.get().model },
|
|
55
82
|
]);
|
|
56
83
|
}
|
|
57
84
|
return;
|
|
58
85
|
}
|
|
59
86
|
sendMessage(input);
|
|
60
|
-
}, [handleCommand, sendMessage, app, showModelSelector]);
|
|
87
|
+
}, [handleCommand, sendMessage, app, showModelSelector, addModelMode]);
|
|
61
88
|
const allMessages = [...systemMessages, ...messages];
|
|
62
89
|
const activeSkills = app.skills.listActive().map(s => s.id);
|
|
63
|
-
return (_jsxs(Box, { flexDirection: "column", paddingX: 1, children: [_jsx(Header, { provider: app.config.get().provider, model: app.config.get().model }), _jsx(ChatList, { messages: allMessages, streaming: streaming, streamingToolCalls: streamingToolCalls, width: terminalWidth - 4 }), _jsx(Thinking, { active: isThinking }), queuedMessage && (_jsxs(Box, { paddingLeft: 1, children: [_jsx(Text, { color: "#F59E0B", children: "\uD83D\uDCCE
|
|
90
|
+
return (_jsxs(Box, { flexDirection: "column", paddingX: 1, children: [_jsx(Header, { provider: app.config.get().provider, model: app.config.get().model }), _jsx(ChatList, { messages: allMessages, streaming: streaming, streamingToolCalls: streamingToolCalls, width: terminalWidth - 4 }), _jsx(Thinking, { active: isThinking }), queuedMessage && (_jsxs(Box, { paddingLeft: 1, children: [_jsx(Text, { color: "#F59E0B", children: "\uD83D\uDCCE \u5DF2\u6392\u961F: " }), _jsx(Text, { color: "#94A3B8", children: queuedMessage })] })), showModelSelector ? (_jsx(ModelSelector, { currentModel: app.config.get().model, onSelect: (model) => {
|
|
64
91
|
app.config.save({ model });
|
|
92
|
+
saveModelToHistory(model);
|
|
65
93
|
setShowModelSelector(false);
|
|
66
94
|
setSystemMessages(prev => [
|
|
67
95
|
...prev,
|
|
68
|
-
{ role: 'assistant', content:
|
|
96
|
+
{ role: 'assistant', content: `模型已切换: ${model}` },
|
|
69
97
|
]);
|
|
70
|
-
},
|
|
98
|
+
}, onAddNew: () => {
|
|
99
|
+
setShowModelSelector(false);
|
|
100
|
+
setAddModelMode(true);
|
|
101
|
+
} })) : addModelMode ? (_jsx(Box, { paddingLeft: 1, children: _jsx(Text, { color: "#06B6D4", bold: true, children: "\u8F93\u5165\u65B0\u6A21\u578B\u540D\u79F0: " }) })) : (_jsx(UserInput, { onSubmit: onSubmit, disabled: false })), _jsx(StatusBar, { messageCount: allMessages.length, skills: activeSkills })] }));
|
|
71
102
|
}
|
|
72
103
|
//# sourceMappingURL=app.js.map
|
package/dist/tui/app.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"app.js","sourceRoot":"","sources":["../../src/tui/app.tsx"],"names":[],"mappings":";AAAA,6BAA6B;AAC7B,OAAc,EAAE,QAAQ,EAAE,WAAW,EAAE,MAAM,OAAO,CAAC;AACrD,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,KAAK,CAAC;AAC3C,OAAO,EAAE,MAAM,EAAE,MAAM,wBAAwB,CAAC;AAChD,OAAO,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAC;AACpD,OAAO,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAC;AACpD,OAAO,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AACtD,OAAO,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AACtD,OAAO,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAC9D,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;AAC7C,OAAO,EAAE,WAAW,
|
|
1
|
+
{"version":3,"file":"app.js","sourceRoot":"","sources":["../../src/tui/app.tsx"],"names":[],"mappings":";AAAA,6BAA6B;AAC7B,OAAc,EAAE,QAAQ,EAAE,WAAW,EAAE,MAAM,OAAO,CAAC;AACrD,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,KAAK,CAAC;AAC3C,OAAO,EAAE,MAAM,EAAE,MAAM,wBAAwB,CAAC;AAChD,OAAO,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAC;AACpD,OAAO,EAAE,QAAQ,EAAE,MAAM,0BAA0B,CAAC;AACpD,OAAO,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AACtD,OAAO,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AACtD,OAAO,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAC9D,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;AAC7C,OAAO,EAAE,WAAW,EAAgB,MAAM,wBAAwB,CAAC;AAGnE,OAAO,EAAE,aAAa,EAAE,YAAY,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AACxE,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EAAE,OAAO,EAAE,MAAM,IAAI,CAAC;AAM7B,SAAS,kBAAkB,CAAC,KAAa;IACvC,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,WAAW,CAAC,CAAC;IACzC,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,EAAE,aAAa,CAAC,CAAC;IACtC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC;QAAE,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAE1D,IAAI,OAAO,GAAa,EAAE,CAAC;IAC3B,IAAI,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC;QACrB,IAAI,CAAC;YAAC,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;IACrE,CAAC;IACD,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;QAC7B,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACpB,aAAa,CAAC,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;IACxD,CAAC;AACH,CAAC;AAED,MAAM,UAAU,MAAM,CAAC,EAAE,GAAG,EAAS;IACnC,MAAM,EAAE,QAAQ,EAAE,UAAU,EAAE,SAAS,EAAE,kBAAkB,EAAE,aAAa,EAAE,WAAW,EAAE,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC;IACzG,MAAM,EAAE,aAAa,EAAE,GAAG,WAAW,CAAC,GAAG,CAAC,CAAC;IAC3C,MAAM,CAAC,cAAc,EAAE,iBAAiB,CAAC,GAAG,QAAQ,CAAgB,EAAE,CAAC,CAAC;IACxE,MAAM,CAAC,iBAAiB,EAAE,oBAAoB,CAAC,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC;IAClE,MAAM,CAAC,YAAY,EAAE,eAAe,CAAC,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC;IACxD,MAAM,EAAE,MAAM,EAAE,GAAG,SAAS,EAAE,CAAC;IAC/B,MAAM,aAAa,GAAG,MAAM,EAAE,OAAO,IAAI,EAAE,CAAC;IAE5C,MAAM,QAAQ,GAAG,WAAW,CAAC,KAAK,EAAE,KAAa,EAAE,EAAE;QACnD,UAAU;QACV,IAAI,YAAY,EAAE,CAAC;YACjB,MAAM,KAAK,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC;YAC3B,IAAI,KAAK,EAAE,CAAC;gBACV,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;gBAC3B,kBAAkB,CAAC,KAAK,CAAC,CAAC;gBAC1B,iBAAiB,CAAC,IAAI,CAAC,EAAE,CAAC;oBACxB,GAAG,IAAI;oBACP,EAAE,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,UAAU,KAAK,EAAE,EAAE;iBAClD,CAAC,CAAC;YACL,CAAC;YACD,eAAe,CAAC,KAAK,CAAC,CAAC;YACvB,OAAO;QACT,CAAC;QAED,IAAI,iBAAiB;YAAE,OAAO;QAE9B,MAAM,MAAM,GAAG,aAAa,CAAC,KAAK,CAAC,CAAC;QAEpC,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;YACnB,IAAI,KAAK,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,KAAK,QAAQ,IAAI,KAAK,CAAC,IAAI,EAAE,KAAK,KAAK,EAAE,CAAC;gBACtE,oBAAoB,CAAC,IAAI,CAAC,CAAC;gBAC3B,OAAO;YACT,CAAC;YAED,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;gBAClB,iBAAiB,CAAC,IAAI,CAAC,EAAE,CAAC;oBACxB,GAAG,IAAI;oBACP,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,KAAK,EAAE;oBAChC,EAAE,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,MAAM,CAAC,MAAO,EAAE;iBAC/C,CAAC,CAAC;YACL,CAAC;YAED,IAAI,MAAM,CAAC,MAAM,KAAK,OAAO;gBAAE,iBAAiB,CAAC,EAAE,CAAC,CAAC;YAErD,IAAI,MAAM,CAAC,MAAM,KAAK,QAAQ,EAAE,CAAC;gBAC/B,MAAM,EAAE,aAAa,EAAE,GAAG,MAAM,MAAM,CAAC,cAAc,CAAC,CAAC;gBACvD,MAAM,aAAa,CAAC,gBAAgB,EAAE,CAAC;gBACvC,MAAM,EAAE,GAAG,EAAE,QAAQ,EAAE,GAAG,MAAM,MAAM,CAAC,iBAAiB,CAAC,CAAC;gBAC1D,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,MAAM,EAAE,CAAC;gBACvC,MAAM,CAAC,MAAM,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;gBAC3B,iBAAiB,CAAC,IAAI,CAAC,EAAE,CAAC;oBACxB,GAAG,IAAI;oBACP,EAAE,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,YAAY,GAAG,GAAG,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,KAAK,EAAE;iBACtE,CAAC,CAAC;YACL,CAAC;YAED,OAAO;QACT,CAAC;QAED,WAAW,CAAC,KAAK,CAAC,CAAC;IACrB,CAAC,EAAE,CAAC,aAAa,EAAE,WAAW,EAAE,GAAG,EAAE,iBAAiB,EAAE,YAAY,CAAC,CAAC,CAAC;IAEvE,MAAM,WAAW,GAAG,CAAC,GAAG,cAAc,EAAE,GAAG,QAAQ,CAAC,CAAC;IACrD,MAAM,YAAY,GAAG,GAAG,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IAE5D,OAAO,CACL,MAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,QAAQ,EAAE,CAAC,aACrC,KAAC,MAAM,IAAC,QAAQ,EAAE,GAAG,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,GAAG,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,KAAK,GAAI,EAC9E,KAAC,QAAQ,IACP,QAAQ,EAAE,WAAW,EACrB,SAAS,EAAE,SAAS,EACpB,kBAAkB,EAAE,kBAAkB,EACtC,KAAK,EAAE,aAAa,GAAG,CAAC,GACxB,EACF,KAAC,QAAQ,IAAC,MAAM,EAAE,UAAU,GAAI,EAC/B,aAAa,IAAI,CAChB,MAAC,GAAG,IAAC,WAAW,EAAE,CAAC,aACjB,KAAC,IAAI,IAAC,KAAK,EAAC,SAAS,kDAAgB,EACrC,KAAC,IAAI,IAAC,KAAK,EAAC,SAAS,YAAE,aAAa,GAAQ,IACxC,CACP,EACA,iBAAiB,CAAC,CAAC,CAAC,CACnB,KAAC,aAAa,IACZ,YAAY,EAAE,GAAG,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,KAAK,EACpC,QAAQ,EAAE,CAAC,KAAK,EAAE,EAAE;oBAClB,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;oBAC3B,kBAAkB,CAAC,KAAK,CAAC,CAAC;oBAC1B,oBAAoB,CAAC,KAAK,CAAC,CAAC;oBAC5B,iBAAiB,CAAC,IAAI,CAAC,EAAE,CAAC;wBACxB,GAAG,IAAI;wBACP,EAAE,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,UAAU,KAAK,EAAE,EAAE;qBAClD,CAAC,CAAC;gBACL,CAAC,EACD,QAAQ,EAAE,GAAG,EAAE;oBACb,oBAAoB,CAAC,KAAK,CAAC,CAAC;oBAC5B,eAAe,CAAC,IAAI,CAAC,CAAC;gBACxB,CAAC,GACD,CACH,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CACjB,KAAC,GAAG,IAAC,WAAW,EAAE,CAAC,YACjB,KAAC,IAAI,IAAC,KAAK,EAAC,SAAS,EAAC,IAAI,mEAAiB,GACvC,CACP,CAAC,CAAC,CAAC,CACF,KAAC,SAAS,IAAC,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,GAAI,CACnD,EACD,KAAC,SAAS,IAAC,YAAY,EAAE,WAAW,CAAC,MAAM,EAAE,MAAM,EAAE,YAAY,GAAI,IACjE,CACP,CAAC;AACJ,CAAC"}
|
|
@@ -3,6 +3,6 @@ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
|
3
3
|
import React from 'react';
|
|
4
4
|
import { Box, Text } from 'ink';
|
|
5
5
|
export const Header = React.memo(function Header({ provider, model }) {
|
|
6
|
-
return (_jsxs(Box, { flexDirection: "column", marginBottom: 0, children: [_jsxs(Box, { justifyContent: "space-between", width: "100%", children: [_jsxs(Box, { children: [_jsx(Text, { color: "#06B6D4", bold: true, children: " \u26A1 " }), _jsx(Text, { color: "#22D3EE", bold: true, children: "THATGFSJ CODE" }), _jsx(Text, { dimColor: true, children: " v0.9.
|
|
6
|
+
return (_jsxs(Box, { flexDirection: "column", marginBottom: 0, children: [_jsxs(Box, { justifyContent: "space-between", width: "100%", children: [_jsxs(Box, { children: [_jsx(Text, { color: "#06B6D4", bold: true, children: " \u26A1 " }), _jsx(Text, { color: "#22D3EE", bold: true, children: "THATGFSJ CODE" }), _jsx(Text, { dimColor: true, children: " v0.9.5" })] }), _jsxs(Box, { children: [_jsxs(Text, { color: "#06B6D4", bold: true, children: [" ", provider, " "] }), _jsx(Text, { dimColor: true, children: "/" }), _jsxs(Text, { color: "#22D3EE", children: [" ", model, " "] })] })] }), _jsx(Text, { color: "#374151", children: '─'.repeat(80) })] }));
|
|
7
7
|
});
|
|
8
8
|
//# sourceMappingURL=Header.js.map
|
|
@@ -3,8 +3,8 @@ import React from 'react';
|
|
|
3
3
|
interface Props {
|
|
4
4
|
currentModel: string;
|
|
5
5
|
onSelect: (model: string) => void;
|
|
6
|
-
|
|
6
|
+
onAddNew: () => void;
|
|
7
7
|
}
|
|
8
|
-
export declare function ModelSelector({ currentModel, onSelect,
|
|
8
|
+
export declare function ModelSelector({ currentModel, onSelect, onAddNew }: Props): React.JSX.Element;
|
|
9
9
|
export {};
|
|
10
10
|
//# sourceMappingURL=ModelSelector.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ModelSelector.d.ts","sourceRoot":"","sources":["../../../src/tui/components/ModelSelector.tsx"],"names":[],"mappings":"AAAA,6BAA6B;AAC7B,OAAO,
|
|
1
|
+
{"version":3,"file":"ModelSelector.d.ts","sourceRoot":"","sources":["../../../src/tui/components/ModelSelector.tsx"],"names":[],"mappings":"AAAA,6BAA6B;AAC7B,OAAO,KAA8B,MAAM,OAAO,CAAC;AAOnD,UAAU,KAAK;IACb,YAAY,EAAE,MAAM,CAAC;IACrB,QAAQ,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;IAClC,QAAQ,EAAE,MAAM,IAAI,CAAC;CACtB;AAgDD,wBAAgB,aAAa,CAAC,EAAE,YAAY,EAAE,QAAQ,EAAE,QAAQ,EAAE,EAAE,KAAK,qBA0BxE"}
|
|
@@ -1,32 +1,60 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { jsxs as _jsxs, jsx as _jsx } from "react/jsx-runtime";
|
|
2
2
|
/** @jsxImportSource react */
|
|
3
|
-
import { useState } from 'react';
|
|
3
|
+
import { useState, useEffect } from 'react';
|
|
4
4
|
import { Box, Text } from 'ink';
|
|
5
5
|
import SelectInput from 'ink-select-input';
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
6
|
+
import { readFileSync, existsSync } from 'fs';
|
|
7
|
+
import { join } from 'path';
|
|
8
|
+
import { homedir } from 'os';
|
|
9
|
+
function loadSavedModels() {
|
|
10
|
+
const configPath = join(homedir(), '.thatgfsj', 'config.json');
|
|
11
|
+
const models = [];
|
|
12
|
+
const seen = new Set();
|
|
13
|
+
// Load from history if exists
|
|
14
|
+
const historyPath = join(homedir(), '.thatgfsj', 'models.json');
|
|
15
|
+
if (existsSync(historyPath)) {
|
|
16
|
+
try {
|
|
17
|
+
const history = JSON.parse(readFileSync(historyPath, 'utf-8'));
|
|
18
|
+
for (const m of history) {
|
|
19
|
+
if (!seen.has(m)) {
|
|
20
|
+
seen.add(m);
|
|
21
|
+
models.push({ label: m, value: m });
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
catch { }
|
|
26
26
|
}
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
27
|
+
// Always include current model
|
|
28
|
+
if (existsSync(configPath)) {
|
|
29
|
+
try {
|
|
30
|
+
const config = JSON.parse(readFileSync(configPath, 'utf-8'));
|
|
31
|
+
if (config.model && !seen.has(config.model)) {
|
|
32
|
+
seen.add(config.model);
|
|
33
|
+
models.push({ label: `${config.model} (当前)`, value: config.model });
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
catch { }
|
|
37
|
+
}
|
|
38
|
+
// Add some common defaults if list is empty
|
|
39
|
+
if (models.length === 0) {
|
|
40
|
+
const defaults = ['deepseek-chat', 'gpt-4o', 'mimo-v2.5-pro'];
|
|
41
|
+
for (const m of defaults) {
|
|
42
|
+
models.push({ label: m, value: m });
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
return models;
|
|
46
|
+
}
|
|
47
|
+
export function ModelSelector({ currentModel, onSelect, onAddNew }) {
|
|
48
|
+
const [items, setItems] = useState([]);
|
|
49
|
+
useEffect(() => {
|
|
50
|
+
const saved = loadSavedModels();
|
|
51
|
+
// Add "add new" option at the end
|
|
52
|
+
saved.push({ label: '+ 添加新模型', value: '__add_new__' });
|
|
53
|
+
setItems(saved);
|
|
54
|
+
}, []);
|
|
55
|
+
return (_jsxs(Box, { flexDirection: "column", paddingLeft: 1, children: [_jsxs(Text, { color: "#06B6D4", bold: true, children: ["\u5F53\u524D\u6A21\u578B: ", currentModel] }), _jsx(Text, { dimColor: true, children: "\u9009\u62E9\u6A21\u578B (\u2191\u2193 \u56DE\u8F66):" }), _jsx(SelectInput, { items: items, onSelect: (item) => {
|
|
56
|
+
if (item.value === '__add_new__') {
|
|
57
|
+
onAddNew();
|
|
30
58
|
}
|
|
31
59
|
else {
|
|
32
60
|
onSelect(item.value);
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ModelSelector.js","sourceRoot":"","sources":["../../../src/tui/components/ModelSelector.tsx"],"names":[],"mappings":";AAAA,6BAA6B;AAC7B,OAAc,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;
|
|
1
|
+
{"version":3,"file":"ModelSelector.js","sourceRoot":"","sources":["../../../src/tui/components/ModelSelector.tsx"],"names":[],"mappings":"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"}
|
|
@@ -5,6 +5,6 @@ import { Box, Text } from 'ink';
|
|
|
5
5
|
export const StatusBar = React.memo(function StatusBar({ messageCount, skills }) {
|
|
6
6
|
const activeSkills = skills.slice(0, 3).join(', ');
|
|
7
7
|
const moreSkills = skills.length > 3 ? ` +${skills.length - 3}` : '';
|
|
8
|
-
return (_jsxs(Box, { flexDirection: "column", marginTop: 0, children: [_jsx(Text, { color: "#374151", children: '─'.repeat(80) }), _jsxs(Box, { justifyContent: "space-between", width: "100%", children: [_jsxs(Box, { children: [_jsx(Text, { backgroundColor: "#06B6D4", color: "#0F172A", bold: true, children: " \u26A1 THATGFSJ CODE " }), _jsxs(Text, { dimColor: true, children: [" \u2502 ", messageCount, "
|
|
8
|
+
return (_jsxs(Box, { flexDirection: "column", marginTop: 0, children: [_jsx(Text, { color: "#374151", children: '─'.repeat(80) }), _jsxs(Box, { justifyContent: "space-between", width: "100%", children: [_jsxs(Box, { children: [_jsx(Text, { backgroundColor: "#06B6D4", color: "#0F172A", bold: true, children: " \u26A1 THATGFSJ CODE " }), _jsxs(Text, { dimColor: true, children: [" \u2502 ", messageCount, " \u6761\u6D88\u606F"] })] }), _jsx(Box, { children: skills.length > 0 && (_jsxs(Text, { dimColor: true, children: [" \u6280\u80FD: ", activeSkills, moreSkills, " "] })) })] })] }));
|
|
9
9
|
});
|
|
10
10
|
//# sourceMappingURL=StatusBar.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"StatusBar.js","sourceRoot":"","sources":["../../../src/tui/components/StatusBar.tsx"],"names":[],"mappings":";AAAA,6BAA6B;AAC7B,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,KAAK,CAAC;AAOhC,MAAM,CAAC,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC,SAAS,SAAS,CAAC,EAAE,YAAY,EAAE,MAAM,EAAS;IACpF,MAAM,YAAY,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACnD,MAAM,UAAU,GAAG,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;IAErE,OAAO,CACL,MAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,SAAS,EAAE,CAAC,aACtC,KAAC,IAAI,IAAC,KAAK,EAAC,SAAS,YAAE,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,GAAQ,EAC7C,MAAC,GAAG,IAAC,cAAc,EAAC,eAAe,EAAC,KAAK,EAAC,MAAM,aAC9C,MAAC,GAAG,eACF,KAAC,IAAI,IAAC,eAAe,EAAC,SAAS,EAAC,KAAK,EAAC,SAAS,EAAC,IAAI,6CAAyB,EAC7E,MAAC,IAAI,IAAC,QAAQ,+BAAK,YAAY,
|
|
1
|
+
{"version":3,"file":"StatusBar.js","sourceRoot":"","sources":["../../../src/tui/components/StatusBar.tsx"],"names":[],"mappings":";AAAA,6BAA6B;AAC7B,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,KAAK,CAAC;AAOhC,MAAM,CAAC,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC,SAAS,SAAS,CAAC,EAAE,YAAY,EAAE,MAAM,EAAS;IACpF,MAAM,YAAY,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACnD,MAAM,UAAU,GAAG,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;IAErE,OAAO,CACL,MAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,SAAS,EAAE,CAAC,aACtC,KAAC,IAAI,IAAC,KAAK,EAAC,SAAS,YAAE,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,GAAQ,EAC7C,MAAC,GAAG,IAAC,cAAc,EAAC,eAAe,EAAC,KAAK,EAAC,MAAM,aAC9C,MAAC,GAAG,eACF,KAAC,IAAI,IAAC,eAAe,EAAC,SAAS,EAAC,KAAK,EAAC,SAAS,EAAC,IAAI,6CAAyB,EAC7E,MAAC,IAAI,IAAC,QAAQ,+BAAK,YAAY,2BAAY,IACvC,EACN,KAAC,GAAG,cACD,MAAM,CAAC,MAAM,GAAG,CAAC,IAAI,CACpB,MAAC,IAAI,IAAC,QAAQ,sCAAO,YAAY,EAAE,UAAU,SAAS,CACvD,GACG,IACF,IACF,CACP,CAAC;AACJ,CAAC,CAAC,CAAC"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"UserInput.d.ts","sourceRoot":"","sources":["../../../src/tui/components/UserInput.tsx"],"names":[],"mappings":"AAAA,6BAA6B;AAC7B,OAAO,
|
|
1
|
+
{"version":3,"file":"UserInput.d.ts","sourceRoot":"","sources":["../../../src/tui/components/UserInput.tsx"],"names":[],"mappings":"AAAA,6BAA6B;AAC7B,OAAO,KAAgC,MAAM,OAAO,CAAC;AAIrD,UAAU,KAAK;IACb,QAAQ,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;IAClC,QAAQ,CAAC,EAAE,OAAO,CAAC;CACpB;AAED,wBAAgB,SAAS,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,EAAE,KAAK,qBAgHtD"}
|
|
@@ -2,15 +2,38 @@ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
|
2
2
|
/** @jsxImportSource react */
|
|
3
3
|
import { useState } from 'react';
|
|
4
4
|
import { Box, Text, useInput, useApp, useFocus } from 'ink';
|
|
5
|
+
import { COMMAND_LIST } from '../hooks/useCommands.js';
|
|
5
6
|
export function UserInput({ onSubmit, disabled }) {
|
|
6
7
|
const [value, setValue] = useState('');
|
|
7
8
|
const [history, setHistory] = useState([]);
|
|
8
9
|
const [historyIdx, setHistoryIdx] = useState(-1);
|
|
10
|
+
const [selectedCmd, setSelectedCmd] = useState(0);
|
|
9
11
|
const { exit } = useApp();
|
|
10
|
-
|
|
12
|
+
useFocus({ autoFocus: true });
|
|
13
|
+
const showCommands = value.startsWith('/') && value.length > 0 && !value.includes(' ');
|
|
14
|
+
const filteredCommands = showCommands
|
|
15
|
+
? COMMAND_LIST.filter(c => c.name.startsWith(value))
|
|
16
|
+
: [];
|
|
11
17
|
useInput((input, key) => {
|
|
12
18
|
if (disabled)
|
|
13
19
|
return;
|
|
20
|
+
// Command selector navigation
|
|
21
|
+
if (showCommands && filteredCommands.length > 0) {
|
|
22
|
+
if (key.upArrow) {
|
|
23
|
+
setSelectedCmd(prev => Math.max(0, prev - 1));
|
|
24
|
+
return;
|
|
25
|
+
}
|
|
26
|
+
if (key.downArrow) {
|
|
27
|
+
setSelectedCmd(prev => Math.min(filteredCommands.length - 1, prev + 1));
|
|
28
|
+
return;
|
|
29
|
+
}
|
|
30
|
+
if (key.tab || (key.return && filteredCommands.length === 1)) {
|
|
31
|
+
const selected = filteredCommands[selectedCmd] || filteredCommands[0];
|
|
32
|
+
setValue(selected.name + ' ');
|
|
33
|
+
setSelectedCmd(0);
|
|
34
|
+
return;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
14
37
|
if (key.return) {
|
|
15
38
|
const trimmed = value.trim();
|
|
16
39
|
if (trimmed) {
|
|
@@ -22,6 +45,7 @@ export function UserInput({ onSubmit, disabled }) {
|
|
|
22
45
|
setHistoryIdx(-1);
|
|
23
46
|
onSubmit(trimmed);
|
|
24
47
|
setValue('');
|
|
48
|
+
setSelectedCmd(0);
|
|
25
49
|
}
|
|
26
50
|
return;
|
|
27
51
|
}
|
|
@@ -45,6 +69,7 @@ export function UserInput({ onSubmit, disabled }) {
|
|
|
45
69
|
}
|
|
46
70
|
if (key.backspace || key.delete) {
|
|
47
71
|
setValue(v => v.slice(0, -1));
|
|
72
|
+
setSelectedCmd(0);
|
|
48
73
|
return;
|
|
49
74
|
}
|
|
50
75
|
if (key.ctrl && input === 'c') {
|
|
@@ -53,8 +78,9 @@ export function UserInput({ onSubmit, disabled }) {
|
|
|
53
78
|
}
|
|
54
79
|
if (input && !key.ctrl && !key.meta) {
|
|
55
80
|
setValue(v => v + input);
|
|
81
|
+
setSelectedCmd(0);
|
|
56
82
|
}
|
|
57
83
|
});
|
|
58
|
-
return (_jsxs(Box, { paddingY: 0, children: [_jsx(Text, { color: "#06B6D4", bold: true, children: disabled ? ' ' : '❯ ' }), _jsx(Text, { children: value }), !disabled && _jsx(Text, { color: "#06B6D4", children: "\u2588" })] }));
|
|
84
|
+
return (_jsxs(Box, { flexDirection: "column", children: [showCommands && filteredCommands.length > 0 && (_jsx(Box, { flexDirection: "column", paddingLeft: 2, marginBottom: 0, children: filteredCommands.map((cmd, i) => (_jsxs(Box, { children: [_jsx(Text, { color: i === selectedCmd ? '#06B6D4' : '#64748B', children: i === selectedCmd ? '▸ ' : ' ' }), _jsx(Text, { color: i === selectedCmd ? '#06B6D4' : '#94A3B8', bold: i === selectedCmd, children: cmd.name }), _jsxs(Text, { color: "#64748B", children: [" ", cmd.desc] })] }, cmd.name))) })), _jsxs(Box, { paddingY: 0, children: [_jsx(Text, { color: "#06B6D4", bold: true, children: disabled ? ' ' : '❯ ' }), _jsx(Text, { children: value }), !disabled && _jsx(Text, { color: "#06B6D4", children: "\u2588" })] })] }));
|
|
59
85
|
}
|
|
60
86
|
//# sourceMappingURL=UserInput.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"UserInput.js","sourceRoot":"","sources":["../../../src/tui/components/UserInput.tsx"],"names":[],"mappings":";AAAA,6BAA6B;AAC7B,OAAc,EAAE,QAAQ,
|
|
1
|
+
{"version":3,"file":"UserInput.js","sourceRoot":"","sources":["../../../src/tui/components/UserInput.tsx"],"names":[],"mappings":";AAAA,6BAA6B;AAC7B,OAAc,EAAE,QAAQ,EAAe,MAAM,OAAO,CAAC;AACrD,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,KAAK,CAAC;AAC5D,OAAO,EAAE,YAAY,EAAE,MAAM,yBAAyB,CAAC;AAOvD,MAAM,UAAU,SAAS,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAS;IACrD,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,CAAC,EAAE,CAAC,CAAC;IACvC,MAAM,CAAC,OAAO,EAAE,UAAU,CAAC,GAAG,QAAQ,CAAW,EAAE,CAAC,CAAC;IACrD,MAAM,CAAC,UAAU,EAAE,aAAa,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;IACjD,MAAM,CAAC,WAAW,EAAE,cAAc,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC;IAClD,MAAM,EAAE,IAAI,EAAE,GAAG,MAAM,EAAE,CAAC;IAC1B,QAAQ,CAAC,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAE9B,MAAM,YAAY,GAAG,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;IACvF,MAAM,gBAAgB,GAAG,YAAY;QACnC,CAAC,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;QACpD,CAAC,CAAC,EAAE,CAAC;IAEP,QAAQ,CAAC,CAAC,KAAK,EAAE,GAAG,EAAE,EAAE;QACtB,IAAI,QAAQ;YAAE,OAAO;QAErB,8BAA8B;QAC9B,IAAI,YAAY,IAAI,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAChD,IAAI,GAAG,CAAC,OAAO,EAAE,CAAC;gBAChB,cAAc,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,GAAG,CAAC,CAAC,CAAC,CAAC;gBAC9C,OAAO;YACT,CAAC;YACD,IAAI,GAAG,CAAC,SAAS,EAAE,CAAC;gBAClB,cAAc,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE,IAAI,GAAG,CAAC,CAAC,CAAC,CAAC;gBACxE,OAAO;YACT,CAAC;YACD,IAAI,GAAG,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,MAAM,IAAI,gBAAgB,CAAC,MAAM,KAAK,CAAC,CAAC,EAAE,CAAC;gBAC7D,MAAM,QAAQ,GAAG,gBAAgB,CAAC,WAAW,CAAC,IAAI,gBAAgB,CAAC,CAAC,CAAC,CAAC;gBACtE,QAAQ,CAAC,QAAQ,CAAC,IAAI,GAAG,GAAG,CAAC,CAAC;gBAC9B,cAAc,CAAC,CAAC,CAAC,CAAC;gBAClB,OAAO;YACT,CAAC;QACH,CAAC;QAED,IAAI,GAAG,CAAC,MAAM,EAAE,CAAC;YACf,MAAM,OAAO,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC;YAC7B,IAAI,OAAO,EAAE,CAAC;gBACZ,IAAI,OAAO,KAAK,MAAM,IAAI,OAAO,KAAK,MAAM,EAAE,CAAC;oBAC7C,IAAI,EAAE,CAAC;oBACP,OAAO;gBACT,CAAC;gBACD,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,GAAG,IAAI,EAAE,OAAO,CAAC,CAAC,CAAC;gBACvC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;gBAClB,QAAQ,CAAC,OAAO,CAAC,CAAC;gBAClB,QAAQ,CAAC,EAAE,CAAC,CAAC;gBACb,cAAc,CAAC,CAAC,CAAC,CAAC;YACpB,CAAC;YACD,OAAO;QACT,CAAC;QAED,IAAI,GAAG,CAAC,OAAO,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtC,MAAM,MAAM,GAAG,UAAU,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,UAAU,GAAG,CAAC,CAAC,CAAC;YACpF,aAAa,CAAC,MAAM,CAAC,CAAC;YACtB,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC;YAC1B,OAAO;QACT,CAAC;QAED,IAAI,GAAG,CAAC,SAAS,IAAI,UAAU,IAAI,CAAC,EAAE,CAAC;YACrC,MAAM,MAAM,GAAG,UAAU,GAAG,CAAC,CAAC;YAC9B,IAAI,MAAM,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;gBAC7B,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;gBAClB,QAAQ,CAAC,EAAE,CAAC,CAAC;YACf,CAAC;iBAAM,CAAC;gBACN,aAAa,CAAC,MAAM,CAAC,CAAC;gBACtB,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC;YAC5B,CAAC;YACD,OAAO;QACT,CAAC;QAED,IAAI,GAAG,CAAC,SAAS,IAAI,GAAG,CAAC,MAAM,EAAE,CAAC;YAChC,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;YAC9B,cAAc,CAAC,CAAC,CAAC,CAAC;YAClB,OAAO;QACT,CAAC;QAED,IAAI,GAAG,CAAC,IAAI,IAAI,KAAK,KAAK,GAAG,EAAE,CAAC;YAC9B,IAAI,EAAE,CAAC;YACP,OAAO;QACT,CAAC;QAED,IAAI,KAAK,IAAI,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC;YACpC,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,KAAK,CAAC,CAAC;YACzB,cAAc,CAAC,CAAC,CAAC,CAAC;QACpB,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,OAAO,CACL,MAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,aAExB,YAAY,IAAI,gBAAgB,CAAC,MAAM,GAAG,CAAC,IAAI,CAC9C,KAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,WAAW,EAAE,CAAC,EAAE,YAAY,EAAE,CAAC,YACxD,gBAAgB,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,CAChC,MAAC,GAAG,eACF,KAAC,IAAI,IAAC,KAAK,EAAE,CAAC,KAAK,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,YACnD,CAAC,KAAK,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,GAC3B,EACP,KAAC,IAAI,IAAC,KAAK,EAAE,CAAC,KAAK,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,EAAE,IAAI,EAAE,CAAC,KAAK,WAAW,YAC5E,GAAG,CAAC,IAAI,GACJ,EACP,MAAC,IAAI,IAAC,KAAK,EAAC,SAAS,mBAAI,GAAG,CAAC,IAAI,IAAQ,KAPjC,GAAG,CAAC,IAAI,CAQZ,CACP,CAAC,GACE,CACP,EAED,MAAC,GAAG,IAAC,QAAQ,EAAE,CAAC,aACd,KAAC,IAAI,IAAC,KAAK,EAAC,SAAS,EAAC,IAAI,kBAAE,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,GAAQ,EAC1D,KAAC,IAAI,cAAE,KAAK,GAAQ,EACnB,CAAC,QAAQ,IAAI,KAAC,IAAI,IAAC,KAAK,EAAC,SAAS,uBAAS,IACxC,IACF,CACP,CAAC;AACJ,CAAC"}
|
|
@@ -4,6 +4,10 @@ interface CommandResult {
|
|
|
4
4
|
output?: string;
|
|
5
5
|
action?: 'clear' | 'reinit';
|
|
6
6
|
}
|
|
7
|
+
export declare const COMMAND_LIST: {
|
|
8
|
+
name: string;
|
|
9
|
+
desc: string;
|
|
10
|
+
}[];
|
|
7
11
|
export declare function useCommands(app: App): {
|
|
8
12
|
handleCommand: (input: string) => CommandResult;
|
|
9
13
|
};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"useCommands.d.ts","sourceRoot":"","sources":["../../../src/tui/hooks/useCommands.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,oBAAoB,CAAC;AAE9C,UAAU,aAAa;IACrB,OAAO,EAAE,OAAO,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,CAAC,EAAE,OAAO,GAAG,QAAQ,CAAC;CAC7B;
|
|
1
|
+
{"version":3,"file":"useCommands.d.ts","sourceRoot":"","sources":["../../../src/tui/hooks/useCommands.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,oBAAoB,CAAC;AAE9C,UAAU,aAAa;IACrB,OAAO,EAAE,OAAO,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,CAAC,EAAE,OAAO,GAAG,QAAQ,CAAC;CAC7B;AAeD,eAAO,MAAM,YAAY;;;GAQxB,CAAC;AAEF,wBAAgB,WAAW,CAAC,GAAG,EAAE,GAAG;2BAEQ,MAAM,KAAG,aAAa;EAuHjE"}
|
|
@@ -1,10 +1,35 @@
|
|
|
1
1
|
import { useCallback } from 'react';
|
|
2
|
+
// 中文命令别名映射
|
|
3
|
+
const CMD_ALIASES = {
|
|
4
|
+
'/模型': '/model',
|
|
5
|
+
'/新建': '/new',
|
|
6
|
+
'/清空': '/new',
|
|
7
|
+
'/压缩': '/compact',
|
|
8
|
+
'/技能': '/skills',
|
|
9
|
+
'/技能管理': '/skills',
|
|
10
|
+
'/mcp': '/mcp',
|
|
11
|
+
'/帮助': '/help',
|
|
12
|
+
'/服务商': '/provider',
|
|
13
|
+
};
|
|
14
|
+
export const COMMAND_LIST = [
|
|
15
|
+
{ name: '/模型', desc: '切换模型' },
|
|
16
|
+
{ name: '/服务商', desc: '更换服务商' },
|
|
17
|
+
{ name: '/新建', desc: '新建会话' },
|
|
18
|
+
{ name: '/压缩', desc: '压缩上下文' },
|
|
19
|
+
{ name: '/技能', desc: '管理技能' },
|
|
20
|
+
{ name: '/mcp', desc: 'MCP 设置' },
|
|
21
|
+
{ name: '/帮助', desc: '查看帮助' },
|
|
22
|
+
];
|
|
2
23
|
export function useCommands(app) {
|
|
3
24
|
const handleCommand = useCallback((input) => {
|
|
4
25
|
const cmd = input.trim();
|
|
5
26
|
const parts = cmd.split(/\s+/);
|
|
6
|
-
|
|
27
|
+
let name = parts[0].toLowerCase();
|
|
7
28
|
const arg = parts.slice(1).join(' ');
|
|
29
|
+
// 中文别名映射
|
|
30
|
+
if (CMD_ALIASES[name]) {
|
|
31
|
+
name = CMD_ALIASES[name];
|
|
32
|
+
}
|
|
8
33
|
// ── /model [name] ───────────────────────────────────
|
|
9
34
|
if (name === '/model') {
|
|
10
35
|
if (!arg) {
|
|
@@ -12,59 +37,57 @@ export function useCommands(app) {
|
|
|
12
37
|
return {
|
|
13
38
|
handled: true,
|
|
14
39
|
output: [
|
|
15
|
-
|
|
40
|
+
`当前: ${c.provider} / ${c.model}`,
|
|
16
41
|
'',
|
|
17
|
-
'
|
|
18
|
-
'
|
|
19
|
-
'
|
|
20
|
-
'
|
|
21
|
-
' /model qwen3-235b-a22b',
|
|
42
|
+
'用法: /模型 <名称>',
|
|
43
|
+
' /模型 deepseek-chat',
|
|
44
|
+
' /模型 gpt-4o',
|
|
45
|
+
' /模型 claude-sonnet-4-20250514',
|
|
22
46
|
'',
|
|
23
|
-
'
|
|
47
|
+
'或: /服务商 更换服务商',
|
|
24
48
|
].join('\n'),
|
|
25
49
|
};
|
|
26
50
|
}
|
|
27
51
|
app.config.save({ model: arg });
|
|
28
|
-
return { handled: true, output:
|
|
52
|
+
return { handled: true, output: `模型 → ${arg}` };
|
|
29
53
|
}
|
|
30
54
|
// ── /provider ───────────────────────────────────────
|
|
31
55
|
if (name === '/provider') {
|
|
32
56
|
return { handled: true, action: 'reinit' };
|
|
33
57
|
}
|
|
34
58
|
// ── /new, /clear ────────────────────────────────────
|
|
35
|
-
if (name === '/new'
|
|
59
|
+
if (name === '/new') {
|
|
36
60
|
app.session.clear();
|
|
37
|
-
return { handled: true, output: '
|
|
61
|
+
return { handled: true, output: '新会话已创建。', action: 'clear' };
|
|
38
62
|
}
|
|
39
63
|
// ── /compact ────────────────────────────────────────
|
|
40
64
|
if (name === '/compact') {
|
|
41
65
|
const before = app.session.getMessageCount();
|
|
42
66
|
app.session.truncate();
|
|
43
67
|
const after = app.session.getMessageCount();
|
|
44
|
-
return { handled: true, output:
|
|
68
|
+
return { handled: true, output: `上下文已压缩: ${before} → ${after} 条消息` };
|
|
45
69
|
}
|
|
46
70
|
// ── /skills [id] ────────────────────────────────────
|
|
47
71
|
if (name === '/skills') {
|
|
48
72
|
if (arg) {
|
|
49
|
-
// Toggle specific skill
|
|
50
73
|
if (app.skills.isActive(arg)) {
|
|
51
74
|
app.skills.deactivate(arg);
|
|
52
|
-
return { handled: true, output: `✗ ${arg}` };
|
|
75
|
+
return { handled: true, output: `✗ 已关闭: ${arg}` };
|
|
53
76
|
}
|
|
54
77
|
else if (app.skills.activate(arg)) {
|
|
55
|
-
return { handled: true, output: `✓ ${arg}` };
|
|
78
|
+
return { handled: true, output: `✓ 已开启: ${arg}` };
|
|
56
79
|
}
|
|
57
|
-
return { handled: true, output:
|
|
80
|
+
return { handled: true, output: `未知技能: ${arg}` };
|
|
58
81
|
}
|
|
59
82
|
const all = app.skills.list();
|
|
60
83
|
const lines = [
|
|
61
|
-
'
|
|
84
|
+
'技能列表:',
|
|
62
85
|
...all.map(s => {
|
|
63
86
|
const mark = app.skills.isActive(s.id) ? '✓' : '·';
|
|
64
|
-
return ` ${mark} ${s.id}`;
|
|
87
|
+
return ` ${mark} ${s.id} ${s.description}`;
|
|
65
88
|
}),
|
|
66
89
|
'',
|
|
67
|
-
'
|
|
90
|
+
'用法: /技能 <id> 切换',
|
|
68
91
|
];
|
|
69
92
|
return { handled: true, output: lines.join('\n') };
|
|
70
93
|
}
|
|
@@ -73,29 +96,34 @@ export function useCommands(app) {
|
|
|
73
96
|
return {
|
|
74
97
|
handled: true,
|
|
75
98
|
output: [
|
|
76
|
-
'MCP
|
|
99
|
+
'MCP 配置: ~/.thatgfsj/mcp.json',
|
|
77
100
|
'',
|
|
78
101
|
' {',
|
|
79
102
|
' "servers": {',
|
|
80
|
-
' "
|
|
103
|
+
' "名称": { "command": "npx", "args": ["-y", "server"] }',
|
|
81
104
|
' }',
|
|
82
105
|
' }',
|
|
83
106
|
].join('\n'),
|
|
84
107
|
};
|
|
85
108
|
}
|
|
86
109
|
// ── /help ───────────────────────────────────────────
|
|
87
|
-
if (name === '/help'
|
|
110
|
+
if (name === '/help') {
|
|
88
111
|
return {
|
|
89
112
|
handled: true,
|
|
90
113
|
output: [
|
|
91
|
-
'
|
|
92
|
-
'
|
|
93
|
-
'
|
|
94
|
-
'
|
|
95
|
-
'
|
|
96
|
-
'
|
|
97
|
-
'/
|
|
98
|
-
'
|
|
114
|
+
'命令列表:',
|
|
115
|
+
' /模型 <名称> 切换模型',
|
|
116
|
+
' /服务商 更换服务商',
|
|
117
|
+
' /新建 新建会话',
|
|
118
|
+
' /压缩 压缩上下文',
|
|
119
|
+
' /技能 [id] 管理技能',
|
|
120
|
+
' /mcp MCP 设置',
|
|
121
|
+
' /帮助 查看帮助',
|
|
122
|
+
' exit 退出',
|
|
123
|
+
'',
|
|
124
|
+
'快捷键:',
|
|
125
|
+
' ↑/↓ 历史记录',
|
|
126
|
+
' Ctrl+C 退出',
|
|
99
127
|
].join('\n'),
|
|
100
128
|
};
|
|
101
129
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"useCommands.js","sourceRoot":"","sources":["../../../src/tui/hooks/useCommands.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,OAAO,CAAC;AASpC,MAAM,UAAU,WAAW,CAAC,GAAQ;IAElC,MAAM,aAAa,GAAG,WAAW,CAAC,CAAC,KAAa,EAAiB,EAAE;QACjE,MAAM,GAAG,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC;QACzB,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAC/B,
|
|
1
|
+
{"version":3,"file":"useCommands.js","sourceRoot":"","sources":["../../../src/tui/hooks/useCommands.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,OAAO,CAAC;AASpC,WAAW;AACX,MAAM,WAAW,GAA2B;IAC1C,KAAK,EAAE,QAAQ;IACf,KAAK,EAAE,MAAM;IACb,KAAK,EAAE,MAAM;IACb,KAAK,EAAE,UAAU;IACjB,KAAK,EAAE,SAAS;IAChB,OAAO,EAAE,SAAS;IAClB,MAAM,EAAE,MAAM;IACd,KAAK,EAAE,OAAO;IACd,MAAM,EAAE,WAAW;CACpB,CAAC;AAEF,MAAM,CAAC,MAAM,YAAY,GAAG;IAC1B,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE;IAC7B,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,OAAO,EAAE;IAC/B,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE;IAC7B,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,OAAO,EAAE;IAC9B,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE;IAC7B,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,QAAQ,EAAE;IAChC,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE;CAC9B,CAAC;AAEF,MAAM,UAAU,WAAW,CAAC,GAAQ;IAElC,MAAM,aAAa,GAAG,WAAW,CAAC,CAAC,KAAa,EAAiB,EAAE;QACjE,MAAM,GAAG,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC;QACzB,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAC/B,IAAI,IAAI,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;QAClC,MAAM,GAAG,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAErC,SAAS;QACT,IAAI,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC;YACtB,IAAI,GAAG,WAAW,CAAC,IAAI,CAAC,CAAC;QAC3B,CAAC;QAED,uDAAuD;QACvD,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;YACtB,IAAI,CAAC,GAAG,EAAE,CAAC;gBACT,MAAM,CAAC,GAAG,GAAG,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC;gBAC3B,OAAO;oBACL,OAAO,EAAE,IAAI;oBACb,MAAM,EAAE;wBACN,OAAO,CAAC,CAAC,QAAQ,MAAM,CAAC,CAAC,KAAK,EAAE;wBAChC,EAAE;wBACF,cAAc;wBACd,qBAAqB;wBACrB,cAAc;wBACd,gCAAgC;wBAChC,EAAE;wBACF,eAAe;qBAChB,CAAC,IAAI,CAAC,IAAI,CAAC;iBACb,CAAC;YACJ,CAAC;YACD,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC;YAChC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,QAAQ,GAAG,EAAE,EAAE,CAAC;QAClD,CAAC;QAED,uDAAuD;QACvD,IAAI,IAAI,KAAK,WAAW,EAAE,CAAC;YACzB,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC;QAC7C,CAAC;QAED,uDAAuD;QACvD,IAAI,IAAI,KAAK,MAAM,EAAE,CAAC;YACpB,GAAG,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;YACpB,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,OAAO,EAAE,CAAC;QAC/D,CAAC;QAED,uDAAuD;QACvD,IAAI,IAAI,KAAK,UAAU,EAAE,CAAC;YACxB,MAAM,MAAM,GAAG,GAAG,CAAC,OAAO,CAAC,eAAe,EAAE,CAAC;YAC7C,GAAG,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;YACvB,MAAM,KAAK,GAAG,GAAG,CAAC,OAAO,CAAC,eAAe,EAAE,CAAC;YAC5C,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,WAAW,MAAM,MAAM,KAAK,MAAM,EAAE,CAAC;QACvE,CAAC;QAED,uDAAuD;QACvD,IAAI,IAAI,KAAK,SAAS,EAAE,CAAC;YACvB,IAAI,GAAG,EAAE,CAAC;gBACR,IAAI,GAAG,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;oBAC7B,GAAG,CAAC,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;oBAC3B,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,UAAU,GAAG,EAAE,EAAE,CAAC;gBACpD,CAAC;qBAAM,IAAI,GAAG,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;oBACpC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,UAAU,GAAG,EAAE,EAAE,CAAC;gBACpD,CAAC;gBACD,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,SAAS,GAAG,EAAE,EAAE,CAAC;YACnD,CAAC;YAED,MAAM,GAAG,GAAG,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;YAC9B,MAAM,KAAK,GAAG;gBACZ,OAAO;gBACP,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;oBACb,MAAM,IAAI,GAAG,GAAG,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;oBACnD,OAAO,KAAK,IAAI,IAAI,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,WAAW,EAAE,CAAC;gBAC/C,CAAC,CAAC;gBACF,EAAE;gBACF,iBAAiB;aAClB,CAAC;YACF,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;QACrD,CAAC;QAED,uDAAuD;QACvD,IAAI,IAAI,KAAK,MAAM,EAAE,CAAC;YACpB,OAAO;gBACL,OAAO,EAAE,IAAI;gBACb,MAAM,EAAE;oBACN,8BAA8B;oBAC9B,EAAE;oBACF,KAAK;oBACL,kBAAkB;oBAClB,4DAA4D;oBAC5D,OAAO;oBACP,KAAK;iBACN,CAAC,IAAI,CAAC,IAAI,CAAC;aACb,CAAC;QACJ,CAAC;QAED,uDAAuD;QACvD,IAAI,IAAI,KAAK,OAAO,EAAE,CAAC;YACrB,OAAO;gBACL,OAAO,EAAE,IAAI;gBACb,MAAM,EAAE;oBACN,OAAO;oBACP,oBAAoB;oBACpB,uBAAuB;oBACvB,uBAAuB;oBACvB,wBAAwB;oBACxB,uBAAuB;oBACvB,2BAA2B;oBAC3B,uBAAuB;oBACvB,uBAAuB;oBACvB,EAAE;oBACF,MAAM;oBACN,yBAAyB;oBACzB,uBAAuB;iBACxB,CAAC,IAAI,CAAC,IAAI,CAAC;aACb,CAAC;QACJ,CAAC;QAED,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;IAC5B,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IAEV,OAAO,EAAE,aAAa,EAAE,CAAC;AAC3B,CAAC"}
|
package/package.json
CHANGED
package/src/cmd/index.tsx
CHANGED
|
@@ -28,7 +28,7 @@ process.on('unhandledRejection', (reason) => {
|
|
|
28
28
|
program
|
|
29
29
|
.name('gfcode')
|
|
30
30
|
.description('Thatgfsj Code - AI Coding Assistant')
|
|
31
|
-
.version('0.9.
|
|
31
|
+
.version('0.9.5')
|
|
32
32
|
.argument('[prompt]', 'Task to execute (omit to start interactive mode)')
|
|
33
33
|
.option('-m, --model <model>', 'Specify model')
|
|
34
34
|
.option('-i, --interactive', 'Force interactive mode')
|
package/src/tui/app.tsx
CHANGED
|
@@ -8,37 +8,63 @@ import { UserInput } from './components/UserInput.js';
|
|
|
8
8
|
import { StatusBar } from './components/StatusBar.js';
|
|
9
9
|
import { ModelSelector } from './components/ModelSelector.js';
|
|
10
10
|
import { useChat } from './hooks/useChat.js';
|
|
11
|
-
import { useCommands } from './hooks/useCommands.js';
|
|
11
|
+
import { useCommands, COMMAND_LIST } from './hooks/useCommands.js';
|
|
12
12
|
import type { App } from '../app/index.js';
|
|
13
13
|
import type { MessageData } from './components/ChatMessage.js';
|
|
14
|
+
import { writeFileSync, readFileSync, existsSync, mkdirSync } from 'fs';
|
|
15
|
+
import { join } from 'path';
|
|
16
|
+
import { homedir } from 'os';
|
|
14
17
|
|
|
15
18
|
interface Props {
|
|
16
19
|
app: App;
|
|
17
20
|
}
|
|
18
21
|
|
|
22
|
+
function saveModelToHistory(model: string) {
|
|
23
|
+
const dir = join(homedir(), '.thatgfsj');
|
|
24
|
+
const path = join(dir, 'models.json');
|
|
25
|
+
if (!existsSync(dir)) mkdirSync(dir, { recursive: true });
|
|
26
|
+
|
|
27
|
+
let history: string[] = [];
|
|
28
|
+
if (existsSync(path)) {
|
|
29
|
+
try { history = JSON.parse(readFileSync(path, 'utf-8')); } catch {}
|
|
30
|
+
}
|
|
31
|
+
if (!history.includes(model)) {
|
|
32
|
+
history.push(model);
|
|
33
|
+
writeFileSync(path, JSON.stringify(history, null, 2));
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
|
|
19
37
|
export function TuiApp({ app }: Props) {
|
|
20
38
|
const { messages, isThinking, streaming, streamingToolCalls, queuedMessage, sendMessage } = useChat(app);
|
|
21
39
|
const { handleCommand } = useCommands(app);
|
|
22
40
|
const [systemMessages, setSystemMessages] = useState<MessageData[]>([]);
|
|
23
41
|
const [showModelSelector, setShowModelSelector] = useState(false);
|
|
42
|
+
const [addModelMode, setAddModelMode] = useState(false);
|
|
24
43
|
const { stdout } = useStdout();
|
|
25
44
|
const terminalWidth = stdout?.columns || 80;
|
|
26
45
|
|
|
27
46
|
const onSubmit = useCallback(async (input: string) => {
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
47
|
+
// 添加新模型模式
|
|
48
|
+
if (addModelMode) {
|
|
49
|
+
const model = input.trim();
|
|
50
|
+
if (model) {
|
|
51
|
+
app.config.save({ model });
|
|
52
|
+
saveModelToHistory(model);
|
|
53
|
+
setSystemMessages(prev => [
|
|
54
|
+
...prev,
|
|
55
|
+
{ role: 'assistant', content: `模型已切换: ${model}` },
|
|
56
|
+
]);
|
|
57
|
+
}
|
|
58
|
+
setAddModelMode(false);
|
|
35
59
|
return;
|
|
36
60
|
}
|
|
37
61
|
|
|
62
|
+
if (showModelSelector) return;
|
|
63
|
+
|
|
38
64
|
const result = handleCommand(input);
|
|
39
65
|
|
|
40
66
|
if (result.handled) {
|
|
41
|
-
if (input.trim().toLowerCase() === '/model') {
|
|
67
|
+
if (input.trim().toLowerCase() === '/model' || input.trim() === '/模型') {
|
|
42
68
|
setShowModelSelector(true);
|
|
43
69
|
return;
|
|
44
70
|
}
|
|
@@ -51,9 +77,7 @@ export function TuiApp({ app }: Props) {
|
|
|
51
77
|
]);
|
|
52
78
|
}
|
|
53
79
|
|
|
54
|
-
if (result.action === 'clear')
|
|
55
|
-
setSystemMessages([]);
|
|
56
|
-
}
|
|
80
|
+
if (result.action === 'clear') setSystemMessages([]);
|
|
57
81
|
|
|
58
82
|
if (result.action === 'reinit') {
|
|
59
83
|
const { WelcomeScreen } = await import('./welcome.js');
|
|
@@ -63,7 +87,7 @@ export function TuiApp({ app }: Props) {
|
|
|
63
87
|
Object.assign(app, newApp);
|
|
64
88
|
setSystemMessages(prev => [
|
|
65
89
|
...prev,
|
|
66
|
-
{ role: 'assistant', content: '
|
|
90
|
+
{ role: 'assistant', content: '配置已更新,模型: ' + app.config.get().model },
|
|
67
91
|
]);
|
|
68
92
|
}
|
|
69
93
|
|
|
@@ -71,7 +95,7 @@ export function TuiApp({ app }: Props) {
|
|
|
71
95
|
}
|
|
72
96
|
|
|
73
97
|
sendMessage(input);
|
|
74
|
-
}, [handleCommand, sendMessage, app, showModelSelector]);
|
|
98
|
+
}, [handleCommand, sendMessage, app, showModelSelector, addModelMode]);
|
|
75
99
|
|
|
76
100
|
const allMessages = [...systemMessages, ...messages];
|
|
77
101
|
const activeSkills = app.skills.listActive().map(s => s.id);
|
|
@@ -88,7 +112,7 @@ export function TuiApp({ app }: Props) {
|
|
|
88
112
|
<Thinking active={isThinking} />
|
|
89
113
|
{queuedMessage && (
|
|
90
114
|
<Box paddingLeft={1}>
|
|
91
|
-
<Text color="#F59E0B">📎
|
|
115
|
+
<Text color="#F59E0B">📎 已排队: </Text>
|
|
92
116
|
<Text color="#94A3B8">{queuedMessage}</Text>
|
|
93
117
|
</Box>
|
|
94
118
|
)}
|
|
@@ -97,14 +121,22 @@ export function TuiApp({ app }: Props) {
|
|
|
97
121
|
currentModel={app.config.get().model}
|
|
98
122
|
onSelect={(model) => {
|
|
99
123
|
app.config.save({ model });
|
|
124
|
+
saveModelToHistory(model);
|
|
100
125
|
setShowModelSelector(false);
|
|
101
126
|
setSystemMessages(prev => [
|
|
102
127
|
...prev,
|
|
103
|
-
{ role: 'assistant', content:
|
|
128
|
+
{ role: 'assistant', content: `模型已切换: ${model}` },
|
|
104
129
|
]);
|
|
105
130
|
}}
|
|
106
|
-
|
|
131
|
+
onAddNew={() => {
|
|
132
|
+
setShowModelSelector(false);
|
|
133
|
+
setAddModelMode(true);
|
|
134
|
+
}}
|
|
107
135
|
/>
|
|
136
|
+
) : addModelMode ? (
|
|
137
|
+
<Box paddingLeft={1}>
|
|
138
|
+
<Text color="#06B6D4" bold>输入新模型名称: </Text>
|
|
139
|
+
</Box>
|
|
108
140
|
) : (
|
|
109
141
|
<UserInput onSubmit={onSubmit} disabled={false} />
|
|
110
142
|
)}
|
|
@@ -14,7 +14,7 @@ export const Header = React.memo(function Header({ provider, model }: Props) {
|
|
|
14
14
|
<Box>
|
|
15
15
|
<Text color="#06B6D4" bold> ⚡ </Text>
|
|
16
16
|
<Text color="#22D3EE" bold>THATGFSJ CODE</Text>
|
|
17
|
-
<Text dimColor> v0.9.
|
|
17
|
+
<Text dimColor> v0.9.5</Text>
|
|
18
18
|
</Box>
|
|
19
19
|
<Box>
|
|
20
20
|
<Text color="#06B6D4" bold> {provider} </Text>
|
|
@@ -1,56 +1,82 @@
|
|
|
1
1
|
/** @jsxImportSource react */
|
|
2
|
-
import React, { useState } from 'react';
|
|
2
|
+
import React, { useState, useEffect } from 'react';
|
|
3
3
|
import { Box, Text } from 'ink';
|
|
4
4
|
import SelectInput from 'ink-select-input';
|
|
5
|
+
import { readFileSync, existsSync } from 'fs';
|
|
6
|
+
import { join } from 'path';
|
|
7
|
+
import { homedir } from 'os';
|
|
5
8
|
|
|
6
9
|
interface Props {
|
|
7
10
|
currentModel: string;
|
|
8
11
|
onSelect: (model: string) => void;
|
|
9
|
-
|
|
12
|
+
onAddNew: () => void;
|
|
10
13
|
}
|
|
11
14
|
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
<Box flexDirection="column">
|
|
35
|
-
<Text color="#06B6D4" bold>Enter model name:</Text>
|
|
36
|
-
<Box>
|
|
37
|
-
<Text color="#06B6D4">❯ </Text>
|
|
38
|
-
<Text>{customValue}</Text>
|
|
39
|
-
<Text color="#06B6D4">█</Text>
|
|
40
|
-
</Box>
|
|
41
|
-
</Box>
|
|
42
|
-
);
|
|
15
|
+
interface SavedModel {
|
|
16
|
+
label: string;
|
|
17
|
+
value: string;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
function loadSavedModels(): SavedModel[] {
|
|
21
|
+
const configPath = join(homedir(), '.thatgfsj', 'config.json');
|
|
22
|
+
const models: SavedModel[] = [];
|
|
23
|
+
const seen = new Set<string>();
|
|
24
|
+
|
|
25
|
+
// Load from history if exists
|
|
26
|
+
const historyPath = join(homedir(), '.thatgfsj', 'models.json');
|
|
27
|
+
if (existsSync(historyPath)) {
|
|
28
|
+
try {
|
|
29
|
+
const history = JSON.parse(readFileSync(historyPath, 'utf-8'));
|
|
30
|
+
for (const m of history) {
|
|
31
|
+
if (!seen.has(m)) {
|
|
32
|
+
seen.add(m);
|
|
33
|
+
models.push({ label: m, value: m });
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
} catch {}
|
|
43
37
|
}
|
|
44
38
|
|
|
39
|
+
// Always include current model
|
|
40
|
+
if (existsSync(configPath)) {
|
|
41
|
+
try {
|
|
42
|
+
const config = JSON.parse(readFileSync(configPath, 'utf-8'));
|
|
43
|
+
if (config.model && !seen.has(config.model)) {
|
|
44
|
+
seen.add(config.model);
|
|
45
|
+
models.push({ label: `${config.model} (当前)`, value: config.model });
|
|
46
|
+
}
|
|
47
|
+
} catch {}
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
// Add some common defaults if list is empty
|
|
51
|
+
if (models.length === 0) {
|
|
52
|
+
const defaults = ['deepseek-chat', 'gpt-4o', 'mimo-v2.5-pro'];
|
|
53
|
+
for (const m of defaults) {
|
|
54
|
+
models.push({ label: m, value: m });
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
return models;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
export function ModelSelector({ currentModel, onSelect, onAddNew }: Props) {
|
|
62
|
+
const [items, setItems] = useState<SavedModel[]>([]);
|
|
63
|
+
|
|
64
|
+
useEffect(() => {
|
|
65
|
+
const saved = loadSavedModels();
|
|
66
|
+
// Add "add new" option at the end
|
|
67
|
+
saved.push({ label: '+ 添加新模型', value: '__add_new__' });
|
|
68
|
+
setItems(saved);
|
|
69
|
+
}, []);
|
|
70
|
+
|
|
45
71
|
return (
|
|
46
|
-
<Box flexDirection="column">
|
|
47
|
-
<Text color="#06B6D4" bold
|
|
48
|
-
<Text dimColor
|
|
72
|
+
<Box flexDirection="column" paddingLeft={1}>
|
|
73
|
+
<Text color="#06B6D4" bold>当前模型: {currentModel}</Text>
|
|
74
|
+
<Text dimColor>选择模型 (↑↓ 回车):</Text>
|
|
49
75
|
<SelectInput
|
|
50
|
-
items={
|
|
76
|
+
items={items}
|
|
51
77
|
onSelect={(item) => {
|
|
52
|
-
if (item.value === '
|
|
53
|
-
|
|
78
|
+
if (item.value === '__add_new__') {
|
|
79
|
+
onAddNew();
|
|
54
80
|
} else {
|
|
55
81
|
onSelect(item.value);
|
|
56
82
|
}
|
|
@@ -17,11 +17,11 @@ export const StatusBar = React.memo(function StatusBar({ messageCount, skills }:
|
|
|
17
17
|
<Box justifyContent="space-between" width="100%">
|
|
18
18
|
<Box>
|
|
19
19
|
<Text backgroundColor="#06B6D4" color="#0F172A" bold> ⚡ THATGFSJ CODE </Text>
|
|
20
|
-
<Text dimColor> │ {messageCount}
|
|
20
|
+
<Text dimColor> │ {messageCount} 条消息</Text>
|
|
21
21
|
</Box>
|
|
22
22
|
<Box>
|
|
23
23
|
{skills.length > 0 && (
|
|
24
|
-
<Text dimColor>
|
|
24
|
+
<Text dimColor> 技能: {activeSkills}{moreSkills} </Text>
|
|
25
25
|
)}
|
|
26
26
|
</Box>
|
|
27
27
|
</Box>
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
/** @jsxImportSource react */
|
|
2
|
-
import React, { useState } from 'react';
|
|
2
|
+
import React, { useState, useCallback } from 'react';
|
|
3
3
|
import { Box, Text, useInput, useApp, useFocus } from 'ink';
|
|
4
|
+
import { COMMAND_LIST } from '../hooks/useCommands.js';
|
|
4
5
|
|
|
5
6
|
interface Props {
|
|
6
7
|
onSubmit: (input: string) => void;
|
|
@@ -11,12 +12,36 @@ export function UserInput({ onSubmit, disabled }: Props) {
|
|
|
11
12
|
const [value, setValue] = useState('');
|
|
12
13
|
const [history, setHistory] = useState<string[]>([]);
|
|
13
14
|
const [historyIdx, setHistoryIdx] = useState(-1);
|
|
15
|
+
const [selectedCmd, setSelectedCmd] = useState(0);
|
|
14
16
|
const { exit } = useApp();
|
|
15
|
-
|
|
17
|
+
useFocus({ autoFocus: true });
|
|
18
|
+
|
|
19
|
+
const showCommands = value.startsWith('/') && value.length > 0 && !value.includes(' ');
|
|
20
|
+
const filteredCommands = showCommands
|
|
21
|
+
? COMMAND_LIST.filter(c => c.name.startsWith(value))
|
|
22
|
+
: [];
|
|
16
23
|
|
|
17
24
|
useInput((input, key) => {
|
|
18
25
|
if (disabled) return;
|
|
19
26
|
|
|
27
|
+
// Command selector navigation
|
|
28
|
+
if (showCommands && filteredCommands.length > 0) {
|
|
29
|
+
if (key.upArrow) {
|
|
30
|
+
setSelectedCmd(prev => Math.max(0, prev - 1));
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
if (key.downArrow) {
|
|
34
|
+
setSelectedCmd(prev => Math.min(filteredCommands.length - 1, prev + 1));
|
|
35
|
+
return;
|
|
36
|
+
}
|
|
37
|
+
if (key.tab || (key.return && filteredCommands.length === 1)) {
|
|
38
|
+
const selected = filteredCommands[selectedCmd] || filteredCommands[0];
|
|
39
|
+
setValue(selected.name + ' ');
|
|
40
|
+
setSelectedCmd(0);
|
|
41
|
+
return;
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
|
|
20
45
|
if (key.return) {
|
|
21
46
|
const trimmed = value.trim();
|
|
22
47
|
if (trimmed) {
|
|
@@ -28,6 +53,7 @@ export function UserInput({ onSubmit, disabled }: Props) {
|
|
|
28
53
|
setHistoryIdx(-1);
|
|
29
54
|
onSubmit(trimmed);
|
|
30
55
|
setValue('');
|
|
56
|
+
setSelectedCmd(0);
|
|
31
57
|
}
|
|
32
58
|
return;
|
|
33
59
|
}
|
|
@@ -53,6 +79,7 @@ export function UserInput({ onSubmit, disabled }: Props) {
|
|
|
53
79
|
|
|
54
80
|
if (key.backspace || key.delete) {
|
|
55
81
|
setValue(v => v.slice(0, -1));
|
|
82
|
+
setSelectedCmd(0);
|
|
56
83
|
return;
|
|
57
84
|
}
|
|
58
85
|
|
|
@@ -63,14 +90,34 @@ export function UserInput({ onSubmit, disabled }: Props) {
|
|
|
63
90
|
|
|
64
91
|
if (input && !key.ctrl && !key.meta) {
|
|
65
92
|
setValue(v => v + input);
|
|
93
|
+
setSelectedCmd(0);
|
|
66
94
|
}
|
|
67
95
|
});
|
|
68
96
|
|
|
69
97
|
return (
|
|
70
|
-
<Box
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
98
|
+
<Box flexDirection="column">
|
|
99
|
+
{/* Command selector */}
|
|
100
|
+
{showCommands && filteredCommands.length > 0 && (
|
|
101
|
+
<Box flexDirection="column" paddingLeft={2} marginBottom={0}>
|
|
102
|
+
{filteredCommands.map((cmd, i) => (
|
|
103
|
+
<Box key={cmd.name}>
|
|
104
|
+
<Text color={i === selectedCmd ? '#06B6D4' : '#64748B'}>
|
|
105
|
+
{i === selectedCmd ? '▸ ' : ' '}
|
|
106
|
+
</Text>
|
|
107
|
+
<Text color={i === selectedCmd ? '#06B6D4' : '#94A3B8'} bold={i === selectedCmd}>
|
|
108
|
+
{cmd.name}
|
|
109
|
+
</Text>
|
|
110
|
+
<Text color="#64748B"> {cmd.desc}</Text>
|
|
111
|
+
</Box>
|
|
112
|
+
))}
|
|
113
|
+
</Box>
|
|
114
|
+
)}
|
|
115
|
+
{/* Input line */}
|
|
116
|
+
<Box paddingY={0}>
|
|
117
|
+
<Text color="#06B6D4" bold>{disabled ? ' ' : '❯ '}</Text>
|
|
118
|
+
<Text>{value}</Text>
|
|
119
|
+
{!disabled && <Text color="#06B6D4">█</Text>}
|
|
120
|
+
</Box>
|
|
74
121
|
</Box>
|
|
75
122
|
);
|
|
76
123
|
}
|
|
@@ -7,14 +7,42 @@ interface CommandResult {
|
|
|
7
7
|
action?: 'clear' | 'reinit';
|
|
8
8
|
}
|
|
9
9
|
|
|
10
|
+
// 中文命令别名映射
|
|
11
|
+
const CMD_ALIASES: Record<string, string> = {
|
|
12
|
+
'/模型': '/model',
|
|
13
|
+
'/新建': '/new',
|
|
14
|
+
'/清空': '/new',
|
|
15
|
+
'/压缩': '/compact',
|
|
16
|
+
'/技能': '/skills',
|
|
17
|
+
'/技能管理': '/skills',
|
|
18
|
+
'/mcp': '/mcp',
|
|
19
|
+
'/帮助': '/help',
|
|
20
|
+
'/服务商': '/provider',
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
export const COMMAND_LIST = [
|
|
24
|
+
{ name: '/模型', desc: '切换模型' },
|
|
25
|
+
{ name: '/服务商', desc: '更换服务商' },
|
|
26
|
+
{ name: '/新建', desc: '新建会话' },
|
|
27
|
+
{ name: '/压缩', desc: '压缩上下文' },
|
|
28
|
+
{ name: '/技能', desc: '管理技能' },
|
|
29
|
+
{ name: '/mcp', desc: 'MCP 设置' },
|
|
30
|
+
{ name: '/帮助', desc: '查看帮助' },
|
|
31
|
+
];
|
|
32
|
+
|
|
10
33
|
export function useCommands(app: App) {
|
|
11
34
|
|
|
12
35
|
const handleCommand = useCallback((input: string): CommandResult => {
|
|
13
36
|
const cmd = input.trim();
|
|
14
37
|
const parts = cmd.split(/\s+/);
|
|
15
|
-
|
|
38
|
+
let name = parts[0].toLowerCase();
|
|
16
39
|
const arg = parts.slice(1).join(' ');
|
|
17
40
|
|
|
41
|
+
// 中文别名映射
|
|
42
|
+
if (CMD_ALIASES[name]) {
|
|
43
|
+
name = CMD_ALIASES[name];
|
|
44
|
+
}
|
|
45
|
+
|
|
18
46
|
// ── /model [name] ───────────────────────────────────
|
|
19
47
|
if (name === '/model') {
|
|
20
48
|
if (!arg) {
|
|
@@ -22,20 +50,19 @@ export function useCommands(app: App) {
|
|
|
22
50
|
return {
|
|
23
51
|
handled: true,
|
|
24
52
|
output: [
|
|
25
|
-
|
|
53
|
+
`当前: ${c.provider} / ${c.model}`,
|
|
26
54
|
'',
|
|
27
|
-
'
|
|
28
|
-
'
|
|
29
|
-
'
|
|
30
|
-
'
|
|
31
|
-
' /model qwen3-235b-a22b',
|
|
55
|
+
'用法: /模型 <名称>',
|
|
56
|
+
' /模型 deepseek-chat',
|
|
57
|
+
' /模型 gpt-4o',
|
|
58
|
+
' /模型 claude-sonnet-4-20250514',
|
|
32
59
|
'',
|
|
33
|
-
'
|
|
60
|
+
'或: /服务商 更换服务商',
|
|
34
61
|
].join('\n'),
|
|
35
62
|
};
|
|
36
63
|
}
|
|
37
64
|
app.config.save({ model: arg });
|
|
38
|
-
return { handled: true, output:
|
|
65
|
+
return { handled: true, output: `模型 → ${arg}` };
|
|
39
66
|
}
|
|
40
67
|
|
|
41
68
|
// ── /provider ───────────────────────────────────────
|
|
@@ -44,9 +71,9 @@ export function useCommands(app: App) {
|
|
|
44
71
|
}
|
|
45
72
|
|
|
46
73
|
// ── /new, /clear ────────────────────────────────────
|
|
47
|
-
if (name === '/new'
|
|
74
|
+
if (name === '/new') {
|
|
48
75
|
app.session.clear();
|
|
49
|
-
return { handled: true, output: '
|
|
76
|
+
return { handled: true, output: '新会话已创建。', action: 'clear' };
|
|
50
77
|
}
|
|
51
78
|
|
|
52
79
|
// ── /compact ────────────────────────────────────────
|
|
@@ -54,31 +81,30 @@ export function useCommands(app: App) {
|
|
|
54
81
|
const before = app.session.getMessageCount();
|
|
55
82
|
app.session.truncate();
|
|
56
83
|
const after = app.session.getMessageCount();
|
|
57
|
-
return { handled: true, output:
|
|
84
|
+
return { handled: true, output: `上下文已压缩: ${before} → ${after} 条消息` };
|
|
58
85
|
}
|
|
59
86
|
|
|
60
87
|
// ── /skills [id] ────────────────────────────────────
|
|
61
88
|
if (name === '/skills') {
|
|
62
89
|
if (arg) {
|
|
63
|
-
// Toggle specific skill
|
|
64
90
|
if (app.skills.isActive(arg)) {
|
|
65
91
|
app.skills.deactivate(arg);
|
|
66
|
-
return { handled: true, output: `✗ ${arg}` };
|
|
92
|
+
return { handled: true, output: `✗ 已关闭: ${arg}` };
|
|
67
93
|
} else if (app.skills.activate(arg)) {
|
|
68
|
-
return { handled: true, output: `✓ ${arg}` };
|
|
94
|
+
return { handled: true, output: `✓ 已开启: ${arg}` };
|
|
69
95
|
}
|
|
70
|
-
return { handled: true, output:
|
|
96
|
+
return { handled: true, output: `未知技能: ${arg}` };
|
|
71
97
|
}
|
|
72
98
|
|
|
73
99
|
const all = app.skills.list();
|
|
74
100
|
const lines = [
|
|
75
|
-
'
|
|
101
|
+
'技能列表:',
|
|
76
102
|
...all.map(s => {
|
|
77
103
|
const mark = app.skills.isActive(s.id) ? '✓' : '·';
|
|
78
|
-
return ` ${mark} ${s.id}`;
|
|
104
|
+
return ` ${mark} ${s.id} ${s.description}`;
|
|
79
105
|
}),
|
|
80
106
|
'',
|
|
81
|
-
'
|
|
107
|
+
'用法: /技能 <id> 切换',
|
|
82
108
|
];
|
|
83
109
|
return { handled: true, output: lines.join('\n') };
|
|
84
110
|
}
|
|
@@ -88,11 +114,11 @@ export function useCommands(app: App) {
|
|
|
88
114
|
return {
|
|
89
115
|
handled: true,
|
|
90
116
|
output: [
|
|
91
|
-
'MCP
|
|
117
|
+
'MCP 配置: ~/.thatgfsj/mcp.json',
|
|
92
118
|
'',
|
|
93
119
|
' {',
|
|
94
120
|
' "servers": {',
|
|
95
|
-
' "
|
|
121
|
+
' "名称": { "command": "npx", "args": ["-y", "server"] }',
|
|
96
122
|
' }',
|
|
97
123
|
' }',
|
|
98
124
|
].join('\n'),
|
|
@@ -100,18 +126,23 @@ export function useCommands(app: App) {
|
|
|
100
126
|
}
|
|
101
127
|
|
|
102
128
|
// ── /help ───────────────────────────────────────────
|
|
103
|
-
if (name === '/help'
|
|
129
|
+
if (name === '/help') {
|
|
104
130
|
return {
|
|
105
131
|
handled: true,
|
|
106
132
|
output: [
|
|
107
|
-
'
|
|
108
|
-
'
|
|
109
|
-
'
|
|
110
|
-
'
|
|
111
|
-
'
|
|
112
|
-
'
|
|
113
|
-
'/
|
|
114
|
-
'
|
|
133
|
+
'命令列表:',
|
|
134
|
+
' /模型 <名称> 切换模型',
|
|
135
|
+
' /服务商 更换服务商',
|
|
136
|
+
' /新建 新建会话',
|
|
137
|
+
' /压缩 压缩上下文',
|
|
138
|
+
' /技能 [id] 管理技能',
|
|
139
|
+
' /mcp MCP 设置',
|
|
140
|
+
' /帮助 查看帮助',
|
|
141
|
+
' exit 退出',
|
|
142
|
+
'',
|
|
143
|
+
'快捷键:',
|
|
144
|
+
' ↑/↓ 历史记录',
|
|
145
|
+
' Ctrl+C 退出',
|
|
115
146
|
].join('\n'),
|
|
116
147
|
};
|
|
117
148
|
}
|