-
Notifications
You must be signed in to change notification settings - Fork 89
/
moment-recur.js
727 lines (586 loc) · 23.3 KB
/
moment-recur.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
(function (root, factory) {
if (typeof exports === 'object') {
module.exports = factory(require('moment'));
} else if (typeof define === 'function' && define.amd) {
define('moment-recur', ['moment'], factory);
} else {
root.moment = factory(root.moment);
}
}(this, function (moment) {
var hasModule;
hasModule = (typeof module !== "undefined" && module !== null) && (module.exports != null);
if (typeof moment === 'undefined') {
throw Error("Can't find moment");
}
// Interval object for creating and matching interval-based rules
var Interval = (function() {
function createInterval(units, measure) {
// Make sure all of the units are integers greater than 0.
for (var unit in units) {
if (units.hasOwnProperty(unit)) {
if (parseInt(unit, 10) <= 0) {
throw Error('Intervals must be greater than zero');
}
}
}
return {
measure: measure.toLowerCase(),
units: units
};
}
function matchInterval(type, units, start, date) {
// Get the difference between the start date and the provided date,
// using the required measure based on the type of rule'
var diff = null;
if (date.isBefore(start)) {
diff = start.diff(date, type, true);
} else {
diff = date.diff(start, type, true);
}
if (type == 'days') {
// if we are dealing with days, we deal with whole days only.
diff = parseInt(diff);
}
// Check to see if any of the units provided match the date
for (var unit in units) {
if (units.hasOwnProperty(unit)) {
unit = parseInt(unit, 10);
// If the units divide evenly into the difference, we have a match
if ((diff % unit) === 0) {
return true;
}
}
}
return false;
}
return {
create: createInterval,
match: matchInterval
};
})();
// Calendar object for creating and matching calendar-based rules
var Calendar = (function (){
// Dictionary of unit types based on measures
var unitTypes = {
"daysOfMonth": "date",
"daysOfWeek": "day",
"weeksOfMonth": "monthWeek",
"weeksOfMonthByDay": "monthWeekByDay",
"weeksOfYear": "week",
"monthsOfYear": "month"
};
// Dictionary of ranges based on measures
var ranges = {
"daysOfMonth" : { low: 1, high: 31 },
"daysOfWeek" : { low: 0, high: 6 },
"weeksOfMonth" : { low: 0, high: 4 },
"weeksOfMonthByDay" : { low: 0, high: 4 },
"weeksOfYear" : { low: 0, high: 52 },
"monthsOfYear" : { low: 0, high: 11 }
};
// Private function for checking the range of calendar values
function checkRange(low, high, list) {
list.forEach(function(v) {
if (v < low || v > high) {
throw Error('Value should be in range ' + low + ' to ' + high);
}
});
}
// Private function to convert day and month names to numbers
function namesToNumbers(list, nameType) {
var unit, unitInt, unitNum;
var newList = {};
for(unit in list) {
if (list.hasOwnProperty(unit)) {
unitInt = parseInt(unit, 10);
if (isNaN(unitInt)) {
unitInt = unit;
}
unitNum = moment().set(nameType, unitInt).get(nameType);
newList[unitNum] = list[unit];
}
}
return newList;
}
function createCalendarRule(list, measure) {
var keys = [];
// Convert day/month names to numbers, if needed
if (measure === "daysOfWeek") {
list = namesToNumbers(list, "days");
}
if (measure === "monthsOfYear") {
list = namesToNumbers(list, "months");
}
for (var key in list) if (hasOwnProperty.call(list, key)) keys.push(key);
// Make sure the listed units are in the measure's range
checkRange(ranges[measure].low,
ranges[measure].high,
keys);
return {
measure: measure,
units: list
};
}
function matchCalendarRule(measure, list, date) {
// Get the unit type (i.e. date, day, week, monthWeek, weeks, months)
var unitType = unitTypes[measure];
// Get the unit based on the required measure of the date
var unit = date[unitType]();
// If the unit is in our list, return true, else return false
if (list[unit]) {
return true;
}
// match on end of month days
if (unitType === 'date' && unit == date.add(1, 'months').date(0).format('D') && unit < 31) {
while (unit <= 31) {
if (list[unit]) {
return true;
}
unit++;
}
}
return false;
}
return {
create: createCalendarRule,
match: matchCalendarRule
};
})();
// The main Recur object to provide an interface for settings, rules, and matching
var Recur = (function() {
// A dictionary used to match rule measures to rule types
var ruleTypes = {
"days": "interval",
"weeks": "interval",
"months": "interval",
"years": "interval",
"daysOfWeek": "calendar",
"daysOfMonth": "calendar",
"weeksOfMonth": "calendar",
"weeksOfMonthByDay": "calendar",
"weeksOfYear": "calendar",
"monthsOfYear": "calendar"
};
// a dictionary of plural and singular measures
var measures = {
"days": "day",
"weeks": "week",
"months": "month",
"years": "year",
"daysOfWeek": "dayOfWeek",
"daysOfMonth": "dayOfMonth",
"weeksOfMonth": "weekOfMonth",
"weeksOfMonthByDay": "weekOfMonthByDay",
"weeksOfYear": "weekOfYear",
"monthsOfYear": "monthOfYear"
};
/////////////////////////////////
// Private Methods //
// Must be called with .call() //
/////////////////////////////////
// Private method that tries to set a rule.
function trigger() {
var rule;
var ruleType = ruleTypes[this.measure];
if (!(this instanceof Recur)) {
throw Error("Private method trigger() was called directly or not called as instance of Recur!");
}
// Make sure units and measure is defined and not null
if ((typeof this.units === "undefined" || this.units === null) || !this.measure) {
return this;
}
// Error if we don't have a valid ruleType
if (ruleType !== "calendar" && ruleType !== "interval") {
throw Error("Invalid measure provided: " + this.measure);
}
// Create the rule
if (ruleType === "interval") {
if (!this.start) {
throw Error("Must have a start date set to set an interval!");
}
rule = Interval.create(this.units, this.measure);
}
if (ruleType === "calendar") {
rule = Calendar.create(this.units, this.measure);
}
// Remove the temporary rule data
this.units = null;
this.measure = null;
if (rule.measure === 'weeksOfMonthByDay' && !this.hasRule('daysOfWeek')) {
throw Error("weeksOfMonthByDay must be combined with daysOfWeek");
}
// Remove existing rule based on measure
for (var i = 0; i < this.rules.length; i++) {
if (this.rules[i].measure === rule.measure) {
this.rules.splice(i, 1);
}
}
this.rules.push(rule);
return this;
}
// Private method to get next, previous or all occurrences
function getOccurrences(num, format, type) {
var currentDate, date;
var dates = [];
if (!(this instanceof Recur)) {
throw Error("Private method trigger() was called directly or not called as instance of Recur!");
}
if (!this.start && !this.from) {
throw Error("Cannot get occurrences without start or from date.");
}
if (type === "all" && !this.end) {
throw Error("Cannot get all occurrences without an end date.");
}
if (!!this.end && (this.start > this.end)) {
throw Error("Start date cannot be later than end date.");
}
// Return empty set if the caller doesn't want any for next/prev
if (type !== "all" && !(num > 0)) {
return dates;
}
// Start from the from date, or the start date if from is not set.
currentDate = (this.from || this.start).clone();
// Include the initial date in the results if wanting all dates
if (type === "all") {
if (this.matches(currentDate, false)) {
date = format ? currentDate.format(format) : currentDate.clone();
dates.push(date);
}
}
// Get the next N dates, if num is null then infinite
while (dates.length < (num===null ? dates.length+1 : num)) {
if (type === "next" || type === "all") {
currentDate.add(1, "day");
} else {
currentDate.subtract(1, "day");
}
//console.log("Match: " + currentDate.format("L") + " - " + this.matches(currentDate, true));
// Don't match outside the date if generating all dates within start/end
if (this.matches(currentDate, (type==="all"?false:true))) {
date = format ? currentDate.format(format) : currentDate.clone();
dates.push(date);
}
if(currentDate >= this.end) {
break;
}
}
return dates;
}
///////////////////////
// Private Functions //
///////////////////////
// Private function to see if a date is within range of start/end
function inRange(start, end, date) {
if (start && date.isBefore(start)) { return false; }
if (end && date.isAfter(end)) { return false; }
return true;
}
// Private function to turn units into objects
function unitsToObject(units) {
var list = {};
if (Object.prototype.toString.call(units) == '[object Array]') {
units.forEach(function(v) {
list[v] = true;
});
} else if (units === Object(units)) {
list = units;
} else if ((Object.prototype.toString.call(units) == '[object Number]') || (Object.prototype.toString.call(units) == '[object String]')) {
list[units] = true;
} else {
throw Error("Provide an array, object, string or number when passing units!");
}
return list;
}
// Private function to check if a date is an exception
function isException(exceptions, date) {
for (var i = 0, len = exceptions.length; i < len; i++) {
if (moment(exceptions[i]).isSame(date)) {
return true;
}
}
return false;
}
// Private function to pluralize measure names for use with dictionaries.
function pluralize(measure) {
switch(measure) {
case "day":
return "days";
case "week":
return "weeks";
case "month":
return "months";
case "year":
return "years";
case "dayOfWeek":
return "daysOfWeek";
case "dayOfMonth":
return "daysOfMonth";
case "weekOfMonth":
return "weeksOfMonth";
case "weekOfMonthByDay":
return "weeksOfMonthByDay";
case "weekOfYear":
return "weeksOfYear";
case "monthOfYear":
return "monthsOfYear";
default:
return measure;
}
}
// Private funtion to see if all rules match
function matchAllRules(rules, date, start) {
var i, len, rule, type;
for (i = 0, len = rules.length; i < len; i++) {
rule = rules[i];
type = ruleTypes[rule.measure];
if (type === "interval") {
if (!Interval.match(rule.measure, rule.units, start, date)) {
return false;
}
} else if (type === "calendar") {
if (!Calendar.match(rule.measure, rule.units, date)) {
return false;
}
} else {
return false;
}
}
return true;
}
// Private function to create measure functions
function createMeasure(measure) {
return function(units) {
this.every.call(this, units, measure);
return this;
};
}
//////////////////////
// Public Functions //
//////////////////////
// Recur Object Constrcutor
var Recur = function(options) {
if (options.start) {
this.start = moment(options.start).dateOnly();
}
if (options.end) {
this.end = moment(options.end).dateOnly();
}
// Our list of rules, all of which must match
this.rules = options.rules || [];
// Our list of exceptions. Match always fails on these dates.
var exceptions = options.exceptions || [];
this.exceptions = [];
for(var i = 0; i < exceptions.length; i++) {
this.except(exceptions[i]);
}
// Temporary units integer, array, or object. Does not get imported/exported.
this.units = null;
// Temporary measure type. Does not get imported/exported.
this.measure = null;
// Temporary from date for next/previous. Does not get imported/exported.
this.from = null;
return this;
};
// Get/Set start date
Recur.prototype.startDate = function(date) {
if (date === null) {
this.start = null;
return this;
}
if (date) {
this.start = moment(date).dateOnly();
return this;
}
return this.start;
};
// Get/Set end date
Recur.prototype.endDate = function(date) {
if (date === null) {
this.end = null;
return this;
}
if (date) {
this.end = moment(date).dateOnly();
return this;
}
return this.end;
};
// Get/Set a temporary from date
Recur.prototype.fromDate = function(date) {
if (date === null) {
this.from = null;
return this;
}
if (date) {
this.from = moment(date).dateOnly();
return this;
}
return this.from;
};
// Export the settings, rules, and exceptions of this recurring date
Recur.prototype.save = function() {
var data = {};
if (this.start && moment(this.start).isValid()) {
data.start = this.start.format("L");
}
if (this.end && moment(this.end).isValid()) {
data.end = this.end.format("L");
}
data.exceptions = [];
for (var i = 0, len = this.exceptions.length; i < len; i++) {
data.exceptions.push(this.exceptions[i].format("L"));
}
data.rules = this.rules;
return data;
};
// Return boolean value based on whether this date repeats (has rules or not)
Recur.prototype.repeats = function() {
if (this.rules.length > 0) {
return true;
}
return false;
};
// Set the units and, optionally, the measure
Recur.prototype.every = function(units, measure) {
if ((typeof units !== "undefined") && (units !== null)) {
this.units = unitsToObject(units);
}
if ((typeof measure !== "undefined") && (measure !== null)) {
this.measure = pluralize(measure);
}
return trigger.call(this);
};
// Creates an exception date to prevent matches, even if rules match
Recur.prototype.except = function(date) {
date = moment(date).dateOnly();
this.exceptions.push(date);
return this;
};
// Forgets rules (by passing measure) and exceptions (by passing date)
Recur.prototype.forget = function(dateOrRule) {
var i, len;
var whatMoment = moment(dateOrRule);
// If valid date, try to remove it from exceptions
if (whatMoment.isValid()) {
whatMoment = whatMoment.dateOnly(); // change to date only for perfect comparison
for (i = 0, len = this.exceptions.length; i < len; i++) {
if (whatMoment.isSame(this.exceptions[i])) {
this.exceptions.splice(i, 1);
return this;
}
}
return this;
}
// Otherwise, try to remove it from the rules
for (i = 0, len = this.rules.length; i < len; i++) {
if (this.rules[i].measure === pluralize(dateOrRule)) {
this.rules.splice(i, 1);
}
}
};
// Checks if a rule has been set on the chain
Recur.prototype.hasRule = function(measure) {
var i, len;
for (i = 0, len = this.rules.length; i < len; i++) {
if (this.rules[i].measure === pluralize(measure)) {
return true;
}
}
return false;
};
// Attempts to match a date to the rules
Recur.prototype.matches = function(dateToMatch, ignoreStartEnd) {
var date = moment(dateToMatch).dateOnly();
if (!date.isValid()) {
throw Error("Invalid date supplied to match method: " + dateToMatch);
}
if (!ignoreStartEnd && !inRange(this.start, this.end, date)) { return false }
if (isException(this.exceptions, date)) { return false; }
if (!matchAllRules(this.rules, date, this.start)) { return false; }
// if we passed everything above, then this date matches
return true;
};
// Get next N occurrences
Recur.prototype.next = function(num, format) {
return getOccurrences.call(this, num, format, "next");
};
// Get previous N occurrences
Recur.prototype.previous = function(num, format) {
return getOccurrences.call(this, num, format, "previous");
};
// Get all occurrences between start and end date
Recur.prototype.all = function(format) {
return getOccurrences.call(this, null, format, "all");
};
// Create the measure functions (days(), months(), daysOfMonth(), monthsOfYear(), etc.)
for (var measure in measures) {
if (ruleTypes.hasOwnProperty(measure)) {
Recur.prototype[measure] = Recur.prototype[measures[measure]] = createMeasure(measure);
}
}
return Recur;
}());
// Recur can be created the following ways:
// moment.recur()
// moment.recur(options)
// moment.recur(start)
// moment.recur(start, end)
moment.recur = function(start, end) {
// If we have an object, use it as a set of options
if (start === Object(start) && !moment.isMoment(start)) {
return new Recur(start);
}
// else, use the values passed
return new Recur({ start: start, end: end });
};
// Recur can also be created the following ways:
// moment().recur()
// moment().recur(options)
// moment().recur(start, end)
// moment(start).recur(end)
// moment().recur(end)
moment.fn.recur = function(start, end) {
// If we have an object, use it as a set of options
if (start === Object(start) && !moment.isMoment(start)) {
// if we have no start date, use the moment
if (typeof start.start === 'undefined') {
start.start = this;
}
return new Recur( start );
}
// if there is no end value, use the start value as the end
if (!end) {
end = start;
start = undefined;
}
// use the moment for the start value
if (!start) {
start = this;
}
return new Recur({ start: start, end: end, moment: this });
};
// Plugin for calculating the week of the month of a date
moment.fn.monthWeek = function() {
// First day of the first week of the month
var week0 = this.clone().startOf("month").startOf("week");
// First day of week
var day0 = this.clone().startOf("week");
return day0.diff(week0, "weeks");
};
// Plugin for calculating the occurrence of the day of the week in the month.
// Similar to `moment().monthWeek()`, the return value is zero-indexed.
// A return value of 2 means the date is the 3rd occurence of that day
// of the week in the month.
moment.fn.monthWeekByDay = function(date) {
return Math.floor((this.date()-1)/7);
};
// Plugin for removing all time information from a given date
moment.fn.dateOnly = function() {
if (this.tz && typeof(moment.tz) == 'function') {
return moment.tz(this.format('YYYY-MM-DD[T]00:00:00Z'), 'UTC');
} else {
return this.hours(0).minutes(0).seconds(0).milliseconds(0).add(this.utcOffset(), "minute").utcOffset(0);
}
};
return moment;
}));