UNPKG

js-add-to-calendar-buttons

Version:

A vanilla javascript lib to create add to calendar buttons

437 lines (354 loc) 18.9 kB
(function(exports) { /* -------------- config --------------- */ var MS_IN_MINUTES = 60 * 1000; var CONFIG = { selector : ".add-to-calendar", duration : 60, texts : { title : "New event", download: "Calendar-event.ics", google : "Google Calendar", yahoo : "Yahoo! Calendar", off365 : "Office 365", ical : "Download iCal", outlook : "Download Outlook" } }; if (typeof ADDTOCAL_CONFIG != "undefined") { CONFIG = ADDTOCAL_CONFIG; } /* -------------- generators --------------- */ var calendarGenerators = { google: function(event) { var startTime,endTime; if (event.allday) { // google wants 2 consecutive days at 00:00 startTime = formatTime(event.tzstart); endTime = formatTime(getEndDate(event.tzstart,60*24)); startTime = stripISOTime(startTime); endTime = stripISOTime(endTime); } else { if (event.timezone) { // google is somehow weird with timezones. // it works better when giving the local // time in the given timezone without the zulu, // and pass timezone as argument. // but then the dates we have loaded // need to shift inverse with tzoffset the // browser gave us. // so var shiftstart, shiftend; shiftstart = new Date(event.start.getTime()-event.start.getTimezoneOffset()*MS_IN_MINUTES); if (event.end) { shiftend = new Date(event.end.getTime()-event.end.getTimezoneOffset()*MS_IN_MINUTES); } startTime = formatTime(shiftstart); endTime = formatTime(shiftend); // strip the zulu and pass the tz as argument later startTime = startTime.substring(0,startTime.length-1); endTime = endTime.substring(0,endTime.length-1); } else { // use regular times startTime = formatTime(event.start); endTime = formatTime(event.end); } } var href = encodeURI([ 'https://www.google.com/calendar/render', '?action=TEMPLATE', '&text=' + (event.title || ''), '&dates=' + (startTime || ''), '/' + (endTime || ''), (event.timezone)?'&ctz='+event.timezone:'', '&details=' + (event.description || ''), '&location=' + (event.address || ''), '&sprop=&sprop=name:' ].join('')); return '<a class="icon-google" target="_blank" href="' + href + '">'+CONFIG.texts.google+'</a>'; }, yahoo: function(event) { if (event.allday) { var yahooEventDuration = 'allday'; } else { var eventDuration = event.tzend ? ((event.tzend.getTime() - event.tzstart.getTime())/ MS_IN_MINUTES) : event.duration; // Yahoo dates are crazy, we need to convert the duration from minutes to hh:mm var yahooHourDuration = eventDuration < 600 ? '0' + Math.floor((eventDuration / 60)) : Math.floor((eventDuration / 60)) + ''; var yahooMinuteDuration = eventDuration % 60 < 10 ? '0' + eventDuration % 60 : eventDuration % 60 + ''; var yahooEventDuration = yahooHourDuration + yahooMinuteDuration; } // Remove timezone from event time // var st = formatTime(new Date(event.start - (event.start.getTimezoneOffset() * MS_IN_MINUTES))) || ''; var st = formatTime(event.tzstart) || ''; var href = encodeURI([ 'http://calendar.yahoo.com/?v=60&view=d&type=20', '&title=' + (event.title || ''), '&st=' + st, '&dur=' + (yahooEventDuration || ''), '&desc=' + (event.description || ''), '&in_loc=' + (event.address || '') ].join('')); return '<a class="icon-yahoo" target="_blank" href="' + href + '">'+CONFIG.texts.yahoo+'</a>'; }, off365: function(event) { var startTime = formatTime(event.tzstart); var endTime = formatTime(event.tzend); var href = encodeURI([ 'https://outlook.office365.com/owa/', '?path=/calendar/action/compose', '&rru=addevent', '&subject=' + (event.title || ''), '&startdt=' + (startTime || ''), '&enddt=' + (endTime || ''), '&body=' + (event.description || ''), '&location=' + (event.address || ''), '&allday=' + (event.allday)?'true':'false' ].join('')); return '<a class="icon-outlook" target="_blank" href="' + href + '">'+CONFIG.texts.off365+'</a>'; }, ics: function(event, eClass, calendarName) { var startTime,endTime; if (event.allday) { // DTSTART and DTEND need to be equal and 0 startTime = formatTime(event.tzstart); endTime = startTime = stripISOTime(startTime)+'T000000'; } else { startTime = formatTime(event.tzstart); endTime = formatTime(event.tzend); } var href = encodeURI( 'data:text/calendar;charset=utf8,' + [ 'BEGIN:VCALENDAR', 'VERSION:2.0', 'BEGIN:VEVENT', 'URL:' + document.URL, 'DTSTART:' + (startTime || ''), 'DTEND:' + (endTime || ''), 'SUMMARY:' + (event.title || ''), 'DESCRIPTION:' + (event.description || ''), 'LOCATION:' + (event.address || ''), 'UID:' + (event.id || '') + '-' + document.URL, 'END:VEVENT', 'END:VCALENDAR'].join('\n')); return '<a class="' + eClass + '" download="'+CONFIG.texts.download+'" href="' + href + '">' + calendarName + '</a>'; }, ical: function(event) { return this.ics(event, 'icon-ical', CONFIG.texts.ical); }, outlook: function(event) { return this.ics(event, 'icon-outlook', CONFIG.texts.outlook); } }; /* -------------- helpers --------------- */ var changeTimezone = function(date,timezone) { if (date) { if (timezone) { var invdate = new Date(date.toLocaleString('en-US', { timeZone: timezone })); var diff = date.getTime()-invdate.getTime(); return new Date(date.getTime()+diff); } return date; } return; } var formatTime = function(date) { return date?date.toISOString().replace(/-|:|\.\d+/g, ''):''; }; var getEndDate = function(start,duration) { return new Date(start.getTime() + duration * MS_IN_MINUTES); }; var stripISOTime = function(isodatestr) { return isodatestr.substr(0,isodatestr.indexOf('T')); }; /* -------------- output handling --------------- */ var generateMarkup = function(calendars, clazz, calendarId, label) { var result = document.createElement('div'); result.innerHTML = '<label for="checkbox-for-' + calendarId + '" class="add-to-calendar-label">'+label+'</label>'; result.innerHTML += '<input name="add-to-calendar-checkbox" class="add-to-calendar-checkbox" id="checkbox-for-' + calendarId + '" type="checkbox" ' + ' onclick="closeCalenderOnMouseDown(this)">'; var dropdown = document.createElement('div'); dropdown.className = 'add-to-calendar-dropdown'; Object.keys(calendars).forEach(function(services) { dropdown.innerHTML += calendars[services]; }); result.appendChild(dropdown); result.className = 'add-to-calendar-widget'; if (clazz !== undefined) { result.className += (' ' + clazz); } addCSS(); result.id = calendarId; return result; }; var generateCalendars = function(event) { return { google: calendarGenerators.google(event), yahoo: calendarGenerators.yahoo(event), off365: calendarGenerators.off365(event), ical: calendarGenerators.ical(event), outlook: calendarGenerators.outlook(event) }; }; var addCSS = function() { if (!document.getElementById('add-to-calendar-css')) { document.getElementsByTagName('head')[0].appendChild(generateCSS()); } }; var generateCSS = function() { var styles = document.createElement('style'); styles.id = 'add-to-calendar-css'; styles.innerHTML = ".add-to-calendar{position:relative;text-align:left}.add-to-calendar>*{display:none}.add-to-calendar>.add-to-calendar-widget{display:block}.add-to-calendar-checkbox+div.add-to-calendar-dropdown{display:none;margin-left:20px}.add-to-calendar-checkbox+div.add-to-calendar-dropdown a,.add-to-calendar-checkbox:checked+div.add-to-calendar-dropdown{display:block}input[type=checkbox].add-to-calendar-checkbox{position:absolute;visibility:hidden}.add-to-calendar-checkbox+div.add-to-calendar-dropdown a:before{width:16px;height:16px;display:inline-block;background-image:url();margin-right:.5em;content:' '}.icon-ical:before{background-position:-68px 0}.icon-yahoo:before{background-position:-36px +4px}.icon-google:before{background-position:-52px 0}.add-to-calendar-widget{font-family:sans-serif;position:relative}.add-to-calendar-label{cursor:pointer;display:inline-block;padding:0;font-weight:400!important}.add-to-calendar-dropdown{position:absolute;z-index:99;background-color:#fff;top:0;left:0;padding:1em;margin:0!important;border-radius:3px;box-shadow:0 0 0 .5px rgba(50,50,93,.17),0 2px 5px 0 rgba(50,50,93,.1),0 1px 1.5px 0 rgba(0,0,0,.07),0 1px 2px 0 rgba(0,0,0,.08),0 0 0 0 transparent!important}.add-to-calendar-dropdown a{display:block;line-height:1.75em;text-decoration:none;color:inherit;opacity:.7}.add-to-calendar-dropdown a:hover{opacity:1}"; return styles; }; /* -------------- input handling --------------- */ var sanitizeParams = function(params) { if (!params.options) { params.options = {} } if (!params.options.id) { params.options.id = Math.floor(Math.random() * 1000000); } if (!params.options.label) { params.options.label = 'Add to calendar' } if (!params.options.class) { params.options.class = ''; } if (!params.data) { params.data = {}; } if (params.data.allday) { delete params.data.end; // may be set later delete params.data.duration; } if (params.data.end) { delete params.data.duration; } else { if (!params.data.duration) { params.data.duration = CONFIG.duration; } } if (params.data.duration) { params.data.end = getEndDate(params.data.start,params.data.duration); } if (params.data.timezone) { params.data.tzstart = changeTimezone(params.data.start,params.data.timezone); params.data.tzend = changeTimezone(params.data.end,params.data.timezone); } else { params.data.tzstart = params.data.start; params.data.tzend = params.data.end; } if (!params.data.title) { params.data.title = CONFIG.texts.title; } }; var validParams = function(params) { return params.data !== undefined && params.data.start !== undefined && (params.data.end !== undefined || params.data.allday !== undefined); }; var parseCalendar = function(elm) { /* <div title="Add to Calendar" class="addtocalendar"> <span class="start">12/18/2018 08:00 AM</span> <span class="end">12/18/2018 10:00 AM</span> <span class="duration">45</span> <span class="allday">true</span> <span class="timezone">America/Los_Angeles</span> <span class="title">Summary of the event</span> <span class="description">Description of the event</span> <span class="location">Location of the event</span> </div> */ var data = {}, node; node = elm.querySelector('.start'); if (node) data.start = new Date(node.textContent); node = elm.querySelector('.end'); if (node) data.end = new Date(node.textContent); node = elm.querySelector('.duration'); if (node) data.duration = 1*node.textContent; node = elm.querySelector('.allday'); if (node) data.allday = true; node = elm.querySelector('.title'); if (node) data.title = node.textContent; node = elm.querySelector('.description'); if (node) data.description = node.textContent; node = elm.querySelector('.address'); if (node) data.address = node.textContent; if (!data.address) { node = elm.querySelector('.location'); if (node) data.address = node.textContent; } node = elm.querySelector('.timezone'); if (node) data.timezone = node.textContent; cal = createCalendar({data:data}); if (cal) elm.appendChild(cal); return cal; } /* -------------- exports --------------- */ exports.closeCalenderOnMouseDown = function(checkbox) { //console.log('check'); var closeCalendar = function() { //console.log('click'); setTimeout(function() { checkbox.checked=false; }, 750); document.removeEventListener("mousedown",closeCalendar); }; document.addEventListener("mousedown",closeCalendar); } // bwc exports.createCalendar = function(params) { return addToCalendar(params); }; exports.addToCalendar = function(params) { if (params instanceof HTMLElement) { //console.log('HTMLElement'); return parseCalendar(params); } if (params instanceof NodeList) { //console.log('NodeList'); var success = (params.length>0); Array.prototype.forEach.call(params, function(node) { success = success && addToCalendar(node); }); return success; } sanitizeParams(params); if (!validParams(params)) { console.log('Event details missing.'); return; } return generateMarkup( generateCalendars(params.data), params.options.class, params.options.id, params.options.label ); }; // document.ready document.addEventListener("DOMContentLoaded", function(event) { createCalendar(document.querySelectorAll(CONFIG.selector)); }); })(this);