diff --git a/add-to-calendar.js b/add-to-calendar.js index 97b6051..8d04f0f 100644 --- a/add-to-calendar.js +++ b/add-to-calendar.js @@ -1,11 +1,11 @@ ;(function(exports) { /* -------------- - config + config --------------- */ - + var MS_IN_MINUTES = 60 * 1000; - + var CONFIG = { selector : ".add-to-calendar", duration : 60, @@ -15,35 +15,36 @@ download : "Calendar-event.ics", google : "Google Calendar", yahoo : "Yahoo! Calendar", + outlook : "Outlook Calendar", off365 : "Office 365", ical : "Download iCal", - outlook : "Download Outlook", + dOutlook : "Download Outlook", ienoblob : "Sorry, your browser does not support downloading Calendar events." } }; - + if (typeof ADDTOCAL_CONFIG != "undefined") { CONFIG = ADDTOCAL_CONFIG; } - + /* -------------- - browser sniffing + browser sniffing --------------- */ - + // ie < edg (=chromium) doesnt support data-uri:text/calendar var ieCanDownload = ('msSaveOrOpenBlob' in window.navigator); var ieMustDownload = /\b(MSIE |Trident.*?rv:|Edge\/)(\d+)/.exec(navigator.userAgent); - - + + /* -------------- - generators + generators --------------- */ - + var calendarGenerators = { - + google: function(event) { var startTime,endTime; - + if (event.allday) { // google wants 2 consecutive days at 00:00 startTime = formatTime(event.tzstart); @@ -52,13 +53,13 @@ endTime = stripISOTime(endTime); } else { if (event.timezone) { - // google is somehow weird with timezones. + // google is somehow weird with timezones. // it works better when giving the local - // time in the given timezone without the zulu, + // 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. + // 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); @@ -76,7 +77,7 @@ endTime = formatTime(event.end); } } - + var href = encodeURI([ 'https://www.google.com/calendar/render', '?action=TEMPLATE', @@ -88,40 +89,40 @@ '&location=' + (event.address || ''), '&sprop=&sprop=name:' ].join('')); - - + + return ''+CONFIG.texts.google+''; }, 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([ @@ -137,10 +138,29 @@ href + '">'+CONFIG.texts.yahoo+''; }, + outlook: function(event) { + var startTime = formatTime(event.tzstart); + var endTime = formatTime(event.tzend); + + var href = encodeURI([ + 'https://outlook.live.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 ''+CONFIG.texts.outlook+''; + }, + 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', @@ -155,7 +175,7 @@ return ''+CONFIG.texts.off365+''; }, - + ics: function(event, eClass, calendarName) { var startTime,endTime; @@ -167,7 +187,7 @@ startTime = formatTime(event.tzstart); endTime = formatTime(event.tzend); } - + var cal = [ 'BEGIN:VCALENDAR', 'VERSION:2.0', @@ -181,51 +201,51 @@ 'UID:' + (event.id || '') + '-' + document.URL, 'END:VEVENT', 'END:VCALENDAR'].join('\n'); - + if (ieMustDownload) { return '' + calendarName + ''; } - + var href = encodeURI('data:text/calendar;charset=utf8,' + cal); - - return '' + calendarName + ''; - - + + }, ical: function(event) { return this.ics(event, 'icon-ical', CONFIG.texts.ical); }, - outlook: function(event) { - return this.ics(event, 'icon-outlook', CONFIG.texts.outlook); + dOutlook: function(event) { + return this.ics(event, 'icon-outlook', CONFIG.texts.dOutlook); } }; - + /* -------------- - helpers + helpers --------------- */ - + var changeTimezone = function(date,timezone) { if (date) { if (timezone) { - var invdate = new Date(date.toLocaleString('en-US', { - timeZone: 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); }; @@ -233,7 +253,7 @@ var stripISOTime = function(isodatestr) { return isodatestr.substr(0,isodatestr.indexOf('T')); }; - + var escapeJSValue = function(text) { return text .replace(/&/g, '&') @@ -243,9 +263,9 @@ .replace(/\'/g, '\\\'') .replace(/(\r?\n|\r)/gm, '\\n'); }; - + /* -------------- - output handling + output handling --------------- */ var generateMarkup = function(calendars, clazz, calendarId) { @@ -258,20 +278,20 @@ 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; }; @@ -280,9 +300,10 @@ return { google: calendarGenerators.google(event), yahoo: calendarGenerators.yahoo(event), + outlook: calendarGenerators.outlook(event), off365: calendarGenerators.off365(event), ical: calendarGenerators.ical(event), - outlook: calendarGenerators.outlook(event) + dOutlook: calendarGenerators.dOutlook(event) }; }; @@ -299,12 +320,12 @@ 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-label{cursor:pointer}.add-to-calendar-checkbox+div.add-to-calendar-dropdown{display:none;margin-left:20px}.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{cursor:pointer;display:block}.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;margin:1em 0;position:relative}.add-to-calendar-label{display:inline-block;background-color:#fff;background-image:url();background-position:10px 45%;background-repeat:no-repeat;padding:1em 1em 1em 40px;background-size:20px 20px;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{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 + input handling --------------- */ - + var sanitizeParams = function(params) { if (!params.options) { params.options = {} @@ -335,7 +356,7 @@ 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); @@ -346,17 +367,17 @@ 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) { - + /*
12/18/2018 08:00 AM @@ -371,43 +392,43 @@ */ 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 --------------- */ exports.ieDownloadCalendar = function(cal) { @@ -421,14 +442,14 @@ exports.closeCalenderOnMouseDown = function(checkbox) { //console.log('check'); - var closeCalendar = function() { + var closeCalendar = function() { //console.log('click'); setTimeout(function() { checkbox.checked=false; }, 750); document.removeEventListener("mousedown",closeCalendar); }; document.addEventListener("mousedown",closeCalendar); } - + exports.addToCalendarData = function(params) { if (!params) params = {}; sanitizeParams(params); @@ -438,32 +459,32 @@ } return generateCalendars(params.data); } - + // bwc exports.createCalendar = function(params) { return addToCalendar(params); }; - + exports.addToCalendar = function(params) { - + if (!params) 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) { + Array.prototype.forEach.call(params, function(node) { success = success && addToCalendar(node); - }); + }); return success; } - + sanitizeParams(params); - + if (!validParams(params)) { console.error('Event details missing.'); return; @@ -474,13 +495,13 @@ params.options.class, params.options.id ); - + }; - + // document.ready - - document.addEventListener("DOMContentLoaded", function(event) { + + document.addEventListener("DOMContentLoaded", function(event) { addToCalendar(document.querySelectorAll(CONFIG.selector)); }); - + })(this); \ No newline at end of file diff --git a/example.html b/example.html index 361e8e6..09191a6 100644 --- a/example.html +++ b/example.html @@ -18,7 +18,7 @@

OUICal2 Add to Calendar

- +

Widget from javascript input

The example below created a calendar from javascript data. Check the source to see how that's done. @@ -26,12 +26,12 @@

Widget from javascript input

Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.

- - + + - +
- +

Widget from HTML input

The example below created a calendar from html content. Check the source to see how that's done.

Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. - - + +

12/18/2018 08:00 AM America/Los_Angeles - + Summary of the event Description of the event Location of the event
- +

- +
- +

Generating javascript data as output

- The example below created the calendar links from javascript data + The example below created the calendar links from javascript data and returned it as a javascript object. Check the source to see how that's done.

- - + + - +