Install
openclaw skills install ink-tuiInk — React for interactive command-line apps. Build rich terminal UIs with React components.
openclaw skills install ink-tuiInk renders React components directly to the terminal using Yoga layout (Flexbox for CLI). It handles diffing, re-rendering, and terminal I/O. Components receive real React state, effects, and hooks — the same mental model as web React.Requires Node.js 22+ and React 19.2+ as peer dependencies.
when the user wants to create CLI/TUI apps, terminal dashboards, interactive prompts, colored terminal output, spinners, progress bars, or tables using JSX/React syntax.
mkdir my-cli && cd my-cli
pnpm init
pnpm add ink react @types/react
pnpm add -D tsx typescript @types/node
// tsconfig.json
{
"compilerOptions": {
"target": "ES2022",
"module": "ESNext",
"moduleResolution": "bundler",
"jsx": "react-jsx",
"strict": true,
"outDir": "build"
}
}
// index.tsx
import { render, Box, Text } from 'ink';
const App = () => (
<Box flexDirection="column">
<Text color="green" bold>Hello, CLI!</Text>
</Box>
);
render(<App />);
Run: node --import=tsx index.tsx
Ink uses react-reconciler as its rendering core and yoga-layout as its layout engine. It mounts components into a virtual terminal, diffs against current output, and writes changes to stdout. This means:
<Box> — Layout Containerimport { Box, Text } from 'ink';
// Row layout (default flexDirection)
<Box gap={2}>
<Text>Left</Text>
<Text>Right</Text>
</Box>
// Column layout with border
<Box flexDirection="column" borderStyle="round" padding={1} width={40}>
<Text bold>Title</Text>
<Text dim>subtitle</Text>
</Box>
Key Box props: width, height, padding/paddingX/paddingY, borderStyle (single/double/round/classic/bold), borderDimColor, gap, flexGrow, flexShrink, flexDirection (row/column), justifyContent, alignItems.
<Text> — Styled Text// Direct styling
<Text color="green" bold underline>Success</Text>
// Nested spans
<Text>
Regular <Text bold>bold</Text> and <Text color="red" inverse>inverse red</Text>
</Text>
// Wrap via width
<Text width={40} wrap="truncate">Truncated long text...</Text>
Colors: black/red/green/yellow/blue/magenta/cyan/white/gray + Bright variants (redBright, etc.), hex (#ff0000), rgb(255,0,0).
Background: prefix with bg (bgGreen, bgCyanBright).
Styles: bold, dim, italic, underline, strikethrough, inverse.
<Newline>, <Spacer><Newline count={2} /> // n blank lines
<Spacer /> // flex spacer
<Spacer height={3} /> // fixed height spacer
useInput — Keyboard Inputimport { useInput } from 'ink';
useInput((input, key) => {
if (key.escape || input === 'q') exit(); // exit on Escape or 'q'
if (key.upArrow) navigateUp();
if (key.downArrow) navigateDown();
if (key.return) selectItem();
});
key object: upArrow, downArrow, leftArrow, rightArrow, return, escape, tab, backspace, delete, ctrl, shift, meta, space, pageUp, pageDown, home, end, f1–f12.
useApp — App-Level Controlimport { useApp } from 'ink';
const { exit } = useApp();
exit(); // or exit(error)
useFocus — Focus Managementimport { useFocus } from 'ink';
const { isFocused } = useFocus({ autoFocus: true });
// Style differently when focused
<Text color={isFocused ? 'blue' : 'dim'}>{label}</Text>
useStdin / useStdout — Stream Accessconst { stdin, isRawModeSupported } = useStdin();
const { stdout } = useStdout();
render(<App />, {
exitOnCtrlC: true, // default true
debug: false, // show Yoga layout debug
patchConsole: true, // suppress console output
});
const Menu = ({ items, onSelect }) => {
const [idx, setIdx] = useState(0);
useInput((_, key) => {
if (key.upArrow) setIdx(i => Math.max(0, i - 1));
if (key.downArrow) setIdx(i => Math.min(items.length - 1, i + 1));
if (key.return) onSelect(items[idx]);
});
return (
<Box flexDirection="column">
{items.map((item, i) => (
<Text key={i} color={i === idx ? 'green' : undefined}>
{i === idx ? '❯ ' : ' '}{item}
</Text>
))}
</Box>
);
};
const Spinner = () => {
const [frame, setFrame] = useState(0);
const chars = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];
useEffect(() => {
const id = setInterval(() => setFrame(f => (f + 1) % chars.length), 80);
return () => clearInterval(id);
}, []);
return <Text>{chars[frame]} Loading...</Text>;
};
const ProgressBar = ({ percent }) => {
const filled = '█'.repeat(Math.round(percent / 5));
const empty = '░'.repeat(20 - filled.length);
return <Text>{filled}{empty} {percent}%</Text>;
};
const Input = ({ onSubmit }) => {
const [value, setValue] = useState('');
useInput((input, key) => {
if (key.return) { onSubmit(value); setValue(''); }
else if (key.backspace) setValue(v => v.slice(0, -1));
else if (input.length === 1 && !key.ctrl) setValue(v => v + input);
});
return <Text>❯ {value}<Text dim>█</Text></Text>;
};
This skill includes: