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 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.4')
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')
@@ -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;AAG3C,UAAU,KAAK;IACb,GAAG,EAAE,GAAG,CAAC;CACV;AAED,wBAAgB,MAAM,CAAC,EAAE,GAAG,EAAE,EAAE,KAAK,qBA+FpC"}
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
- if (showModelSelector) {
22
- app.config.save({ model: input.trim() });
23
- setShowModelSelector(false);
24
- setSystemMessages(prev => [
25
- ...prev,
26
- { role: 'assistant', content: `Model → ${input.trim()}` },
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: 'Config updated. Model: ' + app.config.get().model },
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 Queued: " }), _jsx(Text, { color: "#94A3B8", children: queuedMessage })] })), showModelSelector ? (_jsx(ModelSelector, { currentModel: app.config.get().model, onSelect: (model) => {
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: `Model ${model}` },
96
+ { role: 'assistant', content: `模型已切换: ${model}` },
69
97
  ]);
70
- }, onCancel: () => setShowModelSelector(false) })) : (_jsx(UserInput, { onSubmit: onSubmit, disabled: false })), _jsx(StatusBar, { messageCount: allMessages.length, skills: activeSkills })] }));
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
@@ -1 +1 @@
1
- {"version":3,"file":"app.js","sourceRoot":"","sources":["../../src/tui/app.tsx"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"app.js","sourceRoot":"","sources":["../../src/tui/app.tsx"],"names":[],"mappings":"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"}
@@ -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.4" })] }), _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) })] }));
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
- onCancel: () => void;
6
+ onAddNew: () => void;
7
7
  }
