OpenLand IconOpenLand
Back to Open Source

ink-scroll-list

TypeScript

A high-level ScrollList component for Ink CLI applications, built on top of ink-scroll-view.

ink-scroll-list

A high-level ScrollList component for Ink CLI applications, built on top of ink-scroll-view.

License Version

✨ Features

  • Selection Management: High-level API for managing selection state (selectedIndex).
  • Auto-Scrolling: Automatically scrolls to ensure the selected item is visible.
  • Performance: Optimized to track selection position efficiently without full re-layouts.
  • Navigation: Built-in support for programmatic scrolling and selection (next/previous/first/last).
  • Flexible Alignment: Control how the selected item aligns in the viewport (auto, top, bottom, center).

🎬 Demos

Selection & Navigation

Selection Demo

Scroll Alignment Modes

Alignment Demo

Expand/Collapse

Expand Demo

Dynamic Items

Dynamic Demo

📦 Installation

npm install ink-scroll-list # Peer dependencies npm install ink react

🚀 Usage

ScrollList manages the selectedIndex state internally if you don't provide it, but typically you'll want to control it to respond to user input.

import React, { useRef, useState } from "react"; import { render, Text, Box, useInput } from "ink"; import { ScrollList, ScrollListRef } from "ink-scroll-list"; const App = () => { const listRef = useRef<ScrollListRef>(null); const [selectedIndex, setSelectedIndex] = useState(0); const items = Array.from({ length: 20 }).map((_, i) => `Item ${i + 1}`); useInput((input, key) => { if (key.upArrow) { // selectPrevious returns the new index const newIndex = listRef.current?.selectPrevious() ?? 0; setSelectedIndex(newIndex); } if (key.downArrow) { // selectNext returns the new index const newIndex = listRef.current?.selectNext() ?? 0; setSelectedIndex(newIndex); } if (key.return) { console.log(`Selected: ${items[selectedIndex]}`); } }); return ( <Box borderStyle="single" height={10}> <ScrollList ref={listRef} selectedIndex={selectedIndex} onSelectionChange={setSelectedIndex} > {items.map((item, i) => ( <Box key={i}> <Text color={i === selectedIndex ? "green" : "white"}> {i === selectedIndex ? "> " : " "} {item} </Text> </Box> ))} </ScrollList> </Box> ); }; render(<App />);

📚 API Reference

For detailed API documentation, see API Reference.

Props (ScrollListProps)

Extends ScrollViewProps from ink-scroll-view.

PropTypeDescription
selectedIndexnumberThe currently selected item index.
scrollAlignment'auto' | 'top' | 'bottom' | 'center'Alignment mode for selected item. Default: 'auto'.
onSelectionChange(index: number) => voidCallback when selection changes internally (e.g. via selectNext).
...ScrollViewPropsAll props from ScrollView (onScroll, onViewportSizeChange, etc.).

Ref Methods (ScrollListRef)

Extends ScrollViewRef from ink-scroll-view. Access these via ref.current.

MethodSignatureDescription
scrollToItem(index: number, mode?: Alignment) => voidScroll to a specific item.
select(index: number, mode?: Alignment) => voidSelect an item and scroll to it.
selectNext() => numberSelect the next item. Returns new index.
selectPrevious() => numberSelect the previous item. Returns new index.
selectFirst() => numberSelect the first item. Returns 0.
selectLast() => numberSelect the last item. Returns new index.
getSelectedIndex() => numberGet current selected index.
getItemCount() => numberGet total number of items.
...ScrollViewRefAll methods from ScrollView (scrollTo, scrollBy, remeasure, etc.).

💡 Tips

  1. Controlled vs Uncontrolled: While ScrollList can manage its own state, it's best practice to lift the selectedIndex state up to your component (as shown in the usage example) so you can use it for rendering your items (highlighting, etc.).
  2. Performance: ScrollList uses ink-scroll-view under the hood, so it benefits from the same performance optimizations (virtual viewport, optimistic updates).
  3. Dynamic Heights: Just like ScrollView, ScrollList supports items with variable and changing heights.
  4. Terminal Resizing: Ink components don't automatically know when the terminal window resizes. You need to listen to process.stdout's resize event and call remeasure() on the ref.

🔗 Related Packages

This package is part of a family of Ink scroll components:

PackageDescription
ink-scroll-viewCore scroll container component
ink-scroll-listA scrollable list with focus management and item selection (this package)
ink-scroll-barA standalone scrollbar component for any scroll container

License

MIT