自定义 fullCalendar js 的“显示更多”链接行为

Customize fullCalendar js 'show more' link behaviour

提问人:Hajji Achref 提问时间:10/20/2023 更新时间:10/22/2023 访问量:66

问:

我正在将fullCalendar V6.1.8(https://fullcalendar.io/)集成到我的应用程序中。 在“dayGrid”视图中,我想自定义“显示更多”按钮的行为(默认情况下会打开一个模式)

function setView() {
    const width = $(window).width();
    if (width < 768) {
        return 'dayGrid';
    } else {
        return 'timeGridWeek';
    }
}

function setDayHeaderFormat() {
    const width = $(window).width();
    if (width < 768) {
        return { weekday: 'long', day: 'numeric', month: 'long', year: 'numeric', omitCommas: true };
    } else {
        return {weekday: 'short', day: 'numeric', month: 'numeric' };
    }
}

function setMoreLinkContent(currentLanguageCode) {
    if (currentLanguageCode == 'de') {
        return 'mehr slots';
    }
    else if (currentLanguageCode == 'en') {
        return 'more slots';
    }
    else {
        return '+ de créneaux';
    }
}

$(document).bind("jFormEvents.init", function (e, online) {


    var calendarDiv = $("div.Content.calendar");
    var recapPage = $("h3.validationPage");

    if ($(calendarDiv).length && !$(recapPage).length) {

        var lang = $("input[name='rdv|language;string']").val() ? $("input[name='rdv|language;string']").val() : 'FR';
        var duration = parseInt($("input[name='campaign|duration;string']").val());

        var guCalendar = new GuFullCalendar.GuCalendar(calendarDiv.get(0), {
            campaignCodeEl: 'input[name="campaign|code;string"]',
            addressCodeEl: 'input[name="address|code;string"]',
            durationEl: 'input[name="campaign|duration;string"]',

            rdvDateEl: 'input[name="rdv|date;string"]',
            rdvHourEl: 'input[name="rdv|hour;string"]',
            rdvDurationEl: 'input[name="rdv|duration;string"]',
            useModal: true,

            mainZoneEl: '#main_zone',

            calendarInfoEl: '.calendar-infos',

            canAddEvent: function(date) {
                var start=date.date + 'T' + '00:00:00';
                var dateMeeting = Date.parse(start);
                var todayAtMidnight = new Date();
                todayAtMidnight.setHours(0,0,0,0);
                var compareDate = dateMeeting - todayAtMidnight.getTime();
                var dateApply = compareDate / (1000 * 60 * 60);

                var todayAt20 = new Date();
                todayAt20.setHours(20,0,0,0);
                var now = new Date();
                
                var nbDaySupp = 1;
                if (now >= todayAt20) {
                    nbDaySupp = 2;
                }
                
                return dateApply >= (24*nbDaySupp);
            },

            locale: lang,

            /*
            headerToolbar: {
                start: 'prev,next today',
                center: 'title',
                end: ''
            },
             */
            headerToolbar: {
                start: 'prev',
                center: 'today',
                end: 'next'
            },
            // Set initial view depending on the screen size
            initialView: setView(),
            height: 'auto',
            dayMaxEvents: 5,
            editable: false,
            eventLimit: true, // allow "more" link when too many events
            timeFormat: '',
            displayEventTime: true,
            //initialDate: today,
            slotDuration: {
                minutes: duration
            }, // The frequency for displaying time slots.
            slotLabelInterval: {
                minutes: (duration < 20 ? 30 : 60)
            }, // The frequency that the time slots should be labelled with text.
            allDaySlot: false, // Determines if the “all-day” slot is displayed at the top of the calendar.
            slotEventOverlap: false, // Determines if timed events in agenda view should visually overlap.
            forceEventDuration: true,
            defaultTimedEventDuration: {
                minutes: duration
            },
            eventTimeFormat: { // like '14:30:00'
                hour: '2-digit',
                minute: '2-digit',
                meridiem: false
            },
            displayEventEnd: true,
            listDayFormat: {
                year: 'numeric',
                month: 'long',
                day: 'numeric',
                weekday: 'long'
            },
            listDaySideFormat: false,
            slotMinTime: "08:00:00",
            slotMaxTime: "19:00:00",
            weekends: false,
            navLinks: true,
            listDayAltFormat: false,
            dayHeaderFormat: setDayHeaderFormat(),
            moreLinkContent: setMoreLinkContent(),
            moreLinkClick: function() {
                console.log(this)
                this.setOption('dayMaxEvents', 10);
            },
            //Change view when resizing the window
            windowResize: function() {
                console.log(this)
                this.changeView(setView());
                this.setOption('dayHeaderFormat', setDayHeaderFormat());
            }
        });

        guCalendar.render();
        
    }
});

这是我创建的配置。根据官方文档,我使用了“moreLinkClick”属性,该属性可以接受回调函数来自定义其行为。这里的问题是关键字“this”引用的是窗口对象而不是日历,所以我收到一个错误,说“setOption”不是一个函数。 Rq:你可以看到我在'windowResize'属性中使用了相同的方法,它工作得很好('this'引用了日历)

我的主要目标是覆盖默认行为,该行为会打开一个包含当天所有时间段的模式。所需的行为是显示日历本身中的所有插槽,而不是以模式显示

我试过箭头功能

moreLinkClick: () => {
                console.log(this)
                this.setOption('dayMaxEvents', 10);
            },

并尝试将其绑定到 Calnedar:

moreLinkClick: function() {
                console.log(this)
                this.setOption('dayMaxEvents', 10);
            }.bind(guCalendar),

但两者都没有解决问题

javascript jquery 这个 fullcalendar-6

评论

1赞 CBroe 10/20/2023
moreLink 要么完全转到不同的视图,要么应该显示当天的事件 - 因此开发人员可能认为没有必要为该特定回调提供实际的 fullcalendar 实例。日期及其所有事件数据都会被传入,因此无需访问此按钮的预期目的的任何其他内容。您试图在这里使用它来修改当前日期范围之外的整个日历实例的行为方式,并且它并不是真正用于...
0赞 Hajji Achref 10/20/2023
@CBroe 感谢您对此问题的技术原因的详细说明。那么,您有什么解决方法可以获得我在问题中提到的所需行为吗?

答:

0赞 ADyson 10/22/2023 #1

您可以只引用 calendar 对象而不是 .this

如果您返回已使用视图的名称,这将停止弹出窗口的显示,而是日历将导航到该视图(当然,在这种情况下,这意味着没有更改,因为这已经是选定的视图)。

moreLinkClick: function() {
  guCalendar.setOption('dayMaxEvents', 10);
  return "dayGridMonth";
},

演示:https://codepen.io/ADyson82/pen/dyayzVp

参考资料: https://fullcalendar.io/docs/moreLinkClick

评论

0赞 Hajji Achref 10/23/2023
感谢您的回复,但我已经尝试过了,但出现错误,说guCalendar.setOption不是函数。
0赞 ADyson 10/24/2023
我不确定这是怎么回事。如您所见,在我的演示中,它工作正常。您是否以同样的方式实现它?