2025-09-10 05:45:12 +08:00
|
|
|
/**
|
|
|
|
|
* @license
|
|
|
|
|
* Copyright 2025 Google LLC
|
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
import type React from 'react';
|
2026-03-12 01:44:12 +05:30
|
|
|
import { useState, useEffect } from 'react';
|
2025-09-10 05:45:12 +08:00
|
|
|
import { Box, Text } from 'ink';
|
|
|
|
|
import { Colors } from '../../colors.js';
|
|
|
|
|
import type { SettingEnumOption } from '../../../config/settingsSchema.js';
|
|
|
|
|
|
|
|
|
|
interface EnumSelectorProps {
|
|
|
|
|
options: readonly SettingEnumOption[];
|
|
|
|
|
currentValue: string | number;
|
|
|
|
|
isActive: boolean;
|
|
|
|
|
onValueChange: (value: string | number) => void;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* A left-right scrolling selector for enum values
|
|
|
|
|
*/
|
|
|
|
|
export function EnumSelector({
|
|
|
|
|
options,
|
|
|
|
|
currentValue,
|
|
|
|
|
isActive,
|
|
|
|
|
onValueChange: _onValueChange,
|
|
|
|
|
}: EnumSelectorProps): React.JSX.Element {
|
|
|
|
|
const [currentIndex, setCurrentIndex] = useState(() => {
|
|
|
|
|
// Guard against empty options array
|
|
|
|
|
if (!options || options.length === 0) {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
const index = options.findIndex((option) => option.value === currentValue);
|
|
|
|
|
return index >= 0 ? index : 0;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Update index when currentValue changes externally
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
// Guard against empty options array
|
|
|
|
|
if (!options || options.length === 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
const index = options.findIndex((option) => option.value === currentValue);
|
|
|
|
|
// Always update index, defaulting to 0 if value not found
|
|
|
|
|
setCurrentIndex(index >= 0 ? index : 0);
|
|
|
|
|
}, [currentValue, options]);
|
|
|
|
|
|
|
|
|
|
// Guard against empty options array
|
|
|
|
|
if (!options || options.length === 0) {
|
|
|
|
|
return <Box />;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Left/right navigation is handled by parent component
|
|
|
|
|
// This component is purely for display
|
|
|
|
|
// onValueChange is kept for interface compatibility but not used internally
|
|
|
|
|
|
|
|
|
|
const currentOption = options[currentIndex] || options[0];
|
|
|
|
|
const canScrollLeft = options.length > 1;
|
|
|
|
|
const canScrollRight = options.length > 1;
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<Box flexDirection="row" alignItems="center">
|
|
|
|
|
<Text
|
|
|
|
|
color={isActive && canScrollLeft ? Colors.AccentGreen : Colors.Gray}
|
|
|
|
|
>
|
|
|
|
|
{canScrollLeft ? '←' : ' '}
|
|
|
|
|
</Text>
|
|
|
|
|
<Text> </Text>
|
|
|
|
|
<Text
|
|
|
|
|
color={isActive ? Colors.AccentGreen : Colors.Foreground}
|
|
|
|
|
bold={isActive}
|
|
|
|
|
>
|
|
|
|
|
{currentOption.label}
|
|
|
|
|
</Text>
|
|
|
|
|
<Text> </Text>
|
|
|
|
|
<Text
|
|
|
|
|
color={isActive && canScrollRight ? Colors.AccentGreen : Colors.Gray}
|
|
|
|
|
>
|
|
|
|
|
{canScrollRight ? '→' : ' '}
|
|
|
|
|
</Text>
|
|
|
|
|
</Box>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Export the interface for external use
|
|
|
|
|
export type { EnumSelectorProps };
|