UNPKG

element-plus

Version:

A Component Library for Vue 3

651 lines (648 loc) 26.7 kB
import { defineComponent, useAttrs, useSlots, inject, toRef, ref, computed, nextTick, watch, openBlock, createElementBlock, normalizeClass, unref, createElementVNode, renderSlot, Fragment, renderList, toDisplayString, createCommentVNode, createVNode, withDirectives, withCtx, vShow, withKeys, createBlock, createTextVNode } from 'vue'; import dayjs from 'dayjs'; import { ElButton } from '../../../button/index.mjs'; import '../../../../directives/index.mjs'; import '../../../../hooks/index.mjs'; import { ElInput } from '../../../input/index.mjs'; import '../../../time-picker/index.mjs'; import { ElIcon } from '../../../icon/index.mjs'; import '../../../../utils/index.mjs'; import '../../../../constants/index.mjs'; import { DArrowLeft, ArrowLeft, ArrowRight, DArrowRight } from '@element-plus/icons-vue'; import '../../../../tokens/index.mjs'; import { panelDatePickProps } from '../props/panel-date-pick.mjs'; import DateTable from './basic-date-table.mjs'; import MonthTable from './basic-month-table.mjs'; import YearTable from './basic-year-table.mjs'; import _export_sfc from '../../../../_virtual/plugin-vue_export-helper.mjs'; import { useNamespace } from '../../../../hooks/use-namespace/index.mjs'; import { useLocale } from '../../../../hooks/use-locale/index.mjs'; import { TOOLTIP_INJECTION_KEY } from '../../../../tokens/tooltip.mjs'; import { isArray, isFunction } from '@vue/shared'; import { extractTimeFormat, extractDateFormat } from '../../../time-picker/src/utils.mjs'; import { EVENT_CODE } from '../../../../constants/aria.mjs'; import TimePickPanel from '../../../time-picker/src/time-picker-com/panel-time-pick.mjs'; import ClickOutside from '../../../../directives/click-outside/index.mjs'; const _hoisted_1 = ["onClick"]; const _hoisted_2 = ["aria-label"]; const _hoisted_3 = ["aria-label"]; const _hoisted_4 = ["aria-label"]; const _hoisted_5 = ["aria-label"]; const _sfc_main = /* @__PURE__ */ defineComponent({ __name: "panel-date-pick", props: panelDatePickProps, emits: ["pick", "set-picker-option", "panel-change"], setup(__props, { emit: contextEmit }) { const props = __props; const timeWithinRange = (_, __, ___) => true; const ppNs = useNamespace("picker-panel"); const dpNs = useNamespace("date-picker"); const attrs = useAttrs(); const slots = useSlots(); const { t, lang } = useLocale(); const pickerBase = inject("EP_PICKER_BASE"); const popper = inject(TOOLTIP_INJECTION_KEY); const { shortcuts, disabledDate, cellClassName, defaultTime, arrowControl } = pickerBase.props; const defaultValue = toRef(pickerBase.props, "defaultValue"); const currentViewRef = ref(); const innerDate = ref(dayjs().locale(lang.value)); const defaultTimeD = computed(() => { return dayjs(defaultTime).locale(lang.value); }); const month = computed(() => { return innerDate.value.month(); }); const year = computed(() => { return innerDate.value.year(); }); const selectableRange = ref([]); const userInputDate = ref(null); const userInputTime = ref(null); const checkDateWithinRange = (date) => { return selectableRange.value.length > 0 ? timeWithinRange(date, selectableRange.value, props.format || "HH:mm:ss") : true; }; const formatEmit = (emitDayjs) => { if (defaultTime && !visibleTime.value) { return defaultTimeD.value.year(emitDayjs.year()).month(emitDayjs.month()).date(emitDayjs.date()); } if (showTime.value) return emitDayjs.millisecond(0); return emitDayjs.startOf("day"); }; const emit = (value, ...args) => { if (!value) { contextEmit("pick", value, ...args); } else if (isArray(value)) { const dates = value.map(formatEmit); contextEmit("pick", dates, ...args); } else { contextEmit("pick", formatEmit(value), ...args); } userInputDate.value = null; userInputTime.value = null; }; const handleDatePick = (value, keepOpen) => { if (selectionMode.value === "date") { value = value; let newDate = props.parsedValue ? props.parsedValue.year(value.year()).month(value.month()).date(value.date()) : value; if (!checkDateWithinRange(newDate)) { newDate = selectableRange.value[0][0].year(value.year()).month(value.month()).date(value.date()); } innerDate.value = newDate; emit(newDate, showTime.value || keepOpen); } else if (selectionMode.value === "week") { emit(value.date); } else if (selectionMode.value === "dates") { emit(value, true); } }; const moveByMonth = (forward) => { const action = forward ? "add" : "subtract"; innerDate.value = innerDate.value[action](1, "month"); handlePanelChange("month"); }; const moveByYear = (forward) => { const currentDate = innerDate.value; const action = forward ? "add" : "subtract"; innerDate.value = currentView.value === "year" ? currentDate[action](10, "year") : currentDate[action](1, "year"); handlePanelChange("year"); }; const currentView = ref("date"); const yearLabel = computed(() => { const yearTranslation = t("el.datepicker.year"); if (currentView.value === "year") { const startYear = Math.floor(year.value / 10) * 10; if (yearTranslation) { return `${startYear} ${yearTranslation} - ${startYear + 9} ${yearTranslation}`; } return `${startYear} - ${startYear + 9}`; } return `${year.value} ${yearTranslation}`; }); const handleShortcutClick = (shortcut) => { const shortcutValue = isFunction(shortcut.value) ? shortcut.value() : shortcut.value; if (shortcutValue) { emit(dayjs(shortcutValue).locale(lang.value)); return; } if (shortcut.onClick) { shortcut.onClick({ attrs, slots, emit: contextEmit }); } }; const selectionMode = computed(() => { const { type } = props; if (["week", "month", "year", "dates"].includes(type)) return type; return "date"; }); const keyboardMode = computed(() => { return selectionMode.value === "date" ? currentView.value : selectionMode.value; }); const hasShortcuts = computed(() => !!shortcuts.length); const handleMonthPick = async (month2) => { innerDate.value = innerDate.value.startOf("month").month(month2); if (selectionMode.value === "month") { emit(innerDate.value, false); } else { currentView.value = "date"; if (["month", "year", "date", "week"].includes(selectionMode.value)) { emit(innerDate.value, true); await nextTick(); handleFocusPicker(); } } handlePanelChange("month"); }; const handleYearPick = async (year2) => { if (selectionMode.value === "year") { innerDate.value = innerDate.value.startOf("year").year(year2); emit(innerDate.value, false); } else { innerDate.value = innerDate.value.year(year2); currentView.value = "month"; if (["month", "year", "date", "week"].includes(selectionMode.value)) { emit(innerDate.value, true); await nextTick(); handleFocusPicker(); } } handlePanelChange("year"); }; const showPicker = async (view) => { currentView.value = view; await nextTick(); handleFocusPicker(); }; const showTime = computed(() => props.type === "datetime" || props.type === "datetimerange"); const footerVisible = computed(() => { return showTime.value || selectionMode.value === "dates"; }); const onConfirm = () => { if (selectionMode.value === "dates") { emit(props.parsedValue); } else { let result = props.parsedValue; if (!result) { const defaultTimeD2 = dayjs(defaultTime).locale(lang.value); const defaultValueD = getDefaultValue(); result = defaultTimeD2.year(defaultValueD.year()).month(defaultValueD.month()).date(defaultValueD.date()); } innerDate.value = result; emit(result); } }; const changeToNow = () => { const now = dayjs().locale(lang.value); const nowDate = now.toDate(); if ((!disabledDate || !disabledDate(nowDate)) && checkDateWithinRange(nowDate)) { innerDate.value = dayjs().locale(lang.value); emit(innerDate.value); } }; const timeFormat = computed(() => { return extractTimeFormat(props.format); }); const dateFormat = computed(() => { return extractDateFormat(props.format); }); const visibleTime = computed(() => { if (userInputTime.value) return userInputTime.value; if (!props.parsedValue && !defaultValue.value) return; return (props.parsedValue || innerDate.value).format(timeFormat.value); }); const visibleDate = computed(() => { if (userInputDate.value) return userInputDate.value; if (!props.parsedValue && !defaultValue.value) return; return (props.parsedValue || innerDate.value).format(dateFormat.value); }); const timePickerVisible = ref(false); const onTimePickerInputFocus = () => { timePickerVisible.value = true; }; const handleTimePickClose = () => { timePickerVisible.value = false; }; const getUnits = (date) => { return { hour: date.hour(), minute: date.minute(), second: date.second(), year: date.year(), month: date.month(), date: date.date() }; }; const handleTimePick = (value, visible, first) => { const { hour, minute, second } = getUnits(value); const newDate = props.parsedValue ? props.parsedValue.hour(hour).minute(minute).second(second) : value; innerDate.value = newDate; emit(innerDate.value, true); if (!first) { timePickerVisible.value = visible; } }; const handleVisibleTimeChange = (value) => { const newDate = dayjs(value, timeFormat.value).locale(lang.value); if (newDate.isValid() && checkDateWithinRange(newDate)) { const { year: year2, month: month2, date } = getUnits(innerDate.value); innerDate.value = newDate.year(year2).month(month2).date(date); userInputTime.value = null; timePickerVisible.value = false; emit(innerDate.value, true); } }; const handleVisibleDateChange = (value) => { const newDate = dayjs(value, dateFormat.value).locale(lang.value); if (newDate.isValid()) { if (disabledDate && disabledDate(newDate.toDate())) { return; } const { hour, minute, second } = getUnits(innerDate.value); innerDate.value = newDate.hour(hour).minute(minute).second(second); userInputDate.value = null; emit(innerDate.value, true); } }; const isValidValue = (date) => { return dayjs.isDayjs(date) && date.isValid() && (disabledDate ? !disabledDate(date.toDate()) : true); }; const formatToString = (value) => { if (selectionMode.value === "dates") { return value.map((_) => _.format(props.format)); } return value.format(props.format); }; const parseUserInput = (value) => { return dayjs(value, props.format).locale(lang.value); }; const getDefaultValue = () => { const parseDate = dayjs(defaultValue.value).locale(lang.value); if (!defaultValue.value) { const defaultTimeDValue = defaultTimeD.value; return dayjs().hour(defaultTimeDValue.hour()).minute(defaultTimeDValue.minute()).second(defaultTimeDValue.second()).locale(lang.value); } return parseDate; }; const handleFocusPicker = async () => { var _a; if (["week", "month", "year", "date"].includes(selectionMode.value)) { (_a = currentViewRef.value) == null ? void 0 : _a.focus(); if (selectionMode.value === "week") { handleKeyControl(EVENT_CODE.down); } } }; const handleKeydownTable = (event) => { const { code } = event; const validCode = [ EVENT_CODE.up, EVENT_CODE.down, EVENT_CODE.left, EVENT_CODE.right, EVENT_CODE.home, EVENT_CODE.end, EVENT_CODE.pageUp, EVENT_CODE.pageDown ]; if (validCode.includes(code)) { handleKeyControl(code); event.stopPropagation(); event.preventDefault(); } if ([EVENT_CODE.enter, EVENT_CODE.space].includes(code) && userInputDate.value === null && userInputTime.value === null) { event.preventDefault(); emit(innerDate.value, false); } }; const handleKeyControl = (code) => { var _a; const { up, down, left, right, home, end, pageUp, pageDown } = EVENT_CODE; const mapping = { year: { [up]: -4, [down]: 4, [left]: -1, [right]: 1, offset: (date, step) => date.setFullYear(date.getFullYear() + step) }, month: { [up]: -4, [down]: 4, [left]: -1, [right]: 1, offset: (date, step) => date.setMonth(date.getMonth() + step) }, week: { [up]: -1, [down]: 1, [left]: -1, [right]: 1, offset: (date, step) => date.setDate(date.getDate() + step * 7) }, date: { [up]: -7, [down]: 7, [left]: -1, [right]: 1, [home]: (date) => -date.getDay(), [end]: (date) => -date.getDay() + 6, [pageUp]: (date) => -new Date(date.getFullYear(), date.getMonth(), 0).getDate(), [pageDown]: (date) => new Date(date.getFullYear(), date.getMonth() + 1, 0).getDate(), offset: (date, step) => date.setDate(date.getDate() + step) } }; const newDate = innerDate.value.toDate(); while (Math.abs(innerDate.value.diff(newDate, "year", true)) < 1) { const map = mapping[keyboardMode.value]; if (!map) return; map.offset(newDate, isFunction(map[code]) ? map[code](newDate) : (_a = map[code]) != null ? _a : 0); if (disabledDate && disabledDate(newDate)) { break; } const result = dayjs(newDate).locale(lang.value); innerDate.value = result; contextEmit("pick", result, true); break; } }; const handlePanelChange = (mode) => { contextEmit("panel-change", innerDate.value.toDate(), mode, currentView.value); }; watch(() => selectionMode.value, (val) => { if (["month", "year"].includes(val)) { currentView.value = val; return; } currentView.value = "date"; }, { immediate: true }); watch(() => currentView.value, () => { popper == null ? void 0 : popper.updatePopper(); }); watch(() => defaultValue.value, (val) => { if (val) { innerDate.value = getDefaultValue(); } }, { immediate: true }); watch(() => props.parsedValue, (val) => { if (val) { if (selectionMode.value === "dates") return; if (Array.isArray(val)) return; innerDate.value = val; } else { innerDate.value = getDefaultValue(); } }, { immediate: true }); contextEmit("set-picker-option", ["isValidValue", isValidValue]); contextEmit("set-picker-option", ["formatToString", formatToString]); contextEmit("set-picker-option", ["parseUserInput", parseUserInput]); contextEmit("set-picker-option", ["handleFocusPicker", handleFocusPicker]); return (_ctx, _cache) => { return openBlock(), createElementBlock("div", { class: normalizeClass([ unref(ppNs).b(), unref(dpNs).b(), { "has-sidebar": _ctx.$slots.sidebar || unref(hasShortcuts), "has-time": unref(showTime) } ]) }, [ createElementVNode("div", { class: normalizeClass(unref(ppNs).e("body-wrapper")) }, [ renderSlot(_ctx.$slots, "sidebar", { class: normalizeClass(unref(ppNs).e("sidebar")) }), unref(hasShortcuts) ? (openBlock(), createElementBlock("div", { key: 0, class: normalizeClass(unref(ppNs).e("sidebar")) }, [ (openBlock(true), createElementBlock(Fragment, null, renderList(unref(shortcuts), (shortcut, key) => { return openBlock(), createElementBlock("button", { key, type: "button", class: normalizeClass(unref(ppNs).e("shortcut")), onClick: ($event) => handleShortcutClick(shortcut) }, toDisplayString(shortcut.text), 11, _hoisted_1); }), 128)) ], 2)) : createCommentVNode("v-if", true), createElementVNode("div", { class: normalizeClass(unref(ppNs).e("body")) }, [ unref(showTime) ? (openBlock(), createElementBlock("div", { key: 0, class: normalizeClass(unref(dpNs).e("time-header")) }, [ createElementVNode("span", { class: normalizeClass(unref(dpNs).e("editor-wrap")) }, [ createVNode(unref(ElInput), { placeholder: unref(t)("el.datepicker.selectDate"), "model-value": unref(visibleDate), size: "small", "validate-event": false, onInput: _cache[0] || (_cache[0] = (val) => userInputDate.value = val), onChange: handleVisibleDateChange }, null, 8, ["placeholder", "model-value"]) ], 2), withDirectives((openBlock(), createElementBlock("span", { class: normalizeClass(unref(dpNs).e("editor-wrap")) }, [ createVNode(unref(ElInput), { placeholder: unref(t)("el.datepicker.selectTime"), "model-value": unref(visibleTime), size: "small", "validate-event": false, onFocus: onTimePickerInputFocus, onInput: _cache[1] || (_cache[1] = (val) => userInputTime.value = val), onChange: handleVisibleTimeChange }, null, 8, ["placeholder", "model-value"]), createVNode(unref(TimePickPanel), { visible: timePickerVisible.value, format: unref(timeFormat), "time-arrow-control": unref(arrowControl), "parsed-value": innerDate.value, onPick: handleTimePick }, null, 8, ["visible", "format", "time-arrow-control", "parsed-value"]) ], 2)), [ [unref(ClickOutside), handleTimePickClose] ]) ], 2)) : createCommentVNode("v-if", true), withDirectives(createElementVNode("div", { class: normalizeClass([ unref(dpNs).e("header"), (currentView.value === "year" || currentView.value === "month") && unref(dpNs).e("header--bordered") ]) }, [ createElementVNode("span", { class: normalizeClass(unref(dpNs).e("prev-btn")) }, [ createElementVNode("button", { type: "button", "aria-label": unref(t)(`el.datepicker.prevYear`), class: normalizeClass(["d-arrow-left", unref(ppNs).e("icon-btn")]), onClick: _cache[2] || (_cache[2] = ($event) => moveByYear(false)) }, [ createVNode(unref(ElIcon), null, { default: withCtx(() => [ createVNode(unref(DArrowLeft)) ]), _: 1 }) ], 10, _hoisted_2), withDirectives(createElementVNode("button", { type: "button", "aria-label": unref(t)(`el.datepicker.prevMonth`), class: normalizeClass([unref(ppNs).e("icon-btn"), "arrow-left"]), onClick: _cache[3] || (_cache[3] = ($event) => moveByMonth(false)) }, [ createVNode(unref(ElIcon), null, { default: withCtx(() => [ createVNode(unref(ArrowLeft)) ]), _: 1 }) ], 10, _hoisted_3), [ [vShow, currentView.value === "date"] ]) ], 2), createElementVNode("span", { role: "button", class: normalizeClass(unref(dpNs).e("header-label")), "aria-live": "polite", tabindex: "0", onKeydown: _cache[4] || (_cache[4] = withKeys(($event) => showPicker("year"), ["enter"])), onClick: _cache[5] || (_cache[5] = ($event) => showPicker("year")) }, toDisplayString(unref(yearLabel)), 35), withDirectives(createElementVNode("span", { role: "button", "aria-live": "polite", tabindex: "0", class: normalizeClass([ unref(dpNs).e("header-label"), { active: currentView.value === "month" } ]), onKeydown: _cache[6] || (_cache[6] = withKeys(($event) => showPicker("month"), ["enter"])), onClick: _cache[7] || (_cache[7] = ($event) => showPicker("month")) }, toDisplayString(unref(t)(`el.datepicker.month${unref(month) + 1}`)), 35), [ [vShow, currentView.value === "date"] ]), createElementVNode("span", { class: normalizeClass(unref(dpNs).e("next-btn")) }, [ withDirectives(createElementVNode("button", { type: "button", "aria-label": unref(t)(`el.datepicker.nextMonth`), class: normalizeClass([unref(ppNs).e("icon-btn"), "arrow-right"]), onClick: _cache[8] || (_cache[8] = ($event) => moveByMonth(true)) }, [ createVNode(unref(ElIcon), null, { default: withCtx(() => [ createVNode(unref(ArrowRight)) ]), _: 1 }) ], 10, _hoisted_4), [ [vShow, currentView.value === "date"] ]), createElementVNode("button", { type: "button", "aria-label": unref(t)(`el.datepicker.nextYear`), class: normalizeClass([unref(ppNs).e("icon-btn"), "d-arrow-right"]), onClick: _cache[9] || (_cache[9] = ($event) => moveByYear(true)) }, [ createVNode(unref(ElIcon), null, { default: withCtx(() => [ createVNode(unref(DArrowRight)) ]), _: 1 }) ], 10, _hoisted_5) ], 2) ], 2), [ [vShow, currentView.value !== "time"] ]), createElementVNode("div", { class: normalizeClass(unref(ppNs).e("content")), onKeydown: handleKeydownTable }, [ currentView.value === "date" ? (openBlock(), createBlock(DateTable, { key: 0, ref_key: "currentViewRef", ref: currentViewRef, "selection-mode": unref(selectionMode), date: innerDate.value, "parsed-value": _ctx.parsedValue, "disabled-date": unref(disabledDate), "cell-class-name": unref(cellClassName), onPick: handleDatePick }, null, 8, ["selection-mode", "date", "parsed-value", "disabled-date", "cell-class-name"])) : createCommentVNode("v-if", true), currentView.value === "year" ? (openBlock(), createBlock(YearTable, { key: 1, ref_key: "currentViewRef", ref: currentViewRef, date: innerDate.value, "disabled-date": unref(disabledDate), "parsed-value": _ctx.parsedValue, onPick: handleYearPick }, null, 8, ["date", "disabled-date", "parsed-value"])) : createCommentVNode("v-if", true), currentView.value === "month" ? (openBlock(), createBlock(MonthTable, { key: 2, ref_key: "currentViewRef", ref: currentViewRef, date: innerDate.value, "parsed-value": _ctx.parsedValue, "disabled-date": unref(disabledDate), onPick: handleMonthPick }, null, 8, ["date", "parsed-value", "disabled-date"])) : createCommentVNode("v-if", true) ], 34) ], 2) ], 2), withDirectives(createElementVNode("div", { class: normalizeClass(unref(ppNs).e("footer")) }, [ withDirectives(createVNode(unref(ElButton), { text: "", size: "small", class: normalizeClass(unref(ppNs).e("link-btn")), onClick: changeToNow }, { default: withCtx(() => [ createTextVNode(toDisplayString(unref(t)("el.datepicker.now")), 1) ]), _: 1 }, 8, ["class"]), [ [vShow, unref(selectionMode) !== "dates"] ]), createVNode(unref(ElButton), { plain: "", size: "small", class: normalizeClass(unref(ppNs).e("link-btn")), onClick: onConfirm }, { default: withCtx(() => [ createTextVNode(toDisplayString(unref(t)("el.datepicker.confirm")), 1) ]), _: 1 }, 8, ["class"]) ], 2), [ [vShow, unref(footerVisible) && currentView.value === "date"] ]) ], 2); }; } }); var DatePickPanel = /* @__PURE__ */ _export_sfc(_sfc_main, [["__file", "/home/runner/work/element-plus/element-plus/packages/components/date-picker/src/date-picker-com/panel-date-pick.vue"]]); export { DatePickPanel as default }; //# sourceMappingURL=panel-date-pick.mjs.map