js-add-to-calendar-buttons
Version:
A vanilla javascript lib to create add to calendar buttons
437 lines (354 loc) • 18.9 kB
JavaScript
(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);