2020-09-16 14:49:47 +08:00
|
|
|
<template>
|
|
|
|
<table
|
|
|
|
:class="{
|
|
|
|
'el-calendar-table': true,
|
|
|
|
'is-range': isInRange
|
|
|
|
}"
|
|
|
|
cellspacing="0"
|
|
|
|
cellpadding="0"
|
|
|
|
>
|
|
|
|
<thead v-if="!hideHeader">
|
|
|
|
<th v-for="day in weekDays" :key="day">{{ day }}</th>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
|
|
|
<tr
|
|
|
|
v-for="(row, index) in rows"
|
|
|
|
:key="index"
|
|
|
|
:class="{
|
|
|
|
'el-calendar-table__row': true,
|
|
|
|
'el-calendar-table__row--hide-border': index === 0 && hideHeader
|
|
|
|
}"
|
|
|
|
>
|
|
|
|
<td
|
|
|
|
v-for="(cell, key) in row"
|
|
|
|
:key="key"
|
|
|
|
:class="getCellClass(cell)"
|
|
|
|
@click="pickDay(cell)"
|
|
|
|
>
|
|
|
|
<div class="el-calendar-day">
|
|
|
|
<slot
|
|
|
|
name="dateCell"
|
|
|
|
:data="getSlotData(cell)"
|
|
|
|
>
|
|
|
|
<span>{{ cell.text }}</span>
|
|
|
|
</slot>
|
|
|
|
</div>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts">
|
|
|
|
import {
|
|
|
|
computed,
|
|
|
|
defineComponent,
|
|
|
|
ref,
|
|
|
|
PropType,
|
|
|
|
} from 'vue'
|
|
|
|
import dayjs, { Dayjs } from 'dayjs'
|
|
|
|
import localeData from 'dayjs/plugin/localeData'
|
2020-10-20 11:45:44 +08:00
|
|
|
import { rangeArr } from '@element-plus/time-picker'
|
2020-09-16 14:49:47 +08:00
|
|
|
dayjs.extend(localeData)
|
|
|
|
|
|
|
|
export const getPrevMonthLastDays = (date: Dayjs, amount) => {
|
|
|
|
const lastDay = date.subtract(1, 'month').endOf('month').date()
|
|
|
|
return rangeArr(amount).map((_, index) => lastDay - (amount - index - 1))
|
|
|
|
}
|
|
|
|
|
|
|
|
export const getMonthDays = (date: Dayjs) => {
|
|
|
|
const days = date.daysInMonth()
|
|
|
|
return rangeArr(days).map((_, index) => index + 1)
|
|
|
|
}
|
|
|
|
|
|
|
|
export default defineComponent({
|
|
|
|
props: {
|
|
|
|
selectedDay: {
|
2020-12-09 12:00:43 +08:00
|
|
|
type: Object as PropType<Dayjs>,
|
2020-09-16 14:49:47 +08:00
|
|
|
},
|
|
|
|
range: {
|
|
|
|
type: Array as PropType<Array<Dayjs>>,
|
|
|
|
},
|
|
|
|
date: {
|
2020-12-09 12:00:43 +08:00
|
|
|
type: Object as PropType<Dayjs>,
|
2020-09-16 14:49:47 +08:00
|
|
|
},
|
|
|
|
hideHeader: {
|
|
|
|
type: Boolean,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
emits: ['pick'],
|
|
|
|
setup(props, ctx) {
|
|
|
|
const WEEK_DAYS = ref(dayjs().localeData().weekdaysShort())
|
|
|
|
|
|
|
|
const now = dayjs()
|
|
|
|
|
|
|
|
// todo better way to get Day.js locale object
|
|
|
|
const firstDayOfWeek = (now as any).$locale().weekStart || 0
|
|
|
|
|
|
|
|
const toNestedArr = days => {
|
|
|
|
return rangeArr(days.length / 7).map((_, index) => {
|
|
|
|
const start = index * 7
|
|
|
|
return days.slice(start, start + 7)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-12-09 18:24:47 +08:00
|
|
|
const getFormattedDate = (day, type): Dayjs => {
|
2020-09-16 14:49:47 +08:00
|
|
|
let result
|
|
|
|
if (type === 'prev') {
|
|
|
|
result = props.date.startOf('month').subtract(1, 'month').date(day)
|
|
|
|
} else if (type === 'next') {
|
|
|
|
result = props.date.startOf('month').add(1, 'month').date(day)
|
|
|
|
} else {
|
|
|
|
result = props.date.date(day)
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
const getCellClass = ({ text, type }) => {
|
|
|
|
const classes = [type]
|
|
|
|
if (type === 'current') {
|
2020-12-09 18:24:47 +08:00
|
|
|
const date_ = getFormattedDate(text, type)
|
2020-09-16 14:49:47 +08:00
|
|
|
if (date_.isSame(props.selectedDay, 'day')) {
|
|
|
|
classes.push('is-selected')
|
|
|
|
}
|
|
|
|
if (date_.isSame(now, 'day')) {
|
|
|
|
classes.push('is-today')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return classes
|
|
|
|
}
|
|
|
|
|
|
|
|
const pickDay = ({ text, type }) => {
|
2020-12-09 18:24:47 +08:00
|
|
|
const date = getFormattedDate(text, type)
|
2020-09-16 14:49:47 +08:00
|
|
|
ctx.emit('pick', date)
|
|
|
|
}
|
|
|
|
|
|
|
|
const getSlotData = ({ text, type }) => {
|
2020-12-09 18:24:47 +08:00
|
|
|
const day = getFormattedDate(text, type)
|
2020-09-16 14:49:47 +08:00
|
|
|
return {
|
|
|
|
isSelected: day.isSame(props.selectedDay),
|
|
|
|
type: `${type}-month`,
|
|
|
|
day: day.format('YYYY-MM-DD'),
|
|
|
|
date: day.toDate(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const isInRange = computed(() => {
|
|
|
|
return props.range && props.range.length
|
|
|
|
})
|
|
|
|
|
|
|
|
const rows = computed(() => {
|
|
|
|
let days = []
|
|
|
|
if (isInRange.value) {
|
|
|
|
const [start, end] = props.range
|
|
|
|
const currentMonthRange = rangeArr(
|
|
|
|
end.date() - start.date() + 1,
|
|
|
|
).map((_, index) => ({
|
|
|
|
text: start.date() + index,
|
|
|
|
type: 'current',
|
|
|
|
}))
|
|
|
|
|
|
|
|
let remaining = currentMonthRange.length % 7
|
|
|
|
remaining = remaining === 0 ? 0 : 7 - remaining
|
|
|
|
const nextMonthRange = rangeArr(remaining).map((_, index) => ({
|
|
|
|
text: index + 1,
|
|
|
|
type: 'next',
|
|
|
|
}))
|
|
|
|
days = currentMonthRange.concat(nextMonthRange)
|
|
|
|
} else {
|
|
|
|
const firstDay = props.date.startOf('month').day() || 7
|
|
|
|
const prevMonthDays = getPrevMonthLastDays(
|
|
|
|
props.date,
|
|
|
|
firstDay - firstDayOfWeek,
|
|
|
|
).map(day => ({
|
|
|
|
text: day,
|
|
|
|
type: 'prev',
|
|
|
|
}))
|
|
|
|
const currentMonthDays = getMonthDays(props.date).map(day => ({
|
|
|
|
text: day,
|
|
|
|
type: 'current',
|
|
|
|
}))
|
|
|
|
days = [...prevMonthDays, ...currentMonthDays]
|
|
|
|
const nextMonthDays = rangeArr(42 - days.length).map((_, index) => ({
|
|
|
|
text: index + 1,
|
|
|
|
type: 'next',
|
|
|
|
}))
|
|
|
|
days = days.concat(nextMonthDays)
|
|
|
|
}
|
|
|
|
return toNestedArr(days)
|
|
|
|
})
|
|
|
|
|
|
|
|
const weekDays = computed(() => {
|
|
|
|
const start = firstDayOfWeek
|
|
|
|
|
|
|
|
if (start === 0) {
|
|
|
|
return WEEK_DAYS.value
|
|
|
|
} else {
|
|
|
|
return WEEK_DAYS.value
|
|
|
|
.slice(start)
|
|
|
|
.concat(WEEK_DAYS.value.slice(0, start))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
return {
|
|
|
|
isInRange,
|
|
|
|
weekDays,
|
|
|
|
rows,
|
|
|
|
getCellClass,
|
|
|
|
pickDay,
|
|
|
|
getSlotData,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
</script>
|