Files
claude-code/packages/@ant/ink/docs/01-getting-started.md
claude-code-best 2fb1c9dcd8 feat: 工具层及 mcp 大重构 (#252)
* feat: 第一版大重构

* fix: 修复类型问题

* chore: 更新版本到 1.3.2

* Add brave as alternative WebSearchTool

* fix: 修正顺序

* fix: 修复对穷鬼模式的 auto dream 和 session memory 越过

* feat: 穷鬼模式去除 session-summary

* feat: 创建 builtin-tools 包,搬运所有工具实现

将 src/tools/ 下的全部 60 个工具目录迁移至 packages/builtin-tools/src/tools/,
内部导入路径已更新为 src/ alias 模式。

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* refactor: 更新 src/ 中所有工具引用至 builtin-tools 包,删除 src/tools/

- src/tools.ts 及 178 个 src/ 文件的 import 路径从 ./tools/ 改为 builtin-tools/tools/
- 删除 src/tools/ 整个目录(已迁移至 packages/builtin-tools/)

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* chore: 添加 builtin-tools 路径别名至 tsconfig,更新 bun.lock

- tsconfig.json 新增 builtin-tools/* 和 builtin-tools 路径映射
- 新增 packages/builtin-tools/src 至 include

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* refactor: 为 builtin-tools、mcp-client、agent-tools 添加 @claude-code-best 作用域前缀

所有包名及 import 路径统一添加 @claude-code-best/ 前缀:
- builtin-tools → @claude-code-best/builtin-tools
- mcp-client → @claude-code-best/mcp-client
- agent-tools → @claude-code-best/agent-tools

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

* fix: 修复 node 环境没有 bun 的问题

---------

Co-authored-by: Eric-Guo <eric.guocz@gmail.com>
Co-authored-by: Claude Opus 4.6 <noreply@anthropic.com>
2026-04-13 09:52:05 +08:00

4.6 KiB

Chapter 1: Getting Started

Installation

@anthropic/ink is a workspace package. It is consumed internally and not published to npm.

{
  "dependencies": {
    "@anthropic/ink": "workspace:*"
  }
}

Peer Dependencies

  • react ^19.2.4
  • react-reconciler ^0.33.0

Key Dependencies

Package Purpose
chalk ANSI color generation
cli-boxes Border style definitions
get-east-asian-width CJK character width measurement
wrap-ansi ANSI-aware word wrapping
bidi-js Bidirectional text support
lodash-es Utility functions (throttle, noop)
signal-exit Process exit handler cleanup
emoji-regex Emoji width handling

Basic Rendering

render(node, options?)

The primary entry point. Renders a React element tree to the terminal.

import { render } from '@anthropic/ink'
import { Box, Text } from '@anthropic/ink'

const { unmount, rerender, waitUntilExit } = await render(
  <Box>
    <Text>Hello, World!</Text>
  </Box>
)

Parameters:

  • node -- ReactNode to render
  • options -- RenderOptions | NodeJS.WriteStream (optional)

Returns: Promise<Instance> with:

  • rerender(node) -- Replace the root node
  • unmount() -- Unmount and clean up
  • waitUntilExit() -- Promise<void> that resolves on unmount
  • cleanup() -- Remove from instance registry

renderSync(node, options?)

Synchronous version of render. Same API, returns Instance directly (no Promise).

import { renderSync } from '@anthropic/ink'

const instance = renderSync(<App />)
// instance.rerender, instance.unmount, etc.

createRoot(options?)

Creates a managed Ink root without immediately rendering. Similar to react-dom's createRoot.

import { createRoot } from '@anthropic/ink'

const root = await createRoot({ exitOnCtrlC: false })

// Later, render into it
root.render(<App />)

// You can re-render into the same root
root.render(<DifferentApp />)

// Clean up
root.unmount()

Returns: Promise<Root> with:

  • render(node) -- Mount or update the tree
  • unmount() -- Unmount
  • waitUntilExit() -- Promise<void>

RenderOptions

type RenderOptions = {
  /** Output stream. Default: process.stdout */
  stdout?: NodeJS.WriteStream

  /** Input stream. Default: process.stdin */
  stdin?: NodeJS.ReadStream

  /** Error stream. Default: process.stderr */
  stderr?: NodeJS.WriteStream

  /** Handle Ctrl+C to exit. Default: true */
  exitOnCtrlC?: boolean

  /** Patch console methods to prevent Ink output mixing. Default: true */
  patchConsole?: boolean

  /** Called after each frame render with timing info. */
  onFrame?: (event: FrameEvent) => void
}

Basic Concepts

Component Tree

Ink renders React components to a terminal using a custom reconciler. The tree structure maps to terminal output:

<Box flexDirection="column">
  <Text bold color="green">Header</Text>
  <Box flexDirection="row" gap={1}>
    <Text>Left</Text>
    <Text>Right</Text>
  </Box>
</Box>

This produces terminal output with Flexbox layout (via Yoga).

Rendering Pipeline

  1. React Reconciler -- Standard React reconciliation; diffs virtual tree
  2. Yoga Layout -- Computes Flexbox positions/ sizes for every node
  3. Render to Output -- Walks the DOM tree, emits styled text into an Output buffer
  4. Screen Diff -- Compares new frame against previous frame in a screen buffer
  5. Terminal Write -- Emits minimal ANSI escape sequences to update only changed cells

Module System

Import everything from the package root:

// Core rendering
import { render, createRoot, renderSync } from '@anthropic/ink'

// Components (base, no theme)
import { BaseBox, BaseText, ScrollBox, Button, Link, Newline, Spacer } from '@anthropic/ink'

// Theme-aware components (recommended)
import { Box, Text } from '@anthropic/ink'

// Hooks
import { useApp, useInput, useTerminalSize, useInterval } from '@anthropic/ink'

// Theme
import { ThemeProvider, useTheme, color } from '@anthropic/ink'

// Keybindings
import { useKeybinding, KeybindingProvider } from '@anthropic/ink'

Naming Convention: Base vs Theme-aware

The package exports both raw and theme-aware versions of core components:

  • BaseBox / BaseText -- Raw components that only accept raw color values (rgb(...), #hex, ansi:..., ansi256(...))
  • Box / Text -- Theme-aware wrappers that accept both theme keys ('claude', 'success', 'error') and raw color values

Always prefer the theme-aware versions unless you have a specific reason to use raw components.