@mui/x-date-pickers
Version:
The community edition of the Date and Time Picker components (MUI X).
417 lines (412 loc) • 17.4 kB
JavaScript
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault").default;
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard").default;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.useFieldV7TextField = void 0;
var React = _interopRequireWildcard(require("react"));
var _useForkRef = _interopRequireDefault(require("@mui/utils/useForkRef"));
var _useEventCallback = _interopRequireDefault(require("@mui/utils/useEventCallback"));
var _useEnhancedEffect = _interopRequireDefault(require("@mui/utils/useEnhancedEffect"));
var _useId = _interopRequireDefault(require("@mui/utils/useId"));
var _useField = require("./useField.utils");
var _utils = require("../../utils/utils");
var _usePickersTranslations = require("../../../hooks/usePickersTranslations");
var _useUtils = require("../useUtils");
const useFieldV7TextField = params => {
const {
internalProps: {
disabled,
readOnly = false
},
forwardedProps: {
sectionListRef: inSectionListRef,
onBlur,
onClick,
onFocus,
onInput,
onPaste,
focused: focusedProp,
autoFocus = false
},
fieldValueManager,
applyCharacterEditing,
resetCharacterQuery,
setSelectedSections,
parsedSelectedSections,
state,
clearActiveSection,
clearValue,
updateSectionValue,
updateValueFromValueStr,
sectionOrder,
areAllSectionsEmpty,
sectionsValueBoundaries
} = params;
const sectionListRef = React.useRef(null);
const handleSectionListRef = (0, _useForkRef.default)(inSectionListRef, sectionListRef);
const translations = (0, _usePickersTranslations.usePickersTranslations)();
const utils = (0, _useUtils.useUtils)();
const id = (0, _useId.default)();
const [focused, setFocused] = React.useState(false);
const interactions = React.useMemo(() => ({
syncSelectionToDOM: () => {
if (!sectionListRef.current) {
return;
}
const selection = document.getSelection();
if (!selection) {
return;
}
if (parsedSelectedSections == null) {
// If the selection contains an element inside the field, we reset it.
if (selection.rangeCount > 0 && sectionListRef.current.getRoot().contains(selection.getRangeAt(0).startContainer)) {
selection.removeAllRanges();
}
if (focused) {
sectionListRef.current.getRoot().blur();
}
return;
}
// On multi input range pickers we want to update selection range only for the active input
if (!sectionListRef.current.getRoot().contains((0, _utils.getActiveElement)(document))) {
return;
}
const range = new window.Range();
let target;
if (parsedSelectedSections === 'all') {
target = sectionListRef.current.getRoot();
} else {
const section = state.sections[parsedSelectedSections];
if (section.type === 'empty') {
target = sectionListRef.current.getSectionContainer(parsedSelectedSections);
} else {
target = sectionListRef.current.getSectionContent(parsedSelectedSections);
}
}
range.selectNodeContents(target);
target.focus();
selection.removeAllRanges();
selection.addRange(range);
},
getActiveSectionIndexFromDOM: () => {
const activeElement = (0, _utils.getActiveElement)(document);
if (!activeElement || !sectionListRef.current || !sectionListRef.current.getRoot().contains(activeElement)) {
return null;
}
return sectionListRef.current.getSectionIndexFromDOMElement(activeElement);
},
focusField: (newSelectedSections = 0) => {
if (!sectionListRef.current ||
// if the field is already focused, we don't need to focus it again
interactions.getActiveSectionIndexFromDOM() != null) {
return;
}
const newParsedSelectedSections = (0, _useField.parseSelectedSections)(newSelectedSections, state.sections);
setFocused(true);
sectionListRef.current.getSectionContent(newParsedSelectedSections).focus();
},
setSelectedSections: newSelectedSections => {
if (!sectionListRef.current) {
return;
}
const newParsedSelectedSections = (0, _useField.parseSelectedSections)(newSelectedSections, state.sections);
const newActiveSectionIndex = newParsedSelectedSections === 'all' ? 0 : newParsedSelectedSections;
setFocused(newActiveSectionIndex !== null);
setSelectedSections(newSelectedSections);
},
isFieldFocused: () => {
const activeElement = (0, _utils.getActiveElement)(document);
return !!sectionListRef.current && sectionListRef.current.getRoot().contains(activeElement);
}
}), [parsedSelectedSections, setSelectedSections, state.sections, focused]);
/**
* If a section content has been updated with a value we don't want to keep,
* Then we need to imperatively revert it (we can't let React do it because the value did not change in his internal representation).
*/
const revertDOMSectionChange = (0, _useEventCallback.default)(sectionIndex => {
if (!sectionListRef.current) {
return;
}
const section = state.sections[sectionIndex];
sectionListRef.current.getSectionContent(sectionIndex).innerHTML = section.value || section.placeholder;
interactions.syncSelectionToDOM();
});
const handleContainerClick = (0, _useEventCallback.default)((event, ...args) => {
// The click event on the clear button would propagate to the input, trigger this handler and result in a wrong section selection.
// We avoid this by checking if the call of `handleContainerClick` is actually intended, or a side effect.
if (event.isDefaultPrevented() || !sectionListRef.current) {
return;
}
setFocused(true);
onClick?.(event, ...args);
if (parsedSelectedSections === 'all') {
setTimeout(() => {
const cursorPosition = document.getSelection().getRangeAt(0).startOffset;
if (cursorPosition === 0) {
setSelectedSections(sectionOrder.startIndex);
return;
}
let sectionIndex = 0;
let cursorOnStartOfSection = 0;
while (cursorOnStartOfSection < cursorPosition && sectionIndex < state.sections.length) {
const section = state.sections[sectionIndex];
sectionIndex += 1;
cursorOnStartOfSection += `${section.startSeparator}${section.value || section.placeholder}${section.endSeparator}`.length;
}
setSelectedSections(sectionIndex - 1);
});
} else if (!focused) {
setFocused(true);
setSelectedSections(sectionOrder.startIndex);
} else {
const hasClickedOnASection = sectionListRef.current.getRoot().contains(event.target);
if (!hasClickedOnASection) {
setSelectedSections(sectionOrder.startIndex);
}
}
});
const handleContainerInput = (0, _useEventCallback.default)(event => {
onInput?.(event);
if (!sectionListRef.current || parsedSelectedSections !== 'all') {
return;
}
const target = event.target;
const keyPressed = target.textContent ?? '';
sectionListRef.current.getRoot().innerHTML = state.sections.map(section => `${section.startSeparator}${section.value || section.placeholder}${section.endSeparator}`).join('');
interactions.syncSelectionToDOM();
if (keyPressed.length === 0 || keyPressed.charCodeAt(0) === 10) {
resetCharacterQuery();
clearValue();
setSelectedSections('all');
} else if (keyPressed.length > 1) {
updateValueFromValueStr(keyPressed);
} else {
if (parsedSelectedSections === 'all') {
setSelectedSections(0);
}
applyCharacterEditing({
keyPressed,
sectionIndex: 0
});
}
});
const handleContainerPaste = (0, _useEventCallback.default)(event => {
onPaste?.(event);
if (readOnly || parsedSelectedSections !== 'all') {
event.preventDefault();
return;
}
const pastedValue = event.clipboardData.getData('text');
event.preventDefault();
resetCharacterQuery();
updateValueFromValueStr(pastedValue);
});
const handleContainerFocus = (0, _useEventCallback.default)((...args) => {
onFocus?.(...args);
if (focused || !sectionListRef.current) {
return;
}
setFocused(true);
const isFocusInsideASection = sectionListRef.current.getSectionIndexFromDOMElement((0, _utils.getActiveElement)(document)) != null;
if (!isFocusInsideASection) {
setSelectedSections(sectionOrder.startIndex);
}
});
const handleContainerBlur = (0, _useEventCallback.default)((...args) => {
onBlur?.(...args);
setTimeout(() => {
if (!sectionListRef.current) {
return;
}
const activeElement = (0, _utils.getActiveElement)(document);
const shouldBlur = !sectionListRef.current.getRoot().contains(activeElement);
if (shouldBlur) {
setFocused(false);
setSelectedSections(null);
}
});
});
const getInputContainerClickHandler = (0, _useEventCallback.default)(sectionIndex => event => {
// The click event on the clear button would propagate to the input, trigger this handler and result in a wrong section selection.
// We avoid this by checking if the call to this function is actually intended, or a side effect.
if (event.isDefaultPrevented()) {
return;
}
setSelectedSections(sectionIndex);
});
const handleInputContentMouseUp = (0, _useEventCallback.default)(event => {
// Without this, the browser will remove the selected when clicking inside an already-selected section.
event.preventDefault();
});
const getInputContentFocusHandler = (0, _useEventCallback.default)(sectionIndex => () => {
setSelectedSections(sectionIndex);
});
const handleInputContentPaste = (0, _useEventCallback.default)(event => {
// prevent default to avoid the input `onInput` handler being called
event.preventDefault();
if (readOnly || disabled || typeof parsedSelectedSections !== 'number') {
return;
}
const activeSection = state.sections[parsedSelectedSections];
const pastedValue = event.clipboardData.getData('text');
const lettersOnly = /^[a-zA-Z]+$/.test(pastedValue);
const digitsOnly = /^[0-9]+$/.test(pastedValue);
const digitsAndLetterOnly = /^(([a-zA-Z]+)|)([0-9]+)(([a-zA-Z]+)|)$/.test(pastedValue);
const isValidPastedValue = activeSection.contentType === 'letter' && lettersOnly || activeSection.contentType === 'digit' && digitsOnly || activeSection.contentType === 'digit-with-letter' && digitsAndLetterOnly;
if (isValidPastedValue) {
resetCharacterQuery();
updateSectionValue({
activeSection,
newSectionValue: pastedValue,
shouldGoToNextSection: true
});
}
// If the pasted value corresponds to a single section, but not the expected type, we skip the modification
else if (!lettersOnly && !digitsOnly) {
resetCharacterQuery();
updateValueFromValueStr(pastedValue);
}
});
const handleInputContentDragOver = (0, _useEventCallback.default)(event => {
event.preventDefault();
event.dataTransfer.dropEffect = 'none';
});
const handleInputContentInput = (0, _useEventCallback.default)(event => {
if (!sectionListRef.current) {
return;
}
const target = event.target;
const keyPressed = target.textContent ?? '';
const sectionIndex = sectionListRef.current.getSectionIndexFromDOMElement(target);
const section = state.sections[sectionIndex];
if (readOnly || !sectionListRef.current) {
revertDOMSectionChange(sectionIndex);
return;
}
if (keyPressed.length === 0) {
if (section.value === '') {
revertDOMSectionChange(sectionIndex);
return;
}
const inputType = event.nativeEvent.inputType;
if (inputType === 'insertParagraph' || inputType === 'insertLineBreak') {
revertDOMSectionChange(sectionIndex);
return;
}
resetCharacterQuery();
clearActiveSection();
return;
}
applyCharacterEditing({
keyPressed,
sectionIndex
});
// The DOM value needs to remain the one React is expecting.
revertDOMSectionChange(sectionIndex);
});
(0, _useEnhancedEffect.default)(() => {
if (!focused || !sectionListRef.current) {
return;
}
if (parsedSelectedSections === 'all') {
sectionListRef.current.getRoot().focus();
} else if (typeof parsedSelectedSections === 'number') {
const domElement = sectionListRef.current.getSectionContent(parsedSelectedSections);
if (domElement) {
domElement.focus();
}
}
}, [parsedSelectedSections, focused]);
const sectionBoundaries = React.useMemo(() => {
return state.sections.reduce((acc, next) => {
acc[next.type] = sectionsValueBoundaries[next.type]({
currentDate: null,
contentType: next.contentType,
format: next.format
});
return acc;
}, {});
}, [sectionsValueBoundaries, state.sections]);
const isContainerEditable = parsedSelectedSections === 'all';
const elements = React.useMemo(() => {
return state.sections.map((section, index) => {
const isEditable = !isContainerEditable && !disabled && !readOnly;
return {
container: {
'data-sectionindex': index,
onClick: getInputContainerClickHandler(index)
},
content: {
tabIndex: isContainerEditable || index > 0 ? -1 : 0,
contentEditable: !isContainerEditable && !disabled && !readOnly,
role: 'spinbutton',
id: `${id}-${section.type}`,
'aria-labelledby': `${id}-${section.type}`,
'aria-readonly': readOnly,
'aria-valuenow': (0, _useField.getSectionValueNow)(section, utils),
'aria-valuemin': sectionBoundaries[section.type].minimum,
'aria-valuemax': sectionBoundaries[section.type].maximum,
'aria-valuetext': section.value ? (0, _useField.getSectionValueText)(section, utils) : translations.empty,
'aria-label': translations[section.type],
'aria-disabled': disabled,
spellCheck: isEditable ? false : undefined,
autoCapitalize: isEditable ? 'off' : undefined,
autoCorrect: isEditable ? 'off' : undefined,
[parseInt(React.version, 10) >= 17 ? 'enterKeyHint' : 'enterkeyhint']: isEditable ? 'next' : undefined,
children: section.value || section.placeholder,
onInput: handleInputContentInput,
onPaste: handleInputContentPaste,
onFocus: getInputContentFocusHandler(index),
onDragOver: handleInputContentDragOver,
onMouseUp: handleInputContentMouseUp,
inputMode: section.contentType === 'letter' ? 'text' : 'numeric'
},
before: {
children: section.startSeparator
},
after: {
children: section.endSeparator
}
};
});
}, [state.sections, getInputContentFocusHandler, handleInputContentPaste, handleInputContentDragOver, handleInputContentInput, getInputContainerClickHandler, handleInputContentMouseUp, disabled, readOnly, isContainerEditable, translations, utils, sectionBoundaries, id]);
const handleValueStrChange = (0, _useEventCallback.default)(event => {
updateValueFromValueStr(event.target.value);
});
const valueStr = React.useMemo(() => areAllSectionsEmpty ? '' : fieldValueManager.getV7HiddenInputValueFromSections(state.sections), [areAllSectionsEmpty, state.sections, fieldValueManager]);
React.useEffect(() => {
if (sectionListRef.current == null) {
throw new Error(['MUI X: The `sectionListRef` prop has not been initialized by `PickersSectionList`', 'You probably tried to pass a component to the `textField` slot that contains an `<input />` element instead of a `PickersSectionList`.', '', 'If you want to keep using an `<input />` HTML element for the editing, please remove the `enableAccessibleFieldDOMStructure` prop from your picker or field component:', '', '<DatePicker slots={{ textField: MyCustomTextField }} />', '', 'Learn more about the field accessible DOM structure on the MUI documentation: https://mui.com/x/react-date-pickers/fields/#fields-to-edit-a-single-element'].join('\n'));
}
if (autoFocus && sectionListRef.current) {
sectionListRef.current.getSectionContent(sectionOrder.startIndex).focus();
}
}, []); // eslint-disable-line react-hooks/exhaustive-deps
return {
interactions,
returnedValue: {
// Forwarded
autoFocus,
readOnly,
focused: focusedProp ?? focused,
sectionListRef: handleSectionListRef,
onBlur: handleContainerBlur,
onClick: handleContainerClick,
onFocus: handleContainerFocus,
onInput: handleContainerInput,
onPaste: handleContainerPaste,
// Additional
enableAccessibleFieldDOMStructure: true,
elements,
// TODO v7: Try to set to undefined when there is a section selected.
tabIndex: parsedSelectedSections === 0 ? -1 : 0,
contentEditable: isContainerEditable,
value: valueStr,
onChange: handleValueStrChange,
areAllSectionsEmpty
}
};
};
exports.useFieldV7TextField = useFieldV7TextField;
;