Search code examples
javascriptreactjsfullcalendarignition

FullCalendar React timeGridWeek broken


I've tried to turn Fullcalendar's React component into a component in Inductive Automation's Ignition platform. This has gone relatively smoothly and everything renders properly, except for the timeGridWeek.

Whenever I try to navigate to the timeGridWeek the console spits out an error that doesn't mean much to me and the entire calendar stops responding. This is the error in question:

RadComponents.js?no_…e=1579268884233:960 Uncaught TypeError: Cannot read property 'years' of null
at wholeDivideDurations (RadComponents.js?no_…e=1579268884233:960)
at TimeGrid.../../node_modules/@fullcalendar/timegrid/main.esm.js.TimeGrid.processOptions (RadComponents.js?no_…1579268884233:13228)
at new TimeGrid (RadComponents.js?no_…1579268884233:13207)
at TimeGridView [as constructor] (RadComponents.js?no_…1579268884233:13757)
at new TimeGridView (RadComponents.js?no_…1579268884233:14010)
at CalendarComponent.../../node_modules/@fullcalendar/core/main.esm.js.CalendarComponent.renderView (RadComponents.js?no_…=1579268884233:6501)
at CalendarComponent.../../node_modules/@fullcalendar/core/main.esm.js.CalendarComponent.render (RadComponents.js?no_…=1579268884233:6453)
at CalendarComponent.../../node_modules/@fullcalendar/core/main.esm.js.Component.receiveProps (RadComponents.js?no_…=1579268884233:4138)
at Calendar.../../node_modules/@fullcalendar/core/main.esm.js.Calendar.renderComponent (RadComponents.js?no_…=1579268884233:7068)
at Calendar.../../node_modules/@fullcalendar/core/main.esm.js.Calendar.executeRender (RadComponents.js?no_…=1579268884233:7025)

Now the reason the error doesn't mean much to me is because it seems to be happening within the TimeGridPlugin itself, which I don't think I have access to.

This is my code:

import * as React from 'react';
import { Component, ComponentMeta, ComponentProps, SizeObject } from '@inductiveautomation/perspective-client';
import { Calendar } from '@fullcalendar/core';
import FullCalendar from '@fullcalendar/react';
import timeGridPlugin from '@fullcalendar/timegrid';
import dayGridPlugin from '@fullcalendar/daygrid';
import interactionPlugin from '@fullcalendar/interaction';

// the 'key' or 'id' for this component type.  Component must be registered with this EXACT key in the Java side as well
// as on the client side.  In the client, this is done in the index file where we import and register through the
// ComponentRegistry provided by the perspective-client API.
export const COMPONENT_TYPE = "react-display-calendar";

// this is the shape of the properties we get from the perspective 'props' property tree.
export interface ReactCalendarProps {
    editable: boolean;
    selectable: boolean;
    defaultView: string;
    events: Array<object>;
    firstDay: number;
    eventLimit: boolean | number;
    minTime: string;
    maxTime: string;
    weekNumbers: boolean;
    weekNumbersWithinDays: boolean;
    nowIndicator: boolean;
    slotDuration: string;
    allDayText: string;
    longPressDelay: number;
}


export class ReactCalendar extends Component<ComponentProps, any> {
    calendar: Calendar;
    private calendarComponentRef  = React.createRef<any>();

