@aexol/spectral 0.7.6 → 0.7.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 (57) hide show
  1. package/dist/agent/index.js +16 -140
  2. package/dist/cli.js +25 -220
  3. package/dist/extensions/spectral-vision-fallback.js +188 -0
  4. package/dist/memory/commands/status.js +5 -5
  5. package/dist/memory/commands/view.js +16 -14
  6. package/dist/memory/compaction.js +31 -3
  7. package/dist/memory/prompts.js +5 -5
  8. package/dist/memory/tools/recall-observation.js +2 -2
  9. package/dist/pi/coding-agent/config.js +0 -11
  10. package/dist/pi/coding-agent/core/agent-session.js +3 -17
  11. package/dist/pi/coding-agent/core/extensions/loader.js +0 -6
  12. package/dist/pi/coding-agent/core/extensions/runner.js +7 -1
  13. package/dist/pi/coding-agent/core/keybindings.js +129 -2
  14. package/dist/pi/coding-agent/core/settings-manager.js +20 -0
  15. package/dist/pi/coding-agent/core/tools/bash.js +17 -63
  16. package/dist/pi/coding-agent/core/tools/edit.js +4 -141
  17. package/dist/pi/coding-agent/core/tools/find.js +0 -11
  18. package/dist/pi/coding-agent/core/tools/grep.js +0 -11
  19. package/dist/pi/coding-agent/core/tools/ls.js +0 -11
  20. package/dist/pi/coding-agent/core/tools/read.js +0 -12
  21. package/dist/pi/coding-agent/core/tools/render-utils.js +1 -14
  22. package/dist/pi/coding-agent/core/tools/write.js +2 -97
  23. package/dist/pi/coding-agent/modes/interactive/components/keybinding-hints.js +1 -1
  24. package/dist/pi/coding-agent/modes/interactive/components/visual-truncate.js +6 -12
  25. package/dist/pi/coding-agent/modes/interactive/theme/theme.js +1 -2
  26. package/dist/relay/models-fetch.js +13 -1
  27. package/dist/server/pi-bridge.js +57 -4
  28. package/dist/server/session-stream.js +7 -1
  29. package/package.json +1 -1
  30. package/dist/pi/coding-agent/core/export-html/ansi-to-html.js +0 -248
  31. package/dist/pi/coding-agent/core/export-html/index.js +0 -225
  32. package/dist/pi/coding-agent/core/export-html/tool-renderer.js +0 -107
  33. package/dist/pi/tui/autocomplete.js +0 -631
  34. package/dist/pi/tui/components/box.js +0 -103
  35. package/dist/pi/tui/components/cancellable-loader.js +0 -34
  36. package/dist/pi/tui/components/editor.js +0 -1915
  37. package/dist/pi/tui/components/image.js +0 -88
  38. package/dist/pi/tui/components/input.js +0 -425
  39. package/dist/pi/tui/components/loader.js +0 -68
  40. package/dist/pi/tui/components/markdown.js +0 -633
  41. package/dist/pi/tui/components/select-list.js +0 -158
  42. package/dist/pi/tui/components/settings-list.js +0 -184
  43. package/dist/pi/tui/components/spacer.js +0 -22
  44. package/dist/pi/tui/components/text.js +0 -88
  45. package/dist/pi/tui/components/truncated-text.js +0 -50
  46. package/dist/pi/tui/editor-component.js +0 -1
  47. package/dist/pi/tui/fuzzy.js +0 -109
  48. package/dist/pi/tui/index.js +0 -31
  49. package/dist/pi/tui/keybindings.js +0 -173
  50. package/dist/pi/tui/keys.js +0 -1172
  51. package/dist/pi/tui/kill-ring.js +0 -43
  52. package/dist/pi/tui/stdin-buffer.js +0 -360
  53. package/dist/pi/tui/terminal-image.js +0 -335
  54. package/dist/pi/tui/terminal.js +0 -324
  55. package/dist/pi/tui/tui.js +0 -1076
  56. package/dist/pi/tui/undo-stack.js +0 -24
  57. package/dist/pi/tui/utils.js +0 -1016
