anvil/striker-ui/components/Slider.tsx

299 lines
7.4 KiB
TypeScript
Raw Normal View History

2022-04-06 05:03:32 +00:00
import { useState } from 'react';
import {
Box,
inputLabelClasses as muiInputLabelClasses,
OutlinedInputProps as MUIOutlinedInputProps,
2022-04-06 05:03:32 +00:00
outlinedInputClasses as muiOutlinedInputClasses,
Slider as MUISlider,
sliderClasses as muiSliderClasses,
SliderProps as MUISliderProps,
FormControl,
2022-04-06 05:03:32 +00:00
} from '@mui/material';
import { BORDER_RADIUS, GREY } from '../lib/consts/DEFAULT_THEME';
import InputMessageBox from './InputMessageBox';
import { MessageBoxProps } from './MessageBox';
import OutlinedInput, { OutlinedInputProps } from './OutlinedInput';
import OutlinedInputLabel, {
OutlinedInputLabelProps,
} from './OutlinedInputLabel';
2022-04-06 05:03:32 +00:00
import { BodyText } from './Text';
type SliderOnBlur = Exclude<MUISliderProps['onBlur'], undefined>;
type SliderOnChange = Exclude<MUISliderProps['onChange'], undefined>;
type SliderOnFocus = Exclude<MUISliderProps['onFocus'], undefined>;
type SliderValue = Exclude<MUISliderProps['value'], undefined>;
type SliderOptionalProps = {
inputLabelProps?: Partial<OutlinedInputLabelProps>;
2022-04-06 05:03:32 +00:00
isAllowTextInput?: boolean;
labelId?: string;
messageBoxProps?: Partial<MessageBoxProps>;
sliderProps?: Omit<MUISliderProps, 'onChange'> & {
onChange?: (value: number | number[]) => void;
};
2022-04-06 05:03:32 +00:00
};
type SliderProps = {
label: string;
value: SliderValue;
} & SliderOptionalProps;
type TextInputOnChange = Exclude<MUIOutlinedInputProps['onChange'], undefined>;
2022-04-06 05:03:32 +00:00
const SLIDER_DEFAULT_PROPS: Required<SliderOptionalProps> = {
inputLabelProps: {},
2022-04-06 05:03:32 +00:00
isAllowTextInput: false,
labelId: '',
messageBoxProps: {},
2022-04-06 05:03:32 +00:00
sliderProps: {},
};
const SLIDER_INPUT_LABEL_DECORATOR_CLASS_PREFIX = 'SliderInputLabelDecorator';
const SLIDER_INPUT_LABEL_DECORATOR_CLASSES = {
root: `${SLIDER_INPUT_LABEL_DECORATOR_CLASS_PREFIX}-root`,
};
const createInputLabelDecorator = ({
isFocused,
label,
}: {
isFocused?: boolean;
label: string;
}) => {
2022-04-06 05:03:32 +00:00
const borderColor = GREY;
const borderStyle = 'solid';
const content = '""';
let rootTop = '0';
let labelGapMargin = '0 .6em 0 .4em';
let borderWidth = '1px 0 0 0';
let opacity = '0.3';
if (isFocused) {
rootTop = '-1px';
labelGapMargin = '0 1em 0 1em';
borderWidth = '2px 0 0 0';
opacity = '1';
}
2022-04-06 05:03:32 +00:00
return (
<Box
className={SLIDER_INPUT_LABEL_DECORATOR_CLASSES.root}
2022-04-06 05:03:32 +00:00
sx={{
display: 'flex',
flexDirection: 'row',
position: 'absolute',
top: rootTop,
width: 'calc(100% - 6px)',
2022-04-06 05:03:32 +00:00
'> :last-child': {
flexGrow: 1,
},
}}
>
<Box
sx={{
borderColor,
borderStyle,
borderWidth,
content,
opacity,
width: '.6em',
}}
/>
<BodyText
sx={{
fontSize: '.75em',
margin: labelGapMargin,
2022-04-06 05:03:32 +00:00
visibility: 'hidden',
}}
text={label}
/>
<Box
sx={{
borderColor,
borderStyle,
borderWidth,
content,
opacity,
}}
/>
</Box>
);
};
const createOutlinedInput = ({
key,
isFocused,
...inputRestProps
}: OutlinedInputProps & {
key: string;
isFocused?: boolean;
}) => (
2022-04-06 05:03:32 +00:00
<OutlinedInput
{...{
key,
2022-04-06 05:03:32 +00:00
className: isFocused ? muiOutlinedInputClasses.focused : '',
type: 'number',
...inputRestProps,
2022-04-06 05:03:32 +00:00
}}
/>
);
const stringToNumber = (value: string, fallback: number) => {
const converted = Number.parseFloat(value);
return Number.isNaN(converted) ? fallback : converted;
};
const toRangeString = (value: SliderValue) =>
value instanceof Array
? value.map((element) => String(element))
: [String(value)];
const toSliderValue = (rangeString: string[], value: SliderValue) =>
value instanceof Array
? rangeString.map((element, index) => stringToNumber(element, value[index]))
: stringToNumber(rangeString[0], value);
2022-04-06 05:03:32 +00:00
const Slider = ({
messageBoxProps = SLIDER_DEFAULT_PROPS.messageBoxProps,
isAllowTextInput = SLIDER_DEFAULT_PROPS.isAllowTextInput,
2022-04-06 05:03:32 +00:00
label,
labelId = SLIDER_DEFAULT_PROPS.labelId,
inputLabelProps = SLIDER_DEFAULT_PROPS.inputLabelProps,
sliderProps = SLIDER_DEFAULT_PROPS.sliderProps,
2022-04-06 05:03:32 +00:00
value,
}: SliderProps): JSX.Element => {
const {
max,
min,
2022-04-06 05:03:32 +00:00
onChange: sliderChangeCallback,
sx: sliderSx,
valueLabelDisplay: sliderValueLabelDisplay,
} = sliderProps;
let assignableValue: SliderValue = value;
const [textRangeValue, SetTextRangeValue] = useState<{ range: string[] }>({
range: toRangeString(value),
});
2022-04-06 05:03:32 +00:00
const [isFocused, setIsFocused] = useState<boolean>(false);
const handleLocalSliderBlur: SliderOnBlur = () => {
setIsFocused(false);
};
const handleLocalSliderFocus: SliderOnFocus = () => {
setIsFocused(true);
};
const handleSliderChange: SliderOnChange = (event, newValue) => {
SetTextRangeValue({
range: toRangeString(newValue),
});
sliderChangeCallback?.call(null, newValue);
};
const handleTextInputChange: TextInputOnChange = () => {
assignableValue = toSliderValue(textRangeValue.range, assignableValue);
sliderChangeCallback?.call(null, assignableValue);
};
2022-04-06 05:03:32 +00:00
return (
<FormControl
sx={{
display: 'flex',
flexDirection: 'column',
'&:hover': {
[`& .${SLIDER_INPUT_LABEL_DECORATOR_CLASSES.root} div`]: {
opacity: 1,
},
[`& .${muiOutlinedInputClasses.notchedOutline}`]: {
borderColor: GREY,
},
},
}}
>
2022-04-06 05:03:32 +00:00
<OutlinedInputLabel
{...{
className: isFocused ? muiInputLabelClasses.focused : '',
id: labelId,
shrink: true,
...inputLabelProps,
2022-04-06 05:03:32 +00:00
}}
>
{label}
</OutlinedInputLabel>
{createInputLabelDecorator({ isFocused, label })}
2022-04-06 05:03:32 +00:00
<Box
sx={{
alignItems: 'center',
display: 'flex',
flexDirection: 'row',
2022-04-06 05:03:32 +00:00
'> :first-child': { flexGrow: 1 },
}}
>
<MUISlider
{...{
'aria-labelledby': labelId,
max,
min,
2022-04-06 05:03:32 +00:00
onBlur: handleLocalSliderBlur,
onChange: handleSliderChange,
onFocus: handleLocalSliderFocus,
sx: {
color: GREY,
marginLeft: '1em',
marginRight: '1em',
2022-04-06 05:03:32 +00:00
[`& .${muiSliderClasses.thumb}`]: {
borderRadius: BORDER_RADIUS,
transform: 'translate(-50%, -50%) rotate(45deg)',
},
...sliderSx,
},
value: assignableValue,
2022-04-06 05:03:32 +00:00
valueLabelDisplay: sliderValueLabelDisplay,
}}
/>
{textRangeValue.range.map((textValue, textValueIndex) =>
createOutlinedInput({
key: `slider-nested-text-input-${textValueIndex}`,
inputProps: { max, min },
isFocused,
onBlur: handleLocalSliderBlur,
onChange: (...args) => {
textRangeValue.range[textValueIndex] = args[0].target.value;
SetTextRangeValue({ ...textRangeValue });
handleTextInputChange(...args);
},
onFocus: handleLocalSliderFocus,
sx: isAllowTextInput
? undefined
: {
visibility: 'collapse',
},
value: textValue,
}),
)}
2022-04-06 05:03:32 +00:00
</Box>
<InputMessageBox {...messageBoxProps} />
</FormControl>
2022-04-06 05:03:32 +00:00
);
};
Slider.defaultProps = SLIDER_DEFAULT_PROPS;
export type { SliderProps };
export default Slider;