8
- export declare function ModelSelector({ currentModel, onSelect, onCancel }: Props): React.JSX.Element;
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,KAAmB,MAAM,OAAO,CAAC;AAIxC,UAAU,KAAK;IACb,YAAY,EAAE,MAAM,CAAC;IACrB,QAAQ,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;IAClC,QAAQ,EAAE,MAAM,IAAI,CAAC;CACtB;AAkBD,wBAAgB,aAAa,CAAC,EAAE,YAAY,EAAE,QAAQ,EAAE,QAAQ,EAAE,EAAE,KAAK,qBAiCxE"}
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 { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
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
- const COMMON_MODELS = [
7
- { label: 'deepseek-chat', value: 'deepseek-chat' },
8
- { label: 'deepseek-reasoner', value: 'deepseek-reasoner' },
9
- { label: 'gpt-4o', value: 'gpt-4o' },
10
- { label: 'gpt-4o-mini', value: 'gpt-4o-mini' },
11
- { label: 'claude-sonnet-4-20250514', value: 'claude-sonnet-4-20250514' },
12
- { label: 'claude-3-5-haiku-20241022', value: 'claude-3-5-haiku-20241022' },
13
- { label: 'gemini-2.0-flash', value: 'gemini-2.0-flash' },
14
- { label: 'gemini-2.5-pro', value: 'gemini-2.5-pro' },
15
- { label: 'qwen3-235b-a22b', value: 'qwen3-235b-a22b' },
16
- { label: 'glm-4-flash', value: 'glm-4-flash' },
17
- { label: 'kimi-k2.6', value: 'kimi-k2.6' },
18
- { label: 'mimo-v2.5-pro', value: 'mimo-v2.5-pro' },
19
- { label: '─ type custom name below ─', value: '__custom__' },
20
- ];
21
- export function ModelSelector({ currentModel, onSelect, onCancel }) {
22
- const [customMode, setCustomMode] = useState(false);
23
- const [customValue, setCustomValue] = useState('');
24
- if (customMode) {
25
- return (_jsxs(Box, { flexDirection: "column", children: [_jsx(Text, { color: "#06B6D4", bold: true, children: "Enter model name:" }), _jsxs(Box, { children: [_jsx(Text, { color: "#06B6D4", children: "\u276F " }), _jsx(Text, { children: customValue }), _jsx(Text, { color: "#06B6D4", children: "\u2588" })] })] }));
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
- return (_jsxs(Box, { flexDirection: "column", children: [_jsxs(Text, { color: "#06B6D4", bold: true, children: ["Current: ", currentModel] }), _jsx(Text, { dimColor: true, children: "Select model (\u2191\u2193 Enter):" }), _jsx(SelectInput, { items: COMMON_MODELS, onSelect: (item) => {
28
- if (item.value === '__custom__') {
29
- setCustomMode(true);
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;AACxC,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,KAAK,CAAC;AAChC,OAAO,WAAW,MAAM,kBAAkB,CAAC;AAQ3C,MAAM,aAAa,GAAG;IACpB,EAAE,KAAK,EAAE,eAAe,EAAE,KAAK,EAAE,eAAe,EAAE;IAClD,EAAE,KAAK,EAAE,mBAAmB,EAAE,KAAK,EAAE,mBAAmB,EAAE;IAC1D,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE;IACpC,EAAE,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,aAAa,EAAE;IAC9C,EAAE,KAAK,EAAE,0BAA0B,EAAE,KAAK,EAAE,0BAA0B,EAAE;IACxE,EAAE,KAAK,EAAE,2BAA2B,EAAE,KAAK,EAAE,2BAA2B,EAAE;IAC1E,EAAE,KAAK,EAAE,kBAAkB,EAAE,KAAK,EAAE,kBAAkB,EAAE;IACxD,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,gBAAgB,EAAE;IACpD,EAAE,KAAK,EAAE,iBAAiB,EAAE,KAAK,EAAE,iBAAiB,EAAE;IACtD,EAAE,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,aAAa,EAAE;IAC9C,EAAE,KAAK,EAAE,WAAW,EAAE,KAAK,EAAE,WAAW,EAAE;IAC1C,EAAE,KAAK,EAAE,eAAe,EAAE,KAAK,EAAE,eAAe,EAAE;IAClD,EAAE,KAAK,EAAE,4BAA4B,EAAE,KAAK,EAAE,YAAY,EAAE;CAC7D,CAAC;AAEF,MAAM,UAAU,aAAa,CAAC,EAAE,YAAY,EAAE,QAAQ,EAAE,QAAQ,EAAS;IACvE,MAAM,CAAC,UAAU,EAAE,aAAa,CAAC,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC;IACpD,MAAM,CAAC,WAAW,EAAE,cAAc,CAAC,GAAG,QAAQ,CAAC,EAAE,CAAC,CAAC;IAEnD,IAAI,UAAU,EAAE,CAAC;QACf,OAAO,CACL,MAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,aACzB,KAAC,IAAI,IAAC,KAAK,EAAC,SAAS,EAAC,IAAI,wCAAyB,EACnD,MAAC,GAAG,eACF,KAAC,IAAI,IAAC,KAAK,EAAC,SAAS,wBAAU,EAC/B,KAAC,IAAI,cAAE,WAAW,GAAQ,EAC1B,KAAC,IAAI,IAAC,KAAK,EAAC,SAAS,uBAAS,IAC1B,IACF,CACP,CAAC;IACJ,CAAC;IAED,OAAO,CACL,MAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,aACzB,MAAC,IAAI,IAAC,KAAK,EAAC,SAAS,EAAC,IAAI,gCAAW,YAAY,IAAQ,EACzD,KAAC,IAAI,IAAC,QAAQ,yDAAgC,EAC9C,KAAC,WAAW,IACV,KAAK,EAAE,aAAa,EACpB,QAAQ,EAAE,CAAC,IAAI,EAAE,EAAE;oBACjB,IAAI,IAAI,CAAC,KAAK,KAAK,YAAY,EAAE,CAAC;wBAChC,aAAa,CAAC,IAAI,CAAC,CAAC;oBACtB,CAAC;yBAAM,CAAC;wBACN,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;oBACvB,CAAC;gBACH,CAAC,GACD,IACE,CACP,CAAC;AACJ,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, " messages"] })] }), _jsx(Box, { children: skills.length > 0 && (_jsxs(Text, { dimColor: true, children: [" skills: ", activeSkills, moreSkills, " "] })) })] })] }));
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,iBAAiB,IAC5C,EACN,KAAC,GAAG,cACD,MAAM,CAAC,MAAM,GAAG,CAAC,IAAI,CACpB,MAAC,IAAI,IAAC,QAAQ,gCAAW,YAAY,EAAE,UAAU,SAAS,CAC3D,GACG,IACF,IACF,CACP,CAAC;AACJ,CAAC,CAAC,CAAC"}
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,KAAmB,MAAM,OAAO,CAAC;AAGxC,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,qBAkEtD"}
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
- const { isFocused } = useFocus({ autoFocus: true });
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":"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"}
1
+ {"version":3,"file":"UserInput.js","sourceRoot":"","sources":["../../../src/tui/components/UserInput.tsx"],"names":[],"mappings":"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"}
@@ -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;AAED,wBAAgB,WAAW,CAAC,GAAG,EAAE,GAAG;2BAEQ,MAAM,KAAG,aAAa;EA+GjE"}
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
- const name = parts[0].toLowerCase();
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
- `Current: ${c.provider} / ${c.model}`,
40
+ `当前: ${c.provider} / ${c.model}`,
16
41
  '',
