diff --git a/package-lock.json b/package-lock.json index 4bcfd1e4a2..3b65318f40 100644 --- a/package-lock.json +++ b/package-lock.json @@ -9112,9 +9112,9 @@ } }, "flatpickr": { - "version": "4.5.2", - "resolved": "http://npm.lundalogik.com:4873/flatpickr/-/flatpickr-4.5.2.tgz", - "integrity": "sha512-jDy4QYGpmiy7+Qk8QvKJ4spjDdxcx9cxMydmq1x427HkKWBw0qizLYeYM2F6tMcvvqGjU5VpJS55j4LnsaBblA==" + "version": "4.5.3-lime1", + "resolved": "http://npm.lundalogik.com:4873/flatpickr/-/flatpickr-4.5.3-lime1.tgz", + "integrity": "sha512-abzjWpxKrXBsZErIEXkEVlae48YJqqvfe1kpAodyrfqEtOMiPYRy0NV9Gf6joRrftu9WxdSLcifOADTH4enndQ==" }, "flatted": { "version": "2.0.0", diff --git a/package.json b/package.json index 6c2efb2c33..959dc70742 100644 --- a/package.json +++ b/package.json @@ -46,7 +46,7 @@ "dependencies": { "awesome-debounce-promise": "^2.1.0", "chart.js": "^2.7.3", - "flatpickr": "^4.5.2", + "flatpickr": "4.5.3-lime1", "jsx-dom": "^6.2.1", "lime-material-components-web": "0.43.1", "moment": "^2.24.0", diff --git a/postinstall.js b/postinstall.js index 9bb0f7698b..b2724cb97f 100644 --- a/postinstall.js +++ b/postinstall.js @@ -1,19 +1,5 @@ const fs = require('fs'); -// Copy pre-built version of flatpickr -// TODO: remove this when a new version of flatpickr has been released -fs.exists('src/dev-assets/hack/flatpickr.js', () => { - fs.copyFile( - 'src/dev-assets/hack/flatpickr.js', - 'node_modules/flatpickr/dist/flatpickr.js', - () => { - console.log( - 'copied src/dev-assets/hack/flatpickr.js to node_modules' - ); - } - ); -}); - // Remove type definitions for jsx-dom fs.exists('node_modules/jsx-dom/jsx-dom.d.ts', () => { fs.unlink('node_modules/jsx-dom/jsx-dom.d.ts', () => { diff --git a/src/components/date-picker/pickers/Picker.ts b/src/components/date-picker/pickers/Picker.ts index b950ffc0b4..975a4c7901 100644 --- a/src/components/date-picker/pickers/Picker.ts +++ b/src/components/date-picker/pickers/Picker.ts @@ -48,7 +48,6 @@ export abstract class Picker { defaultDate: value, locale: FlatpickrLanguages[this.language] || 'en', }; - config = { ...config, ...this.getConfig(this.nativePicker) }; this.flatpickr = flatpickr(element, config) as Instance; // tslint:disable-line:no-useless-cast } @@ -73,23 +72,39 @@ export abstract class Picker { } protected handleClose(selectedDates) { - const momentInputDate = moment( - this.flatpickr.input.value, - this.dateFormat - ); let pickerDate = selectedDates[0] ? new Date(selectedDates[0].toJSON()) : null; - const isSameInput = momentInputDate.isSame(moment(pickerDate)); - if (!isSameInput) { - if (momentInputDate.isValid()) { - pickerDate = momentInputDate.toDate(); - this.flatpickr.setDate(pickerDate); - } else { - this.flatpickr.clear(); - } + if (this.nativePicker) { + this.change.emit(pickerDate); + } else { + // With the changes in flatpickr between v4.5.2 and the state of + // the master branch when we created v4.5.3-lime1, this timeout + // had to be added to let `flatpickr.input.value` update before + // reading the value. /Ads + setTimeout(() => { + // We need to set the locale before parsing, in case the + // locale for this picker differs from the locale of the + // app as a whole. For some reason, the fact that we already + // set the locale in the Picker constructor doesn't affect + // the instance used here. /Ads + moment.locale(this.getMomentLang()); + const momentInputDate = moment( + this.flatpickr.input.value, + this.dateFormat + ); + const isSameInput = momentInputDate.isSame(moment(pickerDate)); + if (!isSameInput) { + if (momentInputDate.isValid()) { + pickerDate = momentInputDate.toDate(); + this.flatpickr.setDate(pickerDate); + } else { + this.flatpickr.clear(); + } + } + this.change.emit(pickerDate); + }, 0); } - this.change.emit(pickerDate); } private getMomentLang() { diff --git a/src/dev-assets/hack/flatpickr.js b/src/dev-assets/hack/flatpickr.js deleted file mode 100644 index 548f61f7b0..0000000000 --- a/src/dev-assets/hack/flatpickr.js +++ /dev/null @@ -1,2264 +0,0 @@ -/* flatpickr v4.5.2, @license MIT */ -(function (global, factory) { - typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() : - typeof define === 'function' && define.amd ? define(factory) : - (global.flatpickr = factory()); -}(this, (function () { 'use strict'; - - const pad = (number) => `0${number}`.slice(-2); - const int = (bool) => (bool === true ? 1 : 0); - function debounce(func, wait, immediate = false) { - let timeout; - return function () { - let context = this, args = arguments; - timeout !== null && clearTimeout(timeout); - timeout = window.setTimeout(function () { - timeout = null; - if (!immediate) - func.apply(context, args); - }, wait); - if (immediate && !timeout) - func.apply(context, args); - }; - } - const arrayify = (obj) => obj instanceof Array ? obj : [obj]; - - const do_nothing = () => undefined; - const monthToStr = (monthNumber, shorthand, locale) => locale.months[shorthand ? "shorthand" : "longhand"][monthNumber]; - const revFormat = { - D: do_nothing, - F: function (dateObj, monthName, locale) { - dateObj.setMonth(locale.months.longhand.indexOf(monthName)); - }, - G: (dateObj, hour) => { - dateObj.setHours(parseFloat(hour)); - }, - H: (dateObj, hour) => { - dateObj.setHours(parseFloat(hour)); - }, - J: (dateObj, day) => { - dateObj.setDate(parseFloat(day)); - }, - K: (dateObj, amPM, locale) => { - dateObj.setHours((dateObj.getHours() % 12) + - 12 * int(new RegExp(locale.amPM[1], "i").test(amPM))); - }, - M: function (dateObj, shortMonth, locale) { - dateObj.setMonth(locale.months.shorthand.indexOf(shortMonth)); - }, - S: (dateObj, seconds) => { - dateObj.setSeconds(parseFloat(seconds)); - }, - U: (_, unixSeconds) => new Date(parseFloat(unixSeconds) * 1000), - W: function (dateObj, weekNum) { - const weekNumber = parseInt(weekNum); - return new Date(dateObj.getFullYear(), 0, 2 + (weekNumber - 1) * 7, 0, 0, 0, 0); - }, - Y: (dateObj, year) => { - dateObj.setFullYear(parseFloat(year)); - }, - Z: (_, ISODate) => new Date(ISODate), - d: (dateObj, day) => { - dateObj.setDate(parseFloat(day)); - }, - h: (dateObj, hour) => { - dateObj.setHours(parseFloat(hour)); - }, - i: (dateObj, minutes) => { - dateObj.setMinutes(parseFloat(minutes)); - }, - j: (dateObj, day) => { - dateObj.setDate(parseFloat(day)); - }, - l: do_nothing, - m: (dateObj, month) => { - dateObj.setMonth(parseFloat(month) - 1); - }, - n: (dateObj, month) => { - dateObj.setMonth(parseFloat(month) - 1); - }, - s: (dateObj, seconds) => { - dateObj.setSeconds(parseFloat(seconds)); - }, - u: (_, unixMillSeconds) => new Date(parseFloat(unixMillSeconds)), - w: do_nothing, - y: (dateObj, year) => { - dateObj.setFullYear(2000 + parseFloat(year)); - }, - }; - const tokenRegex = { - D: "(\\w+)", - F: "(\\w+)", - G: "(\\d\\d|\\d)", - H: "(\\d\\d|\\d)", - J: "(\\d\\d|\\d)\\w+", - K: "", - M: "(\\w+)", - S: "(\\d\\d|\\d)", - U: "(.+)", - W: "(\\d\\d|\\d)", - Y: "(\\d{4})", - Z: "(.+)", - d: "(\\d\\d|\\d)", - h: "(\\d\\d|\\d)", - i: "(\\d\\d|\\d)", - j: "(\\d\\d|\\d)", - l: "(\\w+)", - m: "(\\d\\d|\\d)", - n: "(\\d\\d|\\d)", - s: "(\\d\\d|\\d)", - u: "(.+)", - w: "(\\d\\d|\\d)", - y: "(\\d{2})", - }; - const formats = { - Z: (date) => date.toISOString(), - D: function (date, locale, options) { - return locale.weekdays.shorthand[formats.w(date, locale, options)]; - }, - F: function (date, locale, options) { - return monthToStr(formats.n(date, locale, options) - 1, false, locale); - }, - G: function (date, locale, options) { - return pad(formats.h(date, locale, options)); - }, - H: (date) => pad(date.getHours()), - J: function (date, locale) { - return locale.ordinal !== undefined - ? date.getDate() + locale.ordinal(date.getDate()) - : date.getDate(); - }, - K: (date, locale) => locale.amPM[int(date.getHours() > 11)], - M: function (date, locale) { - return monthToStr(date.getMonth(), true, locale); - }, - S: (date) => pad(date.getSeconds()), - U: (date) => date.getTime() / 1000, - W: function (date, _, options) { - return options.getWeek(date); - }, - Y: (date) => date.getFullYear(), - d: (date) => pad(date.getDate()), - h: (date) => (date.getHours() % 12 ? date.getHours() % 12 : 12), - i: (date) => pad(date.getMinutes()), - j: (date) => date.getDate(), - l: function (date, locale) { - return locale.weekdays.longhand[date.getDay()]; - }, - m: (date) => pad(date.getMonth() + 1), - n: (date) => date.getMonth() + 1, - s: (date) => date.getSeconds(), - u: (date) => date.getTime(), - w: (date) => date.getDay(), - y: (date) => String(date.getFullYear()).substring(2), - }; - - const english = { - weekdays: { - shorthand: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"], - longhand: [ - "Sunday", - "Monday", - "Tuesday", - "Wednesday", - "Thursday", - "Friday", - "Saturday", - ], - }, - months: { - shorthand: [ - "Jan", - "Feb", - "Mar", - "Apr", - "May", - "Jun", - "Jul", - "Aug", - "Sep", - "Oct", - "Nov", - "Dec", - ], - longhand: [ - "January", - "February", - "March", - "April", - "May", - "June", - "July", - "August", - "September", - "October", - "November", - "December", - ], - }, - daysInMonth: [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31], - firstDayOfWeek: 0, - ordinal: (nth) => { - const s = nth % 100; - if (s > 3 && s < 21) - return "th"; - switch (s % 10) { - case 1: - return "st"; - case 2: - return "nd"; - case 3: - return "rd"; - default: - return "th"; - } - }, - rangeSeparator: " to ", - weekAbbreviation: "Wk", - scrollTitle: "Scroll to increment", - toggleTitle: "Click to toggle", - amPM: ["AM", "PM"], - yearAriaLabel: "Year", - }; - - const createDateFormatter = ({ config = defaults, l10n = english, }) => (dateObj, frmt, overrideLocale) => { - const locale = overrideLocale || l10n; - if (config.formatDate !== undefined) { - return config.formatDate(dateObj, frmt, locale); - } - return frmt - .split("") - .map((c, i, arr) => formats[c] && arr[i - 1] !== "\\" - ? formats[c](dateObj, locale, config) - : c !== "\\" - ? c - : "") - .join(""); - }; - const createDateParser = ({ config = defaults, l10n = english }) => (date, givenFormat, timeless, customLocale) => { - if (date !== 0 && !date) - return undefined; - const locale = customLocale || l10n; - let parsedDate; - const date_orig = date; - if (date instanceof Date) - parsedDate = new Date(date.getTime()); - else if (typeof date !== "string" && - date.toFixed !== undefined) - parsedDate = new Date(date); - else if (typeof date === "string") { - const format = givenFormat || (config || defaults).dateFormat; - const datestr = String(date).trim(); - if (datestr === "today") { - parsedDate = new Date(); - timeless = true; - } - else if (/Z$/.test(datestr) || - /GMT$/.test(datestr)) - parsedDate = new Date(date); - else if (config && config.parseDate) - parsedDate = config.parseDate(date, format); - else { - parsedDate = - !config || !config.noCalendar - ? new Date(new Date().getFullYear(), 0, 1, 0, 0, 0, 0) - : new Date(new Date().setHours(0, 0, 0, 0)); - let matched, ops = []; - for (let i = 0, matchIndex = 0, regexStr = ""; i < format.length; i++) { - const token = format[i]; - const isBackSlash = token === "\\"; - const escaped = format[i - 1] === "\\" || isBackSlash; - if (tokenRegex[token] && !escaped) { - regexStr += tokenRegex[token]; - const match = new RegExp(regexStr).exec(date); - if (match && (matched = true)) { - ops[token !== "Y" ? "push" : "unshift"]({ - fn: revFormat[token], - val: match[++matchIndex], - }); - } - } - else if (!isBackSlash) - regexStr += "."; - ops.forEach(({ fn, val }) => (parsedDate = fn(parsedDate, val, locale) || parsedDate)); - } - parsedDate = matched ? parsedDate : undefined; - } - } - if (!(parsedDate instanceof Date && !isNaN(parsedDate.getTime()))) { - config.errorHandler(new Error(`Invalid date provided: ${date_orig}`)); - return undefined; - } - if (timeless === true) - parsedDate.setHours(0, 0, 0, 0); - return parsedDate; - }; - function compareDates(date1, date2, timeless = true) { - if (timeless !== false) { - return (new Date(date1.getTime()).setHours(0, 0, 0, 0) - - new Date(date2.getTime()).setHours(0, 0, 0, 0)); - } - return date1.getTime() - date2.getTime(); - } - const getWeek = (givenDate) => { - const date = new Date(givenDate.getTime()); - date.setHours(0, 0, 0, 0); - date.setDate(date.getDate() + 3 - ((date.getDay() + 6) % 7)); - var week1 = new Date(date.getFullYear(), 0, 4); - return (1 + - Math.round(((date.getTime() - week1.getTime()) / 86400000 - - 3 + - ((week1.getDay() + 6) % 7)) / - 7)); - }; - const isBetween = (ts, ts1, ts2) => { - return ts > Math.min(ts1, ts2) && ts < Math.max(ts1, ts2); - }; - const duration = { - DAY: 86400000, - }; - - const HOOKS = [ - "onChange", - "onClose", - "onDayCreate", - "onDestroy", - "onKeyDown", - "onMonthChange", - "onOpen", - "onParseConfig", - "onReady", - "onValueUpdate", - "onYearChange", - "onPreCalendarPosition", - ]; - const defaults = { - _disable: [], - _enable: [], - allowInput: false, - altFormat: "F j, Y", - altInput: false, - altInputClass: "form-control input", - animate: typeof window === "object" && - window.navigator.userAgent.indexOf("MSIE") === -1, - ariaDateFormat: "F j, Y", - clickOpens: true, - closeOnSelect: true, - conjunction: ", ", - dateFormat: "Y-m-d", - defaultHour: 12, - defaultMinute: 0, - defaultSeconds: 0, - disable: [], - disableMobile: false, - enable: [], - enableSeconds: false, - enableTime: false, - errorHandler: (err) => typeof console !== "undefined" && console.warn(err), - getWeek, - hourIncrement: 1, - ignoredFocusElements: [], - inline: false, - locale: "default", - minuteIncrement: 5, - mode: "single", - nextArrow: "", - noCalendar: false, - now: new Date(), - onChange: [], - onClose: [], - onDayCreate: [], - onDestroy: [], - onKeyDown: [], - onMonthChange: [], - onOpen: [], - onParseConfig: [], - onReady: [], - onValueUpdate: [], - onYearChange: [], - onPreCalendarPosition: [], - plugins: [], - position: "auto", - positionElement: undefined, - prevArrow: "", - shorthandCurrentMonth: false, - showMonths: 1, - static: false, - time_24hr: false, - weekNumbers: false, - wrap: false, - }; - - function toggleClass(elem, className, bool) { - if (bool === true) - return elem.classList.add(className); - elem.classList.remove(className); - } - function createElement(tag, className, content) { - const e = window.document.createElement(tag); - className = className || ""; - content = content || ""; - e.className = className; - if (content !== undefined) - e.textContent = content; - return e; - } - function clearNode(node) { - while (node.firstChild) - node.removeChild(node.firstChild); - } - function findParent(node, condition) { - if (condition(node)) - return node; - else if (node.parentNode) - return findParent(node.parentNode, condition); - return undefined; - } - function createNumberInput(inputClassName, opts) { - const wrapper = createElement("div", "numInputWrapper"), numInput = createElement("input", "numInput " + inputClassName), arrowUp = createElement("span", "arrowUp"), arrowDown = createElement("span", "arrowDown"); - numInput.type = "text"; - numInput.pattern = "\\d*"; - if (opts !== undefined) - for (const key in opts) - numInput.setAttribute(key, opts[key]); - wrapper.appendChild(numInput); - wrapper.appendChild(arrowUp); - wrapper.appendChild(arrowDown); - return wrapper; - } - function getEventTarget(event) { - if (typeof event.composedPath === "function") { - const path = event.composedPath(); - return path[0]; - } - return event.target; - } - - if (typeof Object.assign !== "function") { - Object.assign = function (target, ...args) { - if (!target) { - throw TypeError("Cannot convert undefined or null to object"); - } - for (const source of args) { - if (source) { - Object.keys(source).forEach(key => (target[key] = source[key])); - } - } - return target; - }; - } - - const DEBOUNCED_CHANGE_MS = 300; - function FlatpickrInstance(element, instanceConfig) { - const self = { - config: Object.assign({}, flatpickr.defaultConfig), - l10n: english, - }; - self.parseDate = createDateParser({ config: self.config, l10n: self.l10n }); - self._handlers = []; - self._bind = bind; - self._setHoursFromDate = setHoursFromDate; - self._positionCalendar = positionCalendar; - self.changeMonth = changeMonth; - self.changeYear = changeYear; - self.clear = clear; - self.close = close; - self._createElement = createElement; - self.destroy = destroy; - self.isEnabled = isEnabled; - self.jumpToDate = jumpToDate; - self.open = open; - self.redraw = redraw; - self.set = set; - self.setDate = setDate; - self.toggle = toggle; - function setupHelperFunctions() { - self.utils = { - getDaysInMonth(month = self.currentMonth, yr = self.currentYear) { - if (month === 1 && ((yr % 4 === 0 && yr % 100 !== 0) || yr % 400 === 0)) - return 29; - return self.l10n.daysInMonth[month]; - }, - }; - } - function init() { - self.element = self.input = element; - self.isOpen = false; - parseConfig(); - setupLocale(); - setupInputs(); - setupDates(); - setupHelperFunctions(); - if (!self.isMobile) - build(); - bindEvents(); - if (self.selectedDates.length || self.config.noCalendar) { - if (self.config.enableTime) { - setHoursFromDate(self.config.noCalendar - ? self.latestSelectedDateObj || self.config.minDate - : undefined); - } - updateValue(false); - } - setCalendarWidth(); - self.showTimeInput = - self.selectedDates.length > 0 || self.config.noCalendar; - const isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent); - if (!self.isMobile && isSafari) { - positionCalendar(); - } - triggerEvent("onReady"); - } - function bindToInstance(fn) { - return fn.bind(self); - } - function setCalendarWidth() { - const config = self.config; - if (config.weekNumbers === false && config.showMonths === 1) - return; - else if (config.noCalendar !== true) { - window.requestAnimationFrame(function () { - if (self.calendarContainer !== undefined) { - self.calendarContainer.style.visibility = "hidden"; - self.calendarContainer.style.display = "block"; - } - if (self.daysContainer !== undefined) { - const daysWidth = (self.days.offsetWidth + 1) * config.showMonths; - self.daysContainer.style.width = daysWidth + "px"; - self.calendarContainer.style.width = - daysWidth + - (self.weekWrapper !== undefined - ? self.weekWrapper.offsetWidth - : 0) + - "px"; - self.calendarContainer.style.removeProperty("visibility"); - self.calendarContainer.style.removeProperty("display"); - } - }); - } - } - function updateTime(e) { - if (self.selectedDates.length === 0) - return; - if (e !== undefined && e.type !== "blur") { - timeWrapper(e); - } - const prevValue = self._input.value; - setHoursFromInputs(); - updateValue(); - if (self._input.value !== prevValue) { - self._debouncedChange(); - } - } - function ampm2military(hour, amPM) { - return (hour % 12) + 12 * int(amPM === self.l10n.amPM[1]); - } - function military2ampm(hour) { - switch (hour % 24) { - case 0: - case 12: - return 12; - default: - return hour % 12; - } - } - function setHoursFromInputs() { - if (self.hourElement === undefined || self.minuteElement === undefined) - return; - let hours = (parseInt(self.hourElement.value.slice(-2), 10) || 0) % 24, minutes = (parseInt(self.minuteElement.value, 10) || 0) % 60, seconds = self.secondElement !== undefined - ? (parseInt(self.secondElement.value, 10) || 0) % 60 - : 0; - if (self.amPM !== undefined) { - hours = ampm2military(hours, self.amPM.textContent); - } - const limitMinHours = self.config.minTime !== undefined || - (self.config.minDate && - self.minDateHasTime && - self.latestSelectedDateObj && - compareDates(self.latestSelectedDateObj, self.config.minDate, true) === - 0); - const limitMaxHours = self.config.maxTime !== undefined || - (self.config.maxDate && - self.maxDateHasTime && - self.latestSelectedDateObj && - compareDates(self.latestSelectedDateObj, self.config.maxDate, true) === - 0); - if (limitMaxHours) { - const maxTime = self.config.maxTime !== undefined - ? self.config.maxTime - : self.config.maxDate; - hours = Math.min(hours, maxTime.getHours()); - if (hours === maxTime.getHours()) - minutes = Math.min(minutes, maxTime.getMinutes()); - if (minutes === maxTime.getMinutes()) - seconds = Math.min(seconds, maxTime.getSeconds()); - } - if (limitMinHours) { - const minTime = self.config.minTime !== undefined - ? self.config.minTime - : self.config.minDate; - hours = Math.max(hours, minTime.getHours()); - if (hours === minTime.getHours()) - minutes = Math.max(minutes, minTime.getMinutes()); - if (minutes === minTime.getMinutes()) - seconds = Math.max(seconds, minTime.getSeconds()); - } - setHours(hours, minutes, seconds); - } - function setHoursFromDate(dateObj) { - const date = dateObj || self.latestSelectedDateObj; - if (date) - setHours(date.getHours(), date.getMinutes(), date.getSeconds()); - } - function setDefaultHours() { - let hours = self.config.defaultHour; - let minutes = self.config.defaultMinute; - let seconds = self.config.defaultSeconds; - if (self.config.minDate !== undefined) { - const min_hr = self.config.minDate.getHours(); - const min_minutes = self.config.minDate.getMinutes(); - hours = Math.max(hours, min_hr); - if (hours === min_hr) - minutes = Math.max(min_minutes, minutes); - if (hours === min_hr && minutes === min_minutes) - seconds = self.config.minDate.getSeconds(); - } - if (self.config.maxDate !== undefined) { - const max_hr = self.config.maxDate.getHours(); - const max_minutes = self.config.maxDate.getMinutes(); - hours = Math.min(hours, max_hr); - if (hours === max_hr) - minutes = Math.min(max_minutes, minutes); - if (hours === max_hr && minutes === max_minutes) - seconds = self.config.maxDate.getSeconds(); - } - setHours(hours, minutes, seconds); - } - function setHours(hours, minutes, seconds) { - if (self.latestSelectedDateObj !== undefined) { - self.latestSelectedDateObj.setHours(hours % 24, minutes, seconds || 0, 0); - } - if (!self.hourElement || !self.minuteElement || self.isMobile) - return; - self.hourElement.value = pad(!self.config.time_24hr - ? ((12 + hours) % 12) + 12 * int(hours % 12 === 0) - : hours); - self.minuteElement.value = pad(minutes); - if (self.amPM !== undefined) - self.amPM.textContent = self.l10n.amPM[int(hours >= 12)]; - if (self.secondElement !== undefined) - self.secondElement.value = pad(seconds); - } - function onYearInput(event) { - const year = parseInt(event.target.value) + (event.delta || 0); - if (year / 1000 > 1 || - (event.key === "Enter" && !/[^\d]/.test(year.toString()))) { - changeYear(year); - } - } - function bind(element, event, handler, options) { - if (event instanceof Array) - return event.forEach(ev => bind(element, ev, handler, options)); - if (element instanceof Array) - return element.forEach(el => bind(el, event, handler, options)); - element.addEventListener(event, handler, options); - self._handlers.push({ - element: element, - event, - handler, - options, - }); - } - function onClick(handler) { - return evt => { - evt.which === 1 && handler(evt); - }; - } - function triggerChange() { - triggerEvent("onChange"); - } - function bindEvents() { - if (self.config.wrap) { - ["open", "close", "toggle", "clear"].forEach(evt => { - Array.prototype.forEach.call(self.element.querySelectorAll(`[data-${evt}]`), (el) => bind(el, "click", self[evt])); - }); - } - if (self.isMobile) { - setupMobile(); - return; - } - const debouncedResize = debounce(onResize, 50); - self._debouncedChange = debounce(triggerChange, DEBOUNCED_CHANGE_MS); - if (self.daysContainer && !/iPhone|iPad|iPod/i.test(navigator.userAgent)) - bind(self.daysContainer, "mouseover", (e) => { - if (self.config.mode === "range") - onMouseOver(e.target); - }); - bind(window.document.body, "keydown", onKeyDown); - if (!self.config.static) - bind(self._input, "keydown", onKeyDown); - if (!self.config.inline && !self.config.static) - bind(window, "resize", debouncedResize); - if (window.ontouchstart !== undefined) - bind(window.document, "click", documentClick); - else - bind(window.document, "mousedown", onClick(documentClick)); - bind(window.document, "focus", documentClick, { capture: true }); - if (self.config.clickOpens === true) { - bind(self._input, "focus", self.open); - bind(self._input, "mousedown", onClick(self.open)); - } - if (self.daysContainer !== undefined) { - bind(self.monthNav, "mousedown", onClick(onMonthNavClick)); - bind(self.monthNav, ["keyup", "increment"], onYearInput); - bind(self.daysContainer, "mousedown", onClick(selectDate)); - } - if (self.timeContainer !== undefined && - self.minuteElement !== undefined && - self.hourElement !== undefined) { - const selText = (e) => e.target.select(); - bind(self.timeContainer, ["increment"], updateTime); - bind(self.timeContainer, "blur", updateTime, { capture: true }); - bind(self.timeContainer, "mousedown", onClick(timeIncrement)); - bind([self.hourElement, self.minuteElement], ["focus", "click"], selText); - if (self.secondElement !== undefined) - bind(self.secondElement, "focus", () => self.secondElement && self.secondElement.select()); - if (self.amPM !== undefined) { - bind(self.amPM, "mousedown", onClick(e => { - updateTime(e); - triggerChange(); - })); - } - } - } - function jumpToDate(jumpDate) { - const jumpTo = jumpDate !== undefined - ? self.parseDate(jumpDate) - : self.latestSelectedDateObj || - (self.config.minDate && self.config.minDate > self.now - ? self.config.minDate - : self.config.maxDate && self.config.maxDate < self.now - ? self.config.maxDate - : self.now); - try { - if (jumpTo !== undefined) { - self.currentYear = jumpTo.getFullYear(); - self.currentMonth = jumpTo.getMonth(); - } - } - catch (e) { - e.message = "Invalid date supplied: " + jumpTo; - self.config.errorHandler(e); - } - self.redraw(); - } - function timeIncrement(e) { - if (~e.target.className.indexOf("arrow")) - incrementNumInput(e, e.target.classList.contains("arrowUp") ? 1 : -1); - } - function incrementNumInput(e, delta, inputElem) { - const target = e && e.target; - const input = inputElem || - (target && target.parentNode && target.parentNode.firstChild); - const event = createEvent("increment"); - event.delta = delta; - input && input.dispatchEvent(event); - } - function build() { - const fragment = window.document.createDocumentFragment(); - self.calendarContainer = createElement("div", "flatpickr-calendar"); - self.calendarContainer.tabIndex = -1; - if (!self.config.noCalendar) { - fragment.appendChild(buildMonthNav()); - self.innerContainer = createElement("div", "flatpickr-innerContainer"); - if (self.config.weekNumbers) { - const { weekWrapper, weekNumbers } = buildWeeks(); - self.innerContainer.appendChild(weekWrapper); - self.weekNumbers = weekNumbers; - self.weekWrapper = weekWrapper; - } - self.rContainer = createElement("div", "flatpickr-rContainer"); - self.rContainer.appendChild(buildWeekdays()); - if (!self.daysContainer) { - self.daysContainer = createElement("div", "flatpickr-days"); - self.daysContainer.tabIndex = -1; - } - buildDays(); - self.rContainer.appendChild(self.daysContainer); - self.innerContainer.appendChild(self.rContainer); - fragment.appendChild(self.innerContainer); - } - if (self.config.enableTime) { - fragment.appendChild(buildTime()); - } - toggleClass(self.calendarContainer, "rangeMode", self.config.mode === "range"); - toggleClass(self.calendarContainer, "animate", self.config.animate === true); - toggleClass(self.calendarContainer, "multiMonth", self.config.showMonths > 1); - self.calendarContainer.appendChild(fragment); - const customAppend = self.config.appendTo !== undefined && - self.config.appendTo.nodeType !== undefined; - if (self.config.inline || self.config.static) { - self.calendarContainer.classList.add(self.config.inline ? "inline" : "static"); - if (self.config.inline) { - if (!customAppend && self.element.parentNode) - self.element.parentNode.insertBefore(self.calendarContainer, self._input.nextSibling); - else if (self.config.appendTo !== undefined) - self.config.appendTo.appendChild(self.calendarContainer); - } - if (self.config.static) { - const wrapper = createElement("div", "flatpickr-wrapper"); - if (self.element.parentNode) - self.element.parentNode.insertBefore(wrapper, self.element); - wrapper.appendChild(self.element); - if (self.altInput) - wrapper.appendChild(self.altInput); - wrapper.appendChild(self.calendarContainer); - } - } - if (!self.config.static && !self.config.inline) - (self.config.appendTo !== undefined - ? self.config.appendTo - : window.document.body).appendChild(self.calendarContainer); - } - function createDay(className, date, dayNumber, i) { - const dateIsEnabled = isEnabled(date, true), dayElement = createElement("span", "flatpickr-day " + className, date.getDate().toString()); - dayElement.dateObj = date; - dayElement.$i = i; - dayElement.setAttribute("aria-label", self.formatDate(date, self.config.ariaDateFormat)); - if (className.indexOf("hidden") === -1 && - compareDates(date, self.now) === 0) { - self.todayDateElem = dayElement; - dayElement.classList.add("today"); - dayElement.setAttribute("aria-current", "date"); - } - if (dateIsEnabled) { - dayElement.tabIndex = -1; - if (isDateSelected(date)) { - dayElement.classList.add("selected"); - self.selectedDateElem = dayElement; - if (self.config.mode === "range") { - toggleClass(dayElement, "startRange", self.selectedDates[0] && - compareDates(date, self.selectedDates[0], true) === 0); - toggleClass(dayElement, "endRange", self.selectedDates[1] && - compareDates(date, self.selectedDates[1], true) === 0); - if (className === "nextMonthDay") - dayElement.classList.add("inRange"); - } - } - } - else { - dayElement.classList.add("disabled"); - } - if (self.config.mode === "range") { - if (isDateInRange(date) && !isDateSelected(date)) - dayElement.classList.add("inRange"); - } - if (self.weekNumbers && - self.config.showMonths === 1 && - className !== "prevMonthDay" && - dayNumber % 7 === 1) { - self.weekNumbers.insertAdjacentHTML("beforeend", "" + self.config.getWeek(date) + ""); - } - triggerEvent("onDayCreate", dayElement); - return dayElement; - } - function focusOnDayElem(targetNode) { - targetNode.focus(); - if (self.config.mode === "range") - onMouseOver(targetNode); - } - function getFirstAvailableDay(delta) { - const startMonth = delta > 0 ? 0 : self.config.showMonths - 1; - const endMonth = delta > 0 ? self.config.showMonths : -1; - for (let m = startMonth; m != endMonth; m += delta) { - const month = self.daysContainer.children[m]; - const startIndex = delta > 0 ? 0 : month.children.length - 1; - const endIndex = delta > 0 ? month.children.length : -1; - for (let i = startIndex; i != endIndex; i += delta) { - const c = month.children[i]; - if (c.className.indexOf("hidden") === -1 && isEnabled(c.dateObj)) - return c; - } - } - return undefined; - } - function getNextAvailableDay(current, delta) { - const givenMonth = current.className.indexOf("Month") === -1 - ? current.dateObj.getMonth() - : self.currentMonth; - const endMonth = delta > 0 ? self.config.showMonths : -1; - const loopDelta = delta > 0 ? 1 : -1; - for (let m = givenMonth - self.currentMonth; m != endMonth; m += loopDelta) { - const month = self.daysContainer.children[m]; - const startIndex = givenMonth - self.currentMonth === m - ? current.$i + delta - : delta < 0 - ? month.children.length - 1 - : 0; - const numMonthDays = month.children.length; - for (let i = startIndex; i >= 0 && i < numMonthDays && i != (delta > 0 ? numMonthDays : -1); i += loopDelta) { - const c = month.children[i]; - if (c.className.indexOf("hidden") === -1 && - isEnabled(c.dateObj) && - Math.abs(current.$i - i) >= Math.abs(delta)) - return focusOnDayElem(c); - } - } - self.changeMonth(loopDelta); - focusOnDay(getFirstAvailableDay(loopDelta), 0); - return undefined; - } - function focusOnDay(current, offset) { - const dayFocused = isInView(document.activeElement || document.body); - const startElem = current !== undefined - ? current - : dayFocused - ? document.activeElement - : self.selectedDateElem !== undefined && isInView(self.selectedDateElem) - ? self.selectedDateElem - : self.todayDateElem !== undefined && isInView(self.todayDateElem) - ? self.todayDateElem - : getFirstAvailableDay(offset > 0 ? 1 : -1); - if (startElem === undefined) - return self._input.focus(); - if (!dayFocused) - return focusOnDayElem(startElem); - getNextAvailableDay(startElem, offset); - } - function buildMonthDays(year, month) { - const firstOfMonth = (new Date(year, month, 1).getDay() - self.l10n.firstDayOfWeek + 7) % 7; - const prevMonthDays = self.utils.getDaysInMonth((month - 1 + 12) % 12); - const daysInMonth = self.utils.getDaysInMonth(month), days = window.document.createDocumentFragment(), isMultiMonth = self.config.showMonths > 1, prevMonthDayClass = isMultiMonth ? "prevMonthDay hidden" : "prevMonthDay", nextMonthDayClass = isMultiMonth ? "nextMonthDay hidden" : "nextMonthDay"; - let dayNumber = prevMonthDays + 1 - firstOfMonth, dayIndex = 0; - for (; dayNumber <= prevMonthDays; dayNumber++, dayIndex++) { - days.appendChild(createDay(prevMonthDayClass, new Date(year, month - 1, dayNumber), dayNumber, dayIndex)); - } - for (dayNumber = 1; dayNumber <= daysInMonth; dayNumber++, dayIndex++) { - days.appendChild(createDay("", new Date(year, month, dayNumber), dayNumber, dayIndex)); - } - for (let dayNum = daysInMonth + 1; dayNum <= 42 - firstOfMonth && - (self.config.showMonths === 1 || dayIndex % 7 !== 0); dayNum++, dayIndex++) { - days.appendChild(createDay(nextMonthDayClass, new Date(year, month + 1, dayNum % daysInMonth), dayNum, dayIndex)); - } - const dayContainer = createElement("div", "dayContainer"); - dayContainer.appendChild(days); - return dayContainer; - } - function buildDays() { - if (self.daysContainer === undefined) { - return; - } - clearNode(self.daysContainer); - if (self.weekNumbers) - clearNode(self.weekNumbers); - const frag = document.createDocumentFragment(); - for (let i = 0; i < self.config.showMonths; i++) { - const d = new Date(self.currentYear, self.currentMonth, 1); - d.setMonth(self.currentMonth + i); - frag.appendChild(buildMonthDays(d.getFullYear(), d.getMonth())); - } - self.daysContainer.appendChild(frag); - self.days = self.daysContainer.firstChild; - if (self.config.mode === "range" && self.selectedDates.length === 1) { - onMouseOver(); - } - } - function buildMonth() { - const container = createElement("div", "flatpickr-month"); - const monthNavFragment = window.document.createDocumentFragment(); - const monthElement = createElement("span", "cur-month"); - const yearInput = createNumberInput("cur-year", { tabindex: "-1" }); - const yearElement = yearInput.getElementsByTagName("input")[0]; - yearElement.setAttribute("aria-label", self.l10n.yearAriaLabel); - if (self.config.minDate) - yearElement.setAttribute("data-min", self.config.minDate.getFullYear().toString()); - if (self.config.maxDate) { - yearElement.setAttribute("data-max", self.config.maxDate.getFullYear().toString()); - yearElement.disabled = - !!self.config.minDate && - self.config.minDate.getFullYear() === self.config.maxDate.getFullYear(); - } - const currentMonth = createElement("div", "flatpickr-current-month"); - currentMonth.appendChild(monthElement); - currentMonth.appendChild(yearInput); - monthNavFragment.appendChild(currentMonth); - container.appendChild(monthNavFragment); - return { - container, - yearElement, - monthElement, - }; - } - function buildMonths() { - clearNode(self.monthNav); - self.monthNav.appendChild(self.prevMonthNav); - if (self.config.showMonths) { - self.yearElements = []; - self.monthElements = []; - } - for (let m = self.config.showMonths; m--;) { - const month = buildMonth(); - self.yearElements.push(month.yearElement); - self.monthElements.push(month.monthElement); - self.monthNav.appendChild(month.container); - } - self.monthNav.appendChild(self.nextMonthNav); - } - function buildMonthNav() { - self.monthNav = createElement("div", "flatpickr-months"); - self.yearElements = []; - self.monthElements = []; - self.prevMonthNav = createElement("span", "flatpickr-prev-month"); - self.prevMonthNav.innerHTML = self.config.prevArrow; - self.nextMonthNav = createElement("span", "flatpickr-next-month"); - self.nextMonthNav.innerHTML = self.config.nextArrow; - buildMonths(); - Object.defineProperty(self, "_hidePrevMonthArrow", { - get: () => self.__hidePrevMonthArrow, - set(bool) { - if (self.__hidePrevMonthArrow !== bool) { - toggleClass(self.prevMonthNav, "disabled", bool); - self.__hidePrevMonthArrow = bool; - } - }, - }); - Object.defineProperty(self, "_hideNextMonthArrow", { - get: () => self.__hideNextMonthArrow, - set(bool) { - if (self.__hideNextMonthArrow !== bool) { - toggleClass(self.nextMonthNav, "disabled", bool); - self.__hideNextMonthArrow = bool; - } - }, - }); - self.currentYearElement = self.yearElements[0]; - updateNavigationCurrentMonth(); - return self.monthNav; - } - function buildTime() { - self.calendarContainer.classList.add("hasTime"); - if (self.config.noCalendar) - self.calendarContainer.classList.add("noCalendar"); - self.timeContainer = createElement("div", "flatpickr-time"); - self.timeContainer.tabIndex = -1; - const separator = createElement("span", "flatpickr-time-separator", ":"); - const hourInput = createNumberInput("flatpickr-hour"); - self.hourElement = hourInput.getElementsByTagName("input")[0]; - const minuteInput = createNumberInput("flatpickr-minute"); - self.minuteElement = minuteInput.getElementsByTagName("input")[0]; - self.hourElement.tabIndex = self.minuteElement.tabIndex = -1; - self.hourElement.value = pad(self.latestSelectedDateObj - ? self.latestSelectedDateObj.getHours() - : self.config.time_24hr - ? self.config.defaultHour - : military2ampm(self.config.defaultHour)); - self.minuteElement.value = pad(self.latestSelectedDateObj - ? self.latestSelectedDateObj.getMinutes() - : self.config.defaultMinute); - self.hourElement.setAttribute("data-step", self.config.hourIncrement.toString()); - self.minuteElement.setAttribute("data-step", self.config.minuteIncrement.toString()); - self.hourElement.setAttribute("data-min", self.config.time_24hr ? "0" : "1"); - self.hourElement.setAttribute("data-max", self.config.time_24hr ? "23" : "12"); - self.minuteElement.setAttribute("data-min", "0"); - self.minuteElement.setAttribute("data-max", "59"); - self.timeContainer.appendChild(hourInput); - self.timeContainer.appendChild(separator); - self.timeContainer.appendChild(minuteInput); - if (self.config.time_24hr) - self.timeContainer.classList.add("time24hr"); - if (self.config.enableSeconds) { - self.timeContainer.classList.add("hasSeconds"); - const secondInput = createNumberInput("flatpickr-second"); - self.secondElement = secondInput.getElementsByTagName("input")[0]; - self.secondElement.value = pad(self.latestSelectedDateObj - ? self.latestSelectedDateObj.getSeconds() - : self.config.defaultSeconds); - self.secondElement.setAttribute("data-step", self.minuteElement.getAttribute("data-step")); - self.secondElement.setAttribute("data-min", self.minuteElement.getAttribute("data-min")); - self.secondElement.setAttribute("data-max", self.minuteElement.getAttribute("data-max")); - self.timeContainer.appendChild(createElement("span", "flatpickr-time-separator", ":")); - self.timeContainer.appendChild(secondInput); - } - if (!self.config.time_24hr) { - self.amPM = createElement("span", "flatpickr-am-pm", self.l10n.amPM[int((self.latestSelectedDateObj - ? self.hourElement.value - : self.config.defaultHour) > 11)]); - self.amPM.title = self.l10n.toggleTitle; - self.amPM.tabIndex = -1; - self.timeContainer.appendChild(self.amPM); - } - return self.timeContainer; - } - function buildWeekdays() { - if (!self.weekdayContainer) - self.weekdayContainer = createElement("div", "flatpickr-weekdays"); - else - clearNode(self.weekdayContainer); - for (let i = self.config.showMonths; i--;) { - const container = createElement("div", "flatpickr-weekdaycontainer"); - self.weekdayContainer.appendChild(container); - } - updateWeekdays(); - return self.weekdayContainer; - } - function updateWeekdays() { - const firstDayOfWeek = self.l10n.firstDayOfWeek; - let weekdays = [...self.l10n.weekdays.shorthand]; - if (firstDayOfWeek > 0 && firstDayOfWeek < weekdays.length) { - weekdays = [ - ...weekdays.splice(firstDayOfWeek, weekdays.length), - ...weekdays.splice(0, firstDayOfWeek), - ]; - } - for (let i = self.config.showMonths; i--;) { - self.weekdayContainer.children[i].innerHTML = ` - - ${weekdays.join("")} - - `; - } - } - function buildWeeks() { - self.calendarContainer.classList.add("hasWeeks"); - const weekWrapper = createElement("div", "flatpickr-weekwrapper"); - weekWrapper.appendChild(createElement("span", "flatpickr-weekday", self.l10n.weekAbbreviation)); - const weekNumbers = createElement("div", "flatpickr-weeks"); - weekWrapper.appendChild(weekNumbers); - return { - weekWrapper, - weekNumbers, - }; - } - function changeMonth(value, is_offset = true) { - const delta = is_offset ? value : value - self.currentMonth; - if ((delta < 0 && self._hidePrevMonthArrow === true) || - (delta > 0 && self._hideNextMonthArrow === true)) - return; - self.currentMonth += delta; - if (self.currentMonth < 0 || self.currentMonth > 11) { - self.currentYear += self.currentMonth > 11 ? 1 : -1; - self.currentMonth = (self.currentMonth + 12) % 12; - triggerEvent("onYearChange"); - } - buildDays(); - triggerEvent("onMonthChange"); - updateNavigationCurrentMonth(); - } - function clear(triggerChangeEvent = true) { - self.input.value = ""; - if (self.altInput !== undefined) - self.altInput.value = ""; - if (self.mobileInput !== undefined) - self.mobileInput.value = ""; - self.selectedDates = []; - self.latestSelectedDateObj = undefined; - self.showTimeInput = false; - if (self.config.enableTime === true) { - setDefaultHours(); - } - self.redraw(); - if (triggerChangeEvent) - triggerEvent("onChange"); - } - function close() { - self.isOpen = false; - if (!self.isMobile) { - self.calendarContainer.classList.remove("open"); - self._input.classList.remove("active"); - } - triggerEvent("onClose"); - } - function destroy() { - if (self.config !== undefined) - triggerEvent("onDestroy"); - for (let i = self._handlers.length; i--;) { - const h = self._handlers[i]; - h.element.removeEventListener(h.event, h.handler, h.options); - } - self._handlers = []; - if (self.mobileInput) { - if (self.mobileInput.parentNode) - self.mobileInput.parentNode.removeChild(self.mobileInput); - self.mobileInput = undefined; - } - else if (self.calendarContainer && self.calendarContainer.parentNode) { - if (self.config.static && self.calendarContainer.parentNode) { - const wrapper = self.calendarContainer.parentNode; - wrapper.lastChild && wrapper.removeChild(wrapper.lastChild); - if (wrapper.parentNode) { - while (wrapper.firstChild) - wrapper.parentNode.insertBefore(wrapper.firstChild, wrapper); - wrapper.parentNode.removeChild(wrapper); - } - } - else - self.calendarContainer.parentNode.removeChild(self.calendarContainer); - } - if (self.altInput) { - self.input.type = "text"; - if (self.altInput.parentNode) - self.altInput.parentNode.removeChild(self.altInput); - delete self.altInput; - } - if (self.input) { - self.input.type = self.input._type; - self.input.classList.remove("flatpickr-input"); - self.input.removeAttribute("readonly"); - self.input.value = ""; - } - [ - "_showTimeInput", - "latestSelectedDateObj", - "_hideNextMonthArrow", - "_hidePrevMonthArrow", - "__hideNextMonthArrow", - "__hidePrevMonthArrow", - "isMobile", - "isOpen", - "selectedDateElem", - "minDateHasTime", - "maxDateHasTime", - "days", - "daysContainer", - "_input", - "_positionElement", - "innerContainer", - "rContainer", - "monthNav", - "todayDateElem", - "calendarContainer", - "weekdayContainer", - "prevMonthNav", - "nextMonthNav", - "currentMonthElement", - "currentYearElement", - "navigationCurrentMonth", - "selectedDateElem", - "config", - ].forEach(k => { - try { - delete self[k]; - } - catch (_) { } - }); - } - function isCalendarElem(elem) { - if (self.config.appendTo && self.config.appendTo.contains(elem)) - return true; - return self.calendarContainer.contains(elem); - } - function documentClick(e) { - if (self.isOpen && !self.config.inline) { - const eventTarget = getEventTarget(e); - const isCalendarElement = isCalendarElem(eventTarget); - const isInput = eventTarget === self.input || - eventTarget === self.altInput || - self.element.contains(eventTarget) || - (e.path && - e.path.indexOf && - (~e.path.indexOf(self.input) || - ~e.path.indexOf(self.altInput))); - const lostFocus = e.type === "blur" - ? isInput && - e.relatedTarget && - !isCalendarElem(e.relatedTarget) - : !isInput && - !isCalendarElement && - !isCalendarElem(e.relatedTarget); - const isIgnored = !self.config.ignoredFocusElements.some(elem => elem.contains(eventTarget)); - if (lostFocus && isIgnored) { - self.close(); - if (self.config.mode === "range" && self.selectedDates.length === 1) { - self.clear(false); - self.redraw(); - } - } - } - } - function changeYear(newYear) { - if (!newYear || - (self.config.minDate && newYear < self.config.minDate.getFullYear()) || - (self.config.maxDate && newYear > self.config.maxDate.getFullYear())) - return; - const newYearNum = newYear, isNewYear = self.currentYear !== newYearNum; - self.currentYear = newYearNum || self.currentYear; - if (self.config.maxDate && - self.currentYear === self.config.maxDate.getFullYear()) { - self.currentMonth = Math.min(self.config.maxDate.getMonth(), self.currentMonth); - } - else if (self.config.minDate && - self.currentYear === self.config.minDate.getFullYear()) { - self.currentMonth = Math.max(self.config.minDate.getMonth(), self.currentMonth); - } - if (isNewYear) { - self.redraw(); - triggerEvent("onYearChange"); - } - } - function isEnabled(date, timeless = true) { - const dateToCheck = self.parseDate(date, undefined, timeless); - if ((self.config.minDate && - dateToCheck && - compareDates(dateToCheck, self.config.minDate, timeless !== undefined ? timeless : !self.minDateHasTime) < 0) || - (self.config.maxDate && - dateToCheck && - compareDates(dateToCheck, self.config.maxDate, timeless !== undefined ? timeless : !self.maxDateHasTime) > 0)) - return false; - if (self.config.enable.length === 0 && self.config.disable.length === 0) - return true; - if (dateToCheck === undefined) - return false; - const bool = self.config.enable.length > 0, array = bool ? self.config.enable : self.config.disable; - for (let i = 0, d; i < array.length; i++) { - d = array[i]; - if (typeof d === "function" && - d(dateToCheck)) - return bool; - else if (d instanceof Date && - dateToCheck !== undefined && - d.getTime() === dateToCheck.getTime()) - return bool; - else if (typeof d === "string" && dateToCheck !== undefined) { - const parsed = self.parseDate(d, undefined, true); - return parsed && parsed.getTime() === dateToCheck.getTime() - ? bool - : !bool; - } - else if (typeof d === "object" && - dateToCheck !== undefined && - d.from && - d.to && - dateToCheck.getTime() >= d.from.getTime() && - dateToCheck.getTime() <= d.to.getTime()) - return bool; - } - return !bool; - } - function isInView(elem) { - if (self.daysContainer !== undefined) - return (elem.className.indexOf("hidden") === -1 && - self.daysContainer.contains(elem)); - return false; - } - function onKeyDown(e) { - const isInput = e.target === self._input; - const allowInput = self.config.allowInput; - const allowKeydown = self.isOpen && (!allowInput || !isInput); - const allowInlineKeydown = self.config.inline && isInput && !allowInput; - if (e.keyCode === 13 && isInput) { - if (allowInput) { - self.setDate(self._input.value, true, e.target === self.altInput - ? self.config.altFormat - : self.config.dateFormat); - return e.target.blur(); - } - else - self.open(); - } - else if (isCalendarElem(e.target) || - allowKeydown || - allowInlineKeydown) { - const isTimeObj = !!self.timeContainer && - self.timeContainer.contains(e.target); - switch (e.keyCode) { - case 13: - if (isTimeObj) - updateTime(); - else - selectDate(e); - break; - case 27: - e.preventDefault(); - focusAndClose(); - break; - case 8: - case 46: - if (isInput && !self.config.allowInput) { - e.preventDefault(); - self.clear(); - } - break; - case 37: - case 39: - if (!isTimeObj) { - e.preventDefault(); - if (self.daysContainer !== undefined && - (allowInput === false || - (document.activeElement && isInView(document.activeElement)))) { - const delta = e.keyCode === 39 ? 1 : -1; - if (!e.ctrlKey) - focusOnDay(undefined, delta); - else { - e.stopPropagation(); - changeMonth(delta); - focusOnDay(getFirstAvailableDay(1), 0); - } - } - } - else if (self.hourElement) - self.hourElement.focus(); - break; - case 38: - case 40: - e.preventDefault(); - const delta = e.keyCode === 40 ? 1 : -1; - if ((self.daysContainer && e.target.$i !== undefined) || - e.target === self.input) { - if (e.ctrlKey) { - e.stopPropagation(); - changeYear(self.currentYear - delta); - focusOnDay(getFirstAvailableDay(1), 0); - } - else if (!isTimeObj) - focusOnDay(undefined, delta * 7); - } - else if (self.config.enableTime) { - if (!isTimeObj && self.hourElement) - self.hourElement.focus(); - updateTime(e); - self._debouncedChange(); - } - break; - case 9: - if (!isTimeObj) { - self.element.focus(); - break; - } - const elems = [ - self.hourElement, - self.minuteElement, - self.secondElement, - self.amPM, - ].filter(x => x); - const i = elems.indexOf(e.target); - if (i !== -1) { - const target = elems[i + (e.shiftKey ? -1 : 1)]; - if (target !== undefined) { - e.preventDefault(); - target.focus(); - } - else { - self.element.focus(); - } - } - break; - default: - break; - } - } - if (self.amPM !== undefined && e.target === self.amPM) { - switch (e.key) { - case self.l10n.amPM[0].charAt(0): - case self.l10n.amPM[0].charAt(0).toLowerCase(): - self.amPM.textContent = self.l10n.amPM[0]; - setHoursFromInputs(); - updateValue(); - break; - case self.l10n.amPM[1].charAt(0): - case self.l10n.amPM[1].charAt(0).toLowerCase(): - self.amPM.textContent = self.l10n.amPM[1]; - setHoursFromInputs(); - updateValue(); - break; - } - } - triggerEvent("onKeyDown", e); - } - function onMouseOver(elem) { - if (self.selectedDates.length !== 1 || - (elem && - (!elem.classList.contains("flatpickr-day") || - elem.classList.contains("disabled")))) - return; - const hoverDate = elem - ? elem.dateObj.getTime() - : self.days.firstElementChild.dateObj.getTime(), initialDate = self.parseDate(self.selectedDates[0], undefined, true).getTime(), rangeStartDate = Math.min(hoverDate, self.selectedDates[0].getTime()), rangeEndDate = Math.max(hoverDate, self.selectedDates[0].getTime()), lastDate = self.daysContainer.lastChild - .lastChild.dateObj.getTime(); - let containsDisabled = false; - let minRange = 0, maxRange = 0; - for (let t = rangeStartDate; t < lastDate; t += duration.DAY) { - if (!isEnabled(new Date(t), true)) { - containsDisabled = - containsDisabled || (t > rangeStartDate && t < rangeEndDate); - if (t < initialDate && (!minRange || t > minRange)) - minRange = t; - else if (t > initialDate && (!maxRange || t < maxRange)) - maxRange = t; - } - } - for (let m = 0; m < self.config.showMonths; m++) { - const month = self.daysContainer.children[m]; - const prevMonth = self.daysContainer.children[m - 1]; - for (let i = 0, l = month.children.length; i < l; i++) { - const dayElem = month.children[i], date = dayElem.dateObj; - const timestamp = date.getTime(); - const outOfRange = (minRange > 0 && timestamp < minRange) || - (maxRange > 0 && timestamp > maxRange); - if (outOfRange) { - dayElem.classList.add("notAllowed"); - ["inRange", "startRange", "endRange"].forEach(c => { - dayElem.classList.remove(c); - }); - continue; - } - else if (containsDisabled && !outOfRange) - continue; - ["startRange", "inRange", "endRange", "notAllowed"].forEach(c => { - dayElem.classList.remove(c); - }); - if (elem !== undefined) { - elem.classList.add(hoverDate < self.selectedDates[0].getTime() - ? "startRange" - : "endRange"); - if (month.contains(elem) || - !(m > 0 && - prevMonth && - prevMonth.lastChild.dateObj.getTime() >= timestamp)) { - if (initialDate < hoverDate && timestamp === initialDate) - dayElem.classList.add("startRange"); - else if (initialDate > hoverDate && timestamp === initialDate) - dayElem.classList.add("endRange"); - if (timestamp >= minRange && - (maxRange === 0 || timestamp <= maxRange) && - isBetween(timestamp, initialDate, hoverDate)) - dayElem.classList.add("inRange"); - } - } - } - } - } - function onResize() { - if (self.isOpen && !self.config.static && !self.config.inline) - positionCalendar(); - } - function open(e, positionElement = self._positionElement) { - if (self.isMobile === true) { - if (e) { - e.preventDefault(); - e.target && e.target.blur(); - } - if (self.mobileInput !== undefined) { - self.mobileInput.focus(); - self.mobileInput.click(); - } - triggerEvent("onOpen"); - return; - } - if (self._input.disabled || self.config.inline) - return; - const wasOpen = self.isOpen; - self.isOpen = true; - if (!wasOpen) { - self.calendarContainer.classList.add("open"); - self._input.classList.add("active"); - triggerEvent("onOpen"); - positionCalendar(positionElement); - self.calendarContainer.focus(); - if (self.config.noCalendar === false) { - focusOnDay(undefined, 0); - } - } - if (self.config.enableTime === true && self.config.noCalendar === true) { - if (self.selectedDates.length === 0) { - self.setDate(self.config.minDate !== undefined - ? new Date(self.config.minDate.getTime()) - : new Date(), false); - setDefaultHours(); - updateValue(); - } - if (self.config.allowInput === false && - (e === undefined || - !self.timeContainer.contains(e.relatedTarget))) { - setTimeout(() => self.hourElement.select(), 50); - } - } - } - function minMaxDateSetter(type) { - return (date) => { - const dateObj = (self.config[`_${type}Date`] = self.parseDate(date, self.config.dateFormat)); - const inverseDateObj = self.config[`_${type === "min" ? "max" : "min"}Date`]; - if (dateObj !== undefined) { - self[type === "min" ? "minDateHasTime" : "maxDateHasTime"] = - dateObj.getHours() > 0 || - dateObj.getMinutes() > 0 || - dateObj.getSeconds() > 0; - } - if (self.selectedDates) { - self.selectedDates = self.selectedDates.filter(d => isEnabled(d)); - if (!self.selectedDates.length && type === "min") - setHoursFromDate(dateObj); - updateValue(); - } - if (self.daysContainer) { - redraw(); - if (dateObj !== undefined) - self.currentYearElement[type] = dateObj.getFullYear().toString(); - else - self.currentYearElement.removeAttribute(type); - self.currentYearElement.disabled = - !!inverseDateObj && - dateObj !== undefined && - inverseDateObj.getFullYear() === dateObj.getFullYear(); - } - }; - } - function parseConfig() { - const boolOpts = [ - "wrap", - "weekNumbers", - "allowInput", - "clickOpens", - "time_24hr", - "enableTime", - "noCalendar", - "altInput", - "shorthandCurrentMonth", - "inline", - "static", - "enableSeconds", - "disableMobile", - ]; - const userConfig = Object.assign({}, instanceConfig, JSON.parse(JSON.stringify(element.dataset || {}))); - const formats$$1 = {}; - self.config.parseDate = userConfig.parseDate; - self.config.formatDate = userConfig.formatDate; - Object.defineProperty(self.config, "enable", { - get: () => self.config._enable, - set: dates => { - self.config._enable = parseDateRules(dates); - }, - }); - Object.defineProperty(self.config, "disable", { - get: () => self.config._disable, - set: dates => { - self.config._disable = parseDateRules(dates); - }, - }); - const timeMode = userConfig.mode === "time"; - if (!userConfig.dateFormat && (userConfig.enableTime || timeMode)) { - formats$$1.dateFormat = - userConfig.noCalendar || timeMode - ? "H:i" + (userConfig.enableSeconds ? ":S" : "") - : flatpickr.defaultConfig.dateFormat + - " H:i" + - (userConfig.enableSeconds ? ":S" : ""); - } - if (userConfig.altInput && - (userConfig.enableTime || timeMode) && - !userConfig.altFormat) { - formats$$1.altFormat = - userConfig.noCalendar || timeMode - ? "h:i" + (userConfig.enableSeconds ? ":S K" : " K") - : flatpickr.defaultConfig.altFormat + - ` h:i${userConfig.enableSeconds ? ":S" : ""} K`; - } - Object.defineProperty(self.config, "minDate", { - get: () => self.config._minDate, - set: minMaxDateSetter("min"), - }); - Object.defineProperty(self.config, "maxDate", { - get: () => self.config._maxDate, - set: minMaxDateSetter("max"), - }); - const minMaxTimeSetter = (type) => (val) => { - self.config[type === "min" ? "_minTime" : "_maxTime"] = self.parseDate(val, "H:i"); - }; - Object.defineProperty(self.config, "minTime", { - get: () => self.config._minTime, - set: minMaxTimeSetter("min"), - }); - Object.defineProperty(self.config, "maxTime", { - get: () => self.config._maxTime, - set: minMaxTimeSetter("max"), - }); - if (userConfig.mode === "time") { - self.config.noCalendar = true; - self.config.enableTime = true; - } - Object.assign(self.config, formats$$1, userConfig); - for (let i = 0; i < boolOpts.length; i++) - self.config[boolOpts[i]] = - self.config[boolOpts[i]] === true || - self.config[boolOpts[i]] === "true"; - HOOKS.filter(hook => self.config[hook] !== undefined).forEach(hook => { - self.config[hook] = arrayify(self.config[hook] || []).map(bindToInstance); - }); - self.isMobile = - !self.config.disableMobile && - !self.config.inline && - self.config.mode === "single" && - !self.config.disable.length && - !self.config.enable.length && - !self.config.weekNumbers && - /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent); - for (let i = 0; i < self.config.plugins.length; i++) { - const pluginConf = self.config.plugins[i](self) || {}; - for (const key in pluginConf) { - if (HOOKS.indexOf(key) > -1) { - self.config[key] = arrayify(pluginConf[key]) - .map(bindToInstance) - .concat(self.config[key]); - } - else if (typeof userConfig[key] === "undefined") - self.config[key] = pluginConf[key]; - } - } - triggerEvent("onParseConfig"); - } - function setupLocale() { - if (typeof self.config.locale !== "object" && - typeof flatpickr.l10ns[self.config.locale] === "undefined") - self.config.errorHandler(new Error(`flatpickr: invalid locale ${self.config.locale}`)); - self.l10n = Object.assign({}, flatpickr.l10ns.default, (typeof self.config.locale === "object" - ? self.config.locale - : self.config.locale !== "default" - ? flatpickr.l10ns[self.config.locale] - : undefined)); - tokenRegex.K = `(${self.l10n.amPM[0]}|${self.l10n.amPM[1]}|${self.l10n.amPM[0].toLowerCase()}|${self.l10n.amPM[1].toLowerCase()})`; - self.formatDate = createDateFormatter(self); - self.parseDate = createDateParser({ config: self.config, l10n: self.l10n }); - } - function positionCalendar(customPositionElement) { - if (self.calendarContainer === undefined) - return; - triggerEvent("onPreCalendarPosition"); - const positionElement = customPositionElement || self._positionElement; - const calendarHeight = Array.prototype.reduce.call(self.calendarContainer.children, (acc, child) => acc + child.offsetHeight, 0), calendarWidth = self.calendarContainer.offsetWidth, configPos = self.config.position.split(" "), configPosVertical = configPos[0], configPosHorizontal = configPos.length > 1 ? configPos[1] : null, inputBounds = positionElement.getBoundingClientRect(), distanceFromBottom = window.innerHeight - inputBounds.bottom, showOnTop = configPosVertical === "above" || - (configPosVertical !== "below" && - distanceFromBottom < calendarHeight && - inputBounds.top > calendarHeight); - let top = window.pageYOffset + - inputBounds.top + - (!showOnTop ? positionElement.offsetHeight + 2 : -calendarHeight - 2); - toggleClass(self.calendarContainer, "arrowTop", !showOnTop); - toggleClass(self.calendarContainer, "arrowBottom", showOnTop); - if (self.config.inline) - return; - const left = window.pageXOffset + - inputBounds.left - - (configPosHorizontal != null && configPosHorizontal === "center" - ? (calendarWidth - inputBounds.width) / 2 - : 0); - const right = window.document.body.offsetWidth - inputBounds.right; - const rightMost = left + calendarWidth > window.document.body.offsetWidth; - const centerMost = right + calendarWidth > window.document.body.offsetWidth; - toggleClass(self.calendarContainer, "rightMost", rightMost); - if (self.config.static) - return; - self.calendarContainer.style.top = `${top}px`; - if (!rightMost) { - self.calendarContainer.style.left = `${left}px`; - self.calendarContainer.style.right = "auto"; - } - else if (!centerMost) { - self.calendarContainer.style.left = "auto"; - self.calendarContainer.style.right = `${right}px`; - } - else { - const doc = document.styleSheets[0]; - const bodyWidth = window.document.body.offsetWidth; - const centerLeft = Math.max(0, bodyWidth / 2 - calendarWidth / 2); - const centerBefore = ".flatpickr-calendar.centerMost:before"; - const centerAfter = ".flatpickr-calendar.centerMost:after"; - const centerIndex = doc.cssRules.length; - const centerStyle = `{left:${inputBounds.left}px;right:auto;}`; - toggleClass(self.calendarContainer, "rightMost", false); - toggleClass(self.calendarContainer, "centerMost", true); - doc.insertRule(`${centerBefore},${centerAfter}${centerStyle}`, centerIndex); - self.calendarContainer.style.left = `${centerLeft}px`; - self.calendarContainer.style.right = "auto"; - } - } - function redraw() { - if (self.config.noCalendar || self.isMobile) - return; - updateNavigationCurrentMonth(); - buildDays(); - } - function focusAndClose() { - self._input.focus(); - if (window.navigator.userAgent.indexOf("MSIE") !== -1 || - navigator.msMaxTouchPoints !== undefined) { - setTimeout(self.close, 0); - } - else { - self.close(); - } - } - function selectDate(e) { - e.preventDefault(); - e.stopPropagation(); - const isSelectable = (day) => day.classList && - day.classList.contains("flatpickr-day") && - !day.classList.contains("disabled") && - !day.classList.contains("notAllowed"); - const t = findParent(e.target, isSelectable); - if (t === undefined) - return; - const target = t; - const selectedDate = (self.latestSelectedDateObj = new Date(target.dateObj.getTime())); - const shouldChangeMonth = (selectedDate.getMonth() < self.currentMonth || - selectedDate.getMonth() > - self.currentMonth + self.config.showMonths - 1) && - self.config.mode !== "range"; - self.selectedDateElem = target; - if (self.config.mode === "single") - self.selectedDates = [selectedDate]; - else if (self.config.mode === "multiple") { - const selectedIndex = isDateSelected(selectedDate); - if (selectedIndex) - self.selectedDates.splice(parseInt(selectedIndex), 1); - else - self.selectedDates.push(selectedDate); - } - else if (self.config.mode === "range") { - if (self.selectedDates.length === 2) - self.clear(false); - self.latestSelectedDateObj = selectedDate; - self.selectedDates.push(selectedDate); - if (compareDates(selectedDate, self.selectedDates[0], true) !== 0) - self.selectedDates.sort((a, b) => a.getTime() - b.getTime()); - } - setHoursFromInputs(); - if (shouldChangeMonth) { - const isNewYear = self.currentYear !== selectedDate.getFullYear(); - self.currentYear = selectedDate.getFullYear(); - self.currentMonth = selectedDate.getMonth(); - if (isNewYear) - triggerEvent("onYearChange"); - triggerEvent("onMonthChange"); - } - updateNavigationCurrentMonth(); - buildDays(); - updateValue(); - if (self.config.enableTime) - setTimeout(() => (self.showTimeInput = true), 50); - if (!shouldChangeMonth && - self.config.mode !== "range" && - self.config.showMonths === 1) - focusOnDayElem(target); - else if (self.selectedDateElem !== undefined && - self.hourElement === undefined) { - self.selectedDateElem && self.selectedDateElem.focus(); - } - if (self.hourElement !== undefined) - self.hourElement !== undefined && self.hourElement.focus(); - if (self.config.closeOnSelect) { - const single = self.config.mode === "single" && !self.config.enableTime; - const range = self.config.mode === "range" && - self.selectedDates.length === 2 && - !self.config.enableTime; - if (single || range) { - focusAndClose(); - } - } - triggerChange(); - } - const CALLBACKS = { - locale: [setupLocale, updateWeekdays], - showMonths: [buildMonths, setCalendarWidth, buildWeekdays], - }; - function set(option, value) { - if (option !== null && typeof option === "object") - Object.assign(self.config, option); - else { - self.config[option] = value; - if (CALLBACKS[option] !== undefined) - CALLBACKS[option].forEach(x => x()); - else if (HOOKS.indexOf(option) > -1) - self.config[option] = arrayify(value); - } - self.redraw(); - jumpToDate(); - updateValue(false); - } - function setSelectedDate(inputDate, format) { - let dates = []; - if (inputDate instanceof Array) - dates = inputDate.map(d => self.parseDate(d, format)); - else if (inputDate instanceof Date || typeof inputDate === "number") - dates = [self.parseDate(inputDate, format)]; - else if (typeof inputDate === "string") { - switch (self.config.mode) { - case "single": - case "time": - dates = [self.parseDate(inputDate, format)]; - break; - case "multiple": - dates = inputDate - .split(self.config.conjunction) - .map(date => self.parseDate(date, format)); - break; - case "range": - dates = inputDate - .split(self.l10n.rangeSeparator) - .map(date => self.parseDate(date, format)); - break; - default: - break; - } - } - else - self.config.errorHandler(new Error(`Invalid date supplied: ${JSON.stringify(inputDate)}`)); - self.selectedDates = dates.filter(d => d instanceof Date && isEnabled(d, false)); - if (self.config.mode === "range") - self.selectedDates.sort((a, b) => a.getTime() - b.getTime()); - } - function setDate(date, triggerChange = false, format = self.config.dateFormat) { - if ((date !== 0 && !date) || (date instanceof Array && date.length === 0)) - return self.clear(triggerChange); - setSelectedDate(date, format); - self.showTimeInput = self.selectedDates.length > 0; - self.latestSelectedDateObj = self.selectedDates[0]; - self.redraw(); - jumpToDate(); - setHoursFromDate(); - updateValue(triggerChange); - if (triggerChange) - triggerEvent("onChange"); - } - function parseDateRules(arr) { - return arr - .slice() - .map(rule => { - if (typeof rule === "string" || - typeof rule === "number" || - rule instanceof Date) { - return self.parseDate(rule, undefined, true); - } - else if (rule && - typeof rule === "object" && - rule.from && - rule.to) - return { - from: self.parseDate(rule.from, undefined), - to: self.parseDate(rule.to, undefined), - }; - return rule; - }) - .filter(x => x); - } - function setupDates() { - self.selectedDates = []; - self.now = self.parseDate(self.config.now) || new Date(); - const preloadedDate = self.config.defaultDate || - ((self.input.nodeName === "INPUT" || - self.input.nodeName === "TEXTAREA") && - self.input.placeholder && - self.input.value === self.input.placeholder - ? null - : self.input.value); - if (preloadedDate) - setSelectedDate(preloadedDate, self.config.dateFormat); - const initialDate = self.selectedDates.length > 0 - ? self.selectedDates[0] - : self.config.minDate && - self.config.minDate.getTime() > self.now.getTime() - ? self.config.minDate - : self.config.maxDate && - self.config.maxDate.getTime() < self.now.getTime() - ? self.config.maxDate - : self.now; - self.currentYear = initialDate.getFullYear(); - self.currentMonth = initialDate.getMonth(); - if (self.selectedDates.length > 0) - self.latestSelectedDateObj = self.selectedDates[0]; - if (self.config.minTime !== undefined) - self.config.minTime = self.parseDate(self.config.minTime, "H:i"); - if (self.config.maxTime !== undefined) - self.config.maxTime = self.parseDate(self.config.maxTime, "H:i"); - self.minDateHasTime = - !!self.config.minDate && - (self.config.minDate.getHours() > 0 || - self.config.minDate.getMinutes() > 0 || - self.config.minDate.getSeconds() > 0); - self.maxDateHasTime = - !!self.config.maxDate && - (self.config.maxDate.getHours() > 0 || - self.config.maxDate.getMinutes() > 0 || - self.config.maxDate.getSeconds() > 0); - Object.defineProperty(self, "showTimeInput", { - get: () => self._showTimeInput, - set(bool) { - self._showTimeInput = bool; - if (self.calendarContainer) - toggleClass(self.calendarContainer, "showTimeInput", bool); - self.isOpen && positionCalendar(); - }, - }); - } - function setupInputs() { - self.input = self.config.wrap - ? element.querySelector("[data-input]") - : element; - if (!self.input) { - self.config.errorHandler(new Error("Invalid input element specified")); - return; - } - self.input._type = self.input.type; - self.input.type = "text"; - self.input.classList.add("flatpickr-input"); - self._input = self.input; - if (self.config.altInput) { - self.altInput = createElement(self.input.nodeName, self.input.className + " " + self.config.altInputClass); - self._input = self.altInput; - self.altInput.placeholder = self.input.placeholder; - self.altInput.disabled = self.input.disabled; - self.altInput.required = self.input.required; - self.altInput.tabIndex = self.input.tabIndex; - self.altInput.type = "text"; - self.input.setAttribute("type", "hidden"); - if (!self.config.static && self.input.parentNode) - self.input.parentNode.insertBefore(self.altInput, self.input.nextSibling); - } - if (!self.config.allowInput) - self._input.setAttribute("readonly", "readonly"); - self._positionElement = self.config.positionElement || self._input; - } - function setupMobile() { - const inputType = self.config.enableTime - ? self.config.noCalendar - ? "time" - : "datetime-local" - : "date"; - self.mobileInput = createElement("input", self.input.className + " flatpickr-mobile"); - self.mobileInput.step = self.input.getAttribute("step") || "any"; - self.mobileInput.tabIndex = 1; - self.mobileInput.type = inputType; - self.mobileInput.disabled = self.input.disabled; - self.mobileInput.required = self.input.required; - self.mobileInput.placeholder = self.input.placeholder; - self.mobileFormatStr = - inputType === "datetime-local" - ? "Y-m-d\\TH:i:S" - : inputType === "date" - ? "Y-m-d" - : "H:i:S"; - if (self.selectedDates.length > 0) { - self.mobileInput.defaultValue = self.mobileInput.value = self.formatDate(self.selectedDates[0], self.mobileFormatStr); - } - if (self.config.minDate) - self.mobileInput.min = self.formatDate(self.config.minDate, "Y-m-d"); - if (self.config.maxDate) - self.mobileInput.max = self.formatDate(self.config.maxDate, "Y-m-d"); - self.input.type = "hidden"; - if (self.altInput !== undefined) - self.altInput.type = "hidden"; - try { - if (self.input.parentNode) - self.input.parentNode.insertBefore(self.mobileInput, self.input.nextSibling); - } - catch (_a) { } - bind(self.mobileInput, "change", (e) => { - self.setDate(e.target.value, false, self.mobileFormatStr); - triggerEvent("onChange"); - triggerEvent("onClose"); - }); - } - function toggle(e) { - if (self.isOpen === true) - return self.close(); - self.open(e); - } - function triggerEvent(event, data) { - if (self.config === undefined) - return; - const hooks = self.config[event]; - if (hooks !== undefined && hooks.length > 0) { - for (let i = 0; hooks[i] && i < hooks.length; i++) - hooks[i](self.selectedDates, self.input.value, self, data); - } - if (event === "onChange") { - self.input.dispatchEvent(createEvent("change")); - self.input.dispatchEvent(createEvent("input")); - } - } - function createEvent(name) { - const e = document.createEvent("Event"); - e.initEvent(name, true, true); - return e; - } - function isDateSelected(date) { - for (let i = 0; i < self.selectedDates.length; i++) { - if (compareDates(self.selectedDates[i], date) === 0) - return "" + i; - } - return false; - } - function isDateInRange(date) { - if (self.config.mode !== "range" || self.selectedDates.length < 2) - return false; - return (compareDates(date, self.selectedDates[0]) >= 0 && - compareDates(date, self.selectedDates[1]) <= 0); - } - function updateNavigationCurrentMonth() { - if (self.config.noCalendar || self.isMobile || !self.monthNav) - return; - self.yearElements.forEach((yearElement, i) => { - const d = new Date(self.currentYear, self.currentMonth, 1); - d.setMonth(self.currentMonth + i); - self.monthElements[i].textContent = - monthToStr(d.getMonth(), self.config.shorthandCurrentMonth, self.l10n) + - " "; - yearElement.value = d.getFullYear().toString(); - }); - self._hidePrevMonthArrow = - self.config.minDate !== undefined && - (self.currentYear === self.config.minDate.getFullYear() - ? self.currentMonth <= self.config.minDate.getMonth() - : self.currentYear < self.config.minDate.getFullYear()); - self._hideNextMonthArrow = - self.config.maxDate !== undefined && - (self.currentYear === self.config.maxDate.getFullYear() - ? self.currentMonth + 1 > self.config.maxDate.getMonth() - : self.currentYear > self.config.maxDate.getFullYear()); - } - function getDateStr(format) { - return self.selectedDates - .map(dObj => self.formatDate(dObj, format)) - .filter((d, i, arr) => self.config.mode !== "range" || - self.config.enableTime || - arr.indexOf(d) === i) - .join(self.config.mode !== "range" - ? self.config.conjunction - : self.l10n.rangeSeparator); - } - function updateValue(triggerChange = true) { - if (self.selectedDates.length === 0) - return self.clear(triggerChange); - if (self.mobileInput !== undefined && self.mobileFormatStr) { - self.mobileInput.value = - self.latestSelectedDateObj !== undefined - ? self.formatDate(self.latestSelectedDateObj, self.mobileFormatStr) - : ""; - } - self.input.value = getDateStr(self.config.dateFormat); - if (self.altInput !== undefined) { - self.altInput.value = getDateStr(self.config.altFormat); - } - if (triggerChange !== false) - triggerEvent("onValueUpdate"); - } - function onMonthNavClick(e) { - e.preventDefault(); - const isPrevMonth = self.prevMonthNav.contains(e.target); - const isNextMonth = self.nextMonthNav.contains(e.target); - if (isPrevMonth || isNextMonth) { - changeMonth(isPrevMonth ? -1 : 1); - } - else if (self.yearElements.indexOf(e.target) >= 0) { - e.target.select(); - } - else if (e.target.classList.contains("arrowUp")) { - self.changeYear(self.currentYear + 1); - } - else if (e.target.classList.contains("arrowDown")) { - self.changeYear(self.currentYear - 1); - } - } - function timeWrapper(e) { - e.preventDefault(); - const isKeyDown = e.type === "keydown", input = e.target; - if (self.amPM !== undefined && e.target === self.amPM) { - self.amPM.textContent = - self.l10n.amPM[int(self.amPM.textContent === self.l10n.amPM[0])]; - } - const min = parseFloat(input.getAttribute("data-min")), max = parseFloat(input.getAttribute("data-max")), step = parseFloat(input.getAttribute("data-step")), curValue = parseInt(input.value, 10), delta = e.delta || - (isKeyDown ? (e.which === 38 ? 1 : -1) : 0); - let newValue = curValue + step * delta; - if (typeof input.value !== "undefined" && input.value.length === 2) { - const isHourElem = input === self.hourElement, isMinuteElem = input === self.minuteElement; - if (newValue < min) { - newValue = - max + - newValue + - int(!isHourElem) + - (int(isHourElem) && int(!self.amPM)); - if (isMinuteElem) - incrementNumInput(undefined, -1, self.hourElement); - } - else if (newValue > max) { - newValue = - input === self.hourElement ? newValue - max - int(!self.amPM) : min; - if (isMinuteElem) - incrementNumInput(undefined, 1, self.hourElement); - } - if (self.amPM && - isHourElem && - (step === 1 - ? newValue + curValue === 23 - : Math.abs(newValue - curValue) > step)) { - self.amPM.textContent = - self.l10n.amPM[int(self.amPM.textContent === self.l10n.amPM[0])]; - } - input.value = pad(newValue); - } - } - init(); - return self; - } - function _flatpickr(nodeList, config) { - const nodes = Array.prototype.slice.call(nodeList); - let instances = []; - for (let i = 0; i < nodes.length; i++) { - const node = nodes[i]; - try { - if (node.getAttribute("data-fp-omit") !== null) - continue; - if (node._flatpickr !== undefined) { - node._flatpickr.destroy(); - node._flatpickr = undefined; - } - node._flatpickr = FlatpickrInstance(node, config || {}); - instances.push(node._flatpickr); - } - catch (e) { - console.error(e); - } - } - return instances.length === 1 ? instances[0] : instances; - } - if (typeof HTMLElement !== "undefined") { - HTMLCollection.prototype.flatpickr = NodeList.prototype.flatpickr = function (config) { - return _flatpickr(this, config); - }; - HTMLElement.prototype.flatpickr = function (config) { - return _flatpickr([this], config); - }; - } - var flatpickr = function (selector, config) { - if (selector instanceof NodeList) - return _flatpickr(selector, config); - else if (typeof selector === "string") - return _flatpickr(window.document.querySelectorAll(selector), config); - return _flatpickr([selector], config); - }; - flatpickr.defaultConfig = defaults; - flatpickr.l10ns = { - en: Object.assign({}, english), - default: Object.assign({}, english), - }; - flatpickr.localize = (l10n) => { - flatpickr.l10ns.default = Object.assign({}, flatpickr.l10ns.default, l10n); - }; - flatpickr.setDefaults = (config) => { - flatpickr.defaultConfig = Object.assign({}, flatpickr.defaultConfig, config); - }; - flatpickr.parseDate = createDateParser({}); - flatpickr.formatDate = createDateFormatter({}); - flatpickr.compareDates = compareDates; - if (typeof jQuery !== "undefined") { - jQuery.fn.flatpickr = function (config) { - return _flatpickr(this, config); - }; - } - Date.prototype.fp_incr = function (days) { - return new Date(this.getFullYear(), this.getMonth(), this.getDate() + (typeof days === "string" ? parseInt(days, 10) : days)); - }; - if (typeof window !== "undefined") { - window.flatpickr = flatpickr; - } - - return flatpickr; - -}))); diff --git a/src/examples/date-picker/date-picker-formatted.tsx b/src/examples/date-picker/date-picker-formatted.tsx index 46a512f5d3..4370ead9b3 100644 --- a/src/examples/date-picker/date-picker-formatted.tsx +++ b/src/examples/date-picker/date-picker-formatted.tsx @@ -6,29 +6,40 @@ import { Component, State } from '@stencil/core'; }) export class DatePickerFormattedExample { @State() - private value = new Date('2018-11-04'); - - public handleChange(event) { - this.value = event.detail; - } + private valueNo = new Date('2018-11-04'); + private valueFi = new Date('2018-11-04'); public render() { return [ , +

+ Value: {this.valueNo.toString()} +

, , +

+ Value: {this.valueFi.toString()} +

, ]; } + + private handleChangeNo(event) { + this.valueNo = event.detail; + } + + private handleChangeFi(event) { + this.valueFi = event.detail; + } } diff --git a/src/examples/date-picker/date-picker.tsx b/src/examples/date-picker/date-picker.tsx index 3f6f6ea3aa..32407858f9 100644 --- a/src/examples/date-picker/date-picker.tsx +++ b/src/examples/date-picker/date-picker.tsx @@ -1,40 +1,55 @@ import { Component, State } from '@stencil/core'; import { DateType } from '../../components/date-picker/date-type'; -const types = { - datetime: new Date('2018-12-25 16:00'), - date: new Date('2018-12-25'), - time: new Date('2018-12-25 16:00'), - week: new Date('2018-12-24'), - month: new Date('2018-12-01'), - quarter: new Date('2018-10-01'), - year: new Date('2018-10-01'), -}; +const types = ['datetime', 'date', 'time', 'week', 'month', 'quarter', 'year']; @Component({ tag: 'limel-example-date-picker', shadow: true, }) export class DatePickerExample { - public handleChange = Object.keys(types).map((_, index) => { - return event => { - this.values[index] = event.detail; - }; - }); + @State() + private datetime = new Date('2018-12-25 16:00'); + + @State() + private date = new Date('2018-12-25'); + + @State() + private time = new Date('2018-12-25 16:00'); + + @State() + private week = new Date('2018-12-24'); @State() - private values = (Object as any).values(types); + private month = new Date('2018-12-01'); + + @State() + private quarter = new Date('2018-10-01'); + + @State() + private year = new Date('2018-10-01'); public render() { - return Object.keys(types).map((type: DateType, index) => { + return types.map((type: DateType) => { return ( - +

+ { + this.handleChange(event, type); + }} + /> +

+ Value: {this[type].toString()} +

+

); }); } + + private handleChange(event, type: DateType) { + this[type] = event.detail; + } }