Cleaning up function definitions in Language.php
[mediawiki.git] / resources / moment / moment.js
blobb79da7f26046089dd79458076a9629a7b34d2560
1 //! moment.js
2 //! version : 2.5.1
3 //! authors : Tim Wood, Iskren Chernev, Moment.js contributors
4 //! license : MIT
5 //! momentjs.com
7 (function (undefined) {
9     /************************************
10         Constants
11     ************************************/
13     var moment,
14         VERSION = "2.5.1",
15         global = this,
16         round = Math.round,
17         i,
19         YEAR = 0,
20         MONTH = 1,
21         DATE = 2,
22         HOUR = 3,
23         MINUTE = 4,
24         SECOND = 5,
25         MILLISECOND = 6,
27         // internal storage for language config files
28         languages = {},
30         // moment internal properties
31         momentProperties = {
32             _isAMomentObject: null,
33             _i : null,
34             _f : null,
35             _l : null,
36             _strict : null,
37             _isUTC : null,
38             _offset : null,  // optional. Combine with _isUTC
39             _pf : null,
40             _lang : null  // optional
41         },
43         // check for nodeJS
44         hasModule = (typeof module !== 'undefined' && module.exports && typeof require !== 'undefined'),
46         // ASP.NET json date format regex
47         aspNetJsonRegex = /^\/?Date\((\-?\d+)/i,
48         aspNetTimeSpanJsonRegex = /(\-)?(?:(\d*)\.)?(\d+)\:(\d+)(?:\:(\d+)\.?(\d{3})?)?/,
50         // from http://docs.closure-library.googlecode.com/git/closure_goog_date_date.js.source.html
51         // somewhat more in line with 4.4.3.2 2004 spec, but allows decimal anywhere
52         isoDurationRegex = /^(-)?P(?:(?:([0-9,.]*)Y)?(?:([0-9,.]*)M)?(?:([0-9,.]*)D)?(?:T(?:([0-9,.]*)H)?(?:([0-9,.]*)M)?(?:([0-9,.]*)S)?)?|([0-9,.]*)W)$/,
54         // format tokens
55         formattingTokens = /(\[[^\[]*\])|(\\)?(Mo|MM?M?M?|Do|DDDo|DD?D?D?|ddd?d?|do?|w[o|w]?|W[o|W]?|YYYYYY|YYYYY|YYYY|YY|gg(ggg?)?|GG(GGG?)?|e|E|a|A|hh?|HH?|mm?|ss?|S{1,4}|X|zz?|ZZ?|.)/g,
56         localFormattingTokens = /(\[[^\[]*\])|(\\)?(LT|LL?L?L?|l{1,4})/g,
58         // parsing token regexes
59         parseTokenOneOrTwoDigits = /\d\d?/, // 0 - 99
60         parseTokenOneToThreeDigits = /\d{1,3}/, // 0 - 999
61         parseTokenOneToFourDigits = /\d{1,4}/, // 0 - 9999
62         parseTokenOneToSixDigits = /[+\-]?\d{1,6}/, // -999,999 - 999,999
63         parseTokenDigits = /\d+/, // nonzero number of digits
64         parseTokenWord = /[0-9]*['a-z\u00A0-\u05FF\u0700-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]+|[\u0600-\u06FF\/]+(\s*?[\u0600-\u06FF]+){1,2}/i, // any word (or two) characters or numbers including two/three word month in arabic.
65         parseTokenTimezone = /Z|[\+\-]\d\d:?\d\d/gi, // +00:00 -00:00 +0000 -0000 or Z
66         parseTokenT = /T/i, // T (ISO separator)
67         parseTokenTimestampMs = /[\+\-]?\d+(\.\d{1,3})?/, // 123456789 123456789.123
69         //strict parsing regexes
70         parseTokenOneDigit = /\d/, // 0 - 9
71         parseTokenTwoDigits = /\d\d/, // 00 - 99
72         parseTokenThreeDigits = /\d{3}/, // 000 - 999
73         parseTokenFourDigits = /\d{4}/, // 0000 - 9999
74         parseTokenSixDigits = /[+-]?\d{6}/, // -999,999 - 999,999
75         parseTokenSignedNumber = /[+-]?\d+/, // -inf - inf
77         // iso 8601 regex
78         // 0000-00-00 0000-W00 or 0000-W00-0 + T + 00 or 00:00 or 00:00:00 or 00:00:00.000 + +00:00 or +0000 or +00)
79         isoRegex = /^\s*(?:[+-]\d{6}|\d{4})-(?:(\d\d-\d\d)|(W\d\d$)|(W\d\d-\d)|(\d\d\d))((T| )(\d\d(:\d\d(:\d\d(\.\d+)?)?)?)?([\+\-]\d\d(?::?\d\d)?|\s*Z)?)?$/,
81         isoFormat = 'YYYY-MM-DDTHH:mm:ssZ',
83         isoDates = [
84             ['YYYYYY-MM-DD', /[+-]\d{6}-\d{2}-\d{2}/],
85             ['YYYY-MM-DD', /\d{4}-\d{2}-\d{2}/],
86             ['GGGG-[W]WW-E', /\d{4}-W\d{2}-\d/],
87             ['GGGG-[W]WW', /\d{4}-W\d{2}/],
88             ['YYYY-DDD', /\d{4}-\d{3}/]
89         ],
91         // iso time formats and regexes
92         isoTimes = [
93             ['HH:mm:ss.SSSS', /(T| )\d\d:\d\d:\d\d\.\d{1,3}/],
94             ['HH:mm:ss', /(T| )\d\d:\d\d:\d\d/],
95             ['HH:mm', /(T| )\d\d:\d\d/],
96             ['HH', /(T| )\d\d/]
97         ],
99         // timezone chunker "+10:00" > ["10", "00"] or "-1530" > ["-15", "30"]
100         parseTimezoneChunker = /([\+\-]|\d\d)/gi,
102         // getter and setter names
103         proxyGettersAndSetters = 'Date|Hours|Minutes|Seconds|Milliseconds'.split('|'),
104         unitMillisecondFactors = {
105             'Milliseconds' : 1,
106             'Seconds' : 1e3,
107             'Minutes' : 6e4,
108             'Hours' : 36e5,
109             'Days' : 864e5,
110             'Months' : 2592e6,
111             'Years' : 31536e6
112         },
114         unitAliases = {
115             ms : 'millisecond',
116             s : 'second',
117             m : 'minute',
118             h : 'hour',
119             d : 'day',
120             D : 'date',
121             w : 'week',
122             W : 'isoWeek',
123             M : 'month',
124             y : 'year',
125             DDD : 'dayOfYear',
126             e : 'weekday',
127             E : 'isoWeekday',
128             gg: 'weekYear',
129             GG: 'isoWeekYear'
130         },
132         camelFunctions = {
133             dayofyear : 'dayOfYear',
134             isoweekday : 'isoWeekday',
135             isoweek : 'isoWeek',
136             weekyear : 'weekYear',
137             isoweekyear : 'isoWeekYear'
138         },
140         // format function strings
141         formatFunctions = {},
143         // tokens to ordinalize and pad
144         ordinalizeTokens = 'DDD w W M D d'.split(' '),
145         paddedTokens = 'M D H h m s w W'.split(' '),
147         formatTokenFunctions = {
148             M    : function () {
149                 return this.month() + 1;
150             },
151             MMM  : function (format) {
152                 return this.lang().monthsShort(this, format);
153             },
154             MMMM : function (format) {
155                 return this.lang().months(this, format);
156             },
157             D    : function () {
158                 return this.date();
159             },
160             DDD  : function () {
161                 return this.dayOfYear();
162             },
163             d    : function () {
164                 return this.day();
165             },
166             dd   : function (format) {
167                 return this.lang().weekdaysMin(this, format);
168             },
169             ddd  : function (format) {
170                 return this.lang().weekdaysShort(this, format);
171             },
172             dddd : function (format) {
173                 return this.lang().weekdays(this, format);
174             },
175             w    : function () {
176                 return this.week();
177             },
178             W    : function () {
179                 return this.isoWeek();
180             },
181             YY   : function () {
182                 return leftZeroFill(this.year() % 100, 2);
183             },
184             YYYY : function () {
185                 return leftZeroFill(this.year(), 4);
186             },
187             YYYYY : function () {
188                 return leftZeroFill(this.year(), 5);
189             },
190             YYYYYY : function () {
191                 var y = this.year(), sign = y >= 0 ? '+' : '-';
192                 return sign + leftZeroFill(Math.abs(y), 6);
193             },
194             gg   : function () {
195                 return leftZeroFill(this.weekYear() % 100, 2);
196             },
197             gggg : function () {
198                 return leftZeroFill(this.weekYear(), 4);
199             },
200             ggggg : function () {
201                 return leftZeroFill(this.weekYear(), 5);
202             },
203             GG   : function () {
204                 return leftZeroFill(this.isoWeekYear() % 100, 2);
205             },
206             GGGG : function () {
207                 return leftZeroFill(this.isoWeekYear(), 4);
208             },
209             GGGGG : function () {
210                 return leftZeroFill(this.isoWeekYear(), 5);
211             },
212             e : function () {
213                 return this.weekday();
214             },
215             E : function () {
216                 return this.isoWeekday();
217             },
218             a    : function () {
219                 return this.lang().meridiem(this.hours(), this.minutes(), true);
220             },
221             A    : function () {
222                 return this.lang().meridiem(this.hours(), this.minutes(), false);
223             },
224             H    : function () {
225                 return this.hours();
226             },
227             h    : function () {
228                 return this.hours() % 12 || 12;
229             },
230             m    : function () {
231                 return this.minutes();
232             },
233             s    : function () {
234                 return this.seconds();
235             },
236             S    : function () {
237                 return toInt(this.milliseconds() / 100);
238             },
239             SS   : function () {
240                 return leftZeroFill(toInt(this.milliseconds() / 10), 2);
241             },
242             SSS  : function () {
243                 return leftZeroFill(this.milliseconds(), 3);
244             },
245             SSSS : function () {
246                 return leftZeroFill(this.milliseconds(), 3);
247             },
248             Z    : function () {
249                 var a = -this.zone(),
250                     b = "+";
251                 if (a < 0) {
252                     a = -a;
253                     b = "-";
254                 }
255                 return b + leftZeroFill(toInt(a / 60), 2) + ":" + leftZeroFill(toInt(a) % 60, 2);
256             },
257             ZZ   : function () {
258                 var a = -this.zone(),
259                     b = "+";
260                 if (a < 0) {
261                     a = -a;
262                     b = "-";
263                 }
264                 return b + leftZeroFill(toInt(a / 60), 2) + leftZeroFill(toInt(a) % 60, 2);
265             },
266             z : function () {
267                 return this.zoneAbbr();
268             },
269             zz : function () {
270                 return this.zoneName();
271             },
272             X    : function () {
273                 return this.unix();
274             },
275             Q : function () {
276                 return this.quarter();
277             }
278         },
280         lists = ['months', 'monthsShort', 'weekdays', 'weekdaysShort', 'weekdaysMin'];
282     function defaultParsingFlags() {
283         // We need to deep clone this object, and es5 standard is not very
284         // helpful.
285         return {
286             empty : false,
287             unusedTokens : [],
288             unusedInput : [],
289             overflow : -2,
290             charsLeftOver : 0,
291             nullInput : false,
292             invalidMonth : null,
293             invalidFormat : false,
294             userInvalidated : false,
295             iso: false
296         };
297     }
299     function padToken(func, count) {
300         return function (a) {
301             return leftZeroFill(func.call(this, a), count);
302         };
303     }
304     function ordinalizeToken(func, period) {
305         return function (a) {
306             return this.lang().ordinal(func.call(this, a), period);
307         };
308     }
310     while (ordinalizeTokens.length) {
311         i = ordinalizeTokens.pop();
312         formatTokenFunctions[i + 'o'] = ordinalizeToken(formatTokenFunctions[i], i);
313     }
314     while (paddedTokens.length) {
315         i = paddedTokens.pop();
316         formatTokenFunctions[i + i] = padToken(formatTokenFunctions[i], 2);
317     }
318     formatTokenFunctions.DDDD = padToken(formatTokenFunctions.DDD, 3);
321     /************************************
322         Constructors
323     ************************************/
325     function Language() {
327     }
329     // Moment prototype object
330     function Moment(config) {
331         checkOverflow(config);
332         extend(this, config);
333     }
335     // Duration Constructor
336     function Duration(duration) {
337         var normalizedInput = normalizeObjectUnits(duration),
338             years = normalizedInput.year || 0,
339             months = normalizedInput.month || 0,
340             weeks = normalizedInput.week || 0,
341             days = normalizedInput.day || 0,
342             hours = normalizedInput.hour || 0,
343             minutes = normalizedInput.minute || 0,
344             seconds = normalizedInput.second || 0,
345             milliseconds = normalizedInput.millisecond || 0;
347         // representation for dateAddRemove
348         this._milliseconds = +milliseconds +
349             seconds * 1e3 + // 1000
350             minutes * 6e4 + // 1000 * 60
351             hours * 36e5; // 1000 * 60 * 60
352         // Because of dateAddRemove treats 24 hours as different from a
353         // day when working around DST, we need to store them separately
354         this._days = +days +
355             weeks * 7;
356         // It is impossible translate months into days without knowing
357         // which months you are are talking about, so we have to store
358         // it separately.
359         this._months = +months +
360             years * 12;
362         this._data = {};
364         this._bubble();
365     }
367     /************************************
368         Helpers
369     ************************************/
372     function extend(a, b) {
373         for (var i in b) {
374             if (b.hasOwnProperty(i)) {
375                 a[i] = b[i];
376             }
377         }
379         if (b.hasOwnProperty("toString")) {
380             a.toString = b.toString;
381         }
383         if (b.hasOwnProperty("valueOf")) {
384             a.valueOf = b.valueOf;
385         }
387         return a;
388     }
390     function cloneMoment(m) {
391         var result = {}, i;
392         for (i in m) {
393             if (m.hasOwnProperty(i) && momentProperties.hasOwnProperty(i)) {
394                 result[i] = m[i];
395             }
396         }
398         return result;
399     }
401     function absRound(number) {
402         if (number < 0) {
403             return Math.ceil(number);
404         } else {
405             return Math.floor(number);
406         }
407     }
409     // left zero fill a number
410     // see http://jsperf.com/left-zero-filling for performance comparison
411     function leftZeroFill(number, targetLength, forceSign) {
412         var output = '' + Math.abs(number),
413             sign = number >= 0;
415         while (output.length < targetLength) {
416             output = '0' + output;
417         }
418         return (sign ? (forceSign ? '+' : '') : '-') + output;
419     }
421     // helper function for _.addTime and _.subtractTime
422     function addOrSubtractDurationFromMoment(mom, duration, isAdding, ignoreUpdateOffset) {
423         var milliseconds = duration._milliseconds,
424             days = duration._days,
425             months = duration._months,
426             minutes,
427             hours;
429         if (milliseconds) {
430             mom._d.setTime(+mom._d + milliseconds * isAdding);
431         }
432         // store the minutes and hours so we can restore them
433         if (days || months) {
434             minutes = mom.minute();
435             hours = mom.hour();
436         }
437         if (days) {
438             mom.date(mom.date() + days * isAdding);
439         }
440         if (months) {
441             mom.month(mom.month() + months * isAdding);
442         }
443         if (milliseconds && !ignoreUpdateOffset) {
444             moment.updateOffset(mom);
445         }
446         // restore the minutes and hours after possibly changing dst
447         if (days || months) {
448             mom.minute(minutes);
449             mom.hour(hours);
450         }
451     }
453     // check if is an array
454     function isArray(input) {
455         return Object.prototype.toString.call(input) === '[object Array]';
456     }
458     function isDate(input) {
459         return  Object.prototype.toString.call(input) === '[object Date]' ||
460                 input instanceof Date;
461     }
463     // compare two arrays, return the number of differences
464     function compareArrays(array1, array2, dontConvert) {
465         var len = Math.min(array1.length, array2.length),
466             lengthDiff = Math.abs(array1.length - array2.length),
467             diffs = 0,
468             i;
469         for (i = 0; i < len; i++) {
470             if ((dontConvert && array1[i] !== array2[i]) ||
471                 (!dontConvert && toInt(array1[i]) !== toInt(array2[i]))) {
472                 diffs++;
473             }
474         }
475         return diffs + lengthDiff;
476     }
478     function normalizeUnits(units) {
479         if (units) {
480             var lowered = units.toLowerCase().replace(/(.)s$/, '$1');
481             units = unitAliases[units] || camelFunctions[lowered] || lowered;
482         }
483         return units;
484     }
486     function normalizeObjectUnits(inputObject) {
487         var normalizedInput = {},
488             normalizedProp,
489             prop;
491         for (prop in inputObject) {
492             if (inputObject.hasOwnProperty(prop)) {
493                 normalizedProp = normalizeUnits(prop);
494                 if (normalizedProp) {
495                     normalizedInput[normalizedProp] = inputObject[prop];
496                 }
497             }
498         }
500         return normalizedInput;
501     }
503     function makeList(field) {
504         var count, setter;
506         if (field.indexOf('week') === 0) {
507             count = 7;
508             setter = 'day';
509         }
510         else if (field.indexOf('month') === 0) {
511             count = 12;
512             setter = 'month';
513         }
514         else {
515             return;
516         }
518         moment[field] = function (format, index) {
519             var i, getter,
520                 method = moment.fn._lang[field],
521                 results = [];
523             if (typeof format === 'number') {
524                 index = format;
525                 format = undefined;
526             }
528             getter = function (i) {
529                 var m = moment().utc().set(setter, i);
530                 return method.call(moment.fn._lang, m, format || '');
531             };
533             if (index != null) {
534                 return getter(index);
535             }
536             else {
537                 for (i = 0; i < count; i++) {
538                     results.push(getter(i));
539                 }
540                 return results;
541             }
542         };
543     }
545     function toInt(argumentForCoercion) {
546         var coercedNumber = +argumentForCoercion,
547             value = 0;
549         if (coercedNumber !== 0 && isFinite(coercedNumber)) {
550             if (coercedNumber >= 0) {
551                 value = Math.floor(coercedNumber);
552             } else {
553                 value = Math.ceil(coercedNumber);
554             }
555         }
557         return value;
558     }
560     function daysInMonth(year, month) {
561         return new Date(Date.UTC(year, month + 1, 0)).getUTCDate();
562     }
564     function daysInYear(year) {
565         return isLeapYear(year) ? 366 : 365;
566     }
568     function isLeapYear(year) {
569         return (year % 4 === 0 && year % 100 !== 0) || year % 400 === 0;
570     }
572     function checkOverflow(m) {
573         var overflow;
574         if (m._a && m._pf.overflow === -2) {
575             overflow =
576                 m._a[MONTH] < 0 || m._a[MONTH] > 11 ? MONTH :
577                 m._a[DATE] < 1 || m._a[DATE] > daysInMonth(m._a[YEAR], m._a[MONTH]) ? DATE :
578                 m._a[HOUR] < 0 || m._a[HOUR] > 23 ? HOUR :
579                 m._a[MINUTE] < 0 || m._a[MINUTE] > 59 ? MINUTE :
580                 m._a[SECOND] < 0 || m._a[SECOND] > 59 ? SECOND :
581                 m._a[MILLISECOND] < 0 || m._a[MILLISECOND] > 999 ? MILLISECOND :
582                 -1;
584             if (m._pf._overflowDayOfYear && (overflow < YEAR || overflow > DATE)) {
585                 overflow = DATE;
586             }
588             m._pf.overflow = overflow;
589         }
590     }
592     function isValid(m) {
593         if (m._isValid == null) {
594             m._isValid = !isNaN(m._d.getTime()) &&
595                 m._pf.overflow < 0 &&
596                 !m._pf.empty &&
597                 !m._pf.invalidMonth &&
598                 !m._pf.nullInput &&
599                 !m._pf.invalidFormat &&
600                 !m._pf.userInvalidated;
602             if (m._strict) {
603                 m._isValid = m._isValid &&
604                     m._pf.charsLeftOver === 0 &&
605                     m._pf.unusedTokens.length === 0;
606             }
607         }
608         return m._isValid;
609     }
611     function normalizeLanguage(key) {
612         return key ? key.toLowerCase().replace('_', '-') : key;
613     }
615     // Return a moment from input, that is local/utc/zone equivalent to model.
616     function makeAs(input, model) {
617         return model._isUTC ? moment(input).zone(model._offset || 0) :
618             moment(input).local();
619     }
621     /************************************
622         Languages
623     ************************************/
626     extend(Language.prototype, {
628         set : function (config) {
629             var prop, i;
630             for (i in config) {
631                 prop = config[i];
632                 if (typeof prop === 'function') {
633                     this[i] = prop;
634                 } else {
635                     this['_' + i] = prop;
636                 }
637             }
638         },
640         _months : "January_February_March_April_May_June_July_August_September_October_November_December".split("_"),
641         months : function (m) {
642             return this._months[m.month()];
643         },
645         _monthsShort : "Jan_Feb_Mar_Apr_May_Jun_Jul_Aug_Sep_Oct_Nov_Dec".split("_"),
646         monthsShort : function (m) {
647             return this._monthsShort[m.month()];
648         },
650         monthsParse : function (monthName) {
651             var i, mom, regex;
653             if (!this._monthsParse) {
654                 this._monthsParse = [];
655             }
657             for (i = 0; i < 12; i++) {
658                 // make the regex if we don't have it already
659                 if (!this._monthsParse[i]) {
660                     mom = moment.utc([2000, i]);
661                     regex = '^' + this.months(mom, '') + '|^' + this.monthsShort(mom, '');
662                     this._monthsParse[i] = new RegExp(regex.replace('.', ''), 'i');
663                 }
664                 // test the regex
665                 if (this._monthsParse[i].test(monthName)) {
666                     return i;
667                 }
668             }
669         },
671         _weekdays : "Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday".split("_"),
672         weekdays : function (m) {
673             return this._weekdays[m.day()];
674         },
676         _weekdaysShort : "Sun_Mon_Tue_Wed_Thu_Fri_Sat".split("_"),
677         weekdaysShort : function (m) {
678             return this._weekdaysShort[m.day()];
679         },
681         _weekdaysMin : "Su_Mo_Tu_We_Th_Fr_Sa".split("_"),
682         weekdaysMin : function (m) {
683             return this._weekdaysMin[m.day()];
684         },
686         weekdaysParse : function (weekdayName) {
687             var i, mom, regex;
689             if (!this._weekdaysParse) {
690                 this._weekdaysParse = [];
691             }
693             for (i = 0; i < 7; i++) {
694                 // make the regex if we don't have it already
695                 if (!this._weekdaysParse[i]) {
696                     mom = moment([2000, 1]).day(i);
697                     regex = '^' + this.weekdays(mom, '') + '|^' + this.weekdaysShort(mom, '') + '|^' + this.weekdaysMin(mom, '');
698                     this._weekdaysParse[i] = new RegExp(regex.replace('.', ''), 'i');
699                 }
700                 // test the regex
701                 if (this._weekdaysParse[i].test(weekdayName)) {
702                     return i;
703                 }
704             }
705         },
707         _longDateFormat : {
708             LT : "h:mm A",
709             L : "MM/DD/YYYY",
710             LL : "MMMM D YYYY",
711             LLL : "MMMM D YYYY LT",
712             LLLL : "dddd, MMMM D YYYY LT"
713         },
714         longDateFormat : function (key) {
715             var output = this._longDateFormat[key];
716             if (!output && this._longDateFormat[key.toUpperCase()]) {
717                 output = this._longDateFormat[key.toUpperCase()].replace(/MMMM|MM|DD|dddd/g, function (val) {
718                     return val.slice(1);
719                 });
720                 this._longDateFormat[key] = output;
721             }
722             return output;
723         },
725         isPM : function (input) {
726             // IE8 Quirks Mode & IE7 Standards Mode do not allow accessing strings like arrays
727             // Using charAt should be more compatible.
728             return ((input + '').toLowerCase().charAt(0) === 'p');
729         },
731         _meridiemParse : /[ap]\.?m?\.?/i,
732         meridiem : function (hours, minutes, isLower) {
733             if (hours > 11) {
734                 return isLower ? 'pm' : 'PM';
735             } else {
736                 return isLower ? 'am' : 'AM';
737             }
738         },
740         _calendar : {
741             sameDay : '[Today at] LT',
742             nextDay : '[Tomorrow at] LT',
743             nextWeek : 'dddd [at] LT',
744             lastDay : '[Yesterday at] LT',
745             lastWeek : '[Last] dddd [at] LT',
746             sameElse : 'L'
747         },
748         calendar : function (key, mom) {
749             var output = this._calendar[key];
750             return typeof output === 'function' ? output.apply(mom) : output;
751         },
753         _relativeTime : {
754             future : "in %s",
755             past : "%s ago",
756             s : "a few seconds",
757             m : "a minute",
758             mm : "%d minutes",
759             h : "an hour",
760             hh : "%d hours",
761             d : "a day",
762             dd : "%d days",
763             M : "a month",
764             MM : "%d months",
765             y : "a year",
766             yy : "%d years"
767         },
768         relativeTime : function (number, withoutSuffix, string, isFuture) {
769             var output = this._relativeTime[string];
770             return (typeof output === 'function') ?
771                 output(number, withoutSuffix, string, isFuture) :
772                 output.replace(/%d/i, number);
773         },
774         pastFuture : function (diff, output) {
775             var format = this._relativeTime[diff > 0 ? 'future' : 'past'];
776             return typeof format === 'function' ? format(output) : format.replace(/%s/i, output);
777         },
779         ordinal : function (number) {
780             return this._ordinal.replace("%d", number);
781         },
782         _ordinal : "%d",
784         preparse : function (string) {
785             return string;
786         },
788         postformat : function (string) {
789             return string;
790         },
792         week : function (mom) {
793             return weekOfYear(mom, this._week.dow, this._week.doy).week;
794         },
796         _week : {
797             dow : 0, // Sunday is the first day of the week.
798             doy : 6  // The week that contains Jan 1st is the first week of the year.
799         },
801         _invalidDate: 'Invalid date',
802         invalidDate: function () {
803             return this._invalidDate;
804         }
805     });
807     // Loads a language definition into the `languages` cache.  The function
808     // takes a key and optionally values.  If not in the browser and no values
809     // are provided, it will load the language file module.  As a convenience,
810     // this function also returns the language values.
811     function loadLang(key, values) {
812         values.abbr = key;
813         if (!languages[key]) {
814             languages[key] = new Language();
815         }
816         languages[key].set(values);
817         return languages[key];
818     }
820     // Remove a language from the `languages` cache. Mostly useful in tests.
821     function unloadLang(key) {
822         delete languages[key];
823     }
825     // Determines which language definition to use and returns it.
826     //
827     // With no parameters, it will return the global language.  If you
828     // pass in a language key, such as 'en', it will return the
829     // definition for 'en', so long as 'en' has already been loaded using
830     // moment.lang.
831     function getLangDefinition(key) {
832         var i = 0, j, lang, next, split,
833             get = function (k) {
834                 if (!languages[k] && hasModule) {
835                     try {
836                         require('./lang/' + k);
837                     } catch (e) { }
838                 }
839                 return languages[k];
840             };
842         if (!key) {
843             return moment.fn._lang;
844         }
846         if (!isArray(key)) {
847             //short-circuit everything else
848             lang = get(key);
849             if (lang) {
850                 return lang;
851             }
852             key = [key];
853         }
855         //pick the language from the array
856         //try ['en-au', 'en-gb'] as 'en-au', 'en-gb', 'en', as in move through the list trying each
857         //substring from most specific to least, but move to the next array item if it's a more specific variant than the current root
858         while (i < key.length) {
859             split = normalizeLanguage(key[i]).split('-');
860             j = split.length;
861             next = normalizeLanguage(key[i + 1]);
862             next = next ? next.split('-') : null;
863             while (j > 0) {
864                 lang = get(split.slice(0, j).join('-'));
865                 if (lang) {
866                     return lang;
867                 }
868                 if (next && next.length >= j && compareArrays(split, next, true) >= j - 1) {
869                     //the next array item is better than a shallower substring of this one
870                     break;
871                 }
872                 j--;
873             }
874             i++;
875         }
876         return moment.fn._lang;
877     }
879     /************************************
880         Formatting
881     ************************************/
884     function removeFormattingTokens(input) {
885         if (input.match(/\[[\s\S]/)) {
886             return input.replace(/^\[|\]$/g, "");
887         }
888         return input.replace(/\\/g, "");
889     }
891     function makeFormatFunction(format) {
892         var array = format.match(formattingTokens), i, length;
894         for (i = 0, length = array.length; i < length; i++) {
895             if (formatTokenFunctions[array[i]]) {
896                 array[i] = formatTokenFunctions[array[i]];
897             } else {
898                 array[i] = removeFormattingTokens(array[i]);
899             }
900         }
902         return function (mom) {
903             var output = "";
904             for (i = 0; i < length; i++) {
905                 output += array[i] instanceof Function ? array[i].call(mom, format) : array[i];
906             }
907             return output;
908         };
909     }
911     // format date using native date object
912     function formatMoment(m, format) {
914         if (!m.isValid()) {
915             return m.lang().invalidDate();
916         }
918         format = expandFormat(format, m.lang());
920         if (!formatFunctions[format]) {
921             formatFunctions[format] = makeFormatFunction(format);
922         }
924         return formatFunctions[format](m);
925     }
927     function expandFormat(format, lang) {
928         var i = 5;
930         function replaceLongDateFormatTokens(input) {
931             return lang.longDateFormat(input) || input;
932         }
934         localFormattingTokens.lastIndex = 0;
935         while (i >= 0 && localFormattingTokens.test(format)) {
936             format = format.replace(localFormattingTokens, replaceLongDateFormatTokens);
937             localFormattingTokens.lastIndex = 0;
938             i -= 1;
939         }
941         return format;
942     }
945     /************************************
946         Parsing
947     ************************************/
950     // get the regex to find the next token
951     function getParseRegexForToken(token, config) {
952         var a, strict = config._strict;
953         switch (token) {
954         case 'DDDD':
955             return parseTokenThreeDigits;
956         case 'YYYY':
957         case 'GGGG':
958         case 'gggg':
959             return strict ? parseTokenFourDigits : parseTokenOneToFourDigits;
960         case 'Y':
961         case 'G':
962         case 'g':
963             return parseTokenSignedNumber;
964         case 'YYYYYY':
965         case 'YYYYY':
966         case 'GGGGG':
967         case 'ggggg':
968             return strict ? parseTokenSixDigits : parseTokenOneToSixDigits;
969         case 'S':
970             if (strict) { return parseTokenOneDigit; }
971             /* falls through */
972         case 'SS':
973             if (strict) { return parseTokenTwoDigits; }
974             /* falls through */
975         case 'SSS':
976             if (strict) { return parseTokenThreeDigits; }
977             /* falls through */
978         case 'DDD':
979             return parseTokenOneToThreeDigits;
980         case 'MMM':
981         case 'MMMM':
982         case 'dd':
983         case 'ddd':
984         case 'dddd':
985             return parseTokenWord;
986         case 'a':
987         case 'A':
988             return getLangDefinition(config._l)._meridiemParse;
989         case 'X':
990             return parseTokenTimestampMs;
991         case 'Z':
992         case 'ZZ':
993             return parseTokenTimezone;
994         case 'T':
995             return parseTokenT;
996         case 'SSSS':
997             return parseTokenDigits;
998         case 'MM':
999         case 'DD':
1000         case 'YY':
1001         case 'GG':
1002         case 'gg':
1003         case 'HH':
1004         case 'hh':
1005         case 'mm':
1006         case 'ss':
1007         case 'ww':
1008         case 'WW':
1009             return strict ? parseTokenTwoDigits : parseTokenOneOrTwoDigits;
1010         case 'M':
1011         case 'D':
1012         case 'd':
1013         case 'H':
1014         case 'h':
1015         case 'm':
1016         case 's':
1017         case 'w':
1018         case 'W':
1019         case 'e':
1020         case 'E':
1021             return parseTokenOneOrTwoDigits;
1022         default :
1023             a = new RegExp(regexpEscape(unescapeFormat(token.replace('\\', '')), "i"));
1024             return a;
1025         }
1026     }
1028     function timezoneMinutesFromString(string) {
1029         string = string || "";
1030         var possibleTzMatches = (string.match(parseTokenTimezone) || []),
1031             tzChunk = possibleTzMatches[possibleTzMatches.length - 1] || [],
1032             parts = (tzChunk + '').match(parseTimezoneChunker) || ['-', 0, 0],
1033             minutes = +(parts[1] * 60) + toInt(parts[2]);
1035         return parts[0] === '+' ? -minutes : minutes;
1036     }
1038     // function to convert string input to date
1039     function addTimeToArrayFromToken(token, input, config) {
1040         var a, datePartArray = config._a;
1042         switch (token) {
1043         // MONTH
1044         case 'M' : // fall through to MM
1045         case 'MM' :
1046             if (input != null) {
1047                 datePartArray[MONTH] = toInt(input) - 1;
1048             }
1049             break;
1050         case 'MMM' : // fall through to MMMM
1051         case 'MMMM' :
1052             a = getLangDefinition(config._l).monthsParse(input);
1053             // if we didn't find a month name, mark the date as invalid.
1054             if (a != null) {
1055                 datePartArray[MONTH] = a;
1056             } else {
1057                 config._pf.invalidMonth = input;
1058             }
1059             break;
1060         // DAY OF MONTH
1061         case 'D' : // fall through to DD
1062         case 'DD' :
1063             if (input != null) {
1064                 datePartArray[DATE] = toInt(input);
1065             }
1066             break;
1067         // DAY OF YEAR
1068         case 'DDD' : // fall through to DDDD
1069         case 'DDDD' :
1070             if (input != null) {
1071                 config._dayOfYear = toInt(input);
1072             }
1074             break;
1075         // YEAR
1076         case 'YY' :
1077             datePartArray[YEAR] = toInt(input) + (toInt(input) > 68 ? 1900 : 2000);
1078             break;
1079         case 'YYYY' :
1080         case 'YYYYY' :
1081         case 'YYYYYY' :
1082             datePartArray[YEAR] = toInt(input);
1083             break;
1084         // AM / PM
1085         case 'a' : // fall through to A
1086         case 'A' :
1087             config._isPm = getLangDefinition(config._l).isPM(input);
1088             break;
1089         // 24 HOUR
1090         case 'H' : // fall through to hh
1091         case 'HH' : // fall through to hh
1092         case 'h' : // fall through to hh
1093         case 'hh' :
1094             datePartArray[HOUR] = toInt(input);
1095             break;
1096         // MINUTE
1097         case 'm' : // fall through to mm
1098         case 'mm' :
1099             datePartArray[MINUTE] = toInt(input);
1100             break;
1101         // SECOND
1102         case 's' : // fall through to ss
1103         case 'ss' :
1104             datePartArray[SECOND] = toInt(input);
1105             break;
1106         // MILLISECOND
1107         case 'S' :
1108         case 'SS' :
1109         case 'SSS' :
1110         case 'SSSS' :
1111             datePartArray[MILLISECOND] = toInt(('0.' + input) * 1000);
1112             break;
1113         // UNIX TIMESTAMP WITH MS
1114         case 'X':
1115             config._d = new Date(parseFloat(input) * 1000);
1116             break;
1117         // TIMEZONE
1118         case 'Z' : // fall through to ZZ
1119         case 'ZZ' :
1120             config._useUTC = true;
1121             config._tzm = timezoneMinutesFromString(input);
1122             break;
1123         case 'w':
1124         case 'ww':
1125         case 'W':
1126         case 'WW':
1127         case 'd':
1128         case 'dd':
1129         case 'ddd':
1130         case 'dddd':
1131         case 'e':
1132         case 'E':
1133             token = token.substr(0, 1);
1134             /* falls through */
1135         case 'gg':
1136         case 'gggg':
1137         case 'GG':
1138         case 'GGGG':
1139         case 'GGGGG':
1140             token = token.substr(0, 2);
1141             if (input) {
1142                 config._w = config._w || {};
1143                 config._w[token] = input;
1144             }
1145             break;
1146         }
1147     }
1149     // convert an array to a date.
1150     // the array should mirror the parameters below
1151     // note: all values past the year are optional and will default to the lowest possible value.
1152     // [year, month, day , hour, minute, second, millisecond]
1153     function dateFromConfig(config) {
1154         var i, date, input = [], currentDate,
1155             yearToUse, fixYear, w, temp, lang, weekday, week;
1157         if (config._d) {
1158             return;
1159         }
1161         currentDate = currentDateArray(config);
1163         //compute day of the year from weeks and weekdays
1164         if (config._w && config._a[DATE] == null && config._a[MONTH] == null) {
1165             fixYear = function (val) {
1166                 var int_val = parseInt(val, 10);
1167                 return val ?
1168                   (val.length < 3 ? (int_val > 68 ? 1900 + int_val : 2000 + int_val) : int_val) :
1169                   (config._a[YEAR] == null ? moment().weekYear() : config._a[YEAR]);
1170             };
1172             w = config._w;
1173             if (w.GG != null || w.W != null || w.E != null) {
1174                 temp = dayOfYearFromWeeks(fixYear(w.GG), w.W || 1, w.E, 4, 1);
1175             }
1176             else {
1177                 lang = getLangDefinition(config._l);
1178                 weekday = w.d != null ?  parseWeekday(w.d, lang) :
1179                   (w.e != null ?  parseInt(w.e, 10) + lang._week.dow : 0);
1181                 week = parseInt(w.w, 10) || 1;
1183                 //if we're parsing 'd', then the low day numbers may be next week
1184                 if (w.d != null && weekday < lang._week.dow) {
1185                     week++;
1186                 }
1188                 temp = dayOfYearFromWeeks(fixYear(w.gg), week, weekday, lang._week.doy, lang._week.dow);
1189             }
1191             config._a[YEAR] = temp.year;
1192             config._dayOfYear = temp.dayOfYear;
1193         }
1195         //if the day of the year is set, figure out what it is
1196         if (config._dayOfYear) {
1197             yearToUse = config._a[YEAR] == null ? currentDate[YEAR] : config._a[YEAR];
1199             if (config._dayOfYear > daysInYear(yearToUse)) {
1200                 config._pf._overflowDayOfYear = true;
1201             }
1203             date = makeUTCDate(yearToUse, 0, config._dayOfYear);
1204             config._a[MONTH] = date.getUTCMonth();
1205             config._a[DATE] = date.getUTCDate();
1206         }
1208         // Default to current date.
1209         // * if no year, month, day of month are given, default to today
1210         // * if day of month is given, default month and year
1211         // * if month is given, default only year
1212         // * if year is given, don't default anything
1213         for (i = 0; i < 3 && config._a[i] == null; ++i) {
1214             config._a[i] = input[i] = currentDate[i];
1215         }
1217         // Zero out whatever was not defaulted, including time
1218         for (; i < 7; i++) {
1219             config._a[i] = input[i] = (config._a[i] == null) ? (i === 2 ? 1 : 0) : config._a[i];
1220         }
1222         // add the offsets to the time to be parsed so that we can have a clean array for checking isValid
1223         input[HOUR] += toInt((config._tzm || 0) / 60);
1224         input[MINUTE] += toInt((config._tzm || 0) % 60);
1226         config._d = (config._useUTC ? makeUTCDate : makeDate).apply(null, input);
1227     }
1229     function dateFromObject(config) {
1230         var normalizedInput;
1232         if (config._d) {
1233             return;
1234         }
1236         normalizedInput = normalizeObjectUnits(config._i);
1237         config._a = [
1238             normalizedInput.year,
1239             normalizedInput.month,
1240             normalizedInput.day,
1241             normalizedInput.hour,
1242             normalizedInput.minute,
1243             normalizedInput.second,
1244             normalizedInput.millisecond
1245         ];
1247         dateFromConfig(config);
1248     }
1250     function currentDateArray(config) {
1251         var now = new Date();
1252         if (config._useUTC) {
1253             return [
1254                 now.getUTCFullYear(),
1255                 now.getUTCMonth(),
1256                 now.getUTCDate()
1257             ];
1258         } else {
1259             return [now.getFullYear(), now.getMonth(), now.getDate()];
1260         }
1261     }
1263     // date from string and format string
1264     function makeDateFromStringAndFormat(config) {
1266         config._a = [];
1267         config._pf.empty = true;
1269         // This array is used to make a Date, either with `new Date` or `Date.UTC`
1270         var lang = getLangDefinition(config._l),
1271             string = '' + config._i,
1272             i, parsedInput, tokens, token, skipped,
1273             stringLength = string.length,
1274             totalParsedInputLength = 0;
1276         tokens = expandFormat(config._f, lang).match(formattingTokens) || [];
1278         for (i = 0; i < tokens.length; i++) {
1279             token = tokens[i];
1280             parsedInput = (string.match(getParseRegexForToken(token, config)) || [])[0];
1281             if (parsedInput) {
1282                 skipped = string.substr(0, string.indexOf(parsedInput));
1283                 if (skipped.length > 0) {
1284                     config._pf.unusedInput.push(skipped);
1285                 }
1286                 string = string.slice(string.indexOf(parsedInput) + parsedInput.length);
1287                 totalParsedInputLength += parsedInput.length;
1288             }
1289             // don't parse if it's not a known token
1290             if (formatTokenFunctions[token]) {
1291                 if (parsedInput) {
1292                     config._pf.empty = false;
1293                 }
1294                 else {
1295                     config._pf.unusedTokens.push(token);
1296                 }
1297                 addTimeToArrayFromToken(token, parsedInput, config);
1298             }
1299             else if (config._strict && !parsedInput) {
1300                 config._pf.unusedTokens.push(token);
1301             }
1302         }
1304         // add remaining unparsed input length to the string
1305         config._pf.charsLeftOver = stringLength - totalParsedInputLength;
1306         if (string.length > 0) {
1307             config._pf.unusedInput.push(string);
1308         }
1310         // handle am pm
1311         if (config._isPm && config._a[HOUR] < 12) {
1312             config._a[HOUR] += 12;
1313         }
1314         // if is 12 am, change hours to 0
1315         if (config._isPm === false && config._a[HOUR] === 12) {
1316             config._a[HOUR] = 0;
1317         }
1319         dateFromConfig(config);
1320         checkOverflow(config);
1321     }
1323     function unescapeFormat(s) {
1324         return s.replace(/\\(\[)|\\(\])|\[([^\]\[]*)\]|\\(.)/g, function (matched, p1, p2, p3, p4) {
1325             return p1 || p2 || p3 || p4;
1326         });
1327     }
1329     // Code from http://stackoverflow.com/questions/3561493/is-there-a-regexp-escape-function-in-javascript
1330     function regexpEscape(s) {
1331         return s.replace(/[-\/\\^$*+?.()|[\]{}]/g, '\\$&');
1332     }
1334     // date from string and array of format strings
1335     function makeDateFromStringAndArray(config) {
1336         var tempConfig,
1337             bestMoment,
1339             scoreToBeat,
1340             i,
1341             currentScore;
1343         if (config._f.length === 0) {
1344             config._pf.invalidFormat = true;
1345             config._d = new Date(NaN);
1346             return;
1347         }
1349         for (i = 0; i < config._f.length; i++) {
1350             currentScore = 0;
1351             tempConfig = extend({}, config);
1352             tempConfig._pf = defaultParsingFlags();
1353             tempConfig._f = config._f[i];
1354             makeDateFromStringAndFormat(tempConfig);
1356             if (!isValid(tempConfig)) {
1357                 continue;
1358             }
1360             // if there is any input that was not parsed add a penalty for that format
1361             currentScore += tempConfig._pf.charsLeftOver;
1363             //or tokens
1364             currentScore += tempConfig._pf.unusedTokens.length * 10;
1366             tempConfig._pf.score = currentScore;
1368             if (scoreToBeat == null || currentScore < scoreToBeat) {
1369                 scoreToBeat = currentScore;
1370                 bestMoment = tempConfig;
1371             }
1372         }
1374         extend(config, bestMoment || tempConfig);
1375     }
1377     // date from iso format
1378     function makeDateFromString(config) {
1379         var i, l,
1380             string = config._i,
1381             match = isoRegex.exec(string);
1383         if (match) {
1384             config._pf.iso = true;
1385             for (i = 0, l = isoDates.length; i < l; i++) {
1386                 if (isoDates[i][1].exec(string)) {
1387                     // match[5] should be "T" or undefined
1388                     config._f = isoDates[i][0] + (match[6] || " ");
1389                     break;
1390                 }
1391             }
1392             for (i = 0, l = isoTimes.length; i < l; i++) {
1393                 if (isoTimes[i][1].exec(string)) {
1394                     config._f += isoTimes[i][0];
1395                     break;
1396                 }
1397             }
1398             if (string.match(parseTokenTimezone)) {
1399                 config._f += "Z";
1400             }
1401             makeDateFromStringAndFormat(config);
1402         }
1403         else {
1404             config._d = new Date(string);
1405         }
1406     }
1408     function makeDateFromInput(config) {
1409         var input = config._i,
1410             matched = aspNetJsonRegex.exec(input);
1412         if (input === undefined) {
1413             config._d = new Date();
1414         } else if (matched) {
1415             config._d = new Date(+matched[1]);
1416         } else if (typeof input === 'string') {
1417             makeDateFromString(config);
1418         } else if (isArray(input)) {
1419             config._a = input.slice(0);
1420             dateFromConfig(config);
1421         } else if (isDate(input)) {
1422             config._d = new Date(+input);
1423         } else if (typeof(input) === 'object') {
1424             dateFromObject(config);
1425         } else {
1426             config._d = new Date(input);
1427         }
1428     }
1430     function makeDate(y, m, d, h, M, s, ms) {
1431         //can't just apply() to create a date:
1432         //http://stackoverflow.com/questions/181348/instantiating-a-javascript-object-by-calling-prototype-constructor-apply
1433         var date = new Date(y, m, d, h, M, s, ms);
1435         //the date constructor doesn't accept years < 1970
1436         if (y < 1970) {
1437             date.setFullYear(y);
1438         }
1439         return date;
1440     }
1442     function makeUTCDate(y) {
1443         var date = new Date(Date.UTC.apply(null, arguments));
1444         if (y < 1970) {
1445             date.setUTCFullYear(y);
1446         }
1447         return date;
1448     }
1450     function parseWeekday(input, language) {
1451         if (typeof input === 'string') {
1452             if (!isNaN(input)) {
1453                 input = parseInt(input, 10);
1454             }
1455             else {
1456                 input = language.weekdaysParse(input);
1457                 if (typeof input !== 'number') {
1458                     return null;
1459                 }
1460             }
1461         }
1462         return input;
1463     }
1465     /************************************
1466         Relative Time
1467     ************************************/
1470     // helper function for moment.fn.from, moment.fn.fromNow, and moment.duration.fn.humanize
1471     function substituteTimeAgo(string, number, withoutSuffix, isFuture, lang) {
1472         return lang.relativeTime(number || 1, !!withoutSuffix, string, isFuture);
1473     }
1475     function relativeTime(milliseconds, withoutSuffix, lang) {
1476         var seconds = round(Math.abs(milliseconds) / 1000),
1477             minutes = round(seconds / 60),
1478             hours = round(minutes / 60),
1479             days = round(hours / 24),
1480             years = round(days / 365),
1481             args = seconds < 45 && ['s', seconds] ||
1482                 minutes === 1 && ['m'] ||
1483                 minutes < 45 && ['mm', minutes] ||
1484                 hours === 1 && ['h'] ||
1485                 hours < 22 && ['hh', hours] ||
1486                 days === 1 && ['d'] ||
1487                 days <= 25 && ['dd', days] ||
1488                 days <= 45 && ['M'] ||
1489                 days < 345 && ['MM', round(days / 30)] ||
1490                 years === 1 && ['y'] || ['yy', years];
1491         args[2] = withoutSuffix;
1492         args[3] = milliseconds > 0;
1493         args[4] = lang;
1494         return substituteTimeAgo.apply({}, args);
1495     }
1498     /************************************
1499         Week of Year
1500     ************************************/
1503     // firstDayOfWeek       0 = sun, 6 = sat
1504     //                      the day of the week that starts the week
1505     //                      (usually sunday or monday)
1506     // firstDayOfWeekOfYear 0 = sun, 6 = sat
1507     //                      the first week is the week that contains the first
1508     //                      of this day of the week
1509     //                      (eg. ISO weeks use thursday (4))
1510     function weekOfYear(mom, firstDayOfWeek, firstDayOfWeekOfYear) {
1511         var end = firstDayOfWeekOfYear - firstDayOfWeek,
1512             daysToDayOfWeek = firstDayOfWeekOfYear - mom.day(),
1513             adjustedMoment;
1516         if (daysToDayOfWeek > end) {
1517             daysToDayOfWeek -= 7;
1518         }
1520         if (daysToDayOfWeek < end - 7) {
1521             daysToDayOfWeek += 7;
1522         }
1524         adjustedMoment = moment(mom).add('d', daysToDayOfWeek);
1525         return {
1526             week: Math.ceil(adjustedMoment.dayOfYear() / 7),
1527             year: adjustedMoment.year()
1528         };
1529     }
1531     //http://en.wikipedia.org/wiki/ISO_week_date#Calculating_a_date_given_the_year.2C_week_number_and_weekday
1532     function dayOfYearFromWeeks(year, week, weekday, firstDayOfWeekOfYear, firstDayOfWeek) {
1533         var d = makeUTCDate(year, 0, 1).getUTCDay(), daysToAdd, dayOfYear;
1535         weekday = weekday != null ? weekday : firstDayOfWeek;
1536         daysToAdd = firstDayOfWeek - d + (d > firstDayOfWeekOfYear ? 7 : 0) - (d < firstDayOfWeek ? 7 : 0);
1537         dayOfYear = 7 * (week - 1) + (weekday - firstDayOfWeek) + daysToAdd + 1;
1539         return {
1540             year: dayOfYear > 0 ? year : year - 1,
1541             dayOfYear: dayOfYear > 0 ?  dayOfYear : daysInYear(year - 1) + dayOfYear
1542         };
1543     }
1545     /************************************
1546         Top Level Functions
1547     ************************************/
1549     function makeMoment(config) {
1550         var input = config._i,
1551             format = config._f;
1553         if (input === null) {
1554             return moment.invalid({nullInput: true});
1555         }
1557         if (typeof input === 'string') {
1558             config._i = input = getLangDefinition().preparse(input);
1559         }
1561         if (moment.isMoment(input)) {
1562             config = cloneMoment(input);
1564             config._d = new Date(+input._d);
1565         } else if (format) {
1566             if (isArray(format)) {
1567                 makeDateFromStringAndArray(config);
1568             } else {
1569                 makeDateFromStringAndFormat(config);
1570             }
1571         } else {
1572             makeDateFromInput(config);
1573         }
1575         return new Moment(config);
1576     }
1578     moment = function (input, format, lang, strict) {
1579         var c;
1581         if (typeof(lang) === "boolean") {
1582             strict = lang;
1583             lang = undefined;
1584         }
1585         // object construction must be done this way.
1586         // https://github.com/moment/moment/issues/1423
1587         c = {};
1588         c._isAMomentObject = true;
1589         c._i = input;
1590         c._f = format;
1591         c._l = lang;
1592         c._strict = strict;
1593         c._isUTC = false;
1594         c._pf = defaultParsingFlags();
1596         return makeMoment(c);
1597     };
1599     // creating with utc
1600     moment.utc = function (input, format, lang, strict) {
1601         var c;
1603         if (typeof(lang) === "boolean") {
1604             strict = lang;
1605             lang = undefined;
1606         }
1607         // object construction must be done this way.
1608         // https://github.com/moment/moment/issues/1423
1609         c = {};
1610         c._isAMomentObject = true;
1611         c._useUTC = true;
1612         c._isUTC = true;
1613         c._l = lang;
1614         c._i = input;
1615         c._f = format;
1616         c._strict = strict;
1617         c._pf = defaultParsingFlags();
1619         return makeMoment(c).utc();
1620     };
1622     // creating with unix timestamp (in seconds)
1623     moment.unix = function (input) {
1624         return moment(input * 1000);
1625     };
1627     // duration
1628     moment.duration = function (input, key) {
1629         var duration = input,
1630             // matching against regexp is expensive, do it on demand
1631             match = null,
1632             sign,
1633             ret,
1634             parseIso;
1636         if (moment.isDuration(input)) {
1637             duration = {
1638                 ms: input._milliseconds,
1639                 d: input._days,
1640                 M: input._months
1641             };
1642         } else if (typeof input === 'number') {
1643             duration = {};
1644             if (key) {
1645                 duration[key] = input;
1646             } else {
1647                 duration.milliseconds = input;
1648             }
1649         } else if (!!(match = aspNetTimeSpanJsonRegex.exec(input))) {
1650             sign = (match[1] === "-") ? -1 : 1;
1651             duration = {
1652                 y: 0,
1653                 d: toInt(match[DATE]) * sign,
1654                 h: toInt(match[HOUR]) * sign,
1655                 m: toInt(match[MINUTE]) * sign,
1656                 s: toInt(match[SECOND]) * sign,
1657                 ms: toInt(match[MILLISECOND]) * sign
1658             };
1659         } else if (!!(match = isoDurationRegex.exec(input))) {
1660             sign = (match[1] === "-") ? -1 : 1;
1661             parseIso = function (inp) {
1662                 // We'd normally use ~~inp for this, but unfortunately it also
1663                 // converts floats to ints.
1664                 // inp may be undefined, so careful calling replace on it.
1665                 var res = inp && parseFloat(inp.replace(',', '.'));
1666                 // apply sign while we're at it
1667                 return (isNaN(res) ? 0 : res) * sign;
1668             };
1669             duration = {
1670                 y: parseIso(match[2]),
1671                 M: parseIso(match[3]),
1672                 d: parseIso(match[4]),
1673                 h: parseIso(match[5]),
1674                 m: parseIso(match[6]),
1675                 s: parseIso(match[7]),
1676                 w: parseIso(match[8])
1677             };
1678         }
1680         ret = new Duration(duration);
1682         if (moment.isDuration(input) && input.hasOwnProperty('_lang')) {
1683             ret._lang = input._lang;
1684         }
1686         return ret;
1687     };
1689     // version number
1690     moment.version = VERSION;
1692     // default format
1693     moment.defaultFormat = isoFormat;
1695     // This function will be called whenever a moment is mutated.
1696     // It is intended to keep the offset in sync with the timezone.
1697     moment.updateOffset = function () {};
1699     // This function will load languages and then set the global language.  If
1700     // no arguments are passed in, it will simply return the current global
1701     // language key.
1702     moment.lang = function (key, values) {
1703         var r;
1704         if (!key) {
1705             return moment.fn._lang._abbr;
1706         }
1707         if (values) {
1708             loadLang(normalizeLanguage(key), values);
1709         } else if (values === null) {
1710             unloadLang(key);
1711             key = 'en';
1712         } else if (!languages[key]) {
1713             getLangDefinition(key);
1714         }
1715         r = moment.duration.fn._lang = moment.fn._lang = getLangDefinition(key);
1716         return r._abbr;
1717     };
1719     // returns language data
1720     moment.langData = function (key) {
1721         if (key && key._lang && key._lang._abbr) {
1722             key = key._lang._abbr;
1723         }
1724         return getLangDefinition(key);
1725     };
1727     // compare moment object
1728     moment.isMoment = function (obj) {
1729         return obj instanceof Moment ||
1730             (obj != null &&  obj.hasOwnProperty('_isAMomentObject'));
1731     };
1733     // for typechecking Duration objects
1734     moment.isDuration = function (obj) {
1735         return obj instanceof Duration;
1736     };
1738     for (i = lists.length - 1; i >= 0; --i) {
1739         makeList(lists[i]);
1740     }
1742     moment.normalizeUnits = function (units) {
1743         return normalizeUnits(units);
1744     };
1746     moment.invalid = function (flags) {
1747         var m = moment.utc(NaN);
1748         if (flags != null) {
1749             extend(m._pf, flags);
1750         }
1751         else {
1752             m._pf.userInvalidated = true;
1753         }
1755         return m;
1756     };
1758     moment.parseZone = function (input) {
1759         return moment(input).parseZone();
1760     };
1762     /************************************
1763         Moment Prototype
1764     ************************************/
1767     extend(moment.fn = Moment.prototype, {
1769         clone : function () {
1770             return moment(this);
1771         },
1773         valueOf : function () {
1774             return +this._d + ((this._offset || 0) * 60000);
1775         },
1777         unix : function () {
1778             return Math.floor(+this / 1000);
1779         },
1781         toString : function () {
1782             return this.clone().lang('en').format("ddd MMM DD YYYY HH:mm:ss [GMT]ZZ");
1783         },
1785         toDate : function () {
1786             return this._offset ? new Date(+this) : this._d;
1787         },
1789         toISOString : function () {
1790             var m = moment(this).utc();
1791             if (0 < m.year() && m.year() <= 9999) {
1792                 return formatMoment(m, 'YYYY-MM-DD[T]HH:mm:ss.SSS[Z]');
1793             } else {
1794                 return formatMoment(m, 'YYYYYY-MM-DD[T]HH:mm:ss.SSS[Z]');
1795             }
1796         },
1798         toArray : function () {
1799             var m = this;
1800             return [
1801                 m.year(),
1802                 m.month(),
1803                 m.date(),
1804                 m.hours(),
1805                 m.minutes(),
1806                 m.seconds(),
1807                 m.milliseconds()
1808             ];
1809         },
1811         isValid : function () {
1812             return isValid(this);
1813         },
1815         isDSTShifted : function () {
1817             if (this._a) {
1818                 return this.isValid() && compareArrays(this._a, (this._isUTC ? moment.utc(this._a) : moment(this._a)).toArray()) > 0;
1819             }
1821             return false;
1822         },
1824         parsingFlags : function () {
1825             return extend({}, this._pf);
1826         },
1828         invalidAt: function () {
1829             return this._pf.overflow;
1830         },
1832         utc : function () {
1833             return this.zone(0);
1834         },
1836         local : function () {
1837             this.zone(0);
1838             this._isUTC = false;
1839             return this;
1840         },
1842         format : function (inputString) {
1843             var output = formatMoment(this, inputString || moment.defaultFormat);
1844             return this.lang().postformat(output);
1845         },
1847         add : function (input, val) {
1848             var dur;
1849             // switch args to support add('s', 1) and add(1, 's')
1850             if (typeof input === 'string') {
1851                 dur = moment.duration(+val, input);
1852             } else {
1853                 dur = moment.duration(input, val);
1854             }
1855             addOrSubtractDurationFromMoment(this, dur, 1);
1856             return this;
1857         },
1859         subtract : function (input, val) {
1860             var dur;
1861             // switch args to support subtract('s', 1) and subtract(1, 's')
1862             if (typeof input === 'string') {
1863                 dur = moment.duration(+val, input);
1864             } else {
1865                 dur = moment.duration(input, val);
1866             }
1867             addOrSubtractDurationFromMoment(this, dur, -1);
1868             return this;
1869         },
1871         diff : function (input, units, asFloat) {
1872             var that = makeAs(input, this),
1873                 zoneDiff = (this.zone() - that.zone()) * 6e4,
1874                 diff, output;
1876             units = normalizeUnits(units);
1878             if (units === 'year' || units === 'month') {
1879                 // average number of days in the months in the given dates
1880                 diff = (this.daysInMonth() + that.daysInMonth()) * 432e5; // 24 * 60 * 60 * 1000 / 2
1881                 // difference in months
1882                 output = ((this.year() - that.year()) * 12) + (this.month() - that.month());
1883                 // adjust by taking difference in days, average number of days
1884                 // and dst in the given months.
1885                 output += ((this - moment(this).startOf('month')) -
1886                         (that - moment(that).startOf('month'))) / diff;
1887                 // same as above but with zones, to negate all dst
1888                 output -= ((this.zone() - moment(this).startOf('month').zone()) -
1889                         (that.zone() - moment(that).startOf('month').zone())) * 6e4 / diff;
1890                 if (units === 'year') {
1891                     output = output / 12;
1892                 }
1893             } else {
1894                 diff = (this - that);
1895                 output = units === 'second' ? diff / 1e3 : // 1000
1896                     units === 'minute' ? diff / 6e4 : // 1000 * 60
1897                     units === 'hour' ? diff / 36e5 : // 1000 * 60 * 60
1898                     units === 'day' ? (diff - zoneDiff) / 864e5 : // 1000 * 60 * 60 * 24, negate dst
1899                     units === 'week' ? (diff - zoneDiff) / 6048e5 : // 1000 * 60 * 60 * 24 * 7, negate dst
1900                     diff;
1901             }
1902             return asFloat ? output : absRound(output);
1903         },
1905         from : function (time, withoutSuffix) {
1906             return moment.duration(this.diff(time)).lang(this.lang()._abbr).humanize(!withoutSuffix);
1907         },
1909         fromNow : function (withoutSuffix) {
1910             return this.from(moment(), withoutSuffix);
1911         },
1913         calendar : function () {
1914             // We want to compare the start of today, vs this.
1915             // Getting start-of-today depends on whether we're zone'd or not.
1916             var sod = makeAs(moment(), this).startOf('day'),
1917                 diff = this.diff(sod, 'days', true),
1918                 format = diff < -6 ? 'sameElse' :
1919                     diff < -1 ? 'lastWeek' :
1920                     diff < 0 ? 'lastDay' :
1921                     diff < 1 ? 'sameDay' :
1922                     diff < 2 ? 'nextDay' :
1923                     diff < 7 ? 'nextWeek' : 'sameElse';
1924             return this.format(this.lang().calendar(format, this));
1925         },
1927         isLeapYear : function () {
1928             return isLeapYear(this.year());
1929         },
1931         isDST : function () {
1932             return (this.zone() < this.clone().month(0).zone() ||
1933                 this.zone() < this.clone().month(5).zone());
1934         },
1936         day : function (input) {
1937             var day = this._isUTC ? this._d.getUTCDay() : this._d.getDay();
1938             if (input != null) {
1939                 input = parseWeekday(input, this.lang());
1940                 return this.add({ d : input - day });
1941             } else {
1942                 return day;
1943             }
1944         },
1946         month : function (input) {
1947             var utc = this._isUTC ? 'UTC' : '',
1948                 dayOfMonth;
1950             if (input != null) {
1951                 if (typeof input === 'string') {
1952                     input = this.lang().monthsParse(input);
1953                     if (typeof input !== 'number') {
1954                         return this;
1955                     }
1956                 }
1958                 dayOfMonth = this.date();
1959                 this.date(1);
1960                 this._d['set' + utc + 'Month'](input);
1961                 this.date(Math.min(dayOfMonth, this.daysInMonth()));
1963                 moment.updateOffset(this);
1964                 return this;
1965             } else {
1966                 return this._d['get' + utc + 'Month']();
1967             }
1968         },
1970         startOf: function (units) {
1971             units = normalizeUnits(units);
1972             // the following switch intentionally omits break keywords
1973             // to utilize falling through the cases.
1974             switch (units) {
1975             case 'year':
1976                 this.month(0);
1977                 /* falls through */
1978             case 'month':
1979                 this.date(1);
1980                 /* falls through */
1981             case 'week':
1982             case 'isoWeek':
1983             case 'day':
1984                 this.hours(0);
1985                 /* falls through */
1986             case 'hour':
1987                 this.minutes(0);
1988                 /* falls through */
1989             case 'minute':
1990                 this.seconds(0);
1991                 /* falls through */
1992             case 'second':
1993                 this.milliseconds(0);
1994                 /* falls through */
1995             }
1997             // weeks are a special case
1998             if (units === 'week') {
1999                 this.weekday(0);
2000             } else if (units === 'isoWeek') {
2001                 this.isoWeekday(1);
2002             }
2004             return this;
2005         },
2007         endOf: function (units) {
2008             units = normalizeUnits(units);
2009             return this.startOf(units).add((units === 'isoWeek' ? 'week' : units), 1).subtract('ms', 1);
2010         },
2012         isAfter: function (input, units) {
2013             units = typeof units !== 'undefined' ? units : 'millisecond';
2014             return +this.clone().startOf(units) > +moment(input).startOf(units);
2015         },
2017         isBefore: function (input, units) {
2018             units = typeof units !== 'undefined' ? units : 'millisecond';
2019             return +this.clone().startOf(units) < +moment(input).startOf(units);
2020         },
2022         isSame: function (input, units) {
2023             units = units || 'ms';
2024             return +this.clone().startOf(units) === +makeAs(input, this).startOf(units);
2025         },
2027         min: function (other) {
2028             other = moment.apply(null, arguments);
2029             return other < this ? this : other;
2030         },
2032         max: function (other) {
2033             other = moment.apply(null, arguments);
2034             return other > this ? this : other;
2035         },
2037         zone : function (input) {
2038             var offset = this._offset || 0;
2039             if (input != null) {
2040                 if (typeof input === "string") {
2041                     input = timezoneMinutesFromString(input);
2042                 }
2043                 if (Math.abs(input) < 16) {
2044                     input = input * 60;
2045                 }
2046                 this._offset = input;
2047                 this._isUTC = true;
2048                 if (offset !== input) {
2049                     addOrSubtractDurationFromMoment(this, moment.duration(offset - input, 'm'), 1, true);
2050                 }
2051             } else {
2052                 return this._isUTC ? offset : this._d.getTimezoneOffset();
2053             }
2054             return this;
2055         },
2057         zoneAbbr : function () {
2058             return this._isUTC ? "UTC" : "";
2059         },
2061         zoneName : function () {
2062             return this._isUTC ? "Coordinated Universal Time" : "";
2063         },
2065         parseZone : function () {
2066             if (this._tzm) {
2067                 this.zone(this._tzm);
2068             } else if (typeof this._i === 'string') {
2069                 this.zone(this._i);
2070             }
2071             return this;
2072         },
2074         hasAlignedHourOffset : function (input) {
2075             if (!input) {
2076                 input = 0;
2077             }
2078             else {
2079                 input = moment(input).zone();
2080             }
2082             return (this.zone() - input) % 60 === 0;
2083         },
2085         daysInMonth : function () {
2086             return daysInMonth(this.year(), this.month());
2087         },
2089         dayOfYear : function (input) {
2090             var dayOfYear = round((moment(this).startOf('day') - moment(this).startOf('year')) / 864e5) + 1;
2091             return input == null ? dayOfYear : this.add("d", (input - dayOfYear));
2092         },
2094         quarter : function () {
2095             return Math.ceil((this.month() + 1.0) / 3.0);
2096         },
2098         weekYear : function (input) {
2099             var year = weekOfYear(this, this.lang()._week.dow, this.lang()._week.doy).year;
2100             return input == null ? year : this.add("y", (input - year));
2101         },
2103         isoWeekYear : function (input) {
2104             var year = weekOfYear(this, 1, 4).year;
2105             return input == null ? year : this.add("y", (input - year));
2106         },
2108         week : function (input) {
2109             var week = this.lang().week(this);
2110             return input == null ? week : this.add("d", (input - week) * 7);
2111         },
2113         isoWeek : function (input) {
2114             var week = weekOfYear(this, 1, 4).week;
2115             return input == null ? week : this.add("d", (input - week) * 7);
2116         },
2118         weekday : function (input) {
2119             var weekday = (this.day() + 7 - this.lang()._week.dow) % 7;
2120             return input == null ? weekday : this.add("d", input - weekday);
2121         },
2123         isoWeekday : function (input) {
2124             // behaves the same as moment#day except
2125             // as a getter, returns 7 instead of 0 (1-7 range instead of 0-6)
2126             // as a setter, sunday should belong to the previous week.
2127             return input == null ? this.day() || 7 : this.day(this.day() % 7 ? input : input - 7);
2128         },
2130         get : function (units) {
2131             units = normalizeUnits(units);
2132             return this[units]();
2133         },
2135         set : function (units, value) {
2136             units = normalizeUnits(units);
2137             if (typeof this[units] === 'function') {
2138                 this[units](value);
2139             }
2140             return this;
2141         },
2143         // If passed a language key, it will set the language for this
2144         // instance.  Otherwise, it will return the language configuration
2145         // variables for this instance.
2146         lang : function (key) {
2147             if (key === undefined) {
2148                 return this._lang;
2149             } else {
2150                 this._lang = getLangDefinition(key);
2151                 return this;
2152             }
2153         }
2154     });
2156     // helper for adding shortcuts
2157     function makeGetterAndSetter(name, key) {
2158         moment.fn[name] = moment.fn[name + 's'] = function (input) {
2159             var utc = this._isUTC ? 'UTC' : '';
2160             if (input != null) {
2161                 this._d['set' + utc + key](input);
2162                 moment.updateOffset(this);
2163                 return this;
2164             } else {
2165                 return this._d['get' + utc + key]();
2166             }
2167         };
2168     }
2170     // loop through and add shortcuts (Month, Date, Hours, Minutes, Seconds, Milliseconds)
2171     for (i = 0; i < proxyGettersAndSetters.length; i ++) {
2172         makeGetterAndSetter(proxyGettersAndSetters[i].toLowerCase().replace(/s$/, ''), proxyGettersAndSetters[i]);
2173     }
2175     // add shortcut for year (uses different syntax than the getter/setter 'year' == 'FullYear')
2176     makeGetterAndSetter('year', 'FullYear');
2178     // add plural methods
2179     moment.fn.days = moment.fn.day;
2180     moment.fn.months = moment.fn.month;
2181     moment.fn.weeks = moment.fn.week;
2182     moment.fn.isoWeeks = moment.fn.isoWeek;
2184     // add aliased format methods
2185     moment.fn.toJSON = moment.fn.toISOString;
2187     /************************************
2188         Duration Prototype
2189     ************************************/
2192     extend(moment.duration.fn = Duration.prototype, {
2194         _bubble : function () {
2195             var milliseconds = this._milliseconds,
2196                 days = this._days,
2197                 months = this._months,
2198                 data = this._data,
2199                 seconds, minutes, hours, years;
2201             // The following code bubbles up values, see the tests for
2202             // examples of what that means.
2203             data.milliseconds = milliseconds % 1000;
2205             seconds = absRound(milliseconds / 1000);
2206             data.seconds = seconds % 60;
2208             minutes = absRound(seconds / 60);
2209             data.minutes = minutes % 60;
2211             hours = absRound(minutes / 60);
2212             data.hours = hours % 24;
2214             days += absRound(hours / 24);
2215             data.days = days % 30;
2217             months += absRound(days / 30);
2218             data.months = months % 12;
2220             years = absRound(months / 12);
2221             data.years = years;
2222         },
2224         weeks : function () {
2225             return absRound(this.days() / 7);
2226         },
2228         valueOf : function () {
2229             return this._milliseconds +
2230               this._days * 864e5 +
2231               (this._months % 12) * 2592e6 +
2232               toInt(this._months / 12) * 31536e6;
2233         },
2235         humanize : function (withSuffix) {
2236             var difference = +this,
2237                 output = relativeTime(difference, !withSuffix, this.lang());
2239             if (withSuffix) {
2240                 output = this.lang().pastFuture(difference, output);
2241             }
2243             return this.lang().postformat(output);
2244         },
2246         add : function (input, val) {
2247             // supports only 2.0-style add(1, 's') or add(moment)
2248             var dur = moment.duration(input, val);
2250             this._milliseconds += dur._milliseconds;
2251             this._days += dur._days;
2252             this._months += dur._months;
2254             this._bubble();
2256             return this;
2257         },
2259         subtract : function (input, val) {
2260             var dur = moment.duration(input, val);
2262             this._milliseconds -= dur._milliseconds;
2263             this._days -= dur._days;
2264             this._months -= dur._months;
2266             this._bubble();
2268             return this;
2269         },
2271         get : function (units) {
2272             units = normalizeUnits(units);
2273             return this[units.toLowerCase() + 's']();
2274         },
2276         as : function (units) {
2277             units = normalizeUnits(units);
2278             return this['as' + units.charAt(0).toUpperCase() + units.slice(1) + 's']();
2279         },
2281         lang : moment.fn.lang,
2283         toIsoString : function () {
2284             // inspired by https://github.com/dordille/moment-isoduration/blob/master/moment.isoduration.js
2285             var years = Math.abs(this.years()),
2286                 months = Math.abs(this.months()),
2287                 days = Math.abs(this.days()),
2288                 hours = Math.abs(this.hours()),
2289                 minutes = Math.abs(this.minutes()),
2290                 seconds = Math.abs(this.seconds() + this.milliseconds() / 1000);
2292             if (!this.asSeconds()) {
2293                 // this is the same as C#'s (Noda) and python (isodate)...
2294                 // but not other JS (goog.date)
2295                 return 'P0D';
2296             }
2298             return (this.asSeconds() < 0 ? '-' : '') +
2299                 'P' +
2300                 (years ? years + 'Y' : '') +
2301                 (months ? months + 'M' : '') +
2302                 (days ? days + 'D' : '') +
2303                 ((hours || minutes || seconds) ? 'T' : '') +
2304                 (hours ? hours + 'H' : '') +
2305                 (minutes ? minutes + 'M' : '') +
2306                 (seconds ? seconds + 'S' : '');
2307         }
2308     });
2310     function makeDurationGetter(name) {
2311         moment.duration.fn[name] = function () {
2312             return this._data[name];
2313         };
2314     }
2316     function makeDurationAsGetter(name, factor) {
2317         moment.duration.fn['as' + name] = function () {
2318             return +this / factor;
2319         };
2320     }
2322     for (i in unitMillisecondFactors) {
2323         if (unitMillisecondFactors.hasOwnProperty(i)) {
2324             makeDurationAsGetter(i, unitMillisecondFactors[i]);
2325             makeDurationGetter(i.toLowerCase());
2326         }
2327     }
2329     makeDurationAsGetter('Weeks', 6048e5);
2330     moment.duration.fn.asMonths = function () {
2331         return (+this - this.years() * 31536e6) / 2592e6 + this.years() * 12;
2332     };
2335     /************************************
2336         Default Lang
2337     ************************************/
2340     // Set default language, other languages will inherit from English.
2341     moment.lang('en', {
2342         ordinal : function (number) {
2343             var b = number % 10,
2344                 output = (toInt(number % 100 / 10) === 1) ? 'th' :
2345                 (b === 1) ? 'st' :
2346                 (b === 2) ? 'nd' :
2347                 (b === 3) ? 'rd' : 'th';
2348             return number + output;
2349         }
2350     });
2352     /* EMBED_LANGUAGES */
2354     /************************************
2355         Exposing Moment
2356     ************************************/
2358     function makeGlobal(deprecate) {
2359         var warned = false, local_moment = moment;
2360         /*global ender:false */
2361         if (typeof ender !== 'undefined') {
2362             return;
2363         }
2364         // here, `this` means `window` in the browser, or `global` on the server
2365         // add `moment` as a global object via a string identifier,
2366         // for Closure Compiler "advanced" mode
2367         if (deprecate) {
2368             global.moment = function () {
2369                 if (!warned && console && console.warn) {
2370                     warned = true;
2371                     console.warn(
2372                             "Accessing Moment through the global scope is " +
2373                             "deprecated, and will be removed in an upcoming " +
2374                             "release.");
2375                 }
2376                 return local_moment.apply(null, arguments);
2377             };
2378             extend(global.moment, local_moment);
2379         } else {
2380             global['moment'] = moment;
2381         }
2382     }
2384     // CommonJS module is defined
2385     if (hasModule) {
2386         module.exports = moment;
2387         makeGlobal(true);
2388     } else if (typeof define === "function" && define.amd) {
2389         define("moment", function (require, exports, module) {
2390             if (module.config && module.config() && module.config().noGlobal !== true) {
2391                 // If user provided noGlobal, he is aware of global
2392                 makeGlobal(module.config().noGlobal === undefined);
2393             }
2395             return moment;
2396         });
2397     } else {
2398         makeGlobal();
2399     }
2400 }).call(this);