mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
feat(Calendar): add switch-mode prop (#12836)
This commit is contained in:
parent
92f373c17f
commit
27e080e8aa
@ -31,6 +31,7 @@ import {
|
|||||||
calcDateNum,
|
calcDateNum,
|
||||||
compareMonth,
|
compareMonth,
|
||||||
getDayByOffset,
|
getDayByOffset,
|
||||||
|
getMonthByOffset,
|
||||||
} from './utils';
|
} from './utils';
|
||||||
|
|
||||||
// Composables
|
// Composables
|
||||||
@ -48,6 +49,7 @@ import CalendarHeader from './CalendarHeader';
|
|||||||
// Types
|
// Types
|
||||||
import type {
|
import type {
|
||||||
CalendarType,
|
CalendarType,
|
||||||
|
CalendarSwitchMode,
|
||||||
CalendarExpose,
|
CalendarExpose,
|
||||||
CalendarDayItem,
|
CalendarDayItem,
|
||||||
CalendarMonthInstance,
|
CalendarMonthInstance,
|
||||||
@ -56,6 +58,7 @@ import type {
|
|||||||
export const calendarProps = {
|
export const calendarProps = {
|
||||||
show: Boolean,
|
show: Boolean,
|
||||||
type: makeStringProp<CalendarType>('single'),
|
type: makeStringProp<CalendarType>('single'),
|
||||||
|
switchMode: makeStringProp<CalendarSwitchMode>('none'),
|
||||||
title: String,
|
title: String,
|
||||||
color: String,
|
color: String,
|
||||||
round: truthProp,
|
round: truthProp,
|
||||||
@ -84,15 +87,10 @@ export const calendarProps = {
|
|||||||
minDate: {
|
minDate: {
|
||||||
type: Date,
|
type: Date,
|
||||||
validator: isDate,
|
validator: isDate,
|
||||||
default: getToday,
|
|
||||||
},
|
},
|
||||||
maxDate: {
|
maxDate: {
|
||||||
type: Date,
|
type: Date,
|
||||||
validator: isDate,
|
validator: isDate,
|
||||||
default: () => {
|
|
||||||
const now = getToday();
|
|
||||||
return new Date(now.getFullYear(), now.getMonth() + 6, now.getDate());
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
firstDayOfWeek: {
|
firstDayOfWeek: {
|
||||||
type: numericProp,
|
type: numericProp,
|
||||||
@ -117,25 +115,44 @@ export default defineComponent({
|
|||||||
'update:show',
|
'update:show',
|
||||||
'clickSubtitle',
|
'clickSubtitle',
|
||||||
'clickDisabledDate',
|
'clickDisabledDate',
|
||||||
|
'panelChange',
|
||||||
],
|
],
|
||||||
|
|
||||||
setup(props, { emit, slots }) {
|
setup(props, { emit, slots }) {
|
||||||
|
const canSwitch = computed(() => props.switchMode !== 'none');
|
||||||
|
|
||||||
|
const minDate = computed(() => {
|
||||||
|
if (!props.minDate && !canSwitch.value) {
|
||||||
|
return getToday();
|
||||||
|
}
|
||||||
|
|
||||||
|
return props.minDate;
|
||||||
|
});
|
||||||
|
|
||||||
|
const maxDate = computed(() => {
|
||||||
|
if (!props.maxDate && !canSwitch.value) {
|
||||||
|
return getMonthByOffset(getToday(), 6);
|
||||||
|
}
|
||||||
|
|
||||||
|
return props.maxDate;
|
||||||
|
});
|
||||||
|
|
||||||
const limitDateRange = (
|
const limitDateRange = (
|
||||||
date: Date,
|
date: Date,
|
||||||
minDate = props.minDate,
|
min = minDate.value,
|
||||||
maxDate = props.maxDate,
|
max = maxDate.value,
|
||||||
) => {
|
) => {
|
||||||
if (compareDay(date, minDate) === -1) {
|
if (min && compareDay(date, min) === -1) {
|
||||||
return minDate;
|
return min;
|
||||||
}
|
}
|
||||||
if (compareDay(date, maxDate) === 1) {
|
if (max && compareDay(date, max) === 1) {
|
||||||
return maxDate;
|
return max;
|
||||||
}
|
}
|
||||||
return date;
|
return date;
|
||||||
};
|
};
|
||||||
|
|
||||||
const getInitialDate = (defaultDate = props.defaultDate) => {
|
const getInitialDate = (defaultDate = props.defaultDate) => {
|
||||||
const { type, minDate, maxDate, allowSameDay } = props;
|
const { type, allowSameDay } = props;
|
||||||
|
|
||||||
if (defaultDate === null) {
|
if (defaultDate === null) {
|
||||||
return defaultDate;
|
return defaultDate;
|
||||||
@ -147,15 +164,21 @@ export default defineComponent({
|
|||||||
if (!Array.isArray(defaultDate)) {
|
if (!Array.isArray(defaultDate)) {
|
||||||
defaultDate = [];
|
defaultDate = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const min = minDate.value;
|
||||||
|
const max = maxDate.value;
|
||||||
|
|
||||||
const start = limitDateRange(
|
const start = limitDateRange(
|
||||||
defaultDate[0] || now,
|
defaultDate[0] || now,
|
||||||
minDate,
|
min,
|
||||||
allowSameDay ? maxDate : getPrevDay(maxDate),
|
max ? (allowSameDay ? max : getPrevDay(max)) : undefined,
|
||||||
);
|
);
|
||||||
|
|
||||||
const end = limitDateRange(
|
const end = limitDateRange(
|
||||||
defaultDate[1] || now,
|
defaultDate[1] || (allowSameDay ? now : getNextDay(now)),
|
||||||
allowSameDay ? minDate : getNextDay(minDate),
|
min ? (allowSameDay ? min : getNextDay(min)) : undefined,
|
||||||
);
|
);
|
||||||
|
|
||||||
return [start, end];
|
return [start, end];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -172,16 +195,24 @@ export default defineComponent({
|
|||||||
return limitDateRange(defaultDate);
|
return limitDateRange(defaultDate);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const getInitialPanelDate = () => {
|
||||||
|
const date = Array.isArray(currentDate.value)
|
||||||
|
? currentDate.value[0]
|
||||||
|
: currentDate.value;
|
||||||
|
|
||||||
|
return date ? date : limitDateRange(getToday());
|
||||||
|
};
|
||||||
|
|
||||||
let bodyHeight: number;
|
let bodyHeight: number;
|
||||||
|
|
||||||
const bodyRef = ref<HTMLElement>();
|
const bodyRef = ref<HTMLElement>();
|
||||||
|
|
||||||
const subtitle = ref<{ textFn: () => string; date?: Date }>({
|
|
||||||
textFn: () => '',
|
|
||||||
date: undefined,
|
|
||||||
});
|
|
||||||
const currentDate = ref(getInitialDate());
|
const currentDate = ref(getInitialDate());
|
||||||
|
|
||||||
|
const currentPanelDate = ref<Date>(getInitialPanelDate());
|
||||||
|
|
||||||
|
const currentMonthRef = ref<CalendarMonthInstance>();
|
||||||
|
|
||||||
const [monthRefs, setMonthRefs] = useRefs<CalendarMonthInstance>();
|
const [monthRefs, setMonthRefs] = useRefs<CalendarMonthInstance>();
|
||||||
|
|
||||||
const dayOffset = computed(() =>
|
const dayOffset = computed(() =>
|
||||||
@ -190,14 +221,19 @@ export default defineComponent({
|
|||||||
|
|
||||||
const months = computed(() => {
|
const months = computed(() => {
|
||||||
const months: Date[] = [];
|
const months: Date[] = [];
|
||||||
const cursor = new Date(props.minDate);
|
|
||||||
|
if (!minDate.value || !maxDate.value) {
|
||||||
|
return months;
|
||||||
|
}
|
||||||
|
|
||||||
|
const cursor = new Date(minDate.value);
|
||||||
|
|
||||||
cursor.setDate(1);
|
cursor.setDate(1);
|
||||||
|
|
||||||
do {
|
do {
|
||||||
months.push(new Date(cursor));
|
months.push(new Date(cursor));
|
||||||
cursor.setMonth(cursor.getMonth() + 1);
|
cursor.setMonth(cursor.getMonth() + 1);
|
||||||
} while (compareMonth(cursor, props.maxDate) !== 1);
|
} while (compareMonth(cursor, maxDate.value) !== 1);
|
||||||
|
|
||||||
return months;
|
return months;
|
||||||
});
|
});
|
||||||
@ -271,28 +307,29 @@ export default defineComponent({
|
|||||||
|
|
||||||
/* istanbul ignore else */
|
/* istanbul ignore else */
|
||||||
if (currentMonth) {
|
if (currentMonth) {
|
||||||
subtitle.value = {
|
currentMonthRef.value = currentMonth;
|
||||||
textFn: currentMonth.getTitle,
|
|
||||||
date: currentMonth.date,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const scrollToDate = (targetDate: Date) => {
|
const scrollToDate = (targetDate: Date) => {
|
||||||
raf(() => {
|
if (canSwitch.value) {
|
||||||
months.value.some((month, index) => {
|
currentPanelDate.value = targetDate;
|
||||||
if (compareMonth(month, targetDate) === 0) {
|
} else {
|
||||||
if (bodyRef.value) {
|
raf(() => {
|
||||||
monthRefs.value[index].scrollToDate(bodyRef.value, targetDate);
|
months.value.some((month, index) => {
|
||||||
|
if (compareMonth(month, targetDate) === 0) {
|
||||||
|
if (bodyRef.value) {
|
||||||
|
monthRefs.value[index].scrollToDate(bodyRef.value, targetDate);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
onScroll();
|
||||||
});
|
});
|
||||||
|
}
|
||||||
onScroll();
|
|
||||||
});
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const scrollToCurrentDate = () => {
|
const scrollToCurrentDate = () => {
|
||||||
@ -308,7 +345,7 @@ export default defineComponent({
|
|||||||
if (isDate(targetDate)) {
|
if (isDate(targetDate)) {
|
||||||
scrollToDate(targetDate);
|
scrollToDate(targetDate);
|
||||||
}
|
}
|
||||||
} else {
|
} else if (!canSwitch.value) {
|
||||||
raf(onScroll);
|
raf(onScroll);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -318,11 +355,14 @@ export default defineComponent({
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
raf(() => {
|
if (!canSwitch.value) {
|
||||||
// add Math.floor to avoid decimal height issues
|
raf(() => {
|
||||||
// https://github.com/vant-ui/vant/issues/5640
|
// add Math.floor to avoid decimal height issues
|
||||||
bodyHeight = Math.floor(useRect(bodyRef).height);
|
// https://github.com/vant-ui/vant/issues/5640
|
||||||
});
|
bodyHeight = Math.floor(useRect(bodyRef).height);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
scrollToCurrentDate();
|
scrollToCurrentDate();
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -345,6 +385,11 @@ export default defineComponent({
|
|||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const onPanelChange = (date: Date) => {
|
||||||
|
currentPanelDate.value = date;
|
||||||
|
emit('panelChange', { date });
|
||||||
|
};
|
||||||
|
|
||||||
const onConfirm = () =>
|
const onConfirm = () =>
|
||||||
emit('confirm', currentDate.value ?? cloneDates(currentDate.value!));
|
emit('confirm', currentDate.value ?? cloneDates(currentDate.value!));
|
||||||
|
|
||||||
@ -469,20 +514,20 @@ export default defineComponent({
|
|||||||
return (
|
return (
|
||||||
<CalendarMonth
|
<CalendarMonth
|
||||||
v-slots={pick(slots, ['top-info', 'bottom-info', 'month-title'])}
|
v-slots={pick(slots, ['top-info', 'bottom-info', 'month-title'])}
|
||||||
ref={setMonthRefs(index)}
|
ref={canSwitch.value ? currentMonthRef : setMonthRefs(index)}
|
||||||
date={date}
|
date={date}
|
||||||
currentDate={currentDate.value}
|
currentDate={currentDate.value}
|
||||||
showMonthTitle={showMonthTitle}
|
showMonthTitle={showMonthTitle}
|
||||||
firstDayOfWeek={dayOffset.value}
|
firstDayOfWeek={dayOffset.value}
|
||||||
|
lazyRender={canSwitch.value ? false : props.lazyRender}
|
||||||
|
maxDate={maxDate.value}
|
||||||
|
minDate={minDate.value}
|
||||||
{...pick(props, [
|
{...pick(props, [
|
||||||
'type',
|
'type',
|
||||||
'color',
|
'color',
|
||||||
'minDate',
|
|
||||||
'maxDate',
|
|
||||||
'showMark',
|
'showMark',
|
||||||
'formatter',
|
'formatter',
|
||||||
'rowHeight',
|
'rowHeight',
|
||||||
'lazyRender',
|
|
||||||
'showSubtitle',
|
'showSubtitle',
|
||||||
'allowSameDay',
|
'allowSameDay',
|
||||||
])}
|
])}
|
||||||
@ -529,33 +574,45 @@ export default defineComponent({
|
|||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
const renderCalendar = () => {
|
const renderCalendar = () => (
|
||||||
const subTitle = subtitle.value.textFn();
|
<div class={bem()}>
|
||||||
return (
|
<CalendarHeader
|
||||||
<div class={bem()}>
|
v-slots={pick(slots, [
|
||||||
<CalendarHeader
|
'title',
|
||||||
v-slots={pick(slots, ['title', 'subtitle'])}
|
'subtitle',
|
||||||
date={subtitle.value.date}
|
'prev-month',
|
||||||
title={props.title}
|
'prev-year',
|
||||||
subtitle={subTitle}
|
'next-month',
|
||||||
showTitle={props.showTitle}
|
'next-year',
|
||||||
showSubtitle={props.showSubtitle}
|
])}
|
||||||
firstDayOfWeek={dayOffset.value}
|
date={currentMonthRef.value?.date}
|
||||||
onClickSubtitle={(event: MouseEvent) =>
|
maxDate={maxDate.value}
|
||||||
emit('clickSubtitle', event)
|
minDate={minDate.value}
|
||||||
}
|
title={props.title}
|
||||||
/>
|
subtitle={currentMonthRef.value?.getTitle()}
|
||||||
<div ref={bodyRef} class={bem('body')} onScroll={onScroll}>
|
showTitle={props.showTitle}
|
||||||
{months.value.map(renderMonth)}
|
showSubtitle={props.showSubtitle}
|
||||||
</div>
|
switchMode={props.switchMode}
|
||||||
{renderFooter()}
|
firstDayOfWeek={dayOffset.value}
|
||||||
|
onClickSubtitle={(event: MouseEvent) => emit('clickSubtitle', event)}
|
||||||
|
onPanelChange={onPanelChange}
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
ref={bodyRef}
|
||||||
|
class={bem('body')}
|
||||||
|
onScroll={canSwitch.value ? undefined : onScroll}
|
||||||
|
>
|
||||||
|
{canSwitch.value
|
||||||
|
? renderMonth(currentPanelDate.value, 0)
|
||||||
|
: months.value.map(renderMonth)}
|
||||||
</div>
|
</div>
|
||||||
);
|
{renderFooter()}
|
||||||
};
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
watch(() => props.show, init);
|
watch(() => props.show, init);
|
||||||
watch(
|
watch(
|
||||||
() => [props.type, props.minDate, props.maxDate],
|
() => [props.type, props.minDate, props.maxDate, props.switchMode],
|
||||||
() => reset(getInitialDate(currentDate.value)),
|
() => reset(getInitialDate(currentDate.value)),
|
||||||
);
|
);
|
||||||
watch(
|
watch(
|
||||||
|
@ -1,6 +1,21 @@
|
|||||||
import { defineComponent } from 'vue';
|
import { computed, defineComponent } from 'vue';
|
||||||
import { createNamespace } from '../utils';
|
|
||||||
import { t, bem } from './utils';
|
// Utils
|
||||||
|
import { createNamespace, HAPTICS_FEEDBACK, makeStringProp } from '../utils';
|
||||||
|
import {
|
||||||
|
t,
|
||||||
|
bem,
|
||||||
|
getPrevMonth,
|
||||||
|
getPrevYear,
|
||||||
|
getNextMonth,
|
||||||
|
getNextYear,
|
||||||
|
} from './utils';
|
||||||
|
|
||||||
|
// Components
|
||||||
|
import { Icon } from '../icon';
|
||||||
|
|
||||||
|
// Types
|
||||||
|
import type { CalendarSwitchMode } from './types';
|
||||||
|
|
||||||
const [name] = createNamespace('calendar-header');
|
const [name] = createNamespace('calendar-header');
|
||||||
|
|
||||||
@ -9,16 +24,39 @@ export default defineComponent({
|
|||||||
|
|
||||||
props: {
|
props: {
|
||||||
date: Date,
|
date: Date,
|
||||||
|
minDate: Date,
|
||||||
|
maxDate: Date,
|
||||||
title: String,
|
title: String,
|
||||||
subtitle: String,
|
subtitle: String,
|
||||||
showTitle: Boolean,
|
showTitle: Boolean,
|
||||||
showSubtitle: Boolean,
|
showSubtitle: Boolean,
|
||||||
firstDayOfWeek: Number,
|
firstDayOfWeek: Number,
|
||||||
|
switchMode: makeStringProp<CalendarSwitchMode>('none'),
|
||||||
},
|
},
|
||||||
|
|
||||||
emits: ['clickSubtitle'],
|
emits: ['clickSubtitle', 'panelChange'],
|
||||||
|
|
||||||
setup(props, { slots, emit }) {
|
setup(props, { slots, emit }) {
|
||||||
|
const prevMonthDisabled = computed(() => {
|
||||||
|
const prevMonth = getPrevMonth(props.date!);
|
||||||
|
return props.minDate && prevMonth < props.minDate;
|
||||||
|
});
|
||||||
|
|
||||||
|
const prevYearDisabled = computed(() => {
|
||||||
|
const prevYear = getPrevYear(props.date!);
|
||||||
|
return props.minDate && prevYear < props.minDate;
|
||||||
|
});
|
||||||
|
|
||||||
|
const nextMonthDisabled = computed(() => {
|
||||||
|
const nextMonth = getNextMonth(props.date!);
|
||||||
|
return props.maxDate && nextMonth > props.maxDate;
|
||||||
|
});
|
||||||
|
|
||||||
|
const nextYearDisabled = computed(() => {
|
||||||
|
const nextYear = getNextYear(props.date!);
|
||||||
|
return props.maxDate && nextYear > props.maxDate;
|
||||||
|
});
|
||||||
|
|
||||||
const renderTitle = () => {
|
const renderTitle = () => {
|
||||||
if (props.showTitle) {
|
if (props.showTitle) {
|
||||||
const text = props.title || t('title');
|
const text = props.title || t('title');
|
||||||
@ -29,6 +67,60 @@ export default defineComponent({
|
|||||||
|
|
||||||
const onClickSubtitle = (event: MouseEvent) => emit('clickSubtitle', event);
|
const onClickSubtitle = (event: MouseEvent) => emit('clickSubtitle', event);
|
||||||
|
|
||||||
|
const onPanelChange = (date: Date) => emit('panelChange', date);
|
||||||
|
|
||||||
|
const renderAction = (isNext?: boolean) => {
|
||||||
|
const showYearAction = props.switchMode === 'year-month';
|
||||||
|
const monthSlot = slots[isNext ? 'next-month' : 'prev-month'];
|
||||||
|
const yearSlot = slots[isNext ? 'next-year' : 'prev-year'];
|
||||||
|
const monthDisabled = isNext
|
||||||
|
? nextMonthDisabled.value
|
||||||
|
: prevMonthDisabled.value;
|
||||||
|
const yearDisabled = isNext
|
||||||
|
? nextYearDisabled.value
|
||||||
|
: prevYearDisabled.value;
|
||||||
|
const monthIconName = isNext ? 'arrow' : 'arrow-left';
|
||||||
|
const yearIconName = isNext ? 'arrow-double-right' : 'arrow-double-left';
|
||||||
|
|
||||||
|
const onMonthChange = () =>
|
||||||
|
onPanelChange((isNext ? getNextMonth : getPrevMonth)(props.date!));
|
||||||
|
const onYearChange = () =>
|
||||||
|
onPanelChange((isNext ? getNextYear : getPrevYear)(props.date!));
|
||||||
|
|
||||||
|
const MonthAction = (
|
||||||
|
<view
|
||||||
|
class={bem('header-action', { disabled: monthDisabled })}
|
||||||
|
onClick={monthDisabled ? undefined : onMonthChange}
|
||||||
|
>
|
||||||
|
{monthSlot ? (
|
||||||
|
monthSlot({ disabled: monthDisabled })
|
||||||
|
) : (
|
||||||
|
<Icon
|
||||||
|
class={{ [HAPTICS_FEEDBACK]: !monthDisabled }}
|
||||||
|
name={monthIconName}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</view>
|
||||||
|
);
|
||||||
|
const YearAction = showYearAction && (
|
||||||
|
<view
|
||||||
|
class={bem('header-action', { disabled: yearDisabled })}
|
||||||
|
onClick={yearDisabled ? undefined : onYearChange}
|
||||||
|
>
|
||||||
|
{yearSlot ? (
|
||||||
|
yearSlot({ disabled: yearDisabled })
|
||||||
|
) : (
|
||||||
|
<Icon
|
||||||
|
class={{ [HAPTICS_FEEDBACK]: !yearDisabled }}
|
||||||
|
name={yearIconName}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</view>
|
||||||
|
);
|
||||||
|
|
||||||
|
return isNext ? [MonthAction, YearAction] : [YearAction, MonthAction];
|
||||||
|
};
|
||||||
|
|
||||||
const renderSubtitle = () => {
|
const renderSubtitle = () => {
|
||||||
if (props.showSubtitle) {
|
if (props.showSubtitle) {
|
||||||
const title = slots.subtitle
|
const title = slots.subtitle
|
||||||
@ -37,9 +129,20 @@ export default defineComponent({
|
|||||||
text: props.subtitle,
|
text: props.subtitle,
|
||||||
})
|
})
|
||||||
: props.subtitle;
|
: props.subtitle;
|
||||||
|
const canSwitch = props.switchMode !== 'none';
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div class={bem('header-subtitle')} onClick={onClickSubtitle}>
|
<div
|
||||||
{title}
|
class={bem('header-subtitle', { 'with-swicth': canSwitch })}
|
||||||
|
onClick={onClickSubtitle}
|
||||||
|
>
|
||||||
|
{canSwitch
|
||||||
|
? [
|
||||||
|
renderAction(),
|
||||||
|
<div class={bem('header-subtitle-text')}>{title}</div>,
|
||||||
|
renderAction(true),
|
||||||
|
]
|
||||||
|
: title}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -42,8 +42,8 @@ const calendarMonthProps = {
|
|||||||
date: makeRequiredProp(Date),
|
date: makeRequiredProp(Date),
|
||||||
type: String as PropType<CalendarType>,
|
type: String as PropType<CalendarType>,
|
||||||
color: String,
|
color: String,
|
||||||
minDate: makeRequiredProp(Date),
|
minDate: Date,
|
||||||
maxDate: makeRequiredProp(Date),
|
maxDate: Date,
|
||||||
showMark: Boolean,
|
showMark: Boolean,
|
||||||
rowHeight: numericProp,
|
rowHeight: numericProp,
|
||||||
formatter: Function as PropType<(item: CalendarDayItem) => CalendarDayItem>,
|
formatter: Function as PropType<(item: CalendarDayItem) => CalendarDayItem>,
|
||||||
@ -73,11 +73,14 @@ export default defineComponent({
|
|||||||
const title = computed(() => formatMonthTitle(props.date));
|
const title = computed(() => formatMonthTitle(props.date));
|
||||||
const rowHeight = computed(() => addUnit(props.rowHeight));
|
const rowHeight = computed(() => addUnit(props.rowHeight));
|
||||||
const offset = computed(() => {
|
const offset = computed(() => {
|
||||||
const realDay = props.date.getDay();
|
const date = props.date.getDate();
|
||||||
|
const day = props.date.getDay();
|
||||||
|
const realDay = (day - (date % 7) + 8) % 7;
|
||||||
|
|
||||||
if (props.firstDayOfWeek) {
|
if (props.firstDayOfWeek) {
|
||||||
return (realDay + 7 - props.firstDayOfWeek) % 7;
|
return (realDay + 7 - props.firstDayOfWeek) % 7;
|
||||||
}
|
}
|
||||||
|
|
||||||
return realDay;
|
return realDay;
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -150,7 +153,10 @@ export default defineComponent({
|
|||||||
const getDayType = (day: Date): CalendarDayType => {
|
const getDayType = (day: Date): CalendarDayType => {
|
||||||
const { type, minDate, maxDate, currentDate } = props;
|
const { type, minDate, maxDate, currentDate } = props;
|
||||||
|
|
||||||
if (compareDay(day, minDate) < 0 || compareDay(day, maxDate) > 0) {
|
if (
|
||||||
|
(minDate && compareDay(day, minDate) < 0) ||
|
||||||
|
(maxDate && compareDay(day, maxDate) > 0)
|
||||||
|
) {
|
||||||
return 'disabled';
|
return 'disabled';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -18,6 +18,14 @@ app.use(Calendar);
|
|||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
|
### Select Switch Mode
|
||||||
|
|
||||||
|
By default, all months will be displayed without showing the switch button. When there are too many months, it may affect the page's interactivity performance. You can display the year and month switching buttons by setting the `switch-mode` prop.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-calendar v-model:show="show" switch-mode="year-month" />
|
||||||
|
```
|
||||||
|
|
||||||
### Select Single Date
|
### Select Single Date
|
||||||
|
|
||||||
The `confirm` event will be emitted after the date selection is completed.
|
The `confirm` event will be emitted after the date selection is completed.
|
||||||
@ -250,10 +258,11 @@ Set `poppable` to `false`, the calendar will be displayed directly on the page i
|
|||||||
| Attribute | Description | Type | Default |
|
| Attribute | Description | Type | Default |
|
||||||
| --- | --- | --- | --- |
|
| --- | --- | --- | --- |
|
||||||
| type | Type, can be set to `range` `multiple` | _string_ | `single` |
|
| type | Type, can be set to `range` `multiple` | _string_ | `single` |
|
||||||
|
| switch-mode | Switch mode:<br>`none` Display all months in a tiled format without switch buttons <br>`month` Support switching by month, displaying buttons for previous month/next month <br>`year-month` Support switching by year, as well as by month, displaying buttons for previous year/next year and previous month/next month | _string_ | `none` |
|
||||||
| title | Title of calendar | _string_ | `Calendar` |
|
| title | Title of calendar | _string_ | `Calendar` |
|
||||||
| color | Color for the bottom button and selected date | _string_ | `#1989fa` |
|
| color | Color for the bottom button and selected date | _string_ | `#1989fa` |
|
||||||
| min-date | Min date | _Date_ | Today |
|
| min-date | Min date | _Date_ | When `switch-mode` is set to `none`, the default value is the today |
|
||||||
| max-date | Max date | _Date_ | Six months after the today |
|
| max-date | Max date | _Date_ | When `switch-mode` is set to `none`, the default value is six months after the today |
|
||||||
| default-date | Default selected date | _Date \| Date[] \| null_ | Today |
|
| default-date | Default selected date | _Date \| Date[] \| null_ | Today |
|
||||||
| row-height | Row height | _number \| string_ | `64` |
|
| row-height | Row height | _number \| string_ | `64` |
|
||||||
| formatter | Day formatter | _(day: Day) => Day_ | - |
|
| formatter | Day formatter | _(day: Day) => Day_ | - |
|
||||||
@ -329,6 +338,7 @@ Following props are supported when the type is multiple
|
|||||||
| over-range | Emitted when exceeded max range | - |
|
| over-range | Emitted when exceeded max range | - |
|
||||||
| click-subtitle | Emitted when clicking the subtitle | _event: MouseEvent_ |
|
| click-subtitle | Emitted when clicking the subtitle | _event: MouseEvent_ |
|
||||||
| click-disabled-date `v4.7.0` | Emitted when clicking disabled date | _value: Date \| Date[]_ |
|
| click-disabled-date `v4.7.0` | Emitted when clicking disabled date | _value: Date \| Date[]_ |
|
||||||
|
| panel-change | Emitted when switching calendar panel | _{ date: Date }_ |
|
||||||
|
|
||||||
### Slots
|
### Slots
|
||||||
|
|
||||||
@ -341,6 +351,10 @@ Following props are supported when the type is multiple
|
|||||||
| confirm-text | Custom confirm text | _{ disabled: boolean }_ |
|
| confirm-text | Custom confirm text | _{ disabled: boolean }_ |
|
||||||
| top-info | Custom top info of day | _day: Day_ |
|
| top-info | Custom top info of day | _day: Day_ |
|
||||||
| bottom-info | Custom bottom info of day | _day: Day_ |
|
| bottom-info | Custom bottom info of day | _day: Day_ |
|
||||||
|
| prev-month | Custom previous month button | _{ disabled: boolean }_ |
|
||||||
|
| prev-year | Custom previous year button | _{ disabled: boolean }_ |
|
||||||
|
| next-month | Custom next month button | _{ disabled: boolean }_ |
|
||||||
|
| next-year | Custom next year button | _{ disabled: boolean }_ |
|
||||||
|
|
||||||
### Methods
|
### Methods
|
||||||
|
|
||||||
@ -358,6 +372,7 @@ The component exports the following type definitions:
|
|||||||
|
|
||||||
```ts
|
```ts
|
||||||
import type {
|
import type {
|
||||||
|
CalendarSwitchMode,
|
||||||
CalendarType,
|
CalendarType,
|
||||||
CalendarProps,
|
CalendarProps,
|
||||||
CalendarDayItem,
|
CalendarDayItem,
|
||||||
@ -391,6 +406,9 @@ The component provides the following CSS variables, which can be used to customi
|
|||||||
| --van-calendar-header-title-height | _44px_ | - |
|
| --van-calendar-header-title-height | _44px_ | - |
|
||||||
| --van-calendar-header-title-font-size | _var(--van-font-size-lg)_ | - |
|
| --van-calendar-header-title-font-size | _var(--van-font-size-lg)_ | - |
|
||||||
| --van-calendar-header-subtitle-font-size | _var(--van-font-size-md)_ | - |
|
| --van-calendar-header-subtitle-font-size | _var(--van-font-size-md)_ | - |
|
||||||
|
| --van-calendar-header-action-width | 28px | - |
|
||||||
|
| --van-calendar-header-action-color | _var(--van-text-color)_ | - |
|
||||||
|
| --van-calendar-header-action-disabled-color | _var(--van-text-color-3)_ | - |
|
||||||
| --van-calendar-weekdays-height | _30px_ | - |
|
| --van-calendar-weekdays-height | _30px_ | - |
|
||||||
| --van-calendar-weekdays-font-size | _var(--van-font-size-sm)_ | - |
|
| --van-calendar-weekdays-font-size | _var(--van-font-size-sm)_ | - |
|
||||||
| --van-calendar-month-title-font-size | _var(--van-font-size-md)_ | - |
|
| --van-calendar-month-title-font-size | _var(--van-font-size-md)_ | - |
|
||||||
|
@ -18,6 +18,14 @@ app.use(Calendar);
|
|||||||
|
|
||||||
## 代码演示
|
## 代码演示
|
||||||
|
|
||||||
|
### 选择切换模式
|
||||||
|
|
||||||
|
默认所有月份将以平铺方式展示,不显示切换按钮,当月份过多时可能会影响页面交互性能。可以通过设置 `switch-mode` 属性,展示年月切换按钮。
|
||||||
|
|
||||||
|
```html
|
||||||
|
<van-calendar v-model:show="show" switch-mode="year-month" />
|
||||||
|
```
|
||||||
|
|
||||||
### 选择单个日期
|
### 选择单个日期
|
||||||
|
|
||||||
下面演示了结合单元格来使用日历组件的用法,日期选择完成后会触发 `confirm` 事件。
|
下面演示了结合单元格来使用日历组件的用法,日期选择完成后会触发 `confirm` 事件。
|
||||||
@ -253,11 +261,12 @@ export default {
|
|||||||
|
|
||||||
| 参数 | 说明 | 类型 | 默认值 |
|
| 参数 | 说明 | 类型 | 默认值 |
|
||||||
| --- | --- | --- | --- |
|
| --- | --- | --- | --- |
|
||||||
| type | 选择类型:<br>`single` 表示选择单个日期,<br>`multiple` 表示选择多个日期,<br>`range` 表示选择日期区间 | _string_ | `single` |
|
| type | 选择类型:<br>`single` 表示选择单个日期,<br>`multiple` 表示选择多个日期,<br>`range` 表示选择日期区间 | _string_ | `single` |
|
||||||
|
| switch-mode | 切换模式:<br>`none` 平铺展示所有月份,不展示切换按钮,<br>`month` 支持按月切换,展示上个月/下个月按钮,<br>`year-month` 支持按年切换,也支持按月切换,展示上一年/下一年,上个月/下个月按钮 | _string_ | `none` |
|
||||||
| title | 日历标题 | _string_ | `日期选择` |
|
| title | 日历标题 | _string_ | `日期选择` |
|
||||||
| color | 主题色,对底部按钮和选中日期生效 | _string_ | `#1989fa` |
|
| color | 主题色,对底部按钮和选中日期生效 | _string_ | `#1989fa` |
|
||||||
| min-date | 可选择的最小日期 | _Date_ | 当前日期 |
|
| min-date | 可选择的最小日期 | _Date_ | `switch-mode` 为 `none` 时为当前日期 |
|
||||||
| max-date | 可选择的最大日期 | _Date_ | 当前日期的六个月后 |
|
| max-date | 可选择的最大日期 | _Date_ | `switch-mode` 为 `none` 时为当前日期的六个月后 |
|
||||||
| default-date | 默认选中的日期,`type` 为 `multiple` 或 `range` 时为数组,传入 `null` 表示默认不选择 | _Date \| Date[] \| null_ | 今天 |
|
| default-date | 默认选中的日期,`type` 为 `multiple` 或 `range` 时为数组,传入 `null` 表示默认不选择 | _Date \| Date[] \| null_ | 今天 |
|
||||||
| row-height | 日期行高 | _number \| string_ | `64` |
|
| row-height | 日期行高 | _number \| string_ | `64` |
|
||||||
| formatter | 日期格式化函数 | _(day: Day) => Day_ | - |
|
| formatter | 日期格式化函数 | _(day: Day) => Day_ | - |
|
||||||
@ -335,6 +344,7 @@ export default {
|
|||||||
| over-range | 范围选择超过最多可选天数时触发 | - |
|
| over-range | 范围选择超过最多可选天数时触发 | - |
|
||||||
| click-subtitle | 点击日历副标题时触发 | _event: MouseEvent_ |
|
| click-subtitle | 点击日历副标题时触发 | _event: MouseEvent_ |
|
||||||
| click-disabled-date `v4.7.0` | 点击禁用日期时触发 | _value: Date \| Date[]_ |
|
| click-disabled-date `v4.7.0` | 点击禁用日期时触发 | _value: Date \| Date[]_ |
|
||||||
|
| panel-change | 日历面板切换时触发 | _{ date: Date }_ |
|
||||||
|
|
||||||
### Slots
|
### Slots
|
||||||
|
|
||||||
@ -347,6 +357,10 @@ export default {
|
|||||||
| confirm-text | 自定义确认按钮的内容 | _{ disabled: boolean }_ |
|
| confirm-text | 自定义确认按钮的内容 | _{ disabled: boolean }_ |
|
||||||
| top-info | 自定义日期上方的提示信息 | _day: Day_ |
|
| top-info | 自定义日期上方的提示信息 | _day: Day_ |
|
||||||
| bottom-info | 自定义日期下方的提示信息 | _day: Day_ |
|
| bottom-info | 自定义日期下方的提示信息 | _day: Day_ |
|
||||||
|
| prev-month | 自定义上个月按钮 | _{ disabled: boolean }_ |
|
||||||
|
| prev-year | 自定义上一年按钮 | _{ disabled: boolean }_ |
|
||||||
|
| next-month | 自定义下个月按钮 | _{ disabled: boolean }_ |
|
||||||
|
| next-year | 自定义下一年按钮 | _{ disabled: boolean }_ |
|
||||||
|
|
||||||
### 方法
|
### 方法
|
||||||
|
|
||||||
@ -364,6 +378,7 @@ export default {
|
|||||||
|
|
||||||
```ts
|
```ts
|
||||||
import type {
|
import type {
|
||||||
|
CalendarSwitchMode,
|
||||||
CalendarType,
|
CalendarType,
|
||||||
CalendarProps,
|
CalendarProps,
|
||||||
CalendarDayItem,
|
CalendarDayItem,
|
||||||
@ -397,6 +412,9 @@ calendarRef.value?.reset();
|
|||||||
| --van-calendar-header-title-height | _44px_ | - |
|
| --van-calendar-header-title-height | _44px_ | - |
|
||||||
| --van-calendar-header-title-font-size | _var(--van-font-size-lg)_ | - |
|
| --van-calendar-header-title-font-size | _var(--van-font-size-lg)_ | - |
|
||||||
| --van-calendar-header-subtitle-font-size | _var(--van-font-size-md)_ | - |
|
| --van-calendar-header-subtitle-font-size | _var(--van-font-size-md)_ | - |
|
||||||
|
| --van-calendar-header-action-width | 28px | - |
|
||||||
|
| --van-calendar-header-action-color | _var(--van-text-color)_ | - |
|
||||||
|
| --van-calendar-header-action-disabled-color | _var(--van-text-color-3)_ | - |
|
||||||
| --van-calendar-weekdays-height | _30px_ | - |
|
| --van-calendar-weekdays-height | _30px_ | - |
|
||||||
| --van-calendar-weekdays-font-size | _var(--van-font-size-sm)_ | - |
|
| --van-calendar-weekdays-font-size | _var(--van-font-size-sm)_ | - |
|
||||||
| --van-calendar-month-title-font-size | _var(--van-font-size-md)_ | - |
|
| --van-calendar-month-title-font-size | _var(--van-font-size-md)_ | - |
|
||||||
|
58
packages/vant/src/calendar/demo/SwicthModeField.vue
Normal file
58
packages/vant/src/calendar/demo/SwicthModeField.vue
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import VanCell from '../../cell';
|
||||||
|
import VanPicker, { type PickerConfirmEventParams } from '../../picker';
|
||||||
|
import VanPopup from '../../popup';
|
||||||
|
import { ref } from 'vue';
|
||||||
|
import { useTranslate } from '../../../docs/site';
|
||||||
|
import type { CalendarSwitchMode } from '../types';
|
||||||
|
|
||||||
|
const t = useTranslate({
|
||||||
|
'zh-CN': {
|
||||||
|
switchMode: '选择切换模式',
|
||||||
|
},
|
||||||
|
'en-US': {
|
||||||
|
switchMode: 'Select Switch Mode',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const switchMode = defineModel<CalendarSwitchMode>({
|
||||||
|
default: 'none',
|
||||||
|
});
|
||||||
|
const showPicker = ref(false);
|
||||||
|
const switchModeColumns = [
|
||||||
|
{ text: 'none', value: 'none' },
|
||||||
|
{ text: 'month', value: 'month' },
|
||||||
|
{ text: 'year-month', value: 'year-month' },
|
||||||
|
];
|
||||||
|
|
||||||
|
const onClickField = () => {
|
||||||
|
showPicker.value = true;
|
||||||
|
};
|
||||||
|
|
||||||
|
const onPickerCancel = () => {
|
||||||
|
showPicker.value = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
const onPickerConfirm = ({ selectedOptions }: PickerConfirmEventParams) => {
|
||||||
|
showPicker.value = false;
|
||||||
|
switchMode.value = selectedOptions[0]!.value as CalendarSwitchMode;
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<demo-block card :title="t('switchMode')">
|
||||||
|
<van-cell
|
||||||
|
is-link
|
||||||
|
:title="t('switchMode')"
|
||||||
|
:value="switchMode"
|
||||||
|
@click="onClickField"
|
||||||
|
/>
|
||||||
|
<van-popup v-model:show="showPicker" round position="bottom">
|
||||||
|
<van-picker
|
||||||
|
:columns="switchModeColumns"
|
||||||
|
@cancel="onPickerCancel"
|
||||||
|
@confirm="onPickerConfirm"
|
||||||
|
/>
|
||||||
|
</van-popup>
|
||||||
|
</demo-block>
|
||||||
|
</template>
|
@ -2,8 +2,11 @@
|
|||||||
import VanCalendar from '..';
|
import VanCalendar from '..';
|
||||||
import { useTranslate } from '../../../docs/site';
|
import { useTranslate } from '../../../docs/site';
|
||||||
|
|
||||||
|
const { switchMode } = defineProps({
|
||||||
|
switchMode: String,
|
||||||
|
});
|
||||||
const minDate = new Date(2012, 0, 10);
|
const minDate = new Date(2012, 0, 10);
|
||||||
const maxDate = new Date(2012, 2, 20);
|
const maxDate = new Date(2013, 2, 20);
|
||||||
|
|
||||||
const t = useTranslate({
|
const t = useTranslate({
|
||||||
'zh-CN': {
|
'zh-CN': {
|
||||||
@ -26,6 +29,7 @@ const t = useTranslate({
|
|||||||
:min-date="minDate"
|
:min-date="minDate"
|
||||||
:max-date="maxDate"
|
:max-date="maxDate"
|
||||||
:default-date="minDate"
|
:default-date="minDate"
|
||||||
|
:switch-mode="switchMode"
|
||||||
:style="{ height: '500px' }"
|
:style="{ height: '500px' }"
|
||||||
/>
|
/>
|
||||||
</demo-block>
|
</demo-block>
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import VanCell from '../../cell';
|
import VanCell from '../../cell';
|
||||||
import VanCalendar from '..';
|
import VanCalendar from '..';
|
||||||
import { reactive } from 'vue';
|
import { reactive, ref } from 'vue';
|
||||||
import { useTranslate } from '../../../docs/site';
|
import { useTranslate } from '../../../docs/site';
|
||||||
|
import SwicthModeField from './SwicthModeField.vue';
|
||||||
import TiledDisplay from './TiledDisplay.vue';
|
import TiledDisplay from './TiledDisplay.vue';
|
||||||
import type { CalendarDayItem } from '../types';
|
import type { CalendarDayItem, CalendarSwitchMode } from '../types';
|
||||||
|
|
||||||
const t = useTranslate({
|
const t = useTranslate({
|
||||||
'zh-CN': {
|
'zh-CN': {
|
||||||
@ -191,9 +192,13 @@ const onConfirm = (date: Date | Date[]) => {
|
|||||||
state.showCalendar = false;
|
state.showCalendar = false;
|
||||||
state.date[state.id] = date;
|
state.date[state.id] = date;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const switchMode = ref<CalendarSwitchMode>('none');
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
|
<SwicthModeField v-model="switchMode" />
|
||||||
|
|
||||||
<demo-block card :title="t('basicUsage')">
|
<demo-block card :title="t('basicUsage')">
|
||||||
<van-cell
|
<van-cell
|
||||||
is-link
|
is-link
|
||||||
@ -283,7 +288,7 @@ const onConfirm = (date: Date | Date[]) => {
|
|||||||
/>
|
/>
|
||||||
</demo-block>
|
</demo-block>
|
||||||
|
|
||||||
<TiledDisplay />
|
<TiledDisplay :switch-mode="switchMode" />
|
||||||
|
|
||||||
<van-calendar
|
<van-calendar
|
||||||
v-model:show="state.showCalendar"
|
v-model:show="state.showCalendar"
|
||||||
@ -296,6 +301,7 @@ const onConfirm = (date: Date | Date[]) => {
|
|||||||
:max-range="state.maxRange"
|
:max-range="state.maxRange"
|
||||||
:formatter="state.formatter"
|
:formatter="state.formatter"
|
||||||
:show-confirm="state.showConfirm"
|
:show-confirm="state.showConfirm"
|
||||||
|
:switch-mode="switchMode"
|
||||||
:confirm-text="state.confirmText"
|
:confirm-text="state.confirmText"
|
||||||
:first-day-of-week="state.firstDayOfWeek"
|
:first-day-of-week="state.firstDayOfWeek"
|
||||||
:confirm-disabled-text="state.confirmDisabledText"
|
:confirm-disabled-text="state.confirmDisabledText"
|
||||||
|
@ -6,6 +6,9 @@
|
|||||||
--van-calendar-header-title-height: 44px;
|
--van-calendar-header-title-height: 44px;
|
||||||
--van-calendar-header-title-font-size: var(--van-font-size-lg);
|
--van-calendar-header-title-font-size: var(--van-font-size-lg);
|
||||||
--van-calendar-header-subtitle-font-size: var(--van-font-size-md);
|
--van-calendar-header-subtitle-font-size: var(--van-font-size-md);
|
||||||
|
--van-calendar-header-action-width: 28px;
|
||||||
|
--van-calendar-header-action-color: var(--van-text-color);
|
||||||
|
--van-calendar-header-action-disabled-color: var(--van-text-color-3);
|
||||||
--van-calendar-weekdays-height: 30px;
|
--van-calendar-weekdays-height: 30px;
|
||||||
--van-calendar-weekdays-font-size: var(--van-font-size-sm);
|
--van-calendar-weekdays-font-size: var(--van-font-size-sm);
|
||||||
--van-calendar-month-title-font-size: var(--van-font-size-md);
|
--van-calendar-month-title-font-size: var(--van-font-size-md);
|
||||||
@ -76,6 +79,31 @@
|
|||||||
|
|
||||||
&__header-subtitle {
|
&__header-subtitle {
|
||||||
font-size: var(--van-calendar-header-subtitle-font-size);
|
font-size: var(--van-calendar-header-subtitle-font-size);
|
||||||
|
|
||||||
|
&--with-swicth {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0 var(--van-padding-base);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&__header-subtitle-text {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__header-action {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
min-width: var(--van-calendar-header-action-width);
|
||||||
|
height: 100%;
|
||||||
|
color: var(--van-calendar-header-action-color);
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
&--disabled {
|
||||||
|
color: var(--van-calendar-header-action-disabled-color);
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&__month-title {
|
&__month-title {
|
||||||
|
@ -2,6 +2,35 @@
|
|||||||
|
|
||||||
exports[`should render demo and match snapshot 1`] = `
|
exports[`should render demo and match snapshot 1`] = `
|
||||||
<!--[-->
|
<!--[-->
|
||||||
|
<div>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-cell van-cell--clickable"
|
||||||
|
role="button"
|
||||||
|
tabindex="0"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-cell__title"
|
||||||
|
style
|
||||||
|
>
|
||||||
|
<span>
|
||||||
|
Select Switch Mode
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div class="van-cell__value">
|
||||||
|
<span>
|
||||||
|
none
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<i
|
||||||
|
class="van-badge__wrapper van-icon van-icon-arrow van-cell__right-icon"
|
||||||
|
style
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
</i>
|
||||||
|
</div>
|
||||||
|
<!--[-->
|
||||||
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<!--[-->
|
<!--[-->
|
||||||
<div
|
<div
|
||||||
@ -395,6 +424,453 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/4
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/5
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/6
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/7
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/8
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/9
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/10
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/11
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/12
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2013/1
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2013/2
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2013/3
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<!--[-->
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width:100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="van-calendar__footer van-safe-area-bottom">
|
<div class="van-calendar__footer van-safe-area-bottom">
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,6 +1,42 @@
|
|||||||
// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
|
// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
|
||||||
|
|
||||||
exports[`should render demo and match snapshot 1`] = `
|
exports[`should render demo and match snapshot 1`] = `
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
class="van-cell van-cell--clickable"
|
||||||
|
role="button"
|
||||||
|
tabindex="0"
|
||||||
|
>
|
||||||
|
<div class="van-cell__title">
|
||||||
|
<span>
|
||||||
|
Select Switch Mode
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div class="van-cell__value">
|
||||||
|
<span>
|
||||||
|
none
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<i class="van-badge__wrapper van-icon van-icon-arrow van-cell__right-icon">
|
||||||
|
</i>
|
||||||
|
</div>
|
||||||
|
<transition-stub
|
||||||
|
name="van-fade"
|
||||||
|
appear="true"
|
||||||
|
persisted="false"
|
||||||
|
css="true"
|
||||||
|
role="button"
|
||||||
|
tabindex="0"
|
||||||
|
>
|
||||||
|
</transition-stub>
|
||||||
|
<transition-stub
|
||||||
|
name="van-popup-slide-bottom"
|
||||||
|
appear="false"
|
||||||
|
persisted="false"
|
||||||
|
css="true"
|
||||||
|
>
|
||||||
|
</transition-stub>
|
||||||
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<div
|
<div
|
||||||
class="van-cell van-cell--clickable"
|
class="van-cell van-cell--clickable"
|
||||||
@ -301,6 +337,441 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/4
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/5
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/6
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/7
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/8
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/9
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/10
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/11
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2012/12
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2013/1
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2013/2
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="van-calendar__month">
|
||||||
|
<div class="van-calendar__month-title">
|
||||||
|
2013/3
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="grid"
|
||||||
|
class="van-calendar__days"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="van-calendar__day"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="van-calendar__footer van-safe-area-bottom">
|
<div class="van-calendar__footer van-safe-area-bottom">
|
||||||
</div>
|
</div>
|
||||||
|
205
packages/vant/src/calendar/test/switch-mode.spec.ts
Normal file
205
packages/vant/src/calendar/test/switch-mode.spec.ts
Normal file
@ -0,0 +1,205 @@
|
|||||||
|
import { Calendar, CalendarInstance } from '..';
|
||||||
|
import { mount, later } from '../../../test';
|
||||||
|
import {
|
||||||
|
formatMonthTitle,
|
||||||
|
getMonthByOffset,
|
||||||
|
getPrevMonth,
|
||||||
|
getNextMonth,
|
||||||
|
getPrevYear,
|
||||||
|
getNextYear,
|
||||||
|
getYearByOffset,
|
||||||
|
} from '../utils';
|
||||||
|
import { minDate, maxDate } from './utils';
|
||||||
|
|
||||||
|
const disabledActionClass = 'van-calendar__header-action--disabled';
|
||||||
|
|
||||||
|
test('the action buttons should be displayed correctly', async () => {
|
||||||
|
const wrapper = mount(Calendar, {
|
||||||
|
props: {
|
||||||
|
minDate,
|
||||||
|
maxDate,
|
||||||
|
poppable: false,
|
||||||
|
switchMode: 'month',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await later();
|
||||||
|
expect(wrapper.findAll('.van-calendar__header-action')).toHaveLength(2);
|
||||||
|
|
||||||
|
await wrapper.setProps({ switchMode: 'year-month' });
|
||||||
|
expect(wrapper.findAll('.van-calendar__header-action')).toHaveLength(4);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('disable previous and next month buttons', async () => {
|
||||||
|
const maxDate = getNextMonth(minDate);
|
||||||
|
const wrapper = mount(Calendar, {
|
||||||
|
props: {
|
||||||
|
minDate,
|
||||||
|
maxDate,
|
||||||
|
poppable: false,
|
||||||
|
switchMode: 'month',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await later();
|
||||||
|
const title = wrapper.find('.van-calendar__header-subtitle-text');
|
||||||
|
const [prevMonth, nextMonth] = wrapper.findAll(
|
||||||
|
'.van-calendar__header-action',
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(title.text()).toEqual(formatMonthTitle(maxDate));
|
||||||
|
expect(prevMonth.classes()).not.toContain(disabledActionClass);
|
||||||
|
expect(nextMonth.classes()).toContain(disabledActionClass);
|
||||||
|
|
||||||
|
await nextMonth.trigger('click');
|
||||||
|
expect(title.text()).toEqual(formatMonthTitle(maxDate));
|
||||||
|
expect(prevMonth.classes()).not.toContain(disabledActionClass);
|
||||||
|
expect(nextMonth.classes()).toContain(disabledActionClass);
|
||||||
|
|
||||||
|
await prevMonth.trigger('click');
|
||||||
|
expect(title.text()).toEqual(formatMonthTitle(minDate));
|
||||||
|
expect(prevMonth.classes()).toContain(disabledActionClass);
|
||||||
|
expect(nextMonth.classes()).not.toContain(disabledActionClass);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('disable previous and next year buttons', async () => {
|
||||||
|
const maxDate = getNextYear(minDate);
|
||||||
|
const wrapper = mount(Calendar, {
|
||||||
|
props: {
|
||||||
|
minDate,
|
||||||
|
maxDate,
|
||||||
|
poppable: false,
|
||||||
|
switchMode: 'year-month',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
await later();
|
||||||
|
let currentDate = maxDate;
|
||||||
|
const title = wrapper.find('.van-calendar__header-subtitle-text');
|
||||||
|
const [prevYear, prevMonth, nextMonth, nextYear] = wrapper.findAll(
|
||||||
|
'.van-calendar__header-action',
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(title.text()).toEqual(formatMonthTitle(currentDate));
|
||||||
|
expect(prevYear.classes()).not.toContain(disabledActionClass);
|
||||||
|
expect(prevMonth.classes()).not.toContain(disabledActionClass);
|
||||||
|
expect(nextMonth.classes()).toContain(disabledActionClass);
|
||||||
|
expect(nextYear.classes()).toContain(disabledActionClass);
|
||||||
|
|
||||||
|
await prevMonth.trigger('click');
|
||||||
|
currentDate = getPrevMonth(currentDate);
|
||||||
|
expect(title.text()).toEqual(formatMonthTitle(currentDate));
|
||||||
|
expect(prevYear.classes()).toContain(disabledActionClass);
|
||||||
|
expect(prevMonth.classes()).not.toContain(disabledActionClass);
|
||||||
|
expect(nextMonth.classes()).not.toContain(disabledActionClass);
|
||||||
|
expect(nextYear.classes()).toContain(disabledActionClass);
|
||||||
|
|
||||||
|
await nextMonth.trigger('click');
|
||||||
|
currentDate = getNextMonth(currentDate);
|
||||||
|
expect(title.text()).toEqual(formatMonthTitle(currentDate));
|
||||||
|
|
||||||
|
await prevYear.trigger('click');
|
||||||
|
currentDate = getPrevYear(currentDate);
|
||||||
|
expect(title.text()).toEqual(formatMonthTitle(currentDate));
|
||||||
|
expect(prevYear.classes()).toContain(disabledActionClass);
|
||||||
|
expect(prevMonth.classes()).toContain(disabledActionClass);
|
||||||
|
expect(nextMonth.classes()).not.toContain(disabledActionClass);
|
||||||
|
expect(nextYear.classes()).not.toContain(disabledActionClass);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should switch to the provided date after calling the scrollToDate method', async () => {
|
||||||
|
const maxDate = getNextYear(minDate);
|
||||||
|
const wrapper = mount(Calendar, {
|
||||||
|
props: {
|
||||||
|
minDate,
|
||||||
|
maxDate,
|
||||||
|
poppable: false,
|
||||||
|
switchMode: 'month',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await later();
|
||||||
|
let currentDate = maxDate;
|
||||||
|
const title = wrapper.find('.van-calendar__header-subtitle-text');
|
||||||
|
|
||||||
|
currentDate = getMonthByOffset(currentDate, -4);
|
||||||
|
(wrapper.vm as CalendarInstance).scrollToDate(currentDate);
|
||||||
|
await later();
|
||||||
|
expect(title.text()).toEqual(formatMonthTitle(currentDate));
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should render action slots correctly', async () => {
|
||||||
|
const prevYearSlot = vi.fn(() => 'prev year');
|
||||||
|
const prevMonthSlot = vi.fn(() => 'prev month');
|
||||||
|
const nextMonthSlot = vi.fn(() => 'next month');
|
||||||
|
const nextYearSlot = vi.fn(() => 'next year');
|
||||||
|
const maxDate = getNextYear(minDate);
|
||||||
|
const wrapper = mount(Calendar, {
|
||||||
|
props: {
|
||||||
|
minDate,
|
||||||
|
maxDate,
|
||||||
|
poppable: false,
|
||||||
|
switchMode: 'year-month',
|
||||||
|
},
|
||||||
|
slots: {
|
||||||
|
'prev-year': prevYearSlot,
|
||||||
|
'prev-month': prevMonthSlot,
|
||||||
|
'next-month': nextMonthSlot,
|
||||||
|
'next-year': nextYearSlot,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await later();
|
||||||
|
const [prevYear, prevMonth, nextMonth, nextYear] = wrapper.findAll(
|
||||||
|
'.van-calendar__header-action',
|
||||||
|
);
|
||||||
|
expect(prevYearSlot).toHaveBeenLastCalledWith({ disabled: false });
|
||||||
|
expect(prevMonthSlot).toHaveBeenLastCalledWith({ disabled: false });
|
||||||
|
expect(nextMonthSlot).toHaveBeenLastCalledWith({ disabled: true });
|
||||||
|
expect(nextYearSlot).toHaveBeenLastCalledWith({ disabled: true });
|
||||||
|
expect(prevYear.text()).toEqual('prev year');
|
||||||
|
expect(prevMonth.text()).toEqual('prev month');
|
||||||
|
expect(nextMonth.text()).toEqual('next month');
|
||||||
|
expect(nextYear.text()).toEqual('next year');
|
||||||
|
|
||||||
|
await prevMonth.trigger('click');
|
||||||
|
expect(prevYearSlot).toHaveBeenLastCalledWith({ disabled: true });
|
||||||
|
expect(prevMonthSlot).toHaveBeenLastCalledWith({ disabled: false });
|
||||||
|
expect(nextMonthSlot).toHaveBeenLastCalledWith({ disabled: false });
|
||||||
|
expect(nextYearSlot).toHaveBeenLastCalledWith({ disabled: true });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should emit panelChange event', async () => {
|
||||||
|
const onPanelChange = vi.fn();
|
||||||
|
const maxDate = getYearByOffset(minDate, 10);
|
||||||
|
const wrapper = mount(Calendar, {
|
||||||
|
props: {
|
||||||
|
minDate,
|
||||||
|
maxDate,
|
||||||
|
poppable: false,
|
||||||
|
switchMode: 'year-month',
|
||||||
|
onPanelChange,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await later();
|
||||||
|
let currentDate = maxDate;
|
||||||
|
const [prevYear, prevMonth, nextMonth, nextYear] = wrapper.findAll(
|
||||||
|
'.van-calendar__header-action',
|
||||||
|
);
|
||||||
|
|
||||||
|
await prevMonth.trigger('click');
|
||||||
|
currentDate = getPrevMonth(currentDate);
|
||||||
|
expect(onPanelChange).toHaveBeenLastCalledWith({ date: currentDate });
|
||||||
|
|
||||||
|
await prevYear.trigger('click');
|
||||||
|
currentDate = getPrevYear(currentDate);
|
||||||
|
expect(onPanelChange).toHaveBeenLastCalledWith({ date: currentDate });
|
||||||
|
|
||||||
|
await nextYear.trigger('click');
|
||||||
|
currentDate = getNextYear(currentDate);
|
||||||
|
expect(onPanelChange).toHaveBeenLastCalledWith({ date: currentDate });
|
||||||
|
|
||||||
|
await nextMonth.trigger('click');
|
||||||
|
currentDate = getNextMonth(currentDate);
|
||||||
|
expect(onPanelChange).toHaveBeenLastCalledWith({ date: currentDate });
|
||||||
|
});
|
@ -3,6 +3,8 @@ import type { Numeric } from '../utils';
|
|||||||
import type { CalendarProps } from './Calendar';
|
import type { CalendarProps } from './Calendar';
|
||||||
import type { CalendarMonthProps } from './CalendarMonth';
|
import type { CalendarMonthProps } from './CalendarMonth';
|
||||||
|
|
||||||
|
export type CalendarSwitchMode = 'none' | 'month' | 'year-month';
|
||||||
|
|
||||||
export type CalendarType = 'single' | 'range' | 'multiple';
|
export type CalendarType = 'single' | 'range' | 'multiple';
|
||||||
|
|
||||||
export type CalendarDayType =
|
export type CalendarDayType =
|
||||||
@ -56,6 +58,9 @@ export type CalendarThemeVars = {
|
|||||||
calendarHeaderTitleHeight?: string;
|
calendarHeaderTitleHeight?: string;
|
||||||
calendarHeaderTitleFontSize?: string;
|
calendarHeaderTitleFontSize?: string;
|
||||||
calendarHeaderSubtitleFontSize?: string;
|
calendarHeaderSubtitleFontSize?: string;
|
||||||
|
calendarHeaderActionWidth?: string;
|
||||||
|
calendarHeaderActionColor?: string;
|
||||||
|
calendarHeaderActionDisabledColor?: string;
|
||||||
calendarWeekdaysHeight?: string;
|
calendarWeekdaysHeight?: string;
|
||||||
calendarWeekdaysFontSize?: string;
|
calendarWeekdaysFontSize?: string;
|
||||||
calendarMonthTitleFontSize?: string;
|
calendarMonthTitleFontSize?: string;
|
||||||
|
@ -43,8 +43,24 @@ export function getDayByOffset(date: Date, offset: number) {
|
|||||||
return cloned;
|
return cloned;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function getMonthByOffset(date: Date, offset: number) {
|
||||||
|
const cloned = cloneDate(date);
|
||||||
|
cloned.setMonth(cloned.getMonth() + offset);
|
||||||
|
return cloned;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getYearByOffset(date: Date, offset: number) {
|
||||||
|
const cloned = cloneDate(date);
|
||||||
|
cloned.setFullYear(cloned.getFullYear() + offset);
|
||||||
|
return cloned;
|
||||||
|
}
|
||||||
|
|
||||||
export const getPrevDay = (date: Date) => getDayByOffset(date, -1);
|
export const getPrevDay = (date: Date) => getDayByOffset(date, -1);
|
||||||
export const getNextDay = (date: Date) => getDayByOffset(date, 1);
|
export const getNextDay = (date: Date) => getDayByOffset(date, 1);
|
||||||
|
export const getPrevMonth = (date: Date) => getMonthByOffset(date, -1);
|
||||||
|
export const getNextMonth = (date: Date) => getMonthByOffset(date, 1);
|
||||||
|
export const getPrevYear = (date: Date) => getYearByOffset(date, -1);
|
||||||
|
export const getNextYear = (date: Date) => getYearByOffset(date, 1);
|
||||||
export const getToday = () => {
|
export const getToday = () => {
|
||||||
const today = new Date();
|
const today = new Date();
|
||||||
today.setHours(0, 0, 0, 0);
|
today.setHours(0, 0, 0, 0);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user