activo 0.2.0 → 0.2.2

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/README.md CHANGED
@@ -7,9 +7,11 @@ AI 기반 코드 품질 분석 CLI 도구
7
7
  ## 설치
8
8
 
9
9
  ```bash
10
- pnpm install
11
- pnpm build
12
- npm link
10
+ # 글로벌 설치 (권장)
11
+ npm install -g activo
12
+
13
+ # 또는 npx로 바로 실행
14
+ npx activo
13
15
  ```
14
16
 
15
17
  ## 사용법
@@ -23,6 +25,9 @@ activo "src 폴더 구조 보여줘"
23
25
 
24
26
  # 비대화형 모드
25
27
  activo --print "package.json 분석해줘"
28
+
29
+ # 특정 모델 사용
30
+ activo --model qwen2.5:7b
26
31
  ```
27
32
 
28
33
  ## 주요 기능
@@ -37,7 +42,41 @@ activo --print "package.json 분석해줘"
37
42
  - Node.js 18+
38
43
  - [Ollama](https://ollama.ai) 실행 중
39
44
 
40
- ## 설정
45
+ ## Ollama 설정
46
+
47
+ ### 1. Ollama 설치
48
+
49
+ ```bash
50
+ # macOS
51
+ brew install ollama
52
+
53
+ # Windows
54
+ # https://ollama.ai 에서 다운로드
55
+
56
+ # Linux
57
+ curl -fsSL https://ollama.ai/install.sh | sh
58
+ ```
59
+
60
+ ### 2. 모델 다운로드
61
+
62
+ ```bash
63
+ # 권장 모델
64
+ ollama pull mistral:latest
65
+
66
+ # 한국어 지원 모델
67
+ ollama pull qwen2.5:7b
68
+
69
+ # 코드 특화 모델
70
+ ollama pull codellama:7b
71
+ ```
72
+
73
+ ### 3. Ollama 실행
74
+
75
+ ```bash
76
+ ollama serve
77
+ ```
78
+
79
+ ### 4. activo 설정
41
80
 
42
81
  `~/.activo/config.json`:
43
82
  ```json
@@ -49,6 +88,11 @@ activo --print "package.json 분석해줘"
49
88
  }
