forked from peterbraden/ical.js
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Fix comma-separated EXDATES, change ISO string keys
Added support for comma-separated EXDATES, since those are allowed by the RFC. While testing this, I discovered some flaws with using the timestamp for EXDATE and RECURRENCE lookups. As a result, the lookup key has been changed to be a pure date string, instead of the date time ISO string. Added tests, example code, and documentation showing how to use this.
- Loading branch information
Showing
7 changed files
with
1,225 additions
and
946 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 1,118 @@ | ||
var ical = require('./node-ical') | ||
var moment = require('moment') | ||
|
||
var data = ical.parseFile('./examples/example_rrule.ics'); | ||
|
||
// Complicated example demonstrating how to handle recurrence rules and exceptions. | ||
|
||
for (var k in data) { | ||
|
||
// When dealing with calendar recurrences, you need a range of dates to query against, | ||
// because otherwise you can get an infinite number of calendar events. | ||
var rangeStart = moment("2017-01-01"); | ||
var rangeEnd = moment("2017-12-31"); | ||
|
||
|
||
var event = data[k] | ||
if (event.type === 'VEVENT') { | ||
|
||
var title = event.summary; | ||
var startDate = moment(event.start); | ||
var endDate = moment(event.end); | ||
|
||
// Calculate the duration of the event for use with recurring events. | ||
var duration = parseInt(endDate.format("x")) - parseInt(startDate.format("x")); | ||
|
||
// Simple case - no recurrences, just print out the calendar event. | ||
if (typeof event.rrule === 'undefined') | ||
{ | ||
console.log('title:' title); | ||
console.log('startDate:' startDate.format('MMMM Do YYYY, h:mm:ss a')); | ||
console.log('endDate:' endDate.format('MMMM Do YYYY, h:mm:ss a')); | ||
console.log('duration:' moment.duration(duration).humanize()); | ||
console.log(); | ||
} | ||
|
||
// Complicated case - if an RRULE exists, handle multiple recurrences of the event. | ||
else if (typeof event.rrule !== 'undefined') | ||
{ | ||
// For recurring events, get the set of event start dates that fall within the range | ||
// of dates we're looking for. | ||
var dates = event.rrule.between( | ||
rangeStart.toDate(), | ||
rangeEnd.toDate(), | ||
true, | ||
function(date, i) {return true;} | ||
) | ||
|
||
// The "dates" array contains the set of dates within our desired date range range that are valid | ||
// for the recurrence rule. *However*, it's possible for us to have a specific recurrence that | ||
// had its date changed from outside the range to inside the range. One way to handle this is | ||
// to add *all* recurrence override entries into the set of dates that we check, and then later | ||
// filter out any recurrences that don't actually belong within our range. | ||
if (event.recurrences != undefined) | ||
{ | ||
for (var r in event.recurrences) | ||
{ | ||
// Only add dates that weren't already in the range we added from the rrule so that | ||
// we don't double-add those events. | ||
if (moment(new Date(r)).isBetween(rangeStart, rangeEnd) != true) | ||
{ | ||
dates.push(new Date(r)); | ||
} | ||
} | ||
} | ||
|
||
// Loop through the set of date entries to see which recurrences should be printed. | ||
for(var i in dates) { | ||
|
||
var date = dates[i]; | ||
var curEvent = event; | ||
var showRecurrence = true; | ||
var curDuration = duration; | ||
|
||
startDate = moment(date); | ||
|
||
// Use just the date of the recurrence to look up overrides and exceptions (i.e. chop off time information) | ||
var dateLookupKey = date.toISOString().substring(0, 10); | ||
|
||
// For each date that we're checking, it's possible that there is a recurrence override for that one day. | ||
if ((curEvent.recurrences != undefined) && (curEvent.recurrences[dateLookupKey] != undefined)) | ||
{ | ||
// We found an override, so for this recurrence, use a potentially different title, start date, and duration. | ||
curEvent = curEvent.recurrences[dateLookupKey]; | ||
startDate = moment(curEvent.start); | ||
curDuration = parseInt(moment(curEvent.end).format("x")) - parseInt(startDate.format("x")); | ||
} | ||
// If there's no recurrence override, check for an exception date. Exception dates represent exceptions to the rule. | ||
else if ((curEvent.exdate != undefined) && (curEvent.exdate[dateLookupKey] != undefined)) | ||
{ | ||
// This date is an exception date, which means we should skip it in the recurrence pattern. | ||
showRecurrence = false; | ||
} | ||
|
||
// Set the the title and the end date from either the regular event or the recurrence override. | ||
var recurrenceTitle = curEvent.summary; | ||
endDate = moment(parseInt(startDate.format("x")) curDuration, 'x'); | ||
|
||
// If this recurrence ends before the start of the date range, or starts after the end of the date range, | ||
// don't process it. | ||
if (endDate.isBefore(rangeStart) || startDate.isAfter(rangeEnd)) { | ||
showRecurrence = false; | ||
} | ||
|
||
if (showRecurrence === true) { | ||
|
||
console.log('title:' recurrenceTitle); | ||
console.log('startDate:' startDate.format('MMMM Do YYYY, h:mm:ss a')); | ||
console.log('endDate:' endDate.format('MMMM Do YYYY, h:mm:ss a')); | ||
console.log('duration:' moment.duration(curDuration).humanize()); | ||
console.log(); | ||
} | ||
|
||
} | ||
} | ||
} | ||
} | ||
|
||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 1,40 @@ | ||
BEGIN:VCALENDAR | ||
PRODID:-//Google Inc//Google Calendar 70.9054//EN | ||
VERSION:2.0 | ||
CALSCALE:GREGORIAN | ||
METHOD:PUBLISH | ||
X-WR-CALNAME:ical | ||
X-WR-TIMEZONE:US/Central | ||
X-WR-CALDESC: | ||
BEGIN:VEVENT | ||
UID:98765432-ABCD-DCBB-999A-987765432123 | ||
DTSTART;TZID=US/Central:20170601T095000 | ||
DTEND;TZID=US/Central:20170601T170000 | ||
DTSTAMP:20170727T044436Z | ||
EXDATE;TZID=US/Central:20170706T095000,20170713T095000,20170720T095000,20 | ||
170803T095000 | ||
LAST-MODIFIED:20170727T044435Z | ||
RRULE:FREQ=WEEKLY;WKST=SU;UNTIL=20170814T045959Z;BYDAY=TH | ||
SEQUENCE:0 | ||
SUMMARY:Recurring weekly meeting from June 1 - Aug 14 (except July 6, July 13, July 20, Aug 3) | ||
END:VEVENT | ||
BEGIN:VEVENT | ||
UID:98765432-ABCD-DCBB-999A-987765432123 | ||
RECURRENCE-ID;TZID=US/Central:20170629T095000 | ||
DTSTART;TZID=US/Central:20170703T095000 | ||
DTEND;TZID=US/Central:20170703T120000 | ||
DTSTAMP:20170727T044436Z | ||
LAST-MODIFIED:20170216T143445Z | ||
SEQUENCE:0 | ||
SUMMARY:Last meeting in June moved to Monday July 3 and shortened to half day | ||
END:VEVENT | ||
BEGIN:VEVENT | ||
UID:12354454-ABCD-DCBB-999A-2349872354897 | ||
DTSTART;TZID=US/Central:20171201T130000 | ||
DTEND;TZID=US/Central:20171201T150000 | ||
DTSTAMP:20170727T044436Z | ||
LAST-MODIFIED:20170727T044435Z | ||
SEQUENCE:0 | ||
SUMMARY:Single event on Dec 1 | ||
END:VEVENT | ||
END:VCALENDAR |
Oops, something went wrong.