AgentSkillsCN

inkjs-design

Ink.js(适用于 CLI 的 React)设计与实现指南。 适用场景: (1) 创建或修改 Ink.js 组件; (2) 实现 Ink 特有的自定义 Hook(如 useInput、useApp、useFocus); (3) 处理表情符号/图标宽度相关问题(通过 string-width 提供的变通方案); (4) 构建对终端设备响应灵敏的布局; (5) 管理多屏导航; (6) 实现动画效果(如加载 spinner、进度条); (7) 优化性能(例如使用 React.memo 和 useMemo); (8) 处理键盘输入与快捷键操作; (9) 测试 CLI 用户界面(ink-testing-library)。

SKILL.md
--- frontmatter
name: inkjs-design
description: |
  Ink.js (React for CLI) design and implementation guide.
  Use when:
  (1) Creating or modifying Ink.js components
  (2) Implementing Ink-specific hooks (useInput, useApp, useFocus)
  (3) Handling emoji/icon width issues (string-width workarounds)
  (4) Building terminal-responsive layouts
  (5) Managing multi-screen navigation
  (6) Implementing animations (spinners, progress bars)
  (7) Optimizing performance (React.memo, useMemo)
  (8) Handling keyboard input and shortcuts
  (9) Testing CLI UI (ink-testing-library)

Ink.js Design

Comprehensive guide for building terminal UIs with Ink.js (React for CLI).

Quick Start

Creating a New Component

  1. Determine component type: Screen / Part / Common
  2. Reference component-patterns.md for similar patterns
  3. Add type definitions
  4. Implement component
  5. Write tests

Common Issues & Solutions

IssueReference
Emoji width misalignmentink-gotchas.md
Ctrl+C called twiceink-gotchas.md
useInput conflictsink-gotchas.md
Layout breakingresponsive-layout.md
Screen navigationmulti-screen-navigation.md

Directory Conventions

code
src/cli/ui/
├── components/
│   ├── App.tsx              # Root component with screen management
│   ├── common/              # Common input components (Select, Input)
│   ├── parts/               # Reusable UI parts (Header, Footer)
│   └── screens/             # Full-screen components
├── hooks/                   # Custom hooks
├── utils/                   # Utility functions
└── types.ts                 # Type definitions

Component Classification

Screen (Full-page views)

  • Represents a complete screen/page
  • Handles keyboard input via useInput
  • Implements Header/Content/Footer layout
  • Manages screen-level state

Part (Reusable elements)

  • Reusable UI building blocks
  • Optimized with React.memo
  • Stateless/pure components preferred
  • Accept configuration via props

Common (Input components)

  • Basic input components
  • Support both controlled and uncontrolled modes
  • Handle focus management
  • Provide consistent UX

Essential Patterns

1. Icon Width Override

Fix string-width v8 emoji width calculation issues:

typescript
const WIDTH_OVERRIDES: Record<string, number> = {
  "⚡": 1, "✨": 1, "🐛": 1, "🔥": 1, "🚀": 1,
  "🟢": 1, "🟠": 1, "✅": 1, "⚠️": 1,
};

const getIconWidth = (icon: string): number => {
  const baseWidth = stringWidth(icon);
  const override = WIDTH_OVERRIDES[icon];
  return override !== undefined ? Math.max(baseWidth, override) : baseWidth;
};

2. useInput Conflict Avoidance

Multiple useInput hooks all fire - use early return or isActive:

typescript
useInput((input, key) => {
  if (disabled) return;  // Early return when inactive
  // Handle input...
}, { isActive: isFocused });

3. Ctrl+C Handling

typescript
render(<App />, { exitOnCtrlC: false });

// In component
const { exit } = useApp();
useInput((input, key) => {
  if (key.ctrl && input === "c") {
    cleanup();
    exit();
  }
});

4. Dynamic Height Calculation

typescript
const { rows } = useTerminalSize();
const HEADER_LINES = 3;
const FOOTER_LINES = 2;
const contentHeight = rows - HEADER_LINES - FOOTER_LINES;
const visibleItems = Math.max(5, contentHeight);

5. React.memo with Custom Comparator

typescript
function arePropsEqual<T>(prev: Props<T>, next: Props<T>): boolean {
  if (prev.items.length !== next.items.length) return false;
  for (let i = 0; i < prev.items.length; i++) {
    if (prev.items[i].value !== next.items[i].value) return false;
  }
  return prev.selectedIndex === next.selectedIndex;
}

export const Select = React.memo(SelectComponent, arePropsEqual);

6. Multi-Screen Navigation

typescript
type ScreenType = "main" | "detail" | "settings";

const [screenStack, setScreenStack] = useState<ScreenType[]>(["main"]);
const currentScreen = screenStack[screenStack.length - 1];

const navigateTo = (screen: ScreenType) => {
  setScreenStack(prev => [...prev, screen]);
};

const goBack = () => {
  if (screenStack.length > 1) {
    setScreenStack(prev => prev.slice(0, -1));
  }
};

Detailed References

Core Patterns

Advanced Topics

Troubleshooting

Examples

See examples/ for practical implementation examples.