thatgfsj-code 0.9.6 → 0.9.8

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.6')
27
+ .version('0.9.8')
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;AAM3C,UAAU,KAAK;IACb,GAAG,EAAE,GAAG,CAAC;CACV;AAiBD,wBAAgB,MAAM,CAAC,EAAE,GAAG,EAAE,EAAE,KAAK,qBA6GpC"}
1
+ {"version":3,"file":"app.d.ts","sourceRoot":"","sources":["../../src/tui/app.tsx"],"names":[],"mappings":"AAAA,6BAA6B;AAC7B,OAAO,KAAgC,MAAM,OAAO,CAAC;AAWrD,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,iBAAiB,CAAC;AAO3C,UAAU,KAAK;IACb,GAAG,EAAE,GAAG,CAAC;CACV;AAkBD,wBAAgB,MAAM,CAAC,EAAE,GAAG,EAAE,EAAE,KAAK,qBAkJpC"}
package/dist/tui/app.js CHANGED
@@ -8,6 +8,7 @@ import { Thinking } from './components/Thinking.js';
8
8
  import { UserInput } from './components/UserInput.js';
9
9
  import { StatusBar } from './components/StatusBar.js';
10
10
  import { ModelSelector } from './components/ModelSelector.js';
11
+ import { InitWizard } from './components/InitWizard.js';
11
12
  import { useChat } from './hooks/useChat.js';
12
13
  import { useCommands } from './hooks/useCommands.js';
13
14
  import { writeFileSync, readFileSync, existsSync, mkdirSync } from 'fs';
