lsd-pi 1.3.2 → 1.3.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.
Files changed (205) hide show
  1. package/dist/cli.js +2 -1
  2. package/dist/lsd-settings-manager.d.ts +2 -0
  3. package/dist/lsd-settings-manager.js +5 -0
  4. package/dist/resource-loader.js +33 -3
  5. package/dist/resources/extensions/browser-tools/tools/codegen.js +5 -5
  6. package/dist/resources/extensions/browser-tools/tools/navigation.js +107 -178
  7. package/dist/resources/extensions/browser-tools/tools/network-mock.js +112 -167
  8. package/dist/resources/extensions/browser-tools/tools/pages.js +182 -234
  9. package/dist/resources/extensions/browser-tools/tools/refs.js +202 -461
  10. package/dist/resources/extensions/browser-tools/tools/session.js +176 -323
  11. package/dist/resources/extensions/browser-tools/tools/state-persistence.js +91 -154
  12. package/dist/resources/extensions/browser-tools/utils.js +1 -1
  13. package/dist/resources/extensions/cache-timer/index.js +3 -2
  14. package/dist/resources/extensions/slash-commands/extension-manifest.json +2 -2
  15. package/dist/resources/extensions/slash-commands/fast.js +73 -0
  16. package/dist/resources/extensions/slash-commands/index.js +2 -0
  17. package/dist/resources/extensions/slash-commands/plan.js +37 -12
  18. package/dist/resources/extensions/subagent/background-job-manager.js +13 -0
  19. package/dist/resources/extensions/subagent/in-process-runner.js +387 -0
  20. package/dist/resources/extensions/subagent/index.js +278 -626
  21. package/dist/resources/extensions/subagent/legacy-runner.js +503 -0
  22. package/dist/resources/extensions/voice/index.js +96 -36
  23. package/dist/resources/extensions/voice/push-to-talk.js +26 -0
  24. package/dist/welcome-screen.js +2 -2
  25. package/package.json +1 -1
  26. package/packages/pi-agent-core/dist/agent.d.ts +19 -0
  27. package/packages/pi-agent-core/dist/agent.d.ts.map +1 -1
  28. package/packages/pi-agent-core/dist/agent.js +16 -0
  29. package/packages/pi-agent-core/dist/agent.js.map +1 -1
  30. package/packages/pi-agent-core/src/agent.ts +32 -2
  31. package/packages/pi-ai/dist/providers/openai-codex-responses.d.ts +34 -1
  32. package/packages/pi-ai/dist/providers/openai-codex-responses.d.ts.map +1 -1
  33. package/packages/pi-ai/dist/providers/openai-codex-responses.js +32 -4
  34. package/packages/pi-ai/dist/providers/openai-codex-responses.js.map +1 -1
  35. package/packages/pi-ai/dist/providers/openai-codex-responses.test.js +127 -16
  36. package/packages/pi-ai/dist/providers/openai-codex-responses.test.js.map +1 -1
  37. package/packages/pi-ai/dist/providers/openai-responses.d.ts +8 -1
  38. package/packages/pi-ai/dist/providers/openai-responses.d.ts.map +1 -1
  39. package/packages/pi-ai/dist/providers/openai-responses.fast-mode.test.d.ts +2 -0
  40. package/packages/pi-ai/dist/providers/openai-responses.fast-mode.test.d.ts.map +1 -0
  41. package/packages/pi-ai/dist/providers/openai-responses.fast-mode.test.js +67 -0
  42. package/packages/pi-ai/dist/providers/openai-responses.fast-mode.test.js.map +1 -0
  43. package/packages/pi-ai/dist/providers/openai-responses.js +21 -3
  44. package/packages/pi-ai/dist/providers/openai-responses.js.map +1 -1
  45. package/packages/pi-ai/dist/providers/simple-options.d.ts.map +1 -1
  46. package/packages/pi-ai/dist/providers/simple-options.js +2 -0
  47. package/packages/pi-ai/dist/providers/simple-options.js.map +1 -1
  48. package/packages/pi-ai/dist/types.d.ts +5 -0
  49. package/packages/pi-ai/dist/types.d.ts.map +1 -1
  50. package/packages/pi-ai/dist/types.js.map +1 -1
  51. package/packages/pi-ai/src/providers/openai-codex-responses.test.ts +143 -20
  52. package/packages/pi-ai/src/providers/openai-codex-responses.ts +47 -4
  53. package/packages/pi-ai/src/providers/openai-responses.fast-mode.test.ts +73 -0
  54. package/packages/pi-ai/src/providers/openai-responses.ts +26 -3
  55. package/packages/pi-ai/src/providers/simple-options.ts +2 -0
  56. package/packages/pi-ai/src/types.ts +5 -0
  57. package/packages/pi-coding-agent/dist/core/keybindings.d.ts +1 -1
  58. package/packages/pi-coding-agent/dist/core/keybindings.d.ts.map +1 -1
  59. package/packages/pi-coding-agent/dist/core/keybindings.js +2 -0
  60. package/packages/pi-coding-agent/dist/core/keybindings.js.map +1 -1
  61. package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
  62. package/packages/pi-coding-agent/dist/core/sdk.js +4 -2
  63. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  64. package/packages/pi-coding-agent/dist/core/settings-manager.collapse-tool-calls.test.d.ts +2 -0
  65. package/packages/pi-coding-agent/dist/core/settings-manager.collapse-tool-calls.test.d.ts.map +1 -0
  66. package/packages/pi-coding-agent/dist/core/settings-manager.collapse-tool-calls.test.js +35 -0
  67. package/packages/pi-coding-agent/dist/core/settings-manager.collapse-tool-calls.test.js.map +1 -0
  68. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +12 -0
  69. package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
  70. package/packages/pi-coding-agent/dist/core/settings-manager.fast-mode.test.d.ts +2 -0
  71. package/packages/pi-coding-agent/dist/core/settings-manager.fast-mode.test.d.ts.map +1 -0
  72. package/packages/pi-coding-agent/dist/core/settings-manager.fast-mode.test.js +35 -0
  73. package/packages/pi-coding-agent/dist/core/settings-manager.fast-mode.test.js.map +1 -0
  74. package/packages/pi-coding-agent/dist/core/settings-manager.js +24 -0
  75. package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
  76. package/packages/pi-coding-agent/dist/core/slash-commands.d.ts.map +1 -1
  77. package/packages/pi-coding-agent/dist/core/slash-commands.js +1 -0
  78. package/packages/pi-coding-agent/dist/core/slash-commands.js.map +1 -1
  79. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  80. package/packages/pi-coding-agent/dist/core/system-prompt.js +6 -1
  81. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  82. package/packages/pi-coding-agent/dist/core/tool-priority.d.ts +4 -0
  83. package/packages/pi-coding-agent/dist/core/tool-priority.d.ts.map +1 -0
  84. package/packages/pi-coding-agent/dist/core/tool-priority.js +18 -0
  85. package/packages/pi-coding-agent/dist/core/tool-priority.js.map +1 -0
  86. package/packages/pi-coding-agent/dist/core/tool-priority.test.d.ts +2 -0
  87. package/packages/pi-coding-agent/dist/core/tool-priority.test.d.ts.map +1 -0
  88. package/packages/pi-coding-agent/dist/core/tool-priority.test.js +27 -0
  89. package/packages/pi-coding-agent/dist/core/tool-priority.test.js.map +1 -0
  90. package/packages/pi-coding-agent/dist/core/tools/edit-diff.d.ts +5 -0
  91. package/packages/pi-coding-agent/dist/core/tools/edit-diff.d.ts.map +1 -1
  92. package/packages/pi-coding-agent/dist/core/tools/edit-diff.js +21 -0
  93. package/packages/pi-coding-agent/dist/core/tools/edit-diff.js.map +1 -1
  94. package/packages/pi-coding-agent/dist/core/tools/edit-diff.test.js +16 -1
  95. package/packages/pi-coding-agent/dist/core/tools/edit-diff.test.js.map +1 -1
  96. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-summary-line.test.d.ts +2 -0
  97. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-summary-line.test.d.ts.map +1 -0
  98. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-summary-line.test.js +34 -0
  99. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-summary-line.test.js.map +1 -0
  100. package/packages/pi-coding-agent/dist/modes/interactive/components/btw-overlay.d.ts +45 -0
  101. package/packages/pi-coding-agent/dist/modes/interactive/components/btw-overlay.d.ts.map +1 -0
  102. package/packages/pi-coding-agent/dist/modes/interactive/components/btw-overlay.js +314 -0
  103. package/packages/pi-coding-agent/dist/modes/interactive/components/btw-overlay.js.map +1 -0
  104. package/packages/pi-coding-agent/dist/modes/interactive/components/btw-overlay.test.d.ts +2 -0
  105. package/packages/pi-coding-agent/dist/modes/interactive/components/btw-overlay.test.d.ts.map +1 -0
  106. package/packages/pi-coding-agent/dist/modes/interactive/components/btw-overlay.test.js +122 -0
  107. package/packages/pi-coding-agent/dist/modes/interactive/components/btw-overlay.test.js.map +1 -0
  108. package/packages/pi-coding-agent/dist/modes/interactive/components/diff.d.ts +7 -5
  109. package/packages/pi-coding-agent/dist/modes/interactive/components/diff.d.ts.map +1 -1
  110. package/packages/pi-coding-agent/dist/modes/interactive/components/diff.js +86 -28
  111. package/packages/pi-coding-agent/dist/modes/interactive/components/diff.js.map +1 -1
  112. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts +4 -0
  113. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
  114. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +23 -10
  115. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
  116. package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts +8 -0
  117. package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts.map +1 -1
  118. package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js +52 -6
  119. package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js.map +1 -1
  120. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +19 -0
  121. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  122. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +127 -14
  123. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  124. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-summary-line.d.ts +14 -0
  125. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-summary-line.d.ts.map +1 -0
  126. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-summary-line.js +93 -0
  127. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-summary-line.js.map +1 -0
  128. package/packages/pi-coding-agent/dist/modes/interactive/controllers/__tests__/chat-controller.collapsed-tool-summary.test.d.ts +2 -0
  129. package/packages/pi-coding-agent/dist/modes/interactive/controllers/__tests__/chat-controller.collapsed-tool-summary.test.d.ts.map +1 -0
  130. package/packages/pi-coding-agent/dist/modes/interactive/controllers/__tests__/chat-controller.collapsed-tool-summary.test.js +328 -0
  131. package/packages/pi-coding-agent/dist/modes/interactive/controllers/__tests__/chat-controller.collapsed-tool-summary.test.js.map +1 -0
  132. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  133. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +123 -0
  134. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  135. package/packages/pi-coding-agent/dist/modes/interactive/controllers/extension-ui-controller.js +1 -1
  136. package/packages/pi-coding-agent/dist/modes/interactive/controllers/extension-ui-controller.js.map +1 -1
  137. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
  138. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +7 -0
  139. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  140. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts +4 -0
  141. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts.map +1 -1
  142. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.js.map +1 -1
  143. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +9 -1
  144. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  145. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +103 -23
  146. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  147. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.d.ts.map +1 -1
  148. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js +41 -0
  149. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js.map +1 -1
  150. package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.js +4 -4
  151. package/packages/pi-coding-agent/dist/modes/interactive/theme/themes.js.map +1 -1
  152. package/packages/pi-coding-agent/package.json +1 -1
  153. package/packages/pi-coding-agent/src/core/keybindings.ts +4 -1
  154. package/packages/pi-coding-agent/src/core/sdk.ts +4 -2
  155. package/packages/pi-coding-agent/src/core/settings-manager.collapse-tool-calls.test.ts +46 -0
  156. package/packages/pi-coding-agent/src/core/settings-manager.fast-mode.test.ts +46 -0
  157. package/packages/pi-coding-agent/src/core/settings-manager.ts +36 -0
  158. package/packages/pi-coding-agent/src/core/slash-commands.ts +1 -0
  159. package/packages/pi-coding-agent/src/core/system-prompt.ts +6 -1
  160. package/packages/pi-coding-agent/src/core/tool-priority.test.ts +30 -0
  161. package/packages/pi-coding-agent/src/core/tool-priority.ts +17 -0
  162. package/packages/pi-coding-agent/src/core/tools/edit-diff.test.ts +20 -0
  163. package/packages/pi-coding-agent/src/core/tools/edit-diff.ts +26 -0
  164. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-summary-line.test.ts +41 -0
  165. package/packages/pi-coding-agent/src/modes/interactive/components/btw-overlay.test.ts +172 -0
  166. package/packages/pi-coding-agent/src/modes/interactive/components/btw-overlay.ts +402 -0
  167. package/packages/pi-coding-agent/src/modes/interactive/components/diff.ts +105 -28
  168. package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +21 -6
  169. package/packages/pi-coding-agent/src/modes/interactive/components/settings-selector.ts +63 -6
  170. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +1262 -1138
  171. package/packages/pi-coding-agent/src/modes/interactive/components/tool-summary-line.ts +120 -0
  172. package/packages/pi-coding-agent/src/modes/interactive/controllers/__tests__/chat-controller.collapsed-tool-summary.test.ts +396 -0
  173. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +530 -398
  174. package/packages/pi-coding-agent/src/modes/interactive/controllers/extension-ui-controller.ts +1 -1
  175. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +7 -0
  176. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-state.ts +4 -0
  177. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +109 -23
  178. package/packages/pi-coding-agent/src/modes/interactive/slash-command-handlers.ts +60 -1
  179. package/packages/pi-coding-agent/src/modes/interactive/theme/themes.ts +4 -4
  180. package/packages/pi-tui/dist/components/editor.js +3 -3
  181. package/packages/pi-tui/dist/components/editor.js.map +1 -1
  182. package/packages/pi-tui/src/components/editor.ts +3 -3
  183. package/pkg/dist/modes/interactive/theme/themes.js +4 -4
  184. package/pkg/dist/modes/interactive/theme/themes.js.map +1 -1
  185. package/pkg/package.json +1 -1
  186. package/src/resources/extensions/browser-tools/tools/codegen.ts +5 -5
  187. package/src/resources/extensions/browser-tools/tools/navigation.ts +118 -196
  188. package/src/resources/extensions/browser-tools/tools/network-mock.ts +114 -205
  189. package/src/resources/extensions/browser-tools/tools/pages.ts +183 -237
  190. package/src/resources/extensions/browser-tools/tools/refs.ts +193 -507
  191. package/src/resources/extensions/browser-tools/tools/session.ts +182 -321
  192. package/src/resources/extensions/browser-tools/tools/state-persistence.ts +94 -172
  193. package/src/resources/extensions/browser-tools/utils.ts +1 -1
  194. package/src/resources/extensions/cache-timer/index.ts +3 -2
  195. package/src/resources/extensions/slash-commands/extension-manifest.json +2 -2
  196. package/src/resources/extensions/slash-commands/fast.ts +89 -0
  197. package/src/resources/extensions/slash-commands/index.ts +2 -0
  198. package/src/resources/extensions/slash-commands/plan.ts +42 -12
  199. package/src/resources/extensions/subagent/background-job-manager.ts +28 -0
  200. package/src/resources/extensions/subagent/in-process-runner.ts +534 -0
  201. package/src/resources/extensions/subagent/index.ts +489 -799
  202. package/src/resources/extensions/subagent/legacy-runner.ts +607 -0
  203. package/src/resources/extensions/voice/index.ts +308 -238
  204. package/src/resources/extensions/voice/push-to-talk.ts +42 -0
  205. package/src/resources/extensions/voice/tests/push-to-talk.test.ts +109 -0
