UNPKG

react-day-picker

Version:

Customizable Date Picker for React

1,085 lines (1,015 loc) 99.8 kB
'use strict'; Object.defineProperty(exports, '__esModule', { value: true }); var React = require('react'); var enUS = require('date-fns/locale/en-US'); var format = require('date-fns/format'); var endOfMonth = require('date-fns/endOfMonth'); var startOfDay = require('date-fns/startOfDay'); var startOfMonth = require('date-fns/startOfMonth'); var isSameYear = require('date-fns/isSameYear'); var setMonth = require('date-fns/setMonth'); var setYear = require('date-fns/setYear'); var startOfYear = require('date-fns/startOfYear'); var addMonths = require('date-fns/addMonths'); var isBefore = require('date-fns/isBefore'); var isSameMonth = require('date-fns/isSameMonth'); var differenceInCalendarMonths = require('date-fns/differenceInCalendarMonths'); var addDays = require('date-fns/addDays'); var startOfWeek = require('date-fns/startOfWeek'); var getUnixTime = require('date-fns/getUnixTime'); var isSameDay = require('date-fns/isSameDay'); var addWeeks = require('date-fns/addWeeks'); var addYears = require('date-fns/addYears'); var endOfWeek = require('date-fns/endOfWeek'); var differenceInCalendarDays = require('date-fns/differenceInCalendarDays'); var isAfter = require('date-fns/isAfter'); var isDate = require('date-fns/isDate'); var getWeeksInMonth = require('date-fns/getWeeksInMonth'); var getWeek = require('date-fns/getWeek'); var parse = require('date-fns/parse'); function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; } var React__default = /*#__PURE__*/_interopDefaultLegacy(React); var enUS__default = /*#__PURE__*/_interopDefaultLegacy(enUS); var format__default = /*#__PURE__*/_interopDefaultLegacy(format); var endOfMonth__default = /*#__PURE__*/_interopDefaultLegacy(endOfMonth); var startOfDay__default = /*#__PURE__*/_interopDefaultLegacy(startOfDay); var startOfMonth__default = /*#__PURE__*/_interopDefaultLegacy(startOfMonth); var isSameYear__default = /*#__PURE__*/_interopDefaultLegacy(isSameYear); var setMonth__default = /*#__PURE__*/_interopDefaultLegacy(setMonth); var setYear__default = /*#__PURE__*/_interopDefaultLegacy(setYear); var startOfYear__default = /*#__PURE__*/_interopDefaultLegacy(startOfYear); var addMonths__default = /*#__PURE__*/_interopDefaultLegacy(addMonths); var isBefore__default = /*#__PURE__*/_interopDefaultLegacy(isBefore); var isSameMonth__default = /*#__PURE__*/_interopDefaultLegacy(isSameMonth); var differenceInCalendarMonths__default = /*#__PURE__*/_interopDefaultLegacy(differenceInCalendarMonths); var addDays__default = /*#__PURE__*/_interopDefaultLegacy(addDays); var startOfWeek__default = /*#__PURE__*/_interopDefaultLegacy(startOfWeek); var getUnixTime__default = /*#__PURE__*/_interopDefaultLegacy(getUnixTime); var isSameDay__default = /*#__PURE__*/_interopDefaultLegacy(isSameDay); var addWeeks__default = /*#__PURE__*/_interopDefaultLegacy(addWeeks); var addYears__default = /*#__PURE__*/_interopDefaultLegacy(addYears); var endOfWeek__default = /*#__PURE__*/_interopDefaultLegacy(endOfWeek); var differenceInCalendarDays__default = /*#__PURE__*/_interopDefaultLegacy(differenceInCalendarDays); var isAfter__default = /*#__PURE__*/_interopDefaultLegacy(isAfter); var isDate__default = /*#__PURE__*/_interopDefaultLegacy(isDate); var getWeeksInMonth__default = /*#__PURE__*/_interopDefaultLegacy(getWeeksInMonth); var getWeek__default = /*#__PURE__*/_interopDefaultLegacy(getWeek); var parse__default = /*#__PURE__*/_interopDefaultLegacy(parse); /*! ***************************************************************************** Copyright (c) Microsoft Corporation. Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted. THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. ***************************************************************************** */ var __assign = function() { __assign = Object.assign || function __assign(t) { for (var s, i = 1, n = arguments.length; i < n; i++) { s = arguments[i]; for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p]; } return t; }; return __assign.apply(this, arguments); }; function __rest(s, e) { var t = {}; for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0) t[p] = s[p]; if (s != null && typeof Object.getOwnPropertySymbols === "function") for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) { if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i])) t[p[i]] = s[p[i]]; } return t; } function __spreadArray(to, from, pack) { if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) { if (ar || !(i in from)) { if (!ar) ar = Array.prototype.slice.call(from, 0, i); ar[i] = from[i]; } } return to.concat(ar || Array.prototype.slice.call(from)); } function canUseDOM() { return !!(typeof window !== "undefined" && window.document && window.document.createElement); } /** * React currently throws a warning when using useLayoutEffect on the server. To * get around it, we can conditionally useEffect on the server (no-op) and * useLayoutEffect in the browser. We occasionally need useLayoutEffect to * ensure we don't get a render flash for certain operations, but we may also * need affected components to render on the server. One example is when setting * a component's descendants to retrieve their index values. * * Important to note that using this hook as an escape hatch will break the * eslint dependency warnings unless you rename the import to `useLayoutEffect`. * Use sparingly only when the effect won't effect the rendered HTML to avoid * any server/client mismatch. * * If a useLayoutEffect is needed and the result would create a mismatch, it's * likely that the component in question shouldn't be rendered on the server at * all, so a better approach would be to lazily render those in a parent * component after client-side hydration. * * https://gist.github.com/gaearon/e7d97cdf38a2907924ea12e4ebdf3c85 * https://github.com/reduxjs/react-redux/blob/master/src/utils/useIsomorphicLayoutEffect.js * * @param effect * @param deps */ var useIsomorphicLayoutEffect = /*#__PURE__*/canUseDOM() ? React.useLayoutEffect : React.useEffect; /* * Welcome to @reach/auto-id! * Let's see if we can make sense of why this hook exists and its * implementation. * * Some background: * 1. Accessibility APIs rely heavily on element IDs * 2. Requiring developers to put IDs on every element in Reach UI is both * cumbersome and error-prone * 3. With a component model, we can generate IDs for them! * * Solution 1: Generate random IDs. * * This works great as long as you don't server render your app. When React (in * the client) tries to reuse the markup from the server, the IDs won't match * and React will then recreate the entire DOM tree. * * Solution 2: Increment an integer * * This sounds great. Since we're rendering the exact same tree on the server * and client, we can increment a counter and get a deterministic result between * client and server. Also, JS integers can go up to nine-quadrillion. I'm * pretty sure the tab will be closed before an app never needs * 10 quadrillion IDs! * * Problem solved, right? * * Ah, but there's a catch! React's concurrent rendering makes this approach * non-deterministic. While the client and server will end up with the same * elements in the end, depending on suspense boundaries (and possibly some user * input during the initial render) the incrementing integers won't always match * up. * * Solution 3: Don't use IDs at all on the server; patch after first render. * * What we've done here is solution 2 with some tricks. With this approach, the * ID returned is an empty string on the first render. This way the server and * client have the same markup no matter how wild the concurrent rendering may * have gotten. * * After the render, we patch up the components with an incremented ID. This * causes a double render on any components with `useId`. Shouldn't be a problem * since the components using this hook should be small, and we're only updating * the ID attribute on the DOM, nothing big is happening. * * It doesn't have to be an incremented number, though--we could do generate * random strings instead, but incrementing a number is probably the cheapest * thing we can do. * * Additionally, we only do this patchup on the very first client render ever. * Any calls to `useId` that happen dynamically in the client will be * populated immediately with a value. So, we only get the double render after * server hydration and never again, SO BACK OFF ALRIGHT? */ var serverHandoffComplete = false; var id = 0; var genId = function genId() { return ++id; }; /** * useId * * Autogenerate IDs to facilitate WAI-ARIA and server rendering. * * Note: The returned ID will initially be `null` and will update after a * component mounts. Users may need to supply their own ID if they need * consistent values for SSR. * * @see Docs https://reach.tech/auto-id */ function useId(idFromProps) { /* * If this instance isn't part of the initial render, we don't have to do the * double render/patch-up dance. We can just generate the ID and return it. */ var initialId = idFromProps || (serverHandoffComplete ? genId() : null); var _React$useState = React.useState(initialId), id = _React$useState[0], setId = _React$useState[1]; useIsomorphicLayoutEffect(function () { if (id === null) { /* * Patch the ID after render. We do this in `useLayoutEffect` to avoid any * rendering flicker, though it'll make the first render slower (unlikely * to matter, but you're welcome to measure your app and let us know if * it's a problem). */ setId(genId()); } // eslint-disable-next-line react-hooks/exhaustive-deps }, []); React.useEffect(function () { if (serverHandoffComplete === false) { /* * Flag all future uses of `useId` to skip the update dance. This is in * `useEffect` because it goes after `useLayoutEffect`, ensuring we don't * accidentally bail out of the patch-up dance prematurely. */ serverHandoffComplete = true; } }, []); return id != null ? String(id) : undefined; } /** * The name of the default CSS classes. */ var defaultClassNames = { root: 'rdp', multiple_months: 'rdp-multiple_months', with_weeknumber: 'rdp-with_weeknumber', vhidden: 'rdp-vhidden', button_reset: 'rdp-button_reset', button: 'rdp-button', caption: 'rdp-caption', caption_start: 'rdp-caption_start', caption_end: 'rdp-caption_end', caption_between: 'rdp-caption_between', caption_label: 'rdp-caption_label', caption_dropdowns: 'rdp-caption_dropdowns', dropdown: 'rdp-dropdown', dropdown_month: 'rdp-dropdown_month', dropdown_year: 'rdp-dropdown_year', dropdown_icon: 'rdp-dropdown_icon', months: 'rdp-months', month: 'rdp-month', table: 'rdp-table', tbody: 'rdp-tbody', tfoot: 'rdp-tfoot', head: 'rdp-head', head_row: 'rdp-head_row', head_cell: 'rdp-head_cell', nav: 'rdp-nav', nav_button: 'rdp-nav_button', nav_button_previous: 'rdp-nav_button_previous', nav_button_next: 'rdp-nav_button_next', nav_icon: 'rdp-nav_icon', row: 'rdp-row', weeknumber: 'rdp-weeknumber', cell: 'rdp-cell', day: 'rdp-day', day_today: 'rdp-day_today', day_outside: 'rdp-day_outside', day_selected: 'rdp-day_selected', day_disabled: 'rdp-day_disabled', day_hidden: 'rdp-day_hidden', day_range_start: 'rdp-day_range_start', day_range_end: 'rdp-day_range_end', day_range_middle: 'rdp-day_range_middle' }; /** * The default formatter for the caption. */ function formatCaption(month, options) { return format__default["default"](month, 'LLLL y', options); } /** * The default formatter for the Day button. */ function formatDay(day, options) { return format__default["default"](day, 'd', options); } /** * The default formatter for the Month caption. */ function formatMonthCaption(month, options) { return format__default["default"](month, 'LLLL', options); } /** * The default formatter for the week number. */ function formatWeekNumber(weekNumber) { return "".concat(weekNumber); } /** * The default formatter for the name of the weekday. */ function formatWeekdayName(weekday, options) { return format__default["default"](weekday, 'cccccc', options); } /** * The default formatter for the Year caption. */ function formatYearCaption(year, options) { return format__default["default"](year, 'yyyy', options); } var formatters = /*#__PURE__*/Object.freeze({ __proto__: null, formatCaption: formatCaption, formatDay: formatDay, formatMonthCaption: formatMonthCaption, formatWeekNumber: formatWeekNumber, formatWeekdayName: formatWeekdayName, formatYearCaption: formatYearCaption }); /** * The default ARIA label for the day button. */ var labelDay = function (day, activeModifiers, options) { return format__default["default"](day, 'do MMMM (EEEE)', options); }; /** * The default ARIA label for the WeekNumber element. */ var labelMonthDropdown = function () { return 'Month: '; }; /** * The default ARIA label for next month button in navigation */ var labelNext = function () { return 'Go to next month'; }; /** * The default ARIA label for previous month button in navigation */ var labelPrevious = function () { return 'Go to previous month'; }; /** * The default ARIA label for the Weekday element. */ var labelWeekday = function (day, options) { return format__default["default"](day, 'cccc', options); }; /** * The default ARIA label for the WeekNumber element. */ var labelWeekNumber = function (n) { return "Week n. ".concat(n); }; /** * The default ARIA label for the WeekNumber element. */ var labelYearDropdown = function () { return 'Year: '; }; var labels = /*#__PURE__*/Object.freeze({ __proto__: null, labelDay: labelDay, labelMonthDropdown: labelMonthDropdown, labelNext: labelNext, labelPrevious: labelPrevious, labelWeekday: labelWeekday, labelWeekNumber: labelWeekNumber, labelYearDropdown: labelYearDropdown }); /** * Returns the default values to use in the DayPickerContext, in case they are * not passed down with the DayPicker initial props. */ function getDefaultContextValue() { var captionLayout = 'buttons'; var classNames = defaultClassNames; var locale = enUS__default["default"]; var modifiersClassNames = {}; var modifiers = {}; var numberOfMonths = 1; var styles = {}; var today = new Date(); return { captionLayout: captionLayout, classNames: classNames, formatters: formatters, labels: labels, locale: locale, modifiersClassNames: modifiersClassNames, modifiers: modifiers, numberOfMonths: numberOfMonths, styles: styles, today: today, mode: 'default' }; } /** Return the `fromDate` and `toDate` prop values values parsing the DayPicker props. */ function parseFromToProps(props) { var fromYear = props.fromYear, toYear = props.toYear, fromMonth = props.fromMonth, toMonth = props.toMonth; var fromDate = props.fromDate, toDate = props.toDate; if (fromMonth) { fromDate = startOfMonth__default["default"](fromMonth); } else if (fromYear) { fromDate = new Date(fromYear, 0, 1); } if (toMonth) { toDate = endOfMonth__default["default"](toMonth); } else if (toYear) { toDate = new Date(toYear, 11, 31); } return { fromDate: fromDate ? startOfDay__default["default"](fromDate) : undefined, toDate: toDate ? startOfDay__default["default"](toDate) : undefined }; } /** * The DayPicker Context shares the props passed to DayPicker within internal * and custom components. It is used to set the default values and perform * one-time calculations required to render the days. * * Developers may access this context from the [[useDayPicker]] hook when * using custom components. */ var DayPickerContext = React.createContext(undefined); /** * The provider for the [[DayPickerContext]], assigning the defaults from the * initial DayPicker props. */ function DayPickerProvider(props) { var _a, _b, _c, _d; var initialProps = props.initialProps; var defaults = getDefaultContextValue(); var _e = parseFromToProps(initialProps), fromDate = _e.fromDate, toDate = _e.toDate; var captionLayout = (_a = initialProps.captionLayout) !== null && _a !== void 0 ? _a : defaults.captionLayout; if (captionLayout !== 'buttons' && (!fromDate || !toDate)) { captionLayout = 'buttons'; } var value = { captionLayout: captionLayout, className: initialProps.className, classNames: __assign(__assign({}, defaults.classNames), initialProps.classNames), components: __assign(__assign({}, defaults.components), initialProps.components), defaultMonth: initialProps.defaultMonth, dir: initialProps.dir, disabled: initialProps.disabled, disableNavigation: initialProps.disableNavigation, fixedWeeks: initialProps.fixedWeeks, footer: initialProps.footer, formatters: __assign(__assign({}, defaults.formatters), initialProps.formatters), fromDate: fromDate, hidden: initialProps.hidden, hideHead: initialProps.hideHead, initialFocus: initialProps.initialFocus, labels: __assign(__assign({}, defaults.labels), initialProps.labels), locale: (_b = initialProps.locale) !== null && _b !== void 0 ? _b : defaults.locale, mode: initialProps.mode || 'default', modifiers: __assign(__assign({}, defaults.modifiers), initialProps.modifiers), modifiersClassNames: __assign(__assign({}, defaults.modifiersClassNames), initialProps.modifiersClassNames), modifiersStyles: initialProps.modifiersStyles, month: initialProps.month, numberOfMonths: (_c = initialProps.numberOfMonths) !== null && _c !== void 0 ? _c : defaults.numberOfMonths, onDayBlur: initialProps.onDayBlur, onDayClick: initialProps.onDayClick, onDayFocus: initialProps.onDayFocus, onDayKeyDown: initialProps.onDayKeyDown, onDayKeyPress: initialProps.onDayKeyPress, onDayKeyUp: initialProps.onDayKeyUp, onDayMouseEnter: initialProps.onDayMouseEnter, onDayMouseLeave: initialProps.onDayMouseLeave, onDayTouchCancel: initialProps.onDayTouchCancel, onDayTouchEnd: initialProps.onDayTouchEnd, onDayTouchMove: initialProps.onDayTouchMove, onDayTouchStart: initialProps.onDayTouchStart, onMonthChange: initialProps.onMonthChange, onNextClick: initialProps.onNextClick, onPrevClick: initialProps.onPrevClick, onWeekNumberClick: initialProps.onWeekNumberClick, pagedNavigation: initialProps.pagedNavigation, reverseMonths: initialProps.reverseMonths, selected: initialProps.selected, showOutsideDays: initialProps.showOutsideDays, showWeekNumber: initialProps.showWeekNumber, style: initialProps.style, styles: __assign(__assign({}, defaults.styles), initialProps.styles), toDate: toDate, today: (_d = initialProps.today) !== null && _d !== void 0 ? _d : defaults.today, weekStartsOn: initialProps.weekStartsOn }; return (React__default["default"].createElement(DayPickerContext.Provider, { value: value }, props.children)); } /** * Hook to access the [[DayPickerContext]]. * * To use this hook make sure to wrap the components with a one * [[DayPickerProvider]]. */ function useDayPicker() { var context = React.useContext(DayPickerContext); if (!context) { throw new Error("useDayPicker must be used within a DayPickerProvider."); } return context; } /** Render the caption for the displayed month. This component is used when `captionLayout="buttons"`. */ function CaptionLabel(props) { var _a = useDayPicker(), locale = _a.locale, classNames = _a.classNames, styles = _a.styles, formatCaption = _a.formatters.formatCaption; return (React__default["default"].createElement("h2", { className: classNames.caption_label, style: styles.caption_label, "aria-live": "polite", "aria-atomic": "true", id: props.id }, formatCaption(props.displayMonth, { locale: locale }))); } /** * Render the icon in the styled drop-down. */ function IconDropdown(props) { return (React__default["default"].createElement("svg", __assign({ width: "8px", height: "8px", viewBox: "0 0 120 120", "data-testid": "iconDropdown" }, props), React__default["default"].createElement("path", { d: "M4.22182541,48.2218254 C8.44222828,44.0014225 15.2388494,43.9273804 19.5496459,47.9996989 L19.7781746,48.2218254 L60,88.443 L100.221825,48.2218254 C104.442228,44.0014225 111.238849,43.9273804 115.549646,47.9996989 L115.778175,48.2218254 C119.998577,52.4422283 120.07262,59.2388494 116.000301,63.5496459 L115.778175,63.7781746 L67.7781746,111.778175 C63.5577717,115.998577 56.7611506,116.07262 52.4503541,112.000301 L52.2218254,111.778175 L4.22182541,63.7781746 C-0.0739418023,59.4824074 -0.0739418023,52.5175926 4.22182541,48.2218254 Z", fill: "currentColor", fillRule: "nonzero" }))); } /** * Render a styled select component – displaying a caption and a custom * drop-down icon. */ function Dropdown(props) { var _a, _b; var onChange = props.onChange, value = props.value, children = props.children, caption = props.caption, className = props.className, style = props.style; var dayPicker = useDayPicker(); var IconDropdownComponent = (_b = (_a = dayPicker.components) === null || _a === void 0 ? void 0 : _a.IconDropdown) !== null && _b !== void 0 ? _b : IconDropdown; return (React__default["default"].createElement("div", { className: className, style: style }, React__default["default"].createElement("span", { className: dayPicker.classNames.vhidden }, props['aria-label']), React__default["default"].createElement("select", { "aria-label": props['aria-label'], className: dayPicker.classNames.dropdown, style: dayPicker.styles.dropdown, value: value, onChange: onChange }, children), React__default["default"].createElement("div", { className: dayPicker.classNames.caption_label, style: dayPicker.styles.caption_label, "aria-hidden": "true" }, caption, React__default["default"].createElement(IconDropdownComponent, { className: dayPicker.classNames.dropdown_icon, style: dayPicker.styles.dropdown_icon })))); } /** Render the dropdown to navigate between months. */ function MonthsDropdown(props) { var _a; var _b = useDayPicker(), fromDate = _b.fromDate, toDate = _b.toDate, styles = _b.styles, locale = _b.locale, formatMonthCaption = _b.formatters.formatMonthCaption, classNames = _b.classNames, components = _b.components, labelMonthDropdown = _b.labels.labelMonthDropdown; // Dropdown should appear only when both from/toDate is set if (!fromDate) return React__default["default"].createElement(React__default["default"].Fragment, null); if (!toDate) return React__default["default"].createElement(React__default["default"].Fragment, null); var dropdownMonths = []; if (isSameYear__default["default"](fromDate, toDate)) { // only display the months included in the range var date = startOfMonth__default["default"](fromDate); for (var month = fromDate.getMonth(); month <= toDate.getMonth(); month++) { dropdownMonths.push(setMonth__default["default"](date, month)); } } else { // display all the 12 months var date = startOfMonth__default["default"](new Date()); // Any date should be OK, as we just need the year for (var month = 0; month <= 11; month++) { dropdownMonths.push(setMonth__default["default"](date, month)); } } var handleChange = function (e) { var selectedMonth = Number(e.target.value); var newMonth = setMonth__default["default"](startOfMonth__default["default"](props.displayMonth), selectedMonth); props.onChange(newMonth); }; var DropdownComponent = (_a = components === null || components === void 0 ? void 0 : components.Dropdown) !== null && _a !== void 0 ? _a : Dropdown; return (React__default["default"].createElement(DropdownComponent, { "aria-label": labelMonthDropdown(), className: classNames.dropdown_month, style: styles.dropdown_month, onChange: handleChange, value: props.displayMonth.getMonth(), caption: formatMonthCaption(props.displayMonth, { locale: locale }) }, dropdownMonths.map(function (m) { return (React__default["default"].createElement("option", { key: m.getMonth(), value: m.getMonth() }, formatMonthCaption(m, { locale: locale }))); }))); } /** * Render a dropdown to change the year. Take in account the `nav.fromDate` and * `toDate` from context. */ function YearsDropdown(props) { var _a; var displayMonth = props.displayMonth; var _b = useDayPicker(), fromDate = _b.fromDate, toDate = _b.toDate, locale = _b.locale, styles = _b.styles, classNames = _b.classNames, components = _b.components, formatYearCaption = _b.formatters.formatYearCaption, labelYearDropdown = _b.labels.labelYearDropdown; var years = []; // Dropdown should appear only when both from/toDate is set if (!fromDate) return React__default["default"].createElement(React__default["default"].Fragment, null); if (!toDate) return React__default["default"].createElement(React__default["default"].Fragment, null); var fromYear = fromDate.getFullYear(); var toYear = toDate.getFullYear(); for (var year = fromYear; year <= toYear; year++) { years.push(setYear__default["default"](startOfYear__default["default"](new Date()), year)); } var handleChange = function (e) { var newMonth = setYear__default["default"](startOfMonth__default["default"](displayMonth), Number(e.target.value)); props.onChange(newMonth); }; var DropdownComponent = (_a = components === null || components === void 0 ? void 0 : components.Dropdown) !== null && _a !== void 0 ? _a : Dropdown; return (React__default["default"].createElement(DropdownComponent, { "aria-label": labelYearDropdown(), className: classNames.dropdown_year, style: styles.dropdown_year, onChange: handleChange, value: displayMonth.getFullYear(), caption: formatYearCaption(displayMonth, { locale: locale }) }, years.map(function (year) { return (React__default["default"].createElement("option", { key: year.getFullYear(), value: year.getFullYear() }, formatYearCaption(year, { locale: locale }))); }))); } /** * Helper hook for using controlled/uncontrolled values from a component props. * * When the value is not controlled, pass `undefined` as `controlledValue` and * use the returned setter to update it. * * When the value is controlled, pass the controlled value as second * argument, which will be always returned as `value`. */ function useControlledValue(defaultValue, controlledValue) { var _a = React.useState(defaultValue), uncontrolledValue = _a[0], setValue = _a[1]; var value = controlledValue === undefined ? uncontrolledValue : controlledValue; return [value, setValue]; } /** Return the initial month according to the given options. */ function getInitialMonth(context) { var month = context.month, defaultMonth = context.defaultMonth, today = context.today; var initialMonth = month || defaultMonth || today || new Date(); var toDate = context.toDate, fromDate = context.fromDate, _a = context.numberOfMonths, numberOfMonths = _a === void 0 ? 1 : _a; // Fix the initialMonth if is after the to-date if (toDate && differenceInCalendarMonths__default["default"](toDate, initialMonth) < 0) { var offset = -1 * (numberOfMonths - 1); initialMonth = addMonths__default["default"](toDate, offset); } // Fix the initialMonth if is before the from-date if (fromDate && differenceInCalendarMonths__default["default"](initialMonth, fromDate) < 0) { initialMonth = fromDate; } return startOfMonth__default["default"](initialMonth); } /** Controls the navigation state. */ function useNavigationState() { var context = useDayPicker(); var initialMonth = getInitialMonth(context); var _a = useControlledValue(initialMonth, context.month), month = _a[0], setMonth = _a[1]; var goToMonth = function (date) { if (context.disableNavigation) return; setMonth(startOfMonth__default["default"](date)); }; return [month, goToMonth]; } /** * Return the months to display in the component according to the number of * months and the from/to date. */ function getDisplayMonths(month, _a) { var reverseMonths = _a.reverseMonths, numberOfMonths = _a.numberOfMonths; var start = startOfMonth__default["default"](month); var end = startOfMonth__default["default"](addMonths__default["default"](start, numberOfMonths)); var monthsDiff = differenceInCalendarMonths__default["default"](end, start); var months = []; for (var i = 0; i < monthsDiff; i++) { var nextMonth = addMonths__default["default"](start, i); months.push(nextMonth); } if (reverseMonths) months = months.reverse(); return months; } /** * Returns the next month the user can navigate to according to the given * options. * * Please note that the next month is not always the next calendar month: * * - if after the `toDate` range, is undefined; * - if the navigation is paged, is the number of months displayed ahead. * */ function getNextMonth(startingMonth, options) { if (options.disableNavigation) { return undefined; } var toDate = options.toDate, pagedNavigation = options.pagedNavigation, _a = options.numberOfMonths, numberOfMonths = _a === void 0 ? 1 : _a; var offset = pagedNavigation ? numberOfMonths : 1; var month = startOfMonth__default["default"](startingMonth); if (!toDate) { return addMonths__default["default"](month, offset); } var monthsDiff = differenceInCalendarMonths__default["default"](toDate, startingMonth); if (monthsDiff < numberOfMonths) { return undefined; } // Jump forward as the number of months when paged navigation return addMonths__default["default"](month, offset); } /** * Returns the next previous the user can navigate to, according to the given * options. * * Please note that the previous month is not always the previous calendar * month: * * - if before the `fromDate` date, is `undefined`; * - if the navigation is paged, is the number of months displayed before. * */ function getPreviousMonth(startingMonth, options) { if (options.disableNavigation) { return undefined; } var fromDate = options.fromDate, pagedNavigation = options.pagedNavigation, _a = options.numberOfMonths, numberOfMonths = _a === void 0 ? 1 : _a; var offset = pagedNavigation ? numberOfMonths : 1; var month = startOfMonth__default["default"](startingMonth); if (!fromDate) { return addMonths__default["default"](month, -offset); } var monthsDiff = differenceInCalendarMonths__default["default"](month, fromDate); if (monthsDiff <= 0) { return undefined; } // Jump back as the number of months when paged navigation return addMonths__default["default"](month, -offset); } /** * The Navigation context shares details about the months being navigated in DayPicker. * * Access this context from the [[useNavigation]] hook. */ var NavigationContext = React.createContext(undefined); /** Provides the values for the [[NavigationContext]]. */ function NavigationProvider(props) { var dayPicker = useDayPicker(); var _a = useNavigationState(), currentMonth = _a[0], goToMonth = _a[1]; var displayMonths = getDisplayMonths(currentMonth, dayPicker); var nextMonth = getNextMonth(currentMonth, dayPicker); var previousMonth = getPreviousMonth(currentMonth, dayPicker); var isDateDisplayed = function (date) { return displayMonths.some(function (displayMonth) { return isSameMonth__default["default"](date, displayMonth); }); }; var goToDate = function (date, refDate) { if (isDateDisplayed(date)) { return; } if (refDate && isBefore__default["default"](date, refDate)) { goToMonth(addMonths__default["default"](date, 1 + dayPicker.numberOfMonths * -1)); } else { goToMonth(date); } }; var value = { currentMonth: currentMonth, displayMonths: displayMonths, goToMonth: goToMonth, goToDate: goToDate, previousMonth: previousMonth, nextMonth: nextMonth, isDateDisplayed: isDateDisplayed }; return (React__default["default"].createElement(NavigationContext.Provider, { value: value }, props.children)); } /** Hook to access the [[NavigationContext]]. */ function useNavigation() { var context = React.useContext(NavigationContext); if (!context) { throw new Error('useNavigation must be used within a NavigationProvider'); } return context; } /** * Render a caption with the dropdowns to navigate between months and years. */ function CaptionDropdowns(props) { var _a; var _b = useDayPicker(), classNames = _b.classNames, styles = _b.styles, onMonthChange = _b.onMonthChange, components = _b.components; var goToMonth = useNavigation().goToMonth; var handleMonthChange = function (newMonth) { goToMonth(newMonth); onMonthChange === null || onMonthChange === void 0 ? void 0 : onMonthChange(newMonth); }; var CaptionLabelComponent = (_a = components === null || components === void 0 ? void 0 : components.CaptionLabel) !== null && _a !== void 0 ? _a : CaptionLabel; var captionLabel = (React__default["default"].createElement(CaptionLabelComponent, { id: props.id, displayMonth: props.displayMonth })); return (React__default["default"].createElement("div", { className: classNames.caption_dropdowns, style: styles.caption_dropdowns }, React__default["default"].createElement("div", { className: classNames.vhidden }, captionLabel), React__default["default"].createElement(MonthsDropdown, { onChange: handleMonthChange, displayMonth: props.displayMonth }), React__default["default"].createElement(YearsDropdown, { onChange: handleMonthChange, displayMonth: props.displayMonth }))); } /** * Render the "previous month" button in the navigation. */ function IconLeft(props) { return (React__default["default"].createElement("svg", __assign({ width: "16px", height: "16px", viewBox: "0 0 120 120" }, props), React__default["default"].createElement("path", { d: "M69.490332,3.34314575 C72.6145263,0.218951416 77.6798462,0.218951416 80.8040405,3.34314575 C83.8617626,6.40086786 83.9268205,11.3179931 80.9992143,14.4548388 L80.8040405,14.6568542 L35.461,60 L80.8040405,105.343146 C83.8617626,108.400868 83.9268205,113.317993 80.9992143,116.454839 L80.8040405,116.656854 C77.7463184,119.714576 72.8291931,119.779634 69.6923475,116.852028 L69.490332,116.656854 L18.490332,65.6568542 C15.4326099,62.5991321 15.367552,57.6820069 18.2951583,54.5451612 L18.490332,54.3431458 L69.490332,3.34314575 Z", fill: "currentColor", fillRule: "nonzero" }))); } /** * Render the "next month" button in the navigation. */ function IconRight(props) { return (React__default["default"].createElement("svg", __assign({ width: "16px", height: "16px", viewBox: "0 0 120 120" }, props), React__default["default"].createElement("path", { d: "M49.8040405,3.34314575 C46.6798462,0.218951416 41.6145263,0.218951416 38.490332,3.34314575 C35.4326099,6.40086786 35.367552,11.3179931 38.2951583,14.4548388 L38.490332,14.6568542 L83.8333725,60 L38.490332,105.343146 C35.4326099,108.400868 35.367552,113.317993 38.2951583,116.454839 L38.490332,116.656854 C41.5480541,119.714576 46.4651794,119.779634 49.602025,116.852028 L49.8040405,116.656854 L100.804041,65.6568542 C103.861763,62.5991321 103.926821,57.6820069 100.999214,54.5451612 L100.804041,54.3431458 L49.8040405,3.34314575 Z", fill: "currentColor" }))); } /** * Render a button HTML element applying the reset class name. */ var Button = React.forwardRef(function (props, ref) { var _a = useDayPicker(), classNames = _a.classNames, styles = _a.styles; var classNamesArr = [classNames.button_reset, classNames.button]; if (props.className) { classNamesArr.push(props.className); } var className = classNamesArr.join(' '); var style = __assign(__assign({}, styles.button_reset), styles.button); if (props.style) { Object.assign(style, props.style); } return (React__default["default"].createElement("button", __assign({}, props, { ref: ref, type: "button", className: className, style: style }))); }); /** A component rendering the navigation buttons or the drop-downs. */ function Navigation(props) { var _a, _b; var _c = useDayPicker(), dir = _c.dir, locale = _c.locale, classNames = _c.classNames, styles = _c.styles, _d = _c.labels, labelPrevious = _d.labelPrevious, labelNext = _d.labelNext, components = _c.components; if (!props.nextMonth && !props.previousMonth) { return React__default["default"].createElement(React__default["default"].Fragment, null); } var previousLabel = labelPrevious(props.previousMonth, { locale: locale }); var previousClassName = [ classNames.nav_button, classNames.nav_button_previous ].join(' '); var nextLabel = labelNext(props.nextMonth, { locale: locale }); var nextClassName = [ classNames.nav_button, classNames.nav_button_next ].join(' '); var IconRightComponent = (_a = components === null || components === void 0 ? void 0 : components.IconRight) !== null && _a !== void 0 ? _a : IconRight; var IconLeftComponent = (_b = components === null || components === void 0 ? void 0 : components.IconLeft) !== null && _b !== void 0 ? _b : IconLeft; return (React__default["default"].createElement("div", { className: classNames.nav, style: styles.nav }, !props.hidePrevious && (React__default["default"].createElement(Button, { "aria-label": previousLabel, className: previousClassName, style: styles.nav_button_previous, "aria-disabled": !props.previousMonth, onClick: props.onPreviousClick }, dir === 'rtl' ? (React__default["default"].createElement(IconRightComponent, { className: classNames.nav_icon, style: styles.nav_icon })) : (React__default["default"].createElement(IconLeftComponent, { className: classNames.nav_icon, style: styles.nav_icon })))), !props.hideNext && (React__default["default"].createElement(Button, { "aria-label": nextLabel, className: nextClassName, style: styles.nav_button_next, "aria-disabled": !props.nextMonth, onClick: props.onNextClick }, dir === 'rtl' ? (React__default["default"].createElement(IconLeftComponent, { className: classNames.nav_icon, style: styles.nav_icon })) : (React__default["default"].createElement(IconRightComponent, { className: classNames.nav_icon, style: styles.nav_icon })))))); } /** * Render a caption with a button-based navigation. */ function CaptionNavigation(props) { var _a; var _b; var _c = useDayPicker(), numberOfMonths = _c.numberOfMonths, onMonthChange = _c.onMonthChange, dir = _c.dir, components = _c.components; var _d = useNavigation(), previousMonth = _d.previousMonth, nextMonth = _d.nextMonth, goToMonth = _d.goToMonth, displayMonths = _d.displayMonths; var displayIndex = displayMonths.findIndex(function (month) { return isSameMonth__default["default"](props.displayMonth, month); }); var isFirst = displayIndex === 0; var isLast = displayIndex === displayMonths.length - 1; if (dir === 'rtl') { _a = [isFirst, isLast], isLast = _a[0], isFirst = _a[1]; } var hideNext = numberOfMonths > 1 && (isFirst || !isLast); var hidePrevious = numberOfMonths > 1 && (isLast || !isFirst); var handlePreviousClick = function () { if (!previousMonth) return; goToMonth(previousMonth); onMonthChange === null || onMonthChange === void 0 ? void 0 : onMonthChange(previousMonth); }; var handleNextClick = function () { if (!nextMonth) return; goToMonth(nextMonth); onMonthChange === null || onMonthChange === void 0 ? void 0 : onMonthChange(nextMonth); }; var CaptionLabelComponent = (_b = components === null || components === void 0 ? void 0 : components.CaptionLabel) !== null && _b !== void 0 ? _b : CaptionLabel; var captionLabel = (React__default["default"].createElement(CaptionLabelComponent, { id: props.id, displayMonth: props.displayMonth })); return (React__default["default"].createElement(React__default["default"].Fragment, null, captionLabel, React__default["default"].createElement(Navigation, { displayMonth: props.displayMonth, hideNext: hideNext, hidePrevious: hidePrevious, nextMonth: nextMonth, previousMonth: previousMonth, onPreviousClick: handlePreviousClick, onNextClick: handleNextClick }))); } /** * Render the caption of a month. The caption has a different layout when * setting the [[DayPickerProps.captionLayout]] prop. */ function Caption(props) { var _a; var _b = useDayPicker(), classNames = _b.classNames, disableNavigation = _b.disableNavigation, styles = _b.styles, captionLayout = _b.captionLayout, components = _b.components; var CaptionLabelComponent = (_a = components === null || components === void 0 ? void 0 : components.CaptionLabel) !== null && _a !== void 0 ? _a : CaptionLabel; var caption; if (disableNavigation) { caption = (React__default["default"].createElement(CaptionLabelComponent, { id: props.id, displayMonth: props.displayMonth })); } else if (captionLayout === 'dropdown') { caption = (React__default["default"].createElement(CaptionDropdowns, { displayMonth: props.displayMonth, id: props.id })); } else { caption = (React__default["default"].createElement(CaptionNavigation, { displayMonth: props.displayMonth, id: props.id })); } return (React__default["default"].createElement("div", { className: classNames.caption, style: styles.caption }, caption)); } /** Render the Footer component (empty as default).*/ function Footer() { var _a = useDayPicker(), footer = _a.footer, styles = _a.styles, tfoot = _a.classNames.tfoot; if (!footer) return React__default["default"].createElement(React__default["default"].Fragment, null); return (React__default["default"].createElement("tfoot", { className: tfoot, style: styles.tfoot }, React__default["default"].createElement("tr", null, React__default["default"].createElement("td", { colSpan: 8 }, footer)))); } /** * Generate a series of 7 days, starting from the week, to use for formatting * the weekday names (Monday, Tuesday, etc.). */ function getWeekdays(locale, /** The index of the first day of the week (0 - Sunday) */ weekStartsOn) { var start = startOfWeek__default["default"](new Date(), { locale: locale, weekStartsOn: weekStartsOn }); var days = []; for (var i = 0; i < 7; i++) { var day = addDays__default["default"](start, i); days.push(day); } return days; } /** * Render the Head component - i.e. the table head with the weekday names. */ function Head() { var _a = useDayPicker(), classNames = _a.classNames, styles = _a.styles, showWeekNumber = _a.showWeekNumber, locale = _a.locale, weekStartsOn = _a.weekStartsOn, formatWeekdayName = _a.formatters.formatWeekdayName, labelWeekday = _a.labels.labelWeekday; var weekdays = getWeekdays(locale, weekStartsOn); return (React__default["default"].createElement("thead", { style: styles.head, className: classNames.head }, React__default["default"].createElement("tr", { style: styles.head_row, className: classNames.head_row }, showWeekNumber && (React__default["default"].createElement("th", { scope: "col", style: styles.head_cell, className: classNames.head_cell })), weekdays.map(function (weekday, i) { return (React__default["default"].createElement("th", { key: i, scope: "col", className: classNames.head_cell, style: styles.head_cell }, React__default["default"].createElement("span", { "aria-hidden": true }, formatWeekdayName(weekday, { locale: locale })), React__default["default"].createElement("span", { className: classNames.vhidden }, labelWeekday(weekday, { locale: locale })))); })))); } /** * Render the content of the day cell. */ function DayContent(props) { var _a = useDayPicker(), locale = _a.locale, classNames = _a.classNames, styles = _a.styles, labelDay = _a.labels.labelDay, formatDay = _a.formatters.formatDay; return (React__default["default"].createElement(React__default["default"].Fragment, null, React__default["default"].createElement("span", { "aria-hidden": "true" }, formatDay(props.date, { locale: locale })), React__default["default"].createElement("span", { className: classNames.vhidden, style: styles.vhidden }, labelDay(props.date, props.activeModifiers, { locale: locale })))); } /** Returns true when the props are of type [[DayPickerMultiple]]. */ function isDayPickerMultiple(props) { return props.mode === 'multiple'; } /** * The SelectMultiple context shares details about the selected days when in * multiple selection mode. * * Access this context from the [[useSelectMultiple]] hook. */ var SelectMultipleContext = React.createContext(undefined); /** Provides the values for the [[SelectMultipleContext]]. */ function SelectMultipleProvider(props) { if (!isDayPickerMultiple(props.initialProps)) { var emptyContextValue = { selected: undefined, modifiers: { disabled: [] } }; return (React__default["default"].createElement(SelectMultipleContext.Provider, { value: emptyContextValue }, props.children)); } return (React__default["default"].createElement(SelectMultipleProviderInternal, { initialProps: props.initialProps, children: props.children })); } function SelectMultipleProviderInternal(_a) { var initialProps = _a.initialProps, children = _a.children; var selected = initialProps.selected, min = initialProps.min, max = initialProps.max; var onDayClick = function (day, activeModifiers, e) { var _a, _b; (_a = initialProps.onDayClick) === null || _a === void 0 ? void 0 : _a.call(initialProps, day, activeModifiers, e); var isMinSelected = Boolean(activeModifiers.selected && min && (selected === null || selected === void 0 ? void 0 : selected.length) === min); if (isMinSelected) { return; } var isMaxSelected = Boolean(!activeModifiers.selected && max && (selected === null || selected === void 0 ? void 0 : selected.length) === max); if (isMaxSelected) { return; } var selectedDays = selected ? __spreadArray([], selected, true) : []; if (activeModifiers.selected) { var index = selectedDays.findIndex(function (selectedDay) { return isSameDay__default["default"](day, selectedDay); }); selectedDays.splice(index, 1); } else { selectedDays.push(day); } (_b = initialProps.onSelect) === null || _b === void 0 ? void 0 : _b.call(initialProps, selectedDays, day, activeModifiers, e); }; var modifiers = { disabled: [] }; if (selected) { modifiers.disabled.push(function (day) { var isMaxSelected = max && selected.length > max - 1; var isSelected = selected.some(function (selectedDay) { return isSameDay__default["default"](selectedDay, day); }); return Boolean(isMaxSelected && !isSelected); }); } var contextValue = { selected: selected, onDayClick: onDayClick, modifiers: modifiers }; return (React__default["default"].createElement(SelectMultipleContext.Provider, { value: contextValue }, children)); } /** Hook to access the [[SelectMultipleContext]]. */ function useSelectMultiple() { var context = React.useContext(SelectMultipleContext); if (!context) { throw new Error('useSelectMultiple must be used within a SelectMultipleProvider'); } return context; } /** Returns true when the props are of type [[DayPickerRange]]. */ function isDayPickerRange(props) { return props.mode === 'range'; } /** * Add a day to an existing range. * * The