thatgfsj-code 0.9.6 → 0.9.7
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 +18 -26
- package/dist/tui/app.js.map +1 -1
- package/dist/tui/components/Header.js +1 -1
- package/package.json +1 -1
- package/src/cmd/index.tsx +1 -1
- package/src/tui/app.tsx +29 -32
- package/src/tui/components/Header.tsx +1 -1
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.7')
|
|
28
28
|
.argument('[prompt]', 'Task to execute (omit to start interactive mode)')
|
|
29
29
|
.option('-m, --model <model>', 'Specify model')
|
|
30
30
|
.option('-i, --interactive', 'Force interactive mode')
|
package/dist/tui/app.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"app.d.ts","sourceRoot":"","sources":["../../src/tui/app.tsx"],"names":[],"mappings":"AAAA,6BAA6B;AAC7B,OAAO,KAAgC,MAAM,OAAO,CAAC;AAUrD,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,iBAAiB,CAAC;AAM3C,UAAU,KAAK;IACb,GAAG,EAAE,GAAG,CAAC;CACV;
|
|
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;AAkBD,wBAAgB,MAAM,CAAC,EAAE,GAAG,EAAE,EAAE,KAAK,qBAyGpC"}
|
package/dist/tui/app.js
CHANGED
|
@@ -34,31 +34,32 @@ export function TuiApp({ app }) {
|
|
|
34
34
|
const { messages, isThinking, streaming, streamingToolCalls, queuedMessage, sendMessage } = useChat(app);
|
|
35
35
|
const { handleCommand } = useCommands(app);
|
|
36
36
|
const [systemMessages, setSystemMessages] = useState([]);
|
|
37
|
-
const [
|
|
38
|
-
const [addModelMode, setAddModelMode] = useState(false);
|
|
37
|
+
const [viewMode, setViewMode] = useState('chat');
|
|
39
38
|
const { stdout } = useStdout();
|
|
40
39
|
const terminalWidth = stdout?.columns || 80;
|
|
40
|
+
const addAssistantMsg = useCallback((content) => {
|
|
41
|
+
setSystemMessages(prev => [...prev, { role: 'assistant', content }]);
|
|
42
|
+
}, []);
|
|
41
43
|
const onSubmit = useCallback(async (input) => {
|
|
42
44
|
// 添加新模型模式
|
|
43
|
-
if (
|
|
45
|
+
if (viewMode === 'model_add') {
|
|
44
46
|
const model = input.trim();
|
|
45
47
|
if (model) {
|
|
46
48
|
app.config.save({ model });
|
|
47
49
|
saveModelToHistory(model);
|
|
48
|
-
|
|
49
|
-
...prev,
|
|
50
|
-
{ role: 'assistant', content: `模型已切换: ${model}` },
|
|
51
|
-
]);
|
|
50
|
+
addAssistantMsg(`模型已切换: ${model}`);
|
|
52
51
|
}
|
|
53
|
-
|
|
52
|
+
setViewMode('chat');
|
|
54
53
|
return;
|
|
55
54
|
}
|
|
56
|
-
|
|
55
|
+
// 模型选择器模式 - 不处理输入
|
|
56
|
+
if (viewMode === 'model_select')
|
|
57
57
|
return;
|
|
58
|
+
// 命令处理
|
|
58
59
|
const result = handleCommand(input);
|
|
59
60
|
if (result.handled) {
|
|
60
|
-
if (input.trim()
|
|
61
|
-
|
|
61
|
+
if (input.trim() === '/模型' || input.trim() === '/model') {
|
|
62
|
+
setViewMode('model_select');
|
|
62
63
|
return;
|
|
63
64
|
}
|
|
64
65
|
if (result.output) {
|
|
@@ -76,28 +77,19 @@ export function TuiApp({ app }) {
|
|
|
76
77
|
const { App: AppClass } = await import('../app/index.js');
|
|
77
78
|
const newApp = await AppClass.create();
|
|
78
79
|
Object.assign(app, newApp);
|
|
79
|
-
|
|
80
|
-
...prev,
|
|
81
|
-
{ role: 'assistant', content: '配置已更新,模型: ' + app.config.get().model },
|
|
82
|
-
]);
|
|
80
|
+
addAssistantMsg('配置已更新,模型: ' + app.config.get().model);
|
|
83
81
|
}
|
|
84
82
|
return;
|
|
85
83
|
}
|
|
86
84
|
sendMessage(input);
|
|
87
|
-
}, [handleCommand, sendMessage, app,
|
|
85
|
+
}, [handleCommand, sendMessage, app, viewMode, addAssistantMsg]);
|
|
88
86
|
const allMessages = [...systemMessages, ...messages];
|
|
89
87
|
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 })] })),
|
|
88
|
+
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
89
|
app.config.save({ model });
|
|
92
90
|
saveModelToHistory(model);
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
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 })] }));
|
|
91
|
+
setViewMode('chat');
|
|
92
|
+
addAssistantMsg(`模型已切换: ${model}`);
|
|
93
|
+
}, onAddNew: () => setViewMode('model_add') })) : (_jsxs(Box, { flexDirection: "column", children: [viewMode === 'model_add' && (_jsx(Box, { paddingLeft: 1, marginBottom: 0, 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 })] }));
|
|
102
94
|
}
|
|
103
95
|
//# 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,EAAgB,MAAM,wBAAwB,CAAC;AAGnE,OAAO,EAAE,aAAa,EAAE,YAAY,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AACxE,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EAAE,OAAO,EAAE,MAAM,IAAI,CAAC;AAM7B,SAAS,kBAAkB,CAAC,KAAa;IACvC,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,WAAW,CAAC,CAAC;IACzC,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,EAAE,aAAa,CAAC,CAAC;IACtC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC;QAAE,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;
|
|
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.7" })] }), _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
|
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.7')
|
|
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
|
@@ -23,7 +23,6 @@ function saveModelToHistory(model: string) {
|
|
|
23
23
|
const dir = join(homedir(), '.thatgfsj');
|
|
24
24
|
const path = join(dir, 'models.json');
|
|
25
25
|
if (!existsSync(dir)) mkdirSync(dir, { recursive: true });
|
|
26
|
-
|
|
27
26
|
let history: string[] = [];
|
|
28
27
|
if (existsSync(path)) {
|
|
29
28
|
try { history = JSON.parse(readFileSync(path, 'utf-8')); } catch {}
|
|
@@ -34,38 +33,42 @@ function saveModelToHistory(model: string) {
|
|
|
34
33
|
}
|
|
35
34
|
}
|
|
36
35
|
|
|
36
|
+
type ViewMode = 'chat' | 'model_select' | 'model_add';
|
|
37
|
+
|
|
37
38
|
export function TuiApp({ app }: Props) {
|
|
38
39
|
const { messages, isThinking, streaming, streamingToolCalls, queuedMessage, sendMessage } = useChat(app);
|
|
39
40
|
const { handleCommand } = useCommands(app);
|
|
40
41
|
const [systemMessages, setSystemMessages] = useState<MessageData[]>([]);
|
|
41
|
-
const [
|
|
42
|
-
const [addModelMode, setAddModelMode] = useState(false);
|
|
42
|
+
const [viewMode, setViewMode] = useState<ViewMode>('chat');
|
|
43
43
|
const { stdout } = useStdout();
|
|
44
44
|
const terminalWidth = stdout?.columns || 80;
|
|
45
45
|
|
|
46
|
+
const addAssistantMsg = useCallback((content: string) => {
|
|
47
|
+
setSystemMessages(prev => [...prev, { role: 'assistant', content }]);
|
|
48
|
+
}, []);
|
|
49
|
+
|
|
46
50
|
const onSubmit = useCallback(async (input: string) => {
|
|
47
51
|
// 添加新模型模式
|
|
48
|
-
if (
|
|
52
|
+
if (viewMode === 'model_add') {
|
|
49
53
|
const model = input.trim();
|
|
50
54
|
if (model) {
|
|
51
55
|
app.config.save({ model });
|
|
52
56
|
saveModelToHistory(model);
|
|
53
|
-
|
|
54
|
-
...prev,
|
|
55
|
-
{ role: 'assistant', content: `模型已切换: ${model}` },
|
|
56
|
-
]);
|
|
57
|
+
addAssistantMsg(`模型已切换: ${model}`);
|
|
57
58
|
}
|
|
58
|
-
|
|
59
|
+
setViewMode('chat');
|
|
59
60
|
return;
|
|
60
61
|
}
|
|
61
62
|
|
|
62
|
-
|
|
63
|
+
// 模型选择器模式 - 不处理输入
|
|
64
|
+
if (viewMode === 'model_select') return;
|
|
63
65
|
|
|
66
|
+
// 命令处理
|
|
64
67
|
const result = handleCommand(input);
|
|
65
68
|
|
|
66
69
|
if (result.handled) {
|
|
67
|
-
if (input.trim()
|
|
68
|
-
|
|
70
|
+
if (input.trim() === '/模型' || input.trim() === '/model') {
|
|
71
|
+
setViewMode('model_select');
|
|
69
72
|
return;
|
|
70
73
|
}
|
|
71
74
|
|
|
@@ -85,17 +88,14 @@ export function TuiApp({ app }: Props) {
|
|
|
85
88
|
const { App: AppClass } = await import('../app/index.js');
|
|
86
89
|
const newApp = await AppClass.create();
|
|
87
90
|
Object.assign(app, newApp);
|
|
88
|
-
|
|
89
|
-
...prev,
|
|
90
|
-
{ role: 'assistant', content: '配置已更新,模型: ' + app.config.get().model },
|
|
91
|
-
]);
|
|
91
|
+
addAssistantMsg('配置已更新,模型: ' + app.config.get().model);
|
|
92
92
|
}
|
|
93
93
|
|
|
94
94
|
return;
|
|
95
95
|
}
|
|
96
96
|
|
|
97
97
|
sendMessage(input);
|
|
98
|
-
}, [handleCommand, sendMessage, app,
|
|
98
|
+
}, [handleCommand, sendMessage, app, viewMode, addAssistantMsg]);
|
|
99
99
|
|
|
100
100
|
const allMessages = [...systemMessages, ...messages];
|
|
101
101
|
const activeSkills = app.skills.listActive().map(s => s.id);
|
|
@@ -116,29 +116,26 @@ export function TuiApp({ app }: Props) {
|
|
|
116
116
|
<Text color="#94A3B8">{queuedMessage}</Text>
|
|
117
117
|
</Box>
|
|
118
118
|
)}
|
|
119
|
-
{
|
|
119
|
+
{viewMode === 'model_select' ? (
|
|
120
120
|
<ModelSelector
|
|
121
121
|
currentModel={app.config.get().model}
|
|
122
122
|
onSelect={(model) => {
|
|
123
123
|
app.config.save({ model });
|
|
124
124
|
saveModelToHistory(model);
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
...prev,
|
|
128
|
-
{ role: 'assistant', content: `模型已切换: ${model}` },
|
|
129
|
-
]);
|
|
130
|
-
}}
|
|
131
|
-
onAddNew={() => {
|
|
132
|
-
setShowModelSelector(false);
|
|
133
|
-
setAddModelMode(true);
|
|
125
|
+
setViewMode('chat');
|
|
126
|
+
addAssistantMsg(`模型已切换: ${model}`);
|
|
134
127
|
}}
|
|
128
|
+
onAddNew={() => setViewMode('model_add')}
|
|
135
129
|
/>
|
|
136
|
-
) : addModelMode ? (
|
|
137
|
-
<Box paddingLeft={1}>
|
|
138
|
-
<Text color="#06B6D4" bold>输入新模型名称: </Text>
|
|
139
|
-
</Box>
|
|
140
130
|
) : (
|
|
141
|
-
<
|
|
131
|
+
<Box flexDirection="column">
|
|
132
|
+
{viewMode === 'model_add' && (
|
|
133
|
+
<Box paddingLeft={1} marginBottom={0}>
|
|
134
|
+
<Text color="#06B6D4" bold>输入新模型名称: </Text>
|
|
135
|
+
</Box>
|
|
136
|
+
)}
|
|
137
|
+
<UserInput onSubmit={onSubmit} disabled={false} />
|
|
138
|
+
</Box>
|
|
142
139
|
)}
|
|
143
140
|
<StatusBar messageCount={allMessages.length} skills={activeSkills} />
|
|
144
141
|
</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.
|
|
17
|
+
<Text dimColor> v0.9.7</Text>
|
|
18
18
|
</Box>
|
|
19
19
|
<Box>
|
|
20
20
|
<Text color="#06B6D4" bold> {provider} </Text>
|