    render() {
        const {props} = this.props;
        const allDayTextProp = props.readString('allDayText');
        const eventsProp = props.readArray('events');
        const editableProp = props.readBoolean('editable');
        const selectableProp = props.readBoolean('selectable');
        const firstDayProp = props.readNumber('firstDay', 1);
        const eventLimitProp = props.read('eventLimit');
        const weekNumbersProp = props.readBoolean('weekNumbers');
        const weekNumbersWithinDaysProp = props.readBoolean('weekNumbersWithinDays');
        const nowIndicatorProp = props.readBoolean('nowIndicator');
        const defaultViewProp = props.readString('defaultView');
        const minTimeProp = props.readString('minTime');
        const maxTimeProp = props.readString('maxTime');
        const slotDurationProp = props.readString('slotDuration');
        const longPressDelayProp = props.readNumber('longPressDelay', 1000);

        const pluginsProp = [dayGridPlugin, timeGridPlugin, interactionPlugin];
        const headerProp = { left: 'dayGridMonth,timeGridWeek,timeGridDay', center: 'title', right: 'today, ,prev,next' };
        const eventTimeFormatProp = { hour: '2-digit', minute: '2-digit', meridiem: false, hour12: false };
        const slotLabelFormatProp = { hour: '2-digit', minute: '2-digit', meridiem: false, hour12: false };

        return(
            <FullCalendar
                {...this.props.emit()}
                ref={this.calendarComponentRef}
                plugins={pluginsProp}
                header={headerProp}
                eventTimeFormat={eventTimeFormatProp}
                slotLabelFormat={slotLabelFormatProp}
                defaultView={defaultViewProp}
                height="parent"
                contentHeight="auto"
                allDayText={allDayTextProp}
                events={eventsProp}
                editable={editableProp}
                eventStartEditable={editableProp}
                eventDurationEditable={editableProp}
                selectable={selectableProp}
                firstDay={firstDayProp}
                eventLimit={eventLimitProp}
                weekNumbers={weekNumbersProp}
                weekNumbersWithinDays={weekNumbersWithinDaysProp}
                nowIndicator={nowIndicatorProp}
                minTime={minTimeProp}
                maxTime={maxTimeProp}
                slotDuration={slotDurationProp}
                longPressDelay={longPressDelayProp}

                select={this.handleSelect}
                windowResize={this.handleWindowResize}
                eventDrop={this.handleEventDrop}
                eventClick={this.handleEventClick}
                eventResize={this.handleEventResize}
            />
        );
    }

    handleSelect = (arg: any) => {
        let selectStart = arg.start.valueOf() / 1000 + 3600;
        let selectEnd = arg.end.valueOf() / 1000 + 3600;

        if (!isNaN(arg.start.valueOf())) {
            this.props.componentEvents.fireComponentEvent("onSelectionMade", {start:  selectStart, end: selectEnd});
        } else {
            alert('Invalid date.');
        }
    }

    handleWindowResize = (arg: any) => {
        this.props.componentEvents.fireComponentEvent("onWindowResize", {arg});
    }

    handleEventDrop = (arg: any) => {
        if (arg.event.start != null && arg.event.end != null) {
            let eventId = arg.event.id;
            let startTime = arg.event.start.valueOf() / 1000 + 3600;
            let endTime = arg.event.end.valueOf() / 1000 + 3600;
            this.props.componentEvents.fireComponentEvent("onEventDropped", {
                id: eventId,
                start: startTime,
                end: endTime
            });
        }
    }

    handleEventClick = (arg: any) => {
        let title = arg.event.title;
        let id = arg.event.id;

        this.props.componentEvents.fireComponentEvent("onEventClick", {
            title,
            id
        });
    }

    handleEventResize = (arg: any) => {
        if (arg.event.start != null && arg.event.end != null) {
            let eventId = arg.event.id;
            let startTime = arg.event.start.valueOf() / 1000 + 3600;
            let endTime = arg.event.end.valueOf() / 1000 + 3600;
            this.props.componentEvents.fireComponentEvent("onEventResized", {
                id: eventId,
                start: startTime,
                end: endTime
            });
        }
    }

}


// this is the actual thing that gets registered with the component registry
export class ReactCalendarMeta implements ComponentMeta {

    getComponentType(): string {
        return COMPONENT_TYPE;
    }

    // the class or React Type that this component provides
    getViewClass(): React.ReactType {
        return ReactCalendar;
    }

    getDefaultSize(): SizeObject {
        return ({
            width: 360,
            height: 360
        });
    }
}

Has anyone encountered this problem before? Any ideas are appreciated!


Solution

  • So, it turns out the React version requires a default value for slotDuration. Otherwise tonnes of values will be undefined or null during the first render of the timeGridWeek, breaking the calendar.