@@ -34,31 +35,61 @@ export function TuiApp({ app }) {
34
35
  const { messages, isThinking, streaming, streamingToolCalls, queuedMessage, sendMessage } = useChat(app);
35
36
  const { handleCommand } = useCommands(app);
36
37
  const [systemMessages, setSystemMessages] = useState([]);
37
- const [showModelSelector, setShowModelSelector] = useState(false);
38
- const [addModelMode, setAddModelMode] = useState(false);
38
+ const [viewMode, setViewMode] = useState('chat');
39
+ const [initProvider, setInitProvider] = useState('siliconflow');
40
+ const [initUrl, setInitUrl] = useState('');
39
41
  const { stdout } = useStdout();
40
42
  const terminalWidth = stdout?.columns || 80;
43
+ const addMsg = useCallback((content) => {
44
+ setSystemMessages(prev => [...prev, { role: 'assistant', content }]);
45
+ }, []);
41
46
  const onSubmit = useCallback(async (input) => {
42
- // 添加新模型模式
43
- if (addModelMode) {
47
+ // Init wizard - API Key input
48
+ if (viewMode === 'init_key') {
49
+ const key = input.trim();
50
+ if (!key) {
51
+ setViewMode('chat');
52
+ return;
53
+ }
54
+ // Save and complete
55
+ const { App: AppClass } = await import('../app/index.js');
56
+ const cfg = app.config.get();
57
+ app.config.save({ provider: initProvider, apiKey: key, model: cfg.model || 'default' });
58
+ saveModelToHistory(cfg.model || 'default');
59
+ setViewMode('chat');
60
+ addMsg(`配置已更新: ${initProvider}`);
61
+ return;
62
+ }
63
+ // Init wizard - Custom URL input
64
+ if (viewMode === 'init_url') {
65
+ const url = input.trim();
66
+ if (!url) {
67
+ setViewMode('chat');
68
+ return;
69
+ }
70
+ setInitUrl(url);
71
+ setViewMode('init_key');
72
+ return;
73
+ }
74
+ // Init wizard - Custom model name
75
+ if (viewMode === 'init_custom_model') {
44
76
  const model = input.trim();
45
77
  if (model) {
46
78
  app.config.save({ model });
47
79
  saveModelToHistory(model);
48
- setSystemMessages(prev => [
49
- ...prev,
50
- { role: 'assistant', content: `模型已切换: ${model}` },
51
- ]);
80
+ addMsg(`模型已切换: ${model}`);
52
81
  }
53
- setAddModelMode(false);
82
+ setViewMode('chat');
54
83
  return;
55
84
  }
56
- if (showModelSelector)
85
+ // Model selector - ignore text input
86
+ if (viewMode === 'model_select')
57
87
  return;
88
+ // Normal command handling
58
89
  const result = handleCommand(input);
59
90
  if (result.handled) {
60
- if (input.trim().toLowerCase() === '/model' || input.trim() === '/模型') {
61
- setShowModelSelector(true);
91
+ if (input.trim() === '/模型' || input.trim() === '/model') {
92
+ setViewMode('model_select');
62
93
  return;
63
94
  }
64
95
  if (result.output) {
@@ -71,33 +102,24 @@ export function TuiApp({ app }) {
71
102
  if (result.action === 'clear')
72
103
  setSystemMessages([]);
73
104
  if (result.action === 'reinit') {
74
- const { WelcomeScreen } = await import('./welcome.js');
75
- await WelcomeScreen.interactiveSetup();
76
- const { App: AppClass } = await import('../app/index.js');
77
- const newApp = await AppClass.create();
78
- Object.assign(app, newApp);
79
- setSystemMessages(prev => [
80
- ...prev,
81
- { role: 'assistant', content: '配置已更新,模型: ' + app.config.get().model },
82
- ]);
105
+ setViewMode('init_wizard');
83
106
  }
84
107
  return;
85
108
  }
86
109
  sendMessage(input);
87
- }, [handleCommand, sendMessage, app, showModelSelector, addModelMode]);
110
+ }, [handleCommand, sendMessage, app, viewMode, initProvider, initUrl, addMsg]);
88
111
  const allMessages = [...systemMessages, ...messages];
89
112
  const activeSkills = app.skills.listActive().map(s => s.id);
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) => {
113
+ 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 })] })), viewMode === 'model_select' ? (_jsx(ModelSelector, { currentModel: app.config.get().model, onSelect: (model) => {
91
114
  app.config.save({ model });
92
115
  saveModelToHistory(model);
93
- setShowModelSelector(false);
94
- setSystemMessages(prev => [
95
- ...prev,
96
- { role: 'assistant', content: `模型已切换: ${model}` },
97
- ]);
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 })] }));
116
+ setViewMode('chat');
117
+ addMsg(`模型已切换: ${model}`);
118
+ }, onAddNew: () => setViewMode('init_wizard') })) : viewMode === 'init_wizard' ? (_jsx(InitWizard, { onComplete: (provider, model, apiKey, baseUrl) => {
119
+ app.config.save({ provider, model, apiKey, baseUrl });
120
+ saveModelToHistory(model);
121
+ setViewMode('chat');
122
+ addMsg(`配置完成: ${provider} / ${model}`);
123
+ }, onCancel: () => setViewMode('chat') })) : (_jsxs(Box, { flexDirection: "column", children: [viewMode === 'init_key' && (_jsxs(Box, { paddingLeft: 1, marginBottom: 0, children: [_jsx(Text, { color: "#06B6D4", bold: true, children: "\u8F93\u5165 API Key: " }), _jsxs(Text, { dimColor: true, children: ["(", initProvider, ")"] })] })), viewMode === 'init_url' && (_jsx(Box, { paddingLeft: 1, marginBottom: 0, children: _jsx(Text, { color: "#06B6D4", bold: true, children: "\u8F93\u5165\u4E2D\u8F6C\u7AD9 URL: " }) })), viewMode === 'init_custom_model' && (_jsx(Box, { paddingLeft: 1, marginBottom: 0, children: _jsx(Text, { color: "#06B6D4", bold: true, children: "\u8F93\u5165\u6A21\u578B\u540D\u79F0: " }) })), _jsx(UserInput, { onSubmit: onSubmit, disabled: false })] })), _jsx(StatusBar, { messageCount: allMessages.length, skills: activeSkills })] }));
102
124
  }
103
125
  //# 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.6" })] }), _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.8" })] }), _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
