flapjack 0.8.11 → 0.8.12

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