@mui/x-date-pickers
Version:
The community edition of the Date and Time Picker components (MUI X).
365 lines (363 loc) • 11.4 kB
JavaScript
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard").default;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault").default;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.Clock = Clock;
var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends"));
var React = _interopRequireWildcard(require("react"));
var _clsx = _interopRequireDefault(require("clsx"));
var _IconButton = _interopRequireDefault(require("@mui/material/IconButton"));
var _Typography = _interopRequireDefault(require("@mui/material/Typography"));
var _styles = require("@mui/material/styles");
var _utils = require("@mui/utils");
var _ClockPointer = require("./ClockPointer");
var _usePickersTranslations = require("../hooks/usePickersTranslations");
var _useUtils = require("../internals/hooks/useUtils");
var _shared = require("./shared");
var _clockClasses = require("./clockClasses");
var _dateUtils = require("../internals/utils/date-utils");
var _jsxRuntime = require("react/jsx-runtime");
const useUtilityClasses = ownerState => {
const {
classes,
meridiemMode
} = ownerState;
const slots = {
root: ['root'],
clock: ['clock'],
wrapper: ['wrapper'],
squareMask: ['squareMask'],
pin: ['pin'],
amButton: ['amButton', meridiemMode === 'am' && 'selected'],
pmButton: ['pmButton', meridiemMode === 'pm' && 'selected'],
meridiemText: ['meridiemText']
};
return (0, _utils.unstable_composeClasses)(slots, _clockClasses.getClockUtilityClass, classes);
};
const ClockRoot = (0, _styles.styled)('div', {
name: 'MuiClock',
slot: 'Root',
overridesResolver: (_, styles) => styles.root
})(({
theme
}) => ({
display: 'flex',
justifyContent: 'center',
alignItems: 'center',
margin: theme.spacing(2)
}));
const ClockClock = (0, _styles.styled)('div', {
name: 'MuiClock',
slot: 'Clock',
overridesResolver: (_, styles) => styles.clock
})({
backgroundColor: 'rgba(0,0,0,.07)',
borderRadius: '50%',
height: 220,
width: 220,
flexShrink: 0,
position: 'relative',
pointerEvents: 'none'
});
const ClockWrapper = (0, _styles.styled)('div', {
name: 'MuiClock',
slot: 'Wrapper',
overridesResolver: (_, styles) => styles.wrapper
})({
'&:focus': {
outline: 'none'
}
});
const ClockSquareMask = (0, _styles.styled)('div', {
name: 'MuiClock',
slot: 'SquareMask',
overridesResolver: (_, styles) => styles.squareMask
})({
width: '100%',
height: '100%',
position: 'absolute',
pointerEvents: 'auto',
outline: 0,
// Disable scroll capabilities.
touchAction: 'none',
userSelect: 'none',
variants: [{
props: {
disabled: false
},
style: {
'@media (pointer: fine)': {
cursor: 'pointer',
borderRadius: '50%'
},
'&:active': {
cursor: 'move'
}
}
}]
});
const ClockPin = (0, _styles.styled)('div', {
name: 'MuiClock',
slot: 'Pin',
overridesResolver: (_, styles) => styles.pin
})(({
theme
}) => ({
width: 6,
height: 6,
borderRadius: '50%',
backgroundColor: (theme.vars || theme).palette.primary.main,
position: 'absolute',
top: '50%',
left: '50%',
transform: 'translate(-50%, -50%)'
}));
const meridiemButtonCommonStyles = (theme, meridiemMode) => ({
zIndex: 1,
bottom: 8,
paddingLeft: 4,
paddingRight: 4,
width: _shared.CLOCK_HOUR_WIDTH,
variants: [{
props: {
meridiemMode
},
style: {
backgroundColor: (theme.vars || theme).palette.primary.main,
color: (theme.vars || theme).palette.primary.contrastText,
'&:hover': {
backgroundColor: (theme.vars || theme).palette.primary.light
}
}
}]
});
const ClockAmButton = (0, _styles.styled)(_IconButton.default, {
name: 'MuiClock',
slot: 'AmButton',
overridesResolver: (_, styles) => styles.amButton
})(({
theme
}) => (0, _extends2.default)({}, meridiemButtonCommonStyles(theme, 'am'), {
// keeping it here to make TS happy
position: 'absolute',
left: 8
}));
const ClockPmButton = (0, _styles.styled)(_IconButton.default, {
name: 'MuiClock',
slot: 'PmButton',
overridesResolver: (_, styles) => styles.pmButton
})(({
theme
}) => (0, _extends2.default)({}, meridiemButtonCommonStyles(theme, 'pm'), {
// keeping it here to make TS happy
position: 'absolute',
right: 8
}));
const ClockMeridiemText = (0, _styles.styled)(_Typography.default, {
name: 'MuiClock',
slot: 'meridiemText',
overridesResolver: (_, styles) => styles.meridiemText
})({
overflow: 'hidden',
whiteSpace: 'nowrap',
textOverflow: 'ellipsis'
});
/**
* @ignore - internal component.
*/
function Clock(inProps) {
const props = (0, _styles.useThemeProps)({
props: inProps,
name: 'MuiClock'
});
const {
ampm,
ampmInClock,
autoFocus,
children,
value,
handleMeridiemChange,
isTimeDisabled,
meridiemMode,
minutesStep = 1,
onChange,
selectedId,
type,
viewValue,
viewRange: [minViewValue, maxViewValue],
disabled = false,
readOnly,
className
} = props;
const ownerState = props;
const utils = (0, _useUtils.useUtils)();
const translations = (0, _usePickersTranslations.usePickersTranslations)();
const isMoving = React.useRef(false);
const classes = useUtilityClasses(ownerState);
const isSelectedTimeDisabled = isTimeDisabled(viewValue, type);
const isPointerInner = !ampm && type === 'hours' && (viewValue < 1 || viewValue > 12);
const handleValueChange = (newValue, isFinish) => {
if (disabled || readOnly) {
return;
}
if (isTimeDisabled(newValue, type)) {
return;
}
onChange(newValue, isFinish);
};
const setTime = (event, isFinish) => {
let {
offsetX,
offsetY
} = event;
if (offsetX === undefined) {
const rect = event.target.getBoundingClientRect();
offsetX = event.changedTouches[0].clientX - rect.left;
offsetY = event.changedTouches[0].clientY - rect.top;
}
const newSelectedValue = type === 'seconds' || type === 'minutes' ? (0, _shared.getMinutes)(offsetX, offsetY, minutesStep) : (0, _shared.getHours)(offsetX, offsetY, Boolean(ampm));
handleValueChange(newSelectedValue, isFinish);
};
const handleTouchSelection = event => {
isMoving.current = true;
setTime(event, 'shallow');
};
const handleTouchEnd = event => {
if (isMoving.current) {
setTime(event, 'finish');
isMoving.current = false;
}
event.preventDefault();
};
const handleMouseMove = event => {
// event.buttons & PRIMARY_MOUSE_BUTTON
if (event.buttons > 0) {
setTime(event.nativeEvent, 'shallow');
}
};
const handleMouseUp = event => {
if (isMoving.current) {
isMoving.current = false;
}
setTime(event.nativeEvent, 'finish');
};
const hasSelected = React.useMemo(() => {
if (type === 'hours') {
return true;
}
return viewValue % 5 === 0;
}, [type, viewValue]);
const keyboardControlStep = type === 'minutes' ? minutesStep : 1;
const listboxRef = React.useRef(null);
// Since this is rendered when a Popper is opened we can't use passive effects.
// Focusing in passive effects in Popper causes scroll jump.
(0, _utils.unstable_useEnhancedEffect)(() => {
if (autoFocus) {
// The ref not being resolved would be a bug in MUI.
listboxRef.current.focus();
}
}, [autoFocus]);
const clampValue = newValue => Math.max(minViewValue, Math.min(maxViewValue, newValue));
const circleValue = newValue => (newValue + (maxViewValue + 1)) % (maxViewValue + 1);
const handleKeyDown = event => {
// TODO: Why this early exit?
if (isMoving.current) {
return;
}
switch (event.key) {
case 'Home':
// reset both hours and minutes
handleValueChange(minViewValue, 'partial');
event.preventDefault();
break;
case 'End':
handleValueChange(maxViewValue, 'partial');
event.preventDefault();
break;
case 'ArrowUp':
handleValueChange(circleValue(viewValue + keyboardControlStep), 'partial');
event.preventDefault();
break;
case 'ArrowDown':
handleValueChange(circleValue(viewValue - keyboardControlStep), 'partial');
event.preventDefault();
break;
case 'PageUp':
handleValueChange(clampValue(viewValue + 5), 'partial');
event.preventDefault();
break;
case 'PageDown':
handleValueChange(clampValue(viewValue - 5), 'partial');
event.preventDefault();
break;
case 'Enter':
case ' ':
handleValueChange(viewValue, 'finish');
event.preventDefault();
break;
default:
// do nothing
}
};
return /*#__PURE__*/(0, _jsxRuntime.jsxs)(ClockRoot, {
className: (0, _clsx.default)(classes.root, className),
children: [/*#__PURE__*/(0, _jsxRuntime.jsxs)(ClockClock, {
className: classes.clock,
children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(ClockSquareMask, {
onTouchMove: handleTouchSelection,
onTouchStart: handleTouchSelection,
onTouchEnd: handleTouchEnd,
onMouseUp: handleMouseUp,
onMouseMove: handleMouseMove,
ownerState: {
disabled
},
className: classes.squareMask
}), !isSelectedTimeDisabled && /*#__PURE__*/(0, _jsxRuntime.jsxs)(React.Fragment, {
children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(ClockPin, {
className: classes.pin
}), value != null && /*#__PURE__*/(0, _jsxRuntime.jsx)(_ClockPointer.ClockPointer, {
type: type,
viewValue: viewValue,
isInner: isPointerInner,
hasSelected: hasSelected
})]
}), /*#__PURE__*/(0, _jsxRuntime.jsx)(ClockWrapper, {
"aria-activedescendant": selectedId,
"aria-label": translations.clockLabelText(type, value, utils, value == null ? null : utils.format(value, 'fullTime')),
ref: listboxRef,
role: "listbox",
onKeyDown: handleKeyDown,
tabIndex: 0,
className: classes.wrapper,
children: children
})]
}), ampm && ampmInClock && /*#__PURE__*/(0, _jsxRuntime.jsxs)(React.Fragment, {
children: [/*#__PURE__*/(0, _jsxRuntime.jsx)(ClockAmButton, {
onClick: readOnly ? undefined : () => handleMeridiemChange('am'),
disabled: disabled || meridiemMode === null,
ownerState: ownerState,
className: classes.amButton,
title: (0, _dateUtils.formatMeridiem)(utils, 'am'),
children: /*#__PURE__*/(0, _jsxRuntime.jsx)(ClockMeridiemText, {
variant: "caption",
className: classes.meridiemText,
children: (0, _dateUtils.formatMeridiem)(utils, 'am')
})
}), /*#__PURE__*/(0, _jsxRuntime.jsx)(ClockPmButton, {
disabled: disabled || meridiemMode === null,
onClick: readOnly ? undefined : () => handleMeridiemChange('pm'),
ownerState: ownerState,
className: classes.pmButton,
title: (0, _dateUtils.formatMeridiem)(utils, 'pm'),
children: /*#__PURE__*/(0, _jsxRuntime.jsx)(ClockMeridiemText, {
variant: "caption",
className: classes.meridiemText,
children: (0, _dateUtils.formatMeridiem)(utils, 'pm')
})
})]
})]
});
}
;