[5d6f37a] | 1 | import { useFormContext, Controller } from 'react-hook-form';
|
---|
| 2 | // @mui
|
---|
| 3 | import { Theme, SxProps } from '@mui/material/styles';
|
---|
| 4 | import Box from '@mui/material/Box';
|
---|
| 5 | import Chip from '@mui/material/Chip';
|
---|
| 6 | import MenuItem from '@mui/material/MenuItem';
|
---|
| 7 | import Checkbox from '@mui/material/Checkbox';
|
---|
| 8 | import InputLabel from '@mui/material/InputLabel';
|
---|
| 9 | import FormControl from '@mui/material/FormControl';
|
---|
| 10 | import OutlinedInput from '@mui/material/OutlinedInput';
|
---|
| 11 | import FormHelperText from '@mui/material/FormHelperText';
|
---|
| 12 | import Select, { SelectProps } from '@mui/material/Select';
|
---|
| 13 | import TextField, { TextFieldProps } from '@mui/material/TextField';
|
---|
| 14 |
|
---|
| 15 | // ----------------------------------------------------------------------
|
---|
| 16 |
|
---|
| 17 | type RHFSelectProps = TextFieldProps & {
|
---|
| 18 | name: string;
|
---|
| 19 | native?: boolean;
|
---|
| 20 | maxHeight?: boolean | number;
|
---|
| 21 | children: React.ReactNode;
|
---|
| 22 | PaperPropsSx?: SxProps<Theme>;
|
---|
| 23 | };
|
---|
| 24 |
|
---|
| 25 | export function RHFSelect({
|
---|
| 26 | name,
|
---|
| 27 | native,
|
---|
| 28 | maxHeight = 220,
|
---|
| 29 | helperText,
|
---|
| 30 | children,
|
---|
| 31 | PaperPropsSx,
|
---|
| 32 | ...other
|
---|
| 33 | }: RHFSelectProps) {
|
---|
| 34 | const { control } = useFormContext();
|
---|
| 35 |
|
---|
| 36 | return (
|
---|
| 37 | <Controller
|
---|
| 38 | name={name}
|
---|
| 39 | control={control}
|
---|
| 40 | render={({ field, fieldState: { error } }) => (
|
---|
| 41 | <TextField
|
---|
| 42 | {...field}
|
---|
| 43 | select
|
---|
| 44 | fullWidth
|
---|
| 45 | SelectProps={{
|
---|
| 46 | native,
|
---|
| 47 | MenuProps: {
|
---|
| 48 | PaperProps: {
|
---|
| 49 | sx: {
|
---|
| 50 | ...(!native && {
|
---|
| 51 | maxHeight: typeof maxHeight === 'number' ? maxHeight : 'unset',
|
---|
| 52 | }),
|
---|
| 53 | ...PaperPropsSx,
|
---|
| 54 | },
|
---|
| 55 | },
|
---|
| 56 | },
|
---|
| 57 | sx: { textTransform: 'capitalize' },
|
---|
| 58 | }}
|
---|
| 59 | error={!!error}
|
---|
| 60 | helperText={error ? error?.message : helperText}
|
---|
| 61 | {...other}
|
---|
| 62 | >
|
---|
| 63 | {children}
|
---|
| 64 | </TextField>
|
---|
| 65 | )}
|
---|
| 66 | />
|
---|
| 67 | );
|
---|
| 68 | }
|
---|
| 69 |
|
---|
| 70 | // ----------------------------------------------------------------------
|
---|
| 71 |
|
---|
| 72 | type RHFMultiSelectProps = SelectProps & {
|
---|
| 73 | name: string;
|
---|
| 74 | label?: string;
|
---|
| 75 | chip?: boolean;
|
---|
| 76 | checkbox?: boolean;
|
---|
| 77 | placeholder?: string;
|
---|
| 78 | helperText?: React.ReactNode;
|
---|
| 79 | options: {
|
---|
| 80 | label: string;
|
---|
| 81 | value: string;
|
---|
| 82 | }[];
|
---|
| 83 | };
|
---|
| 84 |
|
---|
| 85 | export function RHFMultiSelect({
|
---|
| 86 | name,
|
---|
| 87 | chip,
|
---|
| 88 | label,
|
---|
| 89 | options,
|
---|
| 90 | checkbox,
|
---|
| 91 | placeholder,
|
---|
| 92 | helperText,
|
---|
| 93 | sx,
|
---|
| 94 | ...other
|
---|
| 95 | }: RHFMultiSelectProps) {
|
---|
| 96 | const { control } = useFormContext();
|
---|
| 97 |
|
---|
| 98 | const renderValues = (selectedIds: string[]) => {
|
---|
| 99 | const selectedItems = options.filter((item) => selectedIds.includes(item.value));
|
---|
| 100 |
|
---|
| 101 | if (!selectedItems.length && placeholder) {
|
---|
| 102 | return (
|
---|
| 103 | <Box component="em" sx={{ color: 'text.disabled' }}>
|
---|
| 104 | {placeholder}
|
---|
| 105 | </Box>
|
---|
| 106 | );
|
---|
| 107 | }
|
---|
| 108 |
|
---|
| 109 | if (chip) {
|
---|
| 110 | return (
|
---|
| 111 | <Box sx={{ display: 'flex', flexWrap: 'wrap', gap: 0.5 }}>
|
---|
| 112 | {selectedItems.map((item) => (
|
---|
| 113 | <Chip key={item.value} size="small" label={item.label} />
|
---|
| 114 | ))}
|
---|
| 115 | </Box>
|
---|
| 116 | );
|
---|
| 117 | }
|
---|
| 118 |
|
---|
| 119 | return selectedItems.map((item) => item.label).join(', ');
|
---|
| 120 | };
|
---|
| 121 |
|
---|
| 122 | return (
|
---|
| 123 | <Controller
|
---|
| 124 | name={name}
|
---|
| 125 | control={control}
|
---|
| 126 | render={({ field, fieldState: { error } }) => (
|
---|
| 127 | <FormControl sx={sx}>
|
---|
| 128 | {label && <InputLabel id={name}> {label} </InputLabel>}
|
---|
| 129 |
|
---|
| 130 | <Select
|
---|
| 131 | {...field}
|
---|
| 132 | multiple
|
---|
| 133 | displayEmpty={!!placeholder}
|
---|
| 134 | labelId={name}
|
---|
| 135 | input={<OutlinedInput fullWidth label={label} error={!!error} />}
|
---|
| 136 | renderValue={renderValues}
|
---|
| 137 | {...other}
|
---|
| 138 | >
|
---|
| 139 | {placeholder && (
|
---|
| 140 | <MenuItem disabled value="">
|
---|
| 141 | <em> {placeholder} </em>
|
---|
| 142 | </MenuItem>
|
---|
| 143 | )}
|
---|
| 144 |
|
---|
| 145 | {options.map((option) => {
|
---|
| 146 | const selected = field.value.includes(option.value);
|
---|
| 147 |
|
---|
| 148 | return (
|
---|
| 149 | <MenuItem key={option.value} value={option.value}>
|
---|
| 150 | {checkbox && <Checkbox size="small" disableRipple checked={selected} />}
|
---|
| 151 |
|
---|
| 152 | {option.label}
|
---|
| 153 | </MenuItem>
|
---|
| 154 | );
|
---|
| 155 | })}
|
---|
| 156 | </Select>
|
---|
| 157 |
|
---|
| 158 | {(!!error || helperText) && (
|
---|
| 159 | <FormHelperText error={!!error}>{error ? error?.message : helperText}</FormHelperText>
|
---|
| 160 | )}
|
---|
| 161 | </FormControl>
|
---|
| 162 | )}
|
---|
| 163 | />
|
---|
| 164 | );
|
---|
| 165 | }
|
---|