/*! * All material copyright ESRI, All Rights Reserved, unless otherwise specified. * See https://github.com/Esri/calcite-components/blob/master/LICENSE.md for details. * v1.0.0-beta.97 */ import { proxyCustomElement, HTMLElement, createEvent, h, Host } from '@stencil/core/internal/client/index.js'; import { c as sameDate, b as dateFromRange, i as inRange } from './date.js'; import { d as defineCustomElement$1 } from './date-picker-day.js'; const datePickerMonthCss = "@keyframes in{0%{opacity:0}100%{opacity:1}}@keyframes in-down{0%{opacity:0;transform:translate3D(0, -5px, 0)}100%{opacity:1;transform:translate3D(0, 0, 0)}}@keyframes in-up{0%{opacity:0;transform:translate3D(0, 5px, 0)}100%{opacity:1;transform:translate3D(0, 0, 0)}}@keyframes in-scale{0%{opacity:0;transform:scale3D(0.95, 0.95, 1)}100%{opacity:1;transform:scale3D(1, 1, 1)}}:root{--calcite-animation-timing:calc(150ms * var(--calcite-internal-duration-factor));--calcite-internal-duration-factor:var(--calcite-duration-factor, 1);--calcite-internal-animation-timing-fast:calc(100ms * var(--calcite-internal-duration-factor));--calcite-internal-animation-timing-medium:calc(200ms * var(--calcite-internal-duration-factor));--calcite-internal-animation-timing-slow:calc(300ms * var(--calcite-internal-duration-factor))}.calcite-animate{opacity:0;animation-fill-mode:both;animation-duration:var(--calcite-animation-timing)}.calcite-animate__in{animation-name:in}.calcite-animate__in-down{animation-name:in-down}.calcite-animate__in-up{animation-name:in-up}.calcite-animate__in-scale{animation-name:in-scale}@media (prefers-reduced-motion: reduce){:root{--calcite-internal-duration-factor:0.01}}:root{--calcite-floating-ui-transition:var(--calcite-animation-timing)}:host([hidden]){display:none}.calender{margin-block-end:0.25rem}.week-headers{display:flex;border-width:0px;border-block-start-width:1px;border-style:solid;border-color:var(--calcite-ui-border-3);padding-block:0px;padding-inline:0.25rem}.week-header{text-align:center;font-weight:var(--calcite-font-weight-bold);color:var(--calcite-ui-text-3);inline-size:14.2857142857%}:host([scale=s]) .week-header{padding-inline:0px;padding-block:0.5rem 0.75rem;font-size:var(--calcite-font-size--2);line-height:1rem}:host([scale=m]) .week-header{padding-inline:0px;padding-block:0.75rem 1rem;font-size:var(--calcite-font-size--2);line-height:1rem}:host([scale=l]) .week-header{padding-inline:0px;padding-block:1rem 1.25rem;font-size:var(--calcite-font-size--1);line-height:1rem}.week-days{display:flex;flex-direction:row;padding-block:0px;padding-inline:6px}.week-days:focus{outline:2px solid transparent;outline-offset:2px}"; const DatePickerMonth = /*@__PURE__*/ proxyCustomElement(class extends HTMLElement { constructor() { super(); this.__registerHost(); this.__attachShadow(); this.calciteDatePickerSelect = createEvent(this, "calciteDatePickerSelect", 6); this.calciteInternalDatePickerHover = createEvent(this, "calciteInternalDatePickerHover", 6); this.calciteDatePickerActiveDateChange = createEvent(this, "calciteDatePickerActiveDateChange", 6); this.calciteInternalDatePickerMouseOut = createEvent(this, "calciteInternalDatePickerMouseOut", 6); /** Date currently active.*/ this.activeDate = new Date(); //-------------------------------------------------------------------------- // // Event Listeners // //-------------------------------------------------------------------------- this.keyDownHandler = (event) => { if (event.defaultPrevented) { return; } const isRTL = this.el.dir === "rtl"; switch (event.key) { case "ArrowUp": event.preventDefault(); this.addDays(-7); break; case "ArrowRight": event.preventDefault(); this.addDays(isRTL ? -1 : 1); break; case "ArrowDown": event.preventDefault(); this.addDays(7); break; case "ArrowLeft": event.preventDefault(); this.addDays(isRTL ? 1 : -1); break; case "PageUp": event.preventDefault(); this.addMonths(-1); break; case "PageDown": event.preventDefault(); this.addMonths(1); break; case "Home": event.preventDefault(); this.activeDate.setDate(1); this.addDays(); break; case "End": event.preventDefault(); this.activeDate.setDate(new Date(this.activeDate.getFullYear(), this.activeDate.getMonth() + 1, 0).getDate()); this.addDays(); break; case "Enter": case " ": event.preventDefault(); break; case "Tab": this.activeFocus = false; } }; /** * Once user is not interacting via keyboard, * disable auto focusing of active date */ this.disableActiveFocus = () => { this.activeFocus = false; }; this.dayHover = (event) => { const target = event.target; if (target.disabled) { this.calciteInternalDatePickerMouseOut.emit(); } else { this.calciteInternalDatePickerHover.emit(target.value); } event.stopPropagation(); }; this.daySelect = (event) => { const target = event.target; this.calciteDatePickerSelect.emit(target.value); }; } mouseoutHandler() { this.calciteInternalDatePickerMouseOut.emit(); } //-------------------------------------------------------------------------- // // Lifecycle // //-------------------------------------------------------------------------- render() { const month = this.activeDate.getMonth(); const year = this.activeDate.getFullYear(); const startOfWeek = this.localeData.weekStart % 7; const { abbreviated, short, narrow } = this.localeData.days; const weekDays = this.scale === "s" ? narrow || short || abbreviated : short || abbreviated || narrow; const adjustedWeekDays = [...weekDays.slice(startOfWeek, 7), ...weekDays.slice(0, startOfWeek)]; const curMonDays = this.getCurrentMonthDays(month, year); const prevMonDays = this.getPrevMonthdays(month, year, startOfWeek); const nextMonDays = this.getNextMonthDays(month, year, startOfWeek); const days = [ ...prevMonDays.map((day) => { const date = new Date(year, month - 1, day); return this.renderDateDay(false, day, date); }), ...curMonDays.map((day) => { const date = new Date(year, month, day); const active = sameDate(date, this.activeDate); return this.renderDateDay(active, day, date, true, true); }), ...nextMonDays.map((day) => { const date = new Date(year, month + 1, day); return this.renderDateDay(false, day, date); }) ]; const weeks = []; for (let i = 0; i < days.length; i += 7) { weeks.push(days.slice(i, i + 7)); } return (h(Host, { onFocusOut: this.disableActiveFocus, onKeyDown: this.keyDownHandler }, h("div", { class: "calender", role: "grid" }, h("div", { class: "week-headers", role: "row" }, adjustedWeekDays.map((weekday) => (h("span", { class: "week-header", role: "columnheader" }, weekday)))), weeks.map((days) => (h("div", { class: "week-days", role: "row" }, days)))))); } //-------------------------------------------------------------------------- // // Private Methods // //-------------------------------------------------------------------------- /** * Add n months to the current month * * @param step */ addMonths(step) { const nextDate = new Date(this.activeDate); nextDate.setMonth(this.activeDate.getMonth() + step); this.calciteDatePickerActiveDateChange.emit(dateFromRange(nextDate, this.min, this.max)); this.activeFocus = true; } /** * Add n days to the current date * * @param step */ addDays(step = 0) { const nextDate = new Date(this.activeDate); nextDate.setDate(this.activeDate.getDate() + step); this.calciteDatePickerActiveDateChange.emit(dateFromRange(nextDate, this.min, this.max)); this.activeFocus = true; } /** * Get dates for last days of the previous month * * @param month * @param year * @param startOfWeek */ getPrevMonthdays(month, year, startOfWeek) { const lastDate = new Date(year, month, 0); const date = lastDate.getDate(); const day = lastDate.getDay(); const days = []; if (day - 6 === startOfWeek) { return days; } for (let i = lastDate.getDay() - startOfWeek; i >= 0; i--) { days.push(date - i); } return days; } /** * Get dates for the current month * * @param month * @param year */ getCurrentMonthDays(month, year) { const num = new Date(year, month + 1, 0).getDate(); const days = []; for (let i = 0; i < num; i++) { days.push(i + 1); } return days; } /** * Get dates for first days of the next month * * @param month * @param year * @param startOfWeek */ getNextMonthDays(month, year, startOfWeek) { const endDay = new Date(year, month + 1, 0).getDay(); const days = []; if (endDay === (startOfWeek + 6) % 7) { return days; } for (let i = 0; i < (6 - (endDay - startOfWeek)) % 7; i++) { days.push(i + 1); } return days; } /** * Determine if the date is in between the start and end dates * * @param date */ betweenSelectedRange(date) { return !!(this.startDate && this.endDate && date > this.startDate && date < this.endDate && !this.isRangeHover(date)); } /** * Determine if the date should be in selected state * * @param date */ isSelected(date) { return !!(sameDate(date, this.selectedDate) || (this.startDate && sameDate(date, this.startDate)) || (this.endDate && sameDate(date, this.endDate))); } /** * Determine if the date is the start of the date range * * @param date */ isStartOfRange(date) { return !!(this.startDate && !sameDate(this.startDate, this.endDate) && sameDate(this.startDate, date) && !this.isEndOfRange(date)); } isEndOfRange(date) { return !!((this.endDate && !sameDate(this.startDate, this.endDate) && sameDate(this.endDate, date)) || (!this.endDate && this.hoverRange && sameDate(this.startDate, this.hoverRange.end) && sameDate(date, this.hoverRange.end))); } /** * Render calcite-date-picker-day * * @param active * @param day * @param date * @param currentMonth * @param ref */ renderDateDay(active, day, date, currentMonth, ref) { var _a; const isFocusedOnStart = this.isFocusedOnStart(); const isHoverInRange = this.isHoverInRange() || (!this.endDate && this.hoverRange && sameDate((_a = this.hoverRange) === null || _a === void 0 ? void 0 : _a.end, this.startDate)); return (h("calcite-date-picker-day", { active: active, class: { "hover--inside-range": this.startDate && isHoverInRange, "hover--outside-range": this.startDate && !isHoverInRange, "focused--start": isFocusedOnStart, "focused--end": !isFocusedOnStart }, currentMonth: currentMonth, day: day, disabled: !inRange(date, this.min, this.max), endOfRange: this.isEndOfRange(date), highlighted: this.betweenSelectedRange(date), key: date.toDateString(), onCalciteDaySelect: this.daySelect, onCalciteInternalDayHover: this.dayHover, range: !!this.startDate && !!this.endDate && !sameDate(this.startDate, this.endDate), rangeHover: this.isRangeHover(date), ref: (el) => { // when moving via keyboard, focus must be updated on active date if (ref && active && this.activeFocus) { el === null || el === void 0 ? void 0 : el.focus(); } }, scale: this.scale, selected: this.isSelected(date), startOfRange: this.isStartOfRange(date), value: date })); } isFocusedOnStart() { var _a; return ((_a = this.hoverRange) === null || _a === void 0 ? void 0 : _a.focused) === "start"; } isHoverInRange() { if (!this.hoverRange) { return false; } const { start, end } = this.hoverRange; return !!((!this.isFocusedOnStart() && this.startDate && (!this.endDate || end < this.endDate)) || (this.isFocusedOnStart() && this.startDate && start > this.startDate)); } isRangeHover(date) { if (!this.hoverRange) { return false; } const { start, end } = this.hoverRange; const isStart = this.isFocusedOnStart(); const insideRange = this.isHoverInRange(); const cond1 = insideRange && ((!isStart && date > this.startDate && (date < end || sameDate(date, end))) || (isStart && date < this.endDate && (date > start || sameDate(date, start)))); const cond2 = !insideRange && ((!isStart && date >= this.endDate && (date < end || sameDate(date, end))) || (isStart && (date < this.startDate || (this.endDate && sameDate(date, this.startDate))) && (date > start || sameDate(date, start)))); return cond1 || cond2; } get el() { return this; } static get style() { return datePickerMonthCss; } }, [1, "calcite-date-picker-month", { "selectedDate": [16], "activeDate": [16], "startDate": [16], "endDate": [16], "min": [16], "max": [16], "scale": [513], "localeData": [16], "hoverRange": [16] }, [[1, "pointerout", "mouseoutHandler"]]]); function defineCustomElement() { if (typeof customElements === "undefined") { return; } const components = ["calcite-date-picker-month", "calcite-date-picker-day"]; components.forEach(tagName => { switch (tagName) { case "calcite-date-picker-month": if (!customElements.get(tagName)) { customElements.define(tagName, DatePickerMonth); } break; case "calcite-date-picker-day": if (!customElements.get(tagName)) { defineCustomElement$1(); } break; } }); } defineCustomElement(); export { DatePickerMonth as D, defineCustomElement as d };