50
89
  ```
51
90
 
91
+ 또는 CLI 옵션으로 모델 지정:
92
+ ```bash
93
+ activo --model qwen2.5:7b
94
+ ```
95
+
52
96
  ## 단축키
53
97
 
54
98
  | 키 | 동작 |
@@ -0,0 +1,11 @@
1
+ import { Config } from "./config.js";
2
+ export interface SlashCommandResult {
3
+ output?: string;
4
+ exit?: boolean;
5
+ clear?: boolean;
6
+ changeModel?: string;
7
+ showHelp?: boolean;
8
+ }
9
+ export declare function handleSlashCommand(input: string, config: Config): SlashCommandResult | null;
10
+ export declare function getAvailableCommands(): string[];
11
+ //# sourceMappingURL=commands.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"commands.d.ts","sourceRoot":"","sources":["../../src/core/commands.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAA0B,MAAM,aAAa,CAAC;AAE7D,MAAM,WAAW,kBAAkB;IACjC,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,OAAO,CAAC;IACf,KAAK,CAAC,EAAE,OAAO,CAAC;IAChB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,QAAQ,CAAC,EAAE,OAAO,CAAC;CACpB;AAkFD,wBAAgB,kBAAkB,CAChC,KAAK,EAAE,MAAM,EACb,MAAM,EAAE,MAAM,GACb,kBAAkB,GAAG,IAAI,CAoB3B;AAED,wBAAgB,oBAAoB,IAAI,MAAM,EAAE,CAE/C"}
@@ -0,0 +1,90 @@
1
+ import { saveConfig } from "./config.js";
2
+ const HELP_MESSAGE = `
3
+ ACTIVO - AI 코드 품질 분석 도구
4
+
5
+ [슬래시 커맨드]
6
+ /help 이 도움말 표시
7
+ /exit, /quit 종료
8
+ /clear 채팅 기록 삭제
9
+ /model <name> 모델 변경 (예: /model qwen2.5:7b)
10
+ /info 현재 설정 정보 표시
11
+
12
+ [단축키]
13
+ Enter 메시지 전송
14
+ ESC 진행 중 작업 취소
15
+ Ctrl+C x2 종료
16
+
17
+ [사용 예시]
18
+ "src 폴더 구조 보여줘"
19
+ "package.json 분석해줘"
20
+ "코드 품질 검사해줘"
21
+ "PDF를 마크다운으로 변환해줘"
22
+ `.trim();
23
+ const commandHandlers = {
24
+ help: () => ({
25
+ output: HELP_MESSAGE,
26
+ showHelp: true,
27
+ }),
28
+ exit: () => ({
29
+ exit: true,
30
+ output: "Goodbye!",
31
+ }),
32
+ quit: () => ({
33
+ exit: true,
34
+ output: "Goodbye!",
35
+ }),
36
+ clear: () => ({
37
+ clear: true,
38
+ output: "채팅 기록이 삭제되었습니다.",
39
+ }),
40
+ model: (args, config) => {
41
+ if (args.length === 0) {
42
+ return {
43
+ output: `현재 모델: ${config.ollama.model}\n\n사용법: /model <model_name>\n예시: /model qwen2.5:7b`,
44
+ };
45
+ }
46
+ const newModel = args[0];
47
+ config.ollama.model = newModel;
48
+ saveConfig(config);
49
+ return {
50
+ changeModel: newModel,
51
+ output: `모델이 "${newModel}"로 변경되었습니다.`,
52
+ };
53
+ },
54
+ info: (args, config) => {
55
+ const info = `
56
+ [ACTIVO 정보]
57
+ 버전: 0.2.1
58
+
59
+ [Ollama 설정]
60
+ URL: ${config.ollama.baseUrl}
61
+ 모델: ${config.ollama.model}
62
+ 컨텍스트: ${config.ollama.contextLength}
63
+
64
+ [표준 디렉토리]
65
+ ${config.standards.directory}
66
+ `.trim();
67
+ return { output: info };
68
+ },
69
+ };
70
+ export function handleSlashCommand(input, config) {
71
+ // "/" 로 시작하지 않으면 null 반환
72
+ if (!input.startsWith("/")) {
73
+ return null;
74
+ }
75
+ // 파싱: "/" 제거 후 공백으로 분할
76
+ const trimmed = input.slice(1).trim();
77
+ const [command, ...args] = trimmed.split(/\s+/);
78
+ if (!command) {
79
+ return { output: "명령어를 입력하세요. /help 로 도움말을 확인하세요." };
80
+ }
81
+ const handler = commandHandlers[command.toLowerCase()];
82
+ if (handler) {
83
+ return handler(args, config);
84
+ }
85
+ return { output: `알 수 없는 명령어: /${command}\n/help 로 사용 가능한 명령어를 확인하세요.` };
86
+ }
87
+ export function getAvailableCommands() {
88
+ return Object.keys(commandHandlers);
89
+ }
90
+ //# sourceMappingURL=commands.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"commands.js","sourceRoot":"","sources":["../../src/core/commands.ts"],"names":[],"mappings":"AAAA,OAAO,EAAsB,UAAU,EAAE,MAAM,aAAa,CAAC;AAY7D,MAAM,YAAY,GAAG;;;;;;;;;;;;;;;;;;;;CAoBpB,CAAC,IAAI,EAAE,CAAC;AAET,MAAM,eAAe,GAAmC;IACtD,IAAI,EAAE,GAAG,EAAE,CAAC,CAAC;QACX,MAAM,EAAE,YAAY;QACpB,QAAQ,EAAE,IAAI;KACf,CAAC;IAEF,IAAI,EAAE,GAAG,EAAE,CAAC,CAAC;QACX,IAAI,EAAE,IAAI;QACV,MAAM,EAAE,UAAU;KACnB,CAAC;IAEF,IAAI,EAAE,GAAG,EAAE,CAAC,CAAC;QACX,IAAI,EAAE,IAAI;QACV,MAAM,EAAE,UAAU;KACnB,CAAC;IAEF,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC;QACZ,KAAK,EAAE,IAAI;QACX,MAAM,EAAE,iBAAiB;KAC1B,CAAC;IAEF,KAAK,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE;QACtB,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACtB,OAAO;gBACL,MAAM,EAAE,UAAU,MAAM,CAAC,MAAM,CAAC,KAAK,qDAAqD;aAC3F,CAAC;QACJ,CAAC;QAED,MAAM,QAAQ,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;QACzB,MAAM,CAAC,MAAM,CAAC,KAAK,GAAG,QAAQ,CAAC;QAC/B,UAAU,CAAC,MAAM,CAAC,CAAC;QAEnB,OAAO;YACL,WAAW,EAAE,QAAQ;YACrB,MAAM,EAAE,QAAQ,QAAQ,aAAa;SACtC,CAAC;IACJ,CAAC;IAED,IAAI,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE;QACrB,MAAM,IAAI,GAAG;;;;;SAKR,MAAM,CAAC,MAAM,CAAC,OAAO;QACtB,MAAM,CAAC,MAAM,CAAC,KAAK;UACjB,MAAM,CAAC,MAAM,CAAC,aAAa;;;IAGjC,MAAM,CAAC,SAAS,CAAC,SAAS;CAC7B,CAAC,IAAI,EAAE,CAAC;QAEL,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;IAC1B,CAAC;CACF,CAAC;AAEF,MAAM,UAAU,kBAAkB,CAChC,KAAa,EACb,MAAc;IAEd,yBAAyB;IACzB,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;QAC3B,OAAO,IAAI,CAAC;IACd,CAAC;IAED,uBAAuB;IACvB,MAAM,OAAO,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;IACtC,MAAM,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC,GAAG,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IAEhD,IAAI,CAAC,OAAO,EAAE,CAAC;QACb,OAAO,EAAE,MAAM,EAAE,iCAAiC,EAAE,CAAC;IACvD,CAAC;IAED,MAAM,OAAO,GAAG,eAAe,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC,CAAC;IACvD,IAAI,OAAO,EAAE,CAAC;QACZ,OAAO,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;IAC/B,CAAC;IAED,OAAO,EAAE,MAAM,EAAE,gBAAgB,OAAO,8BAA8B,EAAE,CAAC;AAC3E,CAAC;AAED,MAAM,UAAU,oBAAoB;IAClC,OAAO,MAAM,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;AACtC,CAAC"}
@@ -1 +1 @@
1
- {"version":3,"file":"App.d.ts","sourceRoot":"","sources":["../../src/ui/App.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAmD,MAAM,OAAO,CAAC;AAExE,OAAO,EAAE,MAAM,EAAE,MAAM,mBAAmB,CAAC;AAQ3C,UAAU,QAAQ;IAChB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,MAAM,EAAE,MAAM,CAAC;IACf,MAAM,CAAC,EAAE,OAAO,CAAC;CAClB;AAYD,wBAAgB,GAAG,CAAC,EAAE,aAAa,EAAE,MAAM,EAAE,MAAM,EAAE,EAAE,QAAQ,GAAG,KAAK,CAAC,YAAY,CAmNnF"}
1
+ {"version":3,"file":"App.d.ts","sourceRoot":"","sources":["../../src/ui/App.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAmD,MAAM,OAAO,CAAC;AAExE,OAAO,EAAE,MAAM,EAAE,MAAM,mBAAmB,CAAC;AAS3C,UAAU,QAAQ;IAChB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,MAAM,EAAE,MAAM,CAAC;IACf,MAAM,CAAC,EAAE,OAAO,CAAC;CAClB;AAYD,wBAAgB,GAAG,CAAC,EAAE,aAAa,EAAE,MAAM,EAAE,MAAM,EAAE,EAAE,QAAQ,GAAG,KAAK,CAAC,YAAY,CAsPnF"}
package/dist/ui/App.js CHANGED
@@ -3,6 +3,7 @@ import { useState, useEffect, useCallback, useRef } from "react";
3
3
  import { Box, Text, useApp, useInput } from "ink";
4
4
  import { OllamaClient } from "../core/llm/ollama.js";
5
5
  import { streamProcessMessage } from "../core/agent.js";
6
+ import { handleSlashCommand } from "../core/commands.js";
6
7
  import { InputBox } from "./components/InputBox.js";
7
8
  import { MessageList } from "./components/MessageList.js";
8
9
  import { StatusBar } from "./components/StatusBar.js";
@@ -16,6 +17,7 @@ export function App({ initialPrompt, config, resume }) {
16
17
  const [toolStatus, setToolStatus] = useState(null);
17
18
  const [error, setError] = useState(null);
18
19
  const [client] = useState(() => new OllamaClient(config.ollama));
20
+ const [currentModel, setCurrentModel] = useState(config.ollama.model);
19
21
  const [exitPending, setExitPending] = useState(false);
20
22
  const [cancelled, setCancelled] = useState(false);
21
23
  const abortControllerRef = useRef(null);
@@ -72,8 +74,35 @@ export function App({ initialPrompt, config, resume }) {
72
74
  if (!text.trim() || isProcessing)
73
75
  return;
74
76
  setInput("");
75
- setIsProcessing(true);
76
77
  setError(null);
78
+ // Handle slash commands first
79
+ if (text.startsWith("/")) {
80
+ const result = handleSlashCommand(text, config);
81
+ if (result) {
82
+ // Add command as user message
83
+ setMessages((prev) => [...prev, { role: "user", content: text }]);
84
+ if (result.exit) {
85
+ const exitMsg = { role: "assistant", content: result.output || "Goodbye!" };
86
+ setMessages((prev) => [...prev, exitMsg]);
87
+ setTimeout(() => exit(), 500);
88
+ return;
89
+ }
90
+ if (result.clear) {
91
+ setMessages([]);
92
+ return;
93
+ }
94
+ if (result.changeModel) {
95
+ setCurrentModel(result.changeModel);
96
+ client.setModel(result.changeModel);
97
+ }
98
+ if (result.output) {
99
+ const outputMsg = { role: "assistant", content: result.output };
100
+ setMessages((prev) => [...prev, outputMsg]);
101
+ }
102
+ return;
103
+ }
104
+ }
105
+ setIsProcessing(true);
77
106
  setCancelled(false);
78
107
  // Create AbortController for this request
79
108
  const abortController = new AbortController();
@@ -162,6 +191,6 @@ export function App({ initialPrompt, config, resume }) {
162
191
  abortControllerRef.current = null;
163
192
  }
164
193
  }, [messages, client, config, isProcessing]);
165
- return (_jsxs(Box, { flexDirection: "column", height: "100%", children: [_jsx(Box, { flexDirection: "column", flexGrow: 1, children: _jsx(MessageList, { messages: messages }) }), currentTool && (_jsx(ToolStatus, { tool: currentTool, status: toolStatus || "running" })), error && (_jsx(Box, { marginY: 1, children: _jsxs(Text, { color: "red", children: ["\u26A0 ", error] }) })), exitPending && (_jsx(Box, { marginY: 1, children: _jsx(Text, { color: "yellow", children: "Press Ctrl+C again to exit" }) })), _jsx(InputBox, { value: input, onChange: setInput, onSubmit: handleSubmit, isProcessing: isProcessing, placeholder: "Type your message..." }), _jsx(StatusBar, { model: config.ollama.model, isProcessing: isProcessing, messageCount: messages.length })] }));
194
+ return (_jsxs(Box, { flexDirection: "column", height: "100%", children: [_jsx(Box, { flexDirection: "column", flexGrow: 1, children: _jsx(MessageList, { messages: messages }) }), currentTool && (_jsx(ToolStatus, { tool: currentTool, status: toolStatus || "running" })), error && (_jsx(Box, { marginY: 1, children: _jsxs(Text, { color: "red", children: ["\u26A0 ", error] }) })), exitPending && (_jsx(Box, { marginY: 1, children: _jsx(Text, { color: "yellow", children: "Press Ctrl+C again to exit" }) })), _jsx(InputBox, { value: input, onChange: setInput, onSubmit: handleSubmit, isProcessing: isProcessing, placeholder: "Type your message..." }), _jsx(StatusBar, { model: currentModel, isProcessing: isProcessing, messageCount: messages.length })] }));
166
195
  }
167
196
  //# sourceMappingURL=App.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"App.js","sourceRoot":"","sources":["../../src/ui/App.tsx"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"App.js","sourceRoot":"","sources":["../../src/ui/App.tsx"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "activo",
3
- "version": "0.2.0",
3
+ "version": "0.2.2",
4
4
  "description": "AI-powered code quality analyzer with React Ink TUI, Tool Calling, and MCP support",
5
5
  "repository": {
6
6
  "type": "git",
@@ -0,0 +1,118 @@
1
+ import { Config, loadConfig, saveConfig } from "./config.js";
2
+
3
+ export interface SlashCommandResult {
4
+ output?: string;
5
+ exit?: boolean;
6
+ clear?: boolean;
7
+ changeModel?: string;
8
+ showHelp?: boolean;
9
+ }
10
+
11
+ type CommandHandler = (args: string[], config: Config) => SlashCommandResult;
12
+
13
+ const HELP_MESSAGE = `
14
+ ACTIVO - AI 코드 품질 분석 도구
15
+
16
+ [슬래시 커맨드]
17
+ /help 이 도움말 표시
18
+ /exit, /quit 종료
19
+ /clear 채팅 기록 삭제
20
+ /model <name> 모델 변경 (예: /model qwen2.5:7b)
21
+ /info 현재 설정 정보 표시
22
+
23
+ [단축키]
24
+ Enter 메시지 전송
25
+ ESC 진행 중 작업 취소
26
+ Ctrl+C x2 종료
27
+
28
+ [사용 예시]
29
+ "src 폴더 구조 보여줘"
30
+ "package.json 분석해줘"
31
+ "코드 품질 검사해줘"
32
+ "PDF를 마크다운으로 변환해줘"
33
+ `.trim();
34
+
35
+ const commandHandlers: Record<string, CommandHandler> = {
36
+ help: () => ({
37
+ output: HELP_MESSAGE,
38
+ showHelp: true,
39
+ }),
40
+
41
+ exit: () => ({
42
+ exit: true,
43
+ output: "Goodbye!",
44
+ }),
45
+
46
+ quit: () => ({
47
+ exit: true,
48
+ output: "Goodbye!",
49
+ }),
50
+
51
+ clear: () => ({
52
+ clear: true,
53
+ output: "채팅 기록이 삭제되었습니다.",
54
+ }),
55
+
56
+ model: (args, config) => {
57
+ if (args.length === 0) {
58
+ return {
59
+ output: `현재 모델: ${config.ollama.model}\n\n사용법: /model <model_name>\n예시: /model qwen2.5:7b`,
60
+ };
61
+ }
62
+
63
+ const newModel = args[0];
64
+ config.ollama.model = newModel;
65
+ saveConfig(config);
66
+
67
+ return {
68
+ changeModel: newModel,
69
+ output: `모델이 "${newModel}"로 변경되었습니다.`,
70
+ };
71
+ },
72
+
73
+ info: (args, config) => {
74
+ const info = `
75
+ [ACTIVO 정보]
76
+ 버전: 0.2.1
77
+
78
+ [Ollama 설정]
79
+ URL: ${config.ollama.baseUrl}
80
+ 모델: ${config.ollama.model}
81
+ 컨텍스트: ${config.ollama.contextLength}
82
+
83
+ [표준 디렉토리]
84
+ ${config.standards.directory}
85
+ `.trim();
86
+
87
+ return { output: info };
88
+ },
89
+ };
90
+
91
+ export function handleSlashCommand(
92
+ input: string,
93
+ config: Config
94
+ ): SlashCommandResult | null {
95
+ // "/" 로 시작하지 않으면 null 반환
96
+ if (!input.startsWith("/")) {
97
+ return null;
98
+ }
99
+
100
+ // 파싱: "/" 제거 후 공백으로 분할
101
+ const trimmed = input.slice(1).trim();
102
+ const [command, ...args] = trimmed.split(/\s+/);
103
+
104
+ if (!command) {
105
+ return { output: "명령어를 입력하세요. /help 로 도움말을 확인하세요." };
106
+ }
107
+
108
+ const handler = commandHandlers[command.toLowerCase()];
109
+ if (handler) {
110
+ return handler(args, config);
111
+ }
112
+
113
+ return { output: `알 수 없는 명령어: /${command}\n/help 로 사용 가능한 명령어를 확인하세요.` };
114
+ }
115
+
116
+ export function getAvailableCommands(): string[] {
117
+ return Object.keys(commandHandlers);
118
+ }
package/src/ui/App.tsx CHANGED
@@ -3,6 +3,7 @@ import { Box, Text, useApp, useInput } from "ink";
3
3
  import { Config } from "../core/config.js";
4
4
  import { OllamaClient, ChatMessage } from "../core/llm/ollama.js";
5
5
  import { streamProcessMessage, AgentEvent } from "../core/agent.js";
6
+ import { handleSlashCommand } from "../core/commands.js";
6
7
  import { InputBox } from "./components/InputBox.js";
7
8
  import { MessageList } from "./components/MessageList.js";
8
9
  import { StatusBar } from "./components/StatusBar.js";
@@ -33,6 +34,7 @@ export function App({ initialPrompt, config, resume }: AppProps): React.ReactEle
33
34
  const [toolStatus, setToolStatus] = useState<"running" | "complete" | "error" | null>(null);
34
35
  const [error, setError] = useState<string | null>(null);
35
36
  const [client] = useState(() => new OllamaClient(config.ollama));
37
+ const [currentModel, setCurrentModel] = useState(config.ollama.model);
36
38
  const [exitPending, setExitPending] = useState(false);
37
39
  const [cancelled, setCancelled] = useState(false);
38
40
  const abortControllerRef = useRef<AbortController | null>(null);
@@ -93,8 +95,42 @@ export function App({ initialPrompt, config, resume }: AppProps): React.ReactEle
93
95
  if (!text.trim() || isProcessing) return;
94
96
 
95
97
  setInput("");
96
- setIsProcessing(true);
97
98
  setError(null);
99
+
100
+ // Handle slash commands first
101
+ if (text.startsWith("/")) {
102
+ const result = handleSlashCommand(text, config);
103
+ if (result) {
104
+ // Add command as user message
105
+ setMessages((prev) => [...prev, { role: "user", content: text }]);
106
+
107
+ if (result.exit) {
108
+ const exitMsg: Message = { role: "assistant", content: result.output || "Goodbye!" };
109
+ setMessages((prev) => [...prev, exitMsg]);
110
+ setTimeout(() => exit(), 500);
111
+ return;
112
+ }
113
+
114
+ if (result.clear) {
115
+ setMessages([]);
116
+ return;
117
+ }
118
+
119
+ if (result.changeModel) {
120
+ setCurrentModel(result.changeModel);
121
+ client.setModel(result.changeModel);
122
+ }
123
+
124
+ if (result.output) {
125
+ const outputMsg: Message = { role: "assistant", content: result.output };
126
+ setMessages((prev) => [...prev, outputMsg]);
127
+ }
128
+
129
+ return;
130
+ }
131
+ }
132
+
133
+ setIsProcessing(true);
98
134
  setCancelled(false);
99
135
 
100
136
  // Create AbortController for this request
@@ -229,7 +265,7 @@ export function App({ initialPrompt, config, resume }: AppProps): React.ReactEle
229
265
 
230
266
  {/* Status Bar */}
231
267
  <StatusBar
232
- model={config.ollama.model}
268
+ model={currentModel}
233
269
  isProcessing={isProcessing}
234
270
  messageCount={messages.length}
235
271
  />