erosolar-cli 1.7.258 → 1.7.259

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 +136 -66
  58. package/dist/shell/terminalInput.d.ts.map +1 -1
  59. package/dist/shell/terminalInput.js +662 -409
  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,84 @@ 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 during streaming.
130
+ * Uses cursor save/restore so content flow is not disrupted.
131
+ * Content writes to terminal scrollback, input area overlays at bottom.
132
+ */
133
+ private renderStreamingInputArea;
134
+ /**
135
+ * Enable or disable flow mode.
136
+ * In flow mode, the input renders immediately after content (wherever cursor is).
137
+ * When disabled, input renders at the absolute bottom of terminal.
138
+ */
139
+ setFlowMode(enabled: boolean): void;
140
+ /**
141
+ * Check if flow mode is enabled.
142
+ */
143
+ isFlowMode(): boolean;
144
+ /**
145
+ * Set available slash commands for auto-complete suggestions.
146
+ */
147
+ setCommands(commands: Array<{
148
+ command: string;
149
+ description: string;
150
+ }>): void;
151
+ /**
152
+ * Update filtered suggestions based on current input.
153
+ */
154
+ private updateSuggestions;
155
+ /**
156
+ * Select next suggestion (arrow down / tab).
157
+ */
158
+ selectNextSuggestion(): void;
159
+ /**
160
+ * Select previous suggestion (arrow up / shift+tab).
161
+ */
162
+ selectPrevSuggestion(): void;
163
+ /**
164
+ * Accept current suggestion and insert into buffer.
165
+ */
166
+ acceptSuggestion(): boolean;
167
+ /**
168
+ * Check if suggestions are visible.
169
+ */
170
+ areSuggestionsVisible(): boolean;
171
+ /**
172
+ * Update token count for metrics display
173
+ */
174
+ setTokensUsed(tokens: number): void;
175
+ /**
176
+ * Toggle thinking/reasoning mode
177
+ */
178
+ toggleThinking(): void;
179
+ /**
180
+ * Get thinking enabled state
181
+ */
182
+ isThinkingEnabled(): boolean;
126
183
  /**
127
184
  * Keep the top N rows pinned outside the scroll region (used for the launch banner).
128
185
  */
129
186
  setPinnedHeaderLines(count: number): void;
187
+ /**
188
+ * Anchor prompt rendering near a specific row (inline layout). Pass null to
189
+ * restore the default bottom-aligned layout.
190
+ */
191
+ setInlineAnchor(row: number | null): void;
192
+ /**
193
+ * Provide a dynamic anchor callback. When set, the prompt will follow the
194
+ * output by re-evaluating the anchor before each render.
195
+ */
196
+ setInlineAnchorProvider(provider: (() => number | null | undefined) | null): void;
130
197
  /**
131
198
  * Get current mode
132
199
  */
