skyloom 1.13.6 → 1.13.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.
Files changed (193) hide show
  1. package/.github/workflows/ci.yml +36 -36
  2. package/README.md +220 -159
  3. package/config/providers.yaml +39 -39
  4. package/config/skills/api_integrator/SKILL.md +15 -15
  5. package/config/skills/arch_designer/SKILL.md +13 -13
  6. package/config/skills/ci_cd_manager/SKILL.md +14 -14
  7. package/config/skills/code_analysis/SKILL.md +13 -13
  8. package/config/skills/code_generator/SKILL.md +12 -12
  9. package/config/skills/code_reviewer/SKILL.md +13 -13
  10. package/config/skills/content_writer/SKILL.md +14 -14
  11. package/config/skills/data_transformer/SKILL.md +15 -15
  12. package/config/skills/document_analysis/SKILL.md +13 -13
  13. package/config/skills/emotional_companion/SKILL.md +15 -15
  14. package/config/skills/performance_checker/SKILL.md +14 -14
  15. package/config/skills/security_auditor/SKILL.md +14 -14
  16. package/config/skills/self_evolve/SKILL.md +13 -13
  17. package/config/skills/sys_operator/SKILL.md +15 -15
  18. package/config/skills/task_planner/SKILL.md +14 -14
  19. package/config/skills/web_research/SKILL.md +14 -14
  20. package/config/skills/workflow_designer/SKILL.md +13 -13
  21. package/dist/agents/dew.js +52 -52
  22. package/dist/agents/fair.js +84 -84
  23. package/dist/agents/fog.js +30 -30
  24. package/dist/agents/frost.js +32 -32
  25. package/dist/agents/rain.js +32 -32
  26. package/dist/agents/snow.js +68 -68
  27. package/dist/cli/commands_md.d.ts +41 -0
  28. package/dist/cli/commands_md.d.ts.map +1 -0
  29. package/dist/cli/commands_md.js +140 -0
  30. package/dist/cli/commands_md.js.map +1 -0
  31. package/dist/cli/input_macros.d.ts +28 -0
  32. package/dist/cli/input_macros.d.ts.map +1 -0
  33. package/dist/cli/input_macros.js +120 -0
  34. package/dist/cli/input_macros.js.map +1 -0
  35. package/dist/cli/loom.d.ts +220 -0
  36. package/dist/cli/loom.d.ts.map +1 -0
  37. package/dist/cli/loom.js +1094 -0
  38. package/dist/cli/loom.js.map +1 -0
  39. package/dist/cli/loom_chat.d.ts +20 -0
  40. package/dist/cli/loom_chat.d.ts.map +1 -0
  41. package/dist/cli/loom_chat.js +685 -0
  42. package/dist/cli/loom_chat.js.map +1 -0
  43. package/dist/cli/main.js +310 -14
  44. package/dist/cli/main.js.map +1 -1
  45. package/dist/cli/tui.d.ts.map +1 -1
  46. package/dist/cli/tui.js +7 -1
  47. package/dist/cli/tui.js.map +1 -1
  48. package/dist/core/agent.d.ts +20 -0
  49. package/dist/core/agent.d.ts.map +1 -1
  50. package/dist/core/agent.js +199 -16
  51. package/dist/core/agent.js.map +1 -1
  52. package/dist/core/factory.d.ts.map +1 -1
  53. package/dist/core/factory.js +34 -2
  54. package/dist/core/factory.js.map +1 -1
  55. package/dist/core/file_checkpoint.d.ts +57 -0
  56. package/dist/core/file_checkpoint.d.ts.map +1 -0
  57. package/dist/core/file_checkpoint.js +162 -0
  58. package/dist/core/file_checkpoint.js.map +1 -0
  59. package/dist/core/hooks.d.ts +43 -0
  60. package/dist/core/hooks.d.ts.map +1 -0
  61. package/dist/core/hooks.js +110 -0
  62. package/dist/core/hooks.js.map +1 -0
  63. package/dist/core/llm.d.ts.map +1 -1
  64. package/dist/core/llm.js +15 -9
  65. package/dist/core/llm.js.map +1 -1
  66. package/dist/core/longdoc.js +5 -5
  67. package/dist/core/mcp.d.ts +16 -0
  68. package/dist/core/mcp.d.ts.map +1 -1
  69. package/dist/core/mcp.js +55 -0
  70. package/dist/core/mcp.js.map +1 -1
  71. package/dist/core/model_config.d.ts +40 -0
  72. package/dist/core/model_config.d.ts.map +1 -0
  73. package/dist/core/model_config.js +191 -0
  74. package/dist/core/model_config.js.map +1 -0
  75. package/dist/core/skill.d.ts +7 -0
  76. package/dist/core/skill.d.ts.map +1 -1
  77. package/dist/core/skill.js +47 -0
  78. package/dist/core/skill.js.map +1 -1
  79. package/dist/core/skymd.d.ts +39 -0
  80. package/dist/core/skymd.d.ts.map +1 -0
  81. package/dist/core/skymd.js +177 -0
  82. package/dist/core/skymd.js.map +1 -0
  83. package/dist/core/tool.d.ts +12 -0
  84. package/dist/core/tool.d.ts.map +1 -1
  85. package/dist/core/tool.js +30 -0
  86. package/dist/core/tool.js.map +1 -1
  87. package/dist/core/verify.d.ts +27 -0
  88. package/dist/core/verify.d.ts.map +1 -0
  89. package/dist/core/verify.js +62 -0
  90. package/dist/core/verify.js.map +1 -0
  91. package/dist/skills/loader.d.ts +22 -2
  92. package/dist/skills/loader.d.ts.map +1 -1
  93. package/dist/skills/loader.js +45 -15
  94. package/dist/skills/loader.js.map +1 -1
  95. package/dist/tools/builtin.d.ts.map +1 -1
  96. package/dist/tools/builtin.js +13 -3
  97. package/dist/tools/builtin.js.map +1 -1
  98. package/dist/tools/model_tool.d.ts +11 -0
  99. package/dist/tools/model_tool.d.ts.map +1 -0
  100. package/dist/tools/model_tool.js +71 -0
  101. package/dist/tools/model_tool.js.map +1 -0
  102. package/dist/tools/todo.d.ts +30 -0
  103. package/dist/tools/todo.d.ts.map +1 -0
  104. package/dist/tools/todo.js +78 -0
  105. package/dist/tools/todo.js.map +1 -0
  106. package/docs/AESTHETIC_DESIGN.md +152 -144
  107. package/docs/OPTIMIZATION_PLAN.md +178 -178
  108. package/package.json +68 -68
  109. package/scripts/install.js +48 -48
  110. package/scripts/link.js +10 -10
  111. package/setup.bat +79 -79
  112. package/skill-test-ty2fOA/test.md +10 -10
  113. package/src/agents/dew.ts +70 -70
  114. package/src/agents/fair.ts +102 -102
  115. package/src/agents/fog.ts +48 -48
  116. package/src/agents/frost.ts +50 -50
  117. package/src/agents/rain.ts +50 -50
  118. package/src/agents/snow.ts +239 -239
  119. package/src/cli/commands_md.ts +112 -0
  120. package/src/cli/input_macros.ts +83 -0
  121. package/src/cli/loom.ts +982 -0
  122. package/src/cli/loom_chat.ts +598 -0
  123. package/src/cli/main.ts +255 -9
  124. package/src/cli/mode.ts +58 -58
  125. package/src/cli/tui.ts +228 -222
  126. package/src/core/agent/guard.ts +134 -134
  127. package/src/core/agent/task.ts +100 -100
  128. package/src/core/agent.ts +195 -16
  129. package/src/core/arbitrate.ts +162 -162
  130. package/src/core/catalog.ts +178 -178
  131. package/src/core/checkpoint.ts +94 -94
  132. package/src/core/estimate.ts +104 -104
  133. package/src/core/evolve.ts +191 -191
  134. package/src/core/factory.ts +31 -2
  135. package/src/core/file_checkpoint.ts +136 -0
  136. package/src/core/filter.ts +103 -103
  137. package/src/core/graph.ts +156 -156
  138. package/src/core/hooks.ts +126 -0
  139. package/src/core/icons.ts +53 -53
  140. package/src/core/index.ts +37 -37
  141. package/src/core/learn.ts +146 -146
  142. package/src/core/llm.ts +15 -9
  143. package/src/core/longdoc.ts +155 -155
  144. package/src/core/mcp.ts +48 -0
  145. package/src/core/mcp_server.ts +176 -176
  146. package/src/core/model_config.ts +157 -0
  147. package/src/core/profile.ts +255 -255
  148. package/src/core/router.ts +124 -124
  149. package/src/core/sandbox.ts +142 -142
  150. package/src/core/security.ts +243 -243
  151. package/src/core/skill.ts +42 -0
  152. package/src/core/skymd.ts +143 -0
  153. package/src/core/theme.ts +65 -65
  154. package/src/core/tool.ts +30 -0
  155. package/src/core/tool_router.ts +193 -193
  156. package/src/core/vector.ts +152 -152
  157. package/src/core/verify.ts +71 -0
  158. package/src/core/workspace.ts +150 -150
  159. package/src/plugins/loader.ts +66 -66
  160. package/src/skills/loader.ts +45 -16
  161. package/src/sql.js.d.ts +29 -29
  162. package/src/tools/builtin.ts +13 -3
  163. package/src/tools/computer.ts +269 -269
  164. package/src/tools/delegate.ts +49 -49
  165. package/src/tools/model_tool.ts +74 -0
  166. package/src/tools/todo.ts +76 -0
  167. package/src/web/tts.ts +93 -93
  168. package/tests/agent.test.ts +159 -159
  169. package/tests/agent_helpers.test.ts +48 -48
  170. package/tests/bus.test.ts +121 -121
  171. package/tests/catalog.test.ts +86 -86
  172. package/tests/checkpoint_commands.test.ts +124 -0
  173. package/tests/claude_compat.test.ts +110 -0
  174. package/tests/config.test.ts +41 -41
  175. package/tests/guard.test.ts +75 -75
  176. package/tests/icons.test.ts +45 -45
  177. package/tests/loom.test.ts +248 -0
  178. package/tests/memory.test.ts +170 -170
  179. package/tests/model_config.test.ts +109 -0
  180. package/tests/router.test.ts +86 -86
  181. package/tests/schemas.test.ts +51 -51
  182. package/tests/semantic.test.ts +83 -83
  183. package/tests/setup.ts +10 -10
  184. package/tests/skill.test.ts +172 -172
  185. package/tests/skymd.test.ts +146 -0
  186. package/tests/task.test.ts +60 -60
  187. package/tests/todo_toolstats.test.ts +94 -0
  188. package/tests/tool.test.ts +108 -108
  189. package/tests/tool_router.test.ts +71 -71
  190. package/tests/tui.test.ts +67 -67
  191. package/vitest.config.ts +17 -17
  192. package/=12 +0 -0
  193. package/=8 +0 -0