@@ -0,0 +1,314 @@
1
+ import { streamSimple } from "@gsd/pi-ai";
2
+ import { Input, Markdown, parseKey, truncateToWidth, visibleWidth, } from "@gsd/pi-tui";
3
+ import { theme } from "../theme/theme.js";
4
+ const EMPTY_USAGE = {
5
+ input: 0,
6
+ output: 0,
7
+ cacheRead: 0,
8
+ cacheWrite: 0,
9
+ totalTokens: 0,
10
+ cost: {
11
+ input: 0,
12
+ output: 0,
13
+ cacheRead: 0,
14
+ cacheWrite: 0,
15
+ total: 0,
16
+ },
17
+ };
18
+ function clamp(value, min, max) {
19
+ return Math.max(min, Math.min(max, value));
20
+ }
21
+ function fitWidth(text, width) {
22
+ if (width <= 0)
23
+ return "";
24
+ const visible = visibleWidth(text);
25
+ if (visible === width)
26
+ return text;
27
+ if (visible < width)
28
+ return text + " ".repeat(width - visible);
29
+ return truncateToWidth(text, width, "");
30
+ }
31
+ function formatAssistantError(message) {
32
+ if (message.errorMessage?.trim()) {
33
+ return message.errorMessage.trim();
34
+ }
35
+ const text = message.content
36
+ .map((part) => (part.type === "text" ? part.text.trim() : ""))
37
+ .filter(Boolean)
38
+ .join("\n")
39
+ .trim();
40
+ return text || `Request failed (${message.stopReason})`;
41
+ }
42
+ export class BtwOverlayComponent {
43
+ get focused() {
44
+ return this._focused;
45
+ }
46
+ set focused(value) {
47
+ this._focused = value;
48
+ this.input.focused = value;
49
+ }
50
+ constructor(question, model, systemPrompt, messages, markdownTheme, ui, onDismiss, requestRender, streamer = streamSimple, streamOptions = {}) {
51
+ this._focused = false;
52
+ this.input = new Input();
53
+ this.btwHistory = [];
54
+ this.turns = [];
55
+ this.disposed = false;
56
+ this.isStreaming = false;
57
+ this.scrollOffset = 0;
58
+ this.followTail = true;
59
+ this.lastWidth = 0;
60
+ this.model = model;
61
+ this.systemPrompt = systemPrompt;
62
+ this.baseMessages = messages;
63
+ this.markdownTheme = markdownTheme;
64
+ this.ui = ui;
65
+ this.onDismiss = onDismiss;
66
+ this.requestRender = requestRender;
67
+ this.streamFn = streamer;
68
+ this.streamOptions = streamOptions;
69
+ this.input.placeholder = "Ask follow-up...";
70
+ this.input.onEscape = () => this.dismiss();
71
+ this.input.onSubmit = (value) => {
72
+ void this.submitTurn(value);
73
+ };
74
+ void this.submitTurn(question);
75
+ }
76
+ handleInput(data) {
77
+ const key = parseKey(data);
78
+ switch (key) {
79
+ case "up":
80
+ this.scrollBy(-1);
81
+ return;
82
+ case "down":
83
+ this.scrollBy(1);
84
+ return;
85
+ case "pageUp":
86
+ this.scrollBy(-(this.getBodyHeight() - 1));
87
+ return;
88
+ case "pageDown":
89
+ this.scrollBy(this.getBodyHeight() - 1);
90
+ return;
91
+ }
92
+ this.input.handleInput(data);
93
+ if (!this.disposed) {
94
+ this.requestRender();
95
+ }
96
+ }
97
+ invalidate() {
98
+ // No cached subtree state to invalidate.
99
+ }
100
+ dispose() {
101
+ if (this.disposed)
102
+ return;
103
+ this.disposed = true;
104
+ this.currentAbortController?.abort();
105
+ }
106
+ render(width) {
107
+ this.lastWidth = width;
108
+ const totalHeight = this.getTotalHeight();
109
+ const bodyHeight = Math.max(1, totalHeight - 4);
110
+ const contentWidth = Math.max(1, width - 4);
111
+ const bodyLines = this.getBodyLines(contentWidth);
112
+ const maxOffset = Math.max(0, bodyLines.length - bodyHeight);
113
+ if (this.followTail) {
114
+ this.scrollOffset = maxOffset;
115
+ }
116
+ else {
117
+ this.scrollOffset = clamp(this.scrollOffset, 0, maxOffset);
118
+ }
119
+ const visibleBody = bodyLines.slice(this.scrollOffset, this.scrollOffset + bodyHeight);
120
+ while (visibleBody.length < bodyHeight) {
121
+ visibleBody.push("");
122
+ }
123
+ const inputLines = this.input.render(contentWidth);
124
+ const inputLine = inputLines[0] ?? "";
125
+ return [
126
+ this.renderTopBorder(width),
127
+ ...visibleBody.map((line) => this.renderBodyLine(line, width)),
128
+ this.renderBodyLine(inputLine, width),
129
+ this.renderFooterLine(width, bodyLines.length > bodyHeight),
130
+ this.renderBottomBorder(width),
131
+ ];
132
+ }
133
+ dismiss() {
134
+ this.dispose();
135
+ this.onDismiss();
136
+ }
137
+ async submitTurn(rawQuestion) {
138
+ const question = rawQuestion.trim();
139
+ if (!question || this.disposed || this.isStreaming) {
140
+ return;
141
+ }
142
+ const userMessage = this.createUserMessage(question);
143
+ const context = {
144
+ systemPrompt: this.systemPrompt,
145
+ messages: [...this.baseMessages, ...this.btwHistory, userMessage],
146
+ };
147
+ this.btwHistory.push(userMessage);
148
+ this.turns.push({ role: "user", text: question });
149
+ const assistantTurn = { role: "assistant", text: "", isStreaming: true };
150
+ this.turns.push(assistantTurn);
151
+ this.input.setValue("");
152
+ this.isStreaming = true;
153
+ this.followTail = true;
154
+ this.requestRender();
155
+ const abortController = new AbortController();
156
+ this.currentAbortController = abortController;
157
+ try {
158
+ const eventStream = this.streamFn(this.model, context, {
159
+ signal: abortController.signal,
160
+ apiKey: this.streamOptions.apiKey,
161
+ sessionId: this.streamOptions.sessionId,
162
+ });
163
+ for await (const event of eventStream) {
164
+ if (this.disposed || this.currentAbortController !== abortController) {
165
+ break;
166
+ }
167
+ if (event.type === "text_delta") {
168
+ assistantTurn.text += event.delta;
169
+ this.requestRender();
170
+ continue;
171
+ }
172
+ if (event.type === "done") {
173
+ this.finishAssistantTurn(assistantTurn, "stop");
174
+ this.requestRender();
175
+ break;
176
+ }
177
+ if (event.type === "error") {
178
+ if (event.reason === "aborted") {
179
+ continue;
180
+ }
181
+ assistantTurn.isError = true;
182
+ if (!assistantTurn.text.trim()) {
183
+ assistantTurn.text = formatAssistantError(event.error);
184
+ }
185
+ this.finishAssistantTurn(assistantTurn, "error");
186
+ this.requestRender();
187
+ break;
188
+ }
189
+ }
190
+ }
191
+ catch (error) {
192
+ if (this.disposed || abortController.signal.aborted || this.currentAbortController !== abortController) {
193
+ return;
194
+ }
195
+ assistantTurn.isError = true;
196
+ assistantTurn.text = error instanceof Error ? error.message : "Unknown error";
197
+ this.finishAssistantTurn(assistantTurn, "error");
198
+ this.requestRender();
199
+ }
200
+ finally {
201
+ if (this.currentAbortController === abortController) {
202
+ this.currentAbortController = undefined;
203
+ this.isStreaming = false;
204
+ assistantTurn.isStreaming = false;
205
+ this.requestRender();
206
+ }
207
+ }
208
+ }
209
+ finishAssistantTurn(turn, stopReason) {
210
+ turn.isStreaming = false;
211
+ if (!turn.text.trim()) {
212
+ return;
213
+ }
214
+ this.btwHistory.push({
215
+ role: "assistant",
216
+ content: [{ type: "text", text: turn.text }],
217
+ api: this.model.api,
218
+ provider: this.model.provider,
219
+ model: this.model.id,
220
+ usage: EMPTY_USAGE,
221
+ stopReason,
222
+ errorMessage: turn.isError ? turn.text : undefined,
223
+ timestamp: Date.now(),
224
+ });
225
+ }
226
+ createUserMessage(question) {
227
+ return {
228
+ role: "user",
229
+ content: [{ type: "text", text: question }],
230
+ timestamp: Date.now(),
231
+ };
232
+ }
233
+ getTotalHeight() {
234
+ return Math.max(6, Math.floor(this.ui.terminal.rows * 0.7));
235
+ }
236
+ getBodyHeight() {
237
+ return Math.max(1, this.getTotalHeight() - 4);
238
+ }
239
+ scrollBy(delta) {
240
+ const bodyHeight = this.getBodyHeight();
241
+ const bodyLines = this.getBodyLines(Math.max(1, this.lastWidth - 4));
242
+ const maxOffset = Math.max(0, bodyLines.length - bodyHeight);
243
+ this.scrollOffset = clamp(this.scrollOffset + delta, 0, maxOffset);
244
+ this.followTail = this.scrollOffset >= maxOffset;
245
+ this.requestRender();
246
+ }
247
+ getBodyLines(contentWidth) {
248
+ const lines = [];
249
+ for (const [index, turn] of this.turns.entries()) {
250
+ if (index > 0) {
251
+ lines.push("");
252
+ }
253
+ lines.push(turn.role === "user" ? theme.fg("accent", "You") : theme.fg("muted", "btw"));
254
+ lines.push(...this.renderTurnContent(turn, Math.max(1, contentWidth - 2)).map((line) => ` ${line}`));
255
+ }
256
+ if (lines.length === 0) {
257
+ lines.push(theme.fg("dim", "Awaiting response…"));
258
+ }
259
+ return lines;
260
+ }
261
+ renderTurnContent(turn, contentWidth) {
262
+ if (!turn.text.trim()) {
263
+ return [theme.fg("dim", turn.isStreaming ? "Awaiting response…" : "No response received.")];
264
+ }
265
+ if (turn.isError) {
266
+ return [theme.fg("error", `Error: ${turn.text}`)];
267
+ }
268
+ const markdown = new Markdown(turn.text, 0, 0, this.markdownTheme, {
269
+ color: (text) => theme.fg("text", text),
270
+ });
271
+ return markdown.render(contentWidth);
272
+ }
273
+ renderTopBorder(width) {
274
+ if (width <= 2) {
275
+ return "┌┐".slice(0, width);
276
+ }
277
+ const innerWidth = width - 2;
278
+ const title = theme.fg("accent", " btw ");
279
+ const titleWidth = visibleWidth(title);
280
+ if (innerWidth <= titleWidth) {
281
+ return fitWidth(`┌${truncateToWidth(title, innerWidth, "")}┐`, width);
282
+ }
283
+ const dashCount = innerWidth - titleWidth;
284
+ const leftDashes = Math.floor(dashCount / 2);
285
+ const rightDashes = dashCount - leftDashes;
286
+ const border = theme.fg("border", "─");
287
+ return `┌${border.repeat(leftDashes)}${title}${border.repeat(rightDashes)}┐`;
288
+ }
289
+ renderBottomBorder(width) {
290
+ if (width <= 2) {
291
+ return "└┘".slice(0, width);
292
+ }
293
+ return `└${theme.fg("border", "─").repeat(width - 2)}┘`;
294
+ }
295
+ renderBodyLine(text, width) {
296
+ const innerWidth = Math.max(1, width - 4);
297
+ const fitted = fitWidth(truncateToWidth(text, innerWidth, ""), innerWidth);
298
+ return fitWidth(`│ ${fitted} │`, width);
299
+ }
300
+ renderFooterLine(width, canScroll) {
301
+ const innerWidth = Math.max(1, width - 4);
302
+ const parts = ["Enter send", "Esc dismiss"];
303
+ if (canScroll) {
304
+ parts.push("↑↓/PgUp/PgDn scroll");
305
+ }
306
+ const status = this.isStreaming
307
+ ? theme.fg("muted", "Streaming...")
308
+ : theme.fg("success", "Ready");
309
+ const footer = `${theme.fg("muted", parts.join(" • "))} ${status}`;
310
+ const fitted = fitWidth(truncateToWidth(footer, innerWidth, ""), innerWidth);
311
+ return fitWidth(`│ ${fitted} │`, width);
312
+ }
313
+ }
314
+ //# sourceMappingURL=btw-overlay.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"btw-overlay.js","sourceRoot":"","sources":["../../../../src/modes/interactive/components/btw-overlay.ts"],"names":[],"mappings":"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","sourcesContent":["import type { AssistantMessage, Context, Message, Model, SimpleStreamOptions, StopReason, Usage } from \"@gsd/pi-ai\";\nimport { streamSimple } from \"@gsd/pi-ai\";\nimport {\n Input,\n Markdown,\n type MarkdownTheme,\n parseKey,\n truncateToWidth,\n type Component,\n type Focusable,\n type TUI,\n visibleWidth,\n} from \"@gsd/pi-tui\";\nimport { theme } from \"../theme/theme.js\";\n\nconst EMPTY_USAGE: Usage = {\n input: 0,\n output: 0,\n cacheRead: 0,\n cacheWrite: 0,\n totalTokens: 0,\n cost: {\n input: 0,\n output: 0,\n cacheRead: 0,\n cacheWrite: 0,\n total: 0,\n },\n};\n\ninterface BtwTurn {\n role: \"user\" | \"assistant\";\n text: string;\n isError?: boolean;\n isStreaming?: boolean;\n}\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.max(min, Math.min(max, value));\n}\n\nfunction fitWidth(text: string, width: number): string {\n if (width <= 0) return \"\";\n const visible = visibleWidth(text);\n if (visible === width) return text;\n if (visible < width) return text + \" \".repeat(width - visible);\n return truncateToWidth(text, width, \"\");\n}\n\nfunction formatAssistantError(message: AssistantMessage): string {\n if (message.errorMessage?.trim()) {\n return message.errorMessage.trim();\n }\n\n const text = message.content\n .map((part) => (part.type === \"text\" ? part.text.trim() : \"\"))\n .filter(Boolean)\n .join(\"\\n\")\n .trim();\n\n return text || `Request failed (${message.stopReason})`;\n}\n\nexport class BtwOverlayComponent implements Component, Focusable {\n private _focused = false;\n\n private readonly markdownTheme: MarkdownTheme;\n private readonly model: Model<any>;\n private readonly systemPrompt: string | undefined;\n private readonly baseMessages: Message[];\n private readonly ui: TUI;\n private readonly input = new Input();\n private readonly onDismiss: () => void;\n private readonly requestRender: () => void;\n private readonly streamFn: typeof streamSimple;\n private readonly streamOptions: Pick<SimpleStreamOptions, \"apiKey\" | \"sessionId\">;\n\n private readonly btwHistory: Message[] = [];\n private readonly turns: BtwTurn[] = [];\n private currentAbortController: AbortController | undefined;\n private disposed = false;\n private isStreaming = false;\n private scrollOffset = 0;\n private followTail = true;\n private lastWidth = 0;\n\n get focused(): boolean {\n return this._focused;\n }\n\n set focused(value: boolean) {\n this._focused = value;\n this.input.focused = value;\n }\n\n constructor(\n question: string,\n model: Model<any>,\n systemPrompt: string | undefined,\n messages: Message[],\n markdownTheme: MarkdownTheme,\n ui: TUI,\n onDismiss: () => void,\n requestRender: () => void,\n streamer: typeof streamSimple = streamSimple,\n streamOptions: Pick<SimpleStreamOptions, \"apiKey\" | \"sessionId\"> = {},\n ) {\n this.model = model;\n this.systemPrompt = systemPrompt;\n this.baseMessages = messages;\n this.markdownTheme = markdownTheme;\n this.ui = ui;\n this.onDismiss = onDismiss;\n this.requestRender = requestRender;\n this.streamFn = streamer;\n this.streamOptions = streamOptions;\n\n this.input.placeholder = \"Ask follow-up...\";\n this.input.onEscape = () => this.dismiss();\n this.input.onSubmit = (value) => {\n void this.submitTurn(value);\n };\n\n void this.submitTurn(question);\n }\n\n handleInput(data: string): void {\n const key = parseKey(data);\n switch (key) {\n case \"up\":\n this.scrollBy(-1);\n return;\n case \"down\":\n this.scrollBy(1);\n return;\n case \"pageUp\":\n this.scrollBy(-(this.getBodyHeight() - 1));\n return;\n case \"pageDown\":\n this.scrollBy(this.getBodyHeight() - 1);\n return;\n }\n\n this.input.handleInput(data);\n if (!this.disposed) {\n this.requestRender();\n }\n }\n\n invalidate(): void {\n // No cached subtree state to invalidate.\n }\n\n dispose(): void {\n if (this.disposed) return;\n this.disposed = true;\n this.currentAbortController?.abort();\n }\n\n render(width: number): string[] {\n this.lastWidth = width;\n\n const totalHeight = this.getTotalHeight();\n const bodyHeight = Math.max(1, totalHeight - 4);\n const contentWidth = Math.max(1, width - 4);\n const bodyLines = this.getBodyLines(contentWidth);\n const maxOffset = Math.max(0, bodyLines.length - bodyHeight);\n if (this.followTail) {\n this.scrollOffset = maxOffset;\n } else {\n this.scrollOffset = clamp(this.scrollOffset, 0, maxOffset);\n }\n\n const visibleBody = bodyLines.slice(this.scrollOffset, this.scrollOffset + bodyHeight);\n while (visibleBody.length < bodyHeight) {\n visibleBody.push(\"\");\n }\n\n const inputLines = this.input.render(contentWidth);\n const inputLine = inputLines[0] ?? \"\";\n\n return [\n this.renderTopBorder(width),\n ...visibleBody.map((line) => this.renderBodyLine(line, width)),\n this.renderBodyLine(inputLine, width),\n this.renderFooterLine(width, bodyLines.length > bodyHeight),\n this.renderBottomBorder(width),\n ];\n }\n\n private dismiss(): void {\n this.dispose();\n this.onDismiss();\n }\n\n private async submitTurn(rawQuestion: string): Promise<void> {\n const question = rawQuestion.trim();\n if (!question || this.disposed || this.isStreaming) {\n return;\n }\n\n const userMessage = this.createUserMessage(question);\n const context: Context = {\n systemPrompt: this.systemPrompt,\n messages: [...this.baseMessages, ...this.btwHistory, userMessage],\n };\n\n this.btwHistory.push(userMessage);\n this.turns.push({ role: \"user\", text: question });\n const assistantTurn: BtwTurn = { role: \"assistant\", text: \"\", isStreaming: true };\n this.turns.push(assistantTurn);\n this.input.setValue(\"\");\n this.isStreaming = true;\n this.followTail = true;\n this.requestRender();\n\n const abortController = new AbortController();\n this.currentAbortController = abortController;\n\n try {\n const eventStream = this.streamFn(this.model, context, {\n signal: abortController.signal,\n apiKey: this.streamOptions.apiKey,\n sessionId: this.streamOptions.sessionId,\n });\n\n for await (const event of eventStream) {\n if (this.disposed || this.currentAbortController !== abortController) {\n break;\n }\n\n if (event.type === \"text_delta\") {\n assistantTurn.text += event.delta;\n this.requestRender();\n continue;\n }\n\n if (event.type === \"done\") {\n this.finishAssistantTurn(assistantTurn, \"stop\");\n this.requestRender();\n break;\n }\n\n if (event.type === \"error\") {\n if (event.reason === \"aborted\") {\n continue;\n }\n\n assistantTurn.isError = true;\n if (!assistantTurn.text.trim()) {\n assistantTurn.text = formatAssistantError(event.error);\n }\n this.finishAssistantTurn(assistantTurn, \"error\");\n this.requestRender();\n break;\n }\n }\n } catch (error: unknown) {\n if (this.disposed || abortController.signal.aborted || this.currentAbortController !== abortController) {\n return;\n }\n\n assistantTurn.isError = true;\n assistantTurn.text = error instanceof Error ? error.message : \"Unknown error\";\n this.finishAssistantTurn(assistantTurn, \"error\");\n this.requestRender();\n } finally {\n if (this.currentAbortController === abortController) {\n this.currentAbortController = undefined;\n this.isStreaming = false;\n assistantTurn.isStreaming = false;\n this.requestRender();\n }\n }\n }\n\n private finishAssistantTurn(turn: BtwTurn, stopReason: StopReason): void {\n turn.isStreaming = false;\n if (!turn.text.trim()) {\n return;\n }\n\n this.btwHistory.push({\n role: \"assistant\",\n content: [{ type: \"text\", text: turn.text }],\n api: this.model.api,\n provider: this.model.provider,\n model: this.model.id,\n usage: EMPTY_USAGE,\n stopReason,\n errorMessage: turn.isError ? turn.text : undefined,\n timestamp: Date.now(),\n });\n }\n\n private createUserMessage(question: string): Message {\n return {\n role: \"user\",\n content: [{ type: \"text\", text: question }],\n timestamp: Date.now(),\n };\n }\n\n private getTotalHeight(): number {\n return Math.max(6, Math.floor(this.ui.terminal.rows * 0.7));\n }\n\n private getBodyHeight(): number {\n return Math.max(1, this.getTotalHeight() - 4);\n }\n\n private scrollBy(delta: number): void {\n const bodyHeight = this.getBodyHeight();\n const bodyLines = this.getBodyLines(Math.max(1, this.lastWidth - 4));\n const maxOffset = Math.max(0, bodyLines.length - bodyHeight);\n this.scrollOffset = clamp(this.scrollOffset + delta, 0, maxOffset);\n this.followTail = this.scrollOffset >= maxOffset;\n this.requestRender();\n }\n\n private getBodyLines(contentWidth: number): string[] {\n const lines: string[] = [];\n\n for (const [index, turn] of this.turns.entries()) {\n if (index > 0) {\n lines.push(\"\");\n }\n lines.push(turn.role === \"user\" ? theme.fg(\"accent\", \"You\") : theme.fg(\"muted\", \"btw\"));\n lines.push(...this.renderTurnContent(turn, Math.max(1, contentWidth - 2)).map((line) => ` ${line}`));\n }\n\n if (lines.length === 0) {\n lines.push(theme.fg(\"dim\", \"Awaiting response…\"));\n }\n\n return lines;\n }\n\n private renderTurnContent(turn: BtwTurn, contentWidth: number): string[] {\n if (!turn.text.trim()) {\n return [theme.fg(\"dim\", turn.isStreaming ? \"Awaiting response…\" : \"No response received.\")];\n }\n\n if (turn.isError) {\n return [theme.fg(\"error\", `Error: ${turn.text}`)];\n }\n\n const markdown = new Markdown(turn.text, 0, 0, this.markdownTheme, {\n color: (text: string) => theme.fg(\"text\", text),\n });\n return markdown.render(contentWidth);\n }\n\n private renderTopBorder(width: number): string {\n if (width <= 2) {\n return \"┌┐\".slice(0, width);\n }\n\n const innerWidth = width - 2;\n const title = theme.fg(\"accent\", \" btw \");\n const titleWidth = visibleWidth(title);\n\n if (innerWidth <= titleWidth) {\n return fitWidth(`┌${truncateToWidth(title, innerWidth, \"\")}┐`, width);\n }\n\n const dashCount = innerWidth - titleWidth;\n const leftDashes = Math.floor(dashCount / 2);\n const rightDashes = dashCount - leftDashes;\n const border = theme.fg(\"border\", \"─\");\n return `┌${border.repeat(leftDashes)}${title}${border.repeat(rightDashes)}┐`;\n }\n\n private renderBottomBorder(width: number): string {\n if (width <= 2) {\n return \"└┘\".slice(0, width);\n }\n return `└${theme.fg(\"border\", \"─\").repeat(width - 2)}┘`;\n }\n\n private renderBodyLine(text: string, width: number): string {\n const innerWidth = Math.max(1, width - 4);\n const fitted = fitWidth(truncateToWidth(text, innerWidth, \"\"), innerWidth);\n return fitWidth(`│ ${fitted} │`, width);\n }\n\n private renderFooterLine(width: number, canScroll: boolean): string {\n const innerWidth = Math.max(1, width - 4);\n const parts: string[] = [\"Enter send\", \"Esc dismiss\"];\n if (canScroll) {\n parts.push(\"↑↓/PgUp/PgDn scroll\");\n }\n\n const status = this.isStreaming\n ? theme.fg(\"muted\", \"Streaming...\")\n : theme.fg(\"success\", \"Ready\");\n\n const footer = `${theme.fg(\"muted\", parts.join(\" • \"))} ${status}`;\n const fitted = fitWidth(truncateToWidth(footer, innerWidth, \"\"), innerWidth);\n return fitWidth(`│ ${fitted} │`, width);\n }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=btw-overlay.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"btw-overlay.test.d.ts","sourceRoot":"","sources":["../../../../src/modes/interactive/components/btw-overlay.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,122 @@
1
+ import test from "node:test";
2
+ import assert from "node:assert/strict";
3
+ import stripAnsi from "strip-ansi";
4
+ import { BtwOverlayComponent } from "./btw-overlay.js";
5
+ import { getMarkdownTheme, initTheme } from "../theme/theme.js";
6
+ initTheme();
7
+ function makeUi(rows = 40) {
8
+ return { terminal: { rows } };
9
+ }
10
+ function makeModel() {
11
+ return {
12
+ id: "test-model",
13
+ name: "Test Model",
14
+ api: "openai-completions",
15
+ provider: "test",
16
+ baseUrl: "",
17
+ reasoning: false,
18
+ input: ["text"],
19
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
20
+ contextWindow: 4096,
21
+ maxTokens: 1024,
22
+ };
23
+ }
24
+ function makeMessages() {
25
+ return [
26
+ {
27
+ role: "user",
28
+ content: [{ type: "text", text: "hello" }],
29
+ timestamp: Date.now(),
30
+ },
31
+ ];
32
+ }
33
+ function renderText(component) {
34
+ return stripAnsi(component.render(80).join("\n"));
35
+ }
36
+ async function flushTurns(count = 3) {
37
+ for (let i = 0; i < count; i++) {
38
+ await Promise.resolve();
39
+ await new Promise((resolve) => setTimeout(resolve, 0));
40
+ }
41
+ }
42
+ test("BtwOverlayComponent renders initial question and placeholder before response", async () => {
43
+ const component = new BtwOverlayComponent("What is btw?", makeModel(), "system", makeMessages(), getMarkdownTheme(), makeUi(), () => undefined, () => undefined, () => ({
44
+ async *[Symbol.asyncIterator]() {
45
+ await new Promise(() => undefined);
46
+ },
47
+ }));
48
+ await flushTurns();
49
+ const text = renderText(component);
50
+ assert.match(text, /btw/);
51
+ assert.match(text, /You/);
52
+ assert.match(text, /What is btw\?/);
53
+ assert.match(text, /Awaiting response/);
54
+ assert.match(text, /Ask follow-up/);
55
+ component.dispose();
56
+ });
57
+ test("BtwOverlayComponent supports follow-up turns and reuses overlay-local history", async () => {
58
+ const calls = [];
59
+ const responses = [
60
+ [
61
+ { type: "text_delta", delta: "First answer" },
62
+ { type: "done" },
63
+ ],
64
+ [
65
+ { type: "text_delta", delta: "Second answer" },
66
+ { type: "done" },
67
+ ],
68
+ ];
69
+ const streamFn = ((_, context, options) => {
70
+ calls.push({ context, options });
71
+ const response = responses[calls.length - 1] ?? [];
72
+ return {
73
+ async *[Symbol.asyncIterator]() {
74
+ for (const event of response) {
75
+ yield event;
76
+ }
77
+ },
78
+ };
79
+ });
80
+ const component = new BtwOverlayComponent("What is btw?", makeModel(), "system", makeMessages(), getMarkdownTheme(), makeUi(), () => undefined, () => undefined, streamFn, { apiKey: "test-key", sessionId: "session-1" });
81
+ await flushTurns();
82
+ for (const ch of "next") {
83
+ component.handleInput(ch);
84
+ }
85
+ component.handleInput("\n");
86
+ await flushTurns();
87
+ assert.equal(calls.length, 2);
88
+ assert.equal(calls[0]?.options?.apiKey, "test-key");
89
+ assert.equal(calls[1]?.options?.sessionId, "session-1");
90
+ assert.equal(calls[0]?.context.messages.length, 2);
91
+ assert.equal(calls[1]?.context.messages.length, 4);
92
+ assert.equal(calls[1]?.context.messages[1]?.role, "user");
93
+ assert.equal(calls[1]?.context.messages[2]?.role, "assistant");
94
+ assert.deepEqual(calls[1]?.context.messages[3]?.content, [{ type: "text", text: "next" }]);
95
+ const text = renderText(component);
96
+ assert.match(text, /First answer/);
97
+ assert.match(text, /Second answer/);
98
+ component.dispose();
99
+ });
100
+ test("BtwOverlayComponent aborts active stream on Escape", async () => {
101
+ let capturedSignal;
102
+ let dismissed = false;
103
+ const streamFn = ((_model, _context, options) => {
104
+ capturedSignal = options?.signal;
105
+ return {
106
+ async *[Symbol.asyncIterator]() {
107
+ await new Promise((resolve) => {
108
+ capturedSignal?.addEventListener("abort", () => resolve(), { once: true });
109
+ });
110
+ },
111
+ };
112
+ });
113
+ const component = new BtwOverlayComponent("What is btw?", makeModel(), "system", makeMessages(), getMarkdownTheme(), makeUi(), () => {
114
+ dismissed = true;
115
+ }, () => undefined, streamFn);
116
+ await flushTurns();
117
+ component.handleInput("\u001b");
118
+ await flushTurns();
119
+ assert.equal(dismissed, true);
120
+ assert.equal(capturedSignal?.aborted, true);
121
+ });
122
+ //# sourceMappingURL=btw-overlay.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"btw-overlay.test.js","sourceRoot":"","sources":["../../../../src/modes/interactive/components/btw-overlay.test.ts"],"names":[],"mappings":"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","sourcesContent":["import test from \"node:test\";\nimport assert from \"node:assert/strict\";\nimport stripAnsi from \"strip-ansi\";\nimport type { Context, Model, SimpleStreamOptions, Message } from \"@gsd/pi-ai\";\nimport type { TUI } from \"@gsd/pi-tui\";\nimport { BtwOverlayComponent } from \"./btw-overlay.js\";\nimport { getMarkdownTheme, initTheme } from \"../theme/theme.js\";\n\ninitTheme();\n\nfunction makeUi(rows = 40): TUI {\n return { terminal: { rows } } as unknown as TUI;\n}\n\nfunction makeModel(): Model<any> {\n return {\n id: \"test-model\",\n name: \"Test Model\",\n api: \"openai-completions\",\n provider: \"test\",\n baseUrl: \"\",\n reasoning: false,\n input: [\"text\"],\n cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },\n contextWindow: 4096,\n maxTokens: 1024,\n } as unknown as Model<any>;\n}\n\nfunction makeMessages(): Message[] {\n return [\n {\n role: \"user\",\n content: [{ type: \"text\", text: \"hello\" }],\n timestamp: Date.now(),\n },\n ];\n}\n\nfunction renderText(component: BtwOverlayComponent): string {\n return stripAnsi(component.render(80).join(\"\\n\"));\n}\n\nasync function flushTurns(count = 3): Promise<void> {\n for (let i = 0; i < count; i++) {\n await Promise.resolve();\n await new Promise((resolve) => setTimeout(resolve, 0));\n }\n}\n\ntest(\"BtwOverlayComponent renders initial question and placeholder before response\", async () => {\n const component = new BtwOverlayComponent(\n \"What is btw?\",\n makeModel(),\n \"system\",\n makeMessages(),\n getMarkdownTheme(),\n makeUi(),\n () => undefined,\n () => undefined,\n () => ({\n async *[Symbol.asyncIterator]() {\n await new Promise(() => undefined);\n },\n }) as any,\n );\n\n await flushTurns();\n const text = renderText(component);\n assert.match(text, /btw/);\n assert.match(text, /You/);\n assert.match(text, /What is btw\\?/);\n assert.match(text, /Awaiting response/);\n assert.match(text, /Ask follow-up/);\n component.dispose();\n});\n\ntest(\"BtwOverlayComponent supports follow-up turns and reuses overlay-local history\", async () => {\n const calls: Array<{ context: Context; options?: SimpleStreamOptions }> = [];\n const responses = [\n [\n { type: \"text_delta\", delta: \"First answer\" },\n { type: \"done\" },\n ],\n [\n { type: \"text_delta\", delta: \"Second answer\" },\n { type: \"done\" },\n ],\n ];\n\n const streamFn = ((_: Model<any>, context: Context, options?: SimpleStreamOptions) => {\n calls.push({ context, options });\n const response = responses[calls.length - 1] ?? [];\n return {\n async *[Symbol.asyncIterator]() {\n for (const event of response) {\n yield event;\n }\n },\n };\n }) as any;\n\n const component = new BtwOverlayComponent(\n \"What is btw?\",\n makeModel(),\n \"system\",\n makeMessages(),\n getMarkdownTheme(),\n makeUi(),\n () => undefined,\n () => undefined,\n streamFn,\n { apiKey: \"test-key\", sessionId: \"session-1\" },\n );\n\n await flushTurns();\n for (const ch of \"next\") {\n component.handleInput(ch);\n }\n component.handleInput(\"\\n\");\n await flushTurns();\n\n assert.equal(calls.length, 2);\n assert.equal(calls[0]?.options?.apiKey, \"test-key\");\n assert.equal(calls[1]?.options?.sessionId, \"session-1\");\n assert.equal(calls[0]?.context.messages.length, 2);\n assert.equal(calls[1]?.context.messages.length, 4);\n assert.equal(calls[1]?.context.messages[1]?.role, \"user\");\n assert.equal(calls[1]?.context.messages[2]?.role, \"assistant\");\n assert.deepEqual(calls[1]?.context.messages[3]?.content, [{ type: \"text\", text: \"next\" }]);\n\n const text = renderText(component);\n assert.match(text, /First answer/);\n assert.match(text, /Second answer/);\n component.dispose();\n});\n\ntest(\"BtwOverlayComponent aborts active stream on Escape\", async () => {\n let capturedSignal: AbortSignal | undefined;\n let dismissed = false;\n const streamFn = ((_model: any, _context: any, options?: { signal?: AbortSignal }) => {\n capturedSignal = options?.signal;\n return {\n async *[Symbol.asyncIterator]() {\n await new Promise<void>((resolve) => {\n capturedSignal?.addEventListener(\"abort\", () => resolve(), { once: true });\n });\n },\n };\n }) as any;\n\n const component = new BtwOverlayComponent(\n \"What is btw?\",\n makeModel(),\n \"system\",\n makeMessages(),\n getMarkdownTheme(),\n makeUi(),\n () => {\n dismissed = true;\n },\n () => undefined,\n streamFn,\n );\n\n await flushTurns();\n component.handleInput(\"\\u001b\");\n await flushTurns();\n\n assert.equal(dismissed, true);\n assert.equal(capturedSignal?.aborted, true);\n});\n"]}
@@ -2,11 +2,13 @@ export interface RenderDiffOptions {
2
2
  /** File path (unused, kept for API compatibility) */
3
3
  filePath?: string;
4
4
  }