@@ -173,16 +240,6 @@ export declare class TerminalInput extends EventEmitter {
173
240
  * Can be shown alongside override status messages.
174
241
  */
175
242
  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
243
  /**
187
244
  * Keep mode toggles (verification/auto-continue) visible in the control bar.
188
245
  * Hotkey labels remain stable so the bar looks the same before/during streaming.
@@ -192,55 +249,62 @@ export declare class TerminalInput extends EventEmitter {
192
249
  autoContinueEnabled: boolean;
193
250
  verificationHotkey?: string;
194
251
  autoContinueHotkey?: string;
195
- thinkingModeLabel?: string | null;
196
- thinkingHotkey?: string;
197
252
  }): void;
198
253
  /**
199
254
  * Clear all status messages at once (convenience method).
200
255
  */
201
256
  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
257
  /**
210
258
  * Render the input area - Claude Code style with mode controls
211
259
  *
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.
260
+ * Same rendering for both normal and streaming modes - just different status bar.
261
+ * During streaming, uses cursor save/restore to preserve streaming position.
215
262
  */
216
263
  render(): void;
217
264
  /**
218
- * Build one or more compact meta lines above the divider (thinking, status, usage).
265
+ * Render in flow mode - delegates to bottom-pinned for stability.
266
+ *
267
+ * Flow mode attempted inline rendering but caused duplicate renders
268
+ * due to unreliable cursor position tracking. Bottom-pinned is reliable.
219
269
  */
220
- private buildMetaLines;
270
+ private renderFlowMode;
221
271
  /**
222
- * Clear the reserved bottom block (meta + divider + input + controls) before repainting.
272
+ * Render in bottom-pinned mode - Claude Code style with suggestions
273
+ *
274
+ * Works for both normal and streaming modes:
275
+ * - During streaming: saves/restores cursor position
276
+ * - Status bar shows streaming info or "Type a message"
277
+ *
278
+ * Layout when suggestions visible:
279
+ * - Top divider
280
+ * - Input line(s)
281
+ * - Bottom divider
282
+ * - Suggestions (command list)
283
+ *
284
+ * Layout when suggestions hidden:
285
+ * - Status bar (Ready/Streaming)
286
+ * - Top divider
287
+ * - Input line(s)
288
+ * - Bottom divider
289
+ * - Mode controls
290
+ */
291
+ private renderBottomPinned;
292
+ /**
293
+ * Build status bar for streaming mode (shows elapsed time, queue count).
294
+ */
295
+ private buildStreamingStatusBar;
296
+ /**
297
+ * Build status bar showing streaming/ready status and key info.
298
+ * This is the TOP line above the input area - minimal Claude Code style.
223
299
  */
224
- private clearReservedArea;
300
+ private buildStatusBar;
225
301
  /**
226
- * Build Claude Code style mode controls line.
227
- * Combines streaming label + override status + main status for simultaneous display.
302
+ * Build mode controls line showing toggles and context info.
303
+ * This is the BOTTOM line below the input area - Claude Code style layout with erosolar features.
304
+ *
305
+ * Layout: [toggles on left] ... [context info on right]
228
306
  */
229
307
  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
308
  /**
245
309
  * Force a re-render
246
310
  */
@@ -257,6 +321,9 @@ export declare class TerminalInput extends EventEmitter {
257
321
  * Register with display's output interceptor to position cursor correctly.
258
322
  * When scroll region is active, output needs to go to the scroll region,
259
323
  * not the protected bottom area where the input is rendered.
324
+ *
325
+ * NOTE: With scroll region properly set, content naturally stays within
326
+ * the region boundaries - no cursor manipulation needed per-write.
260
327
  */
261
328
  registerOutputInterceptor(display: {
262
329
  registerOutputInterceptor: (i: {
@@ -301,18 +368,21 @@ export declare class TerminalInput extends EventEmitter {
301
368
  private shiftPlaceholders;
302
369
  private removeRange;
303
370
  private insertPlainText;
304
- private shouldInlineMultiline;
305
371
  private findPlaceholderAt;
306
372
  private buildPlaceholder;
307
373
  private insertPastePlaceholder;
374
+ /**
375
+ * Toggle expansion of a paste placeholder at the current cursor position.
376
+ * When expanded, shows first 3 and last 2 lines of the content.
377
+ */
378
+ togglePasteExpansion(): boolean;
379
+ private buildExpandedPlaceholder;
308
380
  private deletePlaceholder;
309
- updateContextUsage(value: number | null, autoCompactThreshold?: number): void;
381
+ updateContextUsage(value: number | null): void;
310
382
  getEditMode(): EditGuardMode;
311
383
  applyEditMode(mode: EditGuardMode): void;
312
384
  private setEditMode;
313
385
  toggleEditMode(): void;
314
- private scheduleStreamingRender;
315
- private resetStreamingRenderThrottle;
316
386
  private scheduleRender;
317
387
  private canRender;
318
388
  private isTTY;