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;
}
方法描述
render(width)返回字符串数组(每行一个)。每行不能超过 width
handleInput?(data)组件获得焦点时接收键盘输入。
wantsKeyRelease?如果为 true,组件接收按键释放事件(Kitty 协议)。默认:false。
invalidate()清除缓存的渲染状态。主题变更时调用。

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:

  1. 在组件上设置 focused = true
  2. 在渲染输出中扫描 CURSOR_MARKER(零宽度 APC 转义序列)
  3. 将硬件终端光标定位到该位置
  4. 仅在 showHardwareCursor 启用时显示硬件光标

光标默认隐藏。这保留了假光标渲染,同时仍然为通过隐藏光标追踪 IME 候选窗口的终端定位硬件光标。某些终端需要可见的硬件光标来进行 IME 定位;可以通过 showHardwareCursorsetShowHardwareCursor(true)PI_HARDWARE_CURSOR=1 启用。

包含嵌入式输入的容器组件

当容器组件(对话框、选择器等)包含 InputEditor 子组件时,容器必须实现 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.focus() - 聚焦此覆盖层并将其带到视觉最前
    // handle.unfocus() - 将输入释放回正常回退
    // handle.unfocus({ target }) - 将输入释放给指定组件或 null
    // handle.setHidden(true/false) - 切换可见性
    // handle.hide() - 永久移除
  },
});

覆盖层焦点

聚焦且可见的覆盖层可在临时非覆盖层 UI 期间持续持有输入所有权。如果覆盖层打开了另一个未传 { overlay: true }ctx.ui.custom() 组件,则该替换 UI 在活跃期间会接收输入;待其关闭后,聚焦的覆盖层可重新获取输入。

当你希望可见的覆盖层停止持有输入、并让 TUI 回退至其他可见的捕获型覆盖层或上一焦点目标时,使用 handle.unfocus()。当你希望特定组件在覆盖层保持可见期间接收输入时,使用 handle.unfocus({ target })。传入 { target: null } 会在显式再次设置焦点前有意保留无组件被聚焦的状态。

覆盖层生命周期

覆盖层组件在关闭时被释放。不要重用引用——创建新实例:

// 错误 - 过时的引用
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、Warp)。

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.enterKey.escapeKey.tabKey.spaceKey.backspaceKey.deleteKey.homeKey.end
  • 方向键:Key.upKey.downKey.leftKey.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)):

分类颜色
Generaltextaccentmuteddim
Statussuccesserrorwarning
BordersborderborderAccentborderMuted
MessagesuserMessageTextcustomMessageTextcustomMessageLabel
ToolstoolTitletoolOutput
DiffstoolDiffAddedtoolDiffRemovedtoolDiffContext
MarkdownmdHeadingmdLinkmdLinkUrlmdCodemdCodeBlockmdCodeBlockBordermdQuotemdQuoteBordermdHrmdListBullet
SyntaxsyntaxCommentsyntaxKeywordsyntaxFunctionsyntaxVariablesyntaxStringsyntaxNumbersyntaxTypesyntaxOperatorsyntaxPunctuation
ThinkingthinkingOffthinkingMinimalthinkingLowthinkingMediumthinkingHighthinkingXhigh
ModesbashMode

背景色theme.bg(color, text)):

selectedBguserMessageBgcustomMessageBgtoolPendingBgtoolSuccessBgtoolErrorBg

对于 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();
  }
}

何时需要此模式

以下情况需要此模式:

  1. 预烘焙主题颜色 - 使用 theme.fg()theme.bg() 创建存储在子组件中的样式化字符串
  2. 语法高亮 - 使用 highlightCode(),其应用基于主题的语法颜色
  3. 复杂布局 - 构建嵌入了主题颜色的子组件树

以下情况不需要此模式:

  1. 使用主题回调 - 传递像 (text) => theme.fg("accent", text) 这样在渲染期间被调用的函数
  2. 简单容器 - 仅对其他组件进行分组,不添加主题化内容
  3. 无状态渲染 - 在每次 render() 调用中重新计算主题化输出(无缓存)

常见模式

这些模式涵盖了扩展中最常见的 UI 需求。复制这些模式而非从头构建。

模式 1:选择对话框(SelectList)

用于让用户从选项列表中选择。使用 @earendil-works/pi-tuiSelectList,配合 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\u2193 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.tstools.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.tshandoff.ts

模式 3:设置/切换(SettingsList)

用于切换多项设置。使用 @earendil-works/pi-tuiSettingsList,配合 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.tsplan-mode/index.tspreset.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/index.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 接收一个工厂函数,该函数获取 tuithemekeybindings
  • 传递 undefined 以恢复默认编辑器:ctx.ui.setEditorComponent(undefined)

示例: modal-editor.ts

关键规则

  1. 始终使用回调中的 theme - 不要直接导入主题。使用 ctx.ui.custom((tui, theme, keybindings, done) => ...) 回调中的 theme

  2. 始终为 DynamicBorder 颜色参数指定类型 - 写 (s: string) => theme.fg("accent", s),而不是 (s) => theme.fg("accent", s)

  3. 状态变化后调用 tui.requestRender() - 在 handleInput 中,更新状态后调用 tui.requestRender()

  4. 返回三方法对象 - 自定义组件需要 { render, invalidate, handleInput }

  5. 使用现有组件 - SelectListSettingsListBorderedLoader 覆盖了 90% 的情况。不要重复构建它们。

示例


法律声明:本页面是 pi.dev 官方文档的中文翻译版本,仅供学习参考。本网站与 pi.dev 及 Earendil Inc. 无任何法律关系。