@@ -1,88 +0,0 @@
1
- import { allocateImageId, getCapabilities, getCellDimensions, getImageDimensions, imageFallback, renderImage, } from "../terminal-image.js";
2
- export class Image {
3
- base64Data;
4
- mimeType;
5
- dimensions;
6
- theme;
7
- options;
8
- imageId;
9
- cachedLines;
10
- cachedWidth;
11
- constructor(base64Data, mimeType, theme, options = {}, dimensions) {
12
- this.base64Data = base64Data;
13
- this.mimeType = mimeType;
14
- this.theme = theme;
15
- this.options = options;
16
- this.dimensions = dimensions || getImageDimensions(base64Data, mimeType) || { widthPx: 800, heightPx: 600 };
17
- this.imageId = options.imageId;
18
- }
19
- /** Get the Kitty image ID used by this image (if any). */
20
- getImageId() {
21
- return this.imageId;
22
- }
23
- invalidate() {
24
- this.cachedLines = undefined;
25
- this.cachedWidth = undefined;
26
- }
27
- render(width) {
28
- if (this.cachedLines && this.cachedWidth === width) {
29
- return this.cachedLines;
30
- }
31
- const maxWidth = Math.max(1, Math.min(width - 2, this.options.maxWidthCells ?? 60));
32
- const cellDimensions = getCellDimensions();
33
- const defaultMaxHeight = Math.max(1, Math.ceil((maxWidth * cellDimensions.widthPx) / cellDimensions.heightPx));
34
- const maxHeight = this.options.maxHeightCells ?? defaultMaxHeight;
35
- const caps = getCapabilities();
36
- let lines;
37
- if (caps.images) {
38
- if (caps.images === "kitty" && this.imageId === undefined) {
39
- this.imageId = allocateImageId();
40
- }
41
- const result = renderImage(this.base64Data, this.dimensions, {
42
- maxWidthCells: maxWidth,
43
- maxHeightCells: maxHeight,
44
- imageId: this.imageId,
45
- moveCursor: false,
46
- });
47
- if (result) {
48
- // Store the image ID for later cleanup
49
- if (result.imageId) {
50
- this.imageId = result.imageId;
51
- }
52
- if (caps.images === "kitty") {
53
- // For Kitty: C=1 prevents cursor movement.
54
- // Don't need the cursor movement.
55
- lines = [result.sequence];
56
- // Return `rows` lines so TUI accounts for image height.
57
- for (let i = 0; i < result.rows - 1; i++) {
58
- lines.push("");
59
- }
60
- }
61
- else {
62
- // Return `rows` lines so TUI accounts for image height.
63
- // First (rows-1) lines are empty and cleared before the image is drawn.
64
- // Last line: move cursor back up, draw the image, then move back down
65
- // so TUI cursor accounting stays inside the scroll area.
66
- lines = [];
67
- for (let i = 0; i < result.rows - 1; i++) {
68
- lines.push("");
69
- }
70
- const rowOffset = result.rows - 1;
71
- const moveUp = rowOffset > 0 ? `\x1b[${rowOffset}A` : "";
72
- lines.push(moveUp + result.sequence);
73
- }
74
- }
75
- else {
76
- const fallback = imageFallback(this.mimeType, this.dimensions, this.options.filename);
77
- lines = [this.theme.fallbackColor(fallback)];
78
- }
79
- }
80
- else {
81
- const fallback = imageFallback(this.mimeType, this.dimensions, this.options.filename);
82
- lines = [this.theme.fallbackColor(fallback)];
83
- }
84
- this.cachedLines = lines;
85
- this.cachedWidth = width;
86
- return lines;
87
- }
88
- }
@@ -1,425 +0,0 @@
1
- import { getKeybindings } from "../keybindings.js";
2
- import { decodeKittyPrintable } from "../keys.js";
3
- import { KillRing } from "../kill-ring.js";
4
- import { CURSOR_MARKER } from "../tui.js";
5
- import { UndoStack } from "../undo-stack.js";
6
- import { getSegmenter, isPunctuationChar, isWhitespaceChar, sliceByColumn, visibleWidth } from "../utils.js";
7
- const segmenter = getSegmenter();
8
- /**
9
- * Input component - single-line text input with horizontal scrolling
10
- */
11
- export class Input {
12
- value = "";
13
- cursor = 0; // Cursor position in the value
14
- onSubmit;
15
- onEscape;
16
- /** Focusable interface - set by TUI when focus changes */
17
- focused = false;
18
- // Bracketed paste mode buffering
19
- pasteBuffer = "";
20
- isInPaste = false;
21
- // Kill ring for Emacs-style kill/yank operations
22
- killRing = new KillRing();
23
- lastAction = null;
24
- // Undo support
25
- undoStack = new UndoStack();
26
- getValue() {
27
- return this.value;
28
- }
29
- setValue(value) {
30
- this.value = value;
31
- this.cursor = Math.min(this.cursor, value.length);
32
- }
33
- handleInput(data) {
34
- // Handle bracketed paste mode
35
- // Start of paste: \x1b[200~
36
- // End of paste: \x1b[201~
37
- // Check if we're starting a bracketed paste
38
- if (data.includes("\x1b[200~")) {
39
- this.isInPaste = true;
40
- this.pasteBuffer = "";
41
- data = data.replace("\x1b[200~", "");
42
- }
43
- // If we're in a paste, buffer the data
44
- if (this.isInPaste) {
45
- // Check if this chunk contains the end marker
46
- this.pasteBuffer += data;
47
- const endIndex = this.pasteBuffer.indexOf("\x1b[201~");
48
- if (endIndex !== -1) {
49
- // Extract the pasted content
50
- const pasteContent = this.pasteBuffer.substring(0, endIndex);
51
- // Process the complete paste
52
- this.handlePaste(pasteContent);
53
- // Reset paste state
54
- this.isInPaste = false;
55
- // Handle any remaining input after the paste marker
56
- const remaining = this.pasteBuffer.substring(endIndex + 6); // 6 = length of \x1b[201~
57
- this.pasteBuffer = "";
58
- if (remaining) {
59
- this.handleInput(remaining);
60
- }
61
- }
62
- return;
63
- }
64
- const kb = getKeybindings();
65
- // Escape/Cancel
66
- if (kb.matches(data, "tui.select.cancel")) {
67
- if (this.onEscape)
68
- this.onEscape();
69
- return;
70
- }
71
- // Undo
72
- if (kb.matches(data, "tui.editor.undo")) {
73
- this.undo();
74
- return;
75
- }
76
- // Submit
77
- if (kb.matches(data, "tui.input.submit") || data === "\n") {
78
- if (this.onSubmit)
79
- this.onSubmit(this.value);
80
- return;
81
- }
82
- // Deletion
83
- if (kb.matches(data, "tui.editor.deleteCharBackward")) {
84
- this.handleBackspace();
85
- return;
86
- }
87
- if (kb.matches(data, "tui.editor.deleteCharForward")) {
88
- this.handleForwardDelete();
89
- return;
90
- }
91
- if (kb.matches(data, "tui.editor.deleteWordBackward")) {
92
- this.deleteWordBackwards();
93
- return;
94
- }
95
- if (kb.matches(data, "tui.editor.deleteWordForward")) {
96
- this.deleteWordForward();
97
- return;
98
- }
99
- if (kb.matches(data, "tui.editor.deleteToLineStart")) {
100
- this.deleteToLineStart();
101
- return;
102
- }
103
- if (kb.matches(data, "tui.editor.deleteToLineEnd")) {
104
- this.deleteToLineEnd();
105
- return;
106
- }
107
- // Kill ring actions
108
- if (kb.matches(data, "tui.editor.yank")) {
109
- this.yank();
110
- return;
111
- }
112
- if (kb.matches(data, "tui.editor.yankPop")) {
113
- this.yankPop();
114
- return;
115
- }
116
- // Cursor movement
117
- if (kb.matches(data, "tui.editor.cursorLeft")) {
118
- this.lastAction = null;
119
- if (this.cursor > 0) {
120
- const beforeCursor = this.value.slice(0, this.cursor);
121
- const graphemes = [...segmenter.segment(beforeCursor)];
122
- const lastGrapheme = graphemes[graphemes.length - 1];
123
- this.cursor -= lastGrapheme ? lastGrapheme.segment.length : 1;
124
- }
125
- return;
126
- }
127
- if (kb.matches(data, "tui.editor.cursorRight")) {
128
- this.lastAction = null;
129
- if (this.cursor < this.value.length) {
130
- const afterCursor = this.value.slice(this.cursor);
131
- const graphemes = [...segmenter.segment(afterCursor)];
132
- const firstGrapheme = graphemes[0];
133
- this.cursor += firstGrapheme ? firstGrapheme.segment.length : 1;
134
- }
135
- return;
136
- }
137
- if (kb.matches(data, "tui.editor.cursorLineStart")) {
138
- this.lastAction = null;
139
- this.cursor = 0;
140
- return;
141
- }
142
- if (kb.matches(data, "tui.editor.cursorLineEnd")) {
143
- this.lastAction = null;
144
- this.cursor = this.value.length;
145
- return;
146
- }
147
- if (kb.matches(data, "tui.editor.cursorWordLeft")) {
148
- this.moveWordBackwards();
149
- return;
150
- }
151
- if (kb.matches(data, "tui.editor.cursorWordRight")) {
152
- this.moveWordForwards();
153
- return;
154
- }
155
- // Kitty CSI-u printable character (e.g. \x1b[97u for 'a').
156
- // Terminals with Kitty protocol flag 1 (disambiguate) send CSI-u for all keys,
157
- // including plain printable characters. Decode before the control-char check
158
- // since CSI-u sequences contain \x1b which would be rejected.
159
- const kittyPrintable = decodeKittyPrintable(data);
160
- if (kittyPrintable !== undefined) {
161
- this.insertCharacter(kittyPrintable);
162
- return;
163
- }
164
- // Regular character input - accept printable characters including Unicode,
165
- // but reject control characters (C0: 0x00-0x1F, DEL: 0x7F, C1: 0x80-0x9F)
166
- const hasControlChars = [...data].some((ch) => {
167
- const code = ch.charCodeAt(0);
168
- return code < 32 || code === 0x7f || (code >= 0x80 && code <= 0x9f);
169
- });
170
- if (!hasControlChars) {
171
- this.insertCharacter(data);
172
- }
173
- }
174
- insertCharacter(char) {
175
- // Undo coalescing: consecutive word chars coalesce into one undo unit
176
- if (isWhitespaceChar(char) || this.lastAction !== "type-word") {
177
- this.pushUndo();
178
- }
179
- this.lastAction = "type-word";
180
- this.value = this.value.slice(0, this.cursor) + char + this.value.slice(this.cursor);
181
- this.cursor += char.length;
182
- }
183
- handleBackspace() {
184
- this.lastAction = null;
185
- if (this.cursor > 0) {
186
- this.pushUndo();
187
- const beforeCursor = this.value.slice(0, this.cursor);
188
- const graphemes = [...segmenter.segment(beforeCursor)];
189
- const lastGrapheme = graphemes[graphemes.length - 1];
190
- const graphemeLength = lastGrapheme ? lastGrapheme.segment.length : 1;
191
- this.value = this.value.slice(0, this.cursor - graphemeLength) + this.value.slice(this.cursor);
192
- this.cursor -= graphemeLength;
193
- }
194
- }
195
- handleForwardDelete() {
196
- this.lastAction = null;
197
- if (this.cursor < this.value.length) {
198
- this.pushUndo();
199
- const afterCursor = this.value.slice(this.cursor);
200
- const graphemes = [...segmenter.segment(afterCursor)];
201
- const firstGrapheme = graphemes[0];
202
- const graphemeLength = firstGrapheme ? firstGrapheme.segment.length : 1;
203
- this.value = this.value.slice(0, this.cursor) + this.value.slice(this.cursor + graphemeLength);
204
- }
205
- }
206
- deleteToLineStart() {
207
- if (this.cursor === 0)
208
- return;
209
- this.pushUndo();
210
- const deletedText = this.value.slice(0, this.cursor);
211
- this.killRing.push(deletedText, { prepend: true, accumulate: this.lastAction === "kill" });
212
- this.lastAction = "kill";
213
- this.value = this.value.slice(this.cursor);
214
- this.cursor = 0;
215
- }
216
- deleteToLineEnd() {
217
- if (this.cursor >= this.value.length)
218
- return;
219
- this.pushUndo();
220
- const deletedText = this.value.slice(this.cursor);
221
- this.killRing.push(deletedText, { prepend: false, accumulate: this.lastAction === "kill" });
222
- this.lastAction = "kill";
223
- this.value = this.value.slice(0, this.cursor);
224
- }
225
- deleteWordBackwards() {
226
- if (this.cursor === 0)
227
- return;
228
- // Save lastAction before cursor movement (moveWordBackwards resets it)
229
- const wasKill = this.lastAction === "kill";
230
- this.pushUndo();
231
- const oldCursor = this.cursor;
232
- this.moveWordBackwards();
233
- const deleteFrom = this.cursor;
234
- this.cursor = oldCursor;
235
- const deletedText = this.value.slice(deleteFrom, this.cursor);
236
- this.killRing.push(deletedText, { prepend: true, accumulate: wasKill });
237
- this.lastAction = "kill";
238
- this.value = this.value.slice(0, deleteFrom) + this.value.slice(this.cursor);
239
- this.cursor = deleteFrom;
240
- }
241
- deleteWordForward() {
242
- if (this.cursor >= this.value.length)
243
- return;
244
- // Save lastAction before cursor movement (moveWordForwards resets it)
245
- const wasKill = this.lastAction === "kill";
246
- this.pushUndo();
247
- const oldCursor = this.cursor;
248
- this.moveWordForwards();
249
- const deleteTo = this.cursor;
250
- this.cursor = oldCursor;
251
- const deletedText = this.value.slice(this.cursor, deleteTo);
252
- this.killRing.push(deletedText, { prepend: false, accumulate: wasKill });
253
- this.lastAction = "kill";
254
- this.value = this.value.slice(0, this.cursor) + this.value.slice(deleteTo);
255
- }
256
- yank() {
257
- const text = this.killRing.peek();
258
- if (!text)
259
- return;
260
- this.pushUndo();
261
- this.value = this.value.slice(0, this.cursor) + text + this.value.slice(this.cursor);
262
- this.cursor += text.length;
263
- this.lastAction = "yank";
264
- }
265
- yankPop() {
266
- if (this.lastAction !== "yank" || this.killRing.length <= 1)
267
- return;
268
- this.pushUndo();
269
- // Delete the previously yanked text (still at end of ring before rotation)
270
- const prevText = this.killRing.peek() || "";
271
- this.value = this.value.slice(0, this.cursor - prevText.length) + this.value.slice(this.cursor);
272
- this.cursor -= prevText.length;
273
- // Rotate and insert new entry
274
- this.killRing.rotate();
275
- const text = this.killRing.peek() || "";
276
- this.value = this.value.slice(0, this.cursor) + text + this.value.slice(this.cursor);
277
- this.cursor += text.length;
278
- this.lastAction = "yank";
279
- }
280
- pushUndo() {
281
- this.undoStack.push({ value: this.value, cursor: this.cursor });
282
- }
283
- undo() {
284
- const snapshot = this.undoStack.pop();
285
- if (!snapshot)
286
- return;
287
- this.value = snapshot.value;
288
- this.cursor = snapshot.cursor;
289
- this.lastAction = null;
290
- }
291
- moveWordBackwards() {
292
- if (this.cursor === 0) {
293
- return;
294
- }
295
- this.lastAction = null;
296
- const textBeforeCursor = this.value.slice(0, this.cursor);
297
- const graphemes = [...segmenter.segment(textBeforeCursor)];
298
- // Skip trailing whitespace
299
- while (graphemes.length > 0 && isWhitespaceChar(graphemes[graphemes.length - 1]?.segment || "")) {
300
- this.cursor -= graphemes.pop()?.segment.length || 0;
301
- }
302
- if (graphemes.length > 0) {
303
- const lastGrapheme = graphemes[graphemes.length - 1]?.segment || "";
304
- if (isPunctuationChar(lastGrapheme)) {
305
- // Skip punctuation run
306
- while (graphemes.length > 0 && isPunctuationChar(graphemes[graphemes.length - 1]?.segment || "")) {
307
- this.cursor -= graphemes.pop()?.segment.length || 0;
308
- }
309
- }
310
- else {
311
- // Skip word run
312
- while (graphemes.length > 0 &&
313
- !isWhitespaceChar(graphemes[graphemes.length - 1]?.segment || "") &&
314
- !isPunctuationChar(graphemes[graphemes.length - 1]?.segment || "")) {
315
- this.cursor -= graphemes.pop()?.segment.length || 0;
316
- }
317
- }
318
- }
319
- }
320
- moveWordForwards() {
321
- if (this.cursor >= this.value.length) {
322
- return;
323
- }
324
- this.lastAction = null;
325
- const textAfterCursor = this.value.slice(this.cursor);
326
- const segments = segmenter.segment(textAfterCursor);
327
- const iterator = segments[Symbol.iterator]();
328
- let next = iterator.next();
329
- // Skip leading whitespace
330
- while (!next.done && isWhitespaceChar(next.value.segment)) {
331
- this.cursor += next.value.segment.length;
332
- next = iterator.next();
333
- }
334
- if (!next.done) {
335
- const firstGrapheme = next.value.segment;
336
- if (isPunctuationChar(firstGrapheme)) {
337
- // Skip punctuation run
338
- while (!next.done && isPunctuationChar(next.value.segment)) {
339
- this.cursor += next.value.segment.length;
340
- next = iterator.next();
341
- }
342
- }
343
- else {
344
- // Skip word run
345
- while (!next.done && !isWhitespaceChar(next.value.segment) && !isPunctuationChar(next.value.segment)) {
346
- this.cursor += next.value.segment.length;
347
- next = iterator.next();
348
- }
349
- }
350
- }
351
- }
352
- handlePaste(pastedText) {
353
- this.lastAction = null;
354
- this.pushUndo();
355
- // Clean the pasted text - remove newlines and carriage returns
356
- const cleanText = pastedText.replace(/\r\n/g, "").replace(/\r/g, "").replace(/\n/g, "").replace(/\t/g, " ");
357
- // Insert at cursor position
358
- this.value = this.value.slice(0, this.cursor) + cleanText + this.value.slice(this.cursor);
359
- this.cursor += cleanText.length;
360
- }
361
- invalidate() {
362
- // No cached state to invalidate currently
363
- }
364
- render(width) {
365
- // Calculate visible window
366
- const prompt = "> ";
367
- const availableWidth = width - prompt.length;
368
- if (availableWidth <= 0) {
369
- return [prompt];
370
- }
371
- let visibleText = "";
372
- let cursorDisplay = this.cursor;
373
- const totalWidth = visibleWidth(this.value);
374
- if (totalWidth < availableWidth) {
375
- // Everything fits (leave room for cursor at end)
376
- visibleText = this.value;
377
- }
378
- else {
379
- // Need horizontal scrolling
380
- // Reserve one column for cursor if it's at the end
381
- const scrollWidth = this.cursor === this.value.length ? availableWidth - 1 : availableWidth;
382
- const cursorCol = visibleWidth(this.value.slice(0, this.cursor));
383
- if (scrollWidth > 0) {
384
- const halfWidth = Math.floor(scrollWidth / 2);
385
- let startCol = 0;
386
- if (cursorCol < halfWidth) {
387
- // Cursor near start
388
- startCol = 0;
389
- }
390
- else if (cursorCol > totalWidth - halfWidth) {
391
- // Cursor near end
392
- startCol = Math.max(0, totalWidth - scrollWidth);
393
- }
394
- else {
395
- // Cursor in middle
396
- startCol = Math.max(0, cursorCol - halfWidth);
397
- }
398
- visibleText = sliceByColumn(this.value, startCol, scrollWidth, true);
399
- const beforeCursor = sliceByColumn(this.value, startCol, Math.max(0, cursorCol - startCol), true);
400
- cursorDisplay = beforeCursor.length;
401
- }
402
- else {
403
- visibleText = "";
404
- cursorDisplay = 0;
405
- }
406
- }
407
- // Build line with fake cursor
408
- // Insert cursor character at cursor position
409
- const graphemes = [...segmenter.segment(visibleText.slice(cursorDisplay))];
410
- const cursorGrapheme = graphemes[0];
411
- const beforeCursor = visibleText.slice(0, cursorDisplay);
412
- const atCursor = cursorGrapheme?.segment ?? " "; // Character at cursor, or space if at end
413
- const afterCursor = visibleText.slice(cursorDisplay + atCursor.length);
414
- // Hardware cursor marker (zero-width, emitted before fake cursor for IME positioning)
415
- const marker = this.focused ? CURSOR_MARKER : "";
416
- // Use inverse video to show cursor
417
- const cursorChar = `\x1b[7m${atCursor}\x1b[27m`; // ESC[7m = reverse video, ESC[27m = normal
418
- const textWithCursor = beforeCursor + marker + cursorChar + afterCursor;
419
- // Calculate visual width
420
- const visualLength = visibleWidth(textWithCursor);
421
- const padding = " ".repeat(Math.max(0, availableWidth - visualLength));
422
- const line = prompt + textWithCursor + padding;
423
- return [line];
424
- }
425
- }
@@ -1,68 +0,0 @@
1
- import { Text } from "./text.js";
2
- const DEFAULT_FRAMES = ["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"];
3
- const DEFAULT_INTERVAL_MS = 80;
4
- /**
5
- * Loader component that updates with an optional spinning animation.
6
- */
7
- export class Loader extends Text {
8
- frames = [...DEFAULT_FRAMES];
9
- intervalMs = DEFAULT_INTERVAL_MS;
10
- currentFrame = 0;
11
- intervalId = null;
12
- ui = null;
13
- renderIndicatorVerbatim = false;
14
- spinnerColorFn;
15
- messageColorFn;
16
- message = "Loading...";
17
- constructor(ui, spinnerColorFn, messageColorFn, message = "Loading...", indicator) {
18
- super("", 1, 0);
19
- this.ui = ui;
20
- this.spinnerColorFn = spinnerColorFn;
21
- this.messageColorFn = messageColorFn;
22
- this.message = message;
23
- this.setIndicator(indicator);
24
- }
25
- render(width) {
26
- return ["", ...super.render(width)];
27
- }
28
- start() {
29
- this.updateDisplay();
30
- this.restartAnimation();
31
- }
32
- stop() {
33
- if (this.intervalId) {
34
- clearInterval(this.intervalId);
35
- this.intervalId = null;
36
- }
37
- }
38
- setMessage(message) {
39
- this.message = message;
40
- this.updateDisplay();
41
- }
42
- setIndicator(indicator) {
43
- this.renderIndicatorVerbatim = indicator !== undefined;
44
- this.frames = indicator?.frames !== undefined ? [...indicator.frames] : [...DEFAULT_FRAMES];
45
- this.intervalMs = indicator?.intervalMs && indicator.intervalMs > 0 ? indicator.intervalMs : DEFAULT_INTERVAL_MS;
46
- this.currentFrame = 0;
47
- this.start();
48
- }
49
- restartAnimation() {
50
- this.stop();
51
- if (this.frames.length <= 1) {
52
- return;
53
- }
54
- this.intervalId = setInterval(() => {
55
- this.currentFrame = (this.currentFrame + 1) % this.frames.length;
56
- this.updateDisplay();
57
- }, this.intervalMs);
58
- }
59
- updateDisplay() {
60
- const frame = this.frames[this.currentFrame] ?? "";
61
- const renderedFrame = this.renderIndicatorVerbatim ? frame : this.spinnerColorFn(frame);
62
- const indicator = frame.length > 0 ? `${renderedFrame} ` : "";
63
- this.setText(`${indicator}${this.messageColorFn(this.message)}`);
64
- if (this.ui) {
65
- this.ui.requestRender();
66
- }
67
- }
68
- }