π File detail
components/design-system/ListItem.tsx
π§© .tsxπ 244 linesπΎ 19,535 bytesπ text
β Back to All Filesπ― Use case
This file lives under βcomponents/β, which covers shared React UI pieces. On the API surface it exposes ListItem β mainly types, interfaces, or factory objects. Dependencies touch React UI and figures. It composes internal code from ink (relative imports).
Generated from folder role, exports, dependency roots, and inline comments β not hand-reviewed for every path.
π§ Inline summary
import { c as _c } from "react/compiler-runtime"; import figures from 'figures'; import type { ReactNode } from 'react'; import React from 'react'; import { useDeclaredCursor } from '../../ink/hooks/use-declared-cursor.js';
π€ Exports (heuristic)
ListItem
π External import roots
Package roots from from "β¦" (relative paths omitted).
reactfigures
π₯οΈ Source preview
import { c as _c } from "react/compiler-runtime";
import figures from 'figures';
import type { ReactNode } from 'react';
import React from 'react';
import { useDeclaredCursor } from '../../ink/hooks/use-declared-cursor.js';
import { Box, Text } from '../../ink.js';
type ListItemProps = {
/**
* Whether this item is currently focused (keyboard selection).
* Shows the pointer indicator (β―) when true.
*/
isFocused: boolean;
/**
* Whether this item is selected (chosen/checked).
* Shows the checkmark indicator (β) when true.
* @default false
*/
isSelected?: boolean;
/**
* The content to display for this item.
*/
children: ReactNode;
/**
* Optional description text displayed below the main content.
*/
description?: string;
/**
* Show a down arrow indicator instead of pointer (for scroll hints).
* Only applies when not focused.
*/
showScrollDown?: boolean;
/**
* Show an up arrow indicator instead of pointer (for scroll hints).
* Only applies when not focused.
*/
showScrollUp?: boolean;
/**
* Whether to apply automatic styling to the children based on focus/selection state.
* - When true (default): children are wrapped in Text with state-based colors
* - When false: children are rendered as-is, allowing custom styling
* @default true
*/
styled?: boolean;
/**
* Whether this item is disabled. Disabled items show dimmed text and no indicators.
* @default false
*/
disabled?: boolean;
/**
* Whether this ListItem should declare the terminal cursor position.
* Set false when a child (e.g. BaseTextInput) declares its own cursor.
* @default true
*/
declareCursor?: boolean;
};
/**
* A list item component for selection UIs (dropdowns, multi-selects, menus).
*
* Handles the common pattern of:
* - Pointer indicator (β―) for focused items
* - Checkmark indicator (β) for selected items
* - Scroll indicators (ββ) for truncated lists
* - Color states for focus/selection
*
* @example
* // Basic usage in a selection list
* {options.map((option, i) => (
* <ListItem
* key={option.id}
* isFocused={focusIndex === i}
* isSelected={selectedId === option.id}
* >
* {option.label}
* </ListItem>
* ))}
*
* @example
* // With scroll indicators
* <ListItem isFocused={false} showScrollUp>First visible item</ListItem>
* ...
* <ListItem isFocused={false} showScrollDown>Last visible item</ListItem>
*
* @example
* // With description
* <ListItem isFocused isSelected={false} description="Secondary text here">
* Primary text
* </ListItem>
*
* @example
* // Custom children styling (styled=false)
* <ListItem isFocused styled={false}>
* <Text color="claude">Custom styled content</Text>
* </ListItem>
*/
export function ListItem(t0) {
const $ = _c(32);
const {
isFocused,
isSelected: t1,
children,
description,
showScrollDown,
showScrollUp,
styled: t2,
disabled: t3,
declareCursor
} = t0;
const isSelected = t1 === undefined ? false : t1;
const styled = t2 === undefined ? true : t2;
const disabled = t3 === undefined ? false : t3;
let t4;
if ($[0] !== disabled || $[1] !== isFocused || $[2] !== showScrollDown || $[3] !== showScrollUp) {
t4 = function renderIndicator() {
if (disabled) {
return <Text> </Text>;
}
if (isFocused) {
return <Text color="suggestion">{figures.pointer}</Text>;
}
if (showScrollDown) {
return <Text dimColor={true}>{figures.arrowDown}</Text>;
}
if (showScrollUp) {
return <Text dimColor={true}>{figures.arrowUp}</Text>;
}
return <Text> </Text>;
};
$[0] = disabled;
$[1] = isFocused;
$[2] = showScrollDown;
$[3] = showScrollUp;
$[4] = t4;
} else {
t4 = $[4];
}
const renderIndicator = t4;
let t5;
if ($[5] !== disabled || $[6] !== isFocused || $[7] !== isSelected || $[8] !== styled) {
const getTextColor = function getTextColor() {
if (disabled) {
return "inactive";
}
if (!styled) {
return;
}
if (isSelected) {
return "success";
}
if (isFocused) {
return "suggestion";
}
};
t5 = getTextColor();
$[5] = disabled;
$[6] = isFocused;
$[7] = isSelected;
$[8] = styled;
$[9] = t5;
} else {
t5 = $[9];
}
const textColor = t5;
const t6 = isFocused && !disabled && declareCursor !== false;
let t7;
if ($[10] !== t6) {
t7 = {
line: 0,
column: 0,
active: t6
};
$[10] = t6;
$[11] = t7;
} else {
t7 = $[11];
}
const cursorRef = useDeclaredCursor(t7);
let t8;
if ($[12] !== renderIndicator) {
t8 = renderIndicator();
$[12] = renderIndicator;
$[13] = t8;
} else {
t8 = $[13];
}
let t9;
if ($[14] !== children || $[15] !== disabled || $[16] !== styled || $[17] !== textColor) {
t9 = styled ? <Text color={textColor} dimColor={disabled}>{children}</Text> : children;
$[14] = children;
$[15] = disabled;
$[16] = styled;
$[17] = textColor;
$[18] = t9;
} else {
t9 = $[18];
}
let t10;
if ($[19] !== disabled || $[20] !== isSelected) {
t10 = isSelected && !disabled && <Text color="success">{figures.tick}</Text>;
$[19] = disabled;
$[20] = isSelected;
$[21] = t10;
} else {
t10 = $[21];
}
let t11;
if ($[22] !== t10 || $[23] !== t8 || $[24] !== t9) {
t11 = <Box flexDirection="row" gap={1}>{t8}{t9}{t10}</Box>;
$[22] = t10;
$[23] = t8;
$[24] = t9;
$[25] = t11;
} else {
t11 = $[25];
}
let t12;
if ($[26] !== description) {
t12 = description && <Box paddingLeft={2}><Text color="inactive">{description}</Text></Box>;
$[26] = description;
$[27] = t12;
} else {
t12 = $[27];
}
let t13;
if ($[28] !== cursorRef || $[29] !== t11 || $[30] !== t12) {
t13 = <Box ref={cursorRef} flexDirection="column">{t11}{t12}</Box>;
$[28] = cursorRef;
$[29] = t11;
$[30] = t12;
$[31] = t13;
} else {
t13 = $[31];
}
return t13;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["figures","ReactNode","React","useDeclaredCursor","Box","Text","ListItemProps","isFocused","isSelected","children","description","showScrollDown","showScrollUp","styled","disabled","declareCursor","ListItem","t0","$","_c","t1","t2","t3","undefined","t4","renderIndicator","pointer","arrowDown","arrowUp","t5","getTextColor","textColor","t6","t7","line","column","active","cursorRef","t8","t9","t10","tick","t11","t12","t13"],"sources":["ListItem.tsx"],"sourcesContent":["import figures from 'figures'\nimport type { ReactNode } from 'react'\nimport React from 'react'\nimport { useDeclaredCursor } from '../../ink/hooks/use-declared-cursor.js'\nimport { Box, Text } from '../../ink.js'\n\ntype ListItemProps = {\n  /**\n   * Whether this item is currently focused (keyboard selection).\n   * Shows the pointer indicator (❯) when true.\n   */\n  isFocused: boolean\n\n  /**\n   * Whether this item is selected (chosen/checked).\n   * Shows the checkmark indicator (✓) when true.\n   * @default false\n   */\n  isSelected?: boolean\n\n  /**\n   * The content to display for this item.\n   */\n  children: ReactNode\n\n  /**\n   * Optional description text displayed below the main content.\n   */\n  description?: string\n\n  /**\n   * Show a down arrow indicator instead of pointer (for scroll hints).\n   * Only applies when not focused.\n   */\n  showScrollDown?: boolean\n\n  /**\n   * Show an up arrow indicator instead of pointer (for scroll hints).\n   * Only applies when not focused.\n   */\n  showScrollUp?: boolean\n\n  /**\n   * Whether to apply automatic styling to the children based on focus/selection state.\n   * - When true (default): children are wrapped in Text with state-based colors\n   * - When false: children are rendered as-is, allowing custom styling\n   * @default true\n   */\n  styled?: boolean\n\n  /**\n   * Whether this item is disabled. Disabled items show dimmed text and no indicators.\n   * @default false\n   */\n  disabled?: boolean\n\n  /**\n   * Whether this ListItem should declare the terminal cursor position.\n   * Set false when a child (e.g. BaseTextInput) declares its own cursor.\n   * @default true\n   */\n  declareCursor?: boolean\n}\n\n/**\n * A list item component for selection UIs (dropdowns, multi-selects, menus).\n *\n * Handles the common pattern of:\n * - Pointer indicator (❯) for focused items\n * - Checkmark indicator (✓) for selected items\n * - Scroll indicators (↓↑) for truncated lists\n * - Color states for focus/selection\n *\n * @example\n * // Basic usage in a selection list\n * {options.map((option, i) => (\n *   <ListItem\n *     key={option.id}\n *     isFocused={focusIndex === i}\n *     isSelected={selectedId === option.id}\n *   >\n *     {option.label}\n *   </ListItem>\n * ))}\n *\n * @example\n * // With scroll indicators\n * <ListItem isFocused={false} showScrollUp>First visible item</ListItem>\n * ...\n * <ListItem isFocused={false} showScrollDown>Last visible item</ListItem>\n *\n * @example\n * // With description\n * <ListItem isFocused isSelected={false} description=\"Secondary text here\">\n *   Primary text\n * </ListItem>\n *\n * @example\n * // Custom children styling (styled=false)\n * <ListItem isFocused styled={false}>\n *   <Text color=\"claude\">Custom styled content</Text>\n * </ListItem>\n */\nexport function ListItem({\n  isFocused,\n  isSelected = false,\n  children,\n  description,\n  showScrollDown,\n  showScrollUp,\n  styled = true,\n  disabled = false,\n  declareCursor,\n}: ListItemProps): React.ReactNode {\n  // Determine which indicator to show\n  function renderIndicator(): ReactNode {\n    if (disabled) {\n      return <Text> </Text>\n    }\n\n    if (isFocused) {\n      return <Text color=\"suggestion\">{figures.pointer}</Text>\n    }\n\n    if (showScrollDown) {\n      return <Text dimColor>{figures.arrowDown}</Text>\n    }\n\n    if (showScrollUp) {\n      return <Text dimColor>{figures.arrowUp}</Text>\n    }\n\n    return <Text> </Text>\n  }\n\n  // Determine text color based on state\n  function getTextColor(): 'success' | 'suggestion' | 'inactive' | undefined {\n    if (disabled) {\n      return 'inactive'\n    }\n\n    if (!styled) {\n      return undefined\n    }\n\n    if (isSelected) {\n      return 'success'\n    }\n\n    if (isFocused) {\n      return 'suggestion'\n    }\n\n    return undefined\n  }\n\n  const textColor = getTextColor()\n\n  // Park the native terminal cursor on the pointer indicator so screen\n  // readers / magnifiers track the focused item. (0,0) is the top-left of\n  // this Box, where the pointer renders.\n  const cursorRef = useDeclaredCursor({\n    line: 0,\n    column: 0,\n    active: isFocused && !disabled && declareCursor !== false,\n  })\n\n  return (\n    <Box ref={cursorRef} flexDirection=\"column\">\n      <Box flexDirection=\"row\" gap={1}>\n        {renderIndicator()}\n        {styled ? (\n          <Text color={textColor} dimColor={disabled}>\n            {children}\n          </Text>\n        ) : (\n          children\n        )}\n        {isSelected && !disabled && <Text color=\"success\">{figures.tick}</Text>}\n      </Box>\n      {description && (\n        <Box paddingLeft={2}>\n          <Text color=\"inactive\">{description}</Text>\n        </Box>\n      )}\n    </Box>\n  )\n}\n"],"mappings":";AAAA,OAAOA,OAAO,MAAM,SAAS;AAC7B,cAAcC,SAAS,QAAQ,OAAO;AACtC,OAAOC,KAAK,MAAM,OAAO;AACzB,SAASC,iBAAiB,QAAQ,wCAAwC;AAC1E,SAASC,GAAG,EAAEC,IAAI,QAAQ,cAAc;AAExC,KAAKC,aAAa,GAAG;EACnB;AACF;AACA;AACA;EACEC,SAAS,EAAE,OAAO;;EAElB;AACF;AACA;AACA;AACA;EACEC,UAAU,CAAC,EAAE,OAAO;;EAEpB;AACF;AACA;EACEC,QAAQ,EAAER,SAAS;;EAEnB;AACF;AACA;EACES,WAAW,CAAC,EAAE,MAAM;;EAEpB;AACF;AACA;AACA;EACEC,cAAc,CAAC,EAAE,OAAO;;EAExB;AACF;AACA;AACA;EACEC,YAAY,CAAC,EAAE,OAAO;;EAEtB;AACF;AACA;AACA;AACA;AACA;EACEC,MAAM,CAAC,EAAE,OAAO;;EAEhB;AACF;AACA;AACA;EACEC,QAAQ,CAAC,EAAE,OAAO;;EAElB;AACF;AACA;AACA;AACA;EACEC,aAAa,CAAC,EAAE,OAAO;AACzB,CAAC;;AAED;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO,SAAAC,SAAAC,EAAA;EAAA,MAAAC,CAAA,GAAAC,EAAA;EAAkB;IAAAZ,SAAA;IAAAC,UAAA,EAAAY,EAAA;IAAAX,QAAA;IAAAC,WAAA;IAAAC,cAAA;IAAAC,YAAA;IAAAC,MAAA,EAAAQ,EAAA;IAAAP,QAAA,EAAAQ,EAAA;IAAAP;EAAA,IAAAE,EAUT;EARd,MAAAT,UAAA,GAAAY,EAAkB,KAAlBG,SAAkB,GAAlB,KAAkB,GAAlBH,EAAkB;EAKlB,MAAAP,MAAA,GAAAQ,EAAa,KAAbE,SAAa,GAAb,IAAa,GAAbF,EAAa;EACb,MAAAP,QAAA,GAAAQ,EAAgB,KAAhBC,SAAgB,GAAhB,KAAgB,GAAhBD,EAAgB;EAAA,IAAAE,EAAA;EAAA,IAAAN,CAAA,QAAAJ,QAAA,IAAAI,CAAA,QAAAX,SAAA,IAAAW,CAAA,QAAAP,cAAA,IAAAO,CAAA,QAAAN,YAAA;IAIhBY,EAAA,YAAAC,gBAAA;MACE,IAAIX,QAAQ;QAAA,OACH,CAAC,IAAI,CAAC,CAAC,EAAN,IAAI,CAAS;MAAA;MAGvB,IAAIP,SAAS;QAAA,OACJ,CAAC,IAAI,CAAO,KAAY,CAAZ,YAAY,CAAE,CAAAP,OAAO,CAAA0B,OAAO,CAAE,EAAzC,IAAI,CAA4C;MAAA;MAG1D,IAAIf,cAAc;QAAA,OACT,CAAC,IAAI,CAAC,QAAQ,CAAR,KAAO,CAAC,CAAE,CAAAX,OAAO,CAAA2B,SAAS,CAAE,EAAjC,IAAI,CAAoC;MAAA;MAGlD,IAAIf,YAAY;QAAA,OACP,CAAC,IAAI,CAAC,QAAQ,CAAR,KAAO,CAAC,CAAE,CAAAZ,OAAO,CAAA4B,OAAO,CAAE,EAA/B,IAAI,CAAkC;MAAA;MAC/C,OAEM,CAAC,IAAI,CAAC,CAAC,EAAN,IAAI,CAAS;IAAA,CACtB;IAAAV,CAAA,MAAAJ,QAAA;IAAAI,CAAA,MAAAX,SAAA;IAAAW,CAAA,MAAAP,cAAA;IAAAO,CAAA,MAAAN,YAAA;IAAAM,CAAA,MAAAM,EAAA;EAAA;IAAAA,EAAA,GAAAN,CAAA;EAAA;EAlBD,MAAAO,eAAA,GAAAD,EAkBC;EAAA,IAAAK,EAAA;EAAA,IAAAX,CAAA,QAAAJ,QAAA,IAAAI,CAAA,QAAAX,SAAA,IAAAW,CAAA,QAAAV,UAAA,IAAAU,CAAA,QAAAL,MAAA;IAGD,MAAAiB,YAAA,YAAAA,aAAA;MACE,IAAIhB,QAAQ;QAAA,OACH,UAAU;MAAA;MAGnB,IAAI,CAACD,MAAM;QAAA;MAAA;MAIX,IAAIL,UAAU;QAAA,OACL,SAAS;MAAA;MAGlB,IAAID,SAAS;QAAA,OACJ,YAAY;MAAA;IACpB,CAGF;IAEiBsB,EAAA,GAAAC,YAAY,CAAC,CAAC;IAAAZ,CAAA,MAAAJ,QAAA;IAAAI,CAAA,MAAAX,SAAA;IAAAW,CAAA,MAAAV,UAAA;IAAAU,CAAA,MAAAL,MAAA;IAAAK,CAAA,MAAAW,EAAA;EAAA;IAAAA,EAAA,GAAAX,CAAA;EAAA;EAAhC,MAAAa,SAAA,GAAkBF,EAAc;EAQtB,MAAAG,EAAA,GAAAzB,SAAsB,IAAtB,CAAcO,QAAmC,IAAvBC,aAAa,KAAK,KAAK;EAAA,IAAAkB,EAAA;EAAA,IAAAf,CAAA,SAAAc,EAAA;IAHvBC,EAAA;MAAAC,IAAA,EAC5B,CAAC;MAAAC,MAAA,EACC,CAAC;MAAAC,MAAA,EACDJ;IACV,CAAC;IAAAd,CAAA,OAAAc,EAAA;IAAAd,CAAA,OAAAe,EAAA;EAAA;IAAAA,EAAA,GAAAf,CAAA;EAAA;EAJD,MAAAmB,SAAA,GAAkBlC,iBAAiB,CAAC8B,EAInC,CAAC;EAAA,IAAAK,EAAA;EAAA,IAAApB,CAAA,SAAAO,eAAA;IAKKa,EAAA,GAAAb,eAAe,CAAC,CAAC;IAAAP,CAAA,OAAAO,eAAA;IAAAP,CAAA,OAAAoB,EAAA;EAAA;IAAAA,EAAA,GAAApB,CAAA;EAAA;EAAA,IAAAqB,EAAA;EAAA,IAAArB,CAAA,SAAAT,QAAA,IAAAS,CAAA,SAAAJ,QAAA,IAAAI,CAAA,SAAAL,MAAA,IAAAK,CAAA,SAAAa,SAAA;IACjBQ,EAAA,GAAA1B,MAAM,GACL,CAAC,IAAI,CAAQkB,KAAS,CAATA,UAAQ,CAAC,CAAYjB,QAAQ,CAARA,SAAO,CAAC,CACvCL,SAAO,CACV,EAFC,IAAI,CAKN,GANAA,QAMA;IAAAS,CAAA,OAAAT,QAAA;IAAAS,CAAA,OAAAJ,QAAA;IAAAI,CAAA,OAAAL,MAAA;IAAAK,CAAA,OAAAa,SAAA;IAAAb,CAAA,OAAAqB,EAAA;EAAA;IAAAA,EAAA,GAAArB,CAAA;EAAA;EAAA,IAAAsB,GAAA;EAAA,IAAAtB,CAAA,SAAAJ,QAAA,IAAAI,CAAA,SAAAV,UAAA;IACAgC,GAAA,GAAAhC,UAAuB,IAAvB,CAAeM,QAAuD,IAA3C,CAAC,IAAI,CAAO,KAAS,CAAT,SAAS,CAAE,CAAAd,OAAO,CAAAyC,IAAI,CAAE,EAAnC,IAAI,CAAsC;IAAAvB,CAAA,OAAAJ,QAAA;IAAAI,CAAA,OAAAV,UAAA;IAAAU,CAAA,OAAAsB,GAAA;EAAA;IAAAA,GAAA,GAAAtB,CAAA;EAAA;EAAA,IAAAwB,GAAA;EAAA,IAAAxB,CAAA,SAAAsB,GAAA,IAAAtB,CAAA,SAAAoB,EAAA,IAAApB,CAAA,SAAAqB,EAAA;IATzEG,GAAA,IAAC,GAAG,CAAe,aAAK,CAAL,KAAK,CAAM,GAAC,CAAD,GAAC,CAC5B,CAAAJ,EAAgB,CAChB,CAAAC,EAMD,CACC,CAAAC,GAAqE,CACxE,EAVC,GAAG,CAUE;IAAAtB,CAAA,OAAAsB,GAAA;IAAAtB,CAAA,OAAAoB,EAAA;IAAApB,CAAA,OAAAqB,EAAA;IAAArB,CAAA,OAAAwB,GAAA;EAAA;IAAAA,GAAA,GAAAxB,CAAA;EAAA;EAAA,IAAAyB,GAAA;EAAA,IAAAzB,CAAA,SAAAR,WAAA;IACLiC,GAAA,GAAAjC,WAIA,IAHC,CAAC,GAAG,CAAc,WAAC,CAAD,GAAC,CACjB,CAAC,IAAI,CAAO,KAAU,CAAV,UAAU,CAAEA,YAAU,CAAE,EAAnC,IAAI,CACP,EAFC,GAAG,CAGL;IAAAQ,CAAA,OAAAR,WAAA;IAAAQ,CAAA,OAAAyB,GAAA;EAAA;IAAAA,GAAA,GAAAzB,CAAA;EAAA;EAAA,IAAA0B,GAAA;EAAA,IAAA1B,CAAA,SAAAmB,SAAA,IAAAnB,CAAA,SAAAwB,GAAA,IAAAxB,CAAA,SAAAyB,GAAA;IAhBHC,GAAA,IAAC,GAAG,CAAMP,GAAS,CAATA,UAAQ,CAAC,CAAgB,aAAQ,CAAR,QAAQ,CACzC,CAAAK,GAUK,CACJ,CAAAC,GAID,CACF,EAjBC,GAAG,CAiBE;IAAAzB,CAAA,OAAAmB,SAAA;IAAAnB,CAAA,OAAAwB,GAAA;IAAAxB,CAAA,OAAAyB,GAAA;IAAAzB,CAAA,OAAA0B,GAAA;EAAA;IAAAA,GAAA,GAAA1B,CAAA;EAAA;EAAA,OAjBN0B,GAiBM;AAAA","ignoreList":[]}