anvil/striker-ui/components/SelectWithLabel.tsx

149 lines
3.5 KiB
TypeScript
Raw Normal View History

2022-05-05 20:46:00 +00:00
import {
Checkbox as MUICheckbox,
FormControl as MUIFormControl,
selectClasses as muiSelectClasses,
2022-05-05 20:46:00 +00:00
} from '@mui/material';
import { FC, useCallback, useMemo } from 'react';
2022-05-05 20:46:00 +00:00
import InputMessageBox from './InputMessageBox';
import MenuItem from './MenuItem';
import OutlinedInput from './OutlinedInput';
import OutlinedInputLabel from './OutlinedInputLabel';
import Select from './Select';
2022-05-05 20:46:00 +00:00
const SelectWithLabel = <
Value = string,
Display extends React.ReactNode = React.ReactNode,
>(
...[props]: Parameters<FC<SelectWithLabelProps<Value, Display>>>
): ReturnType<FC<SelectWithLabelProps<Value, Display>>> => {
const {
id,
label,
selectItems,
checkItem,
disableItem,
formControlProps,
hideItem,
inputLabelProps = {},
isReadOnly = false,
messageBoxProps = {},
name,
onBlur,
onChange,
onFocus,
required: isRequired,
selectProps: {
multiple: selectMultiple,
sx: selectSx,
...restSelectProps
} = {},
value: selectValue,
// Props with initial value that depend on others.
isCheckableItems = selectMultiple,
} = props;
const combinedSx = useMemo(
() =>
isReadOnly
? {
[`& .${muiSelectClasses.icon}`]: {
visibility: 'hidden',
},
...selectSx,
}
: selectSx,
[isReadOnly, selectSx],
);
const createCheckbox = useCallback(
(value) =>
isCheckableItems && (
<MUICheckbox checked={checkItem?.call(null, value)} />
),
[checkItem, isCheckableItems],
);
const createMenuItem = useCallback(
(value, displayValue) => (
<MenuItem
disabled={disableItem?.call(null, value)}
key={`${id}-${value}`}
sx={{
display: hideItem?.call(null, value) ? 'none' : undefined,
}}
value={value}
>
{createCheckbox(value)}
{displayValue}
</MenuItem>
),
[createCheckbox, disableItem, hideItem, id],
);
const selectId = useMemo(() => `${id}-select-element`, [id]);
const inputElement = useMemo(
() => <OutlinedInput id={id} label={label} />,
[id, label],
);
const labelElement = useMemo(
() =>
label && (
<OutlinedInputLabel
htmlFor={selectId}
isNotifyRequired={isRequired}
{...inputLabelProps}
>
{label}
</OutlinedInputLabel>
),
[inputLabelProps, isRequired, label, selectId],
);
const menuItemElements = useMemo(
() =>
selectItems.map((item) => {
/**
* Cases:
* 1. item is string
* 2. item is SelectItem with only value
* 3. item is SelectItem with both value, and displayValue
*/
if (typeof item === 'string') return createMenuItem(item, item);
const {
value,
displayValue = String(value),
}: SelectItem<Value, Display> = item;
return createMenuItem(value, displayValue);
}),
[createMenuItem, selectItems],
);
return (
<MUIFormControl fullWidth {...formControlProps}>
{labelElement}
<Select<Value>
id={selectId}
input={inputElement}
multiple={selectMultiple}
name={name}
onBlur={onBlur}
onChange={onChange}
onFocus={onFocus}
readOnly={isReadOnly}
value={selectValue}
{...restSelectProps}
sx={combinedSx}
>
{menuItemElements}
</Select>
<InputMessageBox {...messageBoxProps} />
</MUIFormControl>
);
};
2022-05-05 20:46:00 +00:00
export default SelectWithLabel;