erosolar-cli 1.7.258 → 1.7.260

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 (91) hide show
  1. package/README.md +148 -22
  2. package/dist/core/aiFlowOptimizer.d.ts +26 -0
  3. package/dist/core/aiFlowOptimizer.d.ts.map +1 -0
  4. package/dist/core/aiFlowOptimizer.js +31 -0
  5. package/dist/core/aiFlowOptimizer.js.map +1 -0
  6. package/dist/core/aiOptimizationEngine.d.ts +158 -0
  7. package/dist/core/aiOptimizationEngine.d.ts.map +1 -0
  8. package/dist/core/aiOptimizationEngine.js +428 -0
  9. package/dist/core/aiOptimizationEngine.js.map +1 -0
  10. package/dist/core/aiOptimizationIntegration.d.ts +93 -0
  11. package/dist/core/aiOptimizationIntegration.d.ts.map +1 -0
  12. package/dist/core/aiOptimizationIntegration.js +250 -0
  13. package/dist/core/aiOptimizationIntegration.js.map +1 -0
  14. package/dist/core/customCommands.d.ts +0 -1
  15. package/dist/core/customCommands.d.ts.map +1 -1
  16. package/dist/core/customCommands.js +0 -3
  17. package/dist/core/customCommands.js.map +1 -1
  18. package/dist/core/enhancedErrorRecovery.d.ts +100 -0
  19. package/dist/core/enhancedErrorRecovery.d.ts.map +1 -0
  20. package/dist/core/enhancedErrorRecovery.js +345 -0
  21. package/dist/core/enhancedErrorRecovery.js.map +1 -0
  22. package/dist/core/toolPreconditions.d.ts.map +1 -1
  23. package/dist/core/toolPreconditions.js +14 -0
  24. package/dist/core/toolPreconditions.js.map +1 -1
  25. package/dist/core/toolRuntime.d.ts.map +1 -1
  26. package/dist/core/toolRuntime.js +5 -0
  27. package/dist/core/toolRuntime.js.map +1 -1
  28. package/dist/core/toolValidation.d.ts.map +1 -1
  29. package/dist/core/toolValidation.js +3 -14
  30. package/dist/core/toolValidation.js.map +1 -1
  31. package/dist/core/validationRunner.d.ts +3 -1
  32. package/dist/core/validationRunner.d.ts.map +1 -1
  33. package/dist/core/validationRunner.js.map +1 -1
  34. package/dist/mcp/sseClient.d.ts.map +1 -1
  35. package/dist/mcp/sseClient.js +18 -9
  36. package/dist/mcp/sseClient.js.map +1 -1
  37. package/dist/plugins/tools/build/buildPlugin.d.ts +6 -0
  38. package/dist/plugins/tools/build/buildPlugin.d.ts.map +1 -1
  39. package/dist/plugins/tools/build/buildPlugin.js +10 -4
  40. package/dist/plugins/tools/build/buildPlugin.js.map +1 -1
  41. package/dist/shell/claudeCodeStreamHandler.d.ts +145 -0
  42. package/dist/shell/claudeCodeStreamHandler.d.ts.map +1 -0
  43. package/dist/shell/claudeCodeStreamHandler.js +322 -0
  44. package/dist/shell/claudeCodeStreamHandler.js.map +1 -0
  45. package/dist/shell/inputQueueManager.d.ts +144 -0
  46. package/dist/shell/inputQueueManager.d.ts.map +1 -0
  47. package/dist/shell/inputQueueManager.js +290 -0
  48. package/dist/shell/inputQueueManager.js.map +1 -0
  49. package/dist/shell/interactiveShell.d.ts +2 -10
  50. package/dist/shell/interactiveShell.d.ts.map +1 -1
  51. package/dist/shell/interactiveShell.js +35 -190
  52. package/dist/shell/interactiveShell.js.map +1 -1
  53. package/dist/shell/streamingOutputManager.d.ts +115 -0
  54. package/dist/shell/streamingOutputManager.d.ts.map +1 -0
  55. package/dist/shell/streamingOutputManager.js +225 -0
  56. package/dist/shell/streamingOutputManager.js.map +1 -0
  57. package/dist/shell/terminalInput.d.ts +140 -66
  58. package/dist/shell/terminalInput.d.ts.map +1 -1
  59. package/dist/shell/terminalInput.js +685 -410
  60. package/dist/shell/terminalInput.js.map +1 -1
  61. package/dist/shell/terminalInputAdapter.d.ts +15 -20
  62. package/dist/shell/terminalInputAdapter.d.ts.map +1 -1
  63. package/dist/shell/terminalInputAdapter.js +22 -14
  64. package/dist/shell/terminalInputAdapter.js.map +1 -1
  65. package/dist/ui/ShellUIAdapter.d.ts.map +1 -1
  66. package/dist/ui/ShellUIAdapter.js +12 -13
  67. package/dist/ui/ShellUIAdapter.js.map +1 -1
  68. package/dist/ui/display.d.ts +0 -19
  69. package/dist/ui/display.d.ts.map +1 -1
  70. package/dist/ui/display.js +22 -135
  71. package/dist/ui/display.js.map +1 -1
  72. package/dist/ui/persistentPrompt.d.ts +50 -0
  73. package/dist/ui/persistentPrompt.d.ts.map +1 -0
  74. package/dist/ui/persistentPrompt.js +92 -0
  75. package/dist/ui/persistentPrompt.js.map +1 -0
  76. package/dist/ui/terminalUISchema.d.ts +195 -0
  77. package/dist/ui/terminalUISchema.d.ts.map +1 -0
  78. package/dist/ui/terminalUISchema.js +113 -0
  79. package/dist/ui/terminalUISchema.js.map +1 -0
  80. package/dist/ui/theme.d.ts.map +1 -1
  81. package/dist/ui/theme.js +8 -6
  82. package/dist/ui/theme.js.map +1 -1
  83. package/dist/ui/toolDisplay.d.ts +158 -0
  84. package/dist/ui/toolDisplay.d.ts.map +1 -1
  85. package/dist/ui/toolDisplay.js +348 -0
  86. package/dist/ui/toolDisplay.js.map +1 -1
  87. package/dist/ui/unified/layout.d.ts +0 -1
  88. package/dist/ui/unified/layout.d.ts.map +1 -1
  89. package/dist/ui/unified/layout.js +25 -15
  90. package/dist/ui/unified/layout.js.map +1 -1
  91. package/package.json +1 -1
