An enhanced, customizable select input component for Ink that supports both vertical and horizontal orientations, hotkeys, and flexible rendering. Ideal for building rich, interactive CLI apps with React.
- Orientation: Choose between vertical or horizontal layouts.
- Custom Indicators & Items: Easily swap out the default indicator and item rendering.
- Hotkey Support: Assign single-character hotkeys for quick selection.
- Disabled Items: Gracefully skip unselectable items during navigation.
- Keyboard Navigation: Arrow keys and Vim-like keys (
h/j/k/l
) supported. - Hooks for Highlight & Selection: Run custom logic on highlight and selection changes.
- Limit Displayed Items: Restrict how many options to show at once.
npm install ink-enhanced-select-input
or
yarn add ink-enhanced-select-input
import React from 'react';
import { render, Text } from 'ink';
import EnhancedSelectInput from 'ink-enhanced-select-input';
const items = [
{ label: 'Option 1', value: 'one', hotkey: '1' },
{ label: 'Option 2', value: 'two', hotkey: '2' },
{ label: 'Option 3', value: 'three', disabled: true },
{ label: 'Option 4', value: 'four', hotkey: '4' }
];
function Demo() {
const handleSelect = (item) => {
console.log(`Selected: ${item.value}`);
};
const handleHighlight = (item) => {
console.log(`Highlighted: ${item.value}`);
};
return (
<>
<Text>Select an option:</Text>
<EnhancedSelectInput
items={items}
orientation="horizontal"
onSelect={handleSelect}
onHighlight={handleHighlight}
/>
</>
);
}
render(<Demo />);
- Type:
Array<{ label: string; value: V; hotkey?: string; indicator?: React.ReactNode; disabled?: boolean }>
- Description: The list of items to display. Each item defines its own label, value, an optional hotkey for quick selection, optional custom indicator, and whether it’s disabled.
- Type:
boolean
- Default:
true
- Description: When
false
, the component won’t respond to input.
- Type:
number
- Default:
0
- Description: The index of the item that should be highlighted initially.
- Type:
number
- Default: undefined
- Description: How many items to display at once. If provided, only that many items are rendered.
- Type:
React.FC<{isSelected: boolean; item: Item<V>;}>
- Default: A simple
>
arrow when selected. - Description: Custom component to render in front of selected item labels.
- Type:
React.FC<{isSelected: boolean; label: string; isDisabled: boolean;}>
- Default: Renders the item label in green if selected, gray if disabled, or default otherwise.
- Description: Custom renderer for the individual item line.
- Type:
(item: Item<V>) => void
- Default: undefined
- Description: Called when the user confirms a selection (via
Enter
or a hotkey).
- Type:
(item: Item<V>) => void
- Default: undefined
- Description: Called whenever the highlighted (focused) item changes.
- Type:
'vertical' | 'horizontal'
- Default:
'vertical'
- Description: Sets the layout direction of items.
You can customize how items and indicators are rendered:
function MyIndicator({ isSelected }) {
return (
<Text color={isSelected ? 'magenta' : undefined}>
{isSelected ? '👉' : ' '}
</Text>
);
}
function MyItem({ isSelected, isDisabled, label }) {
return (
<Text
color={isDisabled ? 'gray' : isSelected ? 'yellow' : 'white'}
dimColor={isDisabled}
>
{label}
</Text>
);
}
<EnhancedSelectInput
items={[
{ label: 'First', value: '1' },
{ label: 'Second', value: '2', disabled: true },
{ label: 'Third', value: '3', hotkey: 't' }
]}
indicatorComponent={MyIndicator}
itemComponent={MyItem}
/>
-
Clone the repository and navigate to the project directory.
-
Install dependencies
-
Build and run the storybook-like test application locally:
npm run build npm start
This will run
dist/storybook.js
, a local testing interface to interact with and visualize different configurations of the component. -
Run tests:
npm test
Uses AVA for a fast test suite.
Contributions are welcome! Feel free to open issues, submit pull requests, or provide feedback. Suggestions for improvements, new features, or bug reports are all appreciated.
This project is licensed under the MIT License.