From ink
Builds Flexbox-based terminal layouts in CLI apps using Ink's Box for positioning, spacing, alignment, dimensions, borders and Text for colors.
npx claudepluginhub thebushidocollective/han --plugin inkThis skill cannot use any tools. It operates in read-only mode without the ability to modify files or execute commands.
You are an expert in creating beautiful terminal layouts with Ink using Flexbox-based positioning and styling.
Build terminal UIs for React CLI apps using Ink component patterns: layouts, lists with icons, status messages, progress indicators, and collapsible sections.
Builds CLI apps using Ink (React renderer for terminals), ink-ui components (TextInput, Select, Spinner), and create-ink-app scaffolding for interactive terminal UIs and JSX in command-line tools.
Renders JSON specs into interactive terminal UIs using Ink components like Box, Text, Tables, Charts, Spinners, with data binding and actions. Use for terminal apps, component catalogs, or AI-generated UIs.
Share bugs, ideas, or general feedback.
You are an expert in creating beautiful terminal layouts with Ink using Flexbox-based positioning and styling.
The Box component is the primary layout primitive, using Flexbox properties.
// Vertical stack (default)
<Box flexDirection="column">
<Text>First</Text>
<Text>Second</Text>
</Box>
// Horizontal row
<Box flexDirection="row">
<Text>Left</Text>
<Text>Right</Text>
</Box>
// Reverse order
<Box flexDirection="column-reverse">
<Text>Bottom (renders on top)</Text>
<Text>Top (renders on bottom)</Text>
</Box>
// Margin around element
<Box margin={1}>
<Text>Content with margin</Text>
</Box>
// Directional margins
<Box marginTop={1} marginLeft={2} marginRight={2} marginBottom={1}>
<Text>Custom margins</Text>
</Box>
// Padding inside element
<Box padding={1}>
<Text>Content with padding</Text>
</Box>
// Directional padding
<Box paddingX={2} paddingY={1}>
<Text>Horizontal and vertical padding</Text>
</Box>
// Align items on cross axis
<Box alignItems="center">
<Text>Centered</Text>
</Box>
<Box alignItems="flex-start">
<Text>Top aligned</Text>
</Box>
<Box alignItems="flex-end">
<Text>Bottom aligned</Text>
</Box>
// Justify content on main axis
<Box justifyContent="center">
<Text>Centered horizontally</Text>
</Box>
<Box justifyContent="space-between">
<Text>Left</Text>
<Text>Right</Text>
</Box>
<Box justifyContent="space-around">
<Text>Item 1</Text>
<Text>Item 2</Text>
<Text>Item 3</Text>
</Box>
// Fixed width
<Box width={50}>
<Text>Fixed width content</Text>
</Box>
// Percentage width
<Box width="50%">
<Text>Half width</Text>
</Box>
// Fixed height
<Box height={10}>
<Text>Fixed height</Text>
</Box>
// Min/max dimensions
<Box minWidth={20} maxWidth={80}>
<Text>Constrained width</Text>
</Box>
// Simple border
<Box borderStyle="single">
<Text>Bordered content</Text>
</Box>
// Border styles: single, double, round, bold, singleDouble, doubleSingle, classic
<Box borderStyle="round" borderColor="cyan">
<Text>Rounded border</Text>
</Box>
// Custom border color
<Box borderStyle="double" borderColor="green">
<Text>Green double border</Text>
</Box>
// Foreground colors
<Text color="red">Error message</Text>
<Text color="green">Success message</Text>
<Text color="yellow">Warning message</Text>
<Text color="blue">Info message</Text>
<Text color="cyan">Highlight</Text>
<Text color="magenta">Special</Text>
// Background colors
<Text backgroundColor="red" color="white">
Alert!
</Text>
// Hex colors
<Text color="#FF5733">Custom color</Text>
// Bold
<Text bold>Important text</Text>
// Italic
<Text italic>Emphasized text</Text>
// Underline
<Text underline>Underlined text</Text>
// Strikethrough
<Text strikethrough>Removed text</Text>
// Dim
<Text dimColor>Subtle text</Text>
// Inverse
<Text inverse>Inverted colors</Text>
// Combinations
<Text bold color="cyan" underline>
Multiple styles
</Text>
// Wrap text to fit width
<Box width={40}>
<Text wrap="wrap">
This is a long text that will wrap to fit within the 40 character width.
</Text>
</Box>
// Truncate text
<Box width={20}>
<Text wrap="truncate">This text will be truncated</Text>
</Box>
// Truncate with custom ellipsis
<Box width={20}>
<Text wrap="truncate-end">This text will be truncated...</Text>
</Box>
const Card: React.FC<{ title: string; children: React.ReactNode }> = ({ title, children }) => (
<Box borderStyle="round" borderColor="cyan" padding={1} flexDirection="column">
<Box marginBottom={1}>
<Text bold color="cyan">
{title}
</Text>
</Box>
<Box>{children}</Box>
</Box>
);
const SplitLayout: React.FC<{ left: React.ReactNode; right: React.ReactNode }> = ({
left,
right,
}) => (
<Box>
<Box width="50%" borderStyle="single" padding={1}>
{left}
</Box>
<Box width="50%" borderStyle="single" padding={1}>
{right}
</Box>
</Box>
);
const Layout: React.FC<{
header: React.ReactNode;
content: React.ReactNode;
footer: React.ReactNode;
}> = ({ header, content, footer }) => (
<Box flexDirection="column" height="100%">
<Box borderStyle="single" borderColor="cyan" padding={1}>
{header}
</Box>
<Box flexGrow={1} padding={1}>
{content}
</Box>
<Box borderStyle="single" borderColor="gray" padding={1}>
{footer}
</Box>
</Box>
);
const Grid: React.FC<{ items: React.ReactNode[]; columns: number }> = ({ items, columns }) => {
const rows: React.ReactNode[][] = [];
for (let i = 0; i < items.length; i += columns) {
rows.push(items.slice(i, i + columns));
}
return (
<Box flexDirection="column">
{rows.map((row, i) => (
<Box key={i}>
{row.map((item, j) => (
<Box key={j} width={`${100 / columns}%`} padding={1}>
{item}
</Box>
))}
</Box>
))}
</Box>
);
};
import { useStdout } from 'ink';
const ResponsiveLayout: React.FC<{ children: React.ReactNode }> = ({ children }) => {
const { stdout } = useStdout();
const isNarrow = stdout.columns < 80;
return (
<Box flexDirection={isNarrow ? 'column' : 'row'} width="100%">
{children}
</Box>
);
};
// Push elements apart
<Box>
<Text>Left</Text>
<Spacer />
<Text>Right</Text>
</Box>
<Box flexDirection="column">
<Text>First line</Text>
<Newline />
<Text>After blank line</Text>
</Box>
<Box width="100%" borderStyle="single" padding={1}>
<Text bold>Application Title</Text>
</Box>
<Box justifyContent="center" alignItems="center" height="100%">
<Box borderStyle="round" padding={2} borderColor="cyan">
<Text>Modal content</Text>
</Box>
</Box>
<Box>
<Box width={20} borderStyle="single" padding={1}>
<Text>Sidebar</Text>
</Box>
<Box flexGrow={1} padding={1}>
<Text>Main content</Text>
</Box>
</Box>