@@ -0,0 +1,115 @@
1
+ /**
2
+ * StreamingOutputManager - Buffered output for Claude Code style UI
3
+ *
4
+ * Key design principle:
5
+ * - Buffer streaming output instead of writing directly to stdout
6
+ * - This keeps the cursor at the prompt area (no cursor movement during streaming)
7
+ * - Render complete response when streaming ends
8
+ * - The UI (prompt) stays fixed at the bottom
9
+ *
10
+ * Two modes:
11
+ * 1. Direct mode (bufferedMode=false): Write chunks directly to stdout (original behavior)
12
+ * 2. Buffered mode (bufferedMode=true): Collect chunks, render when complete
13
+ */
14
+ import { EventEmitter } from 'node:events';
15
+ export interface StreamingOutputState {
16
+ isStreaming: boolean;
17
+ charsWritten: number;
18
+ linesWritten: number;
19
+ startTime: number | null;
20
+ }
21
+ export interface StreamingOutputEvents {
22
+ 'stream-start': () => void;
23
+ 'stream-chunk': (chunk: string) => void;
24
+ 'stream-end': (stats: {
25
+ duration: number;
26
+ chars: number;
27
+ lines: number;
28
+ }) => void;
29
+ }
30
+ /**
31
+ * Manages streaming output with optional buffering.
32
+ *
33
+ * Buffered mode:
34
+ * - Chunks are collected in a buffer
35
+ * - Output is rendered all at once when streaming ends
36
+ * - Cursor stays at prompt area (no jumping during streaming)
37
+ *
38
+ * Direct mode:
39
+ * - Chunks written directly to stdout (original behavior)
40
+ * - Cursor moves with output
41
+ */
42
+ export declare class StreamingOutputManager extends EventEmitter {
43
+ private state;
44
+ private readonly writeStream;
45
+ private lastChunk;
46
+ private bufferedMode;
47
+ private outputBuffer;
48
+ constructor(writeStream?: NodeJS.WriteStream);
49
+ /**
50
+ * Enable or disable buffered mode
51
+ * When buffered, chunks are collected and rendered at the end
52
+ */
53
+ setBufferedMode(enabled: boolean): void;
54
+ /**
55
+ * Check if buffered mode is enabled
56
+ */
57
+ isBufferedMode(): boolean;
58
+ /**
59
+ * Get the buffered content (for rendering)
60
+ */
61
+ getBuffer(): string;
62
+ /**
63
+ * Clear the buffer
64
+ */
65
+ clearBuffer(): void;
66
+ /**
67
+ * Check if currently streaming output
68
+ */
69
+ isStreaming(): boolean;
70
+ /**
71
+ * Begin a streaming session
72
+ * Resets counters and marks streaming as active
73
+ * In buffered mode, clears the buffer
74
+ */
75
+ startStream(): void;
76
+ /**
77
+ * Write a chunk of content
78
+ *
79
+ * In buffered mode: Collect in buffer (cursor stays at prompt)
80
+ * In direct mode: Write to stdout immediately
81
+ */
82
+ writeChunk(chunk: string): void;
83
+ /**
84
+ * Write a complete line (adds newline)
85
+ */
86
+ writeLine(line?: string): void;
87
+ /**
88
+ * End the streaming session
89
+ * In buffered mode: Renders the complete buffered output
90
+ * Returns statistics about what was streamed
91
+ */
92
+ endStream(): {
93
+ duration: number;
94
+ chars: number;
95
+ lines: number;
96
+ buffer?: string;
97
+ };
98
+ /**
99
+ * Get current streaming statistics
100
+ */
101
+ getStats(): {
102
+ chars: number;
103
+ lines: number;
104
+ elapsed: number;
105
+ };
106
+ /**
107
+ * Force a flush (for buffered output)
108
+ */
109
+ flush(): void;
110
+ private countNewlines;
111
+ private writeLocked;
112
+ }
113
+ export declare function getStreamingOutputManager(): StreamingOutputManager;
114
+ export declare function resetStreamingOutputManager(): void;
115
+ //# sourceMappingURL=streamingOutputManager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"streamingOutputManager.d.ts","sourceRoot":"","sources":["../../src/shell/streamingOutputManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AAEH,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAG3C,MAAM,WAAW,oBAAoB;IACnC,WAAW,EAAE,OAAO,CAAC;IACrB,YAAY,EAAE,MAAM,CAAC;IACrB,YAAY,EAAE,MAAM,CAAC;IACrB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;CAC1B;AAED,MAAM,WAAW,qBAAqB;IACpC,cAAc,EAAE,MAAM,IAAI,CAAC;IAC3B,cAAc,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;IACxC,YAAY,EAAE,CAAC,KAAK,EAAE;QAAE,QAAQ,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,KAAK,IAAI,CAAC;CACnF;AAED;;;;;;;;;;;GAWG;AACH,qBAAa,sBAAuB,SAAQ,YAAY;IACtD,OAAO,CAAC,KAAK,CAKX;IAEF,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAqB;IACjD,OAAO,CAAC,SAAS,CAAc;IAG/B,OAAO,CAAC,YAAY,CAAkB;IACtC,OAAO,CAAC,YAAY,CAAgB;gBAExB,WAAW,GAAE,MAAM,CAAC,WAA4B;IAK5D;;;OAGG;IACH,eAAe,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI;IAIvC;;OAEG;IACH,cAAc,IAAI,OAAO;IAIzB;;OAEG;IACH,SAAS,IAAI,MAAM;IAInB;;OAEG;IACH,WAAW,IAAI,IAAI;IAInB;;OAEG;IACH,WAAW,IAAI,OAAO;IAItB;;;;OAIG;IACH,WAAW,IAAI,IAAI;IAiBnB;;;;;OAKG;IACH,UAAU,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAqB/B;;OAEG;IACH,SAAS,CAAC,IAAI,GAAE,MAAW,GAAG,IAAI;IAIlC;;;;OAIG;IACH,SAAS,IAAI;QAAE,QAAQ,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAC;QAAC,MAAM,CAAC,EAAE,MAAM,CAAA;KAAE;IA0ChF;;OAEG;IACH,QAAQ,IAAI;QAAE,KAAK,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAA;KAAE;IAQ7D;;OAEG;IACH,KAAK,IAAI,IAAI;IAYb,OAAO,CAAC,aAAa;IAUrB,OAAO,CAAC,WAAW;CAWpB;AAOD,wBAAgB,yBAAyB,IAAI,sBAAsB,CAKlE;AAED,wBAAgB,2BAA2B,IAAI,IAAI,CAElD"}
@@ -0,0 +1,225 @@
1
+ /**
2
+ * StreamingOutputManager - Buffered output for Claude Code style UI
3
+ *
4
+ * Key design principle:
5
+ * - Buffer streaming output instead of writing directly to stdout
6
+ * - This keeps the cursor at the prompt area (no cursor movement during streaming)
7
+ * - Render complete response when streaming ends
8
+ * - The UI (prompt) stays fixed at the bottom
9
+ *
10
+ * Two modes:
11
+ * 1. Direct mode (bufferedMode=false): Write chunks directly to stdout (original behavior)
12
+ * 2. Buffered mode (bufferedMode=true): Collect chunks, render when complete
13
+ */
14
+ import { EventEmitter } from 'node:events';
15
+ import { writeLock } from '../ui/writeLock.js';
16
+ /**
17
+ * Manages streaming output with optional buffering.
18
+ *
19
+ * Buffered mode:
20
+ * - Chunks are collected in a buffer
21
+ * - Output is rendered all at once when streaming ends
22
+ * - Cursor stays at prompt area (no jumping during streaming)
23
+ *
24
+ * Direct mode:
25
+ * - Chunks written directly to stdout (original behavior)
26
+ * - Cursor moves with output
27
+ */
28
+ export class StreamingOutputManager extends EventEmitter {
29
+ state = {
30
+ isStreaming: false,
31
+ charsWritten: 0,
32
+ linesWritten: 0,
33
+ startTime: null,
34
+ };
35
+ writeStream;
36
+ lastChunk = '';
37
+ // Buffered mode support
38
+ bufferedMode = false;
39
+ outputBuffer = [];
40
+ constructor(writeStream = process.stdout) {
41
+ super();
42
+ this.writeStream = writeStream;
43
+ }
44
+ /**
45
+ * Enable or disable buffered mode
46
+ * When buffered, chunks are collected and rendered at the end
47
+ */
48
+ setBufferedMode(enabled) {
49
+ this.bufferedMode = enabled;
50
+ }
51
+ /**
52
+ * Check if buffered mode is enabled
53
+ */
54
+ isBufferedMode() {
55
+ return this.bufferedMode;
56
+ }
57
+ /**
58
+ * Get the buffered content (for rendering)
59
+ */
60
+ getBuffer() {
61
+ return this.outputBuffer.join('');
62
+ }
63
+ /**
64
+ * Clear the buffer
65
+ */
66
+ clearBuffer() {
67
+ this.outputBuffer = [];
68
+ }
69
+ /**
70
+ * Check if currently streaming output
71
+ */
72
+ isStreaming() {
73
+ return this.state.isStreaming;
74
+ }
75
+ /**
76
+ * Begin a streaming session
77
+ * Resets counters and marks streaming as active
78
+ * In buffered mode, clears the buffer
79
+ */
80
+ startStream() {
81
+ if (this.state.isStreaming) {
82
+ return; // Already streaming
83
+ }
84
+ this.state = {
85
+ isStreaming: true,
86
+ charsWritten: 0,
87
+ linesWritten: 0,
88
+ startTime: Date.now(),
89
+ };
90
+ this.lastChunk = '';
91
+ this.outputBuffer = [];
92
+ this.emit('stream-start');
93
+ }
94
+ /**
95
+ * Write a chunk of content
96
+ *
97
+ * In buffered mode: Collect in buffer (cursor stays at prompt)
98
+ * In direct mode: Write to stdout immediately
99
+ */
100
+ writeChunk(chunk) {
101
+ if (!chunk) {
102
+ return;
103
+ }
104
+ // Track statistics
105
+ this.state.charsWritten += chunk.length;
106
+ this.state.linesWritten += this.countNewlines(chunk);
107
+ this.lastChunk = chunk;
108
+ if (this.bufferedMode) {
109
+ // Buffered mode: collect chunks, don't write yet
110
+ this.outputBuffer.push(chunk);
111
+ }
112
+ else {
113
+ // Direct mode: write immediately
114
+ this.writeLocked(chunk);
115
+ }
116
+ this.emit('stream-chunk', chunk);
117
+ }
118
+ /**
119
+ * Write a complete line (adds newline)
120
+ */
121
+ writeLine(line = '') {
122
+ this.writeChunk(`${line}\n`);
123
+ }
124
+ /**
125
+ * End the streaming session
126
+ * In buffered mode: Renders the complete buffered output
127
+ * Returns statistics about what was streamed
128
+ */
129
+ endStream() {
130
+ const stats = {
131
+ duration: this.state.startTime ? Date.now() - this.state.startTime : 0,
132
+ chars: this.state.charsWritten,
133
+ lines: this.state.linesWritten,
134
+ buffer: undefined,
135
+ };
136
+ if (this.bufferedMode) {
137
+ // In buffered mode, render the collected output now
138
+ const bufferedContent = this.outputBuffer.join('');
139
+ stats.buffer = bufferedContent;
140
+ if (bufferedContent) {
141
+ this.writeLocked(bufferedContent);
142
+ // Ensure we end with a newline
143
+ if (!bufferedContent.endsWith('\n')) {
144
+ this.writeLocked('\n');
145
+ stats.lines += 1;
146
+ }
147
+ }
148
+ this.outputBuffer = [];
149
+ }
150
+ else {
151
+ // Direct mode: just ensure newline
152
+ if (this.lastChunk && !this.lastChunk.endsWith('\n')) {
153
+ this.writeLocked('\n');
154
+ stats.lines += 1;
155
+ }
156
+ }
157
+ this.state = {
158
+ isStreaming: false,
159
+ charsWritten: 0,
160
+ linesWritten: 0,
161
+ startTime: null,
162
+ };
163
+ this.lastChunk = '';
164
+ this.emit('stream-end', stats);
165
+ return stats;
166
+ }
167
+ /**
168
+ * Get current streaming statistics
169
+ */
170
+ getStats() {
171
+ return {
172
+ chars: this.state.charsWritten,
173
+ lines: this.state.linesWritten,
174
+ elapsed: this.state.startTime ? Date.now() - this.state.startTime : 0,
175
+ };
176
+ }
177
+ /**
178
+ * Force a flush (for buffered output)
179
+ */
180
+ flush() {
181
+ // Cork/uncork pattern for optimal buffering
182
+ if (typeof this.writeStream.cork === 'function') {
183
+ this.writeStream.cork();
184
+ process.nextTick(() => {
185
+ if (typeof this.writeStream.uncork === 'function') {
186
+ this.writeStream.uncork();
187
+ }
188
+ });
189
+ }
190
+ }
191
+ countNewlines(text) {
192
+ let count = 0;
193
+ for (const char of text) {
194
+ if (char === '\n') {
195
+ count += 1;
196
+ }
197
+ }
198
+ return count;
199
+ }
200
+ writeLocked(chunk) {
201
+ // If lock is already held, write directly - we're in a protected context
202
+ // This prevents queuing issues where content gets delayed during streaming
203
+ if (writeLock.isLocked()) {
204
+ this.writeStream.write(chunk);
205
+ return;
206
+ }
207
+ writeLock.withLock(() => {
208
+ this.writeStream.write(chunk);
209
+ }, 'streamingOutputManager.write');
210
+ }
211
+ }
212
+ /**
213
+ * Singleton instance for global use
214
+ */
215
+ let globalInstance = null;
216
+ export function getStreamingOutputManager() {
217
+ if (!globalInstance) {
218
+ globalInstance = new StreamingOutputManager();
219
+ }
220
+ return globalInstance;
221
+ }
222
+ export function resetStreamingOutputManager() {
223
+ globalInstance = null;
224
+ }
225
+ //# sourceMappingURL=streamingOutputManager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"streamingOutputManager.js","sourceRoot":"","sources":["../../src/shell/streamingOutputManager.ts"],"names":[],"mappings":"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"}
@@ -3,18 +3,14 @@
3
3
  *
