thatgfsj-code 0.9.7 → 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 +1 -1
- package/dist/tui/app.d.ts.map +1 -1
- package/dist/tui/app.js +45 -15
- package/dist/tui/app.js.map +1 -1
- package/dist/tui/components/Header.js +1 -1
- package/dist/tui/components/InitWizard.d.ts +13 -0
- package/dist/tui/components/InitWizard.d.ts.map +1 -0
- package/dist/tui/components/InitWizard.js +86 -0
- package/dist/tui/components/InitWizard.js.map +1 -0
- package/package.json +1 -1
- package/src/cmd/index.tsx +1 -1
- package/src/tui/app.tsx +62 -19
- package/src/tui/components/Header.tsx +1 -1
- package/src/tui/components/InitWizard.tsx +132 -0
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.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')
|
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;
|
|
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';
|
|
@@ -35,27 +36,56 @@ export function TuiApp({ app }) {
|
|
|
35
36
|
const { handleCommand } = useCommands(app);
|
|
36
37
|
const [systemMessages, setSystemMessages] = useState([]);
|
|
37
38
|
const [viewMode, setViewMode] = useState('chat');
|
|
39
|
+
const [initProvider, setInitProvider] = useState('siliconflow');
|
|
40
|
+
const [initUrl, setInitUrl] = useState('');
|
|
38
41
|
const { stdout } = useStdout();
|
|
39
42
|
const terminalWidth = stdout?.columns || 80;
|
|
40
|
-
const
|
|
43
|
+
const addMsg = useCallback((content) => {
|
|
41
44
|
setSystemMessages(prev => [...prev, { role: 'assistant', content }]);
|
|
42
45
|
}, []);
|
|
43
46
|
const onSubmit = useCallback(async (input) => {
|
|
44
|
-
//
|
|
45
|
-
if (viewMode === '
|
|
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') {
|
|
46
76
|
const model = input.trim();
|
|
47
77
|
if (model) {
|
|
48
78
|
app.config.save({ model });
|
|
49
79
|
saveModelToHistory(model);
|
|
50
|
-
|
|
80
|
+
addMsg(`模型已切换: ${model}`);
|
|
51
81
|
}
|
|
52
82
|
setViewMode('chat');
|
|
53
83
|
return;
|
|
54
84
|
}
|
|
55
|
-
//
|
|
85
|
+
// Model selector - ignore text input
|
|
56
86
|
if (viewMode === 'model_select')
|
|
57
87
|
return;
|
|
58
|
-
//
|
|
88
|
+
// Normal command handling
|
|
59
89
|
const result = handleCommand(input);
|
|
60
90
|
if (result.handled) {
|
|
61
91
|
if (input.trim() === '/模型' || input.trim() === '/model') {
|
|
@@ -72,24 +102,24 @@ export function TuiApp({ app }) {
|
|
|
72
102
|
if (result.action === 'clear')
|
|
73
103
|
setSystemMessages([]);
|
|
74
104
|
if (result.action === 'reinit') {
|
|
75
|
-
|
|
76
|
-
await WelcomeScreen.interactiveSetup();
|
|
77
|
-
const { App: AppClass } = await import('../app/index.js');
|
|
78
|
-
const newApp = await AppClass.create();
|
|
79
|
-
Object.assign(app, newApp);
|
|
80
|
-
addAssistantMsg('配置已更新,模型: ' + app.config.get().model);
|
|
105
|
+
setViewMode('init_wizard');
|
|
81
106
|
}
|
|
82
107
|
return;
|
|
83
108
|
}
|
|
84
109
|
sendMessage(input);
|
|
85
|
-
}, [handleCommand, sendMessage, app, viewMode,
|
|
110
|
+
}, [handleCommand, sendMessage, app, viewMode, initProvider, initUrl, addMsg]);
|
|
86
111
|
const allMessages = [...systemMessages, ...messages];
|
|
87
112
|
const activeSkills = app.skills.listActive().map(s => s.id);
|
|
88
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) => {
|
|
89
114
|
app.config.save({ model });
|
|
90
115
|
saveModelToHistory(model);
|
|
91
116
|
setViewMode('chat');
|
|
92
|
-
|
|
93
|
-
}, onAddNew: () => setViewMode('
|
|
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 })] }));
|
|
94
124
|
}
|
|
95
125
|
//# 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":"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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
|
+
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
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.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
|
|
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';
|
|
@@ -33,37 +35,62 @@ function saveModelToHistory(model: string) {
|
|
|
33
35
|
}
|
|
34
36
|
}
|
|
35
37
|
|
|
36
|
-
type ViewMode = 'chat' | 'model_select' | '
|
|
38
|
+
type ViewMode = 'chat' | 'model_select' | 'init_wizard' | 'init_key' | 'init_url' | 'init_custom_model';
|
|
37
39
|
|
|
38
40
|
export function TuiApp({ app }: Props) {
|
|
39
41
|
const { messages, isThinking, streaming, streamingToolCalls, queuedMessage, sendMessage } = useChat(app);
|
|
40
42
|
const { handleCommand } = useCommands(app);
|
|
41
43
|
const [systemMessages, setSystemMessages] = useState<MessageData[]>([]);
|
|
42
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
|
|
|
46
|
-
const
|
|
50
|
+
const addMsg = useCallback((content: string) => {
|
|
47
51
|
setSystemMessages(prev => [...prev, { role: 'assistant', content }]);
|
|
48
52
|
}, []);
|
|
49
53
|
|
|
50
54
|
const onSubmit = useCallback(async (input: string) => {
|
|
51
|
-
//
|
|
52
|
-
if (viewMode === '
|
|
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') {
|
|
53
80
|
const model = input.trim();
|
|
54
81
|
if (model) {
|
|
55
82
|
app.config.save({ model });
|
|
56
83
|
saveModelToHistory(model);
|
|
57
|
-
|
|
84
|
+
addMsg(`模型已切换: ${model}`);
|
|
58
85
|
}
|
|
59
86
|
setViewMode('chat');
|
|
60
87
|
return;
|
|
61
88
|
}
|
|
62
89
|
|
|
63
|
-
//
|
|
90
|
+
// Model selector - ignore text input
|
|
64
91
|
if (viewMode === 'model_select') return;
|
|
65
92
|
|
|
66
|
-
//
|
|
93
|
+
// Normal command handling
|
|
67
94
|
const result = handleCommand(input);
|
|
68
95
|
|
|
69
96
|
if (result.handled) {
|
|
@@ -83,19 +110,14 @@ export function TuiApp({ app }: Props) {
|
|
|
83
110
|
if (result.action === 'clear') setSystemMessages([]);
|
|
84
111
|
|
|
85
112
|
if (result.action === 'reinit') {
|
|
86
|
-
|
|
87
|
-
await WelcomeScreen.interactiveSetup();
|
|
88
|
-
const { App: AppClass } = await import('../app/index.js');
|
|
89
|
-
const newApp = await AppClass.create();
|
|
90
|
-
Object.assign(app, newApp);
|
|
91
|
-
addAssistantMsg('配置已更新,模型: ' + app.config.get().model);
|
|
113
|
+
setViewMode('init_wizard');
|
|
92
114
|
}
|
|
93
115
|
|
|
94
116
|
return;
|
|
95
117
|
}
|
|
96
118
|
|
|
97
119
|
sendMessage(input);
|
|
98
|
-
}, [handleCommand, sendMessage, app, viewMode,
|
|
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);
|
|
@@ -123,15 +145,36 @@ export function TuiApp({ app }: Props) {
|
|
|
123
145
|
app.config.save({ model });
|
|
124
146
|
saveModelToHistory(model);
|
|
125
147
|
setViewMode('chat');
|
|
126
|
-
|
|
148
|
+
addMsg(`模型已切换: ${model}`);
|
|
127
149
|
}}
|
|
128
|
-
onAddNew={() => setViewMode('
|
|
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}`);
|
|
159
|
+
}}
|
|
160
|
+
onCancel={() => setViewMode('chat')}
|
|
129
161
|
/>
|
|
130
162
|
) : (
|
|
131
163
|
<Box flexDirection="column">
|
|
132
|
-
{viewMode === '
|
|
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' && (
|
|
133
176
|
<Box paddingLeft={1} marginBottom={0}>
|
|
134
|
-
<Text color="#06B6D4" bold
|
|
177
|
+
<Text color="#06B6D4" bold>输入模型名称: </Text>
|
|
135
178
|
</Box>
|
|
136
179
|
)}
|
|
137
180
|
<UserInput onSubmit={onSubmit} disabled={false} />
|
|
@@ -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.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 };
|