@@ -0,0 +1,13 @@
1
+ /** @jsxImportSource react */
2
+ import React from 'react';
3
+ import { isCustomProvider } from '../../config/providers.js';
4
+ import type { ProviderName } from '../../config/types.js';
5
+ interface Props {
6
+ onComplete: (provider: ProviderName, model: string, apiKey: string, baseUrl?: string) => void;
7
+ onCancel: () => void;
8
+ }
9
+ type InitStep = 'provider' | 'api_key' | 'model' | 'custom_model' | 'custom_url';
10
+ export declare function InitWizard({ onComplete, onCancel }: Props): React.JSX.Element | null;
11
+ export type { InitStep };
12
+ export { isCustomProvider };
13
+ //# sourceMappingURL=InitWizard.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"InitWizard.d.ts","sourceRoot":"","sources":["../../../src/tui/components/InitWizard.tsx"],"names":[],"mappings":"AAAA,6BAA6B;AAC7B,OAAO,KAA8B,MAAM,OAAO,CAAC;AAMnD,OAAO,EAAuC,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AAClG,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAC;AAE1D,UAAU,KAAK;IACb,UAAU,EAAE,CAAC,QAAQ,EAAE,YAAY,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,MAAM,KAAK,IAAI,CAAC;IAC9F,QAAQ,EAAE,MAAM,IAAI,CAAC;CACtB;AAED,KAAK,QAAQ,GAAG,UAAU,GAAG,SAAS,GAAG,OAAO,GAAG,cAAc,GAAG,YAAY,CAAC;AAEjF,wBAAgB,UAAU,CAAC,EAAE,UAAU,EAAE,QAAQ,EAAE,EAAE,KAAK,4BA8GzD;AAGD,YAAY,EAAE,QAAQ,EAAE,CAAC;AACzB,OAAO,EAAE,gBAAgB,EAAE,CAAC"}
@@ -0,0 +1,86 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ /** @jsxImportSource react */
3
+ import { useState } from 'react';
4
+ import { Box, Text } from 'ink';
5
+ import SelectInput from 'ink-select-input';
6
+ import { readFileSync, writeFileSync, existsSync, mkdirSync } from 'fs';
7
+ import { join } from 'path';
8
+ import { homedir } from 'os';
9
+ import { listProviders, getModelsForProvider, isCustomProvider } from '../../config/providers.js';
10
+ export function InitWizard({ onComplete, onCancel }) {
11
+ const [step, setStep] = useState('provider');
12
+ const [selectedProvider, setSelectedProvider] = useState('siliconflow');
13
+ const [apiKey, setApiKey] = useState('');
14
+ const [selectedModel, setSelectedModel] = useState('');
15
+ const [customUrl, setCustomUrl] = useState('');
16
+ const providers = listProviders();
17
+ const models = getModelsForProvider(selectedProvider);
18
+ const saveConfig = (provider, model, key, url) => {
19
+ const dir = join(homedir(), '.thatgfsj');
20
+ const configPath = join(dir, 'config.json');
21
+ if (!existsSync(dir))
22
+ mkdirSync(dir, { recursive: true });
23
+ const config = { provider, model, apiKey: key, temperature: 0.7, maxTokens: 4096, contextLength: 50 };
24
+ if (url)
25
+ config.baseUrl = url;
26
+ writeFileSync(configPath, JSON.stringify(config, null, 2));
27
+ // Save to models history
28
+ const historyPath = join(dir, 'models.json');
29
+ let history = [];
30
+ if (existsSync(historyPath)) {
31
+ try {
32
+ history = JSON.parse(readFileSync(historyPath, 'utf-8'));
33
+ }
34
+ catch { }
35
+ }
36
+ if (!history.includes(model)) {
37
+ history.push(model);
38
+ writeFileSync(historyPath, JSON.stringify(history, null, 2));
39
+ }
40
+ };
41
+ // Step 1: Provider selection
42
+ if (step === 'provider') {
43
+ const items = providers.map(p => ({ label: p.name, value: p.key }));
44
+ return (_jsxs(Box, { flexDirection: "column", paddingLeft: 1, children: [_jsx(Text, { color: "#06B6D4", bold: true, children: "\u9009\u62E9\u670D\u52A1\u5546 (\u2191\u2193 \u56DE\u8F66):" }), _jsx(SelectInput, { items: items, onSelect: (item) => {
45
+ setSelectedProvider(item.value);
46
+ if (isCustomProvider(item.value)) {
47
+ setStep('custom_url');
48
+ }
49
+ else {
50
+ setStep('api_key');
51
+ }
52
+ } })] }));
53
+ }
54
+ // Custom URL input
55
+ if (step === 'custom_url') {
56
+ return (_jsxs(Box, { flexDirection: "column", paddingLeft: 1, children: [_jsx(Text, { color: "#06B6D4", bold: true, children: "\u8F93\u5165\u4E2D\u8F6C\u7AD9 URL:" }), _jsx(Text, { dimColor: true, children: "\u4F8B\u5982: https://api.example.com/v1" }), _jsx(Text, { color: "#F59E0B", children: "\uFF08\u8BF7\u76F4\u63A5\u8F93\u5165 URL \u5E76\u56DE\u8F66\uFF09" })] }));
57
+ }
58
+ // Step 2: API Key
59
+ if (step === 'api_key') {
60
+ return (_jsxs(Box, { flexDirection: "column", paddingLeft: 1, children: [_jsx(Text, { color: "#06B6D4", bold: true, children: "\u8F93\u5165 API Key:" }), _jsxs(Text, { dimColor: true, children: ["\u670D\u52A1\u5546: ", providers.find(p => p.key === selectedProvider)?.name] }), _jsx(Text, { color: "#F59E0B", children: "\uFF08\u8BF7\u76F4\u63A5\u8F93\u5165 Key \u5E76\u56DE\u8F66\uFF09" })] }));
61
+ }
62
+ // Step 3: Model selection
63
+ if (step === 'model') {
64
+ const items = [
65
+ ...models.map(m => ({ label: `${m.name} - ${m.desc}`, value: m.id })),
66
+ { label: '+ 输入自定义模型名', value: '__custom__' },
67
+ ];
68
+ return (_jsxs(Box, { flexDirection: "column", paddingLeft: 1, children: [_jsx(Text, { color: "#06B6D4", bold: true, children: "\u9009\u62E9\u6A21\u578B (\u2191\u2193 \u56DE\u8F66):" }), _jsx(SelectInput, { items: items, onSelect: (item) => {
69
+ if (item.value === '__custom__') {
70
+ setStep('custom_model');
71
+ }
72
+ else {
73
+ setSelectedModel(item.value);
74
+ saveConfig(selectedProvider, item.value, apiKey, customUrl || undefined);
75
+ onComplete(selectedProvider, item.value, apiKey, customUrl || undefined);
76
+ }
77
+ } })] }));
78
+ }
79
+ // Custom model name
80
+ if (step === 'custom_model') {
81
+ return (_jsxs(Box, { flexDirection: "column", paddingLeft: 1, children: [_jsx(Text, { color: "#06B6D4", bold: true, children: "\u8F93\u5165\u6A21\u578B\u540D\u79F0:" }), _jsx(Text, { color: "#F59E0B", children: "\uFF08\u8BF7\u76F4\u63A5\u8F93\u5165\u6A21\u578B\u540D\u5E76\u56DE\u8F66\uFF09" })] }));
82
+ }
83
+ return null;
84
+ }
85
+ export { isCustomProvider };
86
+ //# sourceMappingURL=InitWizard.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"InitWizard.js","sourceRoot":"","sources":["../../../src/tui/components/InitWizard.tsx"],"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.6",
3
+ "version": "0.9.8",
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.6')
31
+ .version('0.9.8')
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
@@ -7,10 +7,12 @@ import { Thinking } from './components/Thinking.js';
7
7
  import { UserInput } from './components/UserInput.js';
