* 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>
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.4react-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--ReactNodeto renderoptions--RenderOptions | NodeJS.WriteStream(optional)
Returns: Promise<Instance> with:
rerender(node)-- Replace the root nodeunmount()-- Unmount and clean upwaitUntilExit()--Promise<void>that resolves on unmountcleanup()-- 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 treeunmount()-- UnmountwaitUntilExit()--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
- React Reconciler -- Standard React reconciliation; diffs virtual tree
- Yoga Layout -- Computes Flexbox positions/ sizes for every node
- Render to Output -- Walks the DOM tree, emits styled text into an
Outputbuffer - Screen Diff -- Compares new frame against previous frame in a screen buffer
- 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.