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 | }
|
---|