17
- 'Usage: /model <name>',
18
- ' /model deepseek-chat',
19
- ' /model gpt-4o',
20
- ' /model claude-sonnet-4-20250514',
21
- ' /model qwen3-235b-a22b',
42
+ '用法: /模型 <名称>',
43
+ ' /模型 deepseek-chat',
44
+ ' /模型 gpt-4o',
45
+ ' /模型 claude-sonnet-4-20250514',
22
46
  '',
23
- 'Or: /provider to change provider',
47
+ '或: /服务商 更换服务商',
24
48
  ].join('\n'),
25
49
  };
26
50
  }
27
51
  app.config.save({ model: arg });
28
- return { handled: true, output: `Model → ${arg}` };
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' || name === '/clear') {
59
+ if (name === '/new') {
36
60
  app.session.clear();
37
- return { handled: true, output: 'New session.', action: 'clear' };
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: `Compacted: ${before} → ${after} messages` };
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: `Unknown skill: ${arg}` };
80
+ return { handled: true, output: `未知技能: ${arg}` };
58
81
  }
59
82
  const all = app.skills.list();
60
83
  const lines = [
61
- 'Skills:',
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
- 'Usage: /skills <id> to toggle',
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: ~/.thatgfsj/mcp.json',
99
+ 'MCP 配置: ~/.thatgfsj/mcp.json',
77
100
  '',
78
101
  ' {',
79
102
  ' "servers": {',
80
- ' "name": { "command": "npx", "args": ["-y", "server"] }',
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' || name === 'help') {
110
+ if (name === '/help') {
88
111
  return {
89
112
  handled: true,
90
113
  output: [
91
- '/model <name> Switch model',
92
- '/provider Change provider',
93
- '/new New session',
94
- '/compact Compress context',
95
- '/skills [id] List/toggle skills',
96
- '/mcp MCP settings',
97
- '/help This help',
98
- 'exit Quit',
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":"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"}
1
+ {"version":3,"file":"useCommands.js","sourceRoot":"","sources":["../../../src/tui/hooks/useCommands.ts"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "thatgfsj-code",
3
- "version": "0.9.4",
3
+ "version": "0.9.5",
4
4
  "description": "Thatgfsj Code - AI Coding Assistant",
5
5
  "main": "dist/cmd/index.js",
6
6
  "type": "module",
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.4')
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
- if (showModelSelector) {
29
- app.config.save({ model: input.trim() });
30
- setShowModelSelector(false);
31
- setSystemMessages(prev => [
32
- ...prev,
33
- { role: 'assistant', content: `Model → ${input.trim()}` },
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: 'Config updated. Model: ' + app.config.get().model },
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">📎 Queued: </Text>
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: `Model ${model}` },
128
+ { role: 'assistant', content: `模型已切换: ${model}` },
104
129
  ]);
105
130
  }}
106
- onCancel={() => setShowModelSelector(false)}
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.4</Text>
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
- onCancel: () => void;
12
+ onAddNew: () => void;
10
13
  }
11
14
 
12
- const COMMON_MODELS = [
13
- { label: 'deepseek-chat', value: 'deepseek-chat' },
14
- { label: 'deepseek-reasoner', value: 'deepseek-reasoner' },
15
- { label: 'gpt-4o', value: 'gpt-4o' },
16
- { label: 'gpt-4o-mini', value: 'gpt-4o-mini' },
17
- { label: 'claude-sonnet-4-20250514', value: 'claude-sonnet-4-20250514' },
18
- { label: 'claude-3-5-haiku-20241022', value: 'claude-3-5-haiku-20241022' },
19
- { label: 'gemini-2.0-flash', value: 'gemini-2.0-flash' },
20
- { label: 'gemini-2.5-pro', value: 'gemini-2.5-pro' },
21
- { label: 'qwen3-235b-a22b', value: 'qwen3-235b-a22b' },
22
- { label: 'glm-4-flash', value: 'glm-4-flash' },
23
- { label: 'kimi-k2.6', value: 'kimi-k2.6' },
24
- { label: 'mimo-v2.5-pro', value: 'mimo-v2.5-pro' },
25
- { label: '─ type custom name below ─', value: '__custom__' },
26
- ];
27
-
28
- export function ModelSelector({ currentModel, onSelect, onCancel }: Props) {
29
- const [customMode, setCustomMode] = useState(false);
30
- const [customValue, setCustomValue] = useState('');
31
-
32
- if (customMode) {
33
- return (
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>Current: {currentModel}</Text>
48
- <Text dimColor>Select model (↑↓ Enter):</Text>
72
+ <Box flexDirection="column" paddingLeft={1}>
73
+ <Text color="#06B6D4" bold>当前模型: {currentModel}</Text>
74
+ <Text dimColor>选择模型 (↑↓ 回车):</Text>
49
75
  <SelectInput
50
- items={COMMON_MODELS}
76
+ items={items}
51
77
  onSelect={(item) => {
52
- if (item.value === '__custom__') {
53
- setCustomMode(true);
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} messages</Text>
20
+ <Text dimColor> │ {messageCount} 条消息</Text>
21
21
  </Box>
22
22
  <Box>
23
23
  {skills.length > 0 && (
24
- <Text dimColor> skills: {activeSkills}{moreSkills} </Text>
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
- const { isFocused } = useFocus({ autoFocus: true });
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 paddingY={0}>
71
- <Text color="#06B6D4" bold>{disabled ? ' ' : '❯ '}</Text>
72
- <Text>{value}</Text>
73
- {!disabled && <Text color="#06B6D4">█</Text>}
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
- const name = parts[0].toLowerCase();
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
- `Current: ${c.provider} / ${c.model}`,
53
+ `当前: ${c.provider} / ${c.model}`,
26
54
  '',
27
- 'Usage: /model <name>',
28
- ' /model deepseek-chat',
29
- ' /model gpt-4o',
30
- ' /model claude-sonnet-4-20250514',
31
- ' /model qwen3-235b-a22b',
55
+ '用法: /模型 <名称>',
56
+ ' /模型 deepseek-chat',
57
+ ' /模型 gpt-4o',
58
+ ' /模型 claude-sonnet-4-20250514',
32
59
  '',
33
- 'Or: /provider to change provider',
60
+ '或: /服务商 更换服务商',
34
61
  ].join('\n'),
35
62
  };
36
63
  }
37
64
  app.config.save({ model: arg });
38
- return { handled: true, output: `Model → ${arg}` };
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' || name === '/clear') {
74
+ if (name === '/new') {
48
75
  app.session.clear();
49
- return { handled: true, output: 'New session.', action: 'clear' };
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: `Compacted: ${before} → ${after} messages` };
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: `Unknown skill: ${arg}` };
96
+ return { handled: true, output: `未知技能: ${arg}` };
71
97
  }
72
98
 
73
99
  const all = app.skills.list();
74
100
  const lines = [
75
- 'Skills:',
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
- 'Usage: /skills <id> to toggle',
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: ~/.thatgfsj/mcp.json',
117
+ 'MCP 配置: ~/.thatgfsj/mcp.json',
92
118
  '',
93
119
  ' {',
94
120
  ' "servers": {',
95
- ' "name": { "command": "npx", "args": ["-y", "server"] }',
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' || name === 'help') {
129
+ if (name === '/help') {
104
130
  return {
105
131
  handled: true,
106
132
  output: [
107
- '/model <name> Switch model',
108
- '/provider Change provider',
109
- '/new New session',
110
- '/compact Compress context',
111
- '/skills [id] List/toggle skills',
112
- '/mcp MCP settings',
113
- '/help This help',
114
- 'exit Quit',
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
  }