TUI Components(TUI 组件)
本页面是 Pi 官方文档 的中文翻译。仅供学习参考。
pi 可以创建 TUI 组件。让它为你的用例构建一个。
扩展和自定义工具可以渲染自定义 TUI 组件以构建交互式用户界面。本页面介绍组件系统和可用的构建块。
来源: @earendil-works/pi-tui
组件接口
所有组件实现:
interface Component {
render(width: number): string[];
handleInput?(data: string): void;
wantsKeyRelease?: boolean;
invalidate(): void;
}
TUI 会在每行渲染内容的末尾附加完整的 SGR 重置和 OSC 8 重置。样式不会跨行延续。如果你输出带样式的多行文本,需要每行重新应用样式,或使用 wrapTextWithAnsi() 来确保每个换行行的样式得以保留。
Focusable 接口(IME 支持)
需要显示文本光标和支持 IME(输入法编辑器)的组件应实现 Focusable 接口:
import { CURSOR_MARKER, type Component, type Focusable } from "@earendil-works/pi-tui";
class MyInput implements Component, Focusable {
focused: boolean = false; // 由 TUI 在焦点变化时设置
render(width: number): string[] {
const marker = this.focused ? CURSOR_MARKER : "";
// 在假光标前发出 marker
return [`> ${beforeCursor}${marker}\x1b[7m${atCursor}\x1b[27m${afterCursor}`];
}
}
当 Focusable 组件获得焦点时,TUI:
- 在组件上设置
focused = true
- 在渲染输出中扫描
CURSOR_MARKER(零宽度 APC 转义序列)
- 将硬件终端光标定位到该位置
- 显示硬件光标
这使得 IME 候选窗口能够出现在正确位置,适用于中日韩输入法。内置的 Editor 和 Input 组件已实现此接口。
包含嵌入式输入的容器组件
当容器组件(对话框、选择器等)包含 Input 或 Editor 子组件时,容器必须实现 Focusable 并将焦点状态传播给子组件。否则,硬件光标无法正确定位以支持 IME 输入。
import { Container, type Focusable, Input } from "@earendil-works/pi-tui";
class SearchDialog extends Container implements Focusable {
private searchInput: Input;
// Focusable 实现 - 传播到子输入以支持 IME 光标定位
private _focused = false;
get focused(): boolean {
return this._focused;
}
set focused(value: boolean) {
this._focused = value;
this.searchInput.focused = value;
}
constructor() {
super();
this.searchInput = new Input();
this.addChild(this.searchInput);
}
}
如果没有这种传播,使用 IME(中文、日文、韩文等)输入时,候选窗口将出现在屏幕上的错误位置。
使用组件
在扩展中通过 ctx.ui.custom():
pi.on("session_start", async (_event, ctx) => {
const handle = ctx.ui.custom(myComponent);
// handle.requestRender() - 触发重新渲染
// handle.close() - 恢复正常 UI
});
在自定义工具中通过 pi.ui.custom():
async execute(toolCallId, params, onUpdate, ctx, signal) {
const handle = pi.ui.custom(myComponent);
// ...
handle.close();
}
覆盖层(Overlays)
覆盖层在现有内容之上渲染组件,无需清屏。向 ctx.ui.custom() 传递 { overlay: true }:
const result = await ctx.ui.custom<string | null>(
(tui, theme, keybindings, done) => new MyDialog({ onClose: done }),
{ overlay: true }
);
定位和尺寸使用 overlayOptions:
const result = await ctx.ui.custom<string | null>(
(tui, theme, keybindings, done) => new SidePanel({ onClose: done }),
{
overlay: true,
overlayOptions: {
// 尺寸:数字或百分比字符串
width: "50%", // 终端宽度的 50%
minWidth: 40, // 最小 40 列
maxHeight: "80%", // 最大终端高度的 80%
// 位置:基于锚点(默认:"center")
anchor: "right-center", // 9 个位置:center、top-left、top-center 等
offsetX: -2, // 从锚点偏移
offsetY: 0,
// 或百分比/绝对定位
row: "25%", // 距离顶部 25%
col: 10, // 第 10 列
// 边距
margin: 2, // 所有边,或 { top, right, bottom, left }
// 响应式:在窄终端上隐藏
visible: (termWidth, termHeight) => termWidth >= 80,
},
// 获取用于程序化控制可见性的句柄
onHandle: (handle) => {
// handle.setHidden(true/false) - 切换可见性
// handle.hide() - 永久移除
},
}
);
覆盖层生命周期
覆盖层组件在关闭时被释放。不要重用引用——创建新实例:
// 错误 - 过时的引用
let menu: MenuComponent;
await ctx.ui.custom((_, __, ___, done) => {
menu = new MenuComponent(done);
return menu;
}, { overlay: true });
setActiveComponent(menu); // 已释放
// 正确 - 重新调用以重新显示
const showMenu = () => ctx.ui.custom((_, __, ___, done) =>
new MenuComponent(done), { overlay: true });
await showMenu(); // 第一次显示
await showMenu(); // "返回" = 再次调用即可
参见 overlay-qa-tests.ts 获取覆盖锚点、边距、堆叠、响应式可见性和动画的全面示例。
内置组件
从 @earendil-works/pi-tui 导入:
import { Text, Box, Container, Spacer, Markdown } from "@earendil-works/pi-tui";
Text
支持自动换行的多行文本。
const text = new Text(
"Hello World", // 内容
1, // 水平内边距(默认:1)
1, // 垂直内边距(默认:1)
(s) => bgGray(s) // 可选背景函数
);
text.setText("Updated");
Box
带内边距和背景色的容器。
const box = new Box(
1, // 水平内边距
1, // 垂直内边距
(s) => bgGray(s) // 背景函数
);
box.addChild(new Text("Content", 0, 0));
box.setBgFn((s) => bgBlue(s));
Container
垂直分组子组件。
const container = new Container();
container.addChild(component1);
container.addChild(component2);
container.removeChild(component1);
Spacer
空垂直空间。
const spacer = new Spacer(2); // 2 个空行
Markdown
渲染带语法高亮的 Markdown。
const md = new Markdown(
"# Title\n\nSome **bold** text",
1, // 水平内边距
1, // 垂直内边距
theme // MarkdownTheme(见下文)
);
md.setText("Updated markdown");
Image
在支持的终端中渲染图像(Kitty、iTerm2、Ghostty、WezTerm)。
const image = new Image(
base64Data, // base64 编码的图像数据
"image/png", // MIME 类型
theme, // ImageTheme
{ maxWidthCells: 80, maxHeightCells: 24 }
);
键盘输入
使用 matchesKey() 检测按键:
import { matchesKey, Key } from "@earendil-works/pi-tui";
handleInput(data: string) {
if (matchesKey(data, Key.up)) {
this.selectedIndex--;
} else if (matchesKey(data, Key.enter)) {
this.onSelect?.(this.selectedIndex);
} else if (matchesKey(data, Key.escape)) {
this.onCancel?.();
} else if (matchesKey(data, Key.ctrl("c"))) {
// Ctrl+C
}
}
Key 标识符(使用 Key.* 获取自动补全,或使用字符串字面量):
- 基本键:
Key.enter、Key.escape、Key.tab、Key.space、Key.backspace、Key.delete、Key.home、Key.end
- 方向键:
Key.up、Key.down、Key.left、Key.right
- 带修饰键:
Key.ctrl("c")、Key.shift("tab")、Key.alt("left")、Key.ctrlShift("p")
- 字符串格式同样有效:
"enter"、"ctrl+c"、"shift+tab"、"ctrl+shift+p"
行宽
关键规则: render() 返回的每行不能超过 width 参数。
import { visibleWidth, truncateToWidth } from "@earendil-works/pi-tui";
render(width: number): string[] {
// 截断长行
return [truncateToWidth(this.text, width)];
}
实用工具:
visibleWidth(str) - 获取显示宽度(忽略 ANSI 码)
truncateToWidth(str, width, ellipsis?) - 以可选省略号截断
wrapTextWithAnsi(str, width) - 保留 ANSI 码的自动换行
创建自定义组件
示例:交互式选择器
import {
matchesKey, Key,
truncateToWidth, visibleWidth
} from "@earendil-works/pi-tui";
class MySelector {
private items: string[];
private selected = 0;
private cachedWidth?: number;
private cachedLines?: string[];
public onSelect?: (item: string) => void;
public onCancel?: () => void;
constructor(items: string[]) {
this.items = items;
}
handleInput(data: string): void {
if (matchesKey(data, Key.up) && this.selected > 0) {
this.selected--;
this.invalidate();
} else if (matchesKey(data, Key.down) && this.selected < this.items.length - 1) {
this.selected++;
this.invalidate();
} else if (matchesKey(data, Key.enter)) {
this.onSelect?.(this.items[this.selected]);
} else if (matchesKey(data, Key.escape)) {
this.onCancel?.();
}
}
render(width: number): string[] {
if (this.cachedLines && this.cachedWidth === width) {
return this.cachedLines;
}
this.cachedLines = this.items.map((item, i) => {
const prefix = i === this.selected ? "> " : " ";
return truncateToWidth(prefix + item, width);
});
this.cachedWidth = width;
return this.cachedLines;
}
invalidate(): void {
this.cachedWidth = undefined;
this.cachedLines = undefined;
}
}
在扩展中使用:
pi.registerCommand("pick", {
description: "Pick an item",
handler: async (args, ctx) => {
const items = ["Option A", "Option B", "Option C"];
const selector = new MySelector(items);
let handle: { close: () => void; requestRender: () => void };
await new Promise<void>((resolve) => {
selector.onSelect = (item) => {
ctx.ui.notify(`Selected: ${item}`, "info");
handle.close();
resolve();
};
selector.onCancel = () => {
handle.close();
resolve();
};
handle = ctx.ui.custom(selector);
});
}
});
主题
组件接受主题对象进行样式设置。
在 renderCall/renderResult 中,使用 theme 参数:
renderResult(result, options, theme, context) {
// 使用 theme.fg() 设置前景色
return new Text(theme.fg("success", "Done!"), 0, 0);
// 使用 theme.bg() 设置背景色
const styled = theme.bg("toolPendingBg", theme.fg("accent", "text"));
}
前景色(theme.fg(color, text)):
背景色(theme.bg(color, text)):
selectedBg、userMessageBg、customMessageBg、toolPendingBg、toolSuccessBg、toolErrorBg
对于 Markdown,使用 getMarkdownTheme():
import { getMarkdownTheme } from "@earendil-works/pi-coding-agent";
import { Markdown } from "@earendil-works/pi-tui";
renderResult(result, options, theme, context) {
const mdTheme = getMarkdownTheme();
return new Markdown(result.details.markdown, 0, 0, mdTheme);
}
对于自定义组件,定义自己的主题接口:
interface MyTheme {
selected: (s: string) => string;
normal: (s: string) => string;
}
调试日志
设置 PI_TUI_WRITE_LOG 以捕获写入 stdout 的原始 ANSI 流。
PI_TUI_WRITE_LOG=/tmp/tui-ansi.log npx tsx packages/tui/test/chat-simple.ts
性能
尽可能缓存渲染输出:
class CachedComponent {
private cachedWidth?: number;
private cachedLines?: string[];
render(width: number): string[] {
if (this.cachedLines && this.cachedWidth === width) {
return this.cachedLines;
}
// ... 计算行 ...
this.cachedWidth = width;
this.cachedLines = lines;
return lines;
}
invalidate(): void {
this.cachedWidth = undefined;
this.cachedLines = undefined;
}
}
状态变化时调用 invalidate(),然后调用 handle.requestRender() 触发重新渲染。
无效化和主题变更
当主题发生变更时,TUI 会在所有组件上调用 invalidate() 以清除其缓存。组件必须正确实现 invalidate() 以确保主题变更生效。
问题
如果组件通过 theme.fg()、theme.bg() 等将主题颜色预烘焙到字符串中并缓存,缓存的字符串包含旧主题的 ANSI 转义码。仅清除渲染缓存是不够的,如果组件单独存储了主题化内容的话。
错误方法(主题颜色不会更新):
class BadComponent extends Container {
private content: Text;
constructor(message: string, theme: Theme) {
super();
// 预烘焙的主题颜色存储在 Text 组件中
this.content = new Text(theme.fg("accent", message), 1, 0);
this.addChild(this.content);
}
// 没有 invalidate 覆盖 - 父级的 invalidate 仅清除
// 子渲染缓存,不清除预烘焙的内容
}
解决方案
使用主题颜色构建内容的组件必须在 invalidate() 被调用时重建该内容:
class GoodComponent extends Container {
private message: string;
private content: Text;
constructor(message: string) {
super();
this.message = message;
this.content = new Text("", 1, 0);
this.addChild(this.content);
this.updateDisplay();
}
private updateDisplay(): void {
// 使用当前主题重建内容
this.content.setText(theme.fg("accent", this.message));
}
override invalidate(): void {
super.invalidate(); // 清除子缓存
this.updateDisplay(); // 使用新主题重建
}
}
模式:在 Invalidate 时重建
对于包含复杂内容的组件:
class ComplexComponent extends Container {
private data: SomeData;
constructor(data: SomeData) {
super();
this.data = data;
this.rebuild();
}
private rebuild(): void {
this.clear(); // 移除所有子组件
// 使用当前主题构建 UI
this.addChild(new Text(theme.fg("accent", theme.bold("Title")), 1, 0));
this.addChild(new Spacer(1));
for (const item of this.data.items) {
const color = item.active ? "success" : "muted";
this.addChild(new Text(theme.fg(color, item.label), 1, 0));
}
}
override invalidate(): void {
super.invalidate();
this.rebuild();
}
}
何时需要此模式
以下情况需要此模式:
- 预烘焙主题颜色 - 使用
theme.fg() 或 theme.bg() 创建存储在子组件中的样式化字符串
- 语法高亮 - 使用
highlightCode(),其应用基于主题的语法颜色
- 复杂布局 - 构建嵌入了主题颜色的子组件树
以下情况不需要此模式:
- 使用主题回调 - 传递像
(text) => theme.fg("accent", text) 这样在渲染期间被调用的函数
- 简单容器 - 仅对其他组件进行分组,不添加主题化内容
- 无状态渲染 - 在每次
render() 调用中重新计算主题化输出(无缓存)
常见模式
这些模式涵盖了扩展中最常见的 UI 需求。复制这些模式而非从头构建。
模式 1:选择对话框(SelectList)
用于让用户从选项列表中选择。使用 @earendil-works/pi-tui 的 SelectList,配合 DynamicBorder 进行边框装饰。
import type { ExtensionAPI } from "@earendil-works/pi-coding-agent";
import { DynamicBorder } from "@earendil-works/pi-coding-agent";
import { Container, type SelectItem, SelectList, Text } from "@earendil-works/pi-tui";
pi.registerCommand("pick", {
handler: async (_args, ctx) => {
const items: SelectItem[] = [
{ value: "opt1", label: "Option 1", description: "First option" },
{ value: "opt2", label: "Option 2", description: "Second option" },
{ value: "opt3", label: "Option 3" }, // description 可选
];
const result = await ctx.ui.custom<string | null>((tui, theme, _kb, done) => {
const container = new Container();
// 顶部边框
container.addChild(new DynamicBorder((s: string) => theme.fg("accent", s)));
// 标题
container.addChild(new Text(theme.fg("accent", theme.bold("Pick an Option")), 1, 0));
// 带主题的 SelectList
const selectList = new SelectList(items, Math.min(items.length, 10), {
selectedPrefix: (t) => theme.fg("accent", t),
selectedText: (t) => theme.fg("accent", t),
description: (t) => theme.fg("muted", t),
scrollInfo: (t) => theme.fg("dim", t),
noMatch: (t) => theme.fg("warning", t),
});
selectList.onSelect = (item) => done(item.value);
selectList.onCancel = () => done(null);
container.addChild(selectList);
// 帮助文本
container.addChild(new Text(theme.fg("dim", "\u2191\u2195 navigate \u2022 enter select \u2022 esc cancel"), 1, 0));
// 底部边框
container.addChild(new DynamicBorder((s: string) => theme.fg("accent", s)));
return {
render: (w) => container.render(w),
invalidate: () => container.invalidate(),
handleInput: (data) => { selectList.handleInput(data); tui.requestRender(); },
};
});
if (result) {
ctx.ui.notify(`Selected: ${result}`, "info");
}
},
});
示例: preset.ts、tools.ts
模式 2:带取消的异步操作(BorderedLoader)
用于需要时间且应可取消的操作。BorderedLoader 显示旋转器并支持 escape 取消。
import { BorderedLoader } from "@earendil-works/pi-coding-agent";
pi.registerCommand("fetch", {
handler: async (_args, ctx) => {
const result = await ctx.ui.custom<string | null>((tui, theme, _kb, done) => {
const loader = new BorderedLoader(tui, theme, "Fetching data...");
loader.onAbort = () => done(null);
// 执行异步工作
fetchData(loader.signal)
.then((data) => done(data))
.catch(() => done(null));
return loader;
});
if (result === null) {
ctx.ui.notify("Cancelled", "info");
} else {
ctx.ui.setEditorText(result);
}
},
});
示例: qna.ts、handoff.ts
模式 3:设置/切换(SettingsList)
用于切换多项设置。使用 @earendil-works/pi-tui 的 SettingsList,配合 getSettingsListTheme()。
import { getSettingsListTheme } from "@earendil-works/pi-coding-agent";
import { Container, type SettingItem, SettingsList, Text } from "@earendil-works/pi-tui";
pi.registerCommand("settings", {
handler: async (_args, ctx) => {
const items: SettingItem[] = [
{ id: "verbose", label: "Verbose mode", currentValue: "off", values: ["on", "off"] },
{ id: "color", label: "Color output", currentValue: "on", values: ["on", "off"] },
];
await ctx.ui.custom((_tui, theme, _kb, done) => {
const container = new Container();
container.addChild(new Text(theme.fg("accent", theme.bold("Settings")), 1, 1));
const settingsList = new SettingsList(
items,
Math.min(items.length + 2, 15),
getSettingsListTheme(),
(id, newValue) => {
// 处理值变更
ctx.ui.notify(`${id} = ${newValue}`, "info");
},
() => done(undefined), // 关闭时
{ enableSearch: true }, // 可选:启用按标签的模糊搜索
);
container.addChild(settingsList);
return {
render: (w) => container.render(w),
invalidate: () => container.invalidate(),
handleInput: (data) => settingsList.handleInput?.(data),
};
});
},
});
示例: tools.ts
模式 4:持久状态指示器
在底部显示跨渲染保持的状态。适用于模式指示器。
// 设置状态(显示在底部)
ctx.ui.setStatus("my-ext", ctx.ui.theme.fg("accent", "\u25cf active"));
// 清除状态
ctx.ui.setStatus("my-ext", undefined);
示例: status-line.ts、plan-mode.ts、preset.ts
模式 4b:工作指示器自定义
自定义 pi 流式响应时显示的内联工作指示器。
// 静态指示器
ctx.ui.setWorkingIndicator({ frames: [ctx.ui.theme.fg("accent", "\u25cf")] });
// 自定义动画指示器
ctx.ui.setWorkingIndicator({
frames: [
ctx.ui.theme.fg("dim", "\u00b7"),
ctx.ui.theme.fg("muted", "\u2022"),
ctx.ui.theme.fg("accent", "\u25cf"),
ctx.ui.theme.fg("muted", "\u2022"),
],
intervalMs: 120,
});
// 完全隐藏指示器
ctx.ui.setWorkingIndicator({ frames: [] });
// 恢复 pi 默认旋转器
ctx.ui.setWorkingIndicator();
这仅影响正常的流式工作指示器。压缩和重试加载器保留其内置样式。自定义帧会逐字渲染,因此扩展需要在需要时自行添加颜色。
示例: working-indicator.ts
模式 5:编辑器上方/下方的小部件
在输入编辑器上方或下方显示持久内容。适用于待办列表、进度。
// 简单字符串数组(默认在编辑器上方)
ctx.ui.setWidget("my-widget", ["Line 1", "Line 2"]);
// 在编辑器下方渲染
ctx.ui.setWidget("my-widget", ["Line 1", "Line 2"], { placement: "belowEditor" });
// 或带主题
ctx.ui.setWidget("my-widget", (_tui, theme) => {
const lines = items.map((item, i) =>
item.done
? theme.fg("success", "\u2713 ") + theme.fg("muted", item.text)
: theme.fg("dim", "\u25cb ") + item.text
);
return {
render: () => lines,
invalidate: () => {},
};
});
// 清除
ctx.ui.setWidget("my-widget", undefined);
示例: plan-mode.ts
模式 6:自定义底部
替换底部。footerData 暴露了扩展无法通过其他方式访问的数据。
ctx.ui.setFooter((tui, theme, footerData) => ({
invalidate() {},
render(width: number): string[] {
// footerData.getGitBranch(): string | null
// footerData.getExtensionStatuses(): ReadonlyMap<string, string>
return [`${ctx.model?.id} (${footerData.getGitBranch() || "no git"})`];
},
dispose: footerData.onBranchChange(() => tui.requestRender()), // 响应式
}));
ctx.ui.setFooter(undefined); // 恢复默认
Token 统计可通过 ctx.sessionManager.getBranch() 和 ctx.model 获取。
示例: custom-footer.ts
模式 7:自定义编辑器(vim 模式等)
用自定义实现替换主输入编辑器。适用于模态编辑(vim)、不同键绑定(emacs)或特殊输入处理。
import { CustomEditor, type ExtensionAPI } from "@earendil-works/pi-coding-agent";
import { matchesKey, truncateToWidth } from "@earendil-works/pi-tui";
type Mode = "normal" | "insert";
class VimEditor extends CustomEditor {
private mode: Mode = "insert";
handleInput(data: string): void {
// Escape:切换到普通模式,或传递给应用处理
if (matchesKey(data, "escape")) {
if (this.mode === "insert") {
this.mode = "normal";
return;
}
// 普通模式下,escape 中止 Agent(由 CustomEditor 处理)
super.handleInput(data);
return;
}
// 插入模式:传递所有内容给 CustomEditor
if (this.mode === "insert") {
super.handleInput(data);
return;
}
// 普通模式:vim 风格导航
switch (data) {
case "i": this.mode = "insert"; return;
case "h": super.handleInput("\x1b[D"); return; // 左
case "j": super.handleInput("\x1b[B"); return; // 下
case "k": super.handleInput("\x1b[A"); return; // 上
case "l": super.handleInput("\x1b[C"); return; // 右
}
// 将未处理的键传递给 super(ctrl+c 等),但过滤可打印字符
if (data.length === 1 && data.charCodeAt(0) >= 32) return;
super.handleInput(data);
}
render(width: number): string[] {
const lines = super.render(width);
// 在底部边框添加模式指示器(使用 truncateToWidth 进行 ANSI 安全截断)
if (lines.length > 0) {
const label = this.mode === "normal" ? " NORMAL " : " INSERT ";
const lastLine = lines[lines.length - 1]!;
// 传入 "" 作为省略号以避免在截断时添加 "..."
lines[lines.length - 1] = truncateToWidth(lastLine, width - label.length, "") + label;
}
return lines;
}
}
export default function (pi: ExtensionAPI) {
pi.on("session_start", (_event, ctx) => {
// 工厂接收来自应用的主题和键绑定
ctx.ui.setEditorComponent((tui, theme, keybindings) =>
new VimEditor(theme, keybindings)
);
});
}
关键点:
- 扩展
CustomEditor(而非基础 Editor)以获取应用键绑定(escape 中止、ctrl+d 退出、模型切换等)
- 对于未处理的键,调用
super.handleInput(data)
- 工厂模式:
setEditorComponent 接收一个工厂函数,该函数获取 tui、theme 和 keybindings
- 传递
undefined 以恢复默认编辑器:ctx.ui.setEditorComponent(undefined)
示例: modal-editor.ts
关键规则
-
始终使用回调中的 theme - 不要直接导入主题。使用 ctx.ui.custom((tui, theme, keybindings, done) => ...) 回调中的 theme。
-
始终为 DynamicBorder 颜色参数指定类型 - 写 (s: string) => theme.fg("accent", s),而不是 (s) => theme.fg("accent", s)。
-
状态变化后调用 tui.requestRender() - 在 handleInput 中,更新状态后调用 tui.requestRender()。
-
返回三方法对象 - 自定义组件需要 { render, invalidate, handleInput }。
-
使用现有组件 - SelectList、SettingsList、BorderedLoader 覆盖了 90% 的情况。不要重复构建它们。
示例
法律声明:本页面是 pi.dev 官方文档的中文翻译版本,仅供学习参考。本网站与 pi.dev 及 Earendil Inc. 无任何法律关系。