/* global window */ // Vue import { h, computed, defineComponent, getCurrentInstance, onBeforeUpdate, onMounted, nextTick, reactive, ref, Transition, watch, withDirectives, CSSProperties, SetupContext, VNode, } from 'vue' // Utility import { getDayIdentifier, parsed, parseTimestamp, today, type Timestamp } from '../utils/Timestamp' import { convertToUnit, minCharWidth } from '../utils/helpers' // Composables import useCalendar from '../composables/useCalendar' import useCommon, { useCommonProps } from '../composables/useCommon' import useInterval, { useSchedulerProps, type Resource } from '../composables/useInterval' import { useColumnProps } from '../composables/useColumn' import { useMaxDaysProps } from '../composables/useMaxDays' import useTimes, { useTimesProps } from '../composables/useTimes' import useRenderValues from '../composables/useRenderValues' import useMouse, { getRawMouseEvents } from '../composables/useMouse' import useMove, { useMoveEmits } from '../composables/useMove' import useEmitListeners from '../composables/useEmitListeners' import useButton from '../composables/useButton' import useFocusHelper from '../composables/useFocusHelper' import useCellWidth, { useCellWidthProps } from '../composables/useCellWidth' import useCheckChange, { useCheckChangeEmits } from '../composables/useCheckChange' import useEvents from '../composables/useEvents' import useKeyboard, { useNavigationProps } from '../composables/useKeyboard' // Directives import ResizeObserver from '../directives/ResizeObserver' interface Size { width: number height: number } const { renderButton } = useButton() export default defineComponent({ name: 'QCalendarScheduler', directives: { ResizeObserver }, props: { ...useCommonProps, ...useSchedulerProps, ...useColumnProps, ...useMaxDaysProps, ...useTimesProps, ...useCellWidthProps, ...useNavigationProps, }, emits: [ 'update:model-value', 'update:model-resources', 'resource-expanded', ...useCheckChangeEmits, ...useMoveEmits, ...getRawMouseEvents('-date'), ...getRawMouseEvents('-day-resource'), ...getRawMouseEvents('-head-resources'), ...getRawMouseEvents('-head-day'), ...getRawMouseEvents('-resource'), ], setup(props, { slots, emit, expose }: SetupContext) { const scrollArea = ref(null), pane = ref(null), headerColumnRef = ref(null), focusRef = ref(props.modelValue || today()), focusValue = ref(parsed(props.modelValue || today()) as Timestamp), datesRef = ref>({}), resourcesRef = ref>({}), headDayEventsParentRef = ref(), headDayEventsChildRef = ref(), // resourceFocusRef = ref(null), // resourceFocusValue = ref(null), // resourcesHeadRef = ref(null), direction = ref<'next' | 'prev'>('next'), startDate = ref(props.modelValue || today()), endDate = ref('0000-00-00'), maxDaysRendered = ref(0), emittedValue = ref(props.modelValue), size = reactive({ width: 0, height: 0 }), dragOverHeadDayRef = ref(''), dragOverResource = ref(''), // keep track of last seen start and end dates lastStart = ref(null), lastEnd = ref(null) watch( () => props.view, () => { // reset maxDaysRendered maxDaysRendered.value = 0 }, ) const parsedView = computed(() => { if (props.view === 'month') { return 'month-interval' } return props.view }) const vm = getCurrentInstance() if (vm === null) { throw new Error('current instance is null') } const { emitListeners } = useEmitListeners(vm) const { isSticky } = useCellWidth(props) const { times, setCurrent, updateCurrent } = useTimes(props) // update dates updateCurrent() setCurrent() const { // computed parsedStart, parsedEnd, dayFormatter, weekdayFormatter, ariaDateFormatter, // methods dayStyleDefault, getRelativeClasses, } = useCommon(props, { startDate, endDate, times }) const parsedValue = computed(() => { return parseTimestamp(props.modelValue, times.now) || parsedStart.value || times.today }) focusValue.value = parsedValue.value focusRef.value = parsedValue.value.date const { renderValues } = useRenderValues(props, { parsedView, parsedValue, times, }) const { rootRef, scrollWidth, __initCalendar, __renderCalendar } = useCalendar( props, __renderScheduler, { scrollArea, pane, }, ) const { // computed days, // intervals, // intervalFormatter, // ariaDateTimeFormatter, parsedCellWidth, // methods // getResourceClasses, // showResourceLabelDefault, styleDefault, // getTimestampAtEventInterval, // getTimestampAtEvent, // getScopeForSlot // scrollToTime, // timeDurationHeight, // timeStartPos /// @ts-expect-error fix later } = useInterval(props, { times, scrollArea, parsedStart, parsedEnd, maxDays: maxDaysRendered, size, headerColumnRef, }) const { move } = useMove(props, { parsedView, parsedValue, direction, maxDays: maxDaysRendered, times, emittedValue, emit, }) const { getDefaultMouseEventHandlers } = useMouse(emit, emitListeners) const { checkChange } = useCheckChange(emit, { days, lastStart, lastEnd }) const { isKeyCode } = useEvents() /// @ts-expect-error fix later const { tryFocus } = useKeyboard(props, { rootRef, focusRef, focusValue, datesRef, days, parsedView, parsedValue, emittedValue, direction, times, }) const parsedColumnCount = computed(() => { const columnCount = parseInt(String(props.columnCount), 10) if (parsedView.value === 'day' && columnCount > 1) { return columnCount } else if (parsedView.value === 'day' && props.maxDays && props.maxDays > 1) { return props.maxDays } return days.value.length }) const resourcesWidth = computed(() => { if (rootRef.value) { return parseInt( window.getComputedStyle(rootRef.value).getPropertyValue('--calendar-resources-width'), 10, ) } return 0 }) const parsedResourceHeight = computed(() => { const height = parseInt(String(props.resourceHeight), 10) if (height === 0) { return 'auto' } return height }) const parsedResourceMinHeight = computed(() => { return parseInt(String(props.resourceMinHeight), 10) }) const computedWidth = computed(() => { if (rootRef.value) { const width = size.width || rootRef.value.getBoundingClientRect().width if (width && resourcesWidth.value && parsedColumnCount.value) { return (width - scrollWidth.value - resourcesWidth.value) / parsedColumnCount.value + 'px' } } return 100 / parsedColumnCount.value + '%' }) watch([days], checkChange, { deep: true, immediate: true }) watch( () => props.modelValue, (val, oldVal) => { if (emittedValue.value !== props.modelValue) { if (props.animated === true) { const v1 = getDayIdentifier(parsed(val) as Timestamp) const v2 = getDayIdentifier(parsed(oldVal) as Timestamp) direction.value = v1 >= v2 ? 'next' : 'prev' } emittedValue.value = val } focusRef.value = val }, ) watch(emittedValue, (val, oldVal) => { if (emittedValue.value !== props.modelValue) { if (props.animated === true) { const v1 = getDayIdentifier(parsed(val) as Timestamp) const v2 = getDayIdentifier(parsed(oldVal) as Timestamp) direction.value = v1 >= v2 ? 'next' : 'prev' } emit('update:model-value', val) } }) watch(focusRef, (val) => { if (val) { focusValue.value = parseTimestamp(val) as Timestamp } }) watch(focusValue, () => { if (datesRef.value[focusRef.value]) { datesRef.value[focusRef.value].focus() } else { // if focusRef is not in the list of current dates of dateRef, // then assume list of days is changing tryFocus() } }) watch( () => props.maxDays, (val) => { maxDaysRendered.value = val }, ) onBeforeUpdate(() => { datesRef.value = {} headDayEventsParentRef.value = undefined headDayEventsChildRef.value = undefined resourcesRef.value = {} }) onMounted(() => { __initCalendar() }) // public functions function moveToToday(): void { emittedValue.value = today() } function next(amount = 1): void { move(amount) } function prev(amount = 1): void { move(-amount) } // private functions function __onResize({ width, height }: Size): void { size.width = width size.height = height } function __isActiveDate(day: Timestamp): boolean { return day.date === emittedValue.value } // function __isActiveResource (day) { // return __isActiveDate(day) // && day.hasTime // && emittedValue.value.hasTime // && day.time === emittedValue.value.time // } // Render functions function __renderHead(): VNode { return h( 'div', { roll: 'presentation', class: { 'q-calendar-scheduler__head': true, 'q-calendar__sticky': isSticky.value === true, }, style: { marginRight: scrollWidth.value + 'px', }, }, [__renderHeadResources(), __renderHeadDaysColumn()], ) } /* * Outputs the header that is above the resources */ function __renderHeadResources(): VNode { const slot = slots['head-resources'] const scope = { days: days.value, // deprecated timestamps: days.value, date: props.modelValue, resources: props.modelResources, } return h( 'div', { class: { 'q-calendar-scheduler__head--resources': true, 'q-calendar__sticky': isSticky.value === true, }, ...getDefaultMouseEventHandlers('-head-resources', (event) => { return { scope, event } }), }, [slot && slot({ scope })], ) } function __renderHeadDaysColumn(): VNode { return h( 'div', { ref: headerColumnRef, class: { 'q-calendar-scheduler__head--days__column': true, }, }, [__renderHeadDaysRow(), __renderHeadDaysEventsRow()], ) } function __renderHeadDaysRow(): VNode { return h( 'div', { class: { 'q-calendar-scheduler__head--days__weekdays': true, }, }, [...__renderHeadDays()], ) } function __renderHeadDaysEventsRow(): VNode { const slot = slots['head-days-events'] nextTick(() => { if ( headDayEventsChildRef.value && parseInt(String(props.columnCount), 10) === 0 && window ) { try { const styles = window.getComputedStyle(headDayEventsChildRef.value) if (headDayEventsParentRef.value && headDayEventsParentRef.value.parentElement) { headDayEventsParentRef.value.parentElement.style.height = styles.height headDayEventsParentRef.value.style.height = styles.height } } catch { // } } }) return h( 'div', { class: { 'q-calendar-scheduler__head--days__event': true, }, }, [ slot && h( 'div', { ref: headDayEventsParentRef, // TODO: this needs to be in a class style: { position: 'absolute', left: 0, top: 0, right: 0, overflow: 'hidden', zIndex: 1, }, }, [ slot({ scope: { timestamps: days.value, days: days.value, // deprecated ref: headDayEventsChildRef, }, }), ], ), ...__renderHeadDaysEvents(), ], ) } function __renderHeadDays(): VNode[] { const columnCount = parseInt(String(props.columnCount), 10) const columnIndexStart = parseInt(String(props.columnIndexStart), 10) if (days.value.length === 1 && columnCount > 0) { return Array.apply(null, new Array(columnCount)) .map((_, i) => i + columnIndexStart) .map((columnIndex) => __renderHeadDay(days.value[0], columnIndex)) } else { return days.value.map((day) => __renderHeadDay(day, 0)) } } function __renderHeadDaysEvents(): VNode[] { const columnCount = parseInt(String(props.columnCount), 10) const columnIndexStart = parseInt(String(props.columnIndexStart), 10) if (days.value.length === 1 && columnCount > 0) { return Array.apply(null, new Array(columnCount)) .map((_, i) => i + columnIndexStart) .map((columnIndex) => __renderHeadDayEvent(days.value[0], columnIndex)) } else { return days.value.map((day) => __renderHeadDayEvent(day, 0)) } } function __renderHeadDay(day: Timestamp, columnIndex: number): VNode { const headDaySlot = slots['head-day'] const headDateSlot = slots['head-date'] const activeDate = props.noActiveDate !== true && __isActiveDate(day) const scope = { timestamp: day, activeDate, droppable: dragOverHeadDayRef.value === day.date, disabled: props.disabledWeekdays ? props.disabledWeekdays.includes(Number(day.weekday)) : false, columnIndex: columnIndex ?? 0, } const width = isSticky.value === true ? convertToUnit(parsedCellWidth.value) : computedWidth.value const styler = props.weekdayStyle || dayStyleDefault const style: CSSProperties = { width, maxWidth: width, minWidth: width, ...styler({ scope }), } if (isSticky.value === true) { style.minWidth = width } const weekdayClass = typeof props.weekdayClass === 'function' ? props.weekdayClass({ scope }) : {} const isFocusable = props.focusable === true && props.focusType.includes('weekday') const key = day.date + (columnIndex !== undefined ? '-' + columnIndex : '') const data: Record = { key, ref: (el: HTMLElement | null) => { if (el !== null) { datesRef.value[key] = el } }, tabindex: isFocusable === true ? 0 : -1, class: { 'q-calendar-scheduler__head--day': true, ...weekdayClass, ...getRelativeClasses(day), 'q-active-date': activeDate, 'q-calendar__hoverable': props.hoverable === true, 'q-calendar__focusable': isFocusable === true, }, style, onFocus: () => { if (isFocusable === true) { focusRef.value = key } }, onKeydown: (e: KeyboardEvent) => { if (day.disabled !== true && isKeyCode(e, [13, 32])) { e.stopPropagation() e.preventDefault() } }, onKeyup: (e: KeyboardEvent) => { // allow selection of date via Enter or Space keys if (day.disabled !== true && isKeyCode(e, [13, 32])) { emittedValue.value = day.date } }, ...getDefaultMouseEventHandlers('-head-day', (event) => { return { scope, event } }), onDragenter: (e: DragEvent) => { if (props.dragEnterFunc !== undefined && typeof props.dragEnterFunc === 'function') { props.dragEnterFunc(e, 'head-day', { scope }) === true ? (dragOverHeadDayRef.value = day.date) : (dragOverHeadDayRef.value = '') } }, onDragover: (e: DragEvent) => { if (props.dragOverFunc !== undefined && typeof props.dragOverFunc === 'function') { props.dragOverFunc(e, 'head-day', { scope }) === true ? (dragOverHeadDayRef.value = day.date) : (dragOverHeadDayRef.value = '') } }, onDragleave: (e: DragEvent) => { if (props.dragLeaveFunc !== undefined && typeof props.dragLeaveFunc === 'function') { props.dragLeaveFunc(e, 'head-day', { scope }) === true ? (dragOverHeadDayRef.value = day.date) : (dragOverHeadDayRef.value = '') } }, onDrop: (e: DragEvent) => { if (props.dropFunc !== undefined && typeof props.dropFunc === 'function') { props.dropFunc(e, 'head-day', { scope }) === true ? (dragOverHeadDayRef.value = day.date) : (dragOverHeadDayRef.value = '') } }, } return h('div', data, [ // head-day slot replaces everything below it headDaySlot !== undefined && headDaySlot({ scope }), headDaySlot === undefined && __renderColumnHeaderBefore(day, columnIndex), headDaySlot === undefined && __renderDateHeader(day), headDaySlot === undefined && headDateSlot && headDateSlot({ scope }), headDaySlot === undefined && __renderColumnHeaderAfter(day, columnIndex), useFocusHelper(), ]) } function __renderDateHeader(day: Timestamp): VNode | VNode[] | void { if (props.dateHeader === 'stacked') { return [ props.noDefaultHeaderText !== true ? __renderHeadWeekday(day) : [], props.noDefaultHeaderBtn !== true ? __renderHeadDayDate(day) : [], ].flat() } else if (props.dateHeader === 'inline') { if (props.weekdayAlign === 'left' && props.dateAlign === 'right') { return h( 'div', { class: 'q-calendar__header--inline', }, [ props.noDefaultHeaderText !== true && __renderHeadWeekday(day), props.noDefaultHeaderBtn !== true && __renderHeadDayDate(day), ], ) } else if (props.weekdayAlign === 'right' && props.dateAlign === 'left') { return h( 'div', { class: 'q-calendar__header--inline', }, [ props.noDefaultHeaderText !== true && __renderHeadWeekday(day), props.noDefaultHeaderBtn !== true && __renderHeadDayDate(day), ], ) } else { return h( 'div', { class: 'q-calendar__header--inline', }, [ props.noDefaultHeaderText !== true && __renderHeadWeekday(day), props.noDefaultHeaderBtn !== true && __renderHeadDayDate(day), ], ) } } else if (props.dateHeader === 'inverted') { if (props.weekdayAlign === 'left' && props.dateAlign === 'right') { return h( 'div', { class: 'q-calendar__header--inline', }, [ props.noDefaultHeaderBtn !== true && __renderHeadDayDate(day), props.noDefaultHeaderText !== true && __renderHeadWeekday(day), ], ) } else if (props.weekdayAlign === 'right' && props.dateAlign === 'left') { return h( 'div', { class: 'q-calendar__header--inline', }, [ props.noDefaultHeaderBtn !== true && __renderHeadDayDate(day), props.noDefaultHeaderText !== true && __renderHeadWeekday(day), ], ) } else { return h( 'div', { class: 'q-calendar__header--inline', }, [ props.noDefaultHeaderBtn !== true && __renderHeadDayDate(day), props.noDefaultHeaderText !== true && __renderHeadWeekday(day), ], ) } } } function __renderHeadDayEvent(day: Timestamp, columnIndex: number): VNode { const headDayEventSlot = slots['head-day-event'] const activeDate = props.noActiveDate !== true && __isActiveDate(day) const scope = { timestamp: day, activeDate, droppable: dragOverHeadDayRef.value === day.date, disabled: props.disabledWeekdays ? props.disabledWeekdays.includes(Number(day.weekday)) : false, columnIndex: columnIndex ?? 0, } const width = isSticky.value === true ? convertToUnit(parsedCellWidth.value) : computedWidth.value const style: CSSProperties = { width, maxWidth: width, minWidth: width, } if (isSticky.value === true) { style.minWidth = width } return h( 'div', { key: 'event-' + day.date + (columnIndex !== undefined ? '-' + columnIndex : ''), class: { 'q-calendar-scheduler__head--day__event': true, ...getRelativeClasses(day), 'q-active-date': activeDate, }, style, }, [headDayEventSlot && headDayEventSlot({ scope })], ) } function __renderHeadWeekday(day: Timestamp): VNode { const slot = slots['head-weekday-label'] const shortWeekdayLabel = props.shortWeekdayLabel === true // const divisor = props.dateHeader === 'inline' || props.dateHeader === 'inverted' ? 0.5 : 1 // const shortCellWidth = props.weekdayBreakpoints[ 1 ] > 0 && (parsedCellWidth.value * divisor) <= props.weekdayBreakpoints[ 1 ] const scope = { timestamp: day, shortWeekdayLabel } const data: Record = { class: { 'q-calendar-scheduler__head--weekday': true, ['q-calendar__' + props.weekdayAlign]: true, 'q-calendar__ellipsis': true, }, } return h( 'div', data, (slot && slot({ scope })) || __renderHeadWeekdayLabel(day, shortWeekdayLabel), ) } function __renderHeadWeekdayLabel(day: Timestamp, shortWeekdayLabel: boolean): VNode { const weekdayLabel = weekdayFormatter.value( day, shortWeekdayLabel || (props.weekdayBreakpoints[0] > 0 && parsedCellWidth.value <= props.weekdayBreakpoints[0]), ) return h( 'span', { class: 'q-calendar-scheduler__head--weekday-label q-calendar__ellipsis', }, props.weekdayBreakpoints[1] > 0 && parsedCellWidth.value <= props.weekdayBreakpoints[1] ? minCharWidth(weekdayLabel, Number(props.minWeekdayLabel)) : weekdayLabel, ) } function __renderHeadDayDate(day: Timestamp): VNode { const data: Record = { class: { 'q-calendar-scheduler__head--date': true, ['q-calendar__' + props.dateAlign]: true, }, } return h('div', data, __renderHeadDayBtn(day)) } function __renderHeadDayBtn(day: Timestamp): VNode { const activeDate = props.noActiveDate !== true && __isActiveDate(day) const dayLabel = dayFormatter.value(day, false) const headDayLabelSlot = slots['head-day-label'] const headDayButtonSlot = slots['head-day-button'] const scope = { dayLabel, timestamp: day, activeDate, } const data: Record = { class: { 'q-calendar-scheduler__head--day__label': true, 'q-calendar__button': true, 'q-calendar__button--round': props.dateType === 'round', 'q-calendar__button--rounded': props.dateType === 'rounded', 'q-calendar__button--bordered': day.current === true, 'q-calendar__focusable': true, }, disabled: day.disabled, onKeydown: (e: KeyboardEvent) => { if (day.disabled !== true && isKeyCode(e, [13, 32])) { e.stopPropagation() e.preventDefault() } }, onKeyup: (e: KeyboardEvent) => { // allow selection of date via Enter or Space keys if (day.disabled !== true && isKeyCode(e, [13, 32])) { emittedValue.value = day.date if (emitListeners.value.onClickDate !== undefined) { emit('click-date', { scope }) } } }, ...getDefaultMouseEventHandlers('-date', (event, eventName) => { if (eventName === 'click-date' || eventName === 'contextmenu-date') { emittedValue.value = day.date if (eventName === 'click-date') { event.preventDefault() } } return { scope, event } }), } if (props.noAria !== true) { data.ariaLabel = ariaDateFormatter.value(day, false) } return headDayButtonSlot ? h('div', [headDayButtonSlot({ scope })]) : renderButton(props, data, headDayLabelSlot ? headDayLabelSlot({ scope }) : dayLabel) } function __renderColumnHeaderBefore(day: Timestamp, columnIndex: number): VNode | void { const slot = slots['column-header-before'] if (slot) { const scope = { timestamp: day, columnIndex } return h( 'div', { class: 'q-calendar-scheduler__column-header--before', }, [slot({ scope })], ) } } function __renderColumnHeaderAfter(day: Timestamp, columnIndex: number): VNode | void { const slot = slots['column-header-after'] if (slot) { const scope = { timestamp: day, columnIndex } return h( 'div', { class: 'q-calendar-scheduler__column-header--after', }, [slot({ scope })], ) } } function __renderBody(): VNode { return h( 'div', { class: 'q-calendar-scheduler__body', }, [__renderScrollArea()], ) } function __renderScrollArea(): VNode { if (isSticky.value === true) { return h( 'div', { ref: scrollArea, class: { 'q-calendar-scheduler__scroll-area': true, 'q-calendar__scroll': true, }, }, /// @ts-expect-error fix later [isSticky.value !== true && __renderDayResources(), __renderDayContainer()], ) } else if (props.noScroll === true) { return __renderPane() } else { return h( 'div', { ref: scrollArea, class: { 'q-calendar-scheduler__scroll-area': true, 'q-calendar__scroll': true, }, }, [__renderPane()], ) } } function __renderPane(): VNode { return h( 'div', { ref: pane, class: 'q-calendar-scheduler__pane', }, [__renderDayContainer()], ) } function __renderDayContainer(): VNode { return h( 'div', { class: 'q-calendar-scheduler__day--container', }, [isSticky.value === true && props.noHeader !== true && __renderHead(), __renderResources()], ) } function __renderResources( resources: Resource[] | undefined = undefined, indentLevel = 0, expanded = true, ): VNode[] { if (resources === undefined) { resources = props.modelResources } return (resources as Resource[]) .map((resource, resourceIndex) => { return __renderResourceRow( resource, resourceIndex, indentLevel, resource.children !== undefined ? resource.expanded : expanded, ) }) .flat() } function __renderResourceRow( resource: Resource, resourceIndex: number, indentLevel = 0, expanded = true, ): VNode[] { const style: CSSProperties = {} style.height = resource.height !== void 0 ? convertToUnit(parseInt(resource.height, 10)) : parsedResourceHeight.value ? convertToUnit(parsedResourceHeight.value) : 'auto' if (parsedResourceMinHeight.value > 0) { style.minHeight = convertToUnit(parsedResourceMinHeight.value) } const resourceRow = h( 'div', { key: resource[props.resourceKey] + '-' + resourceIndex, class: { 'q-calendar-scheduler__resource--row': true, }, style, }, [ __renderResource(resource, resourceIndex, indentLevel, expanded), __renderDayResources(resource, resourceIndex, indentLevel, expanded), ], ) if (resource.children !== undefined) { return [ resourceRow, h( 'div', { class: { 'q-calendar__child': true, 'q-calendar__child--expanded': expanded === true, 'q-calendar__child--collapsed': expanded !== true, }, }, [ __renderResources( resource.children, indentLevel + 1, expanded === false ? expanded : resource.expanded, ), ], ), ] } return [resourceRow] } function __renderResource( resource: Resource, resourceIndex: number, indentLevel = 0, expanded = true, ): VNode { const slotResourceLabel = slots['resource-label'] const resourceMinHeight = parseInt(String(props.resourceMinHeight), 10) const style: CSSProperties = {} style.height = resource.height !== void 0 ? convertToUnit(parseInt(resource.height, 10)) : parsedResourceHeight.value ? convertToUnit(parsedResourceHeight.value) : 'auto' if (resourceMinHeight > 0) { style.minHeight = convertToUnit(resourceMinHeight) } const styler = props.resourceStyle || styleDefault const label = resource[props.resourceLabel] const isFocusable = props.focusable === true && props.focusType.includes('resource') && expanded === true const dragValue = resource[props.resourceKey] const scope = { resource, timestamps: days.value, days: days.value, // deprecated resourceIndex, indentLevel, label, droppable: dragOverResource.value === dragValue, } const resourceClass = typeof props.resourceClass === 'function' ? props.resourceClass({ scope }) : {} return h( 'div', { key: resource[props.resourceKey] + '-' + resourceIndex, ref: (el) => { if (el !== null) { /// @ts-expect-error fix later resourcesRef.value[resource[props.resourceKey]] = el } }, tabindex: isFocusable === true ? 0 : -1, class: { 'q-calendar-scheduler__resource': indentLevel === 0, 'q-calendar-scheduler__resource--section': indentLevel !== 0, ...resourceClass, 'q-calendar__sticky': isSticky.value === true, 'q-calendar__hoverable': props.hoverable === true, 'q-calendar__focusable': isFocusable === true, }, style: { ...style, /// @ts-expect-error fix later ...styler({ scope }), }, onDragenter: (e: DragEvent) => { if (props.dragEnterFunc !== undefined && typeof props.dragEnterFunc === 'function') { props.dragEnterFunc(e, 'resource', { scope }) === true ? (dragOverResource.value = dragValue) : (dragOverResource.value = '') } }, onDragover: (e: DragEvent) => { if (props.dragOverFunc !== undefined && typeof props.dragOverFunc === 'function') { props.dragOverFunc(e, 'resource', { scope }) === true ? (dragOverResource.value = dragValue) : (dragOverResource.value = '') } }, onDragleave: (e: DragEvent) => { if (props.dragLeaveFunc !== undefined && typeof props.dragLeaveFunc === 'function') { props.dragLeaveFunc(e, 'resource', { scope }) === true ? (dragOverResource.value = dragValue) : (dragOverResource.value = '') } }, onDrop: (e: DragEvent) => { if (props.dropFunc !== undefined && typeof props.dropFunc === 'function') { props.dropFunc(e, 'resource', { scope }) === true ? (dragOverResource.value = dragValue) : (dragOverResource.value = '') } }, onKeydown: (event: KeyboardEvent) => { if (isKeyCode(event, [13, 32])) { event.stopPropagation() event.preventDefault() } }, onKeyup: (event: KeyboardEvent) => { // allow selection of resource via Enter or Space keys if (isKeyCode(event, [13, 32])) { if (emitListeners.value.onClickResource !== undefined) { emit('click-resource', { scope, event }) } } }, ...getDefaultMouseEventHandlers('-resource', (event) => { return { scope, event } }), }, [ [ h('div', { class: { 'q-calendar__parent': resource.children !== undefined, 'q-calendar__parent--expanded': resource.children !== undefined && resource.expanded === true, 'q-calendar__parent--collapsed': resource.children !== undefined && resource.expanded !== true, }, onClick: (e) => { e.stopPropagation() resource.expanded = !resource.expanded // emit('update:model-resources', props.modelResources) emit('resource-expanded', { expanded: resource.expanded, scope }) }, }), h( 'div', { class: { 'q-calendar-scheduler__resource--text': true, 'q-calendar__ellipsis': true, }, style: { paddingLeft: 10 * indentLevel + 2 + 'px', }, }, [slotResourceLabel ? slotResourceLabel({ scope }) : label], ), useFocusHelper(), ], ], ) } function __renderDayResources( resource: Resource, resourceIndex: number, indentLevel = 0, expanded = true, ): VNode { const slot = slots['resource-days'] const width = isSticky.value === true ? convertToUnit(parsedCellWidth.value) : computedWidth.value const scope = { resource, resourceIndex, indentLevel, expanded, cellWidth: width, timestamps: days.value, days: days.value, // deprecated } const style: CSSProperties = {} style.height = parseInt(String(props.resourceHeight), 10) > 0 ? convertToUnit(parseInt(String(props.resourceHeight), 10)) : 'auto' if (parseInt(String(props.resourceMinHeight), 10) > 0) { style.minHeight = convertToUnit(parseInt(String(props.resourceMinHeight), 10)) } const data: Record = { class: 'q-calendar-scheduler__resource--days', style, } return h('div', data, [ ...__renderDays(resource, resourceIndex, indentLevel, expanded), slot && slot({ scope }), ]) } function __renderDays( resource: Resource, resourceIndex: number, indentLevel = 0, expanded = true, ): VNode[] { if (days.value.length === 1 && parseInt(String(props.columnCount), 10) > 0) { return Array.apply(null, new Array(parseInt(String(props.columnCount), 10))) .map((_, i) => i + parseInt(String(props.columnIndexStart), 10)) .map((columnIndex) => __renderDay(days.value[0], columnIndex, resource, resourceIndex, indentLevel, expanded), ) } else { return days.value.map((day) => __renderDay(day, 0, resource, resourceIndex, indentLevel, expanded), ) } } function __renderDay( day: Timestamp, columnIndex: number, resource: Resource, resourceIndex: number, indentLevel = 0, expanded = true, ): VNode { const slot = slots.day const styler = props.dayStyle || dayStyleDefault const activeDate = props.noActiveDate !== true && parsedValue.value.date === day.date const dragValue = day.date + ':' + resource[props.resourceKey] + (columnIndex !== undefined ? ':' + columnIndex : '') const droppable = dragOverResource.value === dragValue const scope = { timestamp: day, columnIndex, resource, resourceIndex, indentLevel, activeDate, droppable, } const width = isSticky.value === true ? convertToUnit(parsedCellWidth.value) : computedWidth.value const style: CSSProperties = { width, maxWidth: width, ...styler({ scope }), } style.height = parseInt(String(props.resourceHeight), 10) > 0 ? convertToUnit(parseInt(String(props.resourceHeight), 10)) : 'auto' if (parseInt(String(props.resourceMinHeight), 10) > 0) { style.minHeight = convertToUnit(parseInt(String(props.resourceMinHeight), 10)) } const dayClass = typeof props.dayClass === 'function' ? props.dayClass({ scope }) : {} const isFocusable = props.focusable === true && props.focusType.includes('day') && expanded === true return h( 'div', { key: day.date + (columnIndex !== undefined ? ':' + columnIndex : ''), tabindex: isFocusable === true ? 0 : -1, class: { 'q-calendar-scheduler__day': indentLevel === 0, 'q-calendar-scheduler__day--section': indentLevel !== 0, ...dayClass, ...getRelativeClasses(day), 'q-calendar__hoverable': props.hoverable === true, 'q-calendar__focusable': isFocusable === true, }, style, onDragenter: (e: DragEvent) => { if (props.dragEnterFunc !== undefined && typeof props.dragEnterFunc === 'function') { props.dragEnterFunc(e, 'day', { scope }) === true ? (dragOverResource.value = dragValue) : (dragOverResource.value = '') } }, onDragover: (e: DragEvent) => { if (props.dragOverFunc !== undefined && typeof props.dragOverFunc === 'function') { props.dragOverFunc(e, 'day', { scope }) === true ? (dragOverResource.value = dragValue) : (dragOverResource.value = '') } }, onDragleave: (e: DragEvent) => { if (props.dragLeaveFunc !== undefined && typeof props.dragLeaveFunc === 'function') { props.dragLeaveFunc(e, 'day', { scope }) === true ? (dragOverResource.value = dragValue) : (dragOverResource.value = '') } }, onDrop: (e: DragEvent) => { if (props.dropFunc !== undefined && typeof props.dropFunc === 'function') { props.dropFunc(e, 'day', { scope }) === true ? (dragOverResource.value = dragValue) : (dragOverResource.value = '') } }, onKeydown: (event: KeyboardEvent) => { if (isKeyCode(event, [13, 32])) { event.stopPropagation() event.preventDefault() } }, onKeyup: (event: KeyboardEvent) => { // allow selection of date via Enter or Space keys if (isKeyCode(event, [13, 32])) { emittedValue.value = scope.timestamp.date if (emitListeners.value.onClickResource !== undefined) { emit('click-resource', { scope, event }) } } }, ...getDefaultMouseEventHandlers('-day-resource', (event) => { return { scope, event } }), }, [slot && slot({ scope }), useFocusHelper()], ) } function __renderResourcesError(): VNode { return h('div', {}, 'No resources have been defined') } function __renderScheduler(): VNode { const { start, end, maxDays } = renderValues.value if ( startDate.value !== start.date || endDate.value !== end.date || maxDaysRendered.value !== maxDays ) { startDate.value = start.date endDate.value = end.date maxDaysRendered.value = maxDays } const hasWidth = size.width > 0 const hasResources = props.modelResources && props.modelResources.length > 0 const scheduler = withDirectives( h( 'div', { key: startDate.value, class: 'q-calendar-scheduler', }, [ hasWidth === true && hasResources === true && isSticky.value !== true && props.noHeader !== true && __renderHead(), hasWidth === true && hasResources === true && __renderBody(), hasResources === false && __renderResourcesError(), ], ), [[ResizeObserver, __onResize]], ) if (props.animated === true) { const transition = 'q-calendar--' + (direction.value === 'prev' ? props.transitionPrev : props.transitionNext) return h( Transition, { name: transition, appear: true, }, () => scheduler, ) } return scheduler } // expose public methods expose({ prev, next, move, moveToToday, updateCurrent, }) // Object.assign(vm.proxy, { // prev, // next, // move, // moveToToday, // updateCurrent // }) return (): VNode => __renderCalendar() }, })