UNPKG

@mui/x-date-pickers

Version:

The community edition of the Date and Time Picker components (MUI X).

417 lines (412 loc) 17.4 kB
"use strict"; 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;