5
+ export declare function renderDiffLines(diffText: string, width: number, _options?: RenderDiffOptions): string[];
5
6
  /**
6
- * Render a diff string with colored lines and intra-line change highlighting.
7
- * - Context lines: dim/gray
8
- * - Removed lines: red, with inverse on changed tokens
9
- * - Added lines: green, with inverse on changed tokens
7
+ * Render a diff string with Claude-like colored lines.
8
+ * - Context lines: muted gray
9
+ * - Removed lines: red text on subtle red background
10
+ * - Added lines: green text on subtle green background
11
+ * - Changed tokens: slightly stronger background + bold
10
12
  */
11
- export declare function renderDiff(diffText: string, _options?: RenderDiffOptions): string;
13
+ export declare function renderDiff(diffText: string, options?: RenderDiffOptions): string;
12
14
  //# sourceMappingURL=diff.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"diff.d.ts","sourceRoot":"","sources":["../../../../src/modes/interactive/components/diff.ts"],"names":[],"mappings":"AAmEA,MAAM,WAAW,iBAAiB;IACjC,qDAAqD;IACrD,QAAQ,CAAC,EAAE,MAAM,CAAC;CAClB;AAED;;;;;GAKG;AACH,wBAAgB,UAAU,CAAC,QAAQ,EAAE,MAAM,EAAE,QAAQ,GAAE,iBAAsB,GAAG,MAAM,CAoErF"}
1
+ {"version":3,"file":"diff.d.ts","sourceRoot":"","sources":["../../../../src/modes/interactive/components/diff.ts"],"names":[],"mappings":"AAoHA,MAAM,WAAW,iBAAiB;IACjC,qDAAqD;IACrD,QAAQ,CAAC,EAAE,MAAM,CAAC;CAClB;AAkBD,wBAAgB,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,GAAE,iBAAsB,GAAG,MAAM,EAAE,CA2E3G;AAED;;;;;;GAMG;AACH,wBAAgB,UAAU,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,GAAE,iBAAsB,GAAG,MAAM,CAEpF"}