refactor: Improve car color selection UI and interaction

- Refined color selection styling with more consistent hover and selection states
- Added dynamic styling for selected and custom color options
- Optimized layout and spacing of color selection buttons
- Improved ref handling for the component
This commit is contained in:
RustamRu 2025-03-11 23:08:08 +03:00
parent 0b9b2f4dbc
commit 2f8a172b12

View File

@ -1,11 +1,5 @@
import React, { forwardRef, useState } from 'react'; import React, { forwardRef, useState } from 'react';
import { import { Input, Box, Stack, Text, Flex } from '@chakra-ui/react';
Input,
Box,
Stack,
Text,
Flex,
} from '@chakra-ui/react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { CAR_COLORS } from './helper'; import { CAR_COLORS } from './helper';
@ -18,7 +12,7 @@ interface CarColorSelectProps {
} }
export const CarColorSelect = forwardRef<HTMLInputElement, CarColorSelectProps>( export const CarColorSelect = forwardRef<HTMLInputElement, CarColorSelectProps>(
function CarColorSelect(props) { function CarColorSelect(props, ref) {
const [customColor, setCustomColor] = useState(''); const [customColor, setCustomColor] = useState('');
const [isCustom, setIsCustom] = useState(false); const [isCustom, setIsCustom] = useState(false);
@ -33,7 +27,9 @@ export const CarColorSelect = forwardRef<HTMLInputElement, CarColorSelectProps>(
} as React.ChangeEvent<HTMLInputElement>); } as React.ChangeEvent<HTMLInputElement>);
}; };
const handleCustomColorChange = (e: React.ChangeEvent<HTMLInputElement>) => { const handleCustomColorChange = (
e: React.ChangeEvent<HTMLInputElement>,
) => {
const value = e.target.value; const value = e.target.value;
setCustomColor(value); setCustomColor(value);
props.onChange?.({ props.onChange?.({
@ -48,46 +44,56 @@ export const CarColorSelect = forwardRef<HTMLInputElement, CarColorSelectProps>(
const currentValue = isCustom ? 'custom' : props.value; const currentValue = isCustom ? 'custom' : props.value;
return ( return (
<Stack spacing={4} width="100%"> <Stack spacing={4} width='100%' ref={ref}>
<Flex gap={3} wrap="nowrap" overflowX="auto" pb={2}> <Flex gap={2} wrap='wrap' pb={2}>
{CAR_COLORS.map(({ name, code }) => ( {CAR_COLORS.map(({ name, code }) => (
<Box <Box
key={name} key={name}
flexShrink={0} flexShrink={0}
as="button" as='button'
type="button" type='button'
onClick={() => handleColorChange(name)} onClick={() => handleColorChange(name)}
> >
<Flex <Flex
align="center" align='center'
gap={2} gap={2}
p={2} p={1}
borderRadius="full" borderRadius='full'
borderWidth="2px" borderWidth='2px'
borderColor={currentValue === name ? 'primary.500' : 'gray.200'} borderColor='gray.200'
bg={currentValue === name ? 'primary.50' : 'white'} bg='white'
_hover={{ _hover={{
borderColor: 'primary.500', borderColor: 'primary.500',
bg: currentValue === name ? 'primary.50' : 'gray.50' bg: 'gray.50',
}} }}
minW={currentValue === name ? '120px' : 'auto'} justify='center'
h="48px" transition='all 0.2s'
justify="center" {...(currentValue === name && {
transition="all 0.2s" borderColor: 'primary.500',
bg: 'primary.50',
paddingInlineEnd: 3,
_hover: {
bg: 'primary.50',
},
})}
> >
<Flex align="center" gap={2}> <Flex align='center' gap={2}>
<Box <Box
w="32px" w='32px'
h="32px" h='32px'
borderRadius="full" borderRadius='full'
bg={code} bg={code}
border="1px" border='1px'
borderColor={currentValue === name ? 'primary.500' : 'gray.200'} borderColor='gray.200'
transition="all 0.2s" transition='all 0.2s'
boxShadow={currentValue === name ? 'sm' : 'none'} boxShadow='none'
{...(currentValue === name && {
borderColor: 'primary.500',
boxShadow: 'sm',
})}
/> />
{currentValue === name && ( {currentValue === name && (
<Text fontSize="xs" color="primary.700" fontWeight="medium"> <Text fontSize='xs' color='primary.700' fontWeight='medium'>
{t(`colors.${name}`)} {t(`colors.${name}`)}
</Text> </Text>
)} )}
@ -95,60 +101,67 @@ export const CarColorSelect = forwardRef<HTMLInputElement, CarColorSelectProps>(
</Flex> </Flex>
</Box> </Box>
))} ))}
<Box <Box
flexShrink={0} flexShrink={0}
as="button" as='button'
type="button" type='button'
onClick={() => handleColorChange('custom')} onClick={() => handleColorChange('custom')}
> >
<Flex <Flex
align="center" align='center'
gap={2} gap={2}
p={2} p={1}
borderRadius="full" paddingInlineEnd={3}
borderWidth="2px" borderRadius='full'
borderColor={isCustom ? 'primary.500' : 'gray.200'} borderWidth='2px'
bg={isCustom ? 'primary.50' : 'white'} borderColor='gray.200'
_hover={{ bg='white'
_hover={{
borderColor: 'primary.500', borderColor: 'primary.500',
bg: isCustom ? 'primary.50' : 'gray.50' bg: 'gray.50',
}} }}
minW={isCustom ? '200px' : 'auto'} justify='center'
h="48px" transition='all 0.2s'
justify="center" {...(isCustom && {
transition="all 0.2s" borderColor: 'primary.500',
paddingInlineStart: 3,
bg: 'primary.50',
_hover: {
bg: 'primary.50',
},
})}
> >
{isCustom ? ( {isCustom ? (
<Flex gap={2} align="center"> <Flex gap={2} align='center'>
<Text fontSize="xs" color="primary.700" fontWeight="medium"> <Text fontSize='xs' color='primary.700' fontWeight='medium'>
{t('custom-label')} {t('custom-label')}
</Text> </Text>
<Input <Input
size="sm" size='sm'
width="120px" width='120px'
value={customColor} value={customColor}
onChange={handleCustomColorChange} onChange={handleCustomColorChange}
placeholder={t('placeholder')} placeholder={t('placeholder')}
onClick={(e) => e.stopPropagation()} onClick={(e) => e.stopPropagation()}
borderColor="primary.200" borderColor='primary.200'
_focus={{ _focus={{
borderColor: 'primary.500', borderColor: 'primary.500',
boxShadow: '0 0 0 1px var(--chakra-colors-primary-500)' boxShadow: '0 0 0 1px var(--chakra-colors-primary-500)',
}} }}
/> />
</Flex> </Flex>
) : ( ) : (
<Flex align="center" gap={2}> <Flex align='center' gap={2}>
<Box <Box
w="32px" w='32px'
h="32px" h='32px'
borderRadius="full" borderRadius='full'
bg="gray.100" bg='gray.100'
border="1px" border='1px'
borderColor="gray.200" borderColor='gray.200'
transition="all 0.2s" transition='all 0.2s'
/> />
<Text fontSize="xs" color="gray.500"> <Text fontSize='xs' color='gray.500'>
{t('custom')} {t('custom')}
</Text> </Text>
</Flex> </Flex>
@ -159,4 +172,4 @@ export const CarColorSelect = forwardRef<HTMLInputElement, CarColorSelectProps>(
</Stack> </Stack>
); );
}, },
); );