home / skills / thebushidocollective / han / ink-layout-styling

This skill helps you design elegant terminal layouts in Ink using flexbox-based positioning and styling for clear CLI interfaces.

npx playbooks add skill thebushidocollective/han --skill ink-layout-styling

Review the files below or copy the command above to add this skill to your agents.

Files (1)
SKILL.md
6.9 KB
---
name: ink-layout-styling
user-invocable: false
description: Use when creating terminal layouts with Ink using Flexbox-based positioning and styling for CLI applications.
allowed-tools: []
---

# Ink Layout and Styling

You are an expert in creating beautiful terminal layouts with Ink using Flexbox-based positioning and styling.

## Box Component - Layout Foundation

The Box component is the primary layout primitive, using Flexbox properties.

### Flexbox Direction

```tsx
// 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>
```

### Spacing and Padding

```tsx
// 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>
```

### Alignment and Justification

```tsx
// 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>
```

### Dimensions

```tsx
// 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>
```

### Borders

```tsx
// 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>
```

## Text Component - Styling Text

### Colors

```tsx
// 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>
```

### Text Formatting

```tsx
// 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>
```

### Text Wrapping

```tsx
// 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>
```

## Layout Patterns

### Card Layout

```tsx
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>
);
```

### Split Layout

```tsx
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>
);
```

### Header/Content/Footer

```tsx
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>
);
```

### Grid Layout

```tsx
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>
  );
};
```

### Responsive Layout

```tsx
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>
  );
};
```

## Utility Components

### Spacer

```tsx
// Push elements apart
<Box>
  <Text>Left</Text>
  <Spacer />
  <Text>Right</Text>
</Box>
```

### Newline

```tsx
<Box flexDirection="column">
  <Text>First line</Text>
  <Newline />
  <Text>After blank line</Text>
</Box>
```

## Best Practices

1. **Use Box for layout**: Don't use Text for layout, use Box
2. **Flexbox thinking**: Think in terms of Flexbox properties
3. **Consistent spacing**: Use consistent margin/padding values
4. **Color sparingly**: Don't overuse colors, use them for emphasis
5. **Test terminals**: Test on different terminal sizes and emulators

## Common Layout Patterns

### Full-width header

```tsx
<Box width="100%" borderStyle="single" padding={1}>
  <Text bold>Application Title</Text>
</Box>
```

### Centered modal

```tsx
<Box justifyContent="center" alignItems="center" height="100%">
  <Box borderStyle="round" padding={2} borderColor="cyan">
    <Text>Modal content</Text>
  </Box>
</Box>
```

### Sidebar layout

```tsx
<Box>
  <Box width={20} borderStyle="single" padding={1}>
    <Text>Sidebar</Text>
  </Box>
  <Box flexGrow={1} padding={1}>
    <Text>Main content</Text>
  </Box>
</Box>
```

Overview

This skill helps you build polished terminal layouts with Ink using Flexbox-style Box primitives and rich Text styling. It provides practical patterns for cards, split screens, headers, grids, and responsive layouts so CLI UIs look and behave consistently. Use it to manage spacing, alignment, borders, colors, wrapping, and simple utility components like spacers and newlines.

How this skill works

The skill exposes Box and Text patterns that map Flexbox concepts to terminal layout: flexDirection, alignItems, justifyContent, flexGrow, width, height, padding and margin. It also covers border styles, color and text formatting, wrapping/truncation, and small helpers (Spacer, Newline). Combine these primitives into reusable components (Card, SplitLayout, Grid, ResponsiveLayout) to compose complex interfaces.

When to use it

  • Building multi-column or stacked terminal interfaces
  • Creating cards, sidebars, headers, footers, or modals
  • Styling text with colors, emphasis, and truncation rules
  • Adapting layouts to different terminal widths (responsive UIs)
  • Enforcing consistent spacing and border styles across the app

Best practices

  • Always use Box for layout; avoid using Text for positioning
  • Think in Flexbox terms: main axis (justifyContent) and cross axis (alignItems)
  • Use consistent margin/padding scales to keep spacing uniform
  • Reserve color and heavy styling for emphasis only
  • Test layouts on multiple terminal sizes and emulators to avoid clipping

Example use cases

  • Card component for grouped content with title and body
  • Split layout for side-by-side panes or comparison screens
  • Header/content/footer scaffolding for full-screen CLIs
  • Grid to render item collections in columns with predictable widths
  • Responsive layout that switches between row and column based on terminal width

FAQ

How do I center a modal on screen?

Wrap a Box with justifyContent and alignItems set to "center" and give the inner Box a border and padding for the modal body.

How can I prevent long text from breaking layout?

Use Text wrap="wrap" with a constrained Box width, or wrap="truncate"/"truncate-end" to cut overflow with an ellipsis.

Which border styles are available?

Common options include single, double, round, bold, singleDouble, doubleSingle and classic; pair with borderColor for emphasis.