1//! moment.js 2//! version : 2.20.1 3//! authors : Tim Wood, Iskren Chernev, Moment.js contributors 4//! license : MIT 5//! momentjs.com 6 7;(function (global, factory) { 8 typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() : 9 typeof define === 'function' && define.amd ? define(factory) : 10 global.moment = factory() 11}(this, (function () { 'use strict'; 12 13var hookCallback; 14 15function hooks () { 16 return hookCallback.apply(null, arguments); 17} 18 19// This is done to register the method called with moment() 20// without creating circular dependencies. 21function setHookCallback (callback) { 22 hookCallback = callback; 23} 24 25function isArray(input) { 26 return input instanceof Array || Object.prototype.toString.call(input) === '[object Array]'; 27} 28 29function isObject(input) { 30 // IE8 will treat undefined and null as object if it wasn't for 31 // input != null 32 return input != null && Object.prototype.toString.call(input) === '[object Object]'; 33} 34 35function isObjectEmpty(obj) { 36 if (Object.getOwnPropertyNames) { 37 return (Object.getOwnPropertyNames(obj).length === 0); 38 } else { 39 var k; 40 for (k in obj) { 41 if (obj.hasOwnProperty(k)) { 42 return false; 43 } 44 } 45 return true; 46 } 47} 48 49function isUndefined(input) { 50 return input === void 0; 51} 52 53function isNumber(input) { 54 return typeof input === 'number' || Object.prototype.toString.call(input) === '[object Number]'; 55} 56 57function isDate(input) { 58 return input instanceof Date || Object.prototype.toString.call(input) === '[object Date]'; 59} 60 61function map(arr, fn) { 62 var res = [], i; 63 for (i = 0; i < arr.length; ++i) { 64 res.push(fn(arr[i], i)); 65 } 66 return res; 67} 68 69function hasOwnProp(a, b) { 70 return Object.prototype.hasOwnProperty.call(a, b); 71} 72 73function extend(a, b) { 74 for (var i in b) { 75 if (hasOwnProp(b, i)) { 76 a[i] = b[i]; 77 } 78 } 79 80 if (hasOwnProp(b, 'toString')) { 81 a.toString = b.toString; 82 } 83 84 if (hasOwnProp(b, 'valueOf')) { 85 a.valueOf = b.valueOf; 86 } 87 88 return a; 89} 90 91function createUTC (input, format, locale, strict) { 92 return createLocalOrUTC(input, format, locale, strict, true).utc(); 93} 94 95function defaultParsingFlags() { 96 // We need to deep clone this object. 97 return { 98 empty : false, 99 unusedTokens : [], 100 unusedInput : [], 101 overflow : -2, 102 charsLeftOver : 0, 103 nullInput : false, 104 invalidMonth : null, 105 invalidFormat : false, 106 userInvalidated : false, 107 iso : false, 108 parsedDateParts : [], 109 meridiem : null, 110 rfc2822 : false, 111 weekdayMismatch : false 112 }; 113} 114 115function getParsingFlags(m) { 116 if (m._pf == null) { 117 m._pf = defaultParsingFlags(); 118 } 119 return m._pf; 120} 121 122var some; 123if (Array.prototype.some) { 124 some = Array.prototype.some; 125} else { 126 some = function (fun) { 127 var t = Object(this); 128 var len = t.length >>> 0; 129 130 for (var i = 0; i < len; i++) { 131 if (i in t && fun.call(this, t[i], i, t)) { 132 return true; 133 } 134 } 135 136 return false; 137 }; 138} 139 140function isValid(m) { 141 if (m._isValid == null) { 142 var flags = getParsingFlags(m); 143 var parsedParts = some.call(flags.parsedDateParts, function (i) { 144 return i != null; 145 }); 146 var isNowValid = !isNaN(m._d.getTime()) && 147 flags.overflow < 0 && 148 !flags.empty && 149 !flags.invalidMonth && 150 !flags.invalidWeekday && 151 !flags.weekdayMismatch && 152 !flags.nullInput && 153 !flags.invalidFormat && 154 !flags.userInvalidated && 155 (!flags.meridiem || (flags.meridiem && parsedParts)); 156 157 if (m._strict) { 158 isNowValid = isNowValid && 159 flags.charsLeftOver === 0 && 160 flags.unusedTokens.length === 0 && 161 flags.bigHour === undefined; 162 } 163 164 if (Object.isFrozen == null || !Object.isFrozen(m)) { 165 m._isValid = isNowValid; 166 } 167 else { 168 return isNowValid; 169 } 170 } 171 return m._isValid; 172} 173 174function createInvalid (flags) { 175 var m = createUTC(NaN); 176 if (flags != null) { 177 extend(getParsingFlags(m), flags); 178 } 179 else { 180 getParsingFlags(m).userInvalidated = true; 181 } 182 183 return m; 184} 185 186// Plugins that add properties should also add the key here (null value), 187// so we can properly clone ourselves. 188var momentProperties = hooks.momentProperties = []; 189 190function copyConfig(to, from) { 191 var i, prop, val; 192 193 if (!isUndefined(from._isAMomentObject)) { 194 to._isAMomentObject = from._isAMomentObject; 195 } 196 if (!isUndefined(from._i)) { 197 to._i = from._i; 198 } 199 if (!isUndefined(from._f)) { 200 to._f = from._f; 201 } 202 if (!isUndefined(from._l)) { 203 to._l = from._l; 204 } 205 if (!isUndefined(from._strict)) { 206 to._strict = from._strict; 207 } 208 if (!isUndefined(from._tzm)) { 209 to._tzm = from._tzm; 210 } 211 if (!isUndefined(from._isUTC)) { 212 to._isUTC = from._isUTC; 213 } 214 if (!isUndefined(from._offset)) { 215 to._offset = from._offset; 216 } 217 if (!isUndefined(from._pf)) { 218 to._pf = getParsingFlags(from); 219 } 220 if (!isUndefined(from._locale)) { 221 to._locale = from._locale; 222 } 223 224 if (momentProperties.length > 0) { 225 for (i = 0; i < momentProperties.length; i++) { 226 prop = momentProperties[i]; 227 val = from[prop]; 228 if (!isUndefined(val)) { 229 to[prop] = val; 230 } 231 } 232 } 233 234 return to; 235} 236 237var updateInProgress = false; 238 239// Moment prototype object 240function Moment(config) { 241 copyConfig(this, config); 242 this._d = new Date(config._d != null ? config._d.getTime() : NaN); 243 if (!this.isValid()) { 244 this._d = new Date(NaN); 245 } 246 // Prevent infinite loop in case updateOffset creates new moment 247 // objects. 248 if (updateInProgress === false) { 249 updateInProgress = true; 250 hooks.updateOffset(this); 251 updateInProgress = false; 252 } 253} 254 255function isMoment (obj) { 256 return obj instanceof Moment || (obj != null && obj._isAMomentObject != null); 257} 258 259function absFloor (number) { 260 if (number < 0) { 261 // -0 -> 0 262 return Math.ceil(number) || 0; 263 } else { 264 return Math.floor(number); 265 } 266} 267 268function toInt(argumentForCoercion) { 269 var coercedNumber = +argumentForCoercion, 270 value = 0; 271 272 if (coercedNumber !== 0 && isFinite(coercedNumber)) { 273 value = absFloor(coercedNumber); 274 } 275 276 return value; 277} 278 279// compare two arrays, return the number of differences 280function compareArrays(array1, array2, dontConvert) { 281 var len = Math.min(array1.length, array2.length), 282 lengthDiff = Math.abs(array1.length - array2.length), 283 diffs = 0, 284 i; 285 for (i = 0; i < len; i++) { 286 if ((dontConvert && array1[i] !== array2[i]) || 287 (!dontConvert && toInt(array1[i]) !== toInt(array2[i]))) { 288 diffs++; 289 } 290 } 291 return diffs + lengthDiff; 292} 293 294function warn(msg) { 295 if (hooks.suppressDeprecationWarnings === false && 296 (typeof console !== 'undefined') && console.warn) { 297 console.warn('Deprecation warning: ' + msg); 298 } 299} 300 301function deprecate(msg, fn) { 302 var firstTime = true; 303 304 return extend(function () { 305 if (hooks.deprecationHandler != null) { 306 hooks.deprecationHandler(null, msg); 307 } 308 if (firstTime) { 309 var args = []; 310 var arg; 311 for (var i = 0; i < arguments.length; i++) { 312 arg = ''; 313 if (typeof arguments[i] === 'object') { 314 arg += '\n[' + i + '] '; 315 for (var key in arguments[0]) { 316 arg += key + ': ' + arguments[0][key] + ', '; 317 } 318 arg = arg.slice(0, -2); // Remove trailing comma and space 319 } else { 320 arg = arguments[i]; 321 } 322 args.push(arg); 323 } 324 warn(msg + '\nArguments: ' + Array.prototype.slice.call(args).join('') + '\n' + (new Error()).stack); 325 firstTime = false; 326 } 327 return fn.apply(this, arguments); 328 }, fn); 329} 330 331var deprecations = {}; 332 333function deprecateSimple(name, msg) { 334 if (hooks.deprecationHandler != null) { 335 hooks.deprecationHandler(name, msg); 336 } 337 if (!deprecations[name]) { 338 warn(msg); 339 deprecations[name] = true; 340 } 341} 342 343hooks.suppressDeprecationWarnings = false; 344hooks.deprecationHandler = null; 345 346function isFunction(input) { 347 return input instanceof Function || Object.prototype.toString.call(input) === '[object Function]'; 348} 349 350function set (config) { 351 var prop, i; 352 for (i in config) { 353 prop = config[i]; 354 if (isFunction(prop)) { 355 this[i] = prop; 356 } else { 357 this['_' + i] = prop; 358 } 359 } 360 this._config = config; 361 // Lenient ordinal parsing accepts just a number in addition to 362 // number + (possibly) stuff coming from _dayOfMonthOrdinalParse. 363 // TODO: Remove "ordinalParse" fallback in next major release. 364 this._dayOfMonthOrdinalParseLenient = new RegExp( 365 (this._dayOfMonthOrdinalParse.source || this._ordinalParse.source) + 366 '|' + (/\d{1,2}/).source); 367} 368 369function mergeConfigs(parentConfig, childConfig) { 370 var res = extend({}, parentConfig), prop; 371 for (prop in childConfig) { 372 if (hasOwnProp(childConfig, prop)) { 373 if (isObject(parentConfig[prop]) && isObject(childConfig[prop])) { 374 res[prop] = {}; 375 extend(res[prop], parentConfig[prop]); 376 extend(res[prop], childConfig[prop]); 377 } else if (childConfig[prop] != null) { 378 res[prop] = childConfig[prop]; 379 } else { 380 delete res[prop]; 381 } 382 } 383 } 384 for (prop in parentConfig) { 385 if (hasOwnProp(parentConfig, prop) && 386 !hasOwnProp(childConfig, prop) && 387 isObject(parentConfig[prop])) { 388 // make sure changes to properties don't modify parent config 389 res[prop] = extend({}, res[prop]); 390 } 391 } 392 return res; 393} 394 395function Locale(config) { 396 if (config != null) { 397 this.set(config); 398 } 399} 400 401var keys; 402 403if (Object.keys) { 404 keys = Object.keys; 405} else { 406 keys = function (obj) { 407 var i, res = []; 408 for (i in obj) { 409 if (hasOwnProp(obj, i)) { 410 res.push(i); 411 } 412 } 413 return res; 414 }; 415} 416 417var defaultCalendar = { 418 sameDay : '[Today at] LT', 419 nextDay : '[Tomorrow at] LT', 420 nextWeek : 'dddd [at] LT', 421 lastDay : '[Yesterday at] LT', 422 lastWeek : '[Last] dddd [at] LT', 423 sameElse : 'L' 424}; 425 426function calendar (key, mom, now) { 427 var output = this._calendar[key] || this._calendar['sameElse']; 428 return isFunction(output) ? output.call(mom, now) : output; 429} 430 431var defaultLongDateFormat = { 432 LTS : 'h:mm:ss A', 433 LT : 'h:mm A', 434 L : 'MM/DD/YYYY', 435 LL : 'MMMM D, YYYY', 436 LLL : 'MMMM D, YYYY h:mm A', 437 LLLL : 'dddd, MMMM D, YYYY h:mm A' 438}; 439 440function longDateFormat (key) { 441 var format = this._longDateFormat[key], 442 formatUpper = this._longDateFormat[key.toUpperCase()]; 443 444 if (format || !formatUpper) { 445 return format; 446 } 447 448 this._longDateFormat[key] = formatUpper.replace(/MMMM|MM|DD|dddd/g, function (val) { 449 return val.slice(1); 450 }); 451 452 return this._longDateFormat[key]; 453} 454 455var defaultInvalidDate = 'Invalid date'; 456 457function invalidDate () { 458 return this._invalidDate; 459} 460 461var defaultOrdinal = '%d'; 462var defaultDayOfMonthOrdinalParse = /\d{1,2}/; 463 464function ordinal (number) { 465 return this._ordinal.replace('%d', number); 466} 467 468var defaultRelativeTime = { 469 future : 'in %s', 470 past : '%s ago', 471 s : 'a few seconds', 472 ss : '%d seconds', 473 m : 'a minute', 474 mm : '%d minutes', 475 h : 'an hour', 476 hh : '%d hours', 477 d : 'a day', 478 dd : '%d days', 479 M : 'a month', 480 MM : '%d months', 481 y : 'a year', 482 yy : '%d years' 483}; 484 485function relativeTime (number, withoutSuffix, string, isFuture) { 486 var output = this._relativeTime[string]; 487 return (isFunction(output)) ? 488 output(number, withoutSuffix, string, isFuture) : 489 output.replace(/%d/i, number); 490} 491 492function pastFuture (diff, output) { 493 var format = this._relativeTime[diff > 0 ? 'future' : 'past']; 494 return isFunction(format) ? format(output) : format.replace(/%s/i, output); 495} 496 497var aliases = {}; 498 499function addUnitAlias (unit, shorthand) { 500 var lowerCase = unit.toLowerCase(); 501 aliases[lowerCase] = aliases[lowerCase + 's'] = aliases[shorthand] = unit; 502} 503 504function normalizeUnits(units) { 505 return typeof units === 'string' ? aliases[units] || aliases[units.toLowerCase()] : undefined; 506} 507 508function normalizeObjectUnits(inputObject) { 509 var normalizedInput = {}, 510 normalizedProp, 511 prop; 512 513 for (prop in inputObject) { 514 if (hasOwnProp(inputObject, prop)) { 515 normalizedProp = normalizeUnits(prop); 516 if (normalizedProp) { 517 normalizedInput[normalizedProp] = inputObject[prop]; 518 } 519 } 520 } 521 522 return normalizedInput; 523} 524 525var priorities = {}; 526 527function addUnitPriority(unit, priority) { 528 priorities[unit] = priority; 529} 530 531function getPrioritizedUnits(unitsObj) { 532 var units = []; 533 for (var u in unitsObj) { 534 units.push({unit: u, priority: priorities[u]}); 535 } 536 units.sort(function (a, b) { 537 return a.priority - b.priority; 538 }); 539 return units; 540} 541 542function zeroFill(number, targetLength, forceSign) { 543 var absNumber = '' + Math.abs(number), 544 zerosToFill = targetLength - absNumber.length, 545 sign = number >= 0; 546 return (sign ? (forceSign ? '+' : '') : '-') + 547 Math.pow(10, Math.max(0, zerosToFill)).toString().substr(1) + absNumber; 548} 549 550var formattingTokens = /(\[[^\[]*\])|(\\)?([Hh]mm(ss)?|Mo|MM?M?M?|Do|DDDo|DD?D?D?|ddd?d?|do?|w[o|w]?|W[o|W]?|Qo?|YYYYYY|YYYYY|YYYY|YY|gg(ggg?)?|GG(GGG?)?|e|E|a|A|hh?|HH?|kk?|mm?|ss?|S{1,9}|x|X|zz?|ZZ?|.)/g; 551 552var localFormattingTokens = /(\[[^\[]*\])|(\\)?(LTS|LT|LL?L?L?|l{1,4})/g; 553 554var formatFunctions = {}; 555 556var formatTokenFunctions = {}; 557 558// token: 'M' 559// padded: ['MM', 2] 560// ordinal: 'Mo' 561// callback: function () { this.month() + 1 } 562function addFormatToken (token, padded, ordinal, callback) { 563 var func = callback; 564 if (typeof callback === 'string') { 565 func = function () { 566 return this[callback](); 567 }; 568 } 569 if (token) { 570 formatTokenFunctions[token] = func; 571 } 572 if (padded) { 573 formatTokenFunctions[padded[0]] = function () { 574 return zeroFill(func.apply(this, arguments), padded[1], padded[2]); 575 }; 576 } 577 if (ordinal) { 578 formatTokenFunctions[ordinal] = function () { 579 return this.localeData().ordinal(func.apply(this, arguments), token); 580 }; 581 } 582} 583 584function removeFormattingTokens(input) { 585 if (input.match(/\[[\s\S]/)) { 586 return input.replace(/^\[|\]$/g, ''); 587 } 588 return input.replace(/\\/g, ''); 589} 590 591function makeFormatFunction(format) { 592 var array = format.match(formattingTokens), i, length; 593 594 for (i = 0, length = array.length; i < length; i++) { 595 if (formatTokenFunctions[array[i]]) { 596 array[i] = formatTokenFunctions[array[i]]; 597 } else { 598 array[i] = removeFormattingTokens(array[i]); 599 } 600 } 601 602 return function (mom) { 603 var output = '', i; 604 for (i = 0; i < length; i++) { 605 output += isFunction(array[i]) ? array[i].call(mom, format) : array[i]; 606 } 607 return output; 608 }; 609} 610 611// format date using native date object 612function formatMoment(m, format) { 613 if (!m.isValid()) { 614 return m.localeData().invalidDate(); 615 } 616 617 format = expandFormat(format, m.localeData()); 618 formatFunctions[format] = formatFunctions[format] || makeFormatFunction(format); 619 620 return formatFunctions[format](m); 621} 622 623function expandFormat(format, locale) { 624 var i = 5; 625 626 function replaceLongDateFormatTokens(input) { 627 return locale.longDateFormat(input) || input; 628 } 629 630 localFormattingTokens.lastIndex = 0; 631 while (i >= 0 && localFormattingTokens.test(format)) { 632 format = format.replace(localFormattingTokens, replaceLongDateFormatTokens); 633 localFormattingTokens.lastIndex = 0; 634 i -= 1; 635 } 636 637 return format; 638} 639 640var match1 = /\d/; // 0 - 9 641var match2 = /\d\d/; // 00 - 99 642var match3 = /\d{3}/; // 000 - 999 643var match4 = /\d{4}/; // 0000 - 9999 644var match6 = /[+-]?\d{6}/; // -999999 - 999999 645var match1to2 = /\d\d?/; // 0 - 99 646var match3to4 = /\d\d\d\d?/; // 999 - 9999 647var match5to6 = /\d\d\d\d\d\d?/; // 99999 - 999999 648var match1to3 = /\d{1,3}/; // 0 - 999 649var match1to4 = /\d{1,4}/; // 0 - 9999 650var match1to6 = /[+-]?\d{1,6}/; // -999999 - 999999 651 652var matchUnsigned = /\d+/; // 0 - inf 653var matchSigned = /[+-]?\d+/; // -inf - inf 654 655var matchOffset = /Z|[+-]\d\d:?\d\d/gi; // +00:00 -00:00 +0000 -0000 or Z 656var matchShortOffset = /Z|[+-]\d\d(?::?\d\d)?/gi; // +00 -00 +00:00 -00:00 +0000 -0000 or Z 657 658var matchTimestamp = /[+-]?\d+(\.\d{1,3})?/; // 123456789 123456789.123 659 660// any word (or two) characters or numbers including two/three word month in arabic. 661// includes scottish gaelic two word and hyphenated months 662var matchWord = /[0-9]{0,256}['a-z\u00A0-\u05FF\u0700-\uD7FF\uF900-\uFDCF\uFDF0-\uFF07\uFF10-\uFFEF]{1,256}|[\u0600-\u06FF\/]{1,256}(\s*?[\u0600-\u06FF]{1,256}){1,2}/i; 663 664 665var regexes = {}; 666 667function addRegexToken (token, regex, strictRegex) { 668 regexes[token] = isFunction(regex) ? regex : function (isStrict, localeData) { 669 return (isStrict && strictRegex) ? strictRegex : regex; 670 }; 671} 672 673function getParseRegexForToken (token, config) { 674 if (!hasOwnProp(regexes, token)) { 675 return new RegExp(unescapeFormat(token)); 676 } 677 678 return regexes[token](config._strict, config._locale); 679} 680 681// Code from http://stackoverflow.com/questions/3561493/is-there-a-regexp-escape-function-in-javascript 682function unescapeFormat(s) { 683 return regexEscape(s.replace('\\', '').replace(/\\(\[)|\\(\])|\[([^\]\[]*)\]|\\(.)/g, function (matched, p1, p2, p3, p4) { 684 return p1 || p2 || p3 || p4; 685 })); 686} 687 688function regexEscape(s) { 689 return s.replace(/[-\/\\^$*+?.()|[\]{}]/g, '\\$&'); 690} 691 692var tokens = {}; 693 694function addParseToken (token, callback) { 695 var i, func = callback; 696 if (typeof token === 'string') { 697 token = [token]; 698 } 699 if (isNumber(callback)) { 700 func = function (input, array) { 701 array[callback] = toInt(input); 702 }; 703 } 704 for (i = 0; i < token.length; i++) { 705 tokens[token[i]] = func; 706 } 707} 708 709function addWeekParseToken (token, callback) { 710 addParseToken(token, function (input, array, config, token) { 711 config._w = config._w || {}; 712 callback(input, config._w, config, token); 713 }); 714} 715 716function addTimeToArrayFromToken(token, input, config) { 717 if (input != null && hasOwnProp(tokens, token)) { 718 tokens[token](input, config._a, config, token); 719 } 720} 721 722var YEAR = 0; 723var MONTH = 1; 724var DATE = 2; 725var HOUR = 3; 726var MINUTE = 4; 727var SECOND = 5; 728var MILLISECOND = 6; 729var WEEK = 7; 730var WEEKDAY = 8; 731 732// FORMATTING 733 734addFormatToken('Y', 0, 0, function () { 735 var y = this.year(); 736 return y <= 9999 ? '' + y : '+' + y; 737}); 738 739addFormatToken(0, ['YY', 2], 0, function () { 740 return this.year() % 100; 741}); 742 743addFormatToken(0, ['YYYY', 4], 0, 'year'); 744addFormatToken(0, ['YYYYY', 5], 0, 'year'); 745addFormatToken(0, ['YYYYYY', 6, true], 0, 'year'); 746 747// ALIASES 748 749addUnitAlias('year', 'y'); 750 751// PRIORITIES 752 753addUnitPriority('year', 1); 754 755// PARSING 756 757addRegexToken('Y', matchSigned); 758addRegexToken('YY', match1to2, match2); 759addRegexToken('YYYY', match1to4, match4); 760addRegexToken('YYYYY', match1to6, match6); 761addRegexToken('YYYYYY', match1to6, match6); 762 763addParseToken(['YYYYY', 'YYYYYY'], YEAR); 764addParseToken('YYYY', function (input, array) { 765 array[YEAR] = input.length === 2 ? hooks.parseTwoDigitYear(input) : toInt(input); 766}); 767addParseToken('YY', function (input, array) { 768 array[YEAR] = hooks.parseTwoDigitYear(input); 769}); 770addParseToken('Y', function (input, array) { 771 array[YEAR] = parseInt(input, 10); 772}); 773 774// HELPERS 775 776function daysInYear(year) { 777 return isLeapYear(year) ? 366 : 365; 778} 779 780function isLeapYear(year) { 781 return (year % 4 === 0 && year % 100 !== 0) || year % 400 === 0; 782} 783 784// HOOKS 785 786hooks.parseTwoDigitYear = function (input) { 787 return toInt(input) + (toInt(input) > 68 ? 1900 : 2000); 788}; 789 790// MOMENTS 791 792var getSetYear = makeGetSet('FullYear', true); 793 794function getIsLeapYear () { 795 return isLeapYear(this.year()); 796} 797 798function makeGetSet (unit, keepTime) { 799 return function (value) { 800 if (value != null) { 801 set$1(this, unit, value); 802 hooks.updateOffset(this, keepTime); 803 return this; 804 } else { 805 return get(this, unit); 806 } 807 }; 808} 809 810function get (mom, unit) { 811 return mom.isValid() ? 812 mom._d['get' + (mom._isUTC ? 'UTC' : '') + unit]() : NaN; 813} 814 815function set$1 (mom, unit, value) { 816 if (mom.isValid() && !isNaN(value)) { 817 if (unit === 'FullYear' && isLeapYear(mom.year()) && mom.month() === 1 && mom.date() === 29) { 818 mom._d['set' + (mom._isUTC ? 'UTC' : '') + unit](value, mom.month(), daysInMonth(value, mom.month())); 819 } 820 else { 821 mom._d['set' + (mom._isUTC ? 'UTC' : '') + unit](value); 822 } 823 } 824} 825 826// MOMENTS 827 828function stringGet (units) { 829 units = normalizeUnits(units); 830 if (isFunction(this[units])) { 831 return this[units](); 832 } 833 return this; 834} 835 836 837function stringSet (units, value) { 838 if (typeof units === 'object') { 839 units = normalizeObjectUnits(units); 840 var prioritized = getPrioritizedUnits(units); 841 for (var i = 0; i < prioritized.length; i++) { 842 this[prioritized[i].unit](units[prioritized[i].unit]); 843 } 844 } else { 845 units = normalizeUnits(units); 846 if (isFunction(this[units])) { 847 return this[units](value); 848 } 849 } 850 return this; 851} 852 853function mod(n, x) { 854 return ((n % x) + x) % x; 855} 856 857var indexOf; 858 859if (Array.prototype.indexOf) { 860 indexOf = Array.prototype.indexOf; 861} else { 862 indexOf = function (o) { 863 // I know 864 var i; 865 for (i = 0; i < this.length; ++i) { 866 if (this[i] === o) { 867 return i; 868 } 869 } 870 return -1; 871 }; 872} 873 874function daysInMonth(year, month) { 875 if (isNaN(year) || isNaN(month)) { 876 return NaN; 877 } 878 var modMonth = mod(month, 12); 879 year += (month - modMonth) / 12; 880 return modMonth === 1 ? (isLeapYear(year) ? 29 : 28) : (31 - modMonth % 7 % 2); 881} 882 883// FORMATTING 884 885addFormatToken('M', ['MM', 2], 'Mo', function () { 886 return this.month() + 1; 887}); 888 889addFormatToken('MMM', 0, 0, function (format) { 890 return this.localeData().monthsShort(this, format); 891}); 892 893addFormatToken('MMMM', 0, 0, function (format) { 894 return this.localeData().months(this, format); 895}); 896 897// ALIASES 898 899addUnitAlias('month', 'M'); 900 901// PRIORITY 902 903addUnitPriority('month', 8); 904 905// PARSING 906 907addRegexToken('M', match1to2); 908addRegexToken('MM', match1to2, match2); 909addRegexToken('MMM', function (isStrict, locale) { 910 return locale.monthsShortRegex(isStrict); 911}); 912addRegexToken('MMMM', function (isStrict, locale) { 913 return locale.monthsRegex(isStrict); 914}); 915 916addParseToken(['M', 'MM'], function (input, array) { 917 array[MONTH] = toInt(input) - 1; 918}); 919 920addParseToken(['MMM', 'MMMM'], function (input, array, config, token) { 921 var month = config._locale.monthsParse(input, token, config._strict); 922 // if we didn't find a month name, mark the date as invalid. 923 if (month != null) { 924 array[MONTH] = month; 925 } else { 926 getParsingFlags(config).invalidMonth = input; 927 } 928}); 929 930// LOCALES 931 932var MONTHS_IN_FORMAT = /D[oD]?(\[[^\[\]]*\]|\s)+MMMM?/; 933var defaultLocaleMonths = 'January_February_March_April_May_June_July_August_September_October_November_December'.split('_'); 934function localeMonths (m, format) { 935 if (!m) { 936 return isArray(this._months) ? this._months : 937 this._months['standalone']; 938 } 939 return isArray(this._months) ? this._months[m.month()] : 940 this._months[(this._months.isFormat || MONTHS_IN_FORMAT).test(format) ? 'format' : 'standalone'][m.month()]; 941} 942 943var defaultLocaleMonthsShort = 'Jan_Feb_Mar_Apr_May_Jun_Jul_Aug_Sep_Oct_Nov_Dec'.split('_'); 944function localeMonthsShort (m, format) { 945 if (!m) { 946 return isArray(this._monthsShort) ? this._monthsShort : 947 this._monthsShort['standalone']; 948 } 949 return isArray(this._monthsShort) ? this._monthsShort[m.month()] : 950 this._monthsShort[MONTHS_IN_FORMAT.test(format) ? 'format' : 'standalone'][m.month()]; 951} 952 953function handleStrictParse(monthName, format, strict) { 954 var i, ii, mom, llc = monthName.toLocaleLowerCase(); 955 if (!this._monthsParse) { 956 // this is not used 957 this._monthsParse = []; 958 this._longMonthsParse = []; 959 this._shortMonthsParse = []; 960 for (i = 0; i < 12; ++i) { 961 mom = createUTC([2000, i]); 962 this._shortMonthsParse[i] = this.monthsShort(mom, '').toLocaleLowerCase(); 963 this._longMonthsParse[i] = this.months(mom, '').toLocaleLowerCase(); 964 } 965 } 966 967 if (strict) { 968 if (format === 'MMM') { 969 ii = indexOf.call(this._shortMonthsParse, llc); 970 return ii !== -1 ? ii : null; 971 } else { 972 ii = indexOf.call(this._longMonthsParse, llc); 973 return ii !== -1 ? ii : null; 974 } 975 } else { 976 if (format === 'MMM') { 977 ii = indexOf.call(this._shortMonthsParse, llc); 978 if (ii !== -1) { 979 return ii; 980 } 981 ii = indexOf.call(this._longMonthsParse, llc); 982 return ii !== -1 ? ii : null; 983 } else { 984 ii = indexOf.call(this._longMonthsParse, llc); 985 if (ii !== -1) { 986 return ii; 987 } 988 ii = indexOf.call(this._shortMonthsParse, llc); 989 return ii !== -1 ? ii : null; 990 } 991 } 992} 993 994function localeMonthsParse (monthName, format, strict) { 995 var i, mom, regex; 996 997 if (this._monthsParseExact) { 998 return handleStrictParse.call(this, monthName, format, strict); 999 } 1000 1001 if (!this._monthsParse) { 1002 this._monthsParse = []; 1003 this._longMonthsParse = []; 1004 this._shortMonthsParse = []; 1005 } 1006 1007 // TODO: add sorting 1008 // Sorting makes sure if one month (or abbr) is a prefix of another 1009 // see sorting in computeMonthsParse 1010 for (i = 0; i < 12; i++) { 1011 // make the regex if we don't have it already 1012 mom = createUTC([2000, i]); 1013 if (strict && !this._longMonthsParse[i]) { 1014 this._longMonthsParse[i] = new RegExp('^' + this.months(mom, '').replace('.', '') + '$', 'i'); 1015 this._shortMonthsParse[i] = new RegExp('^' + this.monthsShort(mom, '').replace('.', '') + '$', 'i'); 1016 } 1017 if (!strict && !this._monthsParse[i]) { 1018 regex = '^' + this.months(mom, '') + '|^' + this.monthsShort(mom, ''); 1019 this._monthsParse[i] = new RegExp(regex.replace('.', ''), 'i'); 1020 } 1021 // test the regex 1022 if (strict && format === 'MMMM' && this._longMonthsParse[i].test(monthName)) { 1023 return i; 1024 } else if (strict && format === 'MMM' && this._shortMonthsParse[i].test(monthName)) { 1025 return i; 1026 } else if (!strict && this._monthsParse[i].test(monthName)) { 1027 return i; 1028 } 1029 } 1030} 1031 1032// MOMENTS 1033 1034function setMonth (mom, value) { 1035 var dayOfMonth; 1036 1037 if (!mom.isValid()) { 1038 // No op 1039 return mom; 1040 } 1041 1042 if (typeof value === 'string') { 1043 if (/^\d+$/.test(value)) { 1044 value = toInt(value); 1045 } else { 1046 value = mom.localeData().monthsParse(value); 1047 // TODO: Another silent failure? 1048 if (!isNumber(value)) { 1049 return mom; 1050 } 1051 } 1052 } 1053 1054 dayOfMonth = Math.min(mom.date(), daysInMonth(mom.year(), value)); 1055 mom._d['set' + (mom._isUTC ? 'UTC' : '') + 'Month'](value, dayOfMonth); 1056 return mom; 1057} 1058 1059function getSetMonth (value) { 1060 if (value != null) { 1061 setMonth(this, value); 1062 hooks.updateOffset(this, true); 1063 return this; 1064 } else { 1065 return get(this, 'Month'); 1066 } 1067} 1068 1069function getDaysInMonth () { 1070 return daysInMonth(this.year(), this.month()); 1071} 1072 1073var defaultMonthsShortRegex = matchWord; 1074function monthsShortRegex (isStrict) { 1075 if (this._monthsParseExact) { 1076 if (!hasOwnProp(this, '_monthsRegex')) { 1077 computeMonthsParse.call(this); 1078 } 1079 if (isStrict) { 1080 return this._monthsShortStrictRegex; 1081 } else { 1082 return this._monthsShortRegex; 1083 } 1084 } else { 1085 if (!hasOwnProp(this, '_monthsShortRegex')) { 1086 this._monthsShortRegex = defaultMonthsShortRegex; 1087 } 1088 return this._monthsShortStrictRegex && isStrict ? 1089 this._monthsShortStrictRegex : this._monthsShortRegex; 1090 } 1091} 1092 1093var defaultMonthsRegex = matchWord; 1094function monthsRegex (isStrict) { 1095 if (this._monthsParseExact) { 1096 if (!hasOwnProp(this, '_monthsRegex')) { 1097 computeMonthsParse.call(this); 1098 } 1099 if (isStrict) { 1100 return this._monthsStrictRegex; 1101 } else { 1102 return this._monthsRegex; 1103 } 1104 } else { 1105 if (!hasOwnProp(this, '_monthsRegex')) { 1106 this._monthsRegex = defaultMonthsRegex; 1107 } 1108 return this._monthsStrictRegex && isStrict ? 1109 this._monthsStrictRegex : this._monthsRegex; 1110 } 1111} 1112 1113function computeMonthsParse () { 1114 function cmpLenRev(a, b) { 1115 return b.length - a.length; 1116 } 1117 1118 var shortPieces = [], longPieces = [], mixedPieces = [], 1119 i, mom; 1120 for (i = 0; i < 12; i++) { 1121 // make the regex if we don't have it already 1122 mom = createUTC([2000, i]); 1123 shortPieces.push(this.monthsShort(mom, '')); 1124 longPieces.push(this.months(mom, '')); 1125 mixedPieces.push(this.months(mom, '')); 1126 mixedPieces.push(this.monthsShort(mom, '')); 1127 } 1128 // Sorting makes sure if one month (or abbr) is a prefix of another it 1129 // will match the longer piece. 1130 shortPieces.sort(cmpLenRev); 1131 longPieces.sort(cmpLenRev); 1132 mixedPieces.sort(cmpLenRev); 1133 for (i = 0; i < 12; i++) { 1134 shortPieces[i] = regexEscape(shortPieces[i]); 1135 longPieces[i] = regexEscape(longPieces[i]); 1136 } 1137 for (i = 0; i < 24; i++) { 1138 mixedPieces[i] = regexEscape(mixedPieces[i]); 1139 } 1140 1141 this._monthsRegex = new RegExp('^(' + mixedPieces.join('|') + ')', 'i'); 1142 this._monthsShortRegex = this._monthsRegex; 1143 this._monthsStrictRegex = new RegExp('^(' + longPieces.join('|') + ')', 'i'); 1144 this._monthsShortStrictRegex = new RegExp('^(' + shortPieces.join('|') + ')', 'i'); 1145} 1146 1147function createDate (y, m, d, h, M, s, ms) { 1148 // can't just apply() to create a date: 1149 // https://stackoverflow.com/q/181348 1150 var date = new Date(y, m, d, h, M, s, ms); 1151 1152 // the date constructor remaps years 0-99 to 1900-1999 1153 if (y < 100 && y >= 0 && isFinite(date.getFullYear())) { 1154 date.setFullYear(y); 1155 } 1156 return date; 1157} 1158 1159function createUTCDate (y) { 1160 var date = new Date(Date.UTC.apply(null, arguments)); 1161 1162 // the Date.UTC function remaps years 0-99 to 1900-1999 1163 if (y < 100 && y >= 0 && isFinite(date.getUTCFullYear())) { 1164 date.setUTCFullYear(y); 1165 } 1166 return date; 1167} 1168 1169// start-of-first-week - start-of-year 1170function firstWeekOffset(year, dow, doy) { 1171 var // first-week day -- which january is always in the first week (4 for iso, 1 for other) 1172 fwd = 7 + dow - doy, 1173 // first-week day local weekday -- which local weekday is fwd 1174 fwdlw = (7 + createUTCDate(year, 0, fwd).getUTCDay() - dow) % 7; 1175 1176 return -fwdlw + fwd - 1; 1177} 1178 1179// https://en.wikipedia.org/wiki/ISO_week_date#Calculating_a_date_given_the_year.2C_week_number_and_weekday 1180function dayOfYearFromWeeks(year, week, weekday, dow, doy) { 1181 var localWeekday = (7 + weekday - dow) % 7, 1182 weekOffset = firstWeekOffset(year, dow, doy), 1183 dayOfYear = 1 + 7 * (week - 1) + localWeekday + weekOffset, 1184 resYear, resDayOfYear; 1185 1186 if (dayOfYear <= 0) { 1187 resYear = year - 1; 1188 resDayOfYear = daysInYear(resYear) + dayOfYear; 1189 } else if (dayOfYear > daysInYear(year)) { 1190 resYear = year + 1; 1191 resDayOfYear = dayOfYear - daysInYear(year); 1192 } else { 1193 resYear = year; 1194 resDayOfYear = dayOfYear; 1195 } 1196 1197 return { 1198 year: resYear, 1199 dayOfYear: resDayOfYear 1200 }; 1201} 1202 1203function weekOfYear(mom, dow, doy) { 1204 var weekOffset = firstWeekOffset(mom.year(), dow, doy), 1205 week = Math.floor((mom.dayOfYear() - weekOffset - 1) / 7) + 1, 1206 resWeek, resYear; 1207 1208 if (week < 1) { 1209 resYear = mom.year() - 1; 1210 resWeek = week + weeksInYear(resYear, dow, doy); 1211 } else if (week > weeksInYear(mom.year(), dow, doy)) { 1212 resWeek = week - weeksInYear(mom.year(), dow, doy); 1213 resYear = mom.year() + 1; 1214 } else { 1215 resYear = mom.year(); 1216 resWeek = week; 1217 } 1218 1219 return { 1220 week: resWeek, 1221 year: resYear 1222 }; 1223} 1224 1225function weeksInYear(year, dow, doy) { 1226 var weekOffset = firstWeekOffset(year, dow, doy), 1227 weekOffsetNext = firstWeekOffset(year + 1, dow, doy); 1228 return (daysInYear(year) - weekOffset + weekOffsetNext) / 7; 1229} 1230 1231// FORMATTING 1232 1233addFormatToken('w', ['ww', 2], 'wo', 'week'); 1234addFormatToken('W', ['WW', 2], 'Wo', 'isoWeek'); 1235 1236// ALIASES 1237 1238addUnitAlias('week', 'w'); 1239addUnitAlias('isoWeek', 'W'); 1240 1241// PRIORITIES 1242 1243addUnitPriority('week', 5); 1244addUnitPriority('isoWeek', 5); 1245 1246// PARSING 1247 1248addRegexToken('w', match1to2); 1249addRegexToken('ww', match1to2, match2); 1250addRegexToken('W', match1to2); 1251addRegexToken('WW', match1to2, match2); 1252 1253addWeekParseToken(['w', 'ww', 'W', 'WW'], function (input, week, config, token) { 1254 week[token.substr(0, 1)] = toInt(input); 1255}); 1256 1257// HELPERS 1258 1259// LOCALES 1260 1261function localeWeek (mom) { 1262 return weekOfYear(mom, this._week.dow, this._week.doy).week; 1263} 1264 1265var defaultLocaleWeek = { 1266 dow : 0, // Sunday is the first day of the week. 1267 doy : 6 // The week that contains Jan 1st is the first week of the year. 1268}; 1269 1270function localeFirstDayOfWeek () { 1271 return this._week.dow; 1272} 1273 1274function localeFirstDayOfYear () { 1275 return this._week.doy; 1276} 1277 1278// MOMENTS 1279 1280function getSetWeek (input) { 1281 var week = this.localeData().week(this); 1282 return input == null ? week : this.add((input - week) * 7, 'd'); 1283} 1284 1285function getSetISOWeek (input) { 1286 var week = weekOfYear(this, 1, 4).week; 1287 return input == null ? week : this.add((input - week) * 7, 'd'); 1288} 1289 1290// FORMATTING 1291 1292addFormatToken('d', 0, 'do', 'day'); 1293 1294addFormatToken('dd', 0, 0, function (format) { 1295 return this.localeData().weekdaysMin(this, format); 1296}); 1297 1298addFormatToken('ddd', 0, 0, function (format) { 1299 return this.localeData().weekdaysShort(this, format); 1300}); 1301 1302addFormatToken('dddd', 0, 0, function (format) { 1303 return this.localeData().weekdays(this, format); 1304}); 1305 1306addFormatToken('e', 0, 0, 'weekday'); 1307addFormatToken('E', 0, 0, 'isoWeekday'); 1308 1309// ALIASES 1310 1311addUnitAlias('day', 'd'); 1312addUnitAlias('weekday', 'e'); 1313addUnitAlias('isoWeekday', 'E'); 1314 1315// PRIORITY 1316addUnitPriority('day', 11); 1317addUnitPriority('weekday', 11); 1318addUnitPriority('isoWeekday', 11); 1319 1320// PARSING 1321 1322addRegexToken('d', match1to2); 1323addRegexToken('e', match1to2); 1324addRegexToken('E', match1to2); 1325addRegexToken('dd', function (isStrict, locale) { 1326 return locale.weekdaysMinRegex(isStrict); 1327}); 1328addRegexToken('ddd', function (isStrict, locale) { 1329 return locale.weekdaysShortRegex(isStrict); 1330}); 1331addRegexToken('dddd', function (isStrict, locale) { 1332 return locale.weekdaysRegex(isStrict); 1333}); 1334 1335addWeekParseToken(['dd', 'ddd', 'dddd'], function (input, week, config, token) { 1336 var weekday = config._locale.weekdaysParse(input, token, config._strict); 1337 // if we didn't get a weekday name, mark the date as invalid 1338 if (weekday != null) { 1339 week.d = weekday; 1340 } else { 1341 getParsingFlags(config).invalidWeekday = input; 1342 } 1343}); 1344 1345addWeekParseToken(['d', 'e', 'E'], function (input, week, config, token) { 1346 week[token] = toInt(input); 1347}); 1348 1349// HELPERS 1350 1351function parseWeekday(input, locale) { 1352 if (typeof input !== 'string') { 1353 return input; 1354 } 1355 1356 if (!isNaN(input)) { 1357 return parseInt(input, 10); 1358 } 1359 1360 input = locale.weekdaysParse(input); 1361 if (typeof input === 'number') { 1362 return input; 1363 } 1364 1365 return null; 1366} 1367 1368function parseIsoWeekday(input, locale) { 1369 if (typeof input === 'string') { 1370 return locale.weekdaysParse(input) % 7 || 7; 1371 } 1372 return isNaN(input) ? null : input; 1373} 1374 1375// LOCALES 1376 1377var defaultLocaleWeekdays = 'Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday'.split('_'); 1378function localeWeekdays (m, format) { 1379 if (!m) { 1380 return isArray(this._weekdays) ? this._weekdays : 1381 this._weekdays['standalone']; 1382 } 1383 return isArray(this._weekdays) ? this._weekdays[m.day()] : 1384 this._weekdays[this._weekdays.isFormat.test(format) ? 'format' : 'standalone'][m.day()]; 1385} 1386 1387var defaultLocaleWeekdaysShort = 'Sun_Mon_Tue_Wed_Thu_Fri_Sat'.split('_'); 1388function localeWeekdaysShort (m) { 1389 return (m) ? this._weekdaysShort[m.day()] : this._weekdaysShort; 1390} 1391 1392var defaultLocaleWeekdaysMin = 'Su_Mo_Tu_We_Th_Fr_Sa'.split('_'); 1393function localeWeekdaysMin (m) { 1394 return (m) ? this._weekdaysMin[m.day()] : this._weekdaysMin; 1395} 1396 1397function handleStrictParse$1(weekdayName, format, strict) { 1398 var i, ii, mom, llc = weekdayName.toLocaleLowerCase(); 1399 if (!this._weekdaysParse) { 1400 this._weekdaysParse = []; 1401 this._shortWeekdaysParse = []; 1402 this._minWeekdaysParse = []; 1403 1404 for (i = 0; i < 7; ++i) { 1405 mom = createUTC([2000, 1]).day(i); 1406 this._minWeekdaysParse[i] = this.weekdaysMin(mom, '').toLocaleLowerCase(); 1407 this._shortWeekdaysParse[i] = this.weekdaysShort(mom, '').toLocaleLowerCase(); 1408 this._weekdaysParse[i] = this.weekdays(mom, '').toLocaleLowerCase(); 1409 } 1410 } 1411 1412 if (strict) { 1413 if (format === 'dddd') { 1414 ii = indexOf.call(this._weekdaysParse, llc); 1415 return ii !== -1 ? ii : null; 1416 } else if (format === 'ddd') { 1417 ii = indexOf.call(this._shortWeekdaysParse, llc); 1418 return ii !== -1 ? ii : null; 1419 } else { 1420 ii = indexOf.call(this._minWeekdaysParse, llc); 1421 return ii !== -1 ? ii : null; 1422 } 1423 } else { 1424 if (format === 'dddd') { 1425 ii = indexOf.call(this._weekdaysParse, llc); 1426 if (ii !== -1) { 1427 return ii; 1428 } 1429 ii = indexOf.call(this._shortWeekdaysParse, llc); 1430 if (ii !== -1) { 1431 return ii; 1432 } 1433 ii = indexOf.call(this._minWeekdaysParse, llc); 1434 return ii !== -1 ? ii : null; 1435 } else if (format === 'ddd') { 1436 ii = indexOf.call(this._shortWeekdaysParse, llc); 1437 if (ii !== -1) { 1438 return ii; 1439 } 1440 ii = indexOf.call(this._weekdaysParse, llc); 1441 if (ii !== -1) { 1442 return ii; 1443 } 1444 ii = indexOf.call(this._minWeekdaysParse, llc); 1445 return ii !== -1 ? ii : null; 1446 } else { 1447 ii = indexOf.call(this._minWeekdaysParse, llc); 1448 if (ii !== -1) { 1449 return ii; 1450 } 1451 ii = indexOf.call(this._weekdaysParse, llc); 1452 if (ii !== -1) { 1453 return ii; 1454 } 1455 ii = indexOf.call(this._shortWeekdaysParse, llc); 1456 return ii !== -1 ? ii : null; 1457 } 1458 } 1459} 1460 1461function localeWeekdaysParse (weekdayName, format, strict) { 1462 var i, mom, regex; 1463 1464 if (this._weekdaysParseExact) { 1465 return handleStrictParse$1.call(this, weekdayName, format, strict); 1466 } 1467 1468 if (!this._weekdaysParse) { 1469 this._weekdaysParse = []; 1470 this._minWeekdaysParse = []; 1471 this._shortWeekdaysParse = []; 1472 this._fullWeekdaysParse = []; 1473 } 1474 1475 for (i = 0; i < 7; i++) { 1476 // make the regex if we don't have it already 1477 1478 mom = createUTC([2000, 1]).day(i); 1479 if (strict && !this._fullWeekdaysParse[i]) { 1480 this._fullWeekdaysParse[i] = new RegExp('^' + this.weekdays(mom, '').replace('.', '\.?') + '$', 'i'); 1481 this._shortWeekdaysParse[i] = new RegExp('^' + this.weekdaysShort(mom, '').replace('.', '\.?') + '$', 'i'); 1482 this._minWeekdaysParse[i] = new RegExp('^' + this.weekdaysMin(mom, '').replace('.', '\.?') + '$', 'i'); 1483 } 1484 if (!this._weekdaysParse[i]) { 1485 regex = '^' + this.weekdays(mom, '') + '|^' + this.weekdaysShort(mom, '') + '|^' + this.weekdaysMin(mom, ''); 1486 this._weekdaysParse[i] = new RegExp(regex.replace('.', ''), 'i'); 1487 } 1488 // test the regex 1489 if (strict && format === 'dddd' && this._fullWeekdaysParse[i].test(weekdayName)) { 1490 return i; 1491 } else if (strict && format === 'ddd' && this._shortWeekdaysParse[i].test(weekdayName)) { 1492 return i; 1493 } else if (strict && format === 'dd' && this._minWeekdaysParse[i].test(weekdayName)) { 1494 return i; 1495 } else if (!strict && this._weekdaysParse[i].test(weekdayName)) { 1496 return i; 1497 } 1498 } 1499} 1500 1501// MOMENTS 1502 1503function getSetDayOfWeek (input) { 1504 if (!this.isValid()) { 1505 return input != null ? this : NaN; 1506 } 1507 var day = this._isUTC ? this._d.getUTCDay() : this._d.getDay(); 1508 if (input != null) { 1509 input = parseWeekday(input, this.localeData()); 1510 return this.add(input - day, 'd'); 1511 } else { 1512 return day; 1513 } 1514} 1515 1516function getSetLocaleDayOfWeek (input) { 1517 if (!this.isValid()) { 1518 return input != null ? this : NaN; 1519 } 1520 var weekday = (this.day() + 7 - this.localeData()._week.dow) % 7; 1521 return input == null ? weekday : this.add(input - weekday, 'd'); 1522} 1523 1524function getSetISODayOfWeek (input) { 1525 if (!this.isValid()) { 1526 return input != null ? this : NaN; 1527 } 1528 1529 // behaves the same as moment#day except 1530 // as a getter, returns 7 instead of 0 (1-7 range instead of 0-6) 1531 // as a setter, sunday should belong to the previous week. 1532 1533 if (input != null) { 1534 var weekday = parseIsoWeekday(input, this.localeData()); 1535 return this.day(this.day() % 7 ? weekday : weekday - 7); 1536 } else { 1537 return this.day() || 7; 1538 } 1539} 1540 1541var defaultWeekdaysRegex = matchWord; 1542function weekdaysRegex (isStrict) { 1543 if (this._weekdaysParseExact) { 1544 if (!hasOwnProp(this, '_weekdaysRegex')) { 1545 computeWeekdaysParse.call(this); 1546 } 1547 if (isStrict) { 1548 return this._weekdaysStrictRegex; 1549 } else { 1550 return this._weekdaysRegex; 1551 } 1552 } else { 1553 if (!hasOwnProp(this, '_weekdaysRegex')) { 1554 this._weekdaysRegex = defaultWeekdaysRegex; 1555 } 1556 return this._weekdaysStrictRegex && isStrict ? 1557 this._weekdaysStrictRegex : this._weekdaysRegex; 1558 } 1559} 1560 1561var defaultWeekdaysShortRegex = matchWord; 1562function weekdaysShortRegex (isStrict) { 1563 if (this._weekdaysParseExact) { 1564 if (!hasOwnProp(this, '_weekdaysRegex')) { 1565 computeWeekdaysParse.call(this); 1566 } 1567 if (isStrict) { 1568 return this._weekdaysShortStrictRegex; 1569 } else { 1570 return this._weekdaysShortRegex; 1571 } 1572 } else { 1573 if (!hasOwnProp(this, '_weekdaysShortRegex')) { 1574 this._weekdaysShortRegex = defaultWeekdaysShortRegex; 1575 } 1576 return this._weekdaysShortStrictRegex && isStrict ? 1577 this._weekdaysShortStrictRegex : this._weekdaysShortRegex; 1578 } 1579} 1580 1581var defaultWeekdaysMinRegex = matchWord; 1582function weekdaysMinRegex (isStrict) { 1583 if (this._weekdaysParseExact) { 1584 if (!hasOwnProp(this, '_weekdaysRegex')) { 1585 computeWeekdaysParse.call(this); 1586 } 1587 if (isStrict) { 1588 return this._weekdaysMinStrictRegex; 1589 } else { 1590 return this._weekdaysMinRegex; 1591 } 1592 } else { 1593 if (!hasOwnProp(this, '_weekdaysMinRegex')) { 1594 this._weekdaysMinRegex = defaultWeekdaysMinRegex; 1595 } 1596 return this._weekdaysMinStrictRegex && isStrict ? 1597 this._weekdaysMinStrictRegex : this._weekdaysMinRegex; 1598 } 1599} 1600 1601 1602function computeWeekdaysParse () { 1603 function cmpLenRev(a, b) { 1604 return b.length - a.length; 1605 } 1606 1607 var minPieces = [], shortPieces = [], longPieces = [], mixedPieces = [], 1608 i, mom, minp, shortp, longp; 1609 for (i = 0; i < 7; i++) { 1610 // make the regex if we don't have it already 1611 mom = createUTC([2000, 1]).day(i); 1612 minp = this.weekdaysMin(mom, ''); 1613 shortp = this.weekdaysShort(mom, ''); 1614 longp = this.weekdays(mom, ''); 1615 minPieces.push(minp); 1616 shortPieces.push(shortp); 1617 longPieces.push(longp); 1618 mixedPieces.push(minp); 1619 mixedPieces.push(shortp); 1620 mixedPieces.push(longp); 1621 } 1622 // Sorting makes sure if one weekday (or abbr) is a prefix of another it 1623 // will match the longer piece. 1624 minPieces.sort(cmpLenRev); 1625 shortPieces.sort(cmpLenRev); 1626 longPieces.sort(cmpLenRev); 1627 mixedPieces.sort(cmpLenRev); 1628 for (i = 0; i < 7; i++) { 1629 shortPieces[i] = regexEscape(shortPieces[i]); 1630 longPieces[i] = regexEscape(longPieces[i]); 1631 mixedPieces[i] = regexEscape(mixedPieces[i]); 1632 } 1633 1634 this._weekdaysRegex = new RegExp('^(' + mixedPieces.join('|') + ')', 'i'); 1635 this._weekdaysShortRegex = this._weekdaysRegex; 1636 this._weekdaysMinRegex = this._weekdaysRegex; 1637 1638 this._weekdaysStrictRegex = new RegExp('^(' + longPieces.join('|') + ')', 'i'); 1639 this._weekdaysShortStrictRegex = new RegExp('^(' + shortPieces.join('|') + ')', 'i'); 1640 this._weekdaysMinStrictRegex = new RegExp('^(' + minPieces.join('|') + ')', 'i'); 1641} 1642 1643// FORMATTING 1644 1645function hFormat() { 1646 return this.hours() % 12 || 12; 1647} 1648 1649function kFormat() { 1650 return this.hours() || 24; 1651} 1652 1653addFormatToken('H', ['HH', 2], 0, 'hour'); 1654addFormatToken('h', ['hh', 2], 0, hFormat); 1655addFormatToken('k', ['kk', 2], 0, kFormat); 1656 1657addFormatToken('hmm', 0, 0, function () { 1658 return '' + hFormat.apply(this) + zeroFill(this.minutes(), 2); 1659}); 1660 1661addFormatToken('hmmss', 0, 0, function () { 1662 return '' + hFormat.apply(this) + zeroFill(this.minutes(), 2) + 1663 zeroFill(this.seconds(), 2); 1664}); 1665 1666addFormatToken('Hmm', 0, 0, function () { 1667 return '' + this.hours() + zeroFill(this.minutes(), 2); 1668}); 1669 1670addFormatToken('Hmmss', 0, 0, function () { 1671 return '' + this.hours() + zeroFill(this.minutes(), 2) + 1672 zeroFill(this.seconds(), 2); 1673}); 1674 1675function meridiem (token, lowercase) { 1676 addFormatToken(token, 0, 0, function () { 1677 return this.localeData().meridiem(this.hours(), this.minutes(), lowercase); 1678 }); 1679} 1680 1681meridiem('a', true); 1682meridiem('A', false); 1683 1684// ALIASES 1685 1686addUnitAlias('hour', 'h'); 1687 1688// PRIORITY 1689addUnitPriority('hour', 13); 1690 1691// PARSING 1692 1693function matchMeridiem (isStrict, locale) { 1694 return locale._meridiemParse; 1695} 1696 1697addRegexToken('a', matchMeridiem); 1698addRegexToken('A', matchMeridiem); 1699addRegexToken('H', match1to2); 1700addRegexToken('h', match1to2); 1701addRegexToken('k', match1to2); 1702addRegexToken('HH', match1to2, match2); 1703addRegexToken('hh', match1to2, match2); 1704addRegexToken('kk', match1to2, match2); 1705 1706addRegexToken('hmm', match3to4); 1707addRegexToken('hmmss', match5to6); 1708addRegexToken('Hmm', match3to4); 1709addRegexToken('Hmmss', match5to6); 1710 1711addParseToken(['H', 'HH'], HOUR); 1712addParseToken(['k', 'kk'], function (input, array, config) { 1713 var kInput = toInt(input); 1714 array[HOUR] = kInput === 24 ? 0 : kInput; 1715}); 1716addParseToken(['a', 'A'], function (input, array, config) { 1717 config._isPm = config._locale.isPM(input); 1718 config._meridiem = input; 1719}); 1720addParseToken(['h', 'hh'], function (input, array, config) { 1721 array[HOUR] = toInt(input); 1722 getParsingFlags(config).bigHour = true; 1723}); 1724addParseToken('hmm', function (input, array, config) { 1725 var pos = input.length - 2; 1726 array[HOUR] = toInt(input.substr(0, pos)); 1727 array[MINUTE] = toInt(input.substr(pos)); 1728 getParsingFlags(config).bigHour = true; 1729}); 1730addParseToken('hmmss', function (input, array, config) { 1731 var pos1 = input.length - 4; 1732 var pos2 = input.length - 2; 1733 array[HOUR] = toInt(input.substr(0, pos1)); 1734 array[MINUTE] = toInt(input.substr(pos1, 2)); 1735 array[SECOND] = toInt(input.substr(pos2)); 1736 getParsingFlags(config).bigHour = true; 1737}); 1738addParseToken('Hmm', function (input, array, config) { 1739 var pos = input.length - 2; 1740 array[HOUR] = toInt(input.substr(0, pos)); 1741 array[MINUTE] = toInt(input.substr(pos)); 1742}); 1743addParseToken('Hmmss', function (input, array, config) { 1744 var pos1 = input.length - 4; 1745 var pos2 = input.length - 2; 1746 array[HOUR] = toInt(input.substr(0, pos1)); 1747 array[MINUTE] = toInt(input.substr(pos1, 2)); 1748 array[SECOND] = toInt(input.substr(pos2)); 1749}); 1750 1751// LOCALES 1752 1753function localeIsPM (input) { 1754 // IE8 Quirks Mode & IE7 Standards Mode do not allow accessing strings like arrays 1755 // Using charAt should be more compatible. 1756 return ((input + '').toLowerCase().charAt(0) === 'p'); 1757} 1758 1759var defaultLocaleMeridiemParse = /[ap]\.?m?\.?/i; 1760function localeMeridiem (hours, minutes, isLower) { 1761 if (hours > 11) { 1762 return isLower ? 'pm' : 'PM'; 1763 } else { 1764 return isLower ? 'am' : 'AM'; 1765 } 1766} 1767 1768 1769// MOMENTS 1770 1771// Setting the hour should keep the time, because the user explicitly 1772// specified which hour he wants. So trying to maintain the same hour (in 1773// a new timezone) makes sense. Adding/subtracting hours does not follow 1774// this rule. 1775var getSetHour = makeGetSet('Hours', true); 1776 1777// months 1778// week 1779// weekdays 1780// meridiem 1781var baseConfig = { 1782 calendar: defaultCalendar, 1783 longDateFormat: defaultLongDateFormat, 1784 invalidDate: defaultInvalidDate, 1785 ordinal: defaultOrdinal, 1786 dayOfMonthOrdinalParse: defaultDayOfMonthOrdinalParse, 1787 relativeTime: defaultRelativeTime, 1788 1789 months: defaultLocaleMonths, 1790 monthsShort: defaultLocaleMonthsShort, 1791 1792 week: defaultLocaleWeek, 1793 1794 weekdays: defaultLocaleWeekdays, 1795 weekdaysMin: defaultLocaleWeekdaysMin, 1796 weekdaysShort: defaultLocaleWeekdaysShort, 1797 1798 meridiemParse: defaultLocaleMeridiemParse 1799}; 1800 1801// internal storage for locale config files 1802var locales = {}; 1803var localeFamilies = {}; 1804var globalLocale; 1805 1806function normalizeLocale(key) { 1807 return key ? key.toLowerCase().replace('_', '-') : key; 1808} 1809 1810// pick the locale from the array 1811// try ['en-au', 'en-gb'] as 'en-au', 'en-gb', 'en', as in move through the list trying each 1812// substring from most specific to least, but move to the next array item if it's a more specific variant than the current root 1813function chooseLocale(names) { 1814 var i = 0, j, next, locale, split; 1815 1816 while (i < names.length) { 1817 split = normalizeLocale(names[i]).split('-'); 1818 j = split.length; 1819 next = normalizeLocale(names[i + 1]); 1820 next = next ? next.split('-') : null; 1821 while (j > 0) { 1822 locale = loadLocale(split.slice(0, j).join('-')); 1823 if (locale) { 1824 return locale; 1825 } 1826 if (next && next.length >= j && compareArrays(split, next, true) >= j - 1) { 1827 //the next array item is better than a shallower substring of this one 1828 break; 1829 } 1830 j--; 1831 } 1832 i++; 1833 } 1834 return null; 1835} 1836 1837function loadLocale(name) { 1838 var oldLocale = null; 1839 // TODO: Find a better way to register and load all the locales in Node 1840 if (!locales[name] && (typeof module !== 'undefined') && 1841 module && module.exports) { 1842 try { 1843 oldLocale = globalLocale._abbr; 1844 var aliasedRequire = require; 1845 aliasedRequire('./locale/' + name); 1846 getSetGlobalLocale(oldLocale); 1847 } catch (e) {} 1848 } 1849 return locales[name]; 1850} 1851 1852// This function will load locale and then set the global locale. If 1853// no arguments are passed in, it will simply return the current global 1854// locale key. 1855function getSetGlobalLocale (key, values) { 1856 var data; 1857 if (key) { 1858 if (isUndefined(values)) { 1859 data = getLocale(key); 1860 } 1861 else { 1862 data = defineLocale(key, values); 1863 } 1864 1865 if (data) { 1866 // moment.duration._locale = moment._locale = data; 1867 globalLocale = data; 1868 } 1869 } 1870 1871 return globalLocale._abbr; 1872} 1873 1874function defineLocale (name, config) { 1875 if (config !== null) { 1876 var parentConfig = baseConfig; 1877 config.abbr = name; 1878 if (locales[name] != null) { 1879 deprecateSimple('defineLocaleOverride', 1880 'use moment.updateLocale(localeName, config) to change ' + 1881 'an existing locale. moment.defineLocale(localeName, ' + 1882 'config) should only be used for creating a new locale ' + 1883 'See http://momentjs.com/guides/#/warnings/define-locale/ for more info.'); 1884 parentConfig = locales[name]._config; 1885 } else if (config.parentLocale != null) { 1886 if (locales[config.parentLocale] != null) { 1887 parentConfig = locales[config.parentLocale]._config; 1888 } else { 1889 if (!localeFamilies[config.parentLocale]) { 1890 localeFamilies[config.parentLocale] = []; 1891 } 1892 localeFamilies[config.parentLocale].push({ 1893 name: name, 1894 config: config 1895 }); 1896 return null; 1897 } 1898 } 1899 locales[name] = new Locale(mergeConfigs(parentConfig, config)); 1900 1901 if (localeFamilies[name]) { 1902 localeFamilies[name].forEach(function (x) { 1903 defineLocale(x.name, x.config); 1904 }); 1905 } 1906 1907 // backwards compat for now: also set the locale 1908 // make sure we set the locale AFTER all child locales have been 1909 // created, so we won't end up with the child locale set. 1910 getSetGlobalLocale(name); 1911 1912 1913 return locales[name]; 1914 } else { 1915 // useful for testing 1916 delete locales[name]; 1917 return null; 1918 } 1919} 1920 1921function updateLocale(name, config) { 1922 if (config != null) { 1923 var locale, tmpLocale, parentConfig = baseConfig; 1924 // MERGE 1925 tmpLocale = loadLocale(name); 1926 if (tmpLocale != null) { 1927 parentConfig = tmpLocale._config; 1928 } 1929 config = mergeConfigs(parentConfig, config); 1930 locale = new Locale(config); 1931 locale.parentLocale = locales[name]; 1932 locales[name] = locale; 1933 1934 // backwards compat for now: also set the locale 1935 getSetGlobalLocale(name); 1936 } else { 1937 // pass null for config to unupdate, useful for tests 1938 if (locales[name] != null) { 1939 if (locales[name].parentLocale != null) { 1940 locales[name] = locales[name].parentLocale; 1941 } else if (locales[name] != null) { 1942 delete locales[name]; 1943 } 1944 } 1945 } 1946 return locales[name]; 1947} 1948 1949// returns locale data 1950function getLocale (key) { 1951 var locale; 1952 1953 if (key && key._locale && key._locale._abbr) { 1954 key = key._locale._abbr; 1955 } 1956 1957 if (!key) { 1958 return globalLocale; 1959 } 1960 1961 if (!isArray(key)) { 1962 //short-circuit everything else 1963 locale = loadLocale(key); 1964 if (locale) { 1965 return locale; 1966 } 1967 key = [key]; 1968 } 1969 1970 return chooseLocale(key); 1971} 1972 1973function listLocales() { 1974 return keys(locales); 1975} 1976 1977function checkOverflow (m) { 1978 var overflow; 1979 var a = m._a; 1980 1981 if (a && getParsingFlags(m).overflow === -2) { 1982 overflow = 1983 a[MONTH] < 0 || a[MONTH] > 11 ? MONTH : 1984 a[DATE] < 1 || a[DATE] > daysInMonth(a[YEAR], a[MONTH]) ? DATE : 1985 a[HOUR] < 0 || a[HOUR] > 24 || (a[HOUR] === 24 && (a[MINUTE] !== 0 || a[SECOND] !== 0 || a[MILLISECOND] !== 0)) ? HOUR : 1986 a[MINUTE] < 0 || a[MINUTE] > 59 ? MINUTE : 1987 a[SECOND] < 0 || a[SECOND] > 59 ? SECOND : 1988 a[MILLISECOND] < 0 || a[MILLISECOND] > 999 ? MILLISECOND : 1989 -1; 1990 1991 if (getParsingFlags(m)._overflowDayOfYear && (overflow < YEAR || overflow > DATE)) { 1992 overflow = DATE; 1993 } 1994 if (getParsingFlags(m)._overflowWeeks && overflow === -1) { 1995 overflow = WEEK; 1996 } 1997 if (getParsingFlags(m)._overflowWeekday && overflow === -1) { 1998 overflow = WEEKDAY; 1999 } 2000 2001 getParsingFlags(m).overflow = overflow; 2002 } 2003 2004 return m; 2005} 2006 2007// Pick the first defined of two or three arguments. 2008function defaults(a, b, c) { 2009 if (a != null) { 2010 return a; 2011 } 2012 if (b != null) { 2013 return b; 2014 } 2015 return c; 2016} 2017 2018function currentDateArray(config) { 2019 // hooks is actually the exported moment object 2020 var nowValue = new Date(hooks.now()); 2021 if (config._useUTC) { 2022 return [nowValue.getUTCFullYear(), nowValue.getUTCMonth(), nowValue.getUTCDate()]; 2023 } 2024 return [nowValue.getFullYear(), nowValue.getMonth(), nowValue.getDate()]; 2025} 2026 2027// convert an array to a date. 2028// the array should mirror the parameters below 2029// note: all values past the year are optional and will default to the lowest possible value. 2030// [year, month, day , hour, minute, second, millisecond] 2031function configFromArray (config) { 2032 var i, date, input = [], currentDate, expectedWeekday, yearToUse; 2033 2034 if (config._d) { 2035 return; 2036 } 2037 2038 currentDate = currentDateArray(config); 2039 2040 //compute day of the year from weeks and weekdays 2041 if (config._w && config._a[DATE] == null && config._a[MONTH] == null) { 2042 dayOfYearFromWeekInfo(config); 2043 } 2044 2045 //if the day of the year is set, figure out what it is 2046 if (config._dayOfYear != null) { 2047 yearToUse = defaults(config._a[YEAR], currentDate[YEAR]); 2048 2049 if (config._dayOfYear > daysInYear(yearToUse) || config._dayOfYear === 0) { 2050 getParsingFlags(config)._overflowDayOfYear = true; 2051 } 2052 2053 date = createUTCDate(yearToUse, 0, config._dayOfYear); 2054 config._a[MONTH] = date.getUTCMonth(); 2055 config._a[DATE] = date.getUTCDate(); 2056 } 2057 2058 // Default to current date. 2059 // * if no year, month, day of month are given, default to today 2060 // * if day of month is given, default month and year 2061 // * if month is given, default only year 2062 // * if year is given, don't default anything 2063 for (i = 0; i < 3 && config._a[i] == null; ++i) { 2064 config._a[i] = input[i] = currentDate[i]; 2065 } 2066 2067 // Zero out whatever was not defaulted, including time 2068 for (; i < 7; i++) { 2069 config._a[i] = input[i] = (config._a[i] == null) ? (i === 2 ? 1 : 0) : config._a[i]; 2070 } 2071 2072 // Check for 24:00:00.000 2073 if (config._a[HOUR] === 24 && 2074 config._a[MINUTE] === 0 && 2075 config._a[SECOND] === 0 && 2076 config._a[MILLISECOND] === 0) { 2077 config._nextDay = true; 2078 config._a[HOUR] = 0; 2079 } 2080 2081 config._d = (config._useUTC ? createUTCDate : createDate).apply(null, input); 2082 expectedWeekday = config._useUTC ? config._d.getUTCDay() : config._d.getDay(); 2083 2084 // Apply timezone offset from input. The actual utcOffset can be changed 2085 // with parseZone. 2086 if (config._tzm != null) { 2087 config._d.setUTCMinutes(config._d.getUTCMinutes() - config._tzm); 2088 } 2089 2090 if (config._nextDay) { 2091 config._a[HOUR] = 24; 2092 } 2093 2094 // check for mismatching day of week 2095 if (config._w && typeof config._w.d !== 'undefined' && config._w.d !== expectedWeekday) { 2096 getParsingFlags(config).weekdayMismatch = true; 2097 } 2098} 2099 2100function dayOfYearFromWeekInfo(config) { 2101 var w, weekYear, week, weekday, dow, doy, temp, weekdayOverflow; 2102 2103 w = config._w; 2104 if (w.GG != null || w.W != null || w.E != null) { 2105 dow = 1; 2106 doy = 4; 2107 2108 // TODO: We need to take the current isoWeekYear, but that depends on 2109 // how we interpret now (local, utc, fixed offset). So create 2110 // a now version of current config (take local/utc/offset flags, and 2111 // create now). 2112 weekYear = defaults(w.GG, config._a[YEAR], weekOfYear(createLocal(), 1, 4).year); 2113 week = defaults(w.W, 1); 2114 weekday = defaults(w.E, 1); 2115 if (weekday < 1 || weekday > 7) { 2116 weekdayOverflow = true; 2117 } 2118 } else { 2119 dow = config._locale._week.dow; 2120 doy = config._locale._week.doy; 2121 2122 var curWeek = weekOfYear(createLocal(), dow, doy); 2123 2124 weekYear = defaults(w.gg, config._a[YEAR], curWeek.year); 2125 2126 // Default to current week. 2127 week = defaults(w.w, curWeek.week); 2128 2129 if (w.d != null) { 2130 // weekday -- low day numbers are considered next week 2131 weekday = w.d; 2132 if (weekday < 0 || weekday > 6) { 2133 weekdayOverflow = true; 2134 } 2135 } else if (w.e != null) { 2136 // local weekday -- counting starts from begining of week 2137 weekday = w.e + dow; 2138 if (w.e < 0 || w.e > 6) { 2139 weekdayOverflow = true; 2140 } 2141 } else { 2142 // default to begining of week 2143 weekday = dow; 2144 } 2145 } 2146 if (week < 1 || week > weeksInYear(weekYear, dow, doy)) { 2147 getParsingFlags(config)._overflowWeeks = true; 2148 } else if (weekdayOverflow != null) { 2149 getParsingFlags(config)._overflowWeekday = true; 2150 } else { 2151 temp = dayOfYearFromWeeks(weekYear, week, weekday, dow, doy); 2152 config._a[YEAR] = temp.year; 2153 config._dayOfYear = temp.dayOfYear; 2154 } 2155} 2156 2157// iso 8601 regex 2158// 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) 2159var extendedIsoRegex = /^\s*((?:[+-]\d{6}|\d{4})-(?:\d\d-\d\d|W\d\d-\d|W\d\d|\d\d\d|\d\d))(?:(T| )(\d\d(?::\d\d(?::\d\d(?:[.,]\d+)?)?)?)([\+\-]\d\d(?::?\d\d)?|\s*Z)?)?$/; 2160var basicIsoRegex = /^\s*((?:[+-]\d{6}|\d{4})(?:\d\d\d\d|W\d\d\d|W\d\d|\d\d\d|\d\d))(?:(T| )(\d\d(?:\d\d(?:\d\d(?:[.,]\d+)?)?)?)([\+\-]\d\d(?::?\d\d)?|\s*Z)?)?$/; 2161 2162var tzRegex = /Z|[+-]\d\d(?::?\d\d)?/; 2163 2164var isoDates = [ 2165 ['YYYYYY-MM-DD', /[+-]\d{6}-\d\d-\d\d/], 2166 ['YYYY-MM-DD', /\d{4}-\d\d-\d\d/], 2167 ['GGGG-[W]WW-E', /\d{4}-W\d\d-\d/], 2168 ['GGGG-[W]WW', /\d{4}-W\d\d/, false], 2169 ['YYYY-DDD', /\d{4}-\d{3}/], 2170 ['YYYY-MM', /\d{4}-\d\d/, false], 2171 ['YYYYYYMMDD', /[+-]\d{10}/], 2172 ['YYYYMMDD', /\d{8}/], 2173 // YYYYMM is NOT allowed by the standard 2174 ['GGGG[W]WWE', /\d{4}W\d{3}/], 2175 ['GGGG[W]WW', /\d{4}W\d{2}/, false], 2176 ['YYYYDDD', /\d{7}/] 2177]; 2178 2179// iso time formats and regexes 2180var isoTimes = [ 2181 ['HH:mm:ss.SSSS', /\d\d:\d\d:\d\d\.\d+/], 2182 ['HH:mm:ss,SSSS', /\d\d:\d\d:\d\d,\d+/], 2183 ['HH:mm:ss', /\d\d:\d\d:\d\d/], 2184 ['HH:mm', /\d\d:\d\d/], 2185 ['HHmmss.SSSS', /\d\d\d\d\d\d\.\d+/], 2186 ['HHmmss,SSSS', /\d\d\d\d\d\d,\d+/], 2187 ['HHmmss', /\d\d\d\d\d\d/], 2188 ['HHmm', /\d\d\d\d/], 2189 ['HH', /\d\d/] 2190]; 2191 2192var aspNetJsonRegex = /^\/?Date\((\-?\d+)/i; 2193 2194// date from iso format 2195function configFromISO(config) { 2196 var i, l, 2197 string = config._i, 2198 match = extendedIsoRegex.exec(string) || basicIsoRegex.exec(string), 2199 allowTime, dateFormat, timeFormat, tzFormat; 2200 2201 if (match) { 2202 getParsingFlags(config).iso = true; 2203 2204 for (i = 0, l = isoDates.length; i < l; i++) { 2205 if (isoDates[i][1].exec(match[1])) { 2206 dateFormat = isoDates[i][0]; 2207 allowTime = isoDates[i][2] !== false; 2208 break; 2209 } 2210 } 2211 if (dateFormat == null) { 2212 config._isValid = false; 2213 return; 2214 } 2215 if (match[3]) { 2216 for (i = 0, l = isoTimes.length; i < l; i++) { 2217 if (isoTimes[i][1].exec(match[3])) { 2218 // match[2] should be 'T' or space 2219 timeFormat = (match[2] || ' ') + isoTimes[i][0]; 2220 break; 2221 } 2222 } 2223 if (timeFormat == null) { 2224 config._isValid = false; 2225 return; 2226 } 2227 } 2228 if (!allowTime && timeFormat != null) { 2229 config._isValid = false; 2230 return; 2231 } 2232 if (match[4]) { 2233 if (tzRegex.exec(match[4])) { 2234 tzFormat = 'Z'; 2235 } else { 2236 config._isValid = false; 2237 return; 2238 } 2239 } 2240 config._f = dateFormat + (timeFormat || '') + (tzFormat || ''); 2241 configFromStringAndFormat(config); 2242 } else { 2243 config._isValid = false; 2244 } 2245} 2246 2247// RFC 2822 regex: For details see https://tools.ietf.org/html/rfc2822#section-3.3 2248var rfc2822 = /^(?:(Mon|Tue|Wed|Thu|Fri|Sat|Sun),?\s)?(\d{1,2})\s(Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)\s(\d{2,4})\s(\d\d):(\d\d)(?::(\d\d))?\s(?:(UT|GMT|[ECMP][SD]T)|([Zz])|([+-]\d{4}))$/; 2249 2250function extractFromRFC2822Strings(yearStr, monthStr, dayStr, hourStr, minuteStr, secondStr) { 2251 var result = [ 2252 untruncateYear(yearStr), 2253 defaultLocaleMonthsShort.indexOf(monthStr), 2254 parseInt(dayStr, 10), 2255 parseInt(hourStr, 10), 2256 parseInt(minuteStr, 10) 2257 ]; 2258 2259 if (secondStr) { 2260 result.push(parseInt(secondStr, 10)); 2261 } 2262 2263 return result; 2264} 2265 2266function untruncateYear(yearStr) { 2267 var year = parseInt(yearStr, 10); 2268 if (year <= 49) { 2269 return 2000 + year; 2270 } else if (year <= 999) { 2271 return 1900 + year; 2272 } 2273 return year; 2274} 2275 2276function preprocessRFC2822(s) { 2277 // Remove comments and folding whitespace and replace multiple-spaces with a single space 2278 return s.replace(/\([^)]*\)|[\n\t]/g, ' ').replace(/(\s\s+)/g, ' ').trim(); 2279} 2280 2281function checkWeekday(weekdayStr, parsedInput, config) { 2282 if (weekdayStr) { 2283 // TODO: Replace the vanilla JS Date object with an indepentent day-of-week check. 2284 var weekdayProvided = defaultLocaleWeekdaysShort.indexOf(weekdayStr), 2285 weekdayActual = new Date(parsedInput[0], parsedInput[1], parsedInput[2]).getDay(); 2286 if (weekdayProvided !== weekdayActual) { 2287 getParsingFlags(config).weekdayMismatch = true; 2288 config._isValid = false; 2289 return false; 2290 } 2291 } 2292 return true; 2293} 2294 2295var obsOffsets = { 2296 UT: 0, 2297 GMT: 0, 2298 EDT: -4 * 60, 2299 EST: -5 * 60, 2300 CDT: -5 * 60, 2301 CST: -6 * 60, 2302 MDT: -6 * 60, 2303 MST: -7 * 60, 2304 PDT: -7 * 60, 2305 PST: -8 * 60 2306}; 2307 2308function calculateOffset(obsOffset, militaryOffset, numOffset) { 2309 if (obsOffset) { 2310 return obsOffsets[obsOffset]; 2311 } else if (militaryOffset) { 2312 // the only allowed military tz is Z 2313 return 0; 2314 } else { 2315 var hm = parseInt(numOffset, 10); 2316 var m = hm % 100, h = (hm - m) / 100; 2317 return h * 60 + m; 2318 } 2319} 2320 2321// date and time from ref 2822 format 2322function configFromRFC2822(config) { 2323 var match = rfc2822.exec(preprocessRFC2822(config._i)); 2324 if (match) { 2325 var parsedArray = extractFromRFC2822Strings(match[4], match[3], match[2], match[5], match[6], match[7]); 2326 if (!checkWeekday(match[1], parsedArray, config)) { 2327 return; 2328 } 2329 2330 config._a = parsedArray; 2331 config._tzm = calculateOffset(match[8], match[9], match[10]); 2332 2333 config._d = createUTCDate.apply(null, config._a); 2334 config._d.setUTCMinutes(config._d.getUTCMinutes() - config._tzm); 2335 2336 getParsingFlags(config).rfc2822 = true; 2337 } else { 2338 config._isValid = false; 2339 } 2340} 2341 2342// date from iso format or fallback 2343function configFromString(config) { 2344 var matched = aspNetJsonRegex.exec(config._i); 2345 2346 if (matched !== null) { 2347 config._d = new Date(+matched[1]); 2348 return; 2349 } 2350 2351 configFromISO(config); 2352 if (config._isValid === false) { 2353 delete config._isValid; 2354 } else { 2355 return; 2356 } 2357 2358 configFromRFC2822(config); 2359 if (config._isValid === false) { 2360 delete config._isValid; 2361 } else { 2362 return; 2363 } 2364 2365 // Final attempt, use Input Fallback 2366 hooks.createFromInputFallback(config); 2367} 2368 2369hooks.createFromInputFallback = deprecate( 2370 'value provided is not in a recognized RFC2822 or ISO format. moment construction falls back to js Date(), ' + 2371 'which is not reliable across all browsers and versions. Non RFC2822/ISO date formats are ' + 2372 'discouraged and will be removed in an upcoming major release. Please refer to ' + 2373 'http://momentjs.com/guides/#/warnings/js-date/ for more info.', 2374 function (config) { 2375 config._d = new Date(config._i + (config._useUTC ? ' UTC' : '')); 2376 } 2377); 2378 2379// constant that refers to the ISO standard 2380hooks.ISO_8601 = function () {}; 2381 2382// constant that refers to the RFC 2822 form 2383hooks.RFC_2822 = function () {}; 2384 2385// date from string and format string 2386function configFromStringAndFormat(config) { 2387 // TODO: Move this to another part of the creation flow to prevent circular deps 2388 if (config._f === hooks.ISO_8601) { 2389 configFromISO(config); 2390 return; 2391 } 2392 if (config._f === hooks.RFC_2822) { 2393 configFromRFC2822(config); 2394 return; 2395 } 2396 config._a = []; 2397 getParsingFlags(config).empty = true; 2398 2399 // This array is used to make a Date, either with `new Date` or `Date.UTC` 2400 var string = '' + config._i, 2401 i, parsedInput, tokens, token, skipped, 2402 stringLength = string.length, 2403 totalParsedInputLength = 0; 2404 2405 tokens = expandFormat(config._f, config._locale).match(formattingTokens) || []; 2406 2407 for (i = 0; i < tokens.length; i++) { 2408 token = tokens[i]; 2409 parsedInput = (string.match(getParseRegexForToken(token, config)) || [])[0]; 2410 // console.log('token', token, 'parsedInput', parsedInput, 2411 // 'regex', getParseRegexForToken(token, config)); 2412 if (parsedInput) { 2413 skipped = string.substr(0, string.indexOf(parsedInput)); 2414 if (skipped.length > 0) { 2415 getParsingFlags(config).unusedInput.push(skipped); 2416 } 2417 string = string.slice(string.indexOf(parsedInput) + parsedInput.length); 2418 totalParsedInputLength += parsedInput.length; 2419 } 2420 // don't parse if it's not a known token 2421 if (formatTokenFunctions[token]) { 2422 if (parsedInput) { 2423 getParsingFlags(config).empty = false; 2424 } 2425 else { 2426 getParsingFlags(config).unusedTokens.push(token); 2427 } 2428 addTimeToArrayFromToken(token, parsedInput, config); 2429 } 2430 else if (config._strict && !parsedInput) { 2431 getParsingFlags(config).unusedTokens.push(token); 2432 } 2433 } 2434 2435 // add remaining unparsed input length to the string 2436 getParsingFlags(config).charsLeftOver = stringLength - totalParsedInputLength; 2437 if (string.length > 0) { 2438 getParsingFlags(config).unusedInput.push(string); 2439 } 2440 2441 // clear _12h flag if hour is <= 12 2442 if (config._a[HOUR] <= 12 && 2443 getParsingFlags(config).bigHour === true && 2444 config._a[HOUR] > 0) { 2445 getParsingFlags(config).bigHour = undefined; 2446 } 2447 2448 getParsingFlags(config).parsedDateParts = config._a.slice(0); 2449 getParsingFlags(config).meridiem = config._meridiem; 2450 // handle meridiem 2451 config._a[HOUR] = meridiemFixWrap(config._locale, config._a[HOUR], config._meridiem); 2452 2453 configFromArray(config); 2454 checkOverflow(config); 2455} 2456 2457 2458function meridiemFixWrap (locale, hour, meridiem) { 2459 var isPm; 2460 2461 if (meridiem == null) { 2462 // nothing to do 2463 return hour; 2464 } 2465 if (locale.meridiemHour != null) { 2466 return locale.meridiemHour(hour, meridiem); 2467 } else if (locale.isPM != null) { 2468 // Fallback 2469 isPm = locale.isPM(meridiem); 2470 if (isPm && hour < 12) { 2471 hour += 12; 2472 } 2473 if (!isPm && hour === 12) { 2474 hour = 0; 2475 } 2476 return hour; 2477 } else { 2478 // this is not supposed to happen 2479 return hour; 2480 } 2481} 2482 2483// date from string and array of format strings 2484function configFromStringAndArray(config) { 2485 var tempConfig, 2486 bestMoment, 2487 2488 scoreToBeat, 2489 i, 2490 currentScore; 2491 2492 if (config._f.length === 0) { 2493 getParsingFlags(config).invalidFormat = true; 2494 config._d = new Date(NaN); 2495 return; 2496 } 2497 2498 for (i = 0; i < config._f.length; i++) { 2499 currentScore = 0; 2500 tempConfig = copyConfig({}, config); 2501 if (config._useUTC != null) { 2502 tempConfig._useUTC = config._useUTC; 2503 } 2504 tempConfig._f = config._f[i]; 2505 configFromStringAndFormat(tempConfig); 2506 2507 if (!isValid(tempConfig)) { 2508 continue; 2509 } 2510 2511 // if there is any input that was not parsed add a penalty for that format 2512 currentScore += getParsingFlags(tempConfig).charsLeftOver; 2513 2514 //or tokens 2515 currentScore += getParsingFlags(tempConfig).unusedTokens.length * 10; 2516 2517 getParsingFlags(tempConfig).score = currentScore; 2518 2519 if (scoreToBeat == null || currentScore < scoreToBeat) { 2520 scoreToBeat = currentScore; 2521 bestMoment = tempConfig; 2522 } 2523 } 2524 2525 extend(config, bestMoment || tempConfig); 2526} 2527 2528function configFromObject(config) { 2529 if (config._d) { 2530 return; 2531 } 2532 2533 var i = normalizeObjectUnits(config._i); 2534 config._a = map([i.year, i.month, i.day || i.date, i.hour, i.minute, i.second, i.millisecond], function (obj) { 2535 return obj && parseInt(obj, 10); 2536 }); 2537 2538 configFromArray(config); 2539} 2540 2541function createFromConfig (config) { 2542 var res = new Moment(checkOverflow(prepareConfig(config))); 2543 if (res._nextDay) { 2544 // Adding is smart enough around DST 2545 res.add(1, 'd'); 2546 res._nextDay = undefined; 2547 } 2548 2549 return res; 2550} 2551 2552function prepareConfig (config) { 2553 var input = config._i, 2554 format = config._f; 2555 2556 config._locale = config._locale || getLocale(config._l); 2557 2558 if (input === null || (format === undefined && input === '')) { 2559 return createInvalid({nullInput: true}); 2560 } 2561 2562 if (typeof input === 'string') { 2563 config._i = input = config._locale.preparse(input); 2564 } 2565 2566 if (isMoment(input)) { 2567 return new Moment(checkOverflow(input)); 2568 } else if (isDate(input)) { 2569 config._d = input; 2570 } else if (isArray(format)) { 2571 configFromStringAndArray(config); 2572 } else if (format) { 2573 configFromStringAndFormat(config); 2574 } else { 2575 configFromInput(config); 2576 } 2577 2578 if (!isValid(config)) { 2579 config._d = null; 2580 } 2581 2582 return config; 2583} 2584 2585function configFromInput(config) { 2586 var input = config._i; 2587 if (isUndefined(input)) { 2588 config._d = new Date(hooks.now()); 2589 } else if (isDate(input)) { 2590 config._d = new Date(input.valueOf()); 2591 } else if (typeof input === 'string') { 2592 configFromString(config); 2593 } else if (isArray(input)) { 2594 config._a = map(input.slice(0), function (obj) { 2595 return parseInt(obj, 10); 2596 }); 2597 configFromArray(config); 2598 } else if (isObject(input)) { 2599 configFromObject(config); 2600 } else if (isNumber(input)) { 2601 // from milliseconds 2602 config._d = new Date(input); 2603 } else { 2604 hooks.createFromInputFallback(config); 2605 } 2606} 2607 2608function createLocalOrUTC (input, format, locale, strict, isUTC) { 2609 var c = {}; 2610 2611 if (locale === true || locale === false) { 2612 strict = locale; 2613 locale = undefined; 2614 } 2615 2616 if ((isObject(input) && isObjectEmpty(input)) || 2617 (isArray(input) && input.length === 0)) { 2618 input = undefined; 2619 } 2620 // object construction must be done this way. 2621 // https://github.com/moment/moment/issues/1423 2622 c._isAMomentObject = true; 2623 c._useUTC = c._isUTC = isUTC; 2624 c._l = locale; 2625 c._i = input; 2626 c._f = format; 2627 c._strict = strict; 2628 2629 return createFromConfig(c); 2630} 2631 2632function createLocal (input, format, locale, strict) { 2633 return createLocalOrUTC(input, format, locale, strict, false); 2634} 2635 2636var prototypeMin = deprecate( 2637 'moment().min is deprecated, use moment.max instead. http://momentjs.com/guides/#/warnings/min-max/', 2638 function () { 2639 var other = createLocal.apply(null, arguments); 2640 if (this.isValid() && other.isValid()) { 2641 return other < this ? this : other; 2642 } else { 2643 return createInvalid(); 2644 } 2645 } 2646); 2647 2648var prototypeMax = deprecate( 2649 'moment().max is deprecated, use moment.min instead. http://momentjs.com/guides/#/warnings/min-max/', 2650 function () { 2651 var other = createLocal.apply(null, arguments); 2652 if (this.isValid() && other.isValid()) { 2653 return other > this ? this : other; 2654 } else { 2655 return createInvalid(); 2656 } 2657 } 2658); 2659 2660// Pick a moment m from moments so that m[fn](other) is true for all 2661// other. This relies on the function fn to be transitive. 2662// 2663// moments should either be an array of moment objects or an array, whose 2664// first element is an array of moment objects. 2665function pickBy(fn, moments) { 2666 var res, i; 2667 if (moments.length === 1 && isArray(moments[0])) { 2668 moments = moments[0]; 2669 } 2670 if (!moments.length) { 2671 return createLocal(); 2672 } 2673 res = moments[0]; 2674 for (i = 1; i < moments.length; ++i) { 2675 if (!moments[i].isValid() || moments[i][fn](res)) { 2676 res = moments[i]; 2677 } 2678 } 2679 return res; 2680} 2681 2682// TODO: Use [].sort instead? 2683function min () { 2684 var args = [].slice.call(arguments, 0); 2685 2686 return pickBy('isBefore', args); 2687} 2688 2689function max () { 2690 var args = [].slice.call(arguments, 0); 2691 2692 return pickBy('isAfter', args); 2693} 2694 2695var now = function () { 2696 return Date.now ? Date.now() : +(new Date()); 2697}; 2698 2699var ordering = ['year', 'quarter', 'month', 'week', 'day', 'hour', 'minute', 'second', 'millisecond']; 2700 2701function isDurationValid(m) { 2702 for (var key in m) { 2703 if (!(indexOf.call(ordering, key) !== -1 && (m[key] == null || !isNaN(m[key])))) { 2704 return false; 2705 } 2706 } 2707 2708 var unitHasDecimal = false; 2709 for (var i = 0; i < ordering.length; ++i) { 2710 if (m[ordering[i]]) { 2711 if (unitHasDecimal) { 2712 return false; // only allow non-integers for smallest unit 2713 } 2714 if (parseFloat(m[ordering[i]]) !== toInt(m[ordering[i]])) { 2715 unitHasDecimal = true; 2716 } 2717 } 2718 } 2719 2720 return true; 2721} 2722 2723function isValid$1() { 2724 return this._isValid; 2725} 2726 2727function createInvalid$1() { 2728 return createDuration(NaN); 2729} 2730 2731function Duration (duration) { 2732 var normalizedInput = normalizeObjectUnits(duration), 2733 years = normalizedInput.year || 0, 2734 quarters = normalizedInput.quarter || 0, 2735 months = normalizedInput.month || 0, 2736 weeks = normalizedInput.week || 0, 2737 days = normalizedInput.day || 0, 2738 hours = normalizedInput.hour || 0, 2739 minutes = normalizedInput.minute || 0, 2740 seconds = normalizedInput.second || 0, 2741 milliseconds = normalizedInput.millisecond || 0; 2742 2743 this._isValid = isDurationValid(normalizedInput); 2744 2745 // representation for dateAddRemove 2746 this._milliseconds = +milliseconds + 2747 seconds * 1e3 + // 1000 2748 minutes * 6e4 + // 1000 * 60 2749 hours * 1000 * 60 * 60; //using 1000 * 60 * 60 instead of 36e5 to avoid floating point rounding errors https://github.com/moment/moment/issues/2978 2750 // Because of dateAddRemove treats 24 hours as different from a 2751 // day when working around DST, we need to store them separately 2752 this._days = +days + 2753 weeks * 7; 2754 // It is impossible to translate months into days without knowing 2755 // which months you are are talking about, so we have to store 2756 // it separately. 2757 this._months = +months + 2758 quarters * 3 + 2759 years * 12; 2760 2761 this._data = {}; 2762 2763 this._locale = getLocale(); 2764 2765 this._bubble(); 2766} 2767 2768function isDuration (obj) { 2769 return obj instanceof Duration; 2770} 2771 2772function absRound (number) { 2773 if (number < 0) { 2774 return Math.round(-1 * number) * -1; 2775 } else { 2776 return Math.round(number); 2777 } 2778} 2779 2780// FORMATTING 2781 2782function offset (token, separator) { 2783 addFormatToken(token, 0, 0, function () { 2784 var offset = this.utcOffset(); 2785 var sign = '+'; 2786 if (offset < 0) { 2787 offset = -offset; 2788 sign = '-'; 2789 } 2790 return sign + zeroFill(~~(offset / 60), 2) + separator + zeroFill(~~(offset) % 60, 2); 2791 }); 2792} 2793 2794offset('Z', ':'); 2795offset('ZZ', ''); 2796 2797// PARSING 2798 2799addRegexToken('Z', matchShortOffset); 2800addRegexToken('ZZ', matchShortOffset); 2801addParseToken(['Z', 'ZZ'], function (input, array, config) { 2802 config._useUTC = true; 2803 config._tzm = offsetFromString(matchShortOffset, input); 2804}); 2805 2806// HELPERS 2807 2808// timezone chunker 2809// '+10:00' > ['10', '00'] 2810// '-1530' > ['-15', '30'] 2811var chunkOffset = /([\+\-]|\d\d)/gi; 2812 2813function offsetFromString(matcher, string) { 2814 var matches = (string || '').match(matcher); 2815 2816 if (matches === null) { 2817 return null; 2818 } 2819 2820 var chunk = matches[matches.length - 1] || []; 2821 var parts = (chunk + '').match(chunkOffset) || ['-', 0, 0]; 2822 var minutes = +(parts[1] * 60) + toInt(parts[2]); 2823 2824 return minutes === 0 ? 2825 0 : 2826 parts[0] === '+' ? minutes : -minutes; 2827} 2828 2829// Return a moment from input, that is local/utc/zone equivalent to model. 2830function cloneWithOffset(input, model) { 2831 var res, diff; 2832 if (model._isUTC) { 2833 res = model.clone(); 2834 diff = (isMoment(input) || isDate(input) ? input.valueOf() : createLocal(input).valueOf()) - res.valueOf(); 2835 // Use low-level api, because this fn is low-level api. 2836 res._d.setTime(res._d.valueOf() + diff); 2837 hooks.updateOffset(res, false); 2838 return res; 2839 } else { 2840 return createLocal(input).local(); 2841 } 2842} 2843 2844function getDateOffset (m) { 2845 // On Firefox.24 Date#getTimezoneOffset returns a floating point. 2846 // https://github.com/moment/moment/pull/1871 2847 return -Math.round(m._d.getTimezoneOffset() / 15) * 15; 2848} 2849 2850// HOOKS 2851 2852// This function will be called whenever a moment is mutated. 2853// It is intended to keep the offset in sync with the timezone. 2854hooks.updateOffset = function () {}; 2855 2856// MOMENTS 2857 2858// keepLocalTime = true means only change the timezone, without 2859// affecting the local hour. So 5:31:26 +0300 --[utcOffset(2, true)]--> 2860// 5:31:26 +0200 It is possible that 5:31:26 doesn't exist with offset 2861// +0200, so we adjust the time as needed, to be valid. 2862// 2863// Keeping the time actually adds/subtracts (one hour) 2864// from the actual represented time. That is why we call updateOffset 2865// a second time. In case it wants us to change the offset again 2866// _changeInProgress == true case, then we have to adjust, because 2867// there is no such time in the given timezone. 2868function getSetOffset (input, keepLocalTime, keepMinutes) { 2869 var offset = this._offset || 0, 2870 localAdjust; 2871 if (!this.isValid()) { 2872 return input != null ? this : NaN; 2873 } 2874 if (input != null) { 2875 if (typeof input === 'string') { 2876 input = offsetFromString(matchShortOffset, input); 2877 if (input === null) { 2878 return this; 2879 } 2880 } else if (Math.abs(input) < 16 && !keepMinutes) { 2881 input = input * 60; 2882 } 2883 if (!this._isUTC && keepLocalTime) { 2884 localAdjust = getDateOffset(this); 2885 } 2886 this._offset = input; 2887 this._isUTC = true; 2888 if (localAdjust != null) { 2889 this.add(localAdjust, 'm'); 2890 } 2891 if (offset !== input) { 2892 if (!keepLocalTime || this._changeInProgress) { 2893 addSubtract(this, createDuration(input - offset, 'm'), 1, false); 2894 } else if (!this._changeInProgress) { 2895 this._changeInProgress = true; 2896 hooks.updateOffset(this, true); 2897 this._changeInProgress = null; 2898 } 2899 } 2900 return this; 2901 } else { 2902 return this._isUTC ? offset : getDateOffset(this); 2903 } 2904} 2905 2906function getSetZone (input, keepLocalTime) { 2907 if (input != null) { 2908 if (typeof input !== 'string') { 2909 input = -input; 2910 } 2911 2912 this.utcOffset(input, keepLocalTime); 2913 2914 return this; 2915 } else { 2916 return -this.utcOffset(); 2917 } 2918} 2919 2920function setOffsetToUTC (keepLocalTime) { 2921 return this.utcOffset(0, keepLocalTime); 2922} 2923 2924function setOffsetToLocal (keepLocalTime) { 2925 if (this._isUTC) { 2926 this.utcOffset(0, keepLocalTime); 2927 this._isUTC = false; 2928 2929 if (keepLocalTime) { 2930 this.subtract(getDateOffset(this), 'm'); 2931 } 2932 } 2933 return this; 2934} 2935 2936function setOffsetToParsedOffset () { 2937 if (this._tzm != null) { 2938 this.utcOffset(this._tzm, false, true); 2939 } else if (typeof this._i === 'string') { 2940 var tZone = offsetFromString(matchOffset, this._i); 2941 if (tZone != null) { 2942 this.utcOffset(tZone); 2943 } 2944 else { 2945 this.utcOffset(0, true); 2946 } 2947 } 2948 return this; 2949} 2950 2951function hasAlignedHourOffset (input) { 2952 if (!this.isValid()) { 2953 return false; 2954 } 2955 input = input ? createLocal(input).utcOffset() : 0; 2956 2957 return (this.utcOffset() - input) % 60 === 0; 2958} 2959 2960function isDaylightSavingTime () { 2961 return ( 2962 this.utcOffset() > this.clone().month(0).utcOffset() || 2963 this.utcOffset() > this.clone().month(5).utcOffset() 2964 ); 2965} 2966 2967function isDaylightSavingTimeShifted () { 2968 if (!isUndefined(this._isDSTShifted)) { 2969 return this._isDSTShifted; 2970 } 2971 2972 var c = {}; 2973 2974 copyConfig(c, this); 2975 c = prepareConfig(c); 2976 2977 if (c._a) { 2978 var other = c._isUTC ? createUTC(c._a) : createLocal(c._a); 2979 this._isDSTShifted = this.isValid() && 2980 compareArrays(c._a, other.toArray()) > 0; 2981 } else { 2982 this._isDSTShifted = false; 2983 } 2984 2985 return this._isDSTShifted; 2986} 2987 2988function isLocal () { 2989 return this.isValid() ? !this._isUTC : false; 2990} 2991 2992function isUtcOffset () { 2993 return this.isValid() ? this._isUTC : false; 2994} 2995 2996function isUtc () { 2997 return this.isValid() ? this._isUTC && this._offset === 0 : false; 2998} 2999 3000// ASP.NET json date format regex 3001var aspNetRegex = /^(\-|\+)?(?:(\d*)[. ])?(\d+)\:(\d+)(?:\:(\d+)(\.\d*)?)?$/; 3002 3003// from http://docs.closure-library.googlecode.com/git/closure_goog_date_date.js.source.html 3004// somewhat more in line with 4.4.3.2 2004 spec, but allows decimal anywhere 3005// and further modified to allow for strings containing both week and day 3006var isoRegex = /^(-|\+)?P(?:([-+]?[0-9,.]*)Y)?(?:([-+]?[0-9,.]*)M)?(?:([-+]?[0-9,.]*)W)?(?:([-+]?[0-9,.]*)D)?(?:T(?:([-+]?[0-9,.]*)H)?(?:([-+]?[0-9,.]*)M)?(?:([-+]?[0-9,.]*)S)?)?$/; 3007 3008function createDuration (input, key) { 3009 var duration = input, 3010 // matching against regexp is expensive, do it on demand 3011 match = null, 3012 sign, 3013 ret, 3014 diffRes; 3015 3016 if (isDuration(input)) { 3017 duration = { 3018 ms : input._milliseconds, 3019 d : input._days, 3020 M : input._months 3021 }; 3022 } else if (isNumber(input)) { 3023 duration = {}; 3024 if (key) { 3025 duration[key] = input; 3026 } else { 3027 duration.milliseconds = input; 3028 } 3029 } else if (!!(match = aspNetRegex.exec(input))) { 3030 sign = (match[1] === '-') ? -1 : 1; 3031 duration = { 3032 y : 0, 3033 d : toInt(match[DATE]) * sign, 3034 h : toInt(match[HOUR]) * sign, 3035 m : toInt(match[MINUTE]) * sign, 3036 s : toInt(match[SECOND]) * sign, 3037 ms : toInt(absRound(match[MILLISECOND] * 1000)) * sign // the millisecond decimal point is included in the match 3038 }; 3039 } else if (!!(match = isoRegex.exec(input))) { 3040 sign = (match[1] === '-') ? -1 : (match[1] === '+') ? 1 : 1; 3041 duration = { 3042 y : parseIso(match[2], sign), 3043 M : parseIso(match[3], sign), 3044 w : parseIso(match[4], sign), 3045 d : parseIso(match[5], sign), 3046 h : parseIso(match[6], sign), 3047 m : parseIso(match[7], sign), 3048 s : parseIso(match[8], sign) 3049 }; 3050 } else if (duration == null) {// checks for null or undefined 3051 duration = {}; 3052 } else if (typeof duration === 'object' && ('from' in duration || 'to' in duration)) { 3053 diffRes = momentsDifference(createLocal(duration.from), createLocal(duration.to)); 3054 3055 duration = {}; 3056 duration.ms = diffRes.milliseconds; 3057 duration.M = diffRes.months; 3058 } 3059 3060 ret = new Duration(duration); 3061 3062 if (isDuration(input) && hasOwnProp(input, '_locale')) { 3063 ret._locale = input._locale; 3064 } 3065 3066 return ret; 3067} 3068 3069createDuration.fn = Duration.prototype; 3070createDuration.invalid = createInvalid$1; 3071 3072function parseIso (inp, sign) { 3073 // We'd normally use ~~inp for this, but unfortunately it also 3074 // converts floats to ints. 3075 // inp may be undefined, so careful calling replace on it. 3076 var res = inp && parseFloat(inp.replace(',', '.')); 3077 // apply sign while we're at it 3078 return (isNaN(res) ? 0 : res) * sign; 3079} 3080 3081function positiveMomentsDifference(base, other) { 3082 var res = {milliseconds: 0, months: 0}; 3083 3084 res.months = other.month() - base.month() + 3085 (other.year() - base.year()) * 12; 3086 if (base.clone().add(res.months, 'M').isAfter(other)) { 3087 --res.months; 3088 } 3089 3090 res.milliseconds = +other - +(base.clone().add(res.months, 'M')); 3091 3092 return res; 3093} 3094 3095function momentsDifference(base, other) { 3096 var res; 3097 if (!(base.isValid() && other.isValid())) { 3098 return {milliseconds: 0, months: 0}; 3099 } 3100 3101 other = cloneWithOffset(other, base); 3102 if (base.isBefore(other)) { 3103 res = positiveMomentsDifference(base, other); 3104 } else { 3105 res = positiveMomentsDifference(other, base); 3106 res.milliseconds = -res.milliseconds; 3107 res.months = -res.months; 3108 } 3109 3110 return res; 3111} 3112 3113// TODO: remove 'name' arg after deprecation is removed 3114function createAdder(direction, name) { 3115 return function (val, period) { 3116 var dur, tmp; 3117 //invert the arguments, but complain about it 3118 if (period !== null && !isNaN(+period)) { 3119 deprecateSimple(name, 'moment().' + name + '(period, number) is deprecated. Please use moment().' + name + '(number, period). ' + 3120 'See http://momentjs.com/guides/#/warnings/add-inverted-param/ for more info.'); 3121 tmp = val; val = period; period = tmp; 3122 } 3123 3124 val = typeof val === 'string' ? +val : val; 3125 dur = createDuration(val, period); 3126 addSubtract(this, dur, direction); 3127 return this; 3128 }; 3129} 3130 3131function addSubtract (mom, duration, isAdding, updateOffset) { 3132 var milliseconds = duration._milliseconds, 3133 days = absRound(duration._days), 3134 months = absRound(duration._months); 3135 3136 if (!mom.isValid()) { 3137 // No op 3138 return; 3139 } 3140 3141 updateOffset = updateOffset == null ? true : updateOffset; 3142 3143 if (months) { 3144 setMonth(mom, get(mom, 'Month') + months * isAdding); 3145 } 3146 if (days) { 3147 set$1(mom, 'Date', get(mom, 'Date') + days * isAdding); 3148 } 3149 if (milliseconds) { 3150 mom._d.setTime(mom._d.valueOf() + milliseconds * isAdding); 3151 } 3152 if (updateOffset) { 3153 hooks.updateOffset(mom, days || months); 3154 } 3155} 3156 3157var add = createAdder(1, 'add'); 3158var subtract = createAdder(-1, 'subtract'); 3159 3160function getCalendarFormat(myMoment, now) { 3161 var diff = myMoment.diff(now, 'days', true); 3162 return diff < -6 ? 'sameElse' : 3163 diff < -1 ? 'lastWeek' : 3164 diff < 0 ? 'lastDay' : 3165 diff < 1 ? 'sameDay' : 3166 diff < 2 ? 'nextDay' : 3167 diff < 7 ? 'nextWeek' : 'sameElse'; 3168} 3169 3170function calendar$1 (time, formats) { 3171 // We want to compare the start of today, vs this. 3172 // Getting start-of-today depends on whether we're local/utc/offset or not. 3173 var now = time || createLocal(), 3174 sod = cloneWithOffset(now, this).startOf('day'), 3175 format = hooks.calendarFormat(this, sod) || 'sameElse'; 3176 3177 var output = formats && (isFunction(formats[format]) ? formats[format].call(this, now) : formats[format]); 3178 3179 return this.format(output || this.localeData().calendar(format, this, createLocal(now))); 3180} 3181 3182function clone () { 3183 return new Moment(this); 3184} 3185 3186function isAfter (input, units) { 3187 var localInput = isMoment(input) ? input : createLocal(input); 3188 if (!(this.isValid() && localInput.isValid())) { 3189 return false; 3190 } 3191 units = normalizeUnits(!isUndefined(units) ? units : 'millisecond'); 3192 if (units === 'millisecond') { 3193 return this.valueOf() > localInput.valueOf(); 3194 } else { 3195 return localInput.valueOf() < this.clone().startOf(units).valueOf(); 3196 } 3197} 3198 3199function isBefore (input, units) { 3200 var localInput = isMoment(input) ? input : createLocal(input); 3201 if (!(this.isValid() && localInput.isValid())) { 3202 return false; 3203 } 3204 units = normalizeUnits(!isUndefined(units) ? units : 'millisecond'); 3205 if (units === 'millisecond') { 3206 return this.valueOf() < localInput.valueOf(); 3207 } else { 3208 return this.clone().endOf(units).valueOf() < localInput.valueOf(); 3209 } 3210} 3211 3212function isBetween (from, to, units, inclusivity) { 3213 inclusivity = inclusivity || '()'; 3214 return (inclusivity[0] === '(' ? this.isAfter(from, units) : !this.isBefore(from, units)) && 3215 (inclusivity[1] === ')' ? this.isBefore(to, units) : !this.isAfter(to, units)); 3216} 3217 3218function isSame (input, units) { 3219 var localInput = isMoment(input) ? input : createLocal(input), 3220 inputMs; 3221 if (!(this.isValid() && localInput.isValid())) { 3222 return false; 3223 } 3224 units = normalizeUnits(units || 'millisecond'); 3225 if (units === 'millisecond') { 3226 return this.valueOf() === localInput.valueOf(); 3227 } else { 3228 inputMs = localInput.valueOf(); 3229 return this.clone().startOf(units).valueOf() <= inputMs && inputMs <= this.clone().endOf(units).valueOf(); 3230 } 3231} 3232 3233function isSameOrAfter (input, units) { 3234 return this.isSame(input, units) || this.isAfter(input,units); 3235} 3236 3237function isSameOrBefore (input, units) { 3238 return this.isSame(input, units) || this.isBefore(input,units); 3239} 3240 3241function diff (input, units, asFloat) { 3242 var that, 3243 zoneDelta, 3244 delta, output; 3245 3246 if (!this.isValid()) { 3247 return NaN; 3248 } 3249 3250 that = cloneWithOffset(input, this); 3251 3252 if (!that.isValid()) { 3253 return NaN; 3254 } 3255 3256 zoneDelta = (that.utcOffset() - this.utcOffset()) * 6e4; 3257 3258 units = normalizeUnits(units); 3259 3260 switch (units) { 3261 case 'year': output = monthDiff(this, that) / 12; break; 3262 case 'month': output = monthDiff(this, that); break; 3263 case 'quarter': output = monthDiff(this, that) / 3; break; 3264 case 'second': output = (this - that) / 1e3; break; // 1000 3265 case 'minute': output = (this - that) / 6e4; break; // 1000 * 60 3266 case 'hour': output = (this - that) / 36e5; break; // 1000 * 60 * 60 3267 case 'day': output = (this - that - zoneDelta) / 864e5; break; // 1000 * 60 * 60 * 24, negate dst 3268 case 'week': output = (this - that - zoneDelta) / 6048e5; break; // 1000 * 60 * 60 * 24 * 7, negate dst 3269 default: output = this - that; 3270 } 3271 3272 return asFloat ? output : absFloor(output); 3273} 3274 3275function monthDiff (a, b) { 3276 // difference in months 3277 var wholeMonthDiff = ((b.year() - a.year()) * 12) + (b.month() - a.month()), 3278 // b is in (anchor - 1 month, anchor + 1 month) 3279 anchor = a.clone().add(wholeMonthDiff, 'months'), 3280 anchor2, adjust; 3281 3282 if (b - anchor < 0) { 3283 anchor2 = a.clone().add(wholeMonthDiff - 1, 'months'); 3284 // linear across the month 3285 adjust = (b - anchor) / (anchor - anchor2); 3286 } else { 3287 anchor2 = a.clone().add(wholeMonthDiff + 1, 'months'); 3288 // linear across the month 3289 adjust = (b - anchor) / (anchor2 - anchor); 3290 } 3291 3292 //check for negative zero, return zero if negative zero 3293 return -(wholeMonthDiff + adjust) || 0; 3294} 3295 3296hooks.defaultFormat = 'YYYY-MM-DDTHH:mm:ssZ'; 3297hooks.defaultFormatUtc = 'YYYY-MM-DDTHH:mm:ss[Z]'; 3298 3299function toString () { 3300 return this.clone().locale('en').format('ddd MMM DD YYYY HH:mm:ss [GMT]ZZ'); 3301} 3302 3303function toISOString(keepOffset) { 3304 if (!this.isValid()) { 3305 return null; 3306 } 3307 var utc = keepOffset !== true; 3308 var m = utc ? this.clone().utc() : this; 3309 if (m.year() < 0 || m.year() > 9999) { 3310 return formatMoment(m, utc ? 'YYYYYY-MM-DD[T]HH:mm:ss.SSS[Z]' : 'YYYYYY-MM-DD[T]HH:mm:ss.SSSZ'); 3311 } 3312 if (isFunction(Date.prototype.toISOString)) { 3313 // native implementation is ~50x faster, use it when we can 3314 if (utc) { 3315 return this.toDate().toISOString(); 3316 } else { 3317 return new Date(this._d.valueOf()).toISOString().replace('Z', formatMoment(m, 'Z')); 3318 } 3319 } 3320 return formatMoment(m, utc ? 'YYYY-MM-DD[T]HH:mm:ss.SSS[Z]' : 'YYYY-MM-DD[T]HH:mm:ss.SSSZ'); 3321} 3322 3323/** 3324 * Return a human readable representation of a moment that can 3325 * also be evaluated to get a new moment which is the same 3326 * 3327 * @link https://nodejs.org/dist/latest/docs/api/util.html#util_custom_inspect_function_on_objects 3328 */ 3329function inspect () { 3330 if (!this.isValid()) { 3331 return 'moment.invalid(/* ' + this._i + ' */)'; 3332 } 3333 var func = 'moment'; 3334 var zone = ''; 3335 if (!this.isLocal()) { 3336 func = this.utcOffset() === 0 ? 'moment.utc' : 'moment.parseZone'; 3337 zone = 'Z'; 3338 } 3339 var prefix = '[' + func + '("]'; 3340 var year = (0 <= this.year() && this.year() <= 9999) ? 'YYYY' : 'YYYYYY'; 3341 var datetime = '-MM-DD[T]HH:mm:ss.SSS'; 3342 var suffix = zone + '[")]'; 3343 3344 return this.format(prefix + year + datetime + suffix); 3345} 3346 3347function format (inputString) { 3348 if (!inputString) { 3349 inputString = this.isUtc() ? hooks.defaultFormatUtc : hooks.defaultFormat; 3350 } 3351 var output = formatMoment(this, inputString); 3352 return this.localeData().postformat(output); 3353} 3354 3355function from (time, withoutSuffix) { 3356 if (this.isValid() && 3357 ((isMoment(time) && time.isValid()) || 3358 createLocal(time).isValid())) { 3359 return createDuration({to: this, from: time}).locale(this.locale()).humanize(!withoutSuffix); 3360 } else { 3361 return this.localeData().invalidDate(); 3362 } 3363} 3364 3365function fromNow (withoutSuffix) { 3366 return this.from(createLocal(), withoutSuffix); 3367} 3368 3369function to (time, withoutSuffix) { 3370 if (this.isValid() && 3371 ((isMoment(time) && time.isValid()) || 3372 createLocal(time).isValid())) { 3373 return createDuration({from: this, to: time}).locale(this.locale()).humanize(!withoutSuffix); 3374 } else { 3375 return this.localeData().invalidDate(); 3376 } 3377} 3378 3379function toNow (withoutSuffix) { 3380 return this.to(createLocal(), withoutSuffix); 3381} 3382 3383// If passed a locale key, it will set the locale for this 3384// instance. Otherwise, it will return the locale configuration 3385// variables for this instance. 3386function locale (key) { 3387 var newLocaleData; 3388 3389 if (key === undefined) { 3390 return this._locale._abbr; 3391 } else { 3392 newLocaleData = getLocale(key); 3393 if (newLocaleData != null) { 3394 this._locale = newLocaleData; 3395 } 3396 return this; 3397 } 3398} 3399 3400var lang = deprecate( 3401 'moment().lang() is deprecated. Instead, use moment().localeData() to get the language configuration. Use moment().locale() to change languages.', 3402 function (key) { 3403 if (key === undefined) { 3404 return this.localeData(); 3405 } else { 3406 return this.locale(key); 3407 } 3408 } 3409); 3410 3411function localeData () { 3412 return this._locale; 3413} 3414 3415function startOf (units) { 3416 units = normalizeUnits(units); 3417 // the following switch intentionally omits break keywords 3418 // to utilize falling through the cases. 3419 switch (units) { 3420 case 'year': 3421 this.month(0); 3422 /* falls through */ 3423 case 'quarter': 3424 case 'month': 3425 this.date(1); 3426 /* falls through */ 3427 case 'week': 3428 case 'isoWeek': 3429 case 'day': 3430 case 'date': 3431 this.hours(0); 3432 /* falls through */ 3433 case 'hour': 3434 this.minutes(0); 3435 /* falls through */ 3436 case 'minute': 3437 this.seconds(0); 3438 /* falls through */ 3439 case 'second': 3440 this.milliseconds(0); 3441 } 3442 3443 // weeks are a special case 3444 if (units === 'week') { 3445 this.weekday(0); 3446 } 3447 if (units === 'isoWeek') { 3448 this.isoWeekday(1); 3449 } 3450 3451 // quarters are also special 3452 if (units === 'quarter') { 3453 this.month(Math.floor(this.month() / 3) * 3); 3454 } 3455 3456 return this; 3457} 3458 3459function endOf (units) { 3460 units = normalizeUnits(units); 3461 if (units === undefined || units === 'millisecond') { 3462 return this; 3463 } 3464 3465 // 'date' is an alias for 'day', so it should be considered as such. 3466 if (units === 'date') { 3467 units = 'day'; 3468 } 3469 3470 return this.startOf(units).add(1, (units === 'isoWeek' ? 'week' : units)).subtract(1, 'ms'); 3471} 3472 3473function valueOf () { 3474 return this._d.valueOf() - ((this._offset || 0) * 60000); 3475} 3476 3477function unix () { 3478 return Math.floor(this.valueOf() / 1000); 3479} 3480 3481function toDate () { 3482 return new Date(this.valueOf()); 3483} 3484 3485function toArray () { 3486 var m = this; 3487 return [m.year(), m.month(), m.date(), m.hour(), m.minute(), m.second(), m.millisecond()]; 3488} 3489 3490function toObject () { 3491 var m = this; 3492 return { 3493 years: m.year(), 3494 months: m.month(), 3495 date: m.date(), 3496 hours: m.hours(), 3497 minutes: m.minutes(), 3498 seconds: m.seconds(), 3499 milliseconds: m.milliseconds() 3500 }; 3501} 3502 3503function toJSON () { 3504 // new Date(NaN).toJSON() === null 3505 return this.isValid() ? this.toISOString() : null; 3506} 3507 3508function isValid$2 () { 3509 return isValid(this); 3510} 3511 3512function parsingFlags () { 3513 return extend({}, getParsingFlags(this)); 3514} 3515 3516function invalidAt () { 3517 return getParsingFlags(this).overflow; 3518} 3519 3520function creationData() { 3521 return { 3522 input: this._i, 3523 format: this._f, 3524 locale: this._locale, 3525 isUTC: this._isUTC, 3526 strict: this._strict 3527 }; 3528} 3529 3530// FORMATTING 3531 3532addFormatToken(0, ['gg', 2], 0, function () { 3533 return this.weekYear() % 100; 3534}); 3535 3536addFormatToken(0, ['GG', 2], 0, function () { 3537 return this.isoWeekYear() % 100; 3538}); 3539 3540function addWeekYearFormatToken (token, getter) { 3541 addFormatToken(0, [token, token.length], 0, getter); 3542} 3543 3544addWeekYearFormatToken('gggg', 'weekYear'); 3545addWeekYearFormatToken('ggggg', 'weekYear'); 3546addWeekYearFormatToken('GGGG', 'isoWeekYear'); 3547addWeekYearFormatToken('GGGGG', 'isoWeekYear'); 3548 3549// ALIASES 3550 3551addUnitAlias('weekYear', 'gg'); 3552addUnitAlias('isoWeekYear', 'GG'); 3553 3554// PRIORITY 3555 3556addUnitPriority('weekYear', 1); 3557addUnitPriority('isoWeekYear', 1); 3558 3559 3560// PARSING 3561 3562addRegexToken('G', matchSigned); 3563addRegexToken('g', matchSigned); 3564addRegexToken('GG', match1to2, match2); 3565addRegexToken('gg', match1to2, match2); 3566addRegexToken('GGGG', match1to4, match4); 3567addRegexToken('gggg', match1to4, match4); 3568addRegexToken('GGGGG', match1to6, match6); 3569addRegexToken('ggggg', match1to6, match6); 3570 3571addWeekParseToken(['gggg', 'ggggg', 'GGGG', 'GGGGG'], function (input, week, config, token) { 3572 week[token.substr(0, 2)] = toInt(input); 3573}); 3574 3575addWeekParseToken(['gg', 'GG'], function (input, week, config, token) { 3576 week[token] = hooks.parseTwoDigitYear(input); 3577}); 3578 3579// MOMENTS 3580 3581function getSetWeekYear (input) { 3582 return getSetWeekYearHelper.call(this, 3583 input, 3584 this.week(), 3585 this.weekday(), 3586 this.localeData()._week.dow, 3587 this.localeData()._week.doy); 3588} 3589 3590function getSetISOWeekYear (input) { 3591 return getSetWeekYearHelper.call(this, 3592 input, this.isoWeek(), this.isoWeekday(), 1, 4); 3593} 3594 3595function getISOWeeksInYear () { 3596 return weeksInYear(this.year(), 1, 4); 3597} 3598 3599function getWeeksInYear () { 3600 var weekInfo = this.localeData()._week; 3601 return weeksInYear(this.year(), weekInfo.dow, weekInfo.doy); 3602} 3603 3604function getSetWeekYearHelper(input, week, weekday, dow, doy) { 3605 var weeksTarget; 3606 if (input == null) { 3607 return weekOfYear(this, dow, doy).year; 3608 } else { 3609 weeksTarget = weeksInYear(input, dow, doy); 3610 if (week > weeksTarget) { 3611 week = weeksTarget; 3612 } 3613 return setWeekAll.call(this, input, week, weekday, dow, doy); 3614 } 3615} 3616 3617function setWeekAll(weekYear, week, weekday, dow, doy) { 3618 var dayOfYearData = dayOfYearFromWeeks(weekYear, week, weekday, dow, doy), 3619 date = createUTCDate(dayOfYearData.year, 0, dayOfYearData.dayOfYear); 3620 3621 this.year(date.getUTCFullYear()); 3622 this.month(date.getUTCMonth()); 3623 this.date(date.getUTCDate()); 3624 return this; 3625} 3626 3627// FORMATTING 3628 3629addFormatToken('Q', 0, 'Qo', 'quarter'); 3630 3631// ALIASES 3632 3633addUnitAlias('quarter', 'Q'); 3634 3635// PRIORITY 3636 3637addUnitPriority('quarter', 7); 3638 3639// PARSING 3640 3641addRegexToken('Q', match1); 3642addParseToken('Q', function (input, array) { 3643 array[MONTH] = (toInt(input) - 1) * 3; 3644}); 3645 3646// MOMENTS 3647 3648function getSetQuarter (input) { 3649 return input == null ? Math.ceil((this.month() + 1) / 3) : this.month((input - 1) * 3 + this.month() % 3); 3650} 3651 3652// FORMATTING 3653 3654addFormatToken('D', ['DD', 2], 'Do', 'date'); 3655 3656// ALIASES 3657 3658addUnitAlias('date', 'D'); 3659 3660// PRIOROITY 3661addUnitPriority('date', 9); 3662 3663// PARSING 3664 3665addRegexToken('D', match1to2); 3666addRegexToken('DD', match1to2, match2); 3667addRegexToken('Do', function (isStrict, locale) { 3668 // TODO: Remove "ordinalParse" fallback in next major release. 3669 return isStrict ? 3670 (locale._dayOfMonthOrdinalParse || locale._ordinalParse) : 3671 locale._dayOfMonthOrdinalParseLenient; 3672}); 3673 3674addParseToken(['D', 'DD'], DATE); 3675addParseToken('Do', function (input, array) { 3676 array[DATE] = toInt(input.match(match1to2)[0]); 3677}); 3678 3679// MOMENTS 3680 3681var getSetDayOfMonth = makeGetSet('Date', true); 3682 3683// FORMATTING 3684 3685addFormatToken('DDD', ['DDDD', 3], 'DDDo', 'dayOfYear'); 3686 3687// ALIASES 3688 3689addUnitAlias('dayOfYear', 'DDD'); 3690 3691// PRIORITY 3692addUnitPriority('dayOfYear', 4); 3693 3694// PARSING 3695 3696addRegexToken('DDD', match1to3); 3697addRegexToken('DDDD', match3); 3698addParseToken(['DDD', 'DDDD'], function (input, array, config) { 3699 config._dayOfYear = toInt(input); 3700}); 3701 3702// HELPERS 3703 3704// MOMENTS 3705 3706function getSetDayOfYear (input) { 3707 var dayOfYear = Math.round((this.clone().startOf('day') - this.clone().startOf('year')) / 864e5) + 1; 3708 return input == null ? dayOfYear : this.add((input - dayOfYear), 'd'); 3709} 3710 3711// FORMATTING 3712 3713addFormatToken('m', ['mm', 2], 0, 'minute'); 3714 3715// ALIASES 3716 3717addUnitAlias('minute', 'm'); 3718 3719// PRIORITY 3720 3721addUnitPriority('minute', 14); 3722 3723// PARSING 3724 3725addRegexToken('m', match1to2); 3726addRegexToken('mm', match1to2, match2); 3727addParseToken(['m', 'mm'], MINUTE); 3728 3729// MOMENTS 3730 3731var getSetMinute = makeGetSet('Minutes', false); 3732 3733// FORMATTING 3734 3735addFormatToken('s', ['ss', 2], 0, 'second'); 3736 3737// ALIASES 3738 3739addUnitAlias('second', 's'); 3740 3741// PRIORITY 3742 3743addUnitPriority('second', 15); 3744 3745// PARSING 3746 3747addRegexToken('s', match1to2); 3748addRegexToken('ss', match1to2, match2); 3749addParseToken(['s', 'ss'], SECOND); 3750 3751// MOMENTS 3752 3753var getSetSecond = makeGetSet('Seconds', false); 3754 3755// FORMATTING 3756 3757addFormatToken('S', 0, 0, function () { 3758 return ~~(this.millisecond() / 100); 3759}); 3760 3761addFormatToken(0, ['SS', 2], 0, function () { 3762 return ~~(this.millisecond() / 10); 3763}); 3764 3765addFormatToken(0, ['SSS', 3], 0, 'millisecond'); 3766addFormatToken(0, ['SSSS', 4], 0, function () { 3767 return this.millisecond() * 10; 3768}); 3769addFormatToken(0, ['SSSSS', 5], 0, function () { 3770 return this.millisecond() * 100; 3771}); 3772addFormatToken(0, ['SSSSSS', 6], 0, function () { 3773 return this.millisecond() * 1000; 3774}); 3775addFormatToken(0, ['SSSSSSS', 7], 0, function () { 3776 return this.millisecond() * 10000; 3777}); 3778addFormatToken(0, ['SSSSSSSS', 8], 0, function () { 3779 return this.millisecond() * 100000; 3780}); 3781addFormatToken(0, ['SSSSSSSSS', 9], 0, function () { 3782 return this.millisecond() * 1000000; 3783}); 3784 3785 3786// ALIASES 3787 3788addUnitAlias('millisecond', 'ms'); 3789 3790// PRIORITY 3791 3792addUnitPriority('millisecond', 16); 3793 3794// PARSING 3795 3796addRegexToken('S', match1to3, match1); 3797addRegexToken('SS', match1to3, match2); 3798addRegexToken('SSS', match1to3, match3); 3799 3800var token; 3801for (token = 'SSSS'; token.length <= 9; token += 'S') { 3802 addRegexToken(token, matchUnsigned); 3803} 3804 3805function parseMs(input, array) { 3806 array[MILLISECOND] = toInt(('0.' + input) * 1000); 3807} 3808 3809for (token = 'S'; token.length <= 9; token += 'S') { 3810 addParseToken(token, parseMs); 3811} 3812// MOMENTS 3813 3814var getSetMillisecond = makeGetSet('Milliseconds', false); 3815 3816// FORMATTING 3817 3818addFormatToken('z', 0, 0, 'zoneAbbr'); 3819addFormatToken('zz', 0, 0, 'zoneName'); 3820 3821// MOMENTS 3822 3823function getZoneAbbr () { 3824 return this._isUTC ? 'UTC' : ''; 3825} 3826 3827function getZoneName () { 3828 return this._isUTC ? 'Coordinated Universal Time' : ''; 3829} 3830 3831var proto = Moment.prototype; 3832 3833proto.add = add; 3834proto.calendar = calendar$1; 3835proto.clone = clone; 3836proto.diff = diff; 3837proto.endOf = endOf; 3838proto.format = format; 3839proto.from = from; 3840proto.fromNow = fromNow; 3841proto.to = to; 3842proto.toNow = toNow; 3843proto.get = stringGet; 3844proto.invalidAt = invalidAt; 3845proto.isAfter = isAfter; 3846proto.isBefore = isBefore; 3847proto.isBetween = isBetween; 3848proto.isSame = isSame; 3849proto.isSameOrAfter = isSameOrAfter; 3850proto.isSameOrBefore = isSameOrBefore; 3851proto.isValid = isValid$2; 3852proto.lang = lang; 3853proto.locale = locale; 3854proto.localeData = localeData; 3855proto.max = prototypeMax; 3856proto.min = prototypeMin; 3857proto.parsingFlags = parsingFlags; 3858proto.set = stringSet; 3859proto.startOf = startOf; 3860proto.subtract = subtract; 3861proto.toArray = toArray; 3862proto.toObject = toObject; 3863proto.toDate = toDate; 3864proto.toISOString = toISOString; 3865proto.inspect = inspect; 3866proto.toJSON = toJSON; 3867proto.toString = toString; 3868proto.unix = unix; 3869proto.valueOf = valueOf; 3870proto.creationData = creationData; 3871 3872// Year 3873proto.year = getSetYear; 3874proto.isLeapYear = getIsLeapYear; 3875 3876// Week Year 3877proto.weekYear = getSetWeekYear; 3878proto.isoWeekYear = getSetISOWeekYear; 3879 3880// Quarter 3881proto.quarter = proto.quarters = getSetQuarter; 3882 3883// Month 3884proto.month = getSetMonth; 3885proto.daysInMonth = getDaysInMonth; 3886 3887// Week 3888proto.week = proto.weeks = getSetWeek; 3889proto.isoWeek = proto.isoWeeks = getSetISOWeek; 3890proto.weeksInYear = getWeeksInYear; 3891proto.isoWeeksInYear = getISOWeeksInYear; 3892 3893// Day 3894proto.date = getSetDayOfMonth; 3895proto.day = proto.days = getSetDayOfWeek; 3896proto.weekday = getSetLocaleDayOfWeek; 3897proto.isoWeekday = getSetISODayOfWeek; 3898proto.dayOfYear = getSetDayOfYear; 3899 3900// Hour 3901proto.hour = proto.hours = getSetHour; 3902 3903// Minute 3904proto.minute = proto.minutes = getSetMinute; 3905 3906// Second 3907proto.second = proto.seconds = getSetSecond; 3908 3909// Millisecond 3910proto.millisecond = proto.milliseconds = getSetMillisecond; 3911 3912// Offset 3913proto.utcOffset = getSetOffset; 3914proto.utc = setOffsetToUTC; 3915proto.local = setOffsetToLocal; 3916proto.parseZone = setOffsetToParsedOffset; 3917proto.hasAlignedHourOffset = hasAlignedHourOffset; 3918proto.isDST = isDaylightSavingTime; 3919proto.isLocal = isLocal; 3920proto.isUtcOffset = isUtcOffset; 3921proto.isUtc = isUtc; 3922proto.isUTC = isUtc; 3923 3924// Timezone 3925proto.zoneAbbr = getZoneAbbr; 3926proto.zoneName = getZoneName; 3927 3928// Deprecations 3929proto.dates = deprecate('dates accessor is deprecated. Use date instead.', getSetDayOfMonth); 3930proto.months = deprecate('months accessor is deprecated. Use month instead', getSetMonth); 3931proto.years = deprecate('years accessor is deprecated. Use year instead', getSetYear); 3932proto.zone = deprecate('moment().zone is deprecated, use moment().utcOffset instead. http://momentjs.com/guides/#/warnings/zone/', getSetZone); 3933proto.isDSTShifted = deprecate('isDSTShifted is deprecated. See http://momentjs.com/guides/#/warnings/dst-shifted/ for more information', isDaylightSavingTimeShifted); 3934 3935function createUnix (input) { 3936 return createLocal(input * 1000); 3937} 3938 3939function createInZone () { 3940 return createLocal.apply(null, arguments).parseZone(); 3941} 3942 3943function preParsePostFormat (string) { 3944 return string; 3945} 3946 3947var proto$1 = Locale.prototype; 3948 3949proto$1.calendar = calendar; 3950proto$1.longDateFormat = longDateFormat; 3951proto$1.invalidDate = invalidDate; 3952proto$1.ordinal = ordinal; 3953proto$1.preparse = preParsePostFormat; 3954proto$1.postformat = preParsePostFormat; 3955proto$1.relativeTime = relativeTime; 3956proto$1.pastFuture = pastFuture; 3957proto$1.set = set; 3958 3959// Month 3960proto$1.months = localeMonths; 3961proto$1.monthsShort = localeMonthsShort; 3962proto$1.monthsParse = localeMonthsParse; 3963proto$1.monthsRegex = monthsRegex; 3964proto$1.monthsShortRegex = monthsShortRegex; 3965 3966// Week 3967proto$1.week = localeWeek; 3968proto$1.firstDayOfYear = localeFirstDayOfYear; 3969proto$1.firstDayOfWeek = localeFirstDayOfWeek; 3970 3971// Day of Week 3972proto$1.weekdays = localeWeekdays; 3973proto$1.weekdaysMin = localeWeekdaysMin; 3974proto$1.weekdaysShort = localeWeekdaysShort; 3975proto$1.weekdaysParse = localeWeekdaysParse; 3976 3977proto$1.weekdaysRegex = weekdaysRegex; 3978proto$1.weekdaysShortRegex = weekdaysShortRegex; 3979proto$1.weekdaysMinRegex = weekdaysMinRegex; 3980 3981// Hours 3982proto$1.isPM = localeIsPM; 3983proto$1.meridiem = localeMeridiem; 3984 3985function get$1 (format, index, field, setter) { 3986 var locale = getLocale(); 3987 var utc = createUTC().set(setter, index); 3988 return locale[field](utc, format); 3989} 3990 3991function listMonthsImpl (format, index, field) { 3992 if (isNumber(format)) { 3993 index = format; 3994 format = undefined; 3995 } 3996 3997 format = format || ''; 3998 3999 if (index != null) { 4000 return get$1(format, index, field, 'month'); 4001 } 4002 4003 var i; 4004 var out = []; 4005 for (i = 0; i < 12; i++) { 4006 out[i] = get$1(format, i, field, 'month'); 4007 } 4008 return out; 4009} 4010 4011// () 4012// (5) 4013// (fmt, 5) 4014// (fmt) 4015// (true) 4016// (true, 5) 4017// (true, fmt, 5) 4018// (true, fmt) 4019function listWeekdaysImpl (localeSorted, format, index, field) { 4020 if (typeof localeSorted === 'boolean') { 4021 if (isNumber(format)) { 4022 index = format; 4023 format = undefined; 4024 } 4025 4026 format = format || ''; 4027 } else { 4028 format = localeSorted; 4029 index = format; 4030 localeSorted = false; 4031 4032 if (isNumber(format)) { 4033 index = format; 4034 format = undefined; 4035 } 4036 4037 format = format || ''; 4038 } 4039 4040 var locale = getLocale(), 4041 shift = localeSorted ? locale._week.dow : 0; 4042 4043 if (index != null) { 4044 return get$1(format, (index + shift) % 7, field, 'day'); 4045 } 4046 4047 var i; 4048 var out = []; 4049 for (i = 0; i < 7; i++) { 4050 out[i] = get$1(format, (i + shift) % 7, field, 'day'); 4051 } 4052 return out; 4053} 4054 4055function listMonths (format, index) { 4056 return listMonthsImpl(format, index, 'months'); 4057} 4058 4059function listMonthsShort (format, index) { 4060 return listMonthsImpl(format, index, 'monthsShort'); 4061} 4062 4063function listWeekdays (localeSorted, format, index) { 4064 return listWeekdaysImpl(localeSorted, format, index, 'weekdays'); 4065} 4066 4067function listWeekdaysShort (localeSorted, format, index) { 4068 return listWeekdaysImpl(localeSorted, format, index, 'weekdaysShort'); 4069} 4070 4071function listWeekdaysMin (localeSorted, format, index) { 4072 return listWeekdaysImpl(localeSorted, format, index, 'weekdaysMin'); 4073} 4074 4075getSetGlobalLocale('en', { 4076 dayOfMonthOrdinalParse: /\d{1,2}(th|st|nd|rd)/, 4077 ordinal : function (number) { 4078 var b = number % 10, 4079 output = (toInt(number % 100 / 10) === 1) ? 'th' : 4080 (b === 1) ? 'st' : 4081 (b === 2) ? 'nd' : 4082 (b === 3) ? 'rd' : 'th'; 4083 return number + output; 4084 } 4085}); 4086 4087// Side effect imports 4088hooks.lang = deprecate('moment.lang is deprecated. Use moment.locale instead.', getSetGlobalLocale); 4089hooks.langData = deprecate('moment.langData is deprecated. Use moment.localeData instead.', getLocale); 4090 4091var mathAbs = Math.abs; 4092 4093function abs () { 4094 var data = this._data; 4095 4096 this._milliseconds = mathAbs(this._milliseconds); 4097 this._days = mathAbs(this._days); 4098 this._months = mathAbs(this._months); 4099 4100 data.milliseconds = mathAbs(data.milliseconds); 4101 data.seconds = mathAbs(data.seconds); 4102 data.minutes = mathAbs(data.minutes); 4103 data.hours = mathAbs(data.hours); 4104 data.months = mathAbs(data.months); 4105 data.years = mathAbs(data.years); 4106 4107 return this; 4108} 4109 4110function addSubtract$1 (duration, input, value, direction) { 4111 var other = createDuration(input, value); 4112 4113 duration._milliseconds += direction * other._milliseconds; 4114 duration._days += direction * other._days; 4115 duration._months += direction * other._months; 4116 4117 return duration._bubble(); 4118} 4119 4120// supports only 2.0-style add(1, 's') or add(duration) 4121function add$1 (input, value) { 4122 return addSubtract$1(this, input, value, 1); 4123} 4124 4125// supports only 2.0-style subtract(1, 's') or subtract(duration) 4126function subtract$1 (input, value) { 4127 return addSubtract$1(this, input, value, -1); 4128} 4129 4130function absCeil (number) { 4131 if (number < 0) { 4132 return Math.floor(number); 4133 } else { 4134 return Math.ceil(number); 4135 } 4136} 4137 4138function bubble () { 4139 var milliseconds = this._milliseconds; 4140 var days = this._days; 4141 var months = this._months; 4142 var data = this._data; 4143 var seconds, minutes, hours, years, monthsFromDays; 4144 4145 // if we have a mix of positive and negative values, bubble down first 4146 // check: https://github.com/moment/moment/issues/2166 4147 if (!((milliseconds >= 0 && days >= 0 && months >= 0) || 4148 (milliseconds <= 0 && days <= 0 && months <= 0))) { 4149 milliseconds += absCeil(monthsToDays(months) + days) * 864e5; 4150 days = 0; 4151 months = 0; 4152 } 4153 4154 // The following code bubbles up values, see the tests for 4155 // examples of what that means. 4156 data.milliseconds = milliseconds % 1000; 4157 4158 seconds = absFloor(milliseconds / 1000); 4159 data.seconds = seconds % 60; 4160 4161 minutes = absFloor(seconds / 60); 4162 data.minutes = minutes % 60; 4163 4164 hours = absFloor(minutes / 60); 4165 data.hours = hours % 24; 4166 4167 days += absFloor(hours / 24); 4168 4169 // convert days to months 4170 monthsFromDays = absFloor(daysToMonths(days)); 4171 months += monthsFromDays; 4172 days -= absCeil(monthsToDays(monthsFromDays)); 4173 4174 // 12 months -> 1 year 4175 years = absFloor(months / 12); 4176 months %= 12; 4177 4178 data.days = days; 4179 data.months = months; 4180 data.years = years; 4181 4182 return this; 4183} 4184 4185function daysToMonths (days) { 4186 // 400 years have 146097 days (taking into account leap year rules) 4187 // 400 years have 12 months === 4800 4188 return days * 4800 / 146097; 4189} 4190 4191function monthsToDays (months) { 4192 // the reverse of daysToMonths 4193 return months * 146097 / 4800; 4194} 4195 4196function as (units) { 4197 if (!this.isValid()) { 4198 return NaN; 4199 } 4200 var days; 4201 var months; 4202 var milliseconds = this._milliseconds; 4203 4204 units = normalizeUnits(units); 4205 4206 if (units === 'month' || units === 'year') { 4207 days = this._days + milliseconds / 864e5; 4208 months = this._months + daysToMonths(days); 4209 return units === 'month' ? months : months / 12; 4210 } else { 4211 // handle milliseconds separately because of floating point math errors (issue #1867) 4212 days = this._days + Math.round(monthsToDays(this._months)); 4213 switch (units) { 4214 case 'week' : return days / 7 + milliseconds / 6048e5; 4215 case 'day' : return days + milliseconds / 864e5; 4216 case 'hour' : return days * 24 + milliseconds / 36e5; 4217 case 'minute' : return days * 1440 + milliseconds / 6e4; 4218 case 'second' : return days * 86400 + milliseconds / 1000; 4219 // Math.floor prevents floating point math errors here 4220 case 'millisecond': return Math.floor(days * 864e5) + milliseconds; 4221 default: throw new Error('Unknown unit ' + units); 4222 } 4223 } 4224} 4225 4226// TODO: Use this.as('ms')? 4227function valueOf$1 () { 4228 if (!this.isValid()) { 4229 return NaN; 4230 } 4231 return ( 4232 this._milliseconds + 4233 this._days * 864e5 + 4234 (this._months % 12) * 2592e6 + 4235 toInt(this._months / 12) * 31536e6 4236 ); 4237} 4238 4239function makeAs (alias) { 4240 return function () { 4241 return this.as(alias); 4242 }; 4243} 4244 4245var asMilliseconds = makeAs('ms'); 4246var asSeconds = makeAs('s'); 4247var asMinutes = makeAs('m'); 4248var asHours = makeAs('h'); 4249var asDays = makeAs('d'); 4250var asWeeks = makeAs('w'); 4251var asMonths = makeAs('M'); 4252var asYears = makeAs('y'); 4253 4254function clone$1 () { 4255 return createDuration(this); 4256} 4257 4258function get$2 (units) { 4259 units = normalizeUnits(units); 4260 return this.isValid() ? this[units + 's']() : NaN; 4261} 4262 4263function makeGetter(name) { 4264 return function () { 4265 return this.isValid() ? this._data[name] : NaN; 4266 }; 4267} 4268 4269var milliseconds = makeGetter('milliseconds'); 4270var seconds = makeGetter('seconds'); 4271var minutes = makeGetter('minutes'); 4272var hours = makeGetter('hours'); 4273var days = makeGetter('days'); 4274var months = makeGetter('months'); 4275var years = makeGetter('years'); 4276 4277function weeks () { 4278 return absFloor(this.days() / 7); 4279} 4280 4281var round = Math.round; 4282var thresholds = { 4283 ss: 44, // a few seconds to seconds 4284 s : 45, // seconds to minute 4285 m : 45, // minutes to hour 4286 h : 22, // hours to day 4287 d : 26, // days to month 4288 M : 11 // months to year 4289}; 4290 4291// helper function for moment.fn.from, moment.fn.fromNow, and moment.duration.fn.humanize 4292function substituteTimeAgo(string, number, withoutSuffix, isFuture, locale) { 4293 return locale.relativeTime(number || 1, !!withoutSuffix, string, isFuture); 4294} 4295 4296function relativeTime$1 (posNegDuration, withoutSuffix, locale) { 4297 var duration = createDuration(posNegDuration).abs(); 4298 var seconds = round(duration.as('s')); 4299 var minutes = round(duration.as('m')); 4300 var hours = round(duration.as('h')); 4301 var days = round(duration.as('d')); 4302 var months = round(duration.as('M')); 4303 var years = round(duration.as('y')); 4304 4305 var a = seconds <= thresholds.ss && ['s', seconds] || 4306 seconds < thresholds.s && ['ss', seconds] || 4307 minutes <= 1 && ['m'] || 4308 minutes < thresholds.m && ['mm', minutes] || 4309 hours <= 1 && ['h'] || 4310 hours < thresholds.h && ['hh', hours] || 4311 days <= 1 && ['d'] || 4312 days < thresholds.d && ['dd', days] || 4313 months <= 1 && ['M'] || 4314 months < thresholds.M && ['MM', months] || 4315 years <= 1 && ['y'] || ['yy', years]; 4316 4317 a[2] = withoutSuffix; 4318 a[3] = +posNegDuration > 0; 4319 a[4] = locale; 4320 return substituteTimeAgo.apply(null, a); 4321} 4322 4323// This function allows you to set the rounding function for relative time strings 4324function getSetRelativeTimeRounding (roundingFunction) { 4325 if (roundingFunction === undefined) { 4326 return round; 4327 } 4328 if (typeof(roundingFunction) === 'function') { 4329 round = roundingFunction; 4330 return true; 4331 } 4332 return false; 4333} 4334 4335// This function allows you to set a threshold for relative time strings 4336function getSetRelativeTimeThreshold (threshold, limit) { 4337 if (thresholds[threshold] === undefined) { 4338 return false; 4339 } 4340 if (limit === undefined) { 4341 return thresholds[threshold]; 4342 } 4343 thresholds[threshold] = limit; 4344 if (threshold === 's') { 4345 thresholds.ss = limit - 1; 4346 } 4347 return true; 4348} 4349 4350function humanize (withSuffix) { 4351 if (!this.isValid()) { 4352 return this.localeData().invalidDate(); 4353 } 4354 4355 var locale = this.localeData(); 4356 var output = relativeTime$1(this, !withSuffix, locale); 4357 4358 if (withSuffix) { 4359 output = locale.pastFuture(+this, output); 4360 } 4361 4362 return locale.postformat(output); 4363} 4364 4365var abs$1 = Math.abs; 4366 4367function sign(x) { 4368 return ((x > 0) - (x < 0)) || +x; 4369} 4370 4371function toISOString$1() { 4372 // for ISO strings we do not use the normal bubbling rules: 4373 // * milliseconds bubble up until they become hours 4374 // * days do not bubble at all 4375 // * months bubble up until they become years 4376 // This is because there is no context-free conversion between hours and days 4377 // (think of clock changes) 4378 // and also not between days and months (28-31 days per month) 4379 if (!this.isValid()) { 4380 return this.localeData().invalidDate(); 4381 } 4382 4383 var seconds = abs$1(this._milliseconds) / 1000; 4384 var days = abs$1(this._days); 4385 var months = abs$1(this._months); 4386 var minutes, hours, years; 4387 4388 // 3600 seconds -> 60 minutes -> 1 hour 4389 minutes = absFloor(seconds / 60); 4390 hours = absFloor(minutes / 60); 4391 seconds %= 60; 4392 minutes %= 60; 4393 4394 // 12 months -> 1 year 4395 years = absFloor(months / 12); 4396 months %= 12; 4397 4398 4399 // inspired by https://github.com/dordille/moment-isoduration/blob/master/moment.isoduration.js 4400 var Y = years; 4401 var M = months; 4402 var D = days; 4403 var h = hours; 4404 var m = minutes; 4405 var s = seconds ? seconds.toFixed(3).replace(/\.?0+$/, '') : ''; 4406 var total = this.asSeconds(); 4407 4408 if (!total) { 4409 // this is the same as C#'s (Noda) and python (isodate)... 4410 // but not other JS (goog.date) 4411 return 'P0D'; 4412 } 4413 4414 var totalSign = total < 0 ? '-' : ''; 4415 var ymSign = sign(this._months) !== sign(total) ? '-' : ''; 4416 var daysSign = sign(this._days) !== sign(total) ? '-' : ''; 4417 var hmsSign = sign(this._milliseconds) !== sign(total) ? '-' : ''; 4418 4419 return totalSign + 'P' + 4420 (Y ? ymSign + Y + 'Y' : '') + 4421 (M ? ymSign + M + 'M' : '') + 4422 (D ? daysSign + D + 'D' : '') + 4423 ((h || m || s) ? 'T' : '') + 4424 (h ? hmsSign + h + 'H' : '') + 4425 (m ? hmsSign + m + 'M' : '') + 4426 (s ? hmsSign + s + 'S' : ''); 4427} 4428 4429var proto$2 = Duration.prototype; 4430 4431proto$2.isValid = isValid$1; 4432proto$2.abs = abs; 4433proto$2.add = add$1; 4434proto$2.subtract = subtract$1; 4435proto$2.as = as; 4436proto$2.asMilliseconds = asMilliseconds; 4437proto$2.asSeconds = asSeconds; 4438proto$2.asMinutes = asMinutes; 4439proto$2.asHours = asHours; 4440proto$2.asDays = asDays; 4441proto$2.asWeeks = asWeeks; 4442proto$2.asMonths = asMonths; 4443proto$2.asYears = asYears; 4444proto$2.valueOf = valueOf$1; 4445proto$2._bubble = bubble; 4446proto$2.clone = clone$1; 4447proto$2.get = get$2; 4448proto$2.milliseconds = milliseconds; 4449proto$2.seconds = seconds; 4450proto$2.minutes = minutes; 4451proto$2.hours = hours; 4452proto$2.days = days; 4453proto$2.weeks = weeks; 4454proto$2.months = months; 4455proto$2.years = years; 4456proto$2.humanize = humanize; 4457proto$2.toISOString = toISOString$1; 4458proto$2.toString = toISOString$1; 4459proto$2.toJSON = toISOString$1; 4460proto$2.locale = locale; 4461proto$2.localeData = localeData; 4462 4463// Deprecations 4464proto$2.toIsoString = deprecate('toIsoString() is deprecated. Please use toISOString() instead (notice the capitals)', toISOString$1); 4465proto$2.lang = lang; 4466 4467// Side effect imports 4468 4469// FORMATTING 4470 4471addFormatToken('X', 0, 0, 'unix'); 4472addFormatToken('x', 0, 0, 'valueOf'); 4473 4474// PARSING 4475 4476addRegexToken('x', matchSigned); 4477addRegexToken('X', matchTimestamp); 4478addParseToken('X', function (input, array, config) { 4479 config._d = new Date(parseFloat(input, 10) * 1000); 4480}); 4481addParseToken('x', function (input, array, config) { 4482 config._d = new Date(toInt(input)); 4483}); 4484 4485// Side effect imports 4486 4487 4488hooks.version = '2.20.1'; 4489 4490setHookCallback(createLocal); 4491 4492hooks.fn = proto; 4493hooks.min = min; 4494hooks.max = max; 4495hooks.now = now; 4496hooks.utc = createUTC; 4497hooks.unix = createUnix; 4498hooks.months = listMonths; 4499hooks.isDate = isDate; 4500hooks.locale = getSetGlobalLocale; 4501hooks.invalid = createInvalid; 4502hooks.duration = createDuration; 4503hooks.isMoment = isMoment; 4504hooks.weekdays = listWeekdays; 4505hooks.parseZone = createInZone; 4506hooks.localeData = getLocale; 4507hooks.isDuration = isDuration; 4508hooks.monthsShort = listMonthsShort; 4509hooks.weekdaysMin = listWeekdaysMin; 4510hooks.defineLocale = defineLocale; 4511hooks.updateLocale = updateLocale; 4512hooks.locales = listLocales; 4513hooks.weekdaysShort = listWeekdaysShort; 4514hooks.normalizeUnits = normalizeUnits; 4515hooks.relativeTimeRounding = getSetRelativeTimeRounding; 4516hooks.relativeTimeThreshold = getSetRelativeTimeThreshold; 4517hooks.calendarFormat = getCalendarFormat; 4518hooks.prototype = proto; 4519 4520// currently HTML5 input type only supports 24-hour formats 4521hooks.HTML5_FMT = { 4522 DATETIME_LOCAL: 'YYYY-MM-DDTHH:mm', // <input type="datetime-local" /> 4523 DATETIME_LOCAL_SECONDS: 'YYYY-MM-DDTHH:mm:ss', // <input type="datetime-local" step="1" /> 4524 DATETIME_LOCAL_MS: 'YYYY-MM-DDTHH:mm:ss.SSS', // <input type="datetime-local" step="0.001" /> 4525 DATE: 'YYYY-MM-DD', // <input type="date" /> 4526 TIME: 'HH:mm', // <input type="time" /> 4527 TIME_SECONDS: 'HH:mm:ss', // <input type="time" step="1" /> 4528 TIME_MS: 'HH:mm:ss.SSS', // <input type="time" step="0.001" /> 4529 WEEK: 'YYYY-[W]WW', // <input type="week" /> 4530 MONTH: 'YYYY-MM' // <input type="month" /> 4531}; 4532 4533return hooks; 4534 4535}))); 4536