package/src/cli/tui.ts CHANGED
@@ -1,222 +1,228 @@
1
- /**
2
- * 天空织机 TUI — a polished *linear* terminal interface.
3
- *
4
- * Design note: the previous version tried to be a full-screen app, redrawing
5
- * the whole screen on every keystroke while the reply streamed linearly below
6
- * it — the two fought, the conversation never persisted, and hand-rolled
7
- * raw-mode editing mangled CJK width. This rewrite is linear (like Claude Code
8
- * / opencode): real readline line-editing + a CJK-aware wrapping stream
9
- * renderer. Robust, flicker-free, and it actually reads like a conversation.
10
- */
11
-
12
- import * as readline from "readline";
13
- import chalk from "chalk";
14
- import { agentTheme, PALETTE } from "../core/theme";
15
-
16
- const TUI_VERSION = (() => { try { return require("../../package.json").version; } catch { return ""; } })();
17
-
18
- export interface TUIContext {
19
- agent: any;
20
- agents: Map<string, any>;
21
- model: string;
22
- cost: string;
23
- width: number;
24
- height: number;
25
- }
26
-
27
- /* ── Slash commands (for tab-completion + the inline palette) ── */
28
- export const SLASH_COMMANDS: [string, string][] = [
29
- ["/fog", "≋ 雾 · 探索洞察"],
30
- ["/rain", "⸽ 雨 · 创造产出"],
31
- ["/frost", "✱ 霜 · 精炼品质"],
32
- ["/snow", "❉ 雪 · 架构规划"],
33
- ["/dew", "∘ 露 · 可靠守护"],
34
- ["/fair", "☼ 晴 · 情感陪伴"],
35
- ["/help", "查看所有命令"],
36
- ["/setup", "配置向导"],
37
- ["/model", "模型信息"],
38
- ["/cost", "费用统计"],
39
- ["/status", "状态总览"],
40
- ["/memory", "记忆状态"],
41
- ["/sessions", "会话列表"],
42
- ["/resume ", "恢复会话(序号/id)"],
43
- ["/new", "开始新会话"],
44
- ["/workspace", "工作空间"],
45
- ["/compact", "压缩上下文"],
46
- ["/clear", "清屏"],
47
- ["/task ", "多 Agent 编排"],
48
- ["/mcp", "MCP 服务器"],
49
- ["/version", "版本信息"],
50
- ["/quit", "退出"],
51
- ];
52
-
53
- /* ════════════════════════════════════════
54
- CJK-aware display width
55
- ════════════════════════════════════════ */
56
- /** Visual columns occupied by a single code point (CJK / fullwidth = 2). */
57
- export function charWidth(cp: number): number {
58
- if (cp === 0) return 0;
59
- if (cp < 32 || (cp >= 0x7f && cp < 0xa0)) return 0; // control
60
- // East-Asian wide / fullwidth ranges
61
- if (
62
- (cp >= 0x1100 && cp <= 0x115f) || // Hangul Jamo
63
- (cp >= 0x2e80 && cp <= 0x303e) || // CJK radicals, Kangxi, punctuation
64
- (cp >= 0x3041 && cp <= 0x33ff) || // Hiragana…CJK symbols
65
- (cp >= 0x3400 && cp <= 0x4dbf) || // CJK Ext A
66
- (cp >= 0x4e00 && cp <= 0x9fff) || // CJK Unified
67
- (cp >= 0xa000 && cp <= 0xa4cf) || // Yi
68
- (cp >= 0xac00 && cp <= 0xd7a3) || // Hangul syllables
69
- (cp >= 0xf900 && cp <= 0xfaff) || // CJK compat
70
- (cp >= 0xfe10 && cp <= 0xfe19) ||
71
- (cp >= 0xfe30 && cp <= 0xfe6f) || // CJK compat forms
72
- (cp >= 0xff00 && cp <= 0xff60) || // Fullwidth forms
73
- (cp >= 0xffe0 && cp <= 0xffe6) ||
74
- (cp >= 0x1f300 && cp <= 0x1faff) // emoji / pictographs
75
- ) return 2;
76
- return 1;
77
- }
78
-
79
- const ANSI_RE = /\x1b\[[0-9;]*m/g;
80
-
81
- /** Visual width of a string, ignoring ANSI color codes. */
82
- export function visualWidth(s: string): number {
83
- let w = 0;
84
- for (const ch of s.replace(ANSI_RE, "")) w += charWidth(ch.codePointAt(0) || 0);
85
- return w;
86
- }
87
-
88
- /** Pad a string (containing ANSI) to a visual width. */
89
- export function padVisual(s: string, width: number): string {
90
- const diff = width - visualWidth(s);
91
- return diff > 0 ? s + " ".repeat(diff) : s;
92
- }
93
-
94
- /* ════════════════════════════════════════
95
- Streaming renderer word-wrap aware, CJK aware
96
- ════════════════════════════════════════ */
97
- /**
98
- * Writes streamed text with a fixed left gutter, wrapping at the terminal
99
- * width. English wraps on word boundaries; CJK wraps per glyph. Color is
100
- * applied per flushed chunk so styling survives wrapping.
101
- */
102
- export class StreamRenderer {
103
- private col = 0;
104
- private word = "";
105
- private atLineStart = true;
106
- private out: NodeJS.WriteStream;
107
- private gutter: string;
108
- private maxCols: number;
109
- private color: (s: string) => string;
110
-
111
- constructor(out: NodeJS.WriteStream, opts?: { gutter?: string; color?: (s: string) => string }) {
112
- this.out = out;
113
- this.gutter = opts?.gutter ?? " ";
114
- this.color = opts?.color ?? ((s) => s);
115
- const cols = out.columns || 80;
116
- // content width excludes the gutter; clamp for readability
117
- this.maxCols = Math.max(32, Math.min(cols - visualWidth(this.gutter) - 1, 96));
118
- }
119
-
120
- /** Lazily emit the left gutter at the start of each visual line. */
121
- private startLine() { if (this.atLineStart) { this.out.write(this.gutter); this.atLineStart = false; } }
122
- private newline() { this.out.write("\n"); this.atLineStart = true; this.col = 0; }
123
-
124
- private flushWord() {
125
- if (!this.word) return;
126
- const w = visualWidth(this.word);
127
- if (this.col > 0 && this.col + w > this.maxCols) this.newline();
128
- this.startLine();
129
- this.out.write(this.color(this.word));
130
- this.col += w;
131
- this.word = "";
132
- }
133
-
134
- /** Feed a chunk of streamed text. */
135
- write(text: string) {
136
- for (const ch of text) {
137
- if (ch === "\r") continue; // normalize CRLF / stray CR from providers
138
- if (ch === "\n") { this.flushWord(); this.newline(); continue; }
139
- if (ch === " " || ch === "\t") {
140
- this.flushWord();
141
- if (this.col > 0 && this.col < this.maxCols) { this.startLine(); this.out.write(" "); this.col += 1; }
142
- continue;
143
- }
144
- const cp = ch.codePointAt(0) || 0;
145
- if (charWidth(cp) === 2) {
146
- // CJK / wide: flush any pending latin word, then place this glyph
147
- this.flushWord();
148
- if (this.col > 0 && this.col + 2 > this.maxCols) this.newline();
149
- this.startLine();
150
- this.out.write(this.color(ch));
151
- this.col += 2;
152
- } else {
153
- this.word += ch;
154
- // very long unbroken token: hard-break to avoid overflow
155
- if (visualWidth(this.word) >= this.maxCols) this.flushWord();
156
- }
157
- }
158
- }
159
-
160
- /** Flush any buffered word (call before switching styles / ending). */
161
- flush() { this.flushWord(); }
162
- }
163
-
164
- /* ════════════════════════════════════════
165
- Input — readline-based, robust line editing
166
- ════════════════════════════════════════ */
167
- /** Tab-completer for slash commands. */
168
- function slashCompleter(line: string): [string[], string] {
169
- if (!line.startsWith("/")) return [[], line];
170
- const names = SLASH_COMMANDS.map(([c]) => c.trimEnd());
171
- const hits = names.filter((c) => c.startsWith(line));
172
- return [hits.length ? hits : names, line];
173
- }
174
-
175
- /** The prompt string for an agent: a small mineral seal + chevron. */
176
- export function promptFor(agentName: string): string {
177
- const t = agentTheme(agentName);
178
- return chalk.hex(t.hex)(` ${t.symbol} ${t.kanji} `) + chalk.hex(PALETTE.inkLight)("❯ ");
179
- }
180
-
181
- /** Cross-turn input history (↑/↓), shared by every per-turn reader. */
182
- const inputHistory: string[] = [];
183
-
184
- /**
185
- * Read one line with the agent-themed prompt. A fresh readline interface is
186
- * created and closed per call — this deliberately avoids clashing with the
187
- * separate readline prompts used by the setup wizard and tool-approval flow
188
- * (two live interfaces on one stdin corrupt input). History is preserved
189
- * manually across turns.
190
- */
191
- export function readLine(agentName: string, out: NodeJS.WriteStream = process.stdout): Promise<string> {
192
- return new Promise((resolve) => {
193
- const rl = readline.createInterface({
194
- input: process.stdin,
195
- output: out,
196
- completer: slashCompleter,
197
- terminal: process.stdin.isTTY ?? false,
198
- history: [...inputHistory],
199
- historySize: 200,
200
- } as any);
201
- rl.on("SIGINT", () => { out.write("\n" + chalk.dim(" 再会。\n")); rl.close(); process.exit(0); });
202
- rl.question(promptFor(agentName), (answer) => {
203
- const trimmed = answer.trim();
204
- if (trimmed) inputHistory.unshift(trimmed);
205
- rl.close();
206
- resolve(trimmed);
207
- });
208
- });
209
- }
210
-
211
- /** Render the inline slash-command palette (printed, not full-screen). */
212
- export function renderPalette(filter: string): string {
213
- const f = filter.toLowerCase();
214
- const matches = SLASH_COMMANDS.filter(([c]) => c.toLowerCase().startsWith(f));
215
- const list = matches.length ? matches : SLASH_COMMANDS;
216
- const lines = list.slice(0, 12).map(([cmd, desc]) => {
217
- const isAgent = ["/fog", "/rain", "/frost", "/snow", "/dew", "/fair"].includes(cmd.trim());
218
- const name = isAgent ? chalk.hex(agentTheme(cmd.trim().slice(1)).hex)(cmd.padEnd(12)) : chalk.hex(PALETTE.inkMid)(cmd.padEnd(12));
219
- return " " + name + chalk.hex(PALETTE.inkLight)(desc);
220
- });
221
- return chalk.dim(" 命令 · Tab 补全\n") + lines.join("\n") + "\n";
222
- }
1
+ /**
2
+ * 天空织机 TUI — a polished *linear* terminal interface.
3
+ *
4
+ * Design note: the previous version tried to be a full-screen app, redrawing
5
+ * the whole screen on every keystroke while the reply streamed linearly below
6
+ * it — the two fought, the conversation never persisted, and hand-rolled
7
+ * raw-mode editing mangled CJK width. This rewrite is linear (like Claude Code
8
+ * / opencode): real readline line-editing + a CJK-aware wrapping stream
9
+ * renderer. Robust, flicker-free, and it actually reads like a conversation.
10
+ */
11
+
12
+ import * as readline from "readline";
13
+ import chalk from "chalk";
14
+ import { agentTheme, PALETTE } from "../core/theme";
15
+
16
+ const TUI_VERSION = (() => { try { return require("../../package.json").version; } catch { return ""; } })();
17
+
18
+ export interface TUIContext {
19
+ agent: any;
20
+ agents: Map<string, any>;
21
+ model: string;
22
+ cost: string;
23
+ width: number;
24
+ height: number;
25
+ }
26
+
27
+ /* ── Slash commands (for tab-completion + the inline palette) ── */
28
+ export const SLASH_COMMANDS: [string, string][] = [
29
+ ["/fog", "≋ 雾 · 探索洞察"],
30
+ ["/rain", "⸽ 雨 · 创造产出"],
31
+ ["/frost", "✱ 霜 · 精炼品质"],
32
+ ["/snow", "❉ 雪 · 架构规划"],
33
+ ["/dew", "∘ 露 · 可靠守护"],
34
+ ["/fair", "☼ 晴 · 情感陪伴"],
35
+ ["/help", "查看所有命令"],
36
+ ["/setup", "配置向导"],
37
+ ["/init", "扫描项目生成 SKY.md"],
38
+ ["/plan", "切换计划模式(只读出方案)"],
39
+ ["/verify", "运行项目验证命令"],
40
+ ["/context", "上下文占用明细"],
41
+ ["/rewind", "回退本轮文件改动"],
42
+ ["/tools", "工具调用统计"],
43
+ ["/model", "查看/切换模型(独立/统一)"],
44
+ ["/cost", "费用统计"],
45
+ ["/status", "状态总览"],
46
+ ["/memory", "记忆状态"],
47
+ ["/sessions", "会话列表"],
48
+ ["/resume ", "恢复会话(序号/id)"],
49
+ ["/new", "开始新会话"],
50
+ ["/workspace", "工作空间"],
51
+ ["/compact", "压缩上下文"],
52
+ ["/clear", "清屏"],
53
+ ["/task ", "多 Agent 编排"],
54
+ ["/mcp", "MCP 服务器"],
55
+ ["/version", "版本信息"],
56
+ ["/quit", "退出"],
57
+ ];
58
+
59
+ /* ════════════════════════════════════════
60
+ CJK-aware display width
61
+ ════════════════════════════════════════ */
62
+ /** Visual columns occupied by a single code point (CJK / fullwidth = 2). */
63
+ export function charWidth(cp: number): number {
64
+ if (cp === 0) return 0;
65
+ if (cp < 32 || (cp >= 0x7f && cp < 0xa0)) return 0; // control
66
+ // East-Asian wide / fullwidth ranges
67
+ if (
68
+ (cp >= 0x1100 && cp <= 0x115f) || // Hangul Jamo
69
+ (cp >= 0x2e80 && cp <= 0x303e) || // CJK radicals, Kangxi, punctuation
70
+ (cp >= 0x3041 && cp <= 0x33ff) || // Hiragana…CJK symbols
71
+ (cp >= 0x3400 && cp <= 0x4dbf) || // CJK Ext A
72
+ (cp >= 0x4e00 && cp <= 0x9fff) || // CJK Unified
73
+ (cp >= 0xa000 && cp <= 0xa4cf) || // Yi
74
+ (cp >= 0xac00 && cp <= 0xd7a3) || // Hangul syllables
75
+ (cp >= 0xf900 && cp <= 0xfaff) || // CJK compat
76
+ (cp >= 0xfe10 && cp <= 0xfe19) ||
77
+ (cp >= 0xfe30 && cp <= 0xfe6f) || // CJK compat forms
78
+ (cp >= 0xff00 && cp <= 0xff60) || // Fullwidth forms
79
+ (cp >= 0xffe0 && cp <= 0xffe6) ||
80
+ (cp >= 0x1f300 && cp <= 0x1faff) // emoji / pictographs
81
+ ) return 2;
82
+ return 1;
83
+ }
84
+
85
+ const ANSI_RE = /\x1b\[[0-9;]*m/g;
86
+
87
+ /** Visual width of a string, ignoring ANSI color codes. */
88
+ export function visualWidth(s: string): number {
89
+ let w = 0;
90
+ for (const ch of s.replace(ANSI_RE, "")) w += charWidth(ch.codePointAt(0) || 0);
91
+ return w;
92
+ }
93
+
94
+ /** Pad a string (containing ANSI) to a visual width. */
95
+ export function padVisual(s: string, width: number): string {
96
+ const diff = width - visualWidth(s);
97
+ return diff > 0 ? s + " ".repeat(diff) : s;
98
+ }
99
+
100
+ /* ════════════════════════════════════════
101
+ Streaming renderer — word-wrap aware, CJK aware
102
+ ════════════════════════════════════════ */
103
+ /**
104
+ * Writes streamed text with a fixed left gutter, wrapping at the terminal
105
+ * width. English wraps on word boundaries; CJK wraps per glyph. Color is
106
+ * applied per flushed chunk so styling survives wrapping.
107
+ */
108
+ export class StreamRenderer {
109
+ private col = 0;
110
+ private word = "";
111
+ private atLineStart = true;
112
+ private out: NodeJS.WriteStream;
113
+ private gutter: string;
114
+ private maxCols: number;
115
+ private color: (s: string) => string;
116
+
117
+ constructor(out: NodeJS.WriteStream, opts?: { gutter?: string; color?: (s: string) => string }) {
118
+ this.out = out;
119
+ this.gutter = opts?.gutter ?? " ";
120
+ this.color = opts?.color ?? ((s) => s);
121
+ const cols = out.columns || 80;
122
+ // content width excludes the gutter; clamp for readability
123
+ this.maxCols = Math.max(32, Math.min(cols - visualWidth(this.gutter) - 1, 96));
124
+ }
125
+
126
+ /** Lazily emit the left gutter at the start of each visual line. */
127
+ private startLine() { if (this.atLineStart) { this.out.write(this.gutter); this.atLineStart = false; } }
128
+ private newline() { this.out.write("\n"); this.atLineStart = true; this.col = 0; }
129
+
130
+ private flushWord() {
131
+ if (!this.word) return;
132
+ const w = visualWidth(this.word);
133
+ if (this.col > 0 && this.col + w > this.maxCols) this.newline();
134
+ this.startLine();
135
+ this.out.write(this.color(this.word));
136
+ this.col += w;
137
+ this.word = "";
138
+ }
139
+
140
+ /** Feed a chunk of streamed text. */
141
+ write(text: string) {
142
+ for (const ch of text) {
143
+ if (ch === "\r") continue; // normalize CRLF / stray CR from providers
144
+ if (ch === "\n") { this.flushWord(); this.newline(); continue; }
145
+ if (ch === " " || ch === "\t") {
146
+ this.flushWord();
147
+ if (this.col > 0 && this.col < this.maxCols) { this.startLine(); this.out.write(" "); this.col += 1; }
148
+ continue;
149
+ }
150
+ const cp = ch.codePointAt(0) || 0;
151
+ if (charWidth(cp) === 2) {
152
+ // CJK / wide: flush any pending latin word, then place this glyph
153
+ this.flushWord();
154
+ if (this.col > 0 && this.col + 2 > this.maxCols) this.newline();
155
+ this.startLine();
156
+ this.out.write(this.color(ch));
157
+ this.col += 2;
158
+ } else {
159
+ this.word += ch;
160
+ // very long unbroken token: hard-break to avoid overflow
161
+ if (visualWidth(this.word) >= this.maxCols) this.flushWord();
162
+ }
163
+ }
164
+ }
165
+
166
+ /** Flush any buffered word (call before switching styles / ending). */
167
+ flush() { this.flushWord(); }
168
+ }
169
+
170
+ /* ════════════════════════════════════════
171
+ Input readline-based, robust line editing
172
+ ════════════════════════════════════════ */
173
+ /** Tab-completer for slash commands. */
174
+ function slashCompleter(line: string): [string[], string] {
175
+ if (!line.startsWith("/")) return [[], line];
176
+ const names = SLASH_COMMANDS.map(([c]) => c.trimEnd());
177
+ const hits = names.filter((c) => c.startsWith(line));
178
+ return [hits.length ? hits : names, line];
179
+ }
180
+
181
+ /** The prompt string for an agent: a small mineral seal + chevron. */
182
+ export function promptFor(agentName: string): string {
183
+ const t = agentTheme(agentName);
184
+ return chalk.hex(t.hex)(` ${t.symbol} ${t.kanji} `) + chalk.hex(PALETTE.inkLight)("❯ ");
185
+ }
186
+
187
+ /** Cross-turn input history (↑/↓), shared by every per-turn reader. */
188
+ const inputHistory: string[] = [];
189
+
190
+ /**
191
+ * Read one line with the agent-themed prompt. A fresh readline interface is
192
+ * created and closed per call — this deliberately avoids clashing with the
193
+ * separate readline prompts used by the setup wizard and tool-approval flow
194
+ * (two live interfaces on one stdin corrupt input). History is preserved
195
+ * manually across turns.
196
+ */
197
+ export function readLine(agentName: string, out: NodeJS.WriteStream = process.stdout): Promise<string> {
198
+ return new Promise((resolve) => {
199
+ const rl = readline.createInterface({
200
+ input: process.stdin,
201
+ output: out,
202
+ completer: slashCompleter,
203
+ terminal: process.stdin.isTTY ?? false,
204
+ history: [...inputHistory],
205
+ historySize: 200,
206
+ } as any);
207
+ rl.on("SIGINT", () => { out.write("\n" + chalk.dim(" 再会。\n")); rl.close(); process.exit(0); });
208
+ rl.question(promptFor(agentName), (answer) => {
209
+ const trimmed = answer.trim();
210
+ if (trimmed) inputHistory.unshift(trimmed);
211
+ rl.close();
212
+ resolve(trimmed);
213
+ });
214
+ });
215
+ }
216
+
217
+ /** Render the inline slash-command palette (printed, not full-screen). */
218
+ export function renderPalette(filter: string): string {
219
+ const f = filter.toLowerCase();
220
+ const matches = SLASH_COMMANDS.filter(([c]) => c.toLowerCase().startsWith(f));
221
+ const list = matches.length ? matches : SLASH_COMMANDS;
222
+ const lines = list.slice(0, 12).map(([cmd, desc]) => {
223
+ const isAgent = ["/fog", "/rain", "/frost", "/snow", "/dew", "/fair"].includes(cmd.trim());
224
+ const name = isAgent ? chalk.hex(agentTheme(cmd.trim().slice(1)).hex)(cmd.padEnd(12)) : chalk.hex(PALETTE.inkMid)(cmd.padEnd(12));
225
+ return " " + name + chalk.hex(PALETTE.inkLight)(desc);
226
+ });
227
+ return chalk.dim(" 命令 · Tab 补全\n") + lines.join("\n") + "\n";
228
+ }