2023-02-28 06:31:18 +00:00
|
|
|
import {
|
2023-03-15 06:09:12 +00:00
|
|
|
Add as MUIAddIcon,
|
|
|
|
Close as MUICloseIcon,
|
2023-02-28 06:31:18 +00:00
|
|
|
Done as MUIDoneIcon,
|
|
|
|
Edit as MUIEditIcon,
|
2023-03-28 02:02:36 +00:00
|
|
|
PlayCircle as MUIPlayCircleIcon,
|
2023-02-28 06:31:18 +00:00
|
|
|
Visibility as MUIVisibilityIcon,
|
|
|
|
VisibilityOff as MUIVisibilityOffIcon,
|
|
|
|
} from '@mui/icons-material';
|
2022-05-13 18:46:10 +00:00
|
|
|
import {
|
|
|
|
IconButton as MUIIconButton,
|
|
|
|
IconButtonProps as MUIIconButtonProps,
|
2022-07-20 22:18:48 +00:00
|
|
|
inputClasses as muiInputClasses,
|
2023-02-28 06:31:18 +00:00
|
|
|
styled,
|
2022-05-13 18:46:10 +00:00
|
|
|
} from '@mui/material';
|
2023-02-28 06:31:18 +00:00
|
|
|
import { createElement, FC, ReactNode, useMemo } from 'react';
|
2022-02-17 20:09:29 +00:00
|
|
|
|
|
|
|
import {
|
|
|
|
BLACK,
|
2023-03-15 05:23:18 +00:00
|
|
|
BLUE,
|
2022-02-17 20:09:29 +00:00
|
|
|
BORDER_RADIUS,
|
2022-07-20 22:18:48 +00:00
|
|
|
DISABLED,
|
2022-02-17 20:09:29 +00:00
|
|
|
GREY,
|
|
|
|
} from '../../lib/consts/DEFAULT_THEME';
|
|
|
|
|
2023-02-28 06:31:18 +00:00
|
|
|
type IconButtonProps = IconButtonOptionalProps & MUIIconButtonProps;
|
|
|
|
|
|
|
|
const ContainedIconButton = styled(MUIIconButton)({
|
|
|
|
borderRadius: BORDER_RADIUS,
|
|
|
|
backgroundColor: GREY,
|
|
|
|
color: BLACK,
|
|
|
|
|
|
|
|
'&:hover': {
|
2023-02-28 21:03:32 +00:00
|
|
|
backgroundColor: `${GREY}F0`,
|
2023-02-28 06:31:18 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
[`&.${muiInputClasses.disabled}`]: {
|
|
|
|
backgroundColor: DISABLED,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
const NormalIconButton = styled(MUIIconButton)({
|
|
|
|
color: GREY,
|
|
|
|
});
|
|
|
|
|
2023-03-15 06:09:12 +00:00
|
|
|
const MAP_TO_ADD_ICON: IconButtonMapToStateIconBundle = {
|
|
|
|
none: { iconType: MUIAddIcon },
|
|
|
|
};
|
|
|
|
|
|
|
|
const MAP_TO_CLOSE_ICON: IconButtonMapToStateIconBundle = {
|
|
|
|
none: { iconType: MUICloseIcon },
|
2023-02-28 06:31:18 +00:00
|
|
|
};
|
|
|
|
|
2023-03-15 05:23:18 +00:00
|
|
|
const MAP_TO_EDIT_ICON: IconButtonMapToStateIconBundle = {
|
|
|
|
false: { iconType: MUIEditIcon },
|
|
|
|
true: { iconType: MUIDoneIcon, iconProps: { sx: { color: BLUE } } },
|
2023-02-28 06:31:18 +00:00
|
|
|
};
|
|
|
|
|
2023-03-28 02:02:36 +00:00
|
|
|
const MAP_TO_PLAY_ICON: IconButtonMapToStateIconBundle = {
|
|
|
|
none: { iconType: MUIPlayCircleIcon },
|
|
|
|
};
|
|
|
|
|
2023-03-15 06:09:12 +00:00
|
|
|
const MAP_TO_VISIBILITY_ICON: IconButtonMapToStateIconBundle = {
|
|
|
|
false: { iconType: MUIVisibilityIcon },
|
|
|
|
true: { iconType: MUIVisibilityOffIcon },
|
|
|
|
};
|
|
|
|
|
2023-02-28 06:31:18 +00:00
|
|
|
const MAP_TO_MAP_PRESET: Record<
|
2023-03-15 05:23:18 +00:00
|
|
|
IconButtonPresetMapToStateIconBundle,
|
|
|
|
IconButtonMapToStateIconBundle
|
2023-02-28 06:31:18 +00:00
|
|
|
> = {
|
2023-03-15 06:09:12 +00:00
|
|
|
add: MAP_TO_ADD_ICON,
|
|
|
|
close: MAP_TO_CLOSE_ICON,
|
2023-02-28 06:31:18 +00:00
|
|
|
edit: MAP_TO_EDIT_ICON,
|
2023-03-28 02:02:36 +00:00
|
|
|
play: MAP_TO_PLAY_ICON,
|
2023-02-28 06:31:18 +00:00
|
|
|
visibility: MAP_TO_VISIBILITY_ICON,
|
|
|
|
};
|
|
|
|
|
|
|
|
const MAP_TO_VARIANT: Record<IconButtonVariant, CreatableComponent> = {
|
|
|
|
contained: ContainedIconButton,
|
|
|
|
normal: NormalIconButton,
|
|
|
|
};
|
2022-05-13 18:46:10 +00:00
|
|
|
|
|
|
|
const IconButton: FC<IconButtonProps> = ({
|
|
|
|
children,
|
2023-02-28 06:31:18 +00:00
|
|
|
defaultIcon,
|
|
|
|
iconProps,
|
|
|
|
mapPreset,
|
|
|
|
mapToIcon: externalMapToIcon,
|
2023-03-15 06:09:12 +00:00
|
|
|
state = 'none',
|
2023-02-28 06:31:18 +00:00
|
|
|
variant = 'contained',
|
|
|
|
...restIconButtonProps
|
|
|
|
}) => {
|
2023-03-15 05:23:18 +00:00
|
|
|
const mapToIcon = useMemo<IconButtonMapToStateIconBundle | undefined>(
|
2023-02-28 06:31:18 +00:00
|
|
|
() => externalMapToIcon ?? (mapPreset && MAP_TO_MAP_PRESET[mapPreset]),
|
|
|
|
[externalMapToIcon, mapPreset],
|
|
|
|
);
|
|
|
|
|
|
|
|
const iconButtonContent = useMemo(() => {
|
|
|
|
let result: ReactNode;
|
|
|
|
|
|
|
|
if (mapToIcon) {
|
2023-03-15 06:09:12 +00:00
|
|
|
const { iconType, iconProps: presetIconProps } = mapToIcon[state] ?? {
|
|
|
|
iconType: defaultIcon,
|
|
|
|
};
|
2023-03-15 05:23:18 +00:00
|
|
|
|
|
|
|
if (iconType) {
|
|
|
|
result = createElement(iconType, {
|
|
|
|
...presetIconProps,
|
|
|
|
...iconProps,
|
|
|
|
});
|
2023-02-28 06:31:18 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
result = children;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
2023-03-15 06:09:12 +00:00
|
|
|
}, [mapToIcon, state, defaultIcon, iconProps, children]);
|
2023-02-28 06:31:18 +00:00
|
|
|
const iconButtonElementType = useMemo(
|
|
|
|
() => MAP_TO_VARIANT[variant],
|
|
|
|
[variant],
|
|
|
|
);
|
|
|
|
|
|
|
|
return createElement(
|
|
|
|
iconButtonElementType,
|
|
|
|
restIconButtonProps,
|
|
|
|
iconButtonContent,
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export type { IconButtonProps };
|
2022-02-17 20:09:29 +00:00
|
|
|
|
|
|
|
export default IconButton;
|