4
4
  * Design principles:
5
5
  * - Single source of truth for input state
6
- * - One bottom-pinned chat box for the entire session (no inline anchors)
7
6
  * - Native bracketed paste support (no heuristics)
8
7
  * - Clean cursor model with render-time wrapping
9
8
  * - State machine for different input modes
10
9
  * - No readline dependency for display
11
10
  */
12
11
  import { EventEmitter } from 'node:events';
13
- /**
14
- * Input modes - determines how input is handled
15
- */
16
- export type InputMode = 'idle' | 'streaming' | 'paste';
17
- export type EditGuardMode = 'display-edits' | 'ask-permission';
12
+ import { type InputMode, type EditGuardMode } from '../ui/terminalUISchema.js';
13
+ export type { InputMode, EditGuardMode } from '../ui/terminalUISchema.js';
18
14
  /**
19
15
  * Events emitted by TerminalInput
20
16
  */
@@ -66,11 +62,6 @@ export declare class TerminalInput extends EventEmitter {
66
62
  private statusMessage;
67
63
  private overrideStatusMessage;
68
64
  private streamingLabel;
69
- private metaElapsedSeconds;
70
- private metaTokensUsed;
71
- private metaTokenLimit;
72
- private metaThinkingMs;
73
- private metaThinkingHasContent;
74
65
  private reservedLines;
75
66
  private scrollRegionActive;
76
67
  private lastRenderContent;
@@ -78,22 +69,28 @@ export declare class TerminalInput extends EventEmitter {
78
69
  private renderDirty;
79
70
  private isRendering;
80
71
  private pinnedTopRows;
72
+ private inlineAnchorRow;
73
+ private inlineLayout;
74
+ private anchorProvider;
75
+ private flowMode;
76
+ private flowModeRenderedLines;
77
+ private commandSuggestions;
78
+ private filteredSuggestions;
79
+ private selectedSuggestionIndex;
80
+ private showSuggestions;
81
+ private maxVisibleSuggestions;
81
82
  private disposed;
82
83
  private enabled;
83
84
  private contextUsage;
84
- private contextAutoCompactThreshold;
85
- private thinkingModeLabel;
86
85
  private editMode;
87
86
  private verificationEnabled;
88
87
  private autoContinueEnabled;
89
88
  private verificationHotkey;
90
89
  private autoContinueHotkey;
91
- private thinkingHotkey;
92
- private modelLabel;
93
- private providerLabel;
94
90
  private outputInterceptorCleanup?;
95
- private lastStreamingRender;
96
- private streamingRenderInterval;
91
+ private streamingStartTime;
92
+ private tokensUsed;
93
+ private thinkingEnabled;
97
94
  private streamingRenderTimer;
98
95
  constructor(writeStream?: NodeJS.WriteStream, config?: TerminalInputConfig);
99
96
  /**
@@ -119,14 +116,88 @@ export declare class TerminalInput extends EventEmitter {
119
116
  /**
120
117
  * Set the input mode
121
118
  *
122
- * Streaming keeps the scroll region active so the prompt/status stay pinned
123
- * below the streaming output. When streaming ends, we refresh the input area.
119
+ * Streaming mode disables scroll region and lets content flow naturally.
120
+ * The input area will be re-rendered after streaming ends at wherever
121
+ * the cursor is (below the streamed content).
124
122
  */
125
123
  setMode(mode: InputMode): void;
124
+ /**
125
+ * Update streaming status label (called by timer)
126
+ */
127
+ private updateStreamingStatus;
128
+ /**
129
+ * Render input area at absolute bottom - unified for streaming and normal modes.
130
+ * Uses cursor save/restore during streaming so content flow is not disrupted.
131
+ * In normal mode, cursor is positioned in the input area.
132
+ */
133
+ private renderPinnedInputArea;
134
+ /**
135
+ * Render input area during streaming (alias for unified method)
136
+ */
137
+ private renderStreamingInputArea;
138
+ /**
139
+ * Enable or disable flow mode.
140
+ * In flow mode, the input renders immediately after content (wherever cursor is).
141
+ * When disabled, input renders at the absolute bottom of terminal.
142
+ */
143
+ setFlowMode(enabled: boolean): void;
144
+ /**
145
+ * Check if flow mode is enabled.
146
+ */
147
+ isFlowMode(): boolean;
148
+ /**
149
+ * Set available slash commands for auto-complete suggestions.
150
+ */
151
+ setCommands(commands: Array<{
152
+ command: string;
153
+ description: string;
154
+ }>): void;
155
+ /**
156
+ * Update filtered suggestions based on current input.
157
+ */
158
+ private updateSuggestions;
159
+ /**
160
+ * Select next suggestion (arrow down / tab).
161
+ */
162
+ selectNextSuggestion(): void;
163
+ /**
164
+ * Select previous suggestion (arrow up / shift+tab).
165
+ */
166
+ selectPrevSuggestion(): void;
167
+ /**
168
+ * Accept current suggestion and insert into buffer.
169
+ */
170
+ acceptSuggestion(): boolean;
171
+ /**
172
+ * Check if suggestions are visible.
173
+ */
174
+ areSuggestionsVisible(): boolean;
175
+ /**
176
+ * Update token count for metrics display
177
+ */
178
+ setTokensUsed(tokens: number): void;
179
+ /**
180
+ * Toggle thinking/reasoning mode
181
+ */
182
+ toggleThinking(): void;
183
+ /**
184
+ * Get thinking enabled state
185
+ */
186
+ isThinkingEnabled(): boolean;
126
187
  /**
127
188
  * Keep the top N rows pinned outside the scroll region (used for the launch banner).
128
189
  */
129
190
  setPinnedHeaderLines(count: number): void;
191
+ /**
192
+ * Anchor prompt rendering near a specific row (inline layout). Pass null to
193
+ * restore the default bottom-aligned layout.
194
+ */
195
+ setInlineAnchor(row: number | null): void;
196
+ /**
197
+ * Provide a dynamic anchor callback. When set, the prompt will follow the
198
+ * output by re-evaluating the anchor before each render.
199
+ */
200
+ setInlineAnchorProvider(provider: (() => number | null | undefined) | null): void;
130
201
  /**
131
202
  * Get current mode
132
203
  */
@@ -173,16 +244,6 @@ export declare class TerminalInput extends EventEmitter {
173
244
  * Can be shown alongside override status messages.
174
245
  */
175
246
  setStreamingLabel(label: string | null): void;
176
- /**
177
- * Surface meta status just above the divider (e.g., elapsed time or token usage).
178
- */
179
- setMetaStatus(meta: {
180
- elapsedSeconds?: number | null;
181
- tokensUsed?: number | null;
182
- tokenLimit?: number | null;
183
- thinkingMs?: number | null;
184
- thinkingHasContent?: boolean;
185
- }): void;
186
247
  /**
187
248
  * Keep mode toggles (verification/auto-continue) visible in the control bar.
188
249
  * Hotkey labels remain stable so the bar looks the same before/during streaming.
@@ -192,55 +253,62 @@ export declare class TerminalInput extends EventEmitter {
192
253
  autoContinueEnabled: boolean;
193
254
  verificationHotkey?: string;
194
255
  autoContinueHotkey?: string;
195
- thinkingModeLabel?: string | null;
196
- thinkingHotkey?: string;
197
256
  }): void;
198
257
  /**
199
258
  * Clear all status messages at once (convenience method).
200
259
  */
201
260
  clearAllStatus(): void;
202
- /**
203
- * Surface model/provider context in the controls bar.
204
- */
205
- setModelContext(options: {
206
- model?: string | null;
207
- provider?: string | null;
208
- }): void;
209
261
  /**
210
262
  * Render the input area - Claude Code style with mode controls
211
263
  *
212
- * During streaming we keep the scroll region active and repaint only the
213
- * pinned status/input block (throttled) so streamed content can scroll
214
- * naturally above while elapsed time and status stay fresh.
264
+ * Same rendering for both normal and streaming modes - just different status bar.
265
+ * During streaming, uses cursor save/restore to preserve streaming position.
215
266
  */
216
267
  render(): void;
217
268
  /**
218
- * Build one or more compact meta lines above the divider (thinking, status, usage).
269
+ * Render in flow mode - delegates to bottom-pinned for stability.
270
+ *
271
+ * Flow mode attempted inline rendering but caused duplicate renders
272
+ * due to unreliable cursor position tracking. Bottom-pinned is reliable.
219
273
  */
220
- private buildMetaLines;
274
+ private renderFlowMode;
221
275
  /**
222
- * Clear the reserved bottom block (meta + divider + input + controls) before repainting.
276
+ * Render in bottom-pinned mode - Claude Code style with suggestions
277
+ *
278
+ * Works for both normal and streaming modes:
279
+ * - During streaming: saves/restores cursor position
280
+ * - Status bar shows streaming info or "Type a message"
281
+ *
282
+ * Layout when suggestions visible:
283
+ * - Top divider
284
+ * - Input line(s)
285
+ * - Bottom divider
286
+ * - Suggestions (command list)
287
+ *
288
+ * Layout when suggestions hidden:
289
+ * - Status bar (Ready/Streaming)
290
+ * - Top divider
291
+ * - Input line(s)
292
+ * - Bottom divider
293
+ * - Mode controls
223
294
  */
224
- private clearReservedArea;
295
+ private renderBottomPinned;
225
296
  /**
226
- * Build Claude Code style mode controls line.
227
- * Combines streaming label + override status + main status for simultaneous display.
297
+ * Build status bar for streaming mode (shows elapsed time, queue count).
298
+ */
299
+ private buildStreamingStatusBar;
300
+ /**
301
+ * Build status bar showing streaming/ready status and key info.
302
+ * This is the TOP line above the input area - minimal Claude Code style.
303
+ */
304
+ private buildStatusBar;
305
+ /**
306
+ * Build mode controls line showing toggles and context info.
307
+ * This is the BOTTOM line below the input area - Claude Code style layout with erosolar features.
308
+ *
309
+ * Layout: [toggles on left] ... [context info on right]
228
310
  */
229
311
  private buildModeControls;
230
- private formatHotkey;
231
- private computeContextRemaining;
232
- private computeTokensRemaining;
233
- private formatElapsedLabel;
234
- private formatTokenCount;
235
- private visibleLength;
236
- /**
237
- * Debug-only snapshot used by tests to assert rendered strings without
238
- * needing a TTY. Not used by production code.
239
- */
240
- getDebugUiSnapshot(width?: number): {
241
- meta: string[];
242
- controls: string;
243
- };
244
312
  /**
245
313
  * Force a re-render
246
314
  */
@@ -257,6 +325,9 @@ export declare class TerminalInput extends EventEmitter {
257
325
  * Register with display's output interceptor to position cursor correctly.
258
326
  * When scroll region is active, output needs to go to the scroll region,
259
327
  * not the protected bottom area where the input is rendered.
328
+ *
329
+ * NOTE: With scroll region properly set, content naturally stays within
330
+ * the region boundaries - no cursor manipulation needed per-write.
260
331
  */
261
332
  registerOutputInterceptor(display: {
262
333
  registerOutputInterceptor: (i: {
@@ -301,18 +372,21 @@ export declare class TerminalInput extends EventEmitter {
301
372
  private shiftPlaceholders;
302
373
  private removeRange;
303
374
  private insertPlainText;
304
- private shouldInlineMultiline;
305
375
  private findPlaceholderAt;
306
376
  private buildPlaceholder;
307
377
  private insertPastePlaceholder;
378
+ /**
379
+ * Toggle expansion of a paste placeholder at the current cursor position.
380
+ * When expanded, shows first 3 and last 2 lines of the content.
381
+ */
382
+ togglePasteExpansion(): boolean;
383
+ private buildExpandedPlaceholder;
308
384
  private deletePlaceholder;
309
- updateContextUsage(value: number | null, autoCompactThreshold?: number): void;
385
+ updateContextUsage(value: number | null): void;
310
386
  getEditMode(): EditGuardMode;
311
387
  applyEditMode(mode: EditGuardMode): void;
312
388
  private setEditMode;
313
389
  toggleEditMode(): void;
314
- private scheduleStreamingRender;
315
- private resetStreamingRenderThrottle;
316
390
  private scheduleRender;
317
391
  private canRender;
318
392
  private isTTY;