/** * @license * Copyright 2025 Google LLC * SPDX-License-Identifier: Apache-2.0 */ import React from 'react'; import { Box, Text } from 'ink'; import SelectInput, { type ItemProps as InkSelectItemProps, type IndicatorProps as InkSelectIndicatorProps, } from 'ink-select-input'; import { Colors } from '../../colors.js'; /** * Represents a single option for the RadioButtonSelect. * Requires a label for display and a value to be returned on selection. */ export interface RadioSelectItem { label: string; value: T; } /** * Props for the RadioButtonSelect component. * @template T The type of the value associated with each radio item. */ export interface RadioButtonSelectProps { /** An array of items to display as radio options. */ items: Array>; /** The initial index selected */ initialIndex?: number; /** Function called when an item is selected. Receives the `value` of the selected item. */ onSelect: (value: T) => void; /** Function called when an item is highlighted. Receives the `value` of the selected item. */ onHighlight?: (value: T) => void; /** Whether this select input is currently focused and should respond to input. */ isFocused?: boolean; } /** * Custom indicator component displaying radio button style (◉/○). */ function RadioIndicator({ isSelected = false, }: InkSelectIndicatorProps): React.JSX.Element { return ( {isSelected ? '◉' : '○'} ); } /** * Custom item component for displaying the label with appropriate color. */ function RadioItem({ isSelected = false, label, }: InkSelectItemProps): React.JSX.Element { return ( {label} ); } /** * A specialized SelectInput component styled to look like radio buttons. * It uses '◉' for selected and '○' for unselected items. * * @template T The type of the value associated with each radio item. */ export function RadioButtonSelect({ items, initialIndex, onSelect, onHighlight, isFocused, }: RadioButtonSelectProps): React.JSX.Element { const handleSelect = (item: RadioSelectItem) => { onSelect(item.value); }; const handleHighlight = (item: RadioSelectItem) => { if (onHighlight) { onHighlight(item.value); } }; initialIndex = initialIndex ?? 0; return ( ); }