UNPKG

element-plus

Version:

A Component Library for Vue 3

1 lines 8.7 kB
{"version":3,"file":"basic-year-table.mjs","names":[],"sources":["../../../../../../../packages/components/date-picker-panel/src/date-picker-com/basic-year-table.vue"],"sourcesContent":["<template>\n <table\n role=\"grid\"\n :aria-label=\"t('el.datepicker.yearTablePrompt')\"\n :class=\"ns.b()\"\n @click=\"handleYearTableClick\"\n @mousemove=\"handleMouseMove\"\n >\n <tbody ref=\"tbodyRef\">\n <tr v-for=\"(row, rowKey) in rows\" :key=\"rowKey\">\n <td\n v-for=\"(cell, cellKey) in row\"\n :key=\"`${rowKey}_${cellKey}`\"\n :ref=\"(el) => cell.isSelected && (currentCellRef = el as HTMLElement)\"\n class=\"available\"\n :class=\"getCellKls(cell)\"\n :aria-selected=\"cell.isSelected\"\n :aria-label=\"String(cell.text)\"\n :tabindex=\"cell.isSelected ? 0 : -1\"\n @keydown.space.prevent.stop=\"handleYearTableClick\"\n @keydown.enter.prevent.stop=\"handleYearTableClick\"\n >\n <el-date-picker-cell :cell=\"cell\" />\n </td>\n </tr>\n </tbody>\n </table>\n</template>\n\n<script lang=\"ts\" setup>\nimport { computed, nextTick, ref, watch } from 'vue'\nimport dayjs from 'dayjs'\nimport { useLocale, useNamespace } from '@element-plus/hooks'\nimport { rangeArr } from '@element-plus/components/time-picker'\nimport { castArray, hasClass } from '@element-plus/utils'\nimport { basicYearTableProps } from '../props/basic-year-table'\nimport { getValidDateOfYear } from '../utils'\nimport ElDatePickerCell from './basic-cell-render'\n\nimport type { Dayjs } from 'dayjs'\n\ntype YearCell = {\n column: number\n customClass: string | undefined\n disabled: boolean\n end: boolean\n inRange: boolean\n row: number\n selected: Dayjs | undefined\n isCurrent: boolean | undefined\n isSelected: boolean\n start: boolean\n text: number\n renderText: string | undefined\n timestamp: number | undefined\n type: 'normal' | 'today'\n date: Date | undefined\n dayjs: Dayjs | undefined\n}\n\nconst datesInYear = (year: number, lang: string) => {\n const firstDay = dayjs(String(year)).locale(lang).startOf('year')\n const lastDay = firstDay.endOf('year')\n const numOfDays = lastDay.dayOfYear()\n return rangeArr(numOfDays).map((n) => firstDay.add(n, 'day').toDate())\n}\n\nconst props = defineProps(basicYearTableProps)\nconst emit = defineEmits(['changerange', 'pick', 'select'])\n\nconst ns = useNamespace('year-table')\n\nconst { t, lang } = useLocale()\nconst tbodyRef = ref<HTMLElement>()\nconst currentCellRef = ref<HTMLElement>()\nconst startYear = computed(() => {\n return Math.floor(props.date.year() / 10) * 10\n})\n\nconst tableRows = ref<YearCell[][]>([[], [], []])\nconst lastRow = ref<number>()\nconst lastColumn = ref<number>()\nconst rows = computed(() => {\n const rows = tableRows.value\n const now = dayjs().locale(lang.value).startOf('year')\n\n for (let i = 0; i < 3; i++) {\n const row = rows[i]\n for (let j = 0; j < 4; j++) {\n if (i * 4 + j >= 10) {\n break\n }\n let cell = row[j]\n if (!cell) {\n cell = {\n row: i,\n column: j,\n type: 'normal',\n inRange: false,\n start: false,\n end: false,\n text: -1,\n disabled: false,\n isSelected: false,\n customClass: undefined,\n date: undefined,\n dayjs: undefined,\n isCurrent: undefined,\n selected: undefined,\n renderText: undefined,\n timestamp: undefined,\n }\n }\n cell.type = 'normal'\n const index = i * 4 + j + startYear.value\n const calTime = dayjs().year(index)\n\n const calEndDate =\n props.rangeState.endDate ||\n props.maxDate ||\n (props.rangeState.selecting && props.minDate) ||\n null\n\n cell.inRange =\n !!(\n props.minDate &&\n calTime.isSameOrAfter(props.minDate, 'year') &&\n calEndDate &&\n calTime.isSameOrBefore(calEndDate, 'year')\n ) ||\n !!(\n props.minDate &&\n calTime.isSameOrBefore(props.minDate, 'year') &&\n calEndDate &&\n calTime.isSameOrAfter(calEndDate, 'year')\n )\n\n if (props.minDate?.isSameOrAfter(calEndDate)) {\n cell.start = !!(calEndDate && calTime.isSame(calEndDate, 'year'))\n cell.end = !!(props.minDate && calTime.isSame(props.minDate, 'year'))\n } else {\n cell.start = !!(props.minDate && calTime.isSame(props.minDate, 'year'))\n cell.end = !!(calEndDate && calTime.isSame(calEndDate, 'year'))\n }\n\n const isToday = now.isSame(calTime)\n if (isToday) {\n cell.type = 'today'\n }\n cell.text = index\n const cellDate = calTime.toDate()\n cell.disabled = props.disabledDate?.(cellDate) || false\n cell.date = cellDate\n cell.customClass = props.cellClassName?.(cellDate)\n cell.dayjs = calTime\n cell.timestamp = calTime.valueOf()\n cell.isSelected = isSelectedCell(cell)\n row[j] = cell\n }\n }\n return rows\n})\n\nconst focus = () => {\n currentCellRef.value?.focus()\n}\n\nconst getCellKls = (cell: YearCell) => {\n const kls: Record<string, boolean> = {}\n const today = dayjs().locale(lang.value)\n const year = cell.text\n\n kls.disabled =\n props.disabled ||\n (props.disabledDate\n ? datesInYear(year, lang.value).every(props.disabledDate)\n : false)\n\n kls.today = today.year() === year\n kls.current = castArray(props.parsedValue).some((d) => d!.year() === year)\n\n if (cell.customClass) {\n kls[cell.customClass] = true\n }\n if (cell.inRange) {\n kls['in-range'] = true\n\n if (cell.start) {\n kls['start-date'] = true\n }\n\n if (cell.end) {\n kls['end-date'] = true\n }\n }\n return kls\n}\n\nconst isSelectedCell = (cell: YearCell) => {\n const year = cell.text\n return castArray(props.date).some((date) => date.year() === year)\n}\n\nconst handleYearTableClick = (event: MouseEvent | KeyboardEvent) => {\n if (props.disabled) return\n const target = (event.target as HTMLElement)?.closest(\n 'td'\n ) as HTMLTableCellElement\n if (!target || !target.textContent || hasClass(target, 'disabled')) return\n\n const column = target.cellIndex\n const row = (target.parentNode as HTMLTableRowElement).rowIndex\n const selectedYear = row * 4 + column + startYear.value\n const newDate = dayjs().year(selectedYear)\n if (props.selectionMode === 'range') {\n if (!props.rangeState.selecting) {\n emit('pick', { minDate: newDate, maxDate: null })\n emit('select', true)\n } else {\n if (props.minDate && newDate >= props.minDate) {\n emit('pick', { minDate: props.minDate, maxDate: newDate })\n } else {\n emit('pick', { minDate: newDate, maxDate: props.minDate })\n }\n emit('select', false)\n }\n } else if (props.selectionMode === 'years') {\n if (event.type === 'keydown') {\n emit('pick', castArray(props.parsedValue), false)\n return\n }\n const vaildYear = getValidDateOfYear(\n newDate.startOf('year'),\n lang.value,\n props.disabledDate\n )\n const newValue = hasClass(target, 'current')\n ? castArray(props.parsedValue).filter((d) => d?.year() !== selectedYear)\n : castArray(props.parsedValue).concat([vaildYear])\n emit('pick', newValue)\n } else {\n emit('pick', selectedYear)\n }\n}\n\nconst handleMouseMove = (event: MouseEvent) => {\n if (!props.rangeState.selecting) return\n const target = (event.target as HTMLElement)?.closest(\n 'td'\n ) as HTMLTableCellElement\n if (!target) return\n\n const row = (target.parentNode as HTMLTableRowElement).rowIndex\n const column = (target as HTMLTableCellElement).cellIndex\n\n // can not select disabled date\n if (rows.value[row][column].disabled) return\n\n // only update rangeState when mouse moves to a new cell\n // this avoids frequent Date object creation and improves performance\n if (row !== lastRow.value || column !== lastColumn.value) {\n lastRow.value = row\n lastColumn.value = column\n emit('changerange', {\n selecting: true,\n endDate: dayjs()\n .year(startYear.value)\n .add(row * 4 + column, 'year'),\n })\n }\n}\n\nwatch(\n () => props.date,\n async () => {\n if (tbodyRef.value?.contains(document.activeElement)) {\n await nextTick()\n currentCellRef.value?.focus()\n }\n }\n)\n\ndefineExpose({\n /**\n * @description focus on the current cell\n */\n focus,\n})\n</script>\n"],"mappings":""}