script.js

/**
 * Version 1.8
 **/
"use strict";

renderTimeline();

async function renderTimeline() {
    // Initialize libraries and context
    const vis = require('vis.min.js');
    const container = document.getElementById("timeline");
    const menu = document.getElementById("menu");

    // Get widget attribute values and fetch notes from defined tags
    const events = await api.runOnBackend(() => {
        const parentNote = api.startNote.getParentNotes()[0];
        // fetch what labels to look for from widget attributes
        // event notes (start/end)
        const event_note_label_start = parentNote.getLabelValue('event_label_start').toString();
        const event_note_label_end = parentNote.getLabelValue('event_label_end').toString();
        // agent notes (birth/death)
        const person_note_label_start = parentNote.getLabelValue('person_label_start').toString();
        const person_note_label_end = parentNote.getLabelValue('person_label_end').toString();
        // date types (categories)
        const event_label_type = parentNote.getLabelValue('event_label_type').toString();

        // fetch notes with those labels
        const event_notes = api.getNotesWithLabel(event_note_label_start);
        const person_notes_birth = api.getNotesWithLabel(person_note_label_start);
        const person_notes_death = api.getNotesWithLabel(person_note_label_end);
        let events = [];
        for (let note of event_notes) {
            let id = note.noteId;
            let content = note.title;
            let start = note.getLabelValue(event_note_label_start);
            let end = note.getLabelValue(event_note_label_end);
            let group = note.getLabelValue(event_label_type) ? note.getLabelValue(event_label_type) : (parentNote.getLabelValue('event_type_default') ? parentNote.getLabelValue('event_type_default') : 'default');
            if (content && start) {
                events.push({id, content, start, end, group});
            }
        }
        // make a 'birth' event for each person note with start date
        for (let note of person_notes_birth) {
            let id = note.noteId;
            let start = note.getLabelValue(person_note_label_start);
            let group = note.getLabelValue(event_label_type) ? note.getLabelValue(event_label_type) : (parentNote.getLabelValue('event_type_default') ? parentNote.getLabelValue('event_type_default') : 'default');
            if (start) {
                let content = note.title + ' (Birth)';
                events.push({id, content, start, group});
            }
        }
        // make a 'death' event for each person note with end date
        for (let note of person_notes_death) {
            let id = note.noteId;
            let start = note.getLabelValue(person_note_label_end);
            let group = note.getLabelValue(event_label_type) ? note.getLabelValue(event_label_type) : (parentNote.getLabelValue('event_type_default') ? parentNote.getLabelValue('event_type_default') : 'default');
            if (start) {
                let content = note.title + ' (Death)';
                events.push({id, content, start, group});
            }
        }
        return events;
    });

    // Get event types, make groups
    const types = await api.runOnBackend(() => {
        const parentNote = api.startNote.getParentNotes()[0];
        const event_type_list = parentNote.getLabelValue('event_type_list').toString().split(';');
        let types = [];
        let i = 1;
        for (let type of event_type_list) {
            // type format each type seperated by ;
            // id(string), order(int), label(string), color(string), visible(bool), type(string), forceGroup(int)
            types.push({
                id: type.split(',')[0].toString(),
                order: type.split(',')[1] ? (parseInt(type.split(',')[1], 10)) : i,
                content: type.split(',')[2] ? (type.split(',')[2].toString()) : type.split(',')[0].toString(),
                color: type.split(',')[3] ? (type.split(',')[3].toString()) : null,
                visible: type.split(',')[4] ? (type.split(',')[4] === 'true') : true,
                type: type.split(',')[5] ? type.split(',')[5].toString() : null,
                forceGroup: type.split(',')[6] ? (type.split(',')[6].toString()) : null
            });
            i++;
        }
        return types
    });

    // merge the default group with attribute-added groups
    let groups = new vis.DataSet([{
        id: 'default', order: 100, content: '', color: 'white', visible: true, type: null, forceGroup: null
    }, ...types]);

    // Create a DataSet for all event items
    let items = new vis.DataSet();
    for (let i = 0; i < events.length; i++) {
        // get date values and note link
        let note_id = events[i].id;
        let note_link = await api.createNoteLink(note_id);
        note_link[0].firstChild.innerText = events[i].content;
        let event_start = vis.moment(new Date(events[i].start), 'YYYY-MM-DD-hh:mm:ss');
        let event_end = events[i].end ? vis.moment(new Date(events[i].end), 'YYYY-MM-DD-hh:mm:ss') : null;
        // set event group
        let event_group = events[i].group ? (groups.get({
            filter: function (item) {
                return (item.id === events[i].group)
            }
        })[0] ? groups.get({
            filter: function (item) {
                return (item.id === events[i].group)
            }
        })[0] : groups.get({
            filter: function (item) {
                return (item.id === 'default')
            }
        })[0]) : groups.get({
            filter: function (item) {
                return (item.id === 'default')
            }
        })[0];
        // fill event data
        items.add({
            id: i,
            content: note_link[0].firstChild,
            start: event_start,
            end: event_end,
            group: event_group.forceGroup ? event_group.forceGroup : event_group.id,
            type: (event_group.type === 'box' || event_group.type === 'point' || event_group.type === 'range' || event_group.type === 'background') ? event_group.type : null,
            className: event_group.id,
            style: `background-color:${event_group.color};border-color:${event_group.color};`
        });
    }

    // Timeline options
    const timeline_options = await api.runOnBackend(() => {
        const parentNote = api.startNote.getParentNotes()[0];
        const timeline_start = parentNote.getLabelValue('timeline_start');
        const timeline_end = parentNote.getLabelValue('timeline_end');
        const timeline_present = parentNote.getLabelValue('timeline_present');
        const can_click_day = parentNote.getLabelValue('can_click_day') === 'true';
        return {timeline_start, timeline_end, timeline_present, can_click_day};
    });
    console.log(timeline_options.can_click_day);
    let options = {
        clickToUse: false,
        showCurrentTime: true,
        height: '95%',
        start: timeline_options.timeline_start,
        end: timeline_options.timeline_end
    };

    // Set timeline wrapper height
    container.parentNode.style.height = '100%';
    container.parentNode.parentNode.style.height = '100%';
    // Fix timeline display
    container.parentNode.parentNode.parentNode.parentNode.style.display = 'initial';
    // Create Timeline
    const timeline = new vis.Timeline(container, items, groups, options);
    // Add present time marker based on "timeline_present" label
    timeline.setCurrentTime(timeline_options.timeline_present);
    
    // Add event listener to activate a day note on click
    if(timeline_options.can_click_day) {
        timeline.on("click", async (e) => {
            const timelineHeight = timeline.dom.centerContainer.getBoundingClientRect().height;
            if (e.y <= timelineHeight) {
                // ignore clicks on the timeline
                return;
            }
            const effectiveY = e.y - timelineHeight;
            if (effectiveY > timeline.dom.bottom.children[0].children[0].getBoundingClientRect().height) {
                // ignore clicks on the month axis
                return;
            }
            const day = e.time.toISOString().substr(0, 10);
            const todayNote = await api.getDayNote(day);
            api.waitUntilSynced();
            await api.openTabWithNote(todayNote.noteId, true);
        });
    }

    // Create UI menu buttons from added groups (that are not forced into another group)
    menu.innerHTML = '';
    let toggle_groups = groups.get({
        filter: function (item) {
            return (item.forceGroup === null && item.id !== 'default')
        }
    });
    for (let group of toggle_groups) {
        let button = document.createElement('input');
        button.type = 'button';
        button.id = `toggle-${group.id}`;
        button.value = group.content;
        button.className = (group.visible === true) ? 'toggled' : '';
        menu.appendChild(button);
    }

    // Bind toggle buttons
    for (let group of toggle_groups) {
        document.getElementById(`toggle-${group.id}`).onclick = function () {
            toggleGroupVisibility(group.id);
        }
    }

    // Function to toggle group visibility
    function toggleGroupVisibility(group_id) {
        let visibility = groups.get({
            filter: function (item) {
                return (item.id === group_id)
            }
        })[0].visible;
        groups.update({id: group_id, visible: !visibility});
        console.log(document.getElementById(`toggle-${group_id}`));
        document.getElementById(`toggle-${group_id}`).className = (!visibility === true) ? 'toggled' : '';
        timeline.setGroups(groups);
        timeline.redraw();
    }
}