8
8
  import { StatusBar } from './components/StatusBar.js';
9
9
  import { ModelSelector } from './components/ModelSelector.js';
10
+ import { InitWizard } from './components/InitWizard.js';
10
11
  import { useChat } from './hooks/useChat.js';
11
- import { useCommands, COMMAND_LIST } from './hooks/useCommands.js';
12
+ import { useCommands } from './hooks/useCommands.js';
12
13
  import type { App } from '../app/index.js';
13
14
  import type { MessageData } from './components/ChatMessage.js';
15
+ import type { ProviderName } from '../config/types.js';
14
16
  import { writeFileSync, readFileSync, existsSync, mkdirSync } from 'fs';
15
17
  import { join } from 'path';
16
18
  import { homedir } from 'os';
@@ -23,7 +25,6 @@ function saveModelToHistory(model: string) {
23
25
  const dir = join(homedir(), '.thatgfsj');
24
26
  const path = join(dir, 'models.json');
25
27
  if (!existsSync(dir)) mkdirSync(dir, { recursive: true });
26
-
27
28
  let history: string[] = [];
28
29
  if (existsSync(path)) {
29
30
  try { history = JSON.parse(readFileSync(path, 'utf-8')); } catch {}
@@ -34,38 +35,67 @@ function saveModelToHistory(model: string) {
34
35
  }
35
36
  }
36
37
 
38
+ type ViewMode = 'chat' | 'model_select' | 'init_wizard' | 'init_key' | 'init_url' | 'init_custom_model';
39
+
37
40
  export function TuiApp({ app }: Props) {
38
41
  const { messages, isThinking, streaming, streamingToolCalls, queuedMessage, sendMessage } = useChat(app);
39
42
  const { handleCommand } = useCommands(app);
40
43
  const [systemMessages, setSystemMessages] = useState<MessageData[]>([]);
41
- const [showModelSelector, setShowModelSelector] = useState(false);
42
- const [addModelMode, setAddModelMode] = useState(false);
44
+ const [viewMode, setViewMode] = useState<ViewMode>('chat');
45
+ const [initProvider, setInitProvider] = useState<ProviderName>('siliconflow');
46
+ const [initUrl, setInitUrl] = useState('');
43
47
  const { stdout } = useStdout();
44
48
  const terminalWidth = stdout?.columns || 80;
45
49
 
50
+ const addMsg = useCallback((content: string) => {
51
+ setSystemMessages(prev => [...prev, { role: 'assistant', content }]);
52
+ }, []);
53
+
46
54
  const onSubmit = useCallback(async (input: string) => {
47
- // 添加新模型模式
48
- if (addModelMode) {
55
+ // Init wizard - API Key input
56
+ if (viewMode === 'init_key') {
57
+ const key = input.trim();
58
+ if (!key) { setViewMode('chat'); return; }
59
+ // Save and complete
60
+ const { App: AppClass } = await import('../app/index.js');
61
+ const cfg = app.config.get();
62
+ app.config.save({ provider: initProvider, apiKey: key, model: cfg.model || 'default' });
63
+ saveModelToHistory(cfg.model || 'default');
64
+ setViewMode('chat');
65
+ addMsg(`配置已更新: ${initProvider}`);
66
+ return;
67
+ }
68
+
69
+ // Init wizard - Custom URL input
70
+ if (viewMode === 'init_url') {
71
+ const url = input.trim();
72
+ if (!url) { setViewMode('chat'); return; }
73
+ setInitUrl(url);
74
+ setViewMode('init_key');
75
+ return;
76
+ }
77
+
78
+ // Init wizard - Custom model name
79
+ if (viewMode === 'init_custom_model') {
49
80
  const model = input.trim();
50
81
  if (model) {
51
82
  app.config.save({ model });
52
83
  saveModelToHistory(model);
53
- setSystemMessages(prev => [
54
- ...prev,
55
- { role: 'assistant', content: `模型已切换: ${model}` },
56
- ]);
84
+ addMsg(`模型已切换: ${model}`);
57
85
  }
58
- setAddModelMode(false);
86
+ setViewMode('chat');
59
87
  return;
60
88
  }
61
89
 
62
- if (showModelSelector) return;
90
+ // Model selector - ignore text input
91
+ if (viewMode === 'model_select') return;
63
92
 
93
+ // Normal command handling
64
94
  const result = handleCommand(input);
65
95
 
66
96
  if (result.handled) {
67
- if (input.trim().toLowerCase() === '/model' || input.trim() === '/模型') {
68
- setShowModelSelector(true);
97
+ if (input.trim() === '/模型' || input.trim() === '/model') {
98
+ setViewMode('model_select');
69
99
  return;
70
100
  }
71
101
 
@@ -80,22 +110,14 @@ export function TuiApp({ app }: Props) {
80
110
  if (result.action === 'clear') setSystemMessages([]);
81
111
 
82
112
  if (result.action === 'reinit') {
83
- const { WelcomeScreen } = await import('./welcome.js');
84
- await WelcomeScreen.interactiveSetup();
85
- const { App: AppClass } = await import('../app/index.js');
86
- const newApp = await AppClass.create();
87
- Object.assign(app, newApp);
88
- setSystemMessages(prev => [
89
- ...prev,
90
- { role: 'assistant', content: '配置已更新,模型: ' + app.config.get().model },
91
- ]);
113
+ setViewMode('init_wizard');
92
114
  }
93
115
 
94
116
  return;
95
117
  }
96
118
 
97
119
  sendMessage(input);
98
- }, [handleCommand, sendMessage, app, showModelSelector, addModelMode]);
120
+ }, [handleCommand, sendMessage, app, viewMode, initProvider, initUrl, addMsg]);
99
121
 
100
122
  const allMessages = [...systemMessages, ...messages];
101
123
  const activeSkills = app.skills.listActive().map(s => s.id);
@@ -116,29 +138,47 @@ export function TuiApp({ app }: Props) {
116
138
  <Text color="#94A3B8">{queuedMessage}</Text>
117
139
  </Box>
118
140
  )}
119
- {showModelSelector ? (
141
+ {viewMode === 'model_select' ? (
120
142
  <ModelSelector
121
143
  currentModel={app.config.get().model}
122
144
  onSelect={(model) => {
123
145
  app.config.save({ model });
124
146
  saveModelToHistory(model);
125
- setShowModelSelector(false);
126
- setSystemMessages(prev => [
127
- ...prev,
128
- { role: 'assistant', content: `模型已切换: ${model}` },
129
- ]);
147
+ setViewMode('chat');
148
+ addMsg(`模型已切换: ${model}`);
130
149
  }}
131
- onAddNew={() => {
132
- setShowModelSelector(false);
133
- setAddModelMode(true);
150
+ onAddNew={() => setViewMode('init_wizard')}
151
+ />
152
+ ) : viewMode === 'init_wizard' ? (
153
+ <InitWizard
154
+ onComplete={(provider, model, apiKey, baseUrl) => {
155
+ app.config.save({ provider, model, apiKey, baseUrl });
156
+ saveModelToHistory(model);
157
+ setViewMode('chat');
158
+ addMsg(`配置完成: ${provider} / ${model}`);
134
159
  }}
160
+ onCancel={() => setViewMode('chat')}
135
161
  />
136
- ) : addModelMode ? (
137
- <Box paddingLeft={1}>
138
- <Text color="#06B6D4" bold>输入新模型名称: </Text>
139
- </Box>
140
162
  ) : (
141
- <UserInput onSubmit={onSubmit} disabled={false} />
163
+ <Box flexDirection="column">
164
+ {viewMode === 'init_key' && (
165
+ <Box paddingLeft={1} marginBottom={0}>
166
+ <Text color="#06B6D4" bold>输入 API Key: </Text>
167
+ <Text dimColor>({initProvider})</Text>
168
+ </Box>
169
+ )}
170
+ {viewMode === 'init_url' && (
171
+ <Box paddingLeft={1} marginBottom={0}>
172
+ <Text color="#06B6D4" bold>输入中转站 URL: </Text>
173
+ </Box>
174
+ )}
175
+ {viewMode === 'init_custom_model' && (
176
+ <Box paddingLeft={1} marginBottom={0}>
177
+ <Text color="#06B6D4" bold>输入模型名称: </Text>
178
+ </Box>
179
+ )}
180
+ <UserInput onSubmit={onSubmit} disabled={false} />
181
+ </Box>
142
182
  )}
143
183
  <StatusBar messageCount={allMessages.length} skills={activeSkills} />
144
184
  </Box>
@@ -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.6</Text>
17
+ <Text dimColor> v0.9.8</Text>
18
18
  </Box>
19
19
  <Box>
20
20
  <Text color="#06B6D4" bold> {provider} </Text>
@@ -0,0 +1,132 @@
1
+ /** @jsxImportSource react */
2
+ import React, { useState, useEffect } from 'react';
3
+ import { Box, Text } from 'ink';
4
+ import SelectInput from 'ink-select-input';
5
+ import { readFileSync, writeFileSync, existsSync, mkdirSync } from 'fs';
6
+ import { join } from 'path';
7
+ import { homedir } from 'os';
8
+ import { listProviders, getModelsForProvider, isCustomProvider } from '../../config/providers.js';
9
+ import type { ProviderName } from '../../config/types.js';
10
+
11
+ interface Props {
12
+ onComplete: (provider: ProviderName, model: string, apiKey: string, baseUrl?: string) => void;
13
+ onCancel: () => void;
14
+ }
15
+
16
+ type InitStep = 'provider' | 'api_key' | 'model' | 'custom_model' | 'custom_url';
17
+
18
+ export function InitWizard({ onComplete, onCancel }: Props) {
19
+ const [step, setStep] = useState<InitStep>('provider');
20
+ const [selectedProvider, setSelectedProvider] = useState<ProviderName>('siliconflow');
21
+ const [apiKey, setApiKey] = useState('');
22
+ const [selectedModel, setSelectedModel] = useState('');
23
+ const [customUrl, setCustomUrl] = useState('');
24
+
25
+ const providers = listProviders();
26
+ const models = getModelsForProvider(selectedProvider);
27
+
28
+ const saveConfig = (provider: ProviderName, model: string, key: string, url?: string) => {
29
+ const dir = join(homedir(), '.thatgfsj');
30
+ const configPath = join(dir, 'config.json');
31
+ if (!existsSync(dir)) mkdirSync(dir, { recursive: true });
32
+
33
+ const config: Record<string, any> = { provider, model, apiKey: key, temperature: 0.7, maxTokens: 4096, contextLength: 50 };
34
+ if (url) config.baseUrl = url;
35
+ writeFileSync(configPath, JSON.stringify(config, null, 2));
36
+
37
+ // Save to models history
38
+ const historyPath = join(dir, 'models.json');
39
+ let history: string[] = [];
40
+ if (existsSync(historyPath)) {
41
+ try { history = JSON.parse(readFileSync(historyPath, 'utf-8')); } catch {}
42
+ }
43
+ if (!history.includes(model)) {
44
+ history.push(model);
45
+ writeFileSync(historyPath, JSON.stringify(history, null, 2));
46
+ }
47
+ };
48
+
49
+ // Step 1: Provider selection
50
+ if (step === 'provider') {
51
+ const items = providers.map(p => ({ label: p.name, value: p.key }));
52
+ return (
53
+ <Box flexDirection="column" paddingLeft={1}>
54
+ <Text color="#06B6D4" bold>选择服务商 (↑↓ 回车):</Text>
55
+ <SelectInput
56
+ items={items}
57
+ onSelect={(item) => {
58
+ setSelectedProvider(item.value as ProviderName);
59
+ if (isCustomProvider(item.value as ProviderName)) {
60
+ setStep('custom_url');
61
+ } else {
62
+ setStep('api_key');
63
+ }
64
+ }}
65
+ />
66
+ </Box>
67
+ );
68
+ }
69
+
70
+ // Custom URL input
71
+ if (step === 'custom_url') {
72
+ return (
73
+ <Box flexDirection="column" paddingLeft={1}>
74
+ <Text color="#06B6D4" bold>输入中转站 URL:</Text>
75
+ <Text dimColor>例如: https://api.example.com/v1</Text>
76
+ <Text color="#F59E0B">(请直接输入 URL 并回车)</Text>
77
+ </Box>
78
+ );
79
+ }
80
+
81
+ // Step 2: API Key
82
+ if (step === 'api_key') {
83
+ return (
84
+ <Box flexDirection="column" paddingLeft={1}>
85
+ <Text color="#06B6D4" bold>输入 API Key:</Text>
86
+ <Text dimColor>服务商: {providers.find(p => p.key === selectedProvider)?.name}</Text>
87
+ <Text color="#F59E0B">(请直接输入 Key 并回车)</Text>
88
+ </Box>
89
+ );
90
+ }
91
+
92
+ // Step 3: Model selection
93
+ if (step === 'model') {
94
+ const items = [
95
+ ...models.map(m => ({ label: `${m.name} - ${m.desc}`, value: m.id })),
96
+ { label: '+ 输入自定义模型名', value: '__custom__' },
97
+ ];
98
+ return (
99
+ <Box flexDirection="column" paddingLeft={1}>
100
+ <Text color="#06B6D4" bold>选择模型 (↑↓ 回车):</Text>
101
+ <SelectInput
102
+ items={items}
103
+ onSelect={(item) => {
104
+ if (item.value === '__custom__') {
105
+ setStep('custom_model');
106
+ } else {
107
+ setSelectedModel(item.value);
108
+ saveConfig(selectedProvider, item.value, apiKey, customUrl || undefined);
109
+ onComplete(selectedProvider, item.value, apiKey, customUrl || undefined);
110
+ }
111
+ }}
112
+ />
113
+ </Box>
114
+ );
115
+ }
116
+
117
+ // Custom model name
118
+ if (step === 'custom_model') {
119
+ return (
120
+ <Box flexDirection="column" paddingLeft={1}>
121
+ <Text color="#06B6D4" bold>输入模型名称:</Text>
122
+ <Text color="#F59E0B">(请直接输入模型名并回车)</Text>
123
+ </Box>
124
+ );
125
+ }
126
+
127
+ return null;
128
+ }
129
+
130
+ // Export steps for external handling
131
+ export type { InitStep };
132
+ export { isCustomProvider };