πŸ“„ 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).

  • react
  • figures

πŸ–₯️ 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":"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","ignoreList":[]}