409
codex-cli/src/components/chat/multiline-editor.tsx
Normal file
409
codex-cli/src/components/chat/multiline-editor.tsx
Normal file
@@ -0,0 +1,409 @@
|
||||
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||
|
||||
import { useTerminalSize } from "../../hooks/use-terminal-size";
|
||||
import TextBuffer from "../../lib/text-buffer.js";
|
||||
import chalk from "chalk";
|
||||
import { Box, Text, useInput, useStdin } from "ink";
|
||||
import { EventEmitter } from "node:events";
|
||||
import React, { useRef, useState } from "react";
|
||||
|
||||
/* --------------------------------------------------------------------------
|
||||
* Polyfill missing `ref()` / `unref()` methods on the mock `Stdin` stream
|
||||
* provided by `ink-testing-library`.
|
||||
*
|
||||
* The real `process.stdin` object exposed by Node.js inherits these methods
|
||||
* from `Socket`, but the lightweight stub used in tests only extends
|
||||
* `EventEmitter`. Ink calls the two methods when enabling/disabling raw
|
||||
* mode, so make them harmless no‑ops when they're absent to avoid runtime
|
||||
* failures during unit tests.
|
||||
* ----------------------------------------------------------------------- */
|
||||
|
||||
// Cast through `unknown` ➜ `any` to avoid the `TS2352`/`TS4111` complaints
|
||||
// when augmenting the prototype with the stubbed `ref`/`unref` methods in the
|
||||
// test environment. Using `any` here is acceptable because we purposefully
|
||||
// monkey‑patch internals of Node's `EventEmitter` solely for the benefit of
|
||||
// Ink's stdin stub – type‑safety is not a primary concern at this boundary.
|
||||
//
|
||||
const proto: any = EventEmitter.prototype;
|
||||
|
||||
if (typeof proto["ref"] !== "function") {
|
||||
proto["ref"] = function ref() {};
|
||||
}
|
||||
if (typeof proto["unref"] !== "function") {
|
||||
proto["unref"] = function unref() {};
|
||||
}
|
||||
|
||||
/*
|
||||
* The `ink-testing-library` stub emits only a `data` event when its `stdin`
|
||||
* mock receives `write()` calls. Ink, however, listens for `readable` and
|
||||
* uses the `read()` method to fetch the buffered chunk. Bridge the gap by
|
||||
* hooking into `EventEmitter.emit` so that every `data` emission also:
|
||||
* 1. Buffers the chunk for a subsequent `read()` call, and
|
||||
* 2. Triggers a `readable` event, matching the contract expected by Ink.
|
||||
*/
|
||||
|
||||
// Preserve original emit to avoid infinite recursion.
|
||||
// eslint‑disable‑next‑line @typescript-eslint/no‑unsafe‑assignment
|
||||
const originalEmit = proto["emit"] as (...args: Array<any>) => boolean;
|
||||
|
||||
proto["emit"] = function patchedEmit(
|
||||
this: any,
|
||||
event: string,
|
||||
...args: Array<any>
|
||||
): boolean {
|
||||
if (event === "data") {
|
||||
const chunk = args[0] as string;
|
||||
|
||||
if (
|
||||
process.env["TEXTBUFFER_DEBUG"] === "1" ||
|
||||
process.env["TEXTBUFFER_DEBUG"] === "true"
|
||||
) {
|
||||
// eslint-disable-next-line no-console
|
||||
console.log("[MultilineTextEditor:stdin] data", JSON.stringify(chunk));
|
||||
}
|
||||
// Store carriage returns as‑is so that Ink can distinguish between plain
|
||||
// <Enter> ("\r") and a bare line‑feed ("\n"). This matters because Ink's
|
||||
// `parseKeypress` treats "\r" as key.name === "return", whereas "\n" maps
|
||||
// to "enter" – allowing us to differentiate between plain Enter (submit)
|
||||
// and Shift+Enter (insert newline) inside `useInput`.
|
||||
|
||||
// Identify the lightweight testing stub: lacks `.read()` but exposes
|
||||
// `.setRawMode()` and `isTTY` similar to the real TTY stream.
|
||||
if (
|
||||
!(this as any)._inkIsStub &&
|
||||
typeof (this as any).setRawMode === "function" &&
|
||||
typeof (this as any).isTTY === "boolean" &&
|
||||
typeof (this as any).read !== "function"
|
||||
) {
|
||||
(this as any)._inkIsStub = true;
|
||||
|
||||
// Provide a minimal `read()` shim so Ink can pull queued chunks.
|
||||
(this as any).read = function read() {
|
||||
const ret = (this as any)._inkBuffered ?? null;
|
||||
(this as any)._inkBuffered = null;
|
||||
if (
|
||||
process.env["TEXTBUFFER_DEBUG"] === "1" ||
|
||||
process.env["TEXTBUFFER_DEBUG"] === "true"
|
||||
) {
|
||||
// eslint-disable-next-line no-console
|
||||
console.log("[MultilineTextEditor:stdin.read]", JSON.stringify(ret));
|
||||
}
|
||||
return ret;
|
||||
};
|
||||
}
|
||||
|
||||
if ((this as any)._inkIsStub) {
|
||||
// Buffer the payload so that `read()` can synchronously retrieve it.
|
||||
if (typeof (this as any)._inkBuffered === "string") {
|
||||
(this as any)._inkBuffered += chunk;
|
||||
} else {
|
||||
(this as any)._inkBuffered = chunk;
|
||||
}
|
||||
|
||||
// Notify listeners that data is ready in a way Ink understands.
|
||||
if (
|
||||
process.env["TEXTBUFFER_DEBUG"] === "1" ||
|
||||
process.env["TEXTBUFFER_DEBUG"] === "true"
|
||||
) {
|
||||
// eslint-disable-next-line no-console
|
||||
console.log(
|
||||
"[MultilineTextEditor:stdin] -> readable",
|
||||
JSON.stringify(chunk),
|
||||
);
|
||||
}
|
||||
originalEmit.call(this, "readable");
|
||||
}
|
||||
}
|
||||
|
||||
// Forward the original event.
|
||||
return originalEmit.call(this, event, ...args);
|
||||
};
|
||||
|
||||
export interface MultilineTextEditorProps {
|
||||
// Initial contents.
|
||||
readonly initialText?: string;
|
||||
|
||||
// Visible width.
|
||||
readonly width?: number;
|
||||
|
||||
// Visible height.
|
||||
readonly height?: number;
|
||||
|
||||
// Called when the user submits (plain <Enter> key).
|
||||
readonly onSubmit?: (text: string) => void;
|
||||
|
||||
// Capture keyboard input.
|
||||
readonly focus?: boolean;
|
||||
|
||||
// Called when the internal text buffer updates.
|
||||
readonly onChange?: (text: string) => void;
|
||||
}
|
||||
|
||||
// Expose a minimal imperative API so parent components (e.g. TerminalChatInput)
|
||||
// can query the caret position to implement behaviours like history
|
||||
// navigation that depend on whether the cursor sits on the first/last line.
|
||||
export interface MultilineTextEditorHandle {
|
||||
/** Current caret row */
|
||||
getRow(): number;
|
||||
/** Current caret column */
|
||||
getCol(): number;
|
||||
/** Total number of lines in the buffer */
|
||||
getLineCount(): number;
|
||||
/** Helper: caret is on the very first row */
|
||||
isCursorAtFirstRow(): boolean;
|
||||
/** Helper: caret is on the very last row */
|
||||
isCursorAtLastRow(): boolean;
|
||||
/** Full text contents */
|
||||
getText(): string;
|
||||
}
|
||||
|
||||
const MultilineTextEditorInner = (
|
||||
{
|
||||
initialText = "",
|
||||
// Width can be provided by the caller. When omitted we fall back to the
|
||||
// current terminal size (minus some padding handled by `useTerminalSize`).
|
||||
width,
|
||||
height = 10,
|
||||
onSubmit,
|
||||
focus = true,
|
||||
onChange,
|
||||
}: MultilineTextEditorProps,
|
||||
ref: React.Ref<MultilineTextEditorHandle | null>,
|
||||
): React.ReactElement => {
|
||||
// ---------------------------------------------------------------------------
|
||||
// Editor State
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
const buffer = useRef(new TextBuffer(initialText));
|
||||
const [version, setVersion] = useState(0);
|
||||
|
||||
// Keep track of the current terminal size so that the editor grows/shrinks
|
||||
// with the window. `useTerminalSize` already subtracts a small horizontal
|
||||
// padding so that we don't butt up right against the edge.
|
||||
const terminalSize = useTerminalSize();
|
||||
|
||||
// If the caller didn't specify a width we dynamically choose one based on
|
||||
// the terminal's current column count. We still enforce a reasonable
|
||||
// minimum so that the UI never becomes unusably small.
|
||||
const effectiveWidth = Math.max(20, width ?? terminalSize.columns);
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// External editor integration helpers.
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
// Access to stdin so we can toggle raw‑mode while the external editor is
|
||||
// in control of the terminal.
|
||||
const { stdin, setRawMode } = useStdin();
|
||||
|
||||
/**
|
||||
* Launch the user's preferred $EDITOR, blocking until they close it, then
|
||||
* reload the edited file back into the in‑memory TextBuffer. The heavy
|
||||
* work is delegated to `TextBuffer.openInExternalEditor`, but we are
|
||||
* responsible for temporarily *disabling* raw mode so the child process can
|
||||
* interact with the TTY normally.
|
||||
*/
|
||||
const openExternalEditor = React.useCallback(async () => {
|
||||
// Preserve the current raw‑mode setting so we can restore it afterwards.
|
||||
const wasRaw = stdin?.isRaw ?? false;
|
||||
try {
|
||||
setRawMode?.(false);
|
||||
await buffer.current.openInExternalEditor();
|
||||
} catch (err) {
|
||||
// Surface the error so it doesn't fail silently – for now we log to
|
||||
// stderr. In the future this could surface a toast / overlay.
|
||||
// eslint-disable-next-line no-console
|
||||
console.error("[MultilineTextEditor] external editor error", err);
|
||||
} finally {
|
||||
if (wasRaw) {
|
||||
setRawMode?.(true);
|
||||
}
|
||||
// Force a re‑render so the component reflects the mutated buffer.
|
||||
setVersion((v) => v + 1);
|
||||
}
|
||||
}, [buffer, stdin, setRawMode]);
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Keyboard handling.
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
useInput(
|
||||
(input, key) => {
|
||||
if (!focus) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Single‑step editor shortcut: Ctrl+X or Ctrl+E
|
||||
// Treat both true Ctrl+Key combinations *and* raw control codes so that
|
||||
// the shortcut works consistently in real terminals (raw‑mode) and the
|
||||
// ink‑testing‑library stub which delivers only the raw byte (e.g. 0x05
|
||||
// for Ctrl‑E) without setting `key.ctrl`.
|
||||
const isCtrlX =
|
||||
(key.ctrl && (input === "x" || input === "\x18")) || input === "\x18";
|
||||
const isCtrlE =
|
||||
(key.ctrl && (input === "e" || input === "\x05")) ||
|
||||
input === "\x05" ||
|
||||
(!key.ctrl &&
|
||||
input === "e" &&
|
||||
input.length === 1 &&
|
||||
input.charCodeAt(0) === 5);
|
||||
if (isCtrlX || isCtrlE) {
|
||||
openExternalEditor();
|
||||
return;
|
||||
}
|
||||
|
||||
if (
|
||||
process.env["TEXTBUFFER_DEBUG"] === "1" ||
|
||||
process.env["TEXTBUFFER_DEBUG"] === "true"
|
||||
) {
|
||||
// eslint-disable-next-line no-console
|
||||
console.log("[MultilineTextEditor] event", { input, key });
|
||||
}
|
||||
|
||||
// 1) CSI‑u / modifyOtherKeys (Ink strips initial ESC, so we start with '[')
|
||||
if (input.startsWith("[") && input.endsWith("u")) {
|
||||
const m = input.match(/^\[([0-9]+);([0-9]+)u$/);
|
||||
if (m && m[1] === "13") {
|
||||
const mod = Number(m[2]);
|
||||
// In xterm's encoding: bit‑1 (value 2) is Shift. Everything >1 that
|
||||
// isn't exactly 1 means some modifier was held. We treat *shift
|
||||
// present* (2,4,6,8) as newline; plain (1) as submit.
|
||||
|
||||
// Xterm encodes modifier keys in `mod` – bit‑2 (value 4) indicates
|
||||
// that Ctrl was held. We avoid the `&` bitwise operator (disallowed
|
||||
// by our ESLint config) by using arithmetic instead.
|
||||
const hasCtrl = Math.floor(mod / 4) % 2 === 1;
|
||||
if (hasCtrl) {
|
||||
if (onSubmit) {
|
||||
onSubmit(buffer.current.getText());
|
||||
}
|
||||
} else {
|
||||
// Any variant without Ctrl just inserts newline (Shift, Alt, none)
|
||||
buffer.current.newline();
|
||||
}
|
||||
setVersion((v) => v + 1);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
// 2) Single‑byte control chars ------------------------------------------------
|
||||
if (input === "\n") {
|
||||
// Ctrl+J or pasted newline → insert newline.
|
||||
buffer.current.newline();
|
||||
setVersion((v) => v + 1);
|
||||
return;
|
||||
}
|
||||
|
||||
if (input === "\r") {
|
||||
// Plain Enter – submit (works on all basic terminals).
|
||||
if (onSubmit) {
|
||||
onSubmit(buffer.current.getText());
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
// Let <Esc> fall through so the parent handler (if any) can act on it.
|
||||
|
||||
// Delegate remaining keys to our pure TextBuffer
|
||||
if (
|
||||
process.env["TEXTBUFFER_DEBUG"] === "1" ||
|
||||
process.env["TEXTBUFFER_DEBUG"] === "true"
|
||||
) {
|
||||
// eslint-disable-next-line no-console
|
||||
console.log("[MultilineTextEditor] key event", { input, key });
|
||||
}
|
||||
|
||||
const modified = buffer.current.handleInput(
|
||||
input,
|
||||
key as Record<string, boolean>,
|
||||
{ height, width: effectiveWidth },
|
||||
);
|
||||
if (modified) {
|
||||
setVersion((v) => v + 1);
|
||||
}
|
||||
|
||||
const newText = buffer.current.getText();
|
||||
if (onChange) {
|
||||
onChange(newText);
|
||||
}
|
||||
},
|
||||
{ isActive: focus },
|
||||
);
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Rendering helpers.
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
/* ------------------------------------------------------------------------- */
|
||||
/* Imperative handle – expose a read‑only view of caret & buffer geometry */
|
||||
/* ------------------------------------------------------------------------- */
|
||||
|
||||
React.useImperativeHandle(
|
||||
ref,
|
||||
() => ({
|
||||
getRow: () => buffer.current.getCursor()[0],
|
||||
getCol: () => buffer.current.getCursor()[1],
|
||||
getLineCount: () => buffer.current.getText().split("\n").length,
|
||||
isCursorAtFirstRow: () => buffer.current.getCursor()[0] === 0,
|
||||
isCursorAtLastRow: () => {
|
||||
const [row] = buffer.current.getCursor();
|
||||
const lineCount = buffer.current.getText().split("\n").length;
|
||||
return row === lineCount - 1;
|
||||
},
|
||||
getText: () => buffer.current.getText(),
|
||||
}),
|
||||
[],
|
||||
);
|
||||
|
||||
// Read everything from the buffer
|
||||
const visibleLines = buffer.current.getVisibleLines({
|
||||
height,
|
||||
width: effectiveWidth,
|
||||
});
|
||||
const [cursorRow, cursorCol] = buffer.current.getCursor();
|
||||
const scrollRow = (buffer.current as any).scrollRow as number;
|
||||
const scrollCol = (buffer.current as any).scrollCol as number;
|
||||
|
||||
return (
|
||||
<Box flexDirection="column" key={version}>
|
||||
{visibleLines.map((lineText, idx) => {
|
||||
const absoluteRow = scrollRow + idx;
|
||||
|
||||
// apply horizontal slice
|
||||
let display = lineText.slice(scrollCol, scrollCol + effectiveWidth);
|
||||
if (display.length < effectiveWidth) {
|
||||
display = display.padEnd(effectiveWidth, " ");
|
||||
}
|
||||
|
||||
// Highlight the *character under the caret* (i.e. the one immediately
|
||||
// to the right of the insertion position) so that the block cursor
|
||||
// visually matches the logical caret location. This makes the
|
||||
// highlighted glyph the one that would be replaced by `insert()` and
|
||||
// *not* the one that would be removed by `backspace()`.
|
||||
|
||||
if (absoluteRow === cursorRow) {
|
||||
const relativeCol = cursorCol - scrollCol;
|
||||
const highlightCol = relativeCol;
|
||||
|
||||
if (highlightCol >= 0 && highlightCol < effectiveWidth) {
|
||||
const charToHighlight = display[highlightCol] || " ";
|
||||
const highlighted = chalk.inverse(charToHighlight);
|
||||
display =
|
||||
display.slice(0, highlightCol) +
|
||||
highlighted +
|
||||
display.slice(highlightCol + 1);
|
||||
} else if (relativeCol === effectiveWidth) {
|
||||
// Caret sits just past the right edge; show a block cursor in the
|
||||
// gutter so the user still sees it.
|
||||
display = display.slice(0, effectiveWidth - 1) + chalk.inverse(" ");
|
||||
}
|
||||
}
|
||||
|
||||
return <Text key={idx}>{display}</Text>;
|
||||
})}
|
||||
</Box>
|
||||
);
|
||||
};
|
||||
|
||||
const MultilineTextEditor = React.forwardRef(MultilineTextEditorInner);
|
||||
|
||||
export default MultilineTextEditor;
|
||||
Reference in New Issue
Block a user