diff --git a/README.md b/README.md
index 2ee3587..9714759 100644
--- a/README.md
+++ b/README.md
@@ -1,24 +1,27 @@
-# OuiCal
+# OuiCal2
A simple JS library that enables you to add an "add to calendar" button for your upcoming events.
## Inspiration
-This project was inspired by [Eventbrite's](http://www.eventbrite.com/) add to calendar feature (which should have been open sourced #justSayin).
+This project was inspired by [Eventbrite's](http://www.eventbrite.com/) add to calendar feature (which should have been open sourced #justSayin).
+
+Later, it was adjusted to look and behave more like [AddToCalendar](https://addtocalendar.com), which suddenly became a commercial tool, and very expensive too #justSayin.
## How to use it?
-Call 'createCalendar' with your event info, pass in any optional parameters such as a class and/ or id and boom! Insert your add-to-calendar div wherever you'd like.
+### Method 1
+
+Call 'addToCalendar' with your event info, pass in any optional parameters such as a class and/ or id and boom! Insert your add-to-calendar div wherever you'd like.
The only fields that are mandatory are:
- Event title
- Start time
- - Event duration, in minutes
-## Example
+#### Example
- var myCalendar = createCalendar({
+ var myCalendar = addToCalendar({
options: {
class: 'my-class',
@@ -32,12 +35,19 @@ The only fields that are mandatory are:
// Event start date
start: new Date('June 15, 2013 19:00'),
+ // Event timezone. Will convert the given time to that zone
+ timezone: America/Los_Angeles,
+
// Event duration (IN MINUTES)
duration: 120,
// You can also choose to set an end time
// If an end time is set, this will take precedence over duration
- end: new Date('June 15, 2013 23:00'),
+ // end: new Date('June 15, 2013 23:00'),
+
+ // You can also choose to set 'all day'
+ // If this is set, this will override end time, duration and timezone
+ // allday:true,
// Event Address
address: 'The internet',
@@ -49,23 +59,62 @@ The only fields that are mandatory are:
document.querySelector('#place-where-I-want-this-calendar').appendChild(myCalendar);
-[Here is a live example](http://carlsednaoui.github.io/add-to-calendar-buttons/example.html)
+### Method 2
+
+Write your events data in several hidden HTML tags with the right classnames, and wrap them in a div. Then call 'createCalendar' with the outer div and boom! The calendar is appended to your wrapping div.
+The only fields that are mandatory are:
+
+ - Event title
+ - Start time
+
+
+#### Example
+
+
+ 12/18/2018 08:00 AM
+ America/Los_Angeles
+
+
+
+ Summary of the event
+ Description of the event
+ Location of the event
+
+
+
+### method 3
+
+If you are ony interested in the data, call 'addToCalendarData' with your event info, pass in any optional parameters as described in method #1. An object with some html links is returned.
+
+
+#### Example
+
+ var caldata = addToCalendarData({
+ data: {
+ // Event title
+ title: 'Get on the front page of HN',
+
+ // Event start date
+ start: new Date('June 15, 2013 19:00'),
+
+ //.. and the rest was optional.
+
+ }
+ });
+ console.log(caldata);
-## Looking for Instant Gratification?
-[Copy OuiCal into Chrome's JS console](https://raw.github.com/carlsednaoui/ouical/master/ouical.js) (or whatever browser you're using).
+## Demo
-Then call this:
+[Here is a live example](https://commonpike.github.io/add-to-calendar-buttons/example.html)
- document.getElementsByTagName('body')[0].appendChild(createCalendar({data:{title:"this is the title of my event", start: new Date(), duration: 90}}));
-\#winning!
## Calendar Generator
-Need to generate an add-to-calendar widget on the fly? No problem, [go here](http://carlsednaoui.github.io/add-to-calendar-buttons/generator/generator.html).
+Need to generate an add-to-calendar widget on the fly? No problem, [go here](https://commonpike.github.io/add-to-calendar-buttons/generator/generator.html).
## GitHub Project Page
-[Official Project Page](http://carlsednaoui.github.io/ouical/)
+[Official Project Page](http://commonpike.github.io/ouical/)
## License
[MIT](http://opensource.org/licenses/MIT)
diff --git a/main.css b/add-to-calendar.css
similarity index 60%
rename from main.css
rename to add-to-calendar.css
index f9d318f..a5f882c 100644
--- a/main.css
+++ b/add-to-calendar.css
@@ -1,24 +1,46 @@
-#add-to-calendar-checkbox-label {
+/* --------------------
+ this css is embedded in the script in
+ minified version. it is here for reference
+ only
+-------------------- */
+
+.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 ~ a {
+.add-to-calendar-checkbox + div.add-to-calendar-dropdown {
display: none;
+ margin-left: 20px;
}
-.add-to-calendar-checkbox:checked ~ a {
+.add-to-calendar-checkbox:checked + div.add-to-calendar-dropdown {
display: block;
- width: 150px;
- margin-left: 20px;
}
input[type=checkbox].add-to-calendar-checkbox {
position: absolute;
- top: -9999px;
- left: -9999px;
+ visibility:hidden;
+}
+
+.add-to-calendar-checkbox + div.add-to-calendar-dropdown a {
+ cursor:pointer;
+ display:block;
}
-.add-to-calendar-checkbox ~ a:before {
+.add-to-calendar-checkbox + div.add-to-calendar-dropdown a:before {
width: 16px;
height: 16px;
display: inline-block;
@@ -35,10 +57,70 @@ input[type=checkbox].add-to-calendar-checkbox {
/*this is the default icon*/
}
+.icon-off365:before {
+ /*this is the default icon*/
+}
+
.icon-yahoo:before {
background-position: -36px +4px;
}
.icon-google:before {
background-position: -52px 0;
-}
\ No newline at end of file
+}
+
+/* --------------------
+ prettify
+-------------------- */
+
+.add-to-calendar-widget {
+ font-family:sans-serif;
+ margin: 1em 0;
+ position:relative;
+}
+
+.add-to-calendar-label {
+ display: inline-block;
+ background-color: white;
+ 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 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: white;
+ top: 0;
+ left:0;
+ padding: 1em;
+ margin:0!important;
+ border-radius: 3px;
+ box-shadow: 0 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;
+}
+
+
diff --git a/add-to-calendar.js b/add-to-calendar.js
new file mode 100644
index 0000000..97b6051
--- /dev/null
+++ b/add-to-calendar.js
@@ -0,0 +1,486 @@
+;(function(exports) {
+
+ /* --------------
+ config
+ --------------- */
+
+ var MS_IN_MINUTES = 60 * 1000;
+
+ var CONFIG = {
+ selector : ".add-to-calendar",
+ duration : 60,
+ texts : {
+ label : "Add to Calendar",
+ title : "New event",
+ download : "Calendar-event.ics",
+ google : "Google Calendar",
+ yahoo : "Yahoo! Calendar",
+ off365 : "Office 365",
+ ical : "Download iCal",
+ outlook : "Download Outlook",
+ ienoblob : "Sorry, your browser does not support downloading Calendar events."
+ }
+ };
+
+ if (typeof ADDTOCAL_CONFIG != "undefined") {
+ CONFIG = ADDTOCAL_CONFIG;
+ }
+
+ /* --------------
+ 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
+ --------------- */
+
+ 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 ''+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([
+ '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 ''+CONFIG.texts.yahoo+'';
+ },
+
+ 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 ''+CONFIG.texts.off365+'';
+ },
+
+ 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 cal = [
+ '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');
+
+ 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);
+ }
+ };
+
+ /* --------------
+ 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'));
+ };
+
+ var escapeJSValue = function(text) {
+ return text
+ .replace(/&/g, '&')
+ .replace(//g, '>')
+ .replace(/\"/g, '"')
+ .replace(/\'/g, '\\\'')
+ .replace(/(\r?\n|\r)/gm, '\\n');
+ };
+
+ /* --------------
+ output handling
+ --------------- */
+
+ var generateMarkup = function(calendars, clazz, calendarId) {
+ var result = document.createElement('div');
+
+ result.innerHTML = '';
+ result.innerHTML += '';
+
+ 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-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
+ --------------- */
+
+ var sanitizeParams = function(params) {
+ if (!params.options) {
+ params.options = {}
+ }
+ if (!params.options.id) {
+ params.options.id = Math.floor(Math.random() * 1000000);
+ }
+ if (!params.options.class) {
+ params.options.class = '';
+ }
+ if (!params.data) {
+ params.data = {};
+ }
+ if (!params.data.start) {
+ params.data.start=new Date();
+ }
+ 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) {
+
+ /*
+
+ 12/18/2018 08:00 AM
+ 12/18/2018 10:00 AM
+ 45
+ true
+ America/Los_Angeles
+ Summary of the event
+ Description of the event
+ Location of the event
+
+ 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
+ and returned it as a javascript object. Check the source to see how that's done.
+