=11?e:e+12:"下午"===t||"晚上"===t?e+12:void 0},meridiem:function(e,t,a){var d=100*e+t;return d<600?"凌晨":d<900?"早上":d<1130?"上午":d<1230?"中午":d<1800?"下午":"晚上"},calendar:{sameDay:"[今天]LT",nextDay:"[明天]LT",nextWeek:"[下]ddddLT",lastDay:"[昨天]LT",lastWeek:"[上]ddddLT",sameElse:"L"},ordinalParse:/\d{1,2}(日|月|週)/,ordinal:function(e,t){switch(t){case"d":case"D":case"DDD":return e+"日";case"M":return e+"月";case"w":case"W":return e+"週";default:return e}},relativeTime:{future:"%s內",past:"%s前",s:"幾秒",m:"1 分鐘",mm:"%d 分鐘",h:"1 小時",hh:"%d 小時",d:"1 天",dd:"%d 天",M:"1 個月",MM:"%d 個月",y:"1 年",yy:"%d 年"}});return e}(),e.fullCalendar.datepickerLocale("zh-tw","zh-TW",{closeText:"關閉",prevText:"<上月",nextText:"下月>",currentText:"今天",monthNames:["一月","二月","三月","四月","五月","六月","七月","八月","九月","十月","十一月","十二月"],monthNamesShort:["一月","二月","三月","四月","五月","六月","七月","八月","九月","十月","十一月","十二月"],dayNames:["星期日","星期一","星期二","星期三","星期四","星期五","星期六"],dayNamesShort:["周日","周一","周二","周三","周四","周五","周六"],dayNamesMin:["日","一","二","三","四","五","六"],weekHeader:"周",dateFormat:"yy/mm/dd",firstDay:1,isRTL:!1,showMonthAfterYear:!0,yearSuffix:"年"}),e.fullCalendar.locale("zh-tw",{buttonText:{month:"月",week:"週",day:"天",list:"活動列表"},allDayText:"整天",eventLimitText:"顯示更多",noEventsMessage:"没有任何活動"})});
\ No newline at end of file
diff --git a/fullcalendar-3.2.0/moment.js b/fullcalendar-3.2.0/moment.js
deleted file mode 100644
--- a/fullcalendar-3.2.0/moment.js
+++ /dev/null
@@ -1,4301 +0,0 @@
-//! moment.js
-//! version : 2.17.1
-//! authors : Tim Wood, Iskren Chernev, Moment.js contributors
-//! license : MIT
-//! momentjs.com
-
-;(function (global, factory) {
- typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
- typeof define === 'function' && define.amd ? define(factory) :
- global.moment = factory()
-}(this, (function () { 'use strict';
-
-var hookCallback;
-
-function hooks () {
- return hookCallback.apply(null, arguments);
-}
-
-// This is done to register the method called with moment()
-// without creating circular dependencies.
-function setHookCallback (callback) {
- hookCallback = callback;
-}
-
-function isArray(input) {
- return input instanceof Array || Object.prototype.toString.call(input) === '[object Array]';
-}
-
-function isObject(input) {
- // IE8 will treat undefined and null as object if it wasn't for
- // input != null
- return input != null && Object.prototype.toString.call(input) === '[object Object]';
-}
-
-function isObjectEmpty(obj) {
- var k;
- for (k in obj) {
- // even if its not own property I'd still call it non-empty
- return false;
- }
- return true;
-}
-
-function isNumber(input) {
- return typeof input === 'number' || Object.prototype.toString.call(input) === '[object Number]';
-}
-
-function isDate(input) {
- return input instanceof Date || Object.prototype.toString.call(input) === '[object Date]';
-}
-
-function map(arr, fn) {
- var res = [], i;
- for (i = 0; i < arr.length; ++i) {
- res.push(fn(arr[i], i));
- }
- return res;
-}
-
-function hasOwnProp(a, b) {
- return Object.prototype.hasOwnProperty.call(a, b);
-}
-
-function extend(a, b) {
- for (var i in b) {
- if (hasOwnProp(b, i)) {
- a[i] = b[i];
- }
- }
-
- if (hasOwnProp(b, 'toString')) {
- a.toString = b.toString;
- }
-
- if (hasOwnProp(b, 'valueOf')) {
- a.valueOf = b.valueOf;
- }
-
- return a;
-}
-
-function createUTC (input, format, locale, strict) {
- return createLocalOrUTC(input, format, locale, strict, true).utc();
-}
-
-function defaultParsingFlags() {
- // We need to deep clone this object.
- return {
- empty : false,
- unusedTokens : [],
- unusedInput : [],
- overflow : -2,
- charsLeftOver : 0,
- nullInput : false,
- invalidMonth : null,
- invalidFormat : false,
- userInvalidated : false,
- iso : false,
- parsedDateParts : [],
- meridiem : null
- };
-}
-
-function getParsingFlags(m) {
- if (m._pf == null) {
- m._pf = defaultParsingFlags();
- }
- return m._pf;
-}
-
-var some;
-if (Array.prototype.some) {
- some = Array.prototype.some;
-} else {
- some = function (fun) {
- var t = Object(this);
- var len = t.length >>> 0;
-
- for (var i = 0; i < len; i++) {
- if (i in t && fun.call(this, t[i], i, t)) {
- return true;
- }
- }
-
- return false;
- };
-}
-
-var some$1 = some;
-
-function isValid(m) {
- if (m._isValid == null) {
- var flags = getParsingFlags(m);
- var parsedParts = some$1.call(flags.parsedDateParts, function (i) {
- return i != null;
- });
- var isNowValid = !isNaN(m._d.getTime()) &&
- flags.overflow < 0 &&
- !flags.empty &&
- !flags.invalidMonth &&
- !flags.invalidWeekday &&
- !flags.nullInput &&
- !flags.invalidFormat &&
- !flags.userInvalidated &&
- (!flags.meridiem || (flags.meridiem && parsedParts));
-
- if (m._strict) {
- isNowValid = isNowValid &&
- flags.charsLeftOver === 0 &&
- flags.unusedTokens.length === 0 &&
- flags.bigHour === undefined;
- }
-
- if (Object.isFrozen == null || !Object.isFrozen(m)) {
- m._isValid = isNowValid;
- }
- else {
- return isNowValid;
- }
- }
- return m._isValid;
-}
-
-function createInvalid (flags) {
- var m = createUTC(NaN);
- if (flags != null) {
- extend(getParsingFlags(m), flags);
- }
- else {
- getParsingFlags(m).userInvalidated = true;
- }
-
- return m;
-}
-
-function isUndefined(input) {
- return input === void 0;
-}
-
-// Plugins that add properties should also add the key here (null value),
-// so we can properly clone ourselves.
-var momentProperties = hooks.momentProperties = [];
-
-function copyConfig(to, from) {
- var i, prop, val;
-
- if (!isUndefined(from._isAMomentObject)) {
- to._isAMomentObject = from._isAMomentObject;
- }
- if (!isUndefined(from._i)) {
- to._i = from._i;
- }
- if (!isUndefined(from._f)) {
- to._f = from._f;
- }
- if (!isUndefined(from._l)) {
- to._l = from._l;
- }
- if (!isUndefined(from._strict)) {
- to._strict = from._strict;
- }
- if (!isUndefined(from._tzm)) {
- to._tzm = from._tzm;
- }
- if (!isUndefined(from._isUTC)) {
- to._isUTC = from._isUTC;
- }
- if (!isUndefined(from._offset)) {
- to._offset = from._offset;
- }
- if (!isUndefined(from._pf)) {
- to._pf = getParsingFlags(from);
- }
- if (!isUndefined(from._locale)) {
- to._locale = from._locale;
- }
-
- if (momentProperties.length > 0) {
- for (i in momentProperties) {
- prop = momentProperties[i];
- val = from[prop];
- if (!isUndefined(val)) {
- to[prop] = val;
- }
- }
- }
-
- return to;
-}
-
-var updateInProgress = false;
-
-// Moment prototype object
-function Moment(config) {
- copyConfig(this, config);
- this._d = new Date(config._d != null ? config._d.getTime() : NaN);
- if (!this.isValid()) {
- this._d = new Date(NaN);
- }
- // Prevent infinite loop in case updateOffset creates new moment
- // objects.
- if (updateInProgress === false) {
- updateInProgress = true;
- hooks.updateOffset(this);
- updateInProgress = false;
- }
-}
-
-function isMoment (obj) {
- return obj instanceof Moment || (obj != null && obj._isAMomentObject != null);
-}
-
-function absFloor (number) {
- if (number < 0) {
- // -0 -> 0
- return Math.ceil(number) || 0;
- } else {
- return Math.floor(number);
- }
-}
-
-function toInt(argumentForCoercion) {
- var coercedNumber = +argumentForCoercion,
- value = 0;
-
- if (coercedNumber !== 0 && isFinite(coercedNumber)) {
- value = absFloor(coercedNumber);
- }
-
- return value;
-}
-
-// compare two arrays, return the number of differences
-function compareArrays(array1, array2, dontConvert) {
- var len = Math.min(array1.length, array2.length),
- lengthDiff = Math.abs(array1.length - array2.length),
- diffs = 0,
- i;
- for (i = 0; i < len; i++) {
- if ((dontConvert && array1[i] !== array2[i]) ||
- (!dontConvert && toInt(array1[i]) !== toInt(array2[i]))) {
- diffs++;
- }
- }
- return diffs + lengthDiff;
-}
-
-function warn(msg) {
- if (hooks.suppressDeprecationWarnings === false &&
- (typeof console !== 'undefined') && console.warn) {
- console.warn('Deprecation warning: ' + msg);
- }
-}
-
-function deprecate(msg, fn) {
- var firstTime = true;
-
- return extend(function () {
- if (hooks.deprecationHandler != null) {
- hooks.deprecationHandler(null, msg);
- }
- if (firstTime) {
- var args = [];
- var arg;
- for (var i = 0; i < arguments.length; i++) {
- arg = '';
- if (typeof arguments[i] === 'object') {
- arg += '\n[' + i + '] ';
- for (var key in arguments[0]) {
- arg += key + ': ' + arguments[0][key] + ', ';
- }
- arg = arg.slice(0, -2); // Remove trailing comma and space
- } else {
- arg = arguments[i];
- }
- args.push(arg);
- }
- warn(msg + '\nArguments: ' + Array.prototype.slice.call(args).join('') + '\n' + (new Error()).stack);
- firstTime = false;
- }
- return fn.apply(this, arguments);
- }, fn);
-}
-
-var deprecations = {};
-
-function deprecateSimple(name, msg) {
- if (hooks.deprecationHandler != null) {
- hooks.deprecationHandler(name, msg);
- }
- if (!deprecations[name]) {
- warn(msg);
- deprecations[name] = true;
- }
-}
-
-hooks.suppressDeprecationWarnings = false;
-hooks.deprecationHandler = null;
-
-function isFunction(input) {
- return input instanceof Function || Object.prototype.toString.call(input) === '[object Function]';
-}
-
-function set (config) {
- var prop, i;
- for (i in config) {
- prop = config[i];
- if (isFunction(prop)) {
- this[i] = prop;
- } else {
- this['_' + i] = prop;
- }
- }
- this._config = config;
- // Lenient ordinal parsing accepts just a number in addition to
- // number + (possibly) stuff coming from _ordinalParseLenient.
- this._ordinalParseLenient = new RegExp(this._ordinalParse.source + '|' + (/\d{1,2}/).source);
-}
-
-function mergeConfigs(parentConfig, childConfig) {
- var res = extend({}, parentConfig), prop;
- for (prop in childConfig) {
- if (hasOwnProp(childConfig, prop)) {
- if (isObject(parentConfig[prop]) && isObject(childConfig[prop])) {
- res[prop] = {};
- extend(res[prop], parentConfig[prop]);
- extend(res[prop], childConfig[prop]);
- } else if (childConfig[prop] != null) {
- res[prop] = childConfig[prop];
- } else {
- delete res[prop];
- }
- }
- }
- for (prop in parentConfig) {
- if (hasOwnProp(parentConfig, prop) &&
- !hasOwnProp(childConfig, prop) &&
- isObject(parentConfig[prop])) {
- // make sure changes to properties don't modify parent config
- res[prop] = extend({}, res[prop]);
- }
- }
- return res;
-}
-
-function Locale(config) {
- if (config != null) {
- this.set(config);
- }
-}
-
-var keys;
-
-if (Object.keys) {
- keys = Object.keys;
-} else {
- keys = function (obj) {
- var i, res = [];
- for (i in obj) {
- if (hasOwnProp(obj, i)) {
- res.push(i);
- }
- }
- return res;
- };
-}
-
-var keys$1 = keys;
-
-var defaultCalendar = {
- sameDay : '[Today at] LT',
- nextDay : '[Tomorrow at] LT',
- nextWeek : 'dddd [at] LT',
- lastDay : '[Yesterday at] LT',
- lastWeek : '[Last] dddd [at] LT',
- sameElse : 'L'
-};
-
-function calendar (key, mom, now) {
- var output = this._calendar[key] || this._calendar['sameElse'];
- return isFunction(output) ? output.call(mom, now) : output;
-}
-
-var defaultLongDateFormat = {
- LTS : 'h:mm:ss A',
- LT : 'h:mm A',
- L : 'MM/DD/YYYY',
- LL : 'MMMM D, YYYY',
- LLL : 'MMMM D, YYYY h:mm A',
- LLLL : 'dddd, MMMM D, YYYY h:mm A'
-};
-
-function longDateFormat (key) {
- var format = this._longDateFormat[key],
- formatUpper = this._longDateFormat[key.toUpperCase()];
-
- if (format || !formatUpper) {
- return format;
- }
-
- this._longDateFormat[key] = formatUpper.replace(/MMMM|MM|DD|dddd/g, function (val) {
- return val.slice(1);
- });
-
- return this._longDateFormat[key];
-}
-
-var defaultInvalidDate = 'Invalid date';
-
-function invalidDate () {
- return this._invalidDate;
-}
-
-var defaultOrdinal = '%d';
-var defaultOrdinalParse = /\d{1,2}/;
-
-function ordinal (number) {
- return this._ordinal.replace('%d', number);
-}
-
-var defaultRelativeTime = {
- future : 'in %s',
- past : '%s ago',
- s : 'a few seconds',
- m : 'a minute',
- mm : '%d minutes',
- h : 'an hour',
- hh : '%d hours',
- d : 'a day',
- dd : '%d days',
- M : 'a month',
- MM : '%d months',
- y : 'a year',
- yy : '%d years'
-};
-
-function relativeTime (number, withoutSuffix, string, isFuture) {
- var output = this._relativeTime[string];
- return (isFunction(output)) ?
- output(number, withoutSuffix, string, isFuture) :
- output.replace(/%d/i, number);
-}
-
-function pastFuture (diff, output) {
- var format = this._relativeTime[diff > 0 ? 'future' : 'past'];
- return isFunction(format) ? format(output) : format.replace(/%s/i, output);
-}
-
-var aliases = {};
-
-function addUnitAlias (unit, shorthand) {
- var lowerCase = unit.toLowerCase();
- aliases[lowerCase] = aliases[lowerCase + 's'] = aliases[shorthand] = unit;
-}
-
-function normalizeUnits(units) {
- return typeof units === 'string' ? aliases[units] || aliases[units.toLowerCase()] : undefined;
-}
-
-function normalizeObjectUnits(inputObject) {
- var normalizedInput = {},
- normalizedProp,
- prop;
-
- for (prop in inputObject) {
- if (hasOwnProp(inputObject, prop)) {
- normalizedProp = normalizeUnits(prop);
- if (normalizedProp) {
- normalizedInput[normalizedProp] = inputObject[prop];
- }
- }
- }
-
- return normalizedInput;
-}
-
-var priorities = {};
-
-function addUnitPriority(unit, priority) {
- priorities[unit] = priority;
-}
-
-function getPrioritizedUnits(unitsObj) {
- var units = [];
- for (var u in unitsObj) {
- units.push({unit: u, priority: priorities[u]});
- }
- units.sort(function (a, b) {
- return a.priority - b.priority;
- });
- return units;
-}
-
-function makeGetSet (unit, keepTime) {
- return function (value) {
- if (value != null) {
- set$1(this, unit, value);
- hooks.updateOffset(this, keepTime);
- return this;
- } else {
- return get(this, unit);
- }
- };
-}
-
-function get (mom, unit) {
- return mom.isValid() ?
- mom._d['get' + (mom._isUTC ? 'UTC' : '') + unit]() : NaN;
-}
-
-function set$1 (mom, unit, value) {
- if (mom.isValid()) {
- mom._d['set' + (mom._isUTC ? 'UTC' : '') + unit](value);
- }
-}
-
-// MOMENTS
-
-function stringGet (units) {
- units = normalizeUnits(units);
- if (isFunction(this[units])) {
- return this[units]();
- }
- return this;
-}
-
-
-function stringSet (units, value) {
- if (typeof units === 'object') {
- units = normalizeObjectUnits(units);
- var prioritized = getPrioritizedUnits(units);
- for (var i = 0; i < prioritized.length; i++) {
- this[prioritized[i].unit](units[prioritized[i].unit]);
- }
- } else {
- units = normalizeUnits(units);
- if (isFunction(this[units])) {
- return this[units](value);
- }
- }
- return this;
-}
-
-function zeroFill(number, targetLength, forceSign) {
- var absNumber = '' + Math.abs(number),
- zerosToFill = targetLength - absNumber.length,
- sign = number >= 0;
- return (sign ? (forceSign ? '+' : '') : '-') +
- Math.pow(10, Math.max(0, zerosToFill)).toString().substr(1) + absNumber;
-}
-
-var 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;
-
-var localFormattingTokens = /(\[[^\[]*\])|(\\)?(LTS|LT|LL?L?L?|l{1,4})/g;
-
-var formatFunctions = {};
-
-var formatTokenFunctions = {};
-
-// token: 'M'
-// padded: ['MM', 2]
-// ordinal: 'Mo'
-// callback: function () { this.month() + 1 }
-function addFormatToken (token, padded, ordinal, callback) {
- var func = callback;
- if (typeof callback === 'string') {
- func = function () {
- return this[callback]();
- };
- }
- if (token) {
- formatTokenFunctions[token] = func;
- }
- if (padded) {
- formatTokenFunctions[padded[0]] = function () {
- return zeroFill(func.apply(this, arguments), padded[1], padded[2]);
- };
- }
- if (ordinal) {
- formatTokenFunctions[ordinal] = function () {
- return this.localeData().ordinal(func.apply(this, arguments), token);
- };
- }
-}
-
-function removeFormattingTokens(input) {
- if (input.match(/\[[\s\S]/)) {
- return input.replace(/^\[|\]$/g, '');
- }
- return input.replace(/\\/g, '');
-}
-
-function makeFormatFunction(format) {
- var array = format.match(formattingTokens), i, length;
-
- for (i = 0, length = array.length; i < length; i++) {
- if (formatTokenFunctions[array[i]]) {
- array[i] = formatTokenFunctions[array[i]];
- } else {
- array[i] = removeFormattingTokens(array[i]);
- }
- }
-
- return function (mom) {
- var output = '', i;
- for (i = 0; i < length; i++) {
- output += array[i] instanceof Function ? array[i].call(mom, format) : array[i];
- }
- return output;
- };
-}
-
-// format date using native date object
-function formatMoment(m, format) {
- if (!m.isValid()) {
- return m.localeData().invalidDate();
- }
-
- format = expandFormat(format, m.localeData());
- formatFunctions[format] = formatFunctions[format] || makeFormatFunction(format);
-
- return formatFunctions[format](m);
-}
-
-function expandFormat(format, locale) {
- var i = 5;
-
- function replaceLongDateFormatTokens(input) {
- return locale.longDateFormat(input) || input;
- }
-
- localFormattingTokens.lastIndex = 0;
- while (i >= 0 && localFormattingTokens.test(format)) {
- format = format.replace(localFormattingTokens, replaceLongDateFormatTokens);
- localFormattingTokens.lastIndex = 0;
- i -= 1;
- }
-
- return format;
-}
-
-var match1 = /\d/; // 0 - 9
-var match2 = /\d\d/; // 00 - 99
-var match3 = /\d{3}/; // 000 - 999
-var match4 = /\d{4}/; // 0000 - 9999
-var match6 = /[+-]?\d{6}/; // -999999 - 999999
-var match1to2 = /\d\d?/; // 0 - 99
-var match3to4 = /\d\d\d\d?/; // 999 - 9999
-var match5to6 = /\d\d\d\d\d\d?/; // 99999 - 999999
-var match1to3 = /\d{1,3}/; // 0 - 999
-var match1to4 = /\d{1,4}/; // 0 - 9999
-var match1to6 = /[+-]?\d{1,6}/; // -999999 - 999999
-
-var matchUnsigned = /\d+/; // 0 - inf
-var matchSigned = /[+-]?\d+/; // -inf - inf
-
-var matchOffset = /Z|[+-]\d\d:?\d\d/gi; // +00:00 -00:00 +0000 -0000 or Z
-var matchShortOffset = /Z|[+-]\d\d(?::?\d\d)?/gi; // +00 -00 +00:00 -00:00 +0000 -0000 or Z
-
-var matchTimestamp = /[+-]?\d+(\.\d{1,3})?/; // 123456789 123456789.123
-
-// any word (or two) characters or numbers including two/three word month in arabic.
-// includes scottish gaelic two word and hyphenated months
-var matchWord = /[0-9]*['a-z\u00A0-\u05FF\u0700-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]+|[\u0600-\u06FF\/]+(\s*?[\u0600-\u06FF]+){1,2}/i;
-
-
-var regexes = {};
-
-function addRegexToken (token, regex, strictRegex) {
- regexes[token] = isFunction(regex) ? regex : function (isStrict, localeData) {
- return (isStrict && strictRegex) ? strictRegex : regex;
- };
-}
-
-function getParseRegexForToken (token, config) {
- if (!hasOwnProp(regexes, token)) {
- return new RegExp(unescapeFormat(token));
- }
-
- return regexes[token](config._strict, config._locale);
-}
-
-// Code from http://stackoverflow.com/questions/3561493/is-there-a-regexp-escape-function-in-javascript
-function unescapeFormat(s) {
- return regexEscape(s.replace('\\', '').replace(/\\(\[)|\\(\])|\[([^\]\[]*)\]|\\(.)/g, function (matched, p1, p2, p3, p4) {
- return p1 || p2 || p3 || p4;
- }));
-}
-
-function regexEscape(s) {
- return s.replace(/[-\/\\^$*+?.()|[\]{}]/g, '\\$&');
-}
-
-var tokens = {};
-
-function addParseToken (token, callback) {
- var i, func = callback;
- if (typeof token === 'string') {
- token = [token];
- }
- if (isNumber(callback)) {
- func = function (input, array) {
- array[callback] = toInt(input);
- };
- }
- for (i = 0; i < token.length; i++) {
- tokens[token[i]] = func;
- }
-}
-
-function addWeekParseToken (token, callback) {
- addParseToken(token, function (input, array, config, token) {
- config._w = config._w || {};
- callback(input, config._w, config, token);
- });
-}
-
-function addTimeToArrayFromToken(token, input, config) {
- if (input != null && hasOwnProp(tokens, token)) {
- tokens[token](input, config._a, config, token);
- }
-}
-
-var YEAR = 0;
-var MONTH = 1;
-var DATE = 2;
-var HOUR = 3;
-var MINUTE = 4;
-var SECOND = 5;
-var MILLISECOND = 6;
-var WEEK = 7;
-var WEEKDAY = 8;
-
-var indexOf;
-
-if (Array.prototype.indexOf) {
- indexOf = Array.prototype.indexOf;
-} else {
- indexOf = function (o) {
- // I know
- var i;
- for (i = 0; i < this.length; ++i) {
- if (this[i] === o) {
- return i;
- }
- }
- return -1;
- };
-}
-
-var indexOf$1 = indexOf;
-
-function daysInMonth(year, month) {
- return new Date(Date.UTC(year, month + 1, 0)).getUTCDate();
-}
-
-// FORMATTING
-
-addFormatToken('M', ['MM', 2], 'Mo', function () {
- return this.month() + 1;
-});
-
-addFormatToken('MMM', 0, 0, function (format) {
- return this.localeData().monthsShort(this, format);
-});
-
-addFormatToken('MMMM', 0, 0, function (format) {
- return this.localeData().months(this, format);
-});
-
-// ALIASES
-
-addUnitAlias('month', 'M');
-
-// PRIORITY
-
-addUnitPriority('month', 8);
-
-// PARSING
-
-addRegexToken('M', match1to2);
-addRegexToken('MM', match1to2, match2);
-addRegexToken('MMM', function (isStrict, locale) {
- return locale.monthsShortRegex(isStrict);
-});
-addRegexToken('MMMM', function (isStrict, locale) {
- return locale.monthsRegex(isStrict);
-});
-
-addParseToken(['M', 'MM'], function (input, array) {
- array[MONTH] = toInt(input) - 1;
-});
-
-addParseToken(['MMM', 'MMMM'], function (input, array, config, token) {
- var month = config._locale.monthsParse(input, token, config._strict);
- // if we didn't find a month name, mark the date as invalid.
- if (month != null) {
- array[MONTH] = month;
- } else {
- getParsingFlags(config).invalidMonth = input;
- }
-});
-
-// LOCALES
-
-var MONTHS_IN_FORMAT = /D[oD]?(\[[^\[\]]*\]|\s)+MMMM?/;
-var defaultLocaleMonths = 'January_February_March_April_May_June_July_August_September_October_November_December'.split('_');
-function localeMonths (m, format) {
- if (!m) {
- return this._months;
- }
- return isArray(this._months) ? this._months[m.month()] :
- this._months[(this._months.isFormat || MONTHS_IN_FORMAT).test(format) ? 'format' : 'standalone'][m.month()];
-}
-
-var defaultLocaleMonthsShort = 'Jan_Feb_Mar_Apr_May_Jun_Jul_Aug_Sep_Oct_Nov_Dec'.split('_');
-function localeMonthsShort (m, format) {
- if (!m) {
- return this._monthsShort;
- }
- return isArray(this._monthsShort) ? this._monthsShort[m.month()] :
- this._monthsShort[MONTHS_IN_FORMAT.test(format) ? 'format' : 'standalone'][m.month()];
-}
-
-function handleStrictParse(monthName, format, strict) {
- var i, ii, mom, llc = monthName.toLocaleLowerCase();
- if (!this._monthsParse) {
- // this is not used
- this._monthsParse = [];
- this._longMonthsParse = [];
- this._shortMonthsParse = [];
- for (i = 0; i < 12; ++i) {
- mom = createUTC([2000, i]);
- this._shortMonthsParse[i] = this.monthsShort(mom, '').toLocaleLowerCase();
- this._longMonthsParse[i] = this.months(mom, '').toLocaleLowerCase();
- }
- }
-
- if (strict) {
- if (format === 'MMM') {
- ii = indexOf$1.call(this._shortMonthsParse, llc);
- return ii !== -1 ? ii : null;
- } else {
- ii = indexOf$1.call(this._longMonthsParse, llc);
- return ii !== -1 ? ii : null;
- }
- } else {
- if (format === 'MMM') {
- ii = indexOf$1.call(this._shortMonthsParse, llc);
- if (ii !== -1) {
- return ii;
- }
- ii = indexOf$1.call(this._longMonthsParse, llc);
- return ii !== -1 ? ii : null;
- } else {
- ii = indexOf$1.call(this._longMonthsParse, llc);
- if (ii !== -1) {
- return ii;
- }
- ii = indexOf$1.call(this._shortMonthsParse, llc);
- return ii !== -1 ? ii : null;
- }
- }
-}
-
-function localeMonthsParse (monthName, format, strict) {
- var i, mom, regex;
-
- if (this._monthsParseExact) {
- return handleStrictParse.call(this, monthName, format, strict);
- }
-
- if (!this._monthsParse) {
- this._monthsParse = [];
- this._longMonthsParse = [];
- this._shortMonthsParse = [];
- }
-
- // TODO: add sorting
- // Sorting makes sure if one month (or abbr) is a prefix of another
- // see sorting in computeMonthsParse
- for (i = 0; i < 12; i++) {
- // make the regex if we don't have it already
- mom = createUTC([2000, i]);
- if (strict && !this._longMonthsParse[i]) {
- this._longMonthsParse[i] = new RegExp('^' + this.months(mom, '').replace('.', '') + '$', 'i');
- this._shortMonthsParse[i] = new RegExp('^' + this.monthsShort(mom, '').replace('.', '') + '$', 'i');
- }
- if (!strict && !this._monthsParse[i]) {
- regex = '^' + this.months(mom, '') + '|^' + this.monthsShort(mom, '');
- this._monthsParse[i] = new RegExp(regex.replace('.', ''), 'i');
- }
- // test the regex
- if (strict && format === 'MMMM' && this._longMonthsParse[i].test(monthName)) {
- return i;
- } else if (strict && format === 'MMM' && this._shortMonthsParse[i].test(monthName)) {
- return i;
- } else if (!strict && this._monthsParse[i].test(monthName)) {
- return i;
- }
- }
-}
-
-// MOMENTS
-
-function setMonth (mom, value) {
- var dayOfMonth;
-
- if (!mom.isValid()) {
- // No op
- return mom;
- }
-
- if (typeof value === 'string') {
- if (/^\d+$/.test(value)) {
- value = toInt(value);
- } else {
- value = mom.localeData().monthsParse(value);
- // TODO: Another silent failure?
- if (!isNumber(value)) {
- return mom;
- }
- }
- }
-
- dayOfMonth = Math.min(mom.date(), daysInMonth(mom.year(), value));
- mom._d['set' + (mom._isUTC ? 'UTC' : '') + 'Month'](value, dayOfMonth);
- return mom;
-}
-
-function getSetMonth (value) {
- if (value != null) {
- setMonth(this, value);
- hooks.updateOffset(this, true);
- return this;
- } else {
- return get(this, 'Month');
- }
-}
-
-function getDaysInMonth () {
- return daysInMonth(this.year(), this.month());
-}
-
-var defaultMonthsShortRegex = matchWord;
-function monthsShortRegex (isStrict) {
- if (this._monthsParseExact) {
- if (!hasOwnProp(this, '_monthsRegex')) {
- computeMonthsParse.call(this);
- }
- if (isStrict) {
- return this._monthsShortStrictRegex;
- } else {
- return this._monthsShortRegex;
- }
- } else {
- if (!hasOwnProp(this, '_monthsShortRegex')) {
- this._monthsShortRegex = defaultMonthsShortRegex;
- }
- return this._monthsShortStrictRegex && isStrict ?
- this._monthsShortStrictRegex : this._monthsShortRegex;
- }
-}
-
-var defaultMonthsRegex = matchWord;
-function monthsRegex (isStrict) {
- if (this._monthsParseExact) {
- if (!hasOwnProp(this, '_monthsRegex')) {
- computeMonthsParse.call(this);
- }
- if (isStrict) {
- return this._monthsStrictRegex;
- } else {
- return this._monthsRegex;
- }
- } else {
- if (!hasOwnProp(this, '_monthsRegex')) {
- this._monthsRegex = defaultMonthsRegex;
- }
- return this._monthsStrictRegex && isStrict ?
- this._monthsStrictRegex : this._monthsRegex;
- }
-}
-
-function computeMonthsParse () {
- function cmpLenRev(a, b) {
- return b.length - a.length;
- }
-
- var shortPieces = [], longPieces = [], mixedPieces = [],
- i, mom;
- for (i = 0; i < 12; i++) {
- // make the regex if we don't have it already
- mom = createUTC([2000, i]);
- shortPieces.push(this.monthsShort(mom, ''));
- longPieces.push(this.months(mom, ''));
- mixedPieces.push(this.months(mom, ''));
- mixedPieces.push(this.monthsShort(mom, ''));
- }
- // Sorting makes sure if one month (or abbr) is a prefix of another it
- // will match the longer piece.
- shortPieces.sort(cmpLenRev);
- longPieces.sort(cmpLenRev);
- mixedPieces.sort(cmpLenRev);
- for (i = 0; i < 12; i++) {
- shortPieces[i] = regexEscape(shortPieces[i]);
- longPieces[i] = regexEscape(longPieces[i]);
- }
- for (i = 0; i < 24; i++) {
- mixedPieces[i] = regexEscape(mixedPieces[i]);
- }
-
- this._monthsRegex = new RegExp('^(' + mixedPieces.join('|') + ')', 'i');
- this._monthsShortRegex = this._monthsRegex;
- this._monthsStrictRegex = new RegExp('^(' + longPieces.join('|') + ')', 'i');
- this._monthsShortStrictRegex = new RegExp('^(' + shortPieces.join('|') + ')', 'i');
-}
-
-// FORMATTING
-
-addFormatToken('Y', 0, 0, function () {
- var y = this.year();
- return y <= 9999 ? '' + y : '+' + y;
-});
-
-addFormatToken(0, ['YY', 2], 0, function () {
- return this.year() % 100;
-});
-
-addFormatToken(0, ['YYYY', 4], 0, 'year');
-addFormatToken(0, ['YYYYY', 5], 0, 'year');
-addFormatToken(0, ['YYYYYY', 6, true], 0, 'year');
-
-// ALIASES
-
-addUnitAlias('year', 'y');
-
-// PRIORITIES
-
-addUnitPriority('year', 1);
-
-// PARSING
-
-addRegexToken('Y', matchSigned);
-addRegexToken('YY', match1to2, match2);
-addRegexToken('YYYY', match1to4, match4);
-addRegexToken('YYYYY', match1to6, match6);
-addRegexToken('YYYYYY', match1to6, match6);
-
-addParseToken(['YYYYY', 'YYYYYY'], YEAR);
-addParseToken('YYYY', function (input, array) {
- array[YEAR] = input.length === 2 ? hooks.parseTwoDigitYear(input) : toInt(input);
-});
-addParseToken('YY', function (input, array) {
- array[YEAR] = hooks.parseTwoDigitYear(input);
-});
-addParseToken('Y', function (input, array) {
- array[YEAR] = parseInt(input, 10);
-});
-
-// HELPERS
-
-function daysInYear(year) {
- return isLeapYear(year) ? 366 : 365;
-}
-
-function isLeapYear(year) {
- return (year % 4 === 0 && year % 100 !== 0) || year % 400 === 0;
-}
-
-// HOOKS
-
-hooks.parseTwoDigitYear = function (input) {
- return toInt(input) + (toInt(input) > 68 ? 1900 : 2000);
-};
-
-// MOMENTS
-
-var getSetYear = makeGetSet('FullYear', true);
-
-function getIsLeapYear () {
- return isLeapYear(this.year());
-}
-
-function createDate (y, m, d, h, M, s, ms) {
- //can't just apply() to create a date:
- //http://stackoverflow.com/questions/181348/instantiating-a-javascript-object-by-calling-prototype-constructor-apply
- var date = new Date(y, m, d, h, M, s, ms);
-
- //the date constructor remaps years 0-99 to 1900-1999
- if (y < 100 && y >= 0 && isFinite(date.getFullYear())) {
- date.setFullYear(y);
- }
- return date;
-}
-
-function createUTCDate (y) {
- var date = new Date(Date.UTC.apply(null, arguments));
-
- //the Date.UTC function remaps years 0-99 to 1900-1999
- if (y < 100 && y >= 0 && isFinite(date.getUTCFullYear())) {
- date.setUTCFullYear(y);
- }
- return date;
-}
-
-// start-of-first-week - start-of-year
-function firstWeekOffset(year, dow, doy) {
- var // first-week day -- which january is always in the first week (4 for iso, 1 for other)
- fwd = 7 + dow - doy,
- // first-week day local weekday -- which local weekday is fwd
- fwdlw = (7 + createUTCDate(year, 0, fwd).getUTCDay() - dow) % 7;
-
- return -fwdlw + fwd - 1;
-}
-
-//http://en.wikipedia.org/wiki/ISO_week_date#Calculating_a_date_given_the_year.2C_week_number_and_weekday
-function dayOfYearFromWeeks(year, week, weekday, dow, doy) {
- var localWeekday = (7 + weekday - dow) % 7,
- weekOffset = firstWeekOffset(year, dow, doy),
- dayOfYear = 1 + 7 * (week - 1) + localWeekday + weekOffset,
- resYear, resDayOfYear;
-
- if (dayOfYear <= 0) {
- resYear = year - 1;
- resDayOfYear = daysInYear(resYear) + dayOfYear;
- } else if (dayOfYear > daysInYear(year)) {
- resYear = year + 1;
- resDayOfYear = dayOfYear - daysInYear(year);
- } else {
- resYear = year;
- resDayOfYear = dayOfYear;
- }
-
- return {
- year: resYear,
- dayOfYear: resDayOfYear
- };
-}
-
-function weekOfYear(mom, dow, doy) {
- var weekOffset = firstWeekOffset(mom.year(), dow, doy),
- week = Math.floor((mom.dayOfYear() - weekOffset - 1) / 7) + 1,
- resWeek, resYear;
-
- if (week < 1) {
- resYear = mom.year() - 1;
- resWeek = week + weeksInYear(resYear, dow, doy);
- } else if (week > weeksInYear(mom.year(), dow, doy)) {
- resWeek = week - weeksInYear(mom.year(), dow, doy);
- resYear = mom.year() + 1;
- } else {
- resYear = mom.year();
- resWeek = week;
- }
-
- return {
- week: resWeek,
- year: resYear
- };
-}
-
-function weeksInYear(year, dow, doy) {
- var weekOffset = firstWeekOffset(year, dow, doy),
- weekOffsetNext = firstWeekOffset(year + 1, dow, doy);
- return (daysInYear(year) - weekOffset + weekOffsetNext) / 7;
-}
-
-// FORMATTING
-
-addFormatToken('w', ['ww', 2], 'wo', 'week');
-addFormatToken('W', ['WW', 2], 'Wo', 'isoWeek');
-
-// ALIASES
-
-addUnitAlias('week', 'w');
-addUnitAlias('isoWeek', 'W');
-
-// PRIORITIES
-
-addUnitPriority('week', 5);
-addUnitPriority('isoWeek', 5);
-
-// PARSING
-
-addRegexToken('w', match1to2);
-addRegexToken('ww', match1to2, match2);
-addRegexToken('W', match1to2);
-addRegexToken('WW', match1to2, match2);
-
-addWeekParseToken(['w', 'ww', 'W', 'WW'], function (input, week, config, token) {
- week[token.substr(0, 1)] = toInt(input);
-});
-
-// HELPERS
-
-// LOCALES
-
-function localeWeek (mom) {
- return weekOfYear(mom, this._week.dow, this._week.doy).week;
-}
-
-var defaultLocaleWeek = {
- dow : 0, // Sunday is the first day of the week.
- doy : 6 // The week that contains Jan 1st is the first week of the year.
-};
-
-function localeFirstDayOfWeek () {
- return this._week.dow;
-}
-
-function localeFirstDayOfYear () {
- return this._week.doy;
-}
-
-// MOMENTS
-
-function getSetWeek (input) {
- var week = this.localeData().week(this);
- return input == null ? week : this.add((input - week) * 7, 'd');
-}
-
-function getSetISOWeek (input) {
- var week = weekOfYear(this, 1, 4).week;
- return input == null ? week : this.add((input - week) * 7, 'd');
-}
-
-// FORMATTING
-
-addFormatToken('d', 0, 'do', 'day');
-
-addFormatToken('dd', 0, 0, function (format) {
- return this.localeData().weekdaysMin(this, format);
-});
-
-addFormatToken('ddd', 0, 0, function (format) {
- return this.localeData().weekdaysShort(this, format);
-});
-
-addFormatToken('dddd', 0, 0, function (format) {
- return this.localeData().weekdays(this, format);
-});
-
-addFormatToken('e', 0, 0, 'weekday');
-addFormatToken('E', 0, 0, 'isoWeekday');
-
-// ALIASES
-
-addUnitAlias('day', 'd');
-addUnitAlias('weekday', 'e');
-addUnitAlias('isoWeekday', 'E');
-
-// PRIORITY
-addUnitPriority('day', 11);
-addUnitPriority('weekday', 11);
-addUnitPriority('isoWeekday', 11);
-
-// PARSING
-
-addRegexToken('d', match1to2);
-addRegexToken('e', match1to2);
-addRegexToken('E', match1to2);
-addRegexToken('dd', function (isStrict, locale) {
- return locale.weekdaysMinRegex(isStrict);
-});
-addRegexToken('ddd', function (isStrict, locale) {
- return locale.weekdaysShortRegex(isStrict);
-});
-addRegexToken('dddd', function (isStrict, locale) {
- return locale.weekdaysRegex(isStrict);
-});
-
-addWeekParseToken(['dd', 'ddd', 'dddd'], function (input, week, config, token) {
- var weekday = config._locale.weekdaysParse(input, token, config._strict);
- // if we didn't get a weekday name, mark the date as invalid
- if (weekday != null) {
- week.d = weekday;
- } else {
- getParsingFlags(config).invalidWeekday = input;
- }
-});
-
-addWeekParseToken(['d', 'e', 'E'], function (input, week, config, token) {
- week[token] = toInt(input);
-});
-
-// HELPERS
-
-function parseWeekday(input, locale) {
- if (typeof input !== 'string') {
- return input;
- }
-
- if (!isNaN(input)) {
- return parseInt(input, 10);
- }
-
- input = locale.weekdaysParse(input);
- if (typeof input === 'number') {
- return input;
- }
-
- return null;
-}
-
-function parseIsoWeekday(input, locale) {
- if (typeof input === 'string') {
- return locale.weekdaysParse(input) % 7 || 7;
- }
- return isNaN(input) ? null : input;
-}
-
-// LOCALES
-
-var defaultLocaleWeekdays = 'Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday'.split('_');
-function localeWeekdays (m, format) {
- if (!m) {
- return this._weekdays;
- }
- return isArray(this._weekdays) ? this._weekdays[m.day()] :
- this._weekdays[this._weekdays.isFormat.test(format) ? 'format' : 'standalone'][m.day()];
-}
-
-var defaultLocaleWeekdaysShort = 'Sun_Mon_Tue_Wed_Thu_Fri_Sat'.split('_');
-function localeWeekdaysShort (m) {
- return (m) ? this._weekdaysShort[m.day()] : this._weekdaysShort;
-}
-
-var defaultLocaleWeekdaysMin = 'Su_Mo_Tu_We_Th_Fr_Sa'.split('_');
-function localeWeekdaysMin (m) {
- return (m) ? this._weekdaysMin[m.day()] : this._weekdaysMin;
-}
-
-function handleStrictParse$1(weekdayName, format, strict) {
- var i, ii, mom, llc = weekdayName.toLocaleLowerCase();
- if (!this._weekdaysParse) {
- this._weekdaysParse = [];
- this._shortWeekdaysParse = [];
- this._minWeekdaysParse = [];
-
- for (i = 0; i < 7; ++i) {
- mom = createUTC([2000, 1]).day(i);
- this._minWeekdaysParse[i] = this.weekdaysMin(mom, '').toLocaleLowerCase();
- this._shortWeekdaysParse[i] = this.weekdaysShort(mom, '').toLocaleLowerCase();
- this._weekdaysParse[i] = this.weekdays(mom, '').toLocaleLowerCase();
- }
- }
-
- if (strict) {
- if (format === 'dddd') {
- ii = indexOf$1.call(this._weekdaysParse, llc);
- return ii !== -1 ? ii : null;
- } else if (format === 'ddd') {
- ii = indexOf$1.call(this._shortWeekdaysParse, llc);
- return ii !== -1 ? ii : null;
- } else {
- ii = indexOf$1.call(this._minWeekdaysParse, llc);
- return ii !== -1 ? ii : null;
- }
- } else {
- if (format === 'dddd') {
- ii = indexOf$1.call(this._weekdaysParse, llc);
- if (ii !== -1) {
- return ii;
- }
- ii = indexOf$1.call(this._shortWeekdaysParse, llc);
- if (ii !== -1) {
- return ii;
- }
- ii = indexOf$1.call(this._minWeekdaysParse, llc);
- return ii !== -1 ? ii : null;
- } else if (format === 'ddd') {
- ii = indexOf$1.call(this._shortWeekdaysParse, llc);
- if (ii !== -1) {
- return ii;
- }
- ii = indexOf$1.call(this._weekdaysParse, llc);
- if (ii !== -1) {
- return ii;
- }
- ii = indexOf$1.call(this._minWeekdaysParse, llc);
- return ii !== -1 ? ii : null;
- } else {
- ii = indexOf$1.call(this._minWeekdaysParse, llc);
- if (ii !== -1) {
- return ii;
- }
- ii = indexOf$1.call(this._weekdaysParse, llc);
- if (ii !== -1) {
- return ii;
- }
- ii = indexOf$1.call(this._shortWeekdaysParse, llc);
- return ii !== -1 ? ii : null;
- }
- }
-}
-
-function localeWeekdaysParse (weekdayName, format, strict) {
- var i, mom, regex;
-
- if (this._weekdaysParseExact) {
- return handleStrictParse$1.call(this, weekdayName, format, strict);
- }
-
- if (!this._weekdaysParse) {
- this._weekdaysParse = [];
- this._minWeekdaysParse = [];
- this._shortWeekdaysParse = [];
- this._fullWeekdaysParse = [];
- }
-
- for (i = 0; i < 7; i++) {
- // make the regex if we don't have it already
-
- mom = createUTC([2000, 1]).day(i);
- if (strict && !this._fullWeekdaysParse[i]) {
- this._fullWeekdaysParse[i] = new RegExp('^' + this.weekdays(mom, '').replace('.', '\.?') + '$', 'i');
- this._shortWeekdaysParse[i] = new RegExp('^' + this.weekdaysShort(mom, '').replace('.', '\.?') + '$', 'i');
- this._minWeekdaysParse[i] = new RegExp('^' + this.weekdaysMin(mom, '').replace('.', '\.?') + '$', 'i');
- }
- if (!this._weekdaysParse[i]) {
- regex = '^' + this.weekdays(mom, '') + '|^' + this.weekdaysShort(mom, '') + '|^' + this.weekdaysMin(mom, '');
- this._weekdaysParse[i] = new RegExp(regex.replace('.', ''), 'i');
- }
- // test the regex
- if (strict && format === 'dddd' && this._fullWeekdaysParse[i].test(weekdayName)) {
- return i;
- } else if (strict && format === 'ddd' && this._shortWeekdaysParse[i].test(weekdayName)) {
- return i;
- } else if (strict && format === 'dd' && this._minWeekdaysParse[i].test(weekdayName)) {
- return i;
- } else if (!strict && this._weekdaysParse[i].test(weekdayName)) {
- return i;
- }
- }
-}
-
-// MOMENTS
-
-function getSetDayOfWeek (input) {
- if (!this.isValid()) {
- return input != null ? this : NaN;
- }
- var day = this._isUTC ? this._d.getUTCDay() : this._d.getDay();
- if (input != null) {
- input = parseWeekday(input, this.localeData());
- return this.add(input - day, 'd');
- } else {
- return day;
- }
-}
-
-function getSetLocaleDayOfWeek (input) {
- if (!this.isValid()) {
- return input != null ? this : NaN;
- }
- var weekday = (this.day() + 7 - this.localeData()._week.dow) % 7;
- return input == null ? weekday : this.add(input - weekday, 'd');
-}
-
-function getSetISODayOfWeek (input) {
- if (!this.isValid()) {
- return input != null ? this : NaN;
- }
-
- // behaves the same as moment#day except
- // as a getter, returns 7 instead of 0 (1-7 range instead of 0-6)
- // as a setter, sunday should belong to the previous week.
-
- if (input != null) {
- var weekday = parseIsoWeekday(input, this.localeData());
- return this.day(this.day() % 7 ? weekday : weekday - 7);
- } else {
- return this.day() || 7;
- }
-}
-
-var defaultWeekdaysRegex = matchWord;
-function weekdaysRegex (isStrict) {
- if (this._weekdaysParseExact) {
- if (!hasOwnProp(this, '_weekdaysRegex')) {
- computeWeekdaysParse.call(this);
- }
- if (isStrict) {
- return this._weekdaysStrictRegex;
- } else {
- return this._weekdaysRegex;
- }
- } else {
- if (!hasOwnProp(this, '_weekdaysRegex')) {
- this._weekdaysRegex = defaultWeekdaysRegex;
- }
- return this._weekdaysStrictRegex && isStrict ?
- this._weekdaysStrictRegex : this._weekdaysRegex;
- }
-}
-
-var defaultWeekdaysShortRegex = matchWord;
-function weekdaysShortRegex (isStrict) {
- if (this._weekdaysParseExact) {
- if (!hasOwnProp(this, '_weekdaysRegex')) {
- computeWeekdaysParse.call(this);
- }
- if (isStrict) {
- return this._weekdaysShortStrictRegex;
- } else {
- return this._weekdaysShortRegex;
- }
- } else {
- if (!hasOwnProp(this, '_weekdaysShortRegex')) {
- this._weekdaysShortRegex = defaultWeekdaysShortRegex;
- }
- return this._weekdaysShortStrictRegex && isStrict ?
- this._weekdaysShortStrictRegex : this._weekdaysShortRegex;
- }
-}
-
-var defaultWeekdaysMinRegex = matchWord;
-function weekdaysMinRegex (isStrict) {
- if (this._weekdaysParseExact) {
- if (!hasOwnProp(this, '_weekdaysRegex')) {
- computeWeekdaysParse.call(this);
- }
- if (isStrict) {
- return this._weekdaysMinStrictRegex;
- } else {
- return this._weekdaysMinRegex;
- }
- } else {
- if (!hasOwnProp(this, '_weekdaysMinRegex')) {
- this._weekdaysMinRegex = defaultWeekdaysMinRegex;
- }
- return this._weekdaysMinStrictRegex && isStrict ?
- this._weekdaysMinStrictRegex : this._weekdaysMinRegex;
- }
-}
-
-
-function computeWeekdaysParse () {
- function cmpLenRev(a, b) {
- return b.length - a.length;
- }
-
- var minPieces = [], shortPieces = [], longPieces = [], mixedPieces = [],
- i, mom, minp, shortp, longp;
- for (i = 0; i < 7; i++) {
- // make the regex if we don't have it already
- mom = createUTC([2000, 1]).day(i);
- minp = this.weekdaysMin(mom, '');
- shortp = this.weekdaysShort(mom, '');
- longp = this.weekdays(mom, '');
- minPieces.push(minp);
- shortPieces.push(shortp);
- longPieces.push(longp);
- mixedPieces.push(minp);
- mixedPieces.push(shortp);
- mixedPieces.push(longp);
- }
- // Sorting makes sure if one weekday (or abbr) is a prefix of another it
- // will match the longer piece.
- minPieces.sort(cmpLenRev);
- shortPieces.sort(cmpLenRev);
- longPieces.sort(cmpLenRev);
- mixedPieces.sort(cmpLenRev);
- for (i = 0; i < 7; i++) {
- shortPieces[i] = regexEscape(shortPieces[i]);
- longPieces[i] = regexEscape(longPieces[i]);
- mixedPieces[i] = regexEscape(mixedPieces[i]);
- }
-
- this._weekdaysRegex = new RegExp('^(' + mixedPieces.join('|') + ')', 'i');
- this._weekdaysShortRegex = this._weekdaysRegex;
- this._weekdaysMinRegex = this._weekdaysRegex;
-
- this._weekdaysStrictRegex = new RegExp('^(' + longPieces.join('|') + ')', 'i');
- this._weekdaysShortStrictRegex = new RegExp('^(' + shortPieces.join('|') + ')', 'i');
- this._weekdaysMinStrictRegex = new RegExp('^(' + minPieces.join('|') + ')', 'i');
-}
-
-// FORMATTING
-
-function hFormat() {
- return this.hours() % 12 || 12;
-}
-
-function kFormat() {
- return this.hours() || 24;
-}
-
-addFormatToken('H', ['HH', 2], 0, 'hour');
-addFormatToken('h', ['hh', 2], 0, hFormat);
-addFormatToken('k', ['kk', 2], 0, kFormat);
-
-addFormatToken('hmm', 0, 0, function () {
- return '' + hFormat.apply(this) + zeroFill(this.minutes(), 2);
-});
-
-addFormatToken('hmmss', 0, 0, function () {
- return '' + hFormat.apply(this) + zeroFill(this.minutes(), 2) +
- zeroFill(this.seconds(), 2);
-});
-
-addFormatToken('Hmm', 0, 0, function () {
- return '' + this.hours() + zeroFill(this.minutes(), 2);
-});
-
-addFormatToken('Hmmss', 0, 0, function () {
- return '' + this.hours() + zeroFill(this.minutes(), 2) +
- zeroFill(this.seconds(), 2);
-});
-
-function meridiem (token, lowercase) {
- addFormatToken(token, 0, 0, function () {
- return this.localeData().meridiem(this.hours(), this.minutes(), lowercase);
- });
-}
-
-meridiem('a', true);
-meridiem('A', false);
-
-// ALIASES
-
-addUnitAlias('hour', 'h');
-
-// PRIORITY
-addUnitPriority('hour', 13);
-
-// PARSING
-
-function matchMeridiem (isStrict, locale) {
- return locale._meridiemParse;
-}
-
-addRegexToken('a', matchMeridiem);
-addRegexToken('A', matchMeridiem);
-addRegexToken('H', match1to2);
-addRegexToken('h', match1to2);
-addRegexToken('HH', match1to2, match2);
-addRegexToken('hh', match1to2, match2);
-
-addRegexToken('hmm', match3to4);
-addRegexToken('hmmss', match5to6);
-addRegexToken('Hmm', match3to4);
-addRegexToken('Hmmss', match5to6);
-
-addParseToken(['H', 'HH'], HOUR);
-addParseToken(['a', 'A'], function (input, array, config) {
- config._isPm = config._locale.isPM(input);
- config._meridiem = input;
-});
-addParseToken(['h', 'hh'], function (input, array, config) {
- array[HOUR] = toInt(input);
- getParsingFlags(config).bigHour = true;
-});
-addParseToken('hmm', function (input, array, config) {
- var pos = input.length - 2;
- array[HOUR] = toInt(input.substr(0, pos));
- array[MINUTE] = toInt(input.substr(pos));
- getParsingFlags(config).bigHour = true;
-});
-addParseToken('hmmss', function (input, array, config) {
- var pos1 = input.length - 4;
- var pos2 = input.length - 2;
- array[HOUR] = toInt(input.substr(0, pos1));
- array[MINUTE] = toInt(input.substr(pos1, 2));
- array[SECOND] = toInt(input.substr(pos2));
- getParsingFlags(config).bigHour = true;
-});
-addParseToken('Hmm', function (input, array, config) {
- var pos = input.length - 2;
- array[HOUR] = toInt(input.substr(0, pos));
- array[MINUTE] = toInt(input.substr(pos));
-});
-addParseToken('Hmmss', function (input, array, config) {
- var pos1 = input.length - 4;
- var pos2 = input.length - 2;
- array[HOUR] = toInt(input.substr(0, pos1));
- array[MINUTE] = toInt(input.substr(pos1, 2));
- array[SECOND] = toInt(input.substr(pos2));
-});
-
-// LOCALES
-
-function localeIsPM (input) {
- // IE8 Quirks Mode & IE7 Standards Mode do not allow accessing strings like arrays
- // Using charAt should be more compatible.
- return ((input + '').toLowerCase().charAt(0) === 'p');
-}
-
-var defaultLocaleMeridiemParse = /[ap]\.?m?\.?/i;
-function localeMeridiem (hours, minutes, isLower) {
- if (hours > 11) {
- return isLower ? 'pm' : 'PM';
- } else {
- return isLower ? 'am' : 'AM';
- }
-}
-
-
-// MOMENTS
-
-// Setting the hour should keep the time, because the user explicitly
-// specified which hour he wants. So trying to maintain the same hour (in
-// a new timezone) makes sense. Adding/subtracting hours does not follow
-// this rule.
-var getSetHour = makeGetSet('Hours', true);
-
-// months
-// week
-// weekdays
-// meridiem
-var baseConfig = {
- calendar: defaultCalendar,
- longDateFormat: defaultLongDateFormat,
- invalidDate: defaultInvalidDate,
- ordinal: defaultOrdinal,
- ordinalParse: defaultOrdinalParse,
- relativeTime: defaultRelativeTime,
-
- months: defaultLocaleMonths,
- monthsShort: defaultLocaleMonthsShort,
-
- week: defaultLocaleWeek,
-
- weekdays: defaultLocaleWeekdays,
- weekdaysMin: defaultLocaleWeekdaysMin,
- weekdaysShort: defaultLocaleWeekdaysShort,
-
- meridiemParse: defaultLocaleMeridiemParse
-};
-
-// internal storage for locale config files
-var locales = {};
-var localeFamilies = {};
-var globalLocale;
-
-function normalizeLocale(key) {
- return key ? key.toLowerCase().replace('_', '-') : key;
-}
-
-// pick the locale from the array
-// try ['en-au', 'en-gb'] as 'en-au', 'en-gb', 'en', as in move through the list trying each
-// substring from most specific to least, but move to the next array item if it's a more specific variant than the current root
-function chooseLocale(names) {
- var i = 0, j, next, locale, split;
-
- while (i < names.length) {
- split = normalizeLocale(names[i]).split('-');
- j = split.length;
- next = normalizeLocale(names[i + 1]);
- next = next ? next.split('-') : null;
- while (j > 0) {
- locale = loadLocale(split.slice(0, j).join('-'));
- if (locale) {
- return locale;
- }
- if (next && next.length >= j && compareArrays(split, next, true) >= j - 1) {
- //the next array item is better than a shallower substring of this one
- break;
- }
- j--;
- }
- i++;
- }
- return null;
-}
-
-function loadLocale(name) {
- var oldLocale = null;
- // TODO: Find a better way to register and load all the locales in Node
- if (!locales[name] && (typeof module !== 'undefined') &&
- module && module.exports) {
- try {
- oldLocale = globalLocale._abbr;
- require('./locale/' + name);
- // because defineLocale currently also sets the global locale, we
- // want to undo that for lazy loaded locales
- getSetGlobalLocale(oldLocale);
- } catch (e) { }
- }
- return locales[name];
-}
-
-// This function will load locale and then set the global locale. If
-// no arguments are passed in, it will simply return the current global
-// locale key.
-function getSetGlobalLocale (key, values) {
- var data;
- if (key) {
- if (isUndefined(values)) {
- data = getLocale(key);
- }
- else {
- data = defineLocale(key, values);
- }
-
- if (data) {
- // moment.duration._locale = moment._locale = data;
- globalLocale = data;
- }
- }
-
- return globalLocale._abbr;
-}
-
-function defineLocale (name, config) {
- if (config !== null) {
- var parentConfig = baseConfig;
- config.abbr = name;
- if (locales[name] != null) {
- deprecateSimple('defineLocaleOverride',
- 'use moment.updateLocale(localeName, config) to change ' +
- 'an existing locale. moment.defineLocale(localeName, ' +
- 'config) should only be used for creating a new locale ' +
- 'See http://momentjs.com/guides/#/warnings/define-locale/ for more info.');
- parentConfig = locales[name]._config;
- } else if (config.parentLocale != null) {
- if (locales[config.parentLocale] != null) {
- parentConfig = locales[config.parentLocale]._config;
- } else {
- if (!localeFamilies[config.parentLocale]) {
- localeFamilies[config.parentLocale] = [];
- }
- localeFamilies[config.parentLocale].push({
- name: name,
- config: config
- });
- return null;
- }
- }
- locales[name] = new Locale(mergeConfigs(parentConfig, config));
-
- if (localeFamilies[name]) {
- localeFamilies[name].forEach(function (x) {
- defineLocale(x.name, x.config);
- });
- }
-
- // backwards compat for now: also set the locale
- // make sure we set the locale AFTER all child locales have been
- // created, so we won't end up with the child locale set.
- getSetGlobalLocale(name);
-
-
- return locales[name];
- } else {
- // useful for testing
- delete locales[name];
- return null;
- }
-}
-
-function updateLocale(name, config) {
- if (config != null) {
- var locale, parentConfig = baseConfig;
- // MERGE
- if (locales[name] != null) {
- parentConfig = locales[name]._config;
- }
- config = mergeConfigs(parentConfig, config);
- locale = new Locale(config);
- locale.parentLocale = locales[name];
- locales[name] = locale;
-
- // backwards compat for now: also set the locale
- getSetGlobalLocale(name);
- } else {
- // pass null for config to unupdate, useful for tests
- if (locales[name] != null) {
- if (locales[name].parentLocale != null) {
- locales[name] = locales[name].parentLocale;
- } else if (locales[name] != null) {
- delete locales[name];
- }
- }
- }
- return locales[name];
-}
-
-// returns locale data
-function getLocale (key) {
- var locale;
-
- if (key && key._locale && key._locale._abbr) {
- key = key._locale._abbr;
- }
-
- if (!key) {
- return globalLocale;
- }
-
- if (!isArray(key)) {
- //short-circuit everything else
- locale = loadLocale(key);
- if (locale) {
- return locale;
- }
- key = [key];
- }
-
- return chooseLocale(key);
-}
-
-function listLocales() {
- return keys$1(locales);
-}
-
-function checkOverflow (m) {
- var overflow;
- var a = m._a;
-
- if (a && getParsingFlags(m).overflow === -2) {
- overflow =
- a[MONTH] < 0 || a[MONTH] > 11 ? MONTH :
- a[DATE] < 1 || a[DATE] > daysInMonth(a[YEAR], a[MONTH]) ? DATE :
- a[HOUR] < 0 || a[HOUR] > 24 || (a[HOUR] === 24 && (a[MINUTE] !== 0 || a[SECOND] !== 0 || a[MILLISECOND] !== 0)) ? HOUR :
- a[MINUTE] < 0 || a[MINUTE] > 59 ? MINUTE :
- a[SECOND] < 0 || a[SECOND] > 59 ? SECOND :
- a[MILLISECOND] < 0 || a[MILLISECOND] > 999 ? MILLISECOND :
- -1;
-
- if (getParsingFlags(m)._overflowDayOfYear && (overflow < YEAR || overflow > DATE)) {
- overflow = DATE;
- }
- if (getParsingFlags(m)._overflowWeeks && overflow === -1) {
- overflow = WEEK;
- }
- if (getParsingFlags(m)._overflowWeekday && overflow === -1) {
- overflow = WEEKDAY;
- }
-
- getParsingFlags(m).overflow = overflow;
- }
-
- return m;
-}
-
-// iso 8601 regex
-// 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)
-var 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)?)?$/;
-var 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)?)?$/;
-
-var tzRegex = /Z|[+-]\d\d(?::?\d\d)?/;
-
-var isoDates = [
- ['YYYYYY-MM-DD', /[+-]\d{6}-\d\d-\d\d/],
- ['YYYY-MM-DD', /\d{4}-\d\d-\d\d/],
- ['GGGG-[W]WW-E', /\d{4}-W\d\d-\d/],
- ['GGGG-[W]WW', /\d{4}-W\d\d/, false],
- ['YYYY-DDD', /\d{4}-\d{3}/],
- ['YYYY-MM', /\d{4}-\d\d/, false],
- ['YYYYYYMMDD', /[+-]\d{10}/],
- ['YYYYMMDD', /\d{8}/],
- // YYYYMM is NOT allowed by the standard
- ['GGGG[W]WWE', /\d{4}W\d{3}/],
- ['GGGG[W]WW', /\d{4}W\d{2}/, false],
- ['YYYYDDD', /\d{7}/]
-];
-
-// iso time formats and regexes
-var isoTimes = [
- ['HH:mm:ss.SSSS', /\d\d:\d\d:\d\d\.\d+/],
- ['HH:mm:ss,SSSS', /\d\d:\d\d:\d\d,\d+/],
- ['HH:mm:ss', /\d\d:\d\d:\d\d/],
- ['HH:mm', /\d\d:\d\d/],
- ['HHmmss.SSSS', /\d\d\d\d\d\d\.\d+/],
- ['HHmmss,SSSS', /\d\d\d\d\d\d,\d+/],
- ['HHmmss', /\d\d\d\d\d\d/],
- ['HHmm', /\d\d\d\d/],
- ['HH', /\d\d/]
-];
-
-var aspNetJsonRegex = /^\/?Date\((\-?\d+)/i;
-
-// date from iso format
-function configFromISO(config) {
- var i, l,
- string = config._i,
- match = extendedIsoRegex.exec(string) || basicIsoRegex.exec(string),
- allowTime, dateFormat, timeFormat, tzFormat;
-
- if (match) {
- getParsingFlags(config).iso = true;
-
- for (i = 0, l = isoDates.length; i < l; i++) {
- if (isoDates[i][1].exec(match[1])) {
- dateFormat = isoDates[i][0];
- allowTime = isoDates[i][2] !== false;
- break;
- }
- }
- if (dateFormat == null) {
- config._isValid = false;
- return;
- }
- if (match[3]) {
- for (i = 0, l = isoTimes.length; i < l; i++) {
- if (isoTimes[i][1].exec(match[3])) {
- // match[2] should be 'T' or space
- timeFormat = (match[2] || ' ') + isoTimes[i][0];
- break;
- }
- }
- if (timeFormat == null) {
- config._isValid = false;
- return;
- }
- }
- if (!allowTime && timeFormat != null) {
- config._isValid = false;
- return;
- }
- if (match[4]) {
- if (tzRegex.exec(match[4])) {
- tzFormat = 'Z';
- } else {
- config._isValid = false;
- return;
- }
- }
- config._f = dateFormat + (timeFormat || '') + (tzFormat || '');
- configFromStringAndFormat(config);
- } else {
- config._isValid = false;
- }
-}
-
-// date from iso format or fallback
-function configFromString(config) {
- var matched = aspNetJsonRegex.exec(config._i);
-
- if (matched !== null) {
- config._d = new Date(+matched[1]);
- return;
- }
-
- configFromISO(config);
- if (config._isValid === false) {
- delete config._isValid;
- hooks.createFromInputFallback(config);
- }
-}
-
-hooks.createFromInputFallback = deprecate(
- 'value provided is not in a recognized ISO format. moment construction falls back to js Date(), ' +
- 'which is not reliable across all browsers and versions. Non ISO date formats are ' +
- 'discouraged and will be removed in an upcoming major release. Please refer to ' +
- 'http://momentjs.com/guides/#/warnings/js-date/ for more info.',
- function (config) {
- config._d = new Date(config._i + (config._useUTC ? ' UTC' : ''));
- }
-);
-
-// Pick the first defined of two or three arguments.
-function defaults(a, b, c) {
- if (a != null) {
- return a;
- }
- if (b != null) {
- return b;
- }
- return c;
-}
-
-function currentDateArray(config) {
- // hooks is actually the exported moment object
- var nowValue = new Date(hooks.now());
- if (config._useUTC) {
- return [nowValue.getUTCFullYear(), nowValue.getUTCMonth(), nowValue.getUTCDate()];
- }
- return [nowValue.getFullYear(), nowValue.getMonth(), nowValue.getDate()];
-}
-
-// convert an array to a date.
-// the array should mirror the parameters below
-// note: all values past the year are optional and will default to the lowest possible value.
-// [year, month, day , hour, minute, second, millisecond]
-function configFromArray (config) {
- var i, date, input = [], currentDate, yearToUse;
-
- if (config._d) {
- return;
- }
-
- currentDate = currentDateArray(config);
-
- //compute day of the year from weeks and weekdays
- if (config._w && config._a[DATE] == null && config._a[MONTH] == null) {
- dayOfYearFromWeekInfo(config);
- }
-
- //if the day of the year is set, figure out what it is
- if (config._dayOfYear) {
- yearToUse = defaults(config._a[YEAR], currentDate[YEAR]);
-
- if (config._dayOfYear > daysInYear(yearToUse)) {
- getParsingFlags(config)._overflowDayOfYear = true;
- }
-
- date = createUTCDate(yearToUse, 0, config._dayOfYear);
- config._a[MONTH] = date.getUTCMonth();
- config._a[DATE] = date.getUTCDate();
- }
-
- // Default to current date.
- // * if no year, month, day of month are given, default to today
- // * if day of month is given, default month and year
- // * if month is given, default only year
- // * if year is given, don't default anything
- for (i = 0; i < 3 && config._a[i] == null; ++i) {
- config._a[i] = input[i] = currentDate[i];
- }
-
- // Zero out whatever was not defaulted, including time
- for (; i < 7; i++) {
- config._a[i] = input[i] = (config._a[i] == null) ? (i === 2 ? 1 : 0) : config._a[i];
- }
-
- // Check for 24:00:00.000
- if (config._a[HOUR] === 24 &&
- config._a[MINUTE] === 0 &&
- config._a[SECOND] === 0 &&
- config._a[MILLISECOND] === 0) {
- config._nextDay = true;
- config._a[HOUR] = 0;
- }
-
- config._d = (config._useUTC ? createUTCDate : createDate).apply(null, input);
- // Apply timezone offset from input. The actual utcOffset can be changed
- // with parseZone.
- if (config._tzm != null) {
- config._d.setUTCMinutes(config._d.getUTCMinutes() - config._tzm);
- }
-
- if (config._nextDay) {
- config._a[HOUR] = 24;
- }
-}
-
-function dayOfYearFromWeekInfo(config) {
- var w, weekYear, week, weekday, dow, doy, temp, weekdayOverflow;
-
- w = config._w;
- if (w.GG != null || w.W != null || w.E != null) {
- dow = 1;
- doy = 4;
-
- // TODO: We need to take the current isoWeekYear, but that depends on
- // how we interpret now (local, utc, fixed offset). So create
- // a now version of current config (take local/utc/offset flags, and
- // create now).
- weekYear = defaults(w.GG, config._a[YEAR], weekOfYear(createLocal(), 1, 4).year);
- week = defaults(w.W, 1);
- weekday = defaults(w.E, 1);
- if (weekday < 1 || weekday > 7) {
- weekdayOverflow = true;
- }
- } else {
- dow = config._locale._week.dow;
- doy = config._locale._week.doy;
-
- var curWeek = weekOfYear(createLocal(), dow, doy);
-
- weekYear = defaults(w.gg, config._a[YEAR], curWeek.year);
-
- // Default to current week.
- week = defaults(w.w, curWeek.week);
-
- if (w.d != null) {
- // weekday -- low day numbers are considered next week
- weekday = w.d;
- if (weekday < 0 || weekday > 6) {
- weekdayOverflow = true;
- }
- } else if (w.e != null) {
- // local weekday -- counting starts from begining of week
- weekday = w.e + dow;
- if (w.e < 0 || w.e > 6) {
- weekdayOverflow = true;
- }
- } else {
- // default to begining of week
- weekday = dow;
- }
- }
- if (week < 1 || week > weeksInYear(weekYear, dow, doy)) {
- getParsingFlags(config)._overflowWeeks = true;
- } else if (weekdayOverflow != null) {
- getParsingFlags(config)._overflowWeekday = true;
- } else {
- temp = dayOfYearFromWeeks(weekYear, week, weekday, dow, doy);
- config._a[YEAR] = temp.year;
- config._dayOfYear = temp.dayOfYear;
- }
-}
-
-// constant that refers to the ISO standard
-hooks.ISO_8601 = function () {};
-
-// date from string and format string
-function configFromStringAndFormat(config) {
- // TODO: Move this to another part of the creation flow to prevent circular deps
- if (config._f === hooks.ISO_8601) {
- configFromISO(config);
- return;
- }
-
- config._a = [];
- getParsingFlags(config).empty = true;
-
- // This array is used to make a Date, either with `new Date` or `Date.UTC`
- var string = '' + config._i,
- i, parsedInput, tokens, token, skipped,
- stringLength = string.length,
- totalParsedInputLength = 0;
-
- tokens = expandFormat(config._f, config._locale).match(formattingTokens) || [];
-
- for (i = 0; i < tokens.length; i++) {
- token = tokens[i];
- parsedInput = (string.match(getParseRegexForToken(token, config)) || [])[0];
- // console.log('token', token, 'parsedInput', parsedInput,
- // 'regex', getParseRegexForToken(token, config));
- if (parsedInput) {
- skipped = string.substr(0, string.indexOf(parsedInput));
- if (skipped.length > 0) {
- getParsingFlags(config).unusedInput.push(skipped);
- }
- string = string.slice(string.indexOf(parsedInput) + parsedInput.length);
- totalParsedInputLength += parsedInput.length;
- }
- // don't parse if it's not a known token
- if (formatTokenFunctions[token]) {
- if (parsedInput) {
- getParsingFlags(config).empty = false;
- }
- else {
- getParsingFlags(config).unusedTokens.push(token);
- }
- addTimeToArrayFromToken(token, parsedInput, config);
- }
- else if (config._strict && !parsedInput) {
- getParsingFlags(config).unusedTokens.push(token);
- }
- }
-
- // add remaining unparsed input length to the string
- getParsingFlags(config).charsLeftOver = stringLength - totalParsedInputLength;
- if (string.length > 0) {
- getParsingFlags(config).unusedInput.push(string);
- }
-
- // clear _12h flag if hour is <= 12
- if (config._a[HOUR] <= 12 &&
- getParsingFlags(config).bigHour === true &&
- config._a[HOUR] > 0) {
- getParsingFlags(config).bigHour = undefined;
- }
-
- getParsingFlags(config).parsedDateParts = config._a.slice(0);
- getParsingFlags(config).meridiem = config._meridiem;
- // handle meridiem
- config._a[HOUR] = meridiemFixWrap(config._locale, config._a[HOUR], config._meridiem);
-
- configFromArray(config);
- checkOverflow(config);
-}
-
-
-function meridiemFixWrap (locale, hour, meridiem) {
- var isPm;
-
- if (meridiem == null) {
- // nothing to do
- return hour;
- }
- if (locale.meridiemHour != null) {
- return locale.meridiemHour(hour, meridiem);
- } else if (locale.isPM != null) {
- // Fallback
- isPm = locale.isPM(meridiem);
- if (isPm && hour < 12) {
- hour += 12;
- }
- if (!isPm && hour === 12) {
- hour = 0;
- }
- return hour;
- } else {
- // this is not supposed to happen
- return hour;
- }
-}
-
-// date from string and array of format strings
-function configFromStringAndArray(config) {
- var tempConfig,
- bestMoment,
-
- scoreToBeat,
- i,
- currentScore;
-
- if (config._f.length === 0) {
- getParsingFlags(config).invalidFormat = true;
- config._d = new Date(NaN);
- return;
- }
-
- for (i = 0; i < config._f.length; i++) {
- currentScore = 0;
- tempConfig = copyConfig({}, config);
- if (config._useUTC != null) {
- tempConfig._useUTC = config._useUTC;
- }
- tempConfig._f = config._f[i];
- configFromStringAndFormat(tempConfig);
-
- if (!isValid(tempConfig)) {
- continue;
- }
-
- // if there is any input that was not parsed add a penalty for that format
- currentScore += getParsingFlags(tempConfig).charsLeftOver;
-
- //or tokens
- currentScore += getParsingFlags(tempConfig).unusedTokens.length * 10;
-
- getParsingFlags(tempConfig).score = currentScore;
-
- if (scoreToBeat == null || currentScore < scoreToBeat) {
- scoreToBeat = currentScore;
- bestMoment = tempConfig;
- }
- }
-
- extend(config, bestMoment || tempConfig);
-}
-
-function configFromObject(config) {
- if (config._d) {
- return;
- }
-
- var i = normalizeObjectUnits(config._i);
- config._a = map([i.year, i.month, i.day || i.date, i.hour, i.minute, i.second, i.millisecond], function (obj) {
- return obj && parseInt(obj, 10);
- });
-
- configFromArray(config);
-}
-
-function createFromConfig (config) {
- var res = new Moment(checkOverflow(prepareConfig(config)));
- if (res._nextDay) {
- // Adding is smart enough around DST
- res.add(1, 'd');
- res._nextDay = undefined;
- }
-
- return res;
-}
-
-function prepareConfig (config) {
- var input = config._i,
- format = config._f;
-
- config._locale = config._locale || getLocale(config._l);
-
- if (input === null || (format === undefined && input === '')) {
- return createInvalid({nullInput: true});
- }
-
- if (typeof input === 'string') {
- config._i = input = config._locale.preparse(input);
- }
-
- if (isMoment(input)) {
- return new Moment(checkOverflow(input));
- } else if (isDate(input)) {
- config._d = input;
- } else if (isArray(format)) {
- configFromStringAndArray(config);
- } else if (format) {
- configFromStringAndFormat(config);
- } else {
- configFromInput(config);
- }
-
- if (!isValid(config)) {
- config._d = null;
- }
-
- return config;
-}
-
-function configFromInput(config) {
- var input = config._i;
- if (input === undefined) {
- config._d = new Date(hooks.now());
- } else if (isDate(input)) {
- config._d = new Date(input.valueOf());
- } else if (typeof input === 'string') {
- configFromString(config);
- } else if (isArray(input)) {
- config._a = map(input.slice(0), function (obj) {
- return parseInt(obj, 10);
- });
- configFromArray(config);
- } else if (typeof(input) === 'object') {
- configFromObject(config);
- } else if (isNumber(input)) {
- // from milliseconds
- config._d = new Date(input);
- } else {
- hooks.createFromInputFallback(config);
- }
-}
-
-function createLocalOrUTC (input, format, locale, strict, isUTC) {
- var c = {};
-
- if (locale === true || locale === false) {
- strict = locale;
- locale = undefined;
- }
-
- if ((isObject(input) && isObjectEmpty(input)) ||
- (isArray(input) && input.length === 0)) {
- input = undefined;
- }
- // object construction must be done this way.
- // https://github.com/moment/moment/issues/1423
- c._isAMomentObject = true;
- c._useUTC = c._isUTC = isUTC;
- c._l = locale;
- c._i = input;
- c._f = format;
- c._strict = strict;
-
- return createFromConfig(c);
-}
-
-function createLocal (input, format, locale, strict) {
- return createLocalOrUTC(input, format, locale, strict, false);
-}
-
-var prototypeMin = deprecate(
- 'moment().min is deprecated, use moment.max instead. http://momentjs.com/guides/#/warnings/min-max/',
- function () {
- var other = createLocal.apply(null, arguments);
- if (this.isValid() && other.isValid()) {
- return other < this ? this : other;
- } else {
- return createInvalid();
- }
- }
-);
-
-var prototypeMax = deprecate(
- 'moment().max is deprecated, use moment.min instead. http://momentjs.com/guides/#/warnings/min-max/',
- function () {
- var other = createLocal.apply(null, arguments);
- if (this.isValid() && other.isValid()) {
- return other > this ? this : other;
- } else {
- return createInvalid();
- }
- }
-);
-
-// Pick a moment m from moments so that m[fn](other) is true for all
-// other. This relies on the function fn to be transitive.
-//
-// moments should either be an array of moment objects or an array, whose
-// first element is an array of moment objects.
-function pickBy(fn, moments) {
- var res, i;
- if (moments.length === 1 && isArray(moments[0])) {
- moments = moments[0];
- }
- if (!moments.length) {
- return createLocal();
- }
- res = moments[0];
- for (i = 1; i < moments.length; ++i) {
- if (!moments[i].isValid() || moments[i][fn](res)) {
- res = moments[i];
- }
- }
- return res;
-}
-
-// TODO: Use [].sort instead?
-function min () {
- var args = [].slice.call(arguments, 0);
-
- return pickBy('isBefore', args);
-}
-
-function max () {
- var args = [].slice.call(arguments, 0);
-
- return pickBy('isAfter', args);
-}
-
-var now = function () {
- return Date.now ? Date.now() : +(new Date());
-};
-
-function Duration (duration) {
- var normalizedInput = normalizeObjectUnits(duration),
- years = normalizedInput.year || 0,
- quarters = normalizedInput.quarter || 0,
- months = normalizedInput.month || 0,
- weeks = normalizedInput.week || 0,
- days = normalizedInput.day || 0,
- hours = normalizedInput.hour || 0,
- minutes = normalizedInput.minute || 0,
- seconds = normalizedInput.second || 0,
- milliseconds = normalizedInput.millisecond || 0;
-
- // representation for dateAddRemove
- this._milliseconds = +milliseconds +
- seconds * 1e3 + // 1000
- minutes * 6e4 + // 1000 * 60
- hours * 1000 * 60 * 60; //using 1000 * 60 * 60 instead of 36e5 to avoid floating point rounding errors https://github.com/moment/moment/issues/2978
- // Because of dateAddRemove treats 24 hours as different from a
- // day when working around DST, we need to store them separately
- this._days = +days +
- weeks * 7;
- // It is impossible translate months into days without knowing
- // which months you are are talking about, so we have to store
- // it separately.
- this._months = +months +
- quarters * 3 +
- years * 12;
-
- this._data = {};
-
- this._locale = getLocale();
-
- this._bubble();
-}
-
-function isDuration (obj) {
- return obj instanceof Duration;
-}
-
-function absRound (number) {
- if (number < 0) {
- return Math.round(-1 * number) * -1;
- } else {
- return Math.round(number);
- }
-}
-
-// FORMATTING
-
-function offset (token, separator) {
- addFormatToken(token, 0, 0, function () {
- var offset = this.utcOffset();
- var sign = '+';
- if (offset < 0) {
- offset = -offset;
- sign = '-';
- }
- return sign + zeroFill(~~(offset / 60), 2) + separator + zeroFill(~~(offset) % 60, 2);
- });
-}
-
-offset('Z', ':');
-offset('ZZ', '');
-
-// PARSING
-
-addRegexToken('Z', matchShortOffset);
-addRegexToken('ZZ', matchShortOffset);
-addParseToken(['Z', 'ZZ'], function (input, array, config) {
- config._useUTC = true;
- config._tzm = offsetFromString(matchShortOffset, input);
-});
-
-// HELPERS
-
-// timezone chunker
-// '+10:00' > ['10', '00']
-// '-1530' > ['-15', '30']
-var chunkOffset = /([\+\-]|\d\d)/gi;
-
-function offsetFromString(matcher, string) {
- var matches = (string || '').match(matcher);
-
- if (matches === null) {
- return null;
- }
-
- var chunk = matches[matches.length - 1] || [];
- var parts = (chunk + '').match(chunkOffset) || ['-', 0, 0];
- var minutes = +(parts[1] * 60) + toInt(parts[2]);
-
- return minutes === 0 ?
- 0 :
- parts[0] === '+' ? minutes : -minutes;
-}
-
-// Return a moment from input, that is local/utc/zone equivalent to model.
-function cloneWithOffset(input, model) {
- var res, diff;
- if (model._isUTC) {
- res = model.clone();
- diff = (isMoment(input) || isDate(input) ? input.valueOf() : createLocal(input).valueOf()) - res.valueOf();
- // Use low-level api, because this fn is low-level api.
- res._d.setTime(res._d.valueOf() + diff);
- hooks.updateOffset(res, false);
- return res;
- } else {
- return createLocal(input).local();
- }
-}
-
-function getDateOffset (m) {
- // On Firefox.24 Date#getTimezoneOffset returns a floating point.
- // https://github.com/moment/moment/pull/1871
- return -Math.round(m._d.getTimezoneOffset() / 15) * 15;
-}
-
-// HOOKS
-
-// This function will be called whenever a moment is mutated.
-// It is intended to keep the offset in sync with the timezone.
-hooks.updateOffset = function () {};
-
-// MOMENTS
-
-// keepLocalTime = true means only change the timezone, without
-// affecting the local hour. So 5:31:26 +0300 --[utcOffset(2, true)]-->
-// 5:31:26 +0200 It is possible that 5:31:26 doesn't exist with offset
-// +0200, so we adjust the time as needed, to be valid.
-//
-// Keeping the time actually adds/subtracts (one hour)
-// from the actual represented time. That is why we call updateOffset
-// a second time. In case it wants us to change the offset again
-// _changeInProgress == true case, then we have to adjust, because
-// there is no such time in the given timezone.
-function getSetOffset (input, keepLocalTime) {
- var offset = this._offset || 0,
- localAdjust;
- if (!this.isValid()) {
- return input != null ? this : NaN;
- }
- if (input != null) {
- if (typeof input === 'string') {
- input = offsetFromString(matchShortOffset, input);
- if (input === null) {
- return this;
- }
- } else if (Math.abs(input) < 16) {
- input = input * 60;
- }
- if (!this._isUTC && keepLocalTime) {
- localAdjust = getDateOffset(this);
- }
- this._offset = input;
- this._isUTC = true;
- if (localAdjust != null) {
- this.add(localAdjust, 'm');
- }
- if (offset !== input) {
- if (!keepLocalTime || this._changeInProgress) {
- addSubtract(this, createDuration(input - offset, 'm'), 1, false);
- } else if (!this._changeInProgress) {
- this._changeInProgress = true;
- hooks.updateOffset(this, true);
- this._changeInProgress = null;
- }
- }
- return this;
- } else {
- return this._isUTC ? offset : getDateOffset(this);
- }
-}
-
-function getSetZone (input, keepLocalTime) {
- if (input != null) {
- if (typeof input !== 'string') {
- input = -input;
- }
-
- this.utcOffset(input, keepLocalTime);
-
- return this;
- } else {
- return -this.utcOffset();
- }
-}
-
-function setOffsetToUTC (keepLocalTime) {
- return this.utcOffset(0, keepLocalTime);
-}
-
-function setOffsetToLocal (keepLocalTime) {
- if (this._isUTC) {
- this.utcOffset(0, keepLocalTime);
- this._isUTC = false;
-
- if (keepLocalTime) {
- this.subtract(getDateOffset(this), 'm');
- }
- }
- return this;
-}
-
-function setOffsetToParsedOffset () {
- if (this._tzm != null) {
- this.utcOffset(this._tzm);
- } else if (typeof this._i === 'string') {
- var tZone = offsetFromString(matchOffset, this._i);
- if (tZone != null) {
- this.utcOffset(tZone);
- }
- else {
- this.utcOffset(0, true);
- }
- }
- return this;
-}
-
-function hasAlignedHourOffset (input) {
- if (!this.isValid()) {
- return false;
- }
- input = input ? createLocal(input).utcOffset() : 0;
-
- return (this.utcOffset() - input) % 60 === 0;
-}
-
-function isDaylightSavingTime () {
- return (
- this.utcOffset() > this.clone().month(0).utcOffset() ||
- this.utcOffset() > this.clone().month(5).utcOffset()
- );
-}
-
-function isDaylightSavingTimeShifted () {
- if (!isUndefined(this._isDSTShifted)) {
- return this._isDSTShifted;
- }
-
- var c = {};
-
- copyConfig(c, this);
- c = prepareConfig(c);
-
- if (c._a) {
- var other = c._isUTC ? createUTC(c._a) : createLocal(c._a);
- this._isDSTShifted = this.isValid() &&
- compareArrays(c._a, other.toArray()) > 0;
- } else {
- this._isDSTShifted = false;
- }
-
- return this._isDSTShifted;
-}
-
-function isLocal () {
- return this.isValid() ? !this._isUTC : false;
-}
-
-function isUtcOffset () {
- return this.isValid() ? this._isUTC : false;
-}
-
-function isUtc () {
- return this.isValid() ? this._isUTC && this._offset === 0 : false;
-}
-
-// ASP.NET json date format regex
-var aspNetRegex = /^(\-)?(?:(\d*)[. ])?(\d+)\:(\d+)(?:\:(\d+)(\.\d*)?)?$/;
-
-// from http://docs.closure-library.googlecode.com/git/closure_goog_date_date.js.source.html
-// somewhat more in line with 4.4.3.2 2004 spec, but allows decimal anywhere
-// and further modified to allow for strings containing both week and day
-var isoRegex = /^(-)?P(?:(-?[0-9,.]*)Y)?(?:(-?[0-9,.]*)M)?(?:(-?[0-9,.]*)W)?(?:(-?[0-9,.]*)D)?(?:T(?:(-?[0-9,.]*)H)?(?:(-?[0-9,.]*)M)?(?:(-?[0-9,.]*)S)?)?$/;
-
-function createDuration (input, key) {
- var duration = input,
- // matching against regexp is expensive, do it on demand
- match = null,
- sign,
- ret,
- diffRes;
-
- if (isDuration(input)) {
- duration = {
- ms : input._milliseconds,
- d : input._days,
- M : input._months
- };
- } else if (isNumber(input)) {
- duration = {};
- if (key) {
- duration[key] = input;
- } else {
- duration.milliseconds = input;
- }
- } else if (!!(match = aspNetRegex.exec(input))) {
- sign = (match[1] === '-') ? -1 : 1;
- duration = {
- y : 0,
- d : toInt(match[DATE]) * sign,
- h : toInt(match[HOUR]) * sign,
- m : toInt(match[MINUTE]) * sign,
- s : toInt(match[SECOND]) * sign,
- ms : toInt(absRound(match[MILLISECOND] * 1000)) * sign // the millisecond decimal point is included in the match
- };
- } else if (!!(match = isoRegex.exec(input))) {
- sign = (match[1] === '-') ? -1 : 1;
- duration = {
- y : parseIso(match[2], sign),
- M : parseIso(match[3], sign),
- w : parseIso(match[4], sign),
- d : parseIso(match[5], sign),
- h : parseIso(match[6], sign),
- m : parseIso(match[7], sign),
- s : parseIso(match[8], sign)
- };
- } else if (duration == null) {// checks for null or undefined
- duration = {};
- } else if (typeof duration === 'object' && ('from' in duration || 'to' in duration)) {
- diffRes = momentsDifference(createLocal(duration.from), createLocal(duration.to));
-
- duration = {};
- duration.ms = diffRes.milliseconds;
- duration.M = diffRes.months;
- }
-
- ret = new Duration(duration);
-
- if (isDuration(input) && hasOwnProp(input, '_locale')) {
- ret._locale = input._locale;
- }
-
- return ret;
-}
-
-createDuration.fn = Duration.prototype;
-
-function parseIso (inp, sign) {
- // We'd normally use ~~inp for this, but unfortunately it also
- // converts floats to ints.
- // inp may be undefined, so careful calling replace on it.
- var res = inp && parseFloat(inp.replace(',', '.'));
- // apply sign while we're at it
- return (isNaN(res) ? 0 : res) * sign;
-}
-
-function positiveMomentsDifference(base, other) {
- var res = {milliseconds: 0, months: 0};
-
- res.months = other.month() - base.month() +
- (other.year() - base.year()) * 12;
- if (base.clone().add(res.months, 'M').isAfter(other)) {
- --res.months;
- }
-
- res.milliseconds = +other - +(base.clone().add(res.months, 'M'));
-
- return res;
-}
-
-function momentsDifference(base, other) {
- var res;
- if (!(base.isValid() && other.isValid())) {
- return {milliseconds: 0, months: 0};
- }
-
- other = cloneWithOffset(other, base);
- if (base.isBefore(other)) {
- res = positiveMomentsDifference(base, other);
- } else {
- res = positiveMomentsDifference(other, base);
- res.milliseconds = -res.milliseconds;
- res.months = -res.months;
- }
-
- return res;
-}
-
-// TODO: remove 'name' arg after deprecation is removed
-function createAdder(direction, name) {
- return function (val, period) {
- var dur, tmp;
- //invert the arguments, but complain about it
- if (period !== null && !isNaN(+period)) {
- deprecateSimple(name, 'moment().' + name + '(period, number) is deprecated. Please use moment().' + name + '(number, period). ' +
- 'See http://momentjs.com/guides/#/warnings/add-inverted-param/ for more info.');
- tmp = val; val = period; period = tmp;
- }
-
- val = typeof val === 'string' ? +val : val;
- dur = createDuration(val, period);
- addSubtract(this, dur, direction);
- return this;
- };
-}
-
-function addSubtract (mom, duration, isAdding, updateOffset) {
- var milliseconds = duration._milliseconds,
- days = absRound(duration._days),
- months = absRound(duration._months);
-
- if (!mom.isValid()) {
- // No op
- return;
- }
-
- updateOffset = updateOffset == null ? true : updateOffset;
-
- if (milliseconds) {
- mom._d.setTime(mom._d.valueOf() + milliseconds * isAdding);
- }
- if (days) {
- set$1(mom, 'Date', get(mom, 'Date') + days * isAdding);
- }
- if (months) {
- setMonth(mom, get(mom, 'Month') + months * isAdding);
- }
- if (updateOffset) {
- hooks.updateOffset(mom, days || months);
- }
-}
-
-var add = createAdder(1, 'add');
-var subtract = createAdder(-1, 'subtract');
-
-function getCalendarFormat(myMoment, now) {
- var diff = myMoment.diff(now, 'days', true);
- return diff < -6 ? 'sameElse' :
- diff < -1 ? 'lastWeek' :
- diff < 0 ? 'lastDay' :
- diff < 1 ? 'sameDay' :
- diff < 2 ? 'nextDay' :
- diff < 7 ? 'nextWeek' : 'sameElse';
-}
-
-function calendar$1 (time, formats) {
- // We want to compare the start of today, vs this.
- // Getting start-of-today depends on whether we're local/utc/offset or not.
- var now = time || createLocal(),
- sod = cloneWithOffset(now, this).startOf('day'),
- format = hooks.calendarFormat(this, sod) || 'sameElse';
-
- var output = formats && (isFunction(formats[format]) ? formats[format].call(this, now) : formats[format]);
-
- return this.format(output || this.localeData().calendar(format, this, createLocal(now)));
-}
-
-function clone () {
- return new Moment(this);
-}
-
-function isAfter (input, units) {
- var localInput = isMoment(input) ? input : createLocal(input);
- if (!(this.isValid() && localInput.isValid())) {
- return false;
- }
- units = normalizeUnits(!isUndefined(units) ? units : 'millisecond');
- if (units === 'millisecond') {
- return this.valueOf() > localInput.valueOf();
- } else {
- return localInput.valueOf() < this.clone().startOf(units).valueOf();
- }
-}
-
-function isBefore (input, units) {
- var localInput = isMoment(input) ? input : createLocal(input);
- if (!(this.isValid() && localInput.isValid())) {
- return false;
- }
- units = normalizeUnits(!isUndefined(units) ? units : 'millisecond');
- if (units === 'millisecond') {
- return this.valueOf() < localInput.valueOf();
- } else {
- return this.clone().endOf(units).valueOf() < localInput.valueOf();
- }
-}
-
-function isBetween (from, to, units, inclusivity) {
- inclusivity = inclusivity || '()';
- return (inclusivity[0] === '(' ? this.isAfter(from, units) : !this.isBefore(from, units)) &&
- (inclusivity[1] === ')' ? this.isBefore(to, units) : !this.isAfter(to, units));
-}
-
-function isSame (input, units) {
- var localInput = isMoment(input) ? input : createLocal(input),
- inputMs;
- if (!(this.isValid() && localInput.isValid())) {
- return false;
- }
- units = normalizeUnits(units || 'millisecond');
- if (units === 'millisecond') {
- return this.valueOf() === localInput.valueOf();
- } else {
- inputMs = localInput.valueOf();
- return this.clone().startOf(units).valueOf() <= inputMs && inputMs <= this.clone().endOf(units).valueOf();
- }
-}
-
-function isSameOrAfter (input, units) {
- return this.isSame(input, units) || this.isAfter(input,units);
-}
-
-function isSameOrBefore (input, units) {
- return this.isSame(input, units) || this.isBefore(input,units);
-}
-
-function diff (input, units, asFloat) {
- var that,
- zoneDelta,
- delta, output;
-
- if (!this.isValid()) {
- return NaN;
- }
-
- that = cloneWithOffset(input, this);
-
- if (!that.isValid()) {
- return NaN;
- }
-
- zoneDelta = (that.utcOffset() - this.utcOffset()) * 6e4;
-
- units = normalizeUnits(units);
-
- if (units === 'year' || units === 'month' || units === 'quarter') {
- output = monthDiff(this, that);
- if (units === 'quarter') {
- output = output / 3;
- } else if (units === 'year') {
- output = output / 12;
- }
- } else {
- delta = this - that;
- output = units === 'second' ? delta / 1e3 : // 1000
- units === 'minute' ? delta / 6e4 : // 1000 * 60
- units === 'hour' ? delta / 36e5 : // 1000 * 60 * 60
- units === 'day' ? (delta - zoneDelta) / 864e5 : // 1000 * 60 * 60 * 24, negate dst
- units === 'week' ? (delta - zoneDelta) / 6048e5 : // 1000 * 60 * 60 * 24 * 7, negate dst
- delta;
- }
- return asFloat ? output : absFloor(output);
-}
-
-function monthDiff (a, b) {
- // difference in months
- var wholeMonthDiff = ((b.year() - a.year()) * 12) + (b.month() - a.month()),
- // b is in (anchor - 1 month, anchor + 1 month)
- anchor = a.clone().add(wholeMonthDiff, 'months'),
- anchor2, adjust;
-
- if (b - anchor < 0) {
- anchor2 = a.clone().add(wholeMonthDiff - 1, 'months');
- // linear across the month
- adjust = (b - anchor) / (anchor - anchor2);
- } else {
- anchor2 = a.clone().add(wholeMonthDiff + 1, 'months');
- // linear across the month
- adjust = (b - anchor) / (anchor2 - anchor);
- }
-
- //check for negative zero, return zero if negative zero
- return -(wholeMonthDiff + adjust) || 0;
-}
-
-hooks.defaultFormat = 'YYYY-MM-DDTHH:mm:ssZ';
-hooks.defaultFormatUtc = 'YYYY-MM-DDTHH:mm:ss[Z]';
-
-function toString () {
- return this.clone().locale('en').format('ddd MMM DD YYYY HH:mm:ss [GMT]ZZ');
-}
-
-function toISOString () {
- var m = this.clone().utc();
- if (0 < m.year() && m.year() <= 9999) {
- if (isFunction(Date.prototype.toISOString)) {
- // native implementation is ~50x faster, use it when we can
- return this.toDate().toISOString();
- } else {
- return formatMoment(m, 'YYYY-MM-DD[T]HH:mm:ss.SSS[Z]');
- }
- } else {
- return formatMoment(m, 'YYYYYY-MM-DD[T]HH:mm:ss.SSS[Z]');
- }
-}
-
-/**
- * Return a human readable representation of a moment that can
- * also be evaluated to get a new moment which is the same
- *
- * @link https://nodejs.org/dist/latest/docs/api/util.html#util_custom_inspect_function_on_objects
- */
-function inspect () {
- if (!this.isValid()) {
- return 'moment.invalid(/* ' + this._i + ' */)';
- }
- var func = 'moment';
- var zone = '';
- if (!this.isLocal()) {
- func = this.utcOffset() === 0 ? 'moment.utc' : 'moment.parseZone';
- zone = 'Z';
- }
- var prefix = '[' + func + '("]';
- var year = (0 < this.year() && this.year() <= 9999) ? 'YYYY' : 'YYYYYY';
- var datetime = '-MM-DD[T]HH:mm:ss.SSS';
- var suffix = zone + '[")]';
-
- return this.format(prefix + year + datetime + suffix);
-}
-
-function format (inputString) {
- if (!inputString) {
- inputString = this.isUtc() ? hooks.defaultFormatUtc : hooks.defaultFormat;
- }
- var output = formatMoment(this, inputString);
- return this.localeData().postformat(output);
-}
-
-function from (time, withoutSuffix) {
- if (this.isValid() &&
- ((isMoment(time) && time.isValid()) ||
- createLocal(time).isValid())) {
- return createDuration({to: this, from: time}).locale(this.locale()).humanize(!withoutSuffix);
- } else {
- return this.localeData().invalidDate();
- }
-}
-
-function fromNow (withoutSuffix) {
- return this.from(createLocal(), withoutSuffix);
-}
-
-function to (time, withoutSuffix) {
- if (this.isValid() &&
- ((isMoment(time) && time.isValid()) ||
- createLocal(time).isValid())) {
- return createDuration({from: this, to: time}).locale(this.locale()).humanize(!withoutSuffix);
- } else {
- return this.localeData().invalidDate();
- }
-}
-
-function toNow (withoutSuffix) {
- return this.to(createLocal(), withoutSuffix);
-}
-
-// If passed a locale key, it will set the locale for this
-// instance. Otherwise, it will return the locale configuration
-// variables for this instance.
-function locale (key) {
- var newLocaleData;
-
- if (key === undefined) {
- return this._locale._abbr;
- } else {
- newLocaleData = getLocale(key);
- if (newLocaleData != null) {
- this._locale = newLocaleData;
- }
- return this;
- }
-}
-
-var lang = deprecate(
- 'moment().lang() is deprecated. Instead, use moment().localeData() to get the language configuration. Use moment().locale() to change languages.',
- function (key) {
- if (key === undefined) {
- return this.localeData();
- } else {
- return this.locale(key);
- }
- }
-);
-
-function localeData () {
- return this._locale;
-}
-
-function startOf (units) {
- units = normalizeUnits(units);
- // the following switch intentionally omits break keywords
- // to utilize falling through the cases.
- switch (units) {
- case 'year':
- this.month(0);
- /* falls through */
- case 'quarter':
- case 'month':
- this.date(1);
- /* falls through */
- case 'week':
- case 'isoWeek':
- case 'day':
- case 'date':
- this.hours(0);
- /* falls through */
- case 'hour':
- this.minutes(0);
- /* falls through */
- case 'minute':
- this.seconds(0);
- /* falls through */
- case 'second':
- this.milliseconds(0);
- }
-
- // weeks are a special case
- if (units === 'week') {
- this.weekday(0);
- }
- if (units === 'isoWeek') {
- this.isoWeekday(1);
- }
-
- // quarters are also special
- if (units === 'quarter') {
- this.month(Math.floor(this.month() / 3) * 3);
- }
-
- return this;
-}
-
-function endOf (units) {
- units = normalizeUnits(units);
- if (units === undefined || units === 'millisecond') {
- return this;
- }
-
- // 'date' is an alias for 'day', so it should be considered as such.
- if (units === 'date') {
- units = 'day';
- }
-
- return this.startOf(units).add(1, (units === 'isoWeek' ? 'week' : units)).subtract(1, 'ms');
-}
-
-function valueOf () {
- return this._d.valueOf() - ((this._offset || 0) * 60000);
-}
-
-function unix () {
- return Math.floor(this.valueOf() / 1000);
-}
-
-function toDate () {
- return new Date(this.valueOf());
-}
-
-function toArray () {
- var m = this;
- return [m.year(), m.month(), m.date(), m.hour(), m.minute(), m.second(), m.millisecond()];
-}
-
-function toObject () {
- var m = this;
- return {
- years: m.year(),
- months: m.month(),
- date: m.date(),
- hours: m.hours(),
- minutes: m.minutes(),
- seconds: m.seconds(),
- milliseconds: m.milliseconds()
- };
-}
-
-function toJSON () {
- // new Date(NaN).toJSON() === null
- return this.isValid() ? this.toISOString() : null;
-}
-
-function isValid$1 () {
- return isValid(this);
-}
-
-function parsingFlags () {
- return extend({}, getParsingFlags(this));
-}
-
-function invalidAt () {
- return getParsingFlags(this).overflow;
-}
-
-function creationData() {
- return {
- input: this._i,
- format: this._f,
- locale: this._locale,
- isUTC: this._isUTC,
- strict: this._strict
- };
-}
-
-// FORMATTING
-
-addFormatToken(0, ['gg', 2], 0, function () {
- return this.weekYear() % 100;
-});
-
-addFormatToken(0, ['GG', 2], 0, function () {
- return this.isoWeekYear() % 100;
-});
-
-function addWeekYearFormatToken (token, getter) {
- addFormatToken(0, [token, token.length], 0, getter);
-}
-
-addWeekYearFormatToken('gggg', 'weekYear');
-addWeekYearFormatToken('ggggg', 'weekYear');
-addWeekYearFormatToken('GGGG', 'isoWeekYear');
-addWeekYearFormatToken('GGGGG', 'isoWeekYear');
-
-// ALIASES
-
-addUnitAlias('weekYear', 'gg');
-addUnitAlias('isoWeekYear', 'GG');
-
-// PRIORITY
-
-addUnitPriority('weekYear', 1);
-addUnitPriority('isoWeekYear', 1);
-
-
-// PARSING
-
-addRegexToken('G', matchSigned);
-addRegexToken('g', matchSigned);
-addRegexToken('GG', match1to2, match2);
-addRegexToken('gg', match1to2, match2);
-addRegexToken('GGGG', match1to4, match4);
-addRegexToken('gggg', match1to4, match4);
-addRegexToken('GGGGG', match1to6, match6);
-addRegexToken('ggggg', match1to6, match6);
-
-addWeekParseToken(['gggg', 'ggggg', 'GGGG', 'GGGGG'], function (input, week, config, token) {
- week[token.substr(0, 2)] = toInt(input);
-});
-
-addWeekParseToken(['gg', 'GG'], function (input, week, config, token) {
- week[token] = hooks.parseTwoDigitYear(input);
-});
-
-// MOMENTS
-
-function getSetWeekYear (input) {
- return getSetWeekYearHelper.call(this,
- input,
- this.week(),
- this.weekday(),
- this.localeData()._week.dow,
- this.localeData()._week.doy);
-}
-
-function getSetISOWeekYear (input) {
- return getSetWeekYearHelper.call(this,
- input, this.isoWeek(), this.isoWeekday(), 1, 4);
-}
-
-function getISOWeeksInYear () {
- return weeksInYear(this.year(), 1, 4);
-}
-
-function getWeeksInYear () {
- var weekInfo = this.localeData()._week;
- return weeksInYear(this.year(), weekInfo.dow, weekInfo.doy);
-}
-
-function getSetWeekYearHelper(input, week, weekday, dow, doy) {
- var weeksTarget;
- if (input == null) {
- return weekOfYear(this, dow, doy).year;
- } else {
- weeksTarget = weeksInYear(input, dow, doy);
- if (week > weeksTarget) {
- week = weeksTarget;
- }
- return setWeekAll.call(this, input, week, weekday, dow, doy);
- }
-}
-
-function setWeekAll(weekYear, week, weekday, dow, doy) {
- var dayOfYearData = dayOfYearFromWeeks(weekYear, week, weekday, dow, doy),
- date = createUTCDate(dayOfYearData.year, 0, dayOfYearData.dayOfYear);
-
- this.year(date.getUTCFullYear());
- this.month(date.getUTCMonth());
- this.date(date.getUTCDate());
- return this;
-}
-
-// FORMATTING
-
-addFormatToken('Q', 0, 'Qo', 'quarter');
-
-// ALIASES
-
-addUnitAlias('quarter', 'Q');
-
-// PRIORITY
-
-addUnitPriority('quarter', 7);
-
-// PARSING
-
-addRegexToken('Q', match1);
-addParseToken('Q', function (input, array) {
- array[MONTH] = (toInt(input) - 1) * 3;
-});
-
-// MOMENTS
-
-function getSetQuarter (input) {
- return input == null ? Math.ceil((this.month() + 1) / 3) : this.month((input - 1) * 3 + this.month() % 3);
-}
-
-// FORMATTING
-
-addFormatToken('D', ['DD', 2], 'Do', 'date');
-
-// ALIASES
-
-addUnitAlias('date', 'D');
-
-// PRIOROITY
-addUnitPriority('date', 9);
-
-// PARSING
-
-addRegexToken('D', match1to2);
-addRegexToken('DD', match1to2, match2);
-addRegexToken('Do', function (isStrict, locale) {
- return isStrict ? locale._ordinalParse : locale._ordinalParseLenient;
-});
-
-addParseToken(['D', 'DD'], DATE);
-addParseToken('Do', function (input, array) {
- array[DATE] = toInt(input.match(match1to2)[0], 10);
-});
-
-// MOMENTS
-
-var getSetDayOfMonth = makeGetSet('Date', true);
-
-// FORMATTING
-
-addFormatToken('DDD', ['DDDD', 3], 'DDDo', 'dayOfYear');
-
-// ALIASES
-
-addUnitAlias('dayOfYear', 'DDD');
-
-// PRIORITY
-addUnitPriority('dayOfYear', 4);
-
-// PARSING
-
-addRegexToken('DDD', match1to3);
-addRegexToken('DDDD', match3);
-addParseToken(['DDD', 'DDDD'], function (input, array, config) {
- config._dayOfYear = toInt(input);
-});
-
-// HELPERS
-
-// MOMENTS
-
-function getSetDayOfYear (input) {
- var dayOfYear = Math.round((this.clone().startOf('day') - this.clone().startOf('year')) / 864e5) + 1;
- return input == null ? dayOfYear : this.add((input - dayOfYear), 'd');
-}
-
-// FORMATTING
-
-addFormatToken('m', ['mm', 2], 0, 'minute');
-
-// ALIASES
-
-addUnitAlias('minute', 'm');
-
-// PRIORITY
-
-addUnitPriority('minute', 14);
-
-// PARSING
-
-addRegexToken('m', match1to2);
-addRegexToken('mm', match1to2, match2);
-addParseToken(['m', 'mm'], MINUTE);
-
-// MOMENTS
-
-var getSetMinute = makeGetSet('Minutes', false);
-
-// FORMATTING
-
-addFormatToken('s', ['ss', 2], 0, 'second');
-
-// ALIASES
-
-addUnitAlias('second', 's');
-
-// PRIORITY
-
-addUnitPriority('second', 15);
-
-// PARSING
-
-addRegexToken('s', match1to2);
-addRegexToken('ss', match1to2, match2);
-addParseToken(['s', 'ss'], SECOND);
-
-// MOMENTS
-
-var getSetSecond = makeGetSet('Seconds', false);
-
-// FORMATTING
-
-addFormatToken('S', 0, 0, function () {
- return ~~(this.millisecond() / 100);
-});
-
-addFormatToken(0, ['SS', 2], 0, function () {
- return ~~(this.millisecond() / 10);
-});
-
-addFormatToken(0, ['SSS', 3], 0, 'millisecond');
-addFormatToken(0, ['SSSS', 4], 0, function () {
- return this.millisecond() * 10;
-});
-addFormatToken(0, ['SSSSS', 5], 0, function () {
- return this.millisecond() * 100;
-});
-addFormatToken(0, ['SSSSSS', 6], 0, function () {
- return this.millisecond() * 1000;
-});
-addFormatToken(0, ['SSSSSSS', 7], 0, function () {
- return this.millisecond() * 10000;
-});
-addFormatToken(0, ['SSSSSSSS', 8], 0, function () {
- return this.millisecond() * 100000;
-});
-addFormatToken(0, ['SSSSSSSSS', 9], 0, function () {
- return this.millisecond() * 1000000;
-});
-
-
-// ALIASES
-
-addUnitAlias('millisecond', 'ms');
-
-// PRIORITY
-
-addUnitPriority('millisecond', 16);
-
-// PARSING
-
-addRegexToken('S', match1to3, match1);
-addRegexToken('SS', match1to3, match2);
-addRegexToken('SSS', match1to3, match3);
-
-var token;
-for (token = 'SSSS'; token.length <= 9; token += 'S') {
- addRegexToken(token, matchUnsigned);
-}
-
-function parseMs(input, array) {
- array[MILLISECOND] = toInt(('0.' + input) * 1000);
-}
-
-for (token = 'S'; token.length <= 9; token += 'S') {
- addParseToken(token, parseMs);
-}
-// MOMENTS
-
-var getSetMillisecond = makeGetSet('Milliseconds', false);
-
-// FORMATTING
-
-addFormatToken('z', 0, 0, 'zoneAbbr');
-addFormatToken('zz', 0, 0, 'zoneName');
-
-// MOMENTS
-
-function getZoneAbbr () {
- return this._isUTC ? 'UTC' : '';
-}
-
-function getZoneName () {
- return this._isUTC ? 'Coordinated Universal Time' : '';
-}
-
-var proto = Moment.prototype;
-
-proto.add = add;
-proto.calendar = calendar$1;
-proto.clone = clone;
-proto.diff = diff;
-proto.endOf = endOf;
-proto.format = format;
-proto.from = from;
-proto.fromNow = fromNow;
-proto.to = to;
-proto.toNow = toNow;
-proto.get = stringGet;
-proto.invalidAt = invalidAt;
-proto.isAfter = isAfter;
-proto.isBefore = isBefore;
-proto.isBetween = isBetween;
-proto.isSame = isSame;
-proto.isSameOrAfter = isSameOrAfter;
-proto.isSameOrBefore = isSameOrBefore;
-proto.isValid = isValid$1;
-proto.lang = lang;
-proto.locale = locale;
-proto.localeData = localeData;
-proto.max = prototypeMax;
-proto.min = prototypeMin;
-proto.parsingFlags = parsingFlags;
-proto.set = stringSet;
-proto.startOf = startOf;
-proto.subtract = subtract;
-proto.toArray = toArray;
-proto.toObject = toObject;
-proto.toDate = toDate;
-proto.toISOString = toISOString;
-proto.inspect = inspect;
-proto.toJSON = toJSON;
-proto.toString = toString;
-proto.unix = unix;
-proto.valueOf = valueOf;
-proto.creationData = creationData;
-
-// Year
-proto.year = getSetYear;
-proto.isLeapYear = getIsLeapYear;
-
-// Week Year
-proto.weekYear = getSetWeekYear;
-proto.isoWeekYear = getSetISOWeekYear;
-
-// Quarter
-proto.quarter = proto.quarters = getSetQuarter;
-
-// Month
-proto.month = getSetMonth;
-proto.daysInMonth = getDaysInMonth;
-
-// Week
-proto.week = proto.weeks = getSetWeek;
-proto.isoWeek = proto.isoWeeks = getSetISOWeek;
-proto.weeksInYear = getWeeksInYear;
-proto.isoWeeksInYear = getISOWeeksInYear;
-
-// Day
-proto.date = getSetDayOfMonth;
-proto.day = proto.days = getSetDayOfWeek;
-proto.weekday = getSetLocaleDayOfWeek;
-proto.isoWeekday = getSetISODayOfWeek;
-proto.dayOfYear = getSetDayOfYear;
-
-// Hour
-proto.hour = proto.hours = getSetHour;
-
-// Minute
-proto.minute = proto.minutes = getSetMinute;
-
-// Second
-proto.second = proto.seconds = getSetSecond;
-
-// Millisecond
-proto.millisecond = proto.milliseconds = getSetMillisecond;
-
-// Offset
-proto.utcOffset = getSetOffset;
-proto.utc = setOffsetToUTC;
-proto.local = setOffsetToLocal;
-proto.parseZone = setOffsetToParsedOffset;
-proto.hasAlignedHourOffset = hasAlignedHourOffset;
-proto.isDST = isDaylightSavingTime;
-proto.isLocal = isLocal;
-proto.isUtcOffset = isUtcOffset;
-proto.isUtc = isUtc;
-proto.isUTC = isUtc;
-
-// Timezone
-proto.zoneAbbr = getZoneAbbr;
-proto.zoneName = getZoneName;
-
-// Deprecations
-proto.dates = deprecate('dates accessor is deprecated. Use date instead.', getSetDayOfMonth);
-proto.months = deprecate('months accessor is deprecated. Use month instead', getSetMonth);
-proto.years = deprecate('years accessor is deprecated. Use year instead', getSetYear);
-proto.zone = deprecate('moment().zone is deprecated, use moment().utcOffset instead. http://momentjs.com/guides/#/warnings/zone/', getSetZone);
-proto.isDSTShifted = deprecate('isDSTShifted is deprecated. See http://momentjs.com/guides/#/warnings/dst-shifted/ for more information', isDaylightSavingTimeShifted);
-
-function createUnix (input) {
- return createLocal(input * 1000);
-}
-
-function createInZone () {
- return createLocal.apply(null, arguments).parseZone();
-}
-
-function preParsePostFormat (string) {
- return string;
-}
-
-var proto$1 = Locale.prototype;
-
-proto$1.calendar = calendar;
-proto$1.longDateFormat = longDateFormat;
-proto$1.invalidDate = invalidDate;
-proto$1.ordinal = ordinal;
-proto$1.preparse = preParsePostFormat;
-proto$1.postformat = preParsePostFormat;
-proto$1.relativeTime = relativeTime;
-proto$1.pastFuture = pastFuture;
-proto$1.set = set;
-
-// Month
-proto$1.months = localeMonths;
-proto$1.monthsShort = localeMonthsShort;
-proto$1.monthsParse = localeMonthsParse;
-proto$1.monthsRegex = monthsRegex;
-proto$1.monthsShortRegex = monthsShortRegex;
-
-// Week
-proto$1.week = localeWeek;
-proto$1.firstDayOfYear = localeFirstDayOfYear;
-proto$1.firstDayOfWeek = localeFirstDayOfWeek;
-
-// Day of Week
-proto$1.weekdays = localeWeekdays;
-proto$1.weekdaysMin = localeWeekdaysMin;
-proto$1.weekdaysShort = localeWeekdaysShort;
-proto$1.weekdaysParse = localeWeekdaysParse;
-
-proto$1.weekdaysRegex = weekdaysRegex;
-proto$1.weekdaysShortRegex = weekdaysShortRegex;
-proto$1.weekdaysMinRegex = weekdaysMinRegex;
-
-// Hours
-proto$1.isPM = localeIsPM;
-proto$1.meridiem = localeMeridiem;
-
-function get$1 (format, index, field, setter) {
- var locale = getLocale();
- var utc = createUTC().set(setter, index);
- return locale[field](utc, format);
-}
-
-function listMonthsImpl (format, index, field) {
- if (isNumber(format)) {
- index = format;
- format = undefined;
- }
-
- format = format || '';
-
- if (index != null) {
- return get$1(format, index, field, 'month');
- }
-
- var i;
- var out = [];
- for (i = 0; i < 12; i++) {
- out[i] = get$1(format, i, field, 'month');
- }
- return out;
-}
-
-// ()
-// (5)
-// (fmt, 5)
-// (fmt)
-// (true)
-// (true, 5)
-// (true, fmt, 5)
-// (true, fmt)
-function listWeekdaysImpl (localeSorted, format, index, field) {
- if (typeof localeSorted === 'boolean') {
- if (isNumber(format)) {
- index = format;
- format = undefined;
- }
-
- format = format || '';
- } else {
- format = localeSorted;
- index = format;
- localeSorted = false;
-
- if (isNumber(format)) {
- index = format;
- format = undefined;
- }
-
- format = format || '';
- }
-
- var locale = getLocale(),
- shift = localeSorted ? locale._week.dow : 0;
-
- if (index != null) {
- return get$1(format, (index + shift) % 7, field, 'day');
- }
-
- var i;
- var out = [];
- for (i = 0; i < 7; i++) {
- out[i] = get$1(format, (i + shift) % 7, field, 'day');
- }
- return out;
-}
-
-function listMonths (format, index) {
- return listMonthsImpl(format, index, 'months');
-}
-
-function listMonthsShort (format, index) {
- return listMonthsImpl(format, index, 'monthsShort');
-}
-
-function listWeekdays (localeSorted, format, index) {
- return listWeekdaysImpl(localeSorted, format, index, 'weekdays');
-}
-
-function listWeekdaysShort (localeSorted, format, index) {
- return listWeekdaysImpl(localeSorted, format, index, 'weekdaysShort');
-}
-
-function listWeekdaysMin (localeSorted, format, index) {
- return listWeekdaysImpl(localeSorted, format, index, 'weekdaysMin');
-}
-
-getSetGlobalLocale('en', {
- ordinalParse: /\d{1,2}(th|st|nd|rd)/,
- ordinal : function (number) {
- var b = number % 10,
- output = (toInt(number % 100 / 10) === 1) ? 'th' :
- (b === 1) ? 'st' :
- (b === 2) ? 'nd' :
- (b === 3) ? 'rd' : 'th';
- return number + output;
- }
-});
-
-// Side effect imports
-hooks.lang = deprecate('moment.lang is deprecated. Use moment.locale instead.', getSetGlobalLocale);
-hooks.langData = deprecate('moment.langData is deprecated. Use moment.localeData instead.', getLocale);
-
-var mathAbs = Math.abs;
-
-function abs () {
- var data = this._data;
-
- this._milliseconds = mathAbs(this._milliseconds);
- this._days = mathAbs(this._days);
- this._months = mathAbs(this._months);
-
- data.milliseconds = mathAbs(data.milliseconds);
- data.seconds = mathAbs(data.seconds);
- data.minutes = mathAbs(data.minutes);
- data.hours = mathAbs(data.hours);
- data.months = mathAbs(data.months);
- data.years = mathAbs(data.years);
-
- return this;
-}
-
-function addSubtract$1 (duration, input, value, direction) {
- var other = createDuration(input, value);
-
- duration._milliseconds += direction * other._milliseconds;
- duration._days += direction * other._days;
- duration._months += direction * other._months;
-
- return duration._bubble();
-}
-
-// supports only 2.0-style add(1, 's') or add(duration)
-function add$1 (input, value) {
- return addSubtract$1(this, input, value, 1);
-}
-
-// supports only 2.0-style subtract(1, 's') or subtract(duration)
-function subtract$1 (input, value) {
- return addSubtract$1(this, input, value, -1);
-}
-
-function absCeil (number) {
- if (number < 0) {
- return Math.floor(number);
- } else {
- return Math.ceil(number);
- }
-}
-
-function bubble () {
- var milliseconds = this._milliseconds;
- var days = this._days;
- var months = this._months;
- var data = this._data;
- var seconds, minutes, hours, years, monthsFromDays;
-
- // if we have a mix of positive and negative values, bubble down first
- // check: https://github.com/moment/moment/issues/2166
- if (!((milliseconds >= 0 && days >= 0 && months >= 0) ||
- (milliseconds <= 0 && days <= 0 && months <= 0))) {
- milliseconds += absCeil(monthsToDays(months) + days) * 864e5;
- days = 0;
- months = 0;
- }
-
- // The following code bubbles up values, see the tests for
- // examples of what that means.
- data.milliseconds = milliseconds % 1000;
-
- seconds = absFloor(milliseconds / 1000);
- data.seconds = seconds % 60;
-
- minutes = absFloor(seconds / 60);
- data.minutes = minutes % 60;
-
- hours = absFloor(minutes / 60);
- data.hours = hours % 24;
-
- days += absFloor(hours / 24);
-
- // convert days to months
- monthsFromDays = absFloor(daysToMonths(days));
- months += monthsFromDays;
- days -= absCeil(monthsToDays(monthsFromDays));
-
- // 12 months -> 1 year
- years = absFloor(months / 12);
- months %= 12;
-
- data.days = days;
- data.months = months;
- data.years = years;
-
- return this;
-}
-
-function daysToMonths (days) {
- // 400 years have 146097 days (taking into account leap year rules)
- // 400 years have 12 months === 4800
- return days * 4800 / 146097;
-}
-
-function monthsToDays (months) {
- // the reverse of daysToMonths
- return months * 146097 / 4800;
-}
-
-function as (units) {
- var days;
- var months;
- var milliseconds = this._milliseconds;
-
- units = normalizeUnits(units);
-
- if (units === 'month' || units === 'year') {
- days = this._days + milliseconds / 864e5;
- months = this._months + daysToMonths(days);
- return units === 'month' ? months : months / 12;
- } else {
- // handle milliseconds separately because of floating point math errors (issue #1867)
- days = this._days + Math.round(monthsToDays(this._months));
- switch (units) {
- case 'week' : return days / 7 + milliseconds / 6048e5;
- case 'day' : return days + milliseconds / 864e5;
- case 'hour' : return days * 24 + milliseconds / 36e5;
- case 'minute' : return days * 1440 + milliseconds / 6e4;
- case 'second' : return days * 86400 + milliseconds / 1000;
- // Math.floor prevents floating point math errors here
- case 'millisecond': return Math.floor(days * 864e5) + milliseconds;
- default: throw new Error('Unknown unit ' + units);
- }
- }
-}
-
-// TODO: Use this.as('ms')?
-function valueOf$1 () {
- return (
- this._milliseconds +
- this._days * 864e5 +
- (this._months % 12) * 2592e6 +
- toInt(this._months / 12) * 31536e6
- );
-}
-
-function makeAs (alias) {
- return function () {
- return this.as(alias);
- };
-}
-
-var asMilliseconds = makeAs('ms');
-var asSeconds = makeAs('s');
-var asMinutes = makeAs('m');
-var asHours = makeAs('h');
-var asDays = makeAs('d');
-var asWeeks = makeAs('w');
-var asMonths = makeAs('M');
-var asYears = makeAs('y');
-
-function get$2 (units) {
- units = normalizeUnits(units);
- return this[units + 's']();
-}
-
-function makeGetter(name) {
- return function () {
- return this._data[name];
- };
-}
-
-var milliseconds = makeGetter('milliseconds');
-var seconds = makeGetter('seconds');
-var minutes = makeGetter('minutes');
-var hours = makeGetter('hours');
-var days = makeGetter('days');
-var months = makeGetter('months');
-var years = makeGetter('years');
-
-function weeks () {
- return absFloor(this.days() / 7);
-}
-
-var round = Math.round;
-var thresholds = {
- s: 45, // seconds to minute
- m: 45, // minutes to hour
- h: 22, // hours to day
- d: 26, // days to month
- M: 11 // months to year
-};
-
-// helper function for moment.fn.from, moment.fn.fromNow, and moment.duration.fn.humanize
-function substituteTimeAgo(string, number, withoutSuffix, isFuture, locale) {
- return locale.relativeTime(number || 1, !!withoutSuffix, string, isFuture);
-}
-
-function relativeTime$1 (posNegDuration, withoutSuffix, locale) {
- var duration = createDuration(posNegDuration).abs();
- var seconds = round(duration.as('s'));
- var minutes = round(duration.as('m'));
- var hours = round(duration.as('h'));
- var days = round(duration.as('d'));
- var months = round(duration.as('M'));
- var years = round(duration.as('y'));
-
- var a = seconds < thresholds.s && ['s', seconds] ||
- minutes <= 1 && ['m'] ||
- minutes < thresholds.m && ['mm', minutes] ||
- hours <= 1 && ['h'] ||
- hours < thresholds.h && ['hh', hours] ||
- days <= 1 && ['d'] ||
- days < thresholds.d && ['dd', days] ||
- months <= 1 && ['M'] ||
- months < thresholds.M && ['MM', months] ||
- years <= 1 && ['y'] || ['yy', years];
-
- a[2] = withoutSuffix;
- a[3] = +posNegDuration > 0;
- a[4] = locale;
- return substituteTimeAgo.apply(null, a);
-}
-
-// This function allows you to set the rounding function for relative time strings
-function getSetRelativeTimeRounding (roundingFunction) {
- if (roundingFunction === undefined) {
- return round;
- }
- if (typeof(roundingFunction) === 'function') {
- round = roundingFunction;
- return true;
- }
- return false;
-}
-
-// This function allows you to set a threshold for relative time strings
-function getSetRelativeTimeThreshold (threshold, limit) {
- if (thresholds[threshold] === undefined) {
- return false;
- }
- if (limit === undefined) {
- return thresholds[threshold];
- }
- thresholds[threshold] = limit;
- return true;
-}
-
-function humanize (withSuffix) {
- var locale = this.localeData();
- var output = relativeTime$1(this, !withSuffix, locale);
-
- if (withSuffix) {
- output = locale.pastFuture(+this, output);
- }
-
- return locale.postformat(output);
-}
-
-var abs$1 = Math.abs;
-
-function toISOString$1() {
- // for ISO strings we do not use the normal bubbling rules:
- // * milliseconds bubble up until they become hours
- // * days do not bubble at all
- // * months bubble up until they become years
- // This is because there is no context-free conversion between hours and days
- // (think of clock changes)
- // and also not between days and months (28-31 days per month)
- var seconds = abs$1(this._milliseconds) / 1000;
- var days = abs$1(this._days);
- var months = abs$1(this._months);
- var minutes, hours, years;
-
- // 3600 seconds -> 60 minutes -> 1 hour
- minutes = absFloor(seconds / 60);
- hours = absFloor(minutes / 60);
- seconds %= 60;
- minutes %= 60;
-
- // 12 months -> 1 year
- years = absFloor(months / 12);
- months %= 12;
-
-
- // inspired by https://github.com/dordille/moment-isoduration/blob/master/moment.isoduration.js
- var Y = years;
- var M = months;
- var D = days;
- var h = hours;
- var m = minutes;
- var s = seconds;
- var total = this.asSeconds();
-
- if (!total) {
- // this is the same as C#'s (Noda) and python (isodate)...
- // but not other JS (goog.date)
- return 'P0D';
- }
-
- return (total < 0 ? '-' : '') +
- 'P' +
- (Y ? Y + 'Y' : '') +
- (M ? M + 'M' : '') +
- (D ? D + 'D' : '') +
- ((h || m || s) ? 'T' : '') +
- (h ? h + 'H' : '') +
- (m ? m + 'M' : '') +
- (s ? s + 'S' : '');
-}
-
-var proto$2 = Duration.prototype;
-
-proto$2.abs = abs;
-proto$2.add = add$1;
-proto$2.subtract = subtract$1;
-proto$2.as = as;
-proto$2.asMilliseconds = asMilliseconds;
-proto$2.asSeconds = asSeconds;
-proto$2.asMinutes = asMinutes;
-proto$2.asHours = asHours;
-proto$2.asDays = asDays;
-proto$2.asWeeks = asWeeks;
-proto$2.asMonths = asMonths;
-proto$2.asYears = asYears;
-proto$2.valueOf = valueOf$1;
-proto$2._bubble = bubble;
-proto$2.get = get$2;
-proto$2.milliseconds = milliseconds;
-proto$2.seconds = seconds;
-proto$2.minutes = minutes;
-proto$2.hours = hours;
-proto$2.days = days;
-proto$2.weeks = weeks;
-proto$2.months = months;
-proto$2.years = years;
-proto$2.humanize = humanize;
-proto$2.toISOString = toISOString$1;
-proto$2.toString = toISOString$1;
-proto$2.toJSON = toISOString$1;
-proto$2.locale = locale;
-proto$2.localeData = localeData;
-
-// Deprecations
-proto$2.toIsoString = deprecate('toIsoString() is deprecated. Please use toISOString() instead (notice the capitals)', toISOString$1);
-proto$2.lang = lang;
-
-// Side effect imports
-
-// FORMATTING
-
-addFormatToken('X', 0, 0, 'unix');
-addFormatToken('x', 0, 0, 'valueOf');
-
-// PARSING
-
-addRegexToken('x', matchSigned);
-addRegexToken('X', matchTimestamp);
-addParseToken('X', function (input, array, config) {
- config._d = new Date(parseFloat(input, 10) * 1000);
-});
-addParseToken('x', function (input, array, config) {
- config._d = new Date(toInt(input));
-});
-
-// Side effect imports
-
-
-hooks.version = '2.17.1';
-
-setHookCallback(createLocal);
-
-hooks.fn = proto;
-hooks.min = min;
-hooks.max = max;
-hooks.now = now;
-hooks.utc = createUTC;
-hooks.unix = createUnix;
-hooks.months = listMonths;
-hooks.isDate = isDate;
-hooks.locale = getSetGlobalLocale;
-hooks.invalid = createInvalid;
-hooks.duration = createDuration;
-hooks.isMoment = isMoment;
-hooks.weekdays = listWeekdays;
-hooks.parseZone = createInZone;
-hooks.localeData = getLocale;
-hooks.isDuration = isDuration;
-hooks.monthsShort = listMonthsShort;
-hooks.weekdaysMin = listWeekdaysMin;
-hooks.defineLocale = defineLocale;
-hooks.updateLocale = updateLocale;
-hooks.locales = listLocales;
-hooks.weekdaysShort = listWeekdaysShort;
-hooks.normalizeUnits = normalizeUnits;
-hooks.relativeTimeRounding = getSetRelativeTimeRounding;
-hooks.relativeTimeThreshold = getSetRelativeTimeThreshold;
-hooks.calendarFormat = getCalendarFormat;
-hooks.prototype = proto;
-
-return hooks;
-
-})));
diff --git a/fullcalendar-3.7.0/fullcalendar.js b/fullcalendar-3.7.0/fullcalendar.js
new file mode 100644
--- /dev/null
+++ b/fullcalendar-3.7.0/fullcalendar.js
@@ -0,0 +1,14702 @@
+(function webpackUniversalModuleDefinition(root, factory) {
+ if(typeof exports === 'object' && typeof module === 'object')
+ module.exports = factory(require("jquery"), require("moment"));
+ else if(typeof define === 'function' && define.amd)
+ define(["jquery", "moment"], factory);
+ else {
+ var a = typeof exports === 'object' ? factory(require("jquery"), require("moment")) : factory(root["jQuery"], root["moment"]);
+ for(var i in a) (typeof exports === 'object' ? exports : root)[i] = a[i];
+ }
+})(this, function(__WEBPACK_EXTERNAL_MODULE_1__, __WEBPACK_EXTERNAL_MODULE_3__) {
+return /******/ (function(modules) { // webpackBootstrap
+/******/ // The module cache
+/******/ var installedModules = {};
+/******/
+/******/ // The require function
+/******/ function __webpack_require__(moduleId) {
+/******/
+/******/ // Check if module is in cache
+/******/ if(installedModules[moduleId]) {
+/******/ return installedModules[moduleId].exports;
+/******/ }
+/******/ // Create a new module (and put it into the cache)
+/******/ var module = installedModules[moduleId] = {
+/******/ i: moduleId,
+/******/ l: false,
+/******/ exports: {}
+/******/ };
+/******/
+/******/ // Execute the module function
+/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
+/******/
+/******/ // Flag the module as loaded
+/******/ module.l = true;
+/******/
+/******/ // Return the exports of the module
+/******/ return module.exports;
+/******/ }
+/******/
+/******/
+/******/ // expose the modules object (__webpack_modules__)
+/******/ __webpack_require__.m = modules;
+/******/
+/******/ // expose the module cache
+/******/ __webpack_require__.c = installedModules;
+/******/
+/******/ // define getter function for harmony exports
+/******/ __webpack_require__.d = function(exports, name, getter) {
+/******/ if(!__webpack_require__.o(exports, name)) {
+/******/ Object.defineProperty(exports, name, {
+/******/ configurable: false,
+/******/ enumerable: true,
+/******/ get: getter
+/******/ });
+/******/ }
+/******/ };
+/******/
+/******/ // getDefaultExport function for compatibility with non-harmony modules
+/******/ __webpack_require__.n = function(module) {
+/******/ var getter = module && module.__esModule ?
+/******/ function getDefault() { return module['default']; } :
+/******/ function getModuleExports() { return module; };
+/******/ __webpack_require__.d(getter, 'a', getter);
+/******/ return getter;
+/******/ };
+/******/
+/******/ // Object.prototype.hasOwnProperty.call
+/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
+/******/
+/******/ // __webpack_public_path__
+/******/ __webpack_require__.p = "";
+/******/
+/******/ // Load entry module and return exports
+/******/ return __webpack_require__(__webpack_require__.s = 72);
+/******/ })
+/************************************************************************/
+/******/ ([
+/* 0 */
+/***/ (function(module, exports) {
+
+
+/*
+derived from:
+https://github.com/Microsoft/tslib/blob/v1.6.0/tslib.js
+
+only include the helpers we need, to keep down filesize
+*/
+
+var extendStatics =
+ /* NOTE: tslib's as-is method is not compatible with how CoffeeScript does subclasses.
+ * When CoffeeScript is stripped out, can revert.
+ *
+ * Object.setPrototypeOf ||
+ * ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
+ */
+ function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; };
+
+exports.__extends = function (d, b) {
+ extendStatics(d, b);
+ function __() { this.constructor = d; }
+ d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
+};
+
+
+/***/ }),
+/* 1 */
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE_1__;
+
+/***/ }),
+/* 2 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var moment = __webpack_require__(3);
+var $ = __webpack_require__(1);
+/* FullCalendar-specific DOM Utilities
+----------------------------------------------------------------------------------------------------------------------*/
+// Given the scrollbar widths of some other container, create borders/margins on rowEls in order to match the left
+// and right space that was offset by the scrollbars. A 1-pixel border first, then margin beyond that.
+function compensateScroll(rowEls, scrollbarWidths) {
+ if (scrollbarWidths.left) {
+ rowEls.css({
+ 'border-left-width': 1,
+ 'margin-left': scrollbarWidths.left - 1
+ });
+ }
+ if (scrollbarWidths.right) {
+ rowEls.css({
+ 'border-right-width': 1,
+ 'margin-right': scrollbarWidths.right - 1
+ });
+ }
+}
+exports.compensateScroll = compensateScroll;
+// Undoes compensateScroll and restores all borders/margins
+function uncompensateScroll(rowEls) {
+ rowEls.css({
+ 'margin-left': '',
+ 'margin-right': '',
+ 'border-left-width': '',
+ 'border-right-width': ''
+ });
+}
+exports.uncompensateScroll = uncompensateScroll;
+// Make the mouse cursor express that an event is not allowed in the current area
+function disableCursor() {
+ $('body').addClass('fc-not-allowed');
+}
+exports.disableCursor = disableCursor;
+// Returns the mouse cursor to its original look
+function enableCursor() {
+ $('body').removeClass('fc-not-allowed');
+}
+exports.enableCursor = enableCursor;
+// Given a total available height to fill, have `els` (essentially child rows) expand to accomodate.
+// By default, all elements that are shorter than the recommended height are expanded uniformly, not considering
+// any other els that are already too tall. if `shouldRedistribute` is on, it considers these tall rows and
+// reduces the available height.
+function distributeHeight(els, availableHeight, shouldRedistribute) {
+ // *FLOORING NOTE*: we floor in certain places because zoom can give inaccurate floating-point dimensions,
+ // and it is better to be shorter than taller, to avoid creating unnecessary scrollbars.
+ var minOffset1 = Math.floor(availableHeight / els.length); // for non-last element
+ var minOffset2 = Math.floor(availableHeight - minOffset1 * (els.length - 1)); // for last element *FLOORING NOTE*
+ var flexEls = []; // elements that are allowed to expand. array of DOM nodes
+ var flexOffsets = []; // amount of vertical space it takes up
+ var flexHeights = []; // actual css height
+ var usedHeight = 0;
+ undistributeHeight(els); // give all elements their natural height
+ // find elements that are below the recommended height (expandable).
+ // important to query for heights in a single first pass (to avoid reflow oscillation).
+ els.each(function (i, el) {
+ var minOffset = i === els.length - 1 ? minOffset2 : minOffset1;
+ var naturalOffset = $(el).outerHeight(true);
+ if (naturalOffset < minOffset) {
+ flexEls.push(el);
+ flexOffsets.push(naturalOffset);
+ flexHeights.push($(el).height());
+ }
+ else {
+ // this element stretches past recommended height (non-expandable). mark the space as occupied.
+ usedHeight += naturalOffset;
+ }
+ });
+ // readjust the recommended height to only consider the height available to non-maxed-out rows.
+ if (shouldRedistribute) {
+ availableHeight -= usedHeight;
+ minOffset1 = Math.floor(availableHeight / flexEls.length);
+ minOffset2 = Math.floor(availableHeight - minOffset1 * (flexEls.length - 1)); // *FLOORING NOTE*
+ }
+ // assign heights to all expandable elements
+ $(flexEls).each(function (i, el) {
+ var minOffset = i === flexEls.length - 1 ? minOffset2 : minOffset1;
+ var naturalOffset = flexOffsets[i];
+ var naturalHeight = flexHeights[i];
+ var newHeight = minOffset - (naturalOffset - naturalHeight); // subtract the margin/padding
+ if (naturalOffset < minOffset) {
+ $(el).height(newHeight);
+ }
+ });
+}
+exports.distributeHeight = distributeHeight;
+// Undoes distrubuteHeight, restoring all els to their natural height
+function undistributeHeight(els) {
+ els.height('');
+}
+exports.undistributeHeight = undistributeHeight;
+// Given `els`, a jQuery set of cells, find the cell with the largest natural width and set the widths of all the
+// cells to be that width.
+// PREREQUISITE: if you want a cell to take up width, it needs to have a single inner element w/ display:inline
+function matchCellWidths(els) {
+ var maxInnerWidth = 0;
+ els.find('> *').each(function (i, innerEl) {
+ var innerWidth = $(innerEl).outerWidth();
+ if (innerWidth > maxInnerWidth) {
+ maxInnerWidth = innerWidth;
+ }
+ });
+ maxInnerWidth++; // sometimes not accurate of width the text needs to stay on one line. insurance
+ els.width(maxInnerWidth);
+ return maxInnerWidth;
+}
+exports.matchCellWidths = matchCellWidths;
+// Given one element that resides inside another,
+// Subtracts the height of the inner element from the outer element.
+function subtractInnerElHeight(outerEl, innerEl) {
+ var both = outerEl.add(innerEl);
+ var diff;
+ // effin' IE8/9/10/11 sometimes returns 0 for dimensions. this weird hack was the only thing that worked
+ both.css({
+ position: 'relative',
+ left: -1 // ensure reflow in case the el was already relative. negative is less likely to cause new scroll
+ });
+ diff = outerEl.outerHeight() - innerEl.outerHeight(); // grab the dimensions
+ both.css({ position: '', left: '' }); // undo hack
+ return diff;
+}
+exports.subtractInnerElHeight = subtractInnerElHeight;
+/* Element Geom Utilities
+----------------------------------------------------------------------------------------------------------------------*/
+// borrowed from https://github.com/jquery/jquery-ui/blob/1.11.0/ui/core.js#L51
+function getScrollParent(el) {
+ var position = el.css('position'), scrollParent = el.parents().filter(function () {
+ var parent = $(this);
+ return (/(auto|scroll)/).test(parent.css('overflow') + parent.css('overflow-y') + parent.css('overflow-x'));
+ }).eq(0);
+ return position === 'fixed' || !scrollParent.length ? $(el[0].ownerDocument || document) : scrollParent;
+}
+exports.getScrollParent = getScrollParent;
+// Queries the outer bounding area of a jQuery element.
+// Returns a rectangle with absolute coordinates: left, right (exclusive), top, bottom (exclusive).
+// Origin is optional.
+function getOuterRect(el, origin) {
+ var offset = el.offset();
+ var left = offset.left - (origin ? origin.left : 0);
+ var top = offset.top - (origin ? origin.top : 0);
+ return {
+ left: left,
+ right: left + el.outerWidth(),
+ top: top,
+ bottom: top + el.outerHeight()
+ };
+}
+exports.getOuterRect = getOuterRect;
+// Queries the area within the margin/border/scrollbars of a jQuery element. Does not go within the padding.
+// Returns a rectangle with absolute coordinates: left, right (exclusive), top, bottom (exclusive).
+// Origin is optional.
+// WARNING: given element can't have borders
+// NOTE: should use clientLeft/clientTop, but very unreliable cross-browser.
+function getClientRect(el, origin) {
+ var offset = el.offset();
+ var scrollbarWidths = getScrollbarWidths(el);
+ var left = offset.left + getCssFloat(el, 'border-left-width') + scrollbarWidths.left - (origin ? origin.left : 0);
+ var top = offset.top + getCssFloat(el, 'border-top-width') + scrollbarWidths.top - (origin ? origin.top : 0);
+ return {
+ left: left,
+ right: left + el[0].clientWidth,
+ top: top,
+ bottom: top + el[0].clientHeight // clientHeight includes padding but NOT scrollbars
+ };
+}
+exports.getClientRect = getClientRect;
+// Queries the area within the margin/border/padding of a jQuery element. Assumed not to have scrollbars.
+// Returns a rectangle with absolute coordinates: left, right (exclusive), top, bottom (exclusive).
+// Origin is optional.
+function getContentRect(el, origin) {
+ var offset = el.offset(); // just outside of border, margin not included
+ var left = offset.left + getCssFloat(el, 'border-left-width') + getCssFloat(el, 'padding-left') -
+ (origin ? origin.left : 0);
+ var top = offset.top + getCssFloat(el, 'border-top-width') + getCssFloat(el, 'padding-top') -
+ (origin ? origin.top : 0);
+ return {
+ left: left,
+ right: left + el.width(),
+ top: top,
+ bottom: top + el.height()
+ };
+}
+exports.getContentRect = getContentRect;
+// Returns the computed left/right/top/bottom scrollbar widths for the given jQuery element.
+// WARNING: given element can't have borders (which will cause offsetWidth/offsetHeight to be larger).
+// NOTE: should use clientLeft/clientTop, but very unreliable cross-browser.
+function getScrollbarWidths(el) {
+ var leftRightWidth = el[0].offsetWidth - el[0].clientWidth;
+ var bottomWidth = el[0].offsetHeight - el[0].clientHeight;
+ var widths;
+ leftRightWidth = sanitizeScrollbarWidth(leftRightWidth);
+ bottomWidth = sanitizeScrollbarWidth(bottomWidth);
+ widths = { left: 0, right: 0, top: 0, bottom: bottomWidth };
+ if (getIsLeftRtlScrollbars() && el.css('direction') == 'rtl') {
+ widths.left = leftRightWidth;
+ }
+ else {
+ widths.right = leftRightWidth;
+ }
+ return widths;
+}
+exports.getScrollbarWidths = getScrollbarWidths;
+// The scrollbar width computations in getScrollbarWidths are sometimes flawed when it comes to
+// retina displays, rounding, and IE11. Massage them into a usable value.
+function sanitizeScrollbarWidth(width) {
+ width = Math.max(0, width); // no negatives
+ width = Math.round(width);
+ return width;
+}
+// Logic for determining if, when the element is right-to-left, the scrollbar appears on the left side
+var _isLeftRtlScrollbars = null;
+function getIsLeftRtlScrollbars() {
+ if (_isLeftRtlScrollbars === null) {
+ _isLeftRtlScrollbars = computeIsLeftRtlScrollbars();
+ }
+ return _isLeftRtlScrollbars;
+}
+function computeIsLeftRtlScrollbars() {
+ var el = $('')
+ .css({
+ position: 'absolute',
+ top: -1000,
+ left: 0,
+ border: 0,
+ padding: 0,
+ overflow: 'scroll',
+ direction: 'rtl'
+ })
+ .appendTo('body');
+ var innerEl = el.children();
+ var res = innerEl.offset().left > el.offset().left; // is the inner div shifted to accommodate a left scrollbar?
+ el.remove();
+ return res;
+}
+// Retrieves a jQuery element's computed CSS value as a floating-point number.
+// If the queried value is non-numeric (ex: IE can return "medium" for border width), will just return zero.
+function getCssFloat(el, prop) {
+ return parseFloat(el.css(prop)) || 0;
+}
+/* Mouse / Touch Utilities
+----------------------------------------------------------------------------------------------------------------------*/
+// Returns a boolean whether this was a left mouse click and no ctrl key (which means right click on Mac)
+function isPrimaryMouseButton(ev) {
+ return ev.which == 1 && !ev.ctrlKey;
+}
+exports.isPrimaryMouseButton = isPrimaryMouseButton;
+function getEvX(ev) {
+ var touches = ev.originalEvent.touches;
+ // on mobile FF, pageX for touch events is present, but incorrect,
+ // so, look at touch coordinates first.
+ if (touches && touches.length) {
+ return touches[0].pageX;
+ }
+ return ev.pageX;
+}
+exports.getEvX = getEvX;
+function getEvY(ev) {
+ var touches = ev.originalEvent.touches;
+ // on mobile FF, pageX for touch events is present, but incorrect,
+ // so, look at touch coordinates first.
+ if (touches && touches.length) {
+ return touches[0].pageY;
+ }
+ return ev.pageY;
+}
+exports.getEvY = getEvY;
+function getEvIsTouch(ev) {
+ return /^touch/.test(ev.type);
+}
+exports.getEvIsTouch = getEvIsTouch;
+function preventSelection(el) {
+ el.addClass('fc-unselectable')
+ .on('selectstart', preventDefault);
+}
+exports.preventSelection = preventSelection;
+function allowSelection(el) {
+ el.removeClass('fc-unselectable')
+ .off('selectstart', preventDefault);
+}
+exports.allowSelection = allowSelection;
+// Stops a mouse/touch event from doing it's native browser action
+function preventDefault(ev) {
+ ev.preventDefault();
+}
+exports.preventDefault = preventDefault;
+/* General Geometry Utils
+----------------------------------------------------------------------------------------------------------------------*/
+// Returns a new rectangle that is the intersection of the two rectangles. If they don't intersect, returns false
+function intersectRects(rect1, rect2) {
+ var res = {
+ left: Math.max(rect1.left, rect2.left),
+ right: Math.min(rect1.right, rect2.right),
+ top: Math.max(rect1.top, rect2.top),
+ bottom: Math.min(rect1.bottom, rect2.bottom)
+ };
+ if (res.left < res.right && res.top < res.bottom) {
+ return res;
+ }
+ return false;
+}
+exports.intersectRects = intersectRects;
+// Returns a new point that will have been moved to reside within the given rectangle
+function constrainPoint(point, rect) {
+ return {
+ left: Math.min(Math.max(point.left, rect.left), rect.right),
+ top: Math.min(Math.max(point.top, rect.top), rect.bottom)
+ };
+}
+exports.constrainPoint = constrainPoint;
+// Returns a point that is the center of the given rectangle
+function getRectCenter(rect) {
+ return {
+ left: (rect.left + rect.right) / 2,
+ top: (rect.top + rect.bottom) / 2
+ };
+}
+exports.getRectCenter = getRectCenter;
+// Subtracts point2's coordinates from point1's coordinates, returning a delta
+function diffPoints(point1, point2) {
+ return {
+ left: point1.left - point2.left,
+ top: point1.top - point2.top
+ };
+}
+exports.diffPoints = diffPoints;
+/* Object Ordering by Field
+----------------------------------------------------------------------------------------------------------------------*/
+function parseFieldSpecs(input) {
+ var specs = [];
+ var tokens = [];
+ var i, token;
+ if (typeof input === 'string') {
+ tokens = input.split(/\s*,\s*/);
+ }
+ else if (typeof input === 'function') {
+ tokens = [input];
+ }
+ else if ($.isArray(input)) {
+ tokens = input;
+ }
+ for (i = 0; i < tokens.length; i++) {
+ token = tokens[i];
+ if (typeof token === 'string') {
+ specs.push(token.charAt(0) == '-' ?
+ { field: token.substring(1), order: -1 } :
+ { field: token, order: 1 });
+ }
+ else if (typeof token === 'function') {
+ specs.push({ func: token });
+ }
+ }
+ return specs;
+}
+exports.parseFieldSpecs = parseFieldSpecs;
+function compareByFieldSpecs(obj1, obj2, fieldSpecs) {
+ var i;
+ var cmp;
+ for (i = 0; i < fieldSpecs.length; i++) {
+ cmp = compareByFieldSpec(obj1, obj2, fieldSpecs[i]);
+ if (cmp) {
+ return cmp;
+ }
+ }
+ return 0;
+}
+exports.compareByFieldSpecs = compareByFieldSpecs;
+function compareByFieldSpec(obj1, obj2, fieldSpec) {
+ if (fieldSpec.func) {
+ return fieldSpec.func(obj1, obj2);
+ }
+ return flexibleCompare(obj1[fieldSpec.field], obj2[fieldSpec.field]) *
+ (fieldSpec.order || 1);
+}
+exports.compareByFieldSpec = compareByFieldSpec;
+function flexibleCompare(a, b) {
+ if (!a && !b) {
+ return 0;
+ }
+ if (b == null) {
+ return -1;
+ }
+ if (a == null) {
+ return 1;
+ }
+ if ($.type(a) === 'string' || $.type(b) === 'string') {
+ return String(a).localeCompare(String(b));
+ }
+ return a - b;
+}
+exports.flexibleCompare = flexibleCompare;
+/* Date Utilities
+----------------------------------------------------------------------------------------------------------------------*/
+exports.dayIDs = ['sun', 'mon', 'tue', 'wed', 'thu', 'fri', 'sat'];
+exports.unitsDesc = ['year', 'month', 'week', 'day', 'hour', 'minute', 'second', 'millisecond']; // descending
+// Diffs the two moments into a Duration where full-days are recorded first, then the remaining time.
+// Moments will have their timezones normalized.
+function diffDayTime(a, b) {
+ return moment.duration({
+ days: a.clone().stripTime().diff(b.clone().stripTime(), 'days'),
+ ms: a.time() - b.time() // time-of-day from day start. disregards timezone
+ });
+}
+exports.diffDayTime = diffDayTime;
+// Diffs the two moments via their start-of-day (regardless of timezone). Produces whole-day durations.
+function diffDay(a, b) {
+ return moment.duration({
+ days: a.clone().stripTime().diff(b.clone().stripTime(), 'days')
+ });
+}
+exports.diffDay = diffDay;
+// Diffs two moments, producing a duration, made of a whole-unit-increment of the given unit. Uses rounding.
+function diffByUnit(a, b, unit) {
+ return moment.duration(Math.round(a.diff(b, unit, true)), // returnFloat=true
+ unit);
+}
+exports.diffByUnit = diffByUnit;
+// Computes the unit name of the largest whole-unit period of time.
+// For example, 48 hours will be "days" whereas 49 hours will be "hours".
+// Accepts start/end, a range object, or an original duration object.
+function computeGreatestUnit(start, end) {
+ var i, unit;
+ var val;
+ for (i = 0; i < exports.unitsDesc.length; i++) {
+ unit = exports.unitsDesc[i];
+ val = computeRangeAs(unit, start, end);
+ if (val >= 1 && isInt(val)) {
+ break;
+ }
+ }
+ return unit; // will be "milliseconds" if nothing else matches
+}
+exports.computeGreatestUnit = computeGreatestUnit;
+// like computeGreatestUnit, but has special abilities to interpret the source input for clues
+function computeDurationGreatestUnit(duration, durationInput) {
+ var unit = computeGreatestUnit(duration);
+ // prevent days:7 from being interpreted as a week
+ if (unit === 'week' && typeof durationInput === 'object' && durationInput.days) {
+ unit = 'day';
+ }
+ return unit;
+}
+exports.computeDurationGreatestUnit = computeDurationGreatestUnit;
+// Computes the number of units (like "hours") in the given range.
+// Range can be a {start,end} object, separate start/end args, or a Duration.
+// Results are based on Moment's .as() and .diff() methods, so results can depend on internal handling
+// of month-diffing logic (which tends to vary from version to version).
+function computeRangeAs(unit, start, end) {
+ if (end != null) {
+ return end.diff(start, unit, true);
+ }
+ else if (moment.isDuration(start)) {
+ return start.as(unit);
+ }
+ else {
+ return start.end.diff(start.start, unit, true);
+ }
+}
+// Intelligently divides a range (specified by a start/end params) by a duration
+function divideRangeByDuration(start, end, dur) {
+ var months;
+ if (durationHasTime(dur)) {
+ return (end - start) / dur;
+ }
+ months = dur.asMonths();
+ if (Math.abs(months) >= 1 && isInt(months)) {
+ return end.diff(start, 'months', true) / months;
+ }
+ return end.diff(start, 'days', true) / dur.asDays();
+}
+exports.divideRangeByDuration = divideRangeByDuration;
+// Intelligently divides one duration by another
+function divideDurationByDuration(dur1, dur2) {
+ var months1, months2;
+ if (durationHasTime(dur1) || durationHasTime(dur2)) {
+ return dur1 / dur2;
+ }
+ months1 = dur1.asMonths();
+ months2 = dur2.asMonths();
+ if (Math.abs(months1) >= 1 && isInt(months1) &&
+ Math.abs(months2) >= 1 && isInt(months2)) {
+ return months1 / months2;
+ }
+ return dur1.asDays() / dur2.asDays();
+}
+exports.divideDurationByDuration = divideDurationByDuration;
+// Intelligently multiplies a duration by a number
+function multiplyDuration(dur, n) {
+ var months;
+ if (durationHasTime(dur)) {
+ return moment.duration(dur * n);
+ }
+ months = dur.asMonths();
+ if (Math.abs(months) >= 1 && isInt(months)) {
+ return moment.duration({ months: months * n });
+ }
+ return moment.duration({ days: dur.asDays() * n });
+}
+exports.multiplyDuration = multiplyDuration;
+// Returns a boolean about whether the given duration has any time parts (hours/minutes/seconds/ms)
+function durationHasTime(dur) {
+ return Boolean(dur.hours() || dur.minutes() || dur.seconds() || dur.milliseconds());
+}
+exports.durationHasTime = durationHasTime;
+function isNativeDate(input) {
+ return Object.prototype.toString.call(input) === '[object Date]' || input instanceof Date;
+}
+exports.isNativeDate = isNativeDate;
+// Returns a boolean about whether the given input is a time string, like "06:40:00" or "06:00"
+function isTimeString(str) {
+ return typeof str === 'string' &&
+ /^\d+\:\d+(?:\:\d+\.?(?:\d{3})?)?$/.test(str);
+}
+exports.isTimeString = isTimeString;
+/* Logging and Debug
+----------------------------------------------------------------------------------------------------------------------*/
+function log() {
+ var args = [];
+ for (var _i = 0; _i < arguments.length; _i++) {
+ args[_i] = arguments[_i];
+ }
+ var console = window.console;
+ if (console && console.log) {
+ return console.log.apply(console, args);
+ }
+}
+exports.log = log;
+function warn() {
+ var args = [];
+ for (var _i = 0; _i < arguments.length; _i++) {
+ args[_i] = arguments[_i];
+ }
+ var console = window.console;
+ if (console && console.warn) {
+ return console.warn.apply(console, args);
+ }
+ else {
+ return log.apply(null, args);
+ }
+}
+exports.warn = warn;
+/* General Utilities
+----------------------------------------------------------------------------------------------------------------------*/
+var hasOwnPropMethod = {}.hasOwnProperty;
+// Merges an array of objects into a single object.
+// The second argument allows for an array of property names who's object values will be merged together.
+function mergeProps(propObjs, complexProps) {
+ var dest = {};
+ var i, name;
+ var complexObjs;
+ var j, val;
+ var props;
+ if (complexProps) {
+ for (i = 0; i < complexProps.length; i++) {
+ name = complexProps[i];
+ complexObjs = [];
+ // collect the trailing object values, stopping when a non-object is discovered
+ for (j = propObjs.length - 1; j >= 0; j--) {
+ val = propObjs[j][name];
+ if (typeof val === 'object') {
+ complexObjs.unshift(val);
+ }
+ else if (val !== undefined) {
+ dest[name] = val; // if there were no objects, this value will be used
+ break;
+ }
+ }
+ // if the trailing values were objects, use the merged value
+ if (complexObjs.length) {
+ dest[name] = mergeProps(complexObjs);
+ }
+ }
+ }
+ // copy values into the destination, going from last to first
+ for (i = propObjs.length - 1; i >= 0; i--) {
+ props = propObjs[i];
+ for (name in props) {
+ if (!(name in dest)) {
+ dest[name] = props[name];
+ }
+ }
+ }
+ return dest;
+}
+exports.mergeProps = mergeProps;
+function copyOwnProps(src, dest) {
+ for (var name in src) {
+ if (hasOwnProp(src, name)) {
+ dest[name] = src[name];
+ }
+ }
+}
+exports.copyOwnProps = copyOwnProps;
+function hasOwnProp(obj, name) {
+ return hasOwnPropMethod.call(obj, name);
+}
+exports.hasOwnProp = hasOwnProp;
+function applyAll(functions, thisObj, args) {
+ if ($.isFunction(functions)) {
+ functions = [functions];
+ }
+ if (functions) {
+ var i;
+ var ret;
+ for (i = 0; i < functions.length; i++) {
+ ret = functions[i].apply(thisObj, args) || ret;
+ }
+ return ret;
+ }
+}
+exports.applyAll = applyAll;
+function removeMatching(array, testFunc) {
+ var removeCnt = 0;
+ var i = 0;
+ while (i < array.length) {
+ if (testFunc(array[i])) {
+ array.splice(i, 1);
+ removeCnt++;
+ }
+ else {
+ i++;
+ }
+ }
+ return removeCnt;
+}
+exports.removeMatching = removeMatching;
+function removeExact(array, exactVal) {
+ var removeCnt = 0;
+ var i = 0;
+ while (i < array.length) {
+ if (array[i] === exactVal) {
+ array.splice(i, 1);
+ removeCnt++;
+ }
+ else {
+ i++;
+ }
+ }
+ return removeCnt;
+}
+exports.removeExact = removeExact;
+function isArraysEqual(a0, a1) {
+ var len = a0.length;
+ var i;
+ if (len == null || len !== a1.length) {
+ return false;
+ }
+ for (i = 0; i < len; i++) {
+ if (a0[i] !== a1[i]) {
+ return false;
+ }
+ }
+ return true;
+}
+exports.isArraysEqual = isArraysEqual;
+function firstDefined() {
+ var args = [];
+ for (var _i = 0; _i < arguments.length; _i++) {
+ args[_i] = arguments[_i];
+ }
+ for (var i = 0; i < args.length; i++) {
+ if (args[i] !== undefined) {
+ return args[i];
+ }
+ }
+}
+exports.firstDefined = firstDefined;
+function htmlEscape(s) {
+ return (s + '').replace(/&/g, '&')
+ .replace(//g, '>')
+ .replace(/'/g, ''')
+ .replace(/"/g, '"')
+ .replace(/\n/g, ' ');
+}
+exports.htmlEscape = htmlEscape;
+function stripHtmlEntities(text) {
+ return text.replace(/&.*?;/g, '');
+}
+exports.stripHtmlEntities = stripHtmlEntities;
+// Given a hash of CSS properties, returns a string of CSS.
+// Uses property names as-is (no camel-case conversion). Will not make statements for null/undefined values.
+function cssToStr(cssProps) {
+ var statements = [];
+ $.each(cssProps, function (name, val) {
+ if (val != null) {
+ statements.push(name + ':' + val);
+ }
+ });
+ return statements.join(';');
+}
+exports.cssToStr = cssToStr;
+// Given an object hash of HTML attribute names to values,
+// generates a string that can be injected between < > in HTML
+function attrsToStr(attrs) {
+ var parts = [];
+ $.each(attrs, function (name, val) {
+ if (val != null) {
+ parts.push(name + '="' + htmlEscape(val) + '"');
+ }
+ });
+ return parts.join(' ');
+}
+exports.attrsToStr = attrsToStr;
+function capitaliseFirstLetter(str) {
+ return str.charAt(0).toUpperCase() + str.slice(1);
+}
+exports.capitaliseFirstLetter = capitaliseFirstLetter;
+function compareNumbers(a, b) {
+ return a - b;
+}
+exports.compareNumbers = compareNumbers;
+function isInt(n) {
+ return n % 1 === 0;
+}
+exports.isInt = isInt;
+// Returns a method bound to the given object context.
+// Just like one of the jQuery.proxy signatures, but without the undesired behavior of treating the same method with
+// different contexts as identical when binding/unbinding events.
+function proxy(obj, methodName) {
+ var method = obj[methodName];
+ return function () {
+ return method.apply(obj, arguments);
+ };
+}
+exports.proxy = proxy;
+// Returns a function, that, as long as it continues to be invoked, will not
+// be triggered. The function will be called after it stops being called for
+// N milliseconds. If `immediate` is passed, trigger the function on the
+// leading edge, instead of the trailing.
+// https://github.com/jashkenas/underscore/blob/1.6.0/underscore.js#L714
+function debounce(func, wait, immediate) {
+ if (immediate === void 0) { immediate = false; }
+ var timeout, args, context, timestamp, result;
+ var later = function () {
+ var last = +new Date() - timestamp;
+ if (last < wait) {
+ timeout = setTimeout(later, wait - last);
+ }
+ else {
+ timeout = null;
+ if (!immediate) {
+ result = func.apply(context, args);
+ context = args = null;
+ }
+ }
+ };
+ return function () {
+ context = this;
+ args = arguments;
+ timestamp = +new Date();
+ var callNow = immediate && !timeout;
+ if (!timeout) {
+ timeout = setTimeout(later, wait);
+ }
+ if (callNow) {
+ result = func.apply(context, args);
+ context = args = null;
+ }
+ return result;
+ };
+}
+exports.debounce = debounce;
+
+
+/***/ }),
+/* 3 */
+/***/ (function(module, exports) {
+
+module.exports = __WEBPACK_EXTERNAL_MODULE_3__;
+
+/***/ }),
+/* 4 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var moment = __webpack_require__(3);
+var moment_ext_1 = __webpack_require__(9);
+var UnzonedRange = /** @class */ (function () {
+ function UnzonedRange(startInput, endInput) {
+ // TODO: move these into footprint.
+ // Especially, doesn't make sense for null startMs/endMs.
+ this.isStart = true;
+ this.isEnd = true;
+ if (moment.isMoment(startInput)) {
+ startInput = startInput.clone().stripZone();
+ }
+ if (moment.isMoment(endInput)) {
+ endInput = endInput.clone().stripZone();
+ }
+ if (startInput) {
+ this.startMs = startInput.valueOf();
+ }
+ if (endInput) {
+ this.endMs = endInput.valueOf();
+ }
+ }
+ UnzonedRange.prototype.intersect = function (otherRange) {
+ var startMs = this.startMs;
+ var endMs = this.endMs;
+ var newRange = null;
+ if (otherRange.startMs != null) {
+ if (startMs == null) {
+ startMs = otherRange.startMs;
+ }
+ else {
+ startMs = Math.max(startMs, otherRange.startMs);
+ }
+ }
+ if (otherRange.endMs != null) {
+ if (endMs == null) {
+ endMs = otherRange.endMs;
+ }
+ else {
+ endMs = Math.min(endMs, otherRange.endMs);
+ }
+ }
+ if (startMs == null || endMs == null || startMs < endMs) {
+ newRange = new UnzonedRange(startMs, endMs);
+ newRange.isStart = this.isStart && startMs === this.startMs;
+ newRange.isEnd = this.isEnd && endMs === this.endMs;
+ }
+ return newRange;
+ };
+ UnzonedRange.prototype.intersectsWith = function (otherRange) {
+ return (this.endMs == null || otherRange.startMs == null || this.endMs > otherRange.startMs) &&
+ (this.startMs == null || otherRange.endMs == null || this.startMs < otherRange.endMs);
+ };
+ UnzonedRange.prototype.containsRange = function (innerRange) {
+ return (this.startMs == null || (innerRange.startMs != null && innerRange.startMs >= this.startMs)) &&
+ (this.endMs == null || (innerRange.endMs != null && innerRange.endMs <= this.endMs));
+ };
+ // `date` can be a moment, a Date, or a millisecond time.
+ UnzonedRange.prototype.containsDate = function (date) {
+ var ms = date.valueOf();
+ return (this.startMs == null || ms >= this.startMs) &&
+ (this.endMs == null || ms < this.endMs);
+ };
+ // If the given date is not within the given range, move it inside.
+ // (If it's past the end, make it one millisecond before the end).
+ // `date` can be a moment, a Date, or a millisecond time.
+ // Returns a MS-time.
+ UnzonedRange.prototype.constrainDate = function (date) {
+ var ms = date.valueOf();
+ if (this.startMs != null && ms < this.startMs) {
+ ms = this.startMs;
+ }
+ if (this.endMs != null && ms >= this.endMs) {
+ ms = this.endMs - 1;
+ }
+ return ms;
+ };
+ UnzonedRange.prototype.equals = function (otherRange) {
+ return this.startMs === otherRange.startMs && this.endMs === otherRange.endMs;
+ };
+ UnzonedRange.prototype.clone = function () {
+ var range = new UnzonedRange(this.startMs, this.endMs);
+ range.isStart = this.isStart;
+ range.isEnd = this.isEnd;
+ return range;
+ };
+ // Returns an ambig-zoned moment from startMs.
+ // BEWARE: returned moment is not localized.
+ // Formatting and start-of-week will be default.
+ UnzonedRange.prototype.getStart = function () {
+ if (this.startMs != null) {
+ return moment_ext_1.default.utc(this.startMs).stripZone();
+ }
+ return null;
+ };
+ // Returns an ambig-zoned moment from startMs.
+ // BEWARE: returned moment is not localized.
+ // Formatting and start-of-week will be default.
+ UnzonedRange.prototype.getEnd = function () {
+ if (this.endMs != null) {
+ return moment_ext_1.default.utc(this.endMs).stripZone();
+ }
+ return null;
+ };
+ UnzonedRange.prototype.as = function (unit) {
+ return moment.utc(this.endMs).diff(moment.utc(this.startMs), unit, true);
+ };
+ /*
+ SIDEEFFECT: will mutate eventRanges.
+ Will return a new array result.
+ Only works for non-open-ended ranges.
+ */
+ UnzonedRange.invertRanges = function (ranges, constraintRange) {
+ var invertedRanges = [];
+ var startMs = constraintRange.startMs; // the end of the previous range. the start of the new range
+ var i;
+ var dateRange;
+ // ranges need to be in order. required for our date-walking algorithm
+ ranges.sort(compareUnzonedRanges);
+ for (i = 0; i < ranges.length; i++) {
+ dateRange = ranges[i];
+ // add the span of time before the event (if there is any)
+ if (dateRange.startMs > startMs) {
+ invertedRanges.push(new UnzonedRange(startMs, dateRange.startMs));
+ }
+ if (dateRange.endMs > startMs) {
+ startMs = dateRange.endMs;
+ }
+ }
+ // add the span of time after the last event (if there is any)
+ if (startMs < constraintRange.endMs) {
+ invertedRanges.push(new UnzonedRange(startMs, constraintRange.endMs));
+ }
+ return invertedRanges;
+ };
+ return UnzonedRange;
+}());
+exports.default = UnzonedRange;
+/*
+Only works for non-open-ended ranges.
+*/
+function compareUnzonedRanges(range1, range2) {
+ return range1.startMs - range2.startMs; // earlier ranges go first
+}
+
+
+/***/ }),
+/* 5 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var $ = __webpack_require__(1);
+var ParsableModelMixin_1 = __webpack_require__(50);
+var Class_1 = __webpack_require__(21);
+var EventDefParser_1 = __webpack_require__(33);
+var EventSource = /** @class */ (function (_super) {
+ tslib_1.__extends(EventSource, _super);
+ // can we do away with calendar? at least for the abstract?
+ // useful for buildEventDef
+ function EventSource(calendar) {
+ var _this = _super.call(this) || this;
+ _this.calendar = calendar;
+ _this.className = [];
+ _this.uid = String(EventSource.uuid++);
+ return _this;
+ }
+ EventSource.prototype.fetch = function (start, end, timezone) {
+ // subclasses must implement. must return a promise.
+ };
+ EventSource.prototype.removeEventDefsById = function (eventDefId) {
+ // optional for subclasses to implement
+ };
+ EventSource.prototype.removeAllEventDefs = function () {
+ // optional for subclasses to implement
+ };
+ /*
+ For compairing/matching
+ */
+ EventSource.prototype.getPrimitive = function (otherSource) {
+ // subclasses must implement
+ };
+ EventSource.prototype.parseEventDefs = function (rawEventDefs) {
+ var i;
+ var eventDef;
+ var eventDefs = [];
+ for (i = 0; i < rawEventDefs.length; i++) {
+ eventDef = this.parseEventDef(rawEventDefs[i]);
+ if (eventDef) {
+ eventDefs.push(eventDef);
+ }
+ }
+ return eventDefs;
+ };
+ EventSource.prototype.parseEventDef = function (rawInput) {
+ var calendarTransform = this.calendar.opt('eventDataTransform');
+ var sourceTransform = this.eventDataTransform;
+ if (calendarTransform) {
+ rawInput = calendarTransform(rawInput);
+ }
+ if (sourceTransform) {
+ rawInput = sourceTransform(rawInput);
+ }
+ return EventDefParser_1.default.parse(rawInput, this);
+ };
+ EventSource.prototype.applyManualStandardProps = function (rawProps) {
+ if (rawProps.id != null) {
+ this.id = EventSource.normalizeId(rawProps.id);
+ }
+ // TODO: converge with EventDef
+ if ($.isArray(rawProps.className)) {
+ this.className = rawProps.className;
+ }
+ else if (typeof rawProps.className === 'string') {
+ this.className = rawProps.className.split(/\s+/);
+ }
+ return true;
+ };
+ /*
+ rawInput can be any data type!
+ */
+ EventSource.parse = function (rawInput, calendar) {
+ var source = new this(calendar);
+ if (typeof rawInput === 'object') {
+ if (source.applyProps(rawInput)) {
+ return source;
+ }
+ }
+ return false;
+ };
+ EventSource.normalizeId = function (id) {
+ if (id) {
+ return String(id);
+ }
+ return null;
+ };
+ EventSource.defineStandardProps = ParsableModelMixin_1.default.defineStandardProps;
+ EventSource.copyVerbatimStandardProps = ParsableModelMixin_1.default.copyVerbatimStandardProps;
+ // IDs
+ // -----------------------------------------------------------------------------------------------------------------
+ // TODO: converge with EventDef
+ EventSource.uuid = 0;
+ return EventSource;
+}(Class_1.default));
+exports.default = EventSource;
+ParsableModelMixin_1.default.mixInto(EventSource);
+// Parsing
+// ---------------------------------------------------------------------------------------------------------------------
+EventSource.defineStandardProps({
+ // manually process...
+ id: false,
+ className: false,
+ // automatically transfer...
+ color: true,
+ backgroundColor: true,
+ borderColor: true,
+ textColor: true,
+ editable: true,
+ startEditable: true,
+ durationEditable: true,
+ rendering: true,
+ overlap: true,
+ constraint: true,
+ allDayDefault: true,
+ eventDataTransform: true
+});
+
+
+/***/ }),
+/* 6 */
+/***/ (function(module, exports, __webpack_require__) {
+
+/*
+Utility methods for easily listening to events on another object,
+and more importantly, easily unlistening from them.
+
+USAGE:
+ import { default as ListenerMixin, ListenerInterface } from './ListenerMixin'
+in class:
+ listenTo: ListenerInterface['listenTo']
+ stopListeningTo: ListenerInterface['stopListeningTo']
+after class:
+ ListenerMixin.mixInto(TheClass)
+*/
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var $ = __webpack_require__(1);
+var Mixin_1 = __webpack_require__(12);
+var guid = 0;
+var ListenerMixin = /** @class */ (function (_super) {
+ tslib_1.__extends(ListenerMixin, _super);
+ function ListenerMixin() {
+ var _this = _super !== null && _super.apply(this, arguments) || this;
+ _this.listenerId = null;
+ return _this;
+ }
+ /*
+ Given an `other` object that has on/off methods, bind the given `callback` to an event by the given name.
+ The `callback` will be called with the `this` context of the object that .listenTo is being called on.
+ Can be called:
+ .listenTo(other, eventName, callback)
+ OR
+ .listenTo(other, {
+ eventName1: callback1,
+ eventName2: callback2
+ })
+ */
+ ListenerMixin.prototype.listenTo = function (other, arg, callback) {
+ if (typeof arg === 'object') {
+ for (var eventName in arg) {
+ if (arg.hasOwnProperty(eventName)) {
+ this.listenTo(other, eventName, arg[eventName]);
+ }
+ }
+ }
+ else if (typeof arg === 'string') {
+ other.on(arg + '.' + this.getListenerNamespace(), // use event namespacing to identify this object
+ $.proxy(callback, this) // always use `this` context
+ // the usually-undesired jQuery guid behavior doesn't matter,
+ // because we always unbind via namespace
+ );
+ }
+ };
+ /*
+ Causes the current object to stop listening to events on the `other` object.
+ `eventName` is optional. If omitted, will stop listening to ALL events on `other`.
+ */
+ ListenerMixin.prototype.stopListeningTo = function (other, eventName) {
+ other.off((eventName || '') + '.' + this.getListenerNamespace());
+ };
+ /*
+ Returns a string, unique to this object, to be used for event namespacing
+ */
+ ListenerMixin.prototype.getListenerNamespace = function () {
+ if (this.listenerId == null) {
+ this.listenerId = guid++;
+ }
+ return '_listener' + this.listenerId;
+ };
+ return ListenerMixin;
+}(Mixin_1.default));
+exports.default = ListenerMixin;
+
+
+/***/ }),
+/* 7 */
+/***/ (function(module, exports) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+exports.default = {
+ version: "<%= version %>",
+ // When introducing internal API incompatibilities (where fullcalendar plugins would break),
+ // the minor version of the calendar should be upped (ex: 2.7.2 -> 2.8.0)
+ // and the below integer should be incremented.
+ internalApiVersion: 12,
+ // for GlobalEmitter
+ touchMouseIgnoreWait: 500,
+ // for ExternalDropping
+ // Require all HTML5 data-* attributes used by FullCalendar to have this prefix.
+ // A value of '' will query attributes like data-event. A value of 'fc' will query attributes like data-fc-event.
+ dataAttrPrefix: '',
+ views: {},
+ locales: {}
+};
+
+
+/***/ }),
+/* 8 */
+/***/ (function(module, exports, __webpack_require__) {
+
+/*
+USAGE:
+ import { default as EmitterMixin, EmitterInterface } from './EmitterMixin'
+in class:
+ on: EmitterInterface['on']
+ one: EmitterInterface['one']
+ off: EmitterInterface['off']
+ trigger: EmitterInterface['trigger']
+ triggerWith: EmitterInterface['triggerWith']
+ hasHandlers: EmitterInterface['hasHandlers']
+after class:
+ EmitterMixin.mixInto(TheClass)
+*/
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var $ = __webpack_require__(1);
+var Mixin_1 = __webpack_require__(12);
+var EmitterMixin = /** @class */ (function (_super) {
+ tslib_1.__extends(EmitterMixin, _super);
+ function EmitterMixin() {
+ return _super !== null && _super.apply(this, arguments) || this;
+ }
+ // jQuery-ification via $(this) allows a non-DOM object to have
+ // the same event handling capabilities (including namespaces).
+ EmitterMixin.prototype.on = function (types, handler) {
+ $(this).on(types, this._prepareIntercept(handler));
+ return this; // for chaining
+ };
+ EmitterMixin.prototype.one = function (types, handler) {
+ $(this).one(types, this._prepareIntercept(handler));
+ return this; // for chaining
+ };
+ EmitterMixin.prototype._prepareIntercept = function (handler) {
+ // handlers are always called with an "event" object as their first param.
+ // sneak the `this` context and arguments into the extra parameter object
+ // and forward them on to the original handler.
+ var intercept = function (ev, extra) {
+ return handler.apply(extra.context || this, extra.args || []);
+ };
+ // mimick jQuery's internal "proxy" system (risky, I know)
+ // causing all functions with the same .guid to appear to be the same.
+ // https://github.com/jquery/jquery/blob/2.2.4/src/core.js#L448
+ // this is needed for calling .off with the original non-intercept handler.
+ if (!handler.guid) {
+ handler.guid = $.guid++;
+ }
+ intercept.guid = handler.guid;
+ return intercept;
+ };
+ EmitterMixin.prototype.off = function (types, handler) {
+ $(this).off(types, handler);
+ return this; // for chaining
+ };
+ EmitterMixin.prototype.trigger = function (types) {
+ var args = [];
+ for (var _i = 1; _i < arguments.length; _i++) {
+ args[_i - 1] = arguments[_i];
+ }
+ // pass in "extra" info to the intercept
+ $(this).triggerHandler(types, { args: args });
+ return this; // for chaining
+ };
+ EmitterMixin.prototype.triggerWith = function (types, context, args) {
+ // `triggerHandler` is less reliant on the DOM compared to `trigger`.
+ // pass in "extra" info to the intercept.
+ $(this).triggerHandler(types, { context: context, args: args });
+ return this; // for chaining
+ };
+ EmitterMixin.prototype.hasHandlers = function (type) {
+ var hash = $._data(this, 'events'); // http://blog.jquery.com/2012/08/09/jquery-1-8-released/
+ return hash && hash[type] && hash[type].length > 0;
+ };
+ return EmitterMixin;
+}(Mixin_1.default));
+exports.default = EmitterMixin;
+
+
+/***/ }),
+/* 9 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var moment = __webpack_require__(3);
+var $ = __webpack_require__(1);
+var util_1 = __webpack_require__(2);
+/*
+GENERAL NOTE on moments throughout the *entire rest* of the codebase:
+All moments are assumed to be ambiguously-zoned unless otherwise noted,
+with the NOTABLE EXCEOPTION of start/end dates that live on *Event Objects*.
+Ambiguously-TIMED moments are assumed to be ambiguously-zoned by nature.
+*/
+var ambigDateOfMonthRegex = /^\s*\d{4}-\d\d$/;
+var ambigTimeOrZoneRegex = /^\s*\d{4}-(?:(\d\d-\d\d)|(W\d\d$)|(W\d\d-\d)|(\d\d\d))((T| )(\d\d(:\d\d(:\d\d(\.\d+)?)?)?)?)?$/;
+var newMomentProto = moment.fn; // where we will attach our new methods
+exports.newMomentProto = newMomentProto;
+var oldMomentProto = $.extend({}, newMomentProto); // copy of original moment methods
+exports.oldMomentProto = oldMomentProto;
+// tell momentjs to transfer these properties upon clone
+var momentProperties = moment.momentProperties;
+momentProperties.push('_fullCalendar');
+momentProperties.push('_ambigTime');
+momentProperties.push('_ambigZone');
+/*
+Call this if you want Moment's original format method to be used
+*/
+function oldMomentFormat(mom, formatStr) {
+ return oldMomentProto.format.call(mom, formatStr); // oldMomentProto defined in moment-ext.js
+}
+exports.oldMomentFormat = oldMomentFormat;
+// Creating
+// -------------------------------------------------------------------------------------------------
+// Creates a new moment, similar to the vanilla moment(...) constructor, but with
+// extra features (ambiguous time, enhanced formatting). When given an existing moment,
+// it will function as a clone (and retain the zone of the moment). Anything else will
+// result in a moment in the local zone.
+var momentExt = function () {
+ return makeMoment(arguments);
+};
+exports.default = momentExt;
+// Sames as momentExt, but forces the resulting moment to be in the UTC timezone.
+momentExt.utc = function () {
+ var mom = makeMoment(arguments, true);
+ // Force it into UTC because makeMoment doesn't guarantee it
+ // (if given a pre-existing moment for example)
+ if (mom.hasTime()) {
+ mom.utc();
+ }
+ return mom;
+};
+// Same as momentExt, but when given an ISO8601 string, the timezone offset is preserved.
+// ISO8601 strings with no timezone offset will become ambiguously zoned.
+momentExt.parseZone = function () {
+ return makeMoment(arguments, true, true);
+};
+// Builds an enhanced moment from args. When given an existing moment, it clones. When given a
+// native Date, or called with no arguments (the current time), the resulting moment will be local.
+// Anything else needs to be "parsed" (a string or an array), and will be affected by:
+// parseAsUTC - if there is no zone information, should we parse the input in UTC?
+// parseZone - if there is zone information, should we force the zone of the moment?
+function makeMoment(args, parseAsUTC, parseZone) {
+ if (parseAsUTC === void 0) { parseAsUTC = false; }
+ if (parseZone === void 0) { parseZone = false; }
+ var input = args[0];
+ var isSingleString = args.length == 1 && typeof input === 'string';
+ var isAmbigTime;
+ var isAmbigZone;
+ var ambigMatch;
+ var mom;
+ if (moment.isMoment(input) || util_1.isNativeDate(input) || input === undefined) {
+ mom = moment.apply(null, args);
+ }
+ else {
+ isAmbigTime = false;
+ isAmbigZone = false;
+ if (isSingleString) {
+ if (ambigDateOfMonthRegex.test(input)) {
+ // accept strings like '2014-05', but convert to the first of the month
+ input += '-01';
+ args = [input]; // for when we pass it on to moment's constructor
+ isAmbigTime = true;
+ isAmbigZone = true;
+ }
+ else if ((ambigMatch = ambigTimeOrZoneRegex.exec(input))) {
+ isAmbigTime = !ambigMatch[5]; // no time part?
+ isAmbigZone = true;
+ }
+ }
+ else if ($.isArray(input)) {
+ // arrays have no timezone information, so assume ambiguous zone
+ isAmbigZone = true;
+ }
+ // otherwise, probably a string with a format
+ if (parseAsUTC || isAmbigTime) {
+ mom = moment.utc.apply(moment, args);
+ }
+ else {
+ mom = moment.apply(null, args);
+ }
+ if (isAmbigTime) {
+ mom._ambigTime = true;
+ mom._ambigZone = true; // ambiguous time always means ambiguous zone
+ }
+ else if (parseZone) {
+ if (isAmbigZone) {
+ mom._ambigZone = true;
+ }
+ else if (isSingleString) {
+ mom.utcOffset(input); // if not a valid zone, will assign UTC
+ }
+ }
+ }
+ mom._fullCalendar = true; // flag for extended functionality
+ return mom;
+}
+// Week Number
+// -------------------------------------------------------------------------------------------------
+// Returns the week number, considering the locale's custom week number calcuation
+// `weeks` is an alias for `week`
+newMomentProto.week = newMomentProto.weeks = function (input) {
+ var weekCalc = this._locale._fullCalendar_weekCalc;
+ if (input == null && typeof weekCalc === 'function') {
+ return weekCalc(this);
+ }
+ else if (weekCalc === 'ISO') {
+ return oldMomentProto.isoWeek.apply(this, arguments); // ISO getter/setter
+ }
+ return oldMomentProto.week.apply(this, arguments); // local getter/setter
+};
+// Time-of-day
+// -------------------------------------------------------------------------------------------------
+// GETTER
+// Returns a Duration with the hours/minutes/seconds/ms values of the moment.
+// If the moment has an ambiguous time, a duration of 00:00 will be returned.
+//
+// SETTER
+// You can supply a Duration, a Moment, or a Duration-like argument.
+// When setting the time, and the moment has an ambiguous time, it then becomes unambiguous.
+newMomentProto.time = function (time) {
+ // Fallback to the original method (if there is one) if this moment wasn't created via FullCalendar.
+ // `time` is a generic enough method name where this precaution is necessary to avoid collisions w/ other plugins.
+ if (!this._fullCalendar) {
+ return oldMomentProto.time.apply(this, arguments);
+ }
+ if (time == null) {
+ return moment.duration({
+ hours: this.hours(),
+ minutes: this.minutes(),
+ seconds: this.seconds(),
+ milliseconds: this.milliseconds()
+ });
+ }
+ else {
+ this._ambigTime = false; // mark that the moment now has a time
+ if (!moment.isDuration(time) && !moment.isMoment(time)) {
+ time = moment.duration(time);
+ }
+ // The day value should cause overflow (so 24 hours becomes 00:00:00 of next day).
+ // Only for Duration times, not Moment times.
+ var dayHours = 0;
+ if (moment.isDuration(time)) {
+ dayHours = Math.floor(time.asDays()) * 24;
+ }
+ // We need to set the individual fields.
+ // Can't use startOf('day') then add duration. In case of DST at start of day.
+ return this.hours(dayHours + time.hours())
+ .minutes(time.minutes())
+ .seconds(time.seconds())
+ .milliseconds(time.milliseconds());
+ }
+};
+// Converts the moment to UTC, stripping out its time-of-day and timezone offset,
+// but preserving its YMD. A moment with a stripped time will display no time
+// nor timezone offset when .format() is called.
+newMomentProto.stripTime = function () {
+ if (!this._ambigTime) {
+ this.utc(true); // keepLocalTime=true (for keeping *date* value)
+ // set time to zero
+ this.set({
+ hours: 0,
+ minutes: 0,
+ seconds: 0,
+ ms: 0
+ });
+ // Mark the time as ambiguous. This needs to happen after the .utc() call, which might call .utcOffset(),
+ // which clears all ambig flags.
+ this._ambigTime = true;
+ this._ambigZone = true; // if ambiguous time, also ambiguous timezone offset
+ }
+ return this; // for chaining
+};
+// Returns if the moment has a non-ambiguous time (boolean)
+newMomentProto.hasTime = function () {
+ return !this._ambigTime;
+};
+// Timezone
+// -------------------------------------------------------------------------------------------------
+// Converts the moment to UTC, stripping out its timezone offset, but preserving its
+// YMD and time-of-day. A moment with a stripped timezone offset will display no
+// timezone offset when .format() is called.
+newMomentProto.stripZone = function () {
+ var wasAmbigTime;
+ if (!this._ambigZone) {
+ wasAmbigTime = this._ambigTime;
+ this.utc(true); // keepLocalTime=true (for keeping date and time values)
+ // the above call to .utc()/.utcOffset() unfortunately might clear the ambig flags, so restore
+ this._ambigTime = wasAmbigTime || false;
+ // Mark the zone as ambiguous. This needs to happen after the .utc() call, which might call .utcOffset(),
+ // which clears the ambig flags.
+ this._ambigZone = true;
+ }
+ return this; // for chaining
+};
+// Returns of the moment has a non-ambiguous timezone offset (boolean)
+newMomentProto.hasZone = function () {
+ return !this._ambigZone;
+};
+// implicitly marks a zone
+newMomentProto.local = function (keepLocalTime) {
+ // for when converting from ambiguously-zoned to local,
+ // keep the time values when converting from UTC -> local
+ oldMomentProto.local.call(this, this._ambigZone || keepLocalTime);
+ // ensure non-ambiguous
+ // this probably already happened via local() -> utcOffset(), but don't rely on Moment's internals
+ this._ambigTime = false;
+ this._ambigZone = false;
+ return this; // for chaining
+};
+// implicitly marks a zone
+newMomentProto.utc = function (keepLocalTime) {
+ oldMomentProto.utc.call(this, keepLocalTime);
+ // ensure non-ambiguous
+ // this probably already happened via utc() -> utcOffset(), but don't rely on Moment's internals
+ this._ambigTime = false;
+ this._ambigZone = false;
+ return this;
+};
+// implicitly marks a zone (will probably get called upon .utc() and .local())
+newMomentProto.utcOffset = function (tzo) {
+ if (tzo != null) {
+ // these assignments needs to happen before the original zone method is called.
+ // I forget why, something to do with a browser crash.
+ this._ambigTime = false;
+ this._ambigZone = false;
+ }
+ return oldMomentProto.utcOffset.apply(this, arguments);
+};
+
+
+/***/ }),
+/* 10 */
+/***/ (function(module, exports) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+/*
+Meant to be immutable
+*/
+var ComponentFootprint = /** @class */ (function () {
+ function ComponentFootprint(unzonedRange, isAllDay) {
+ this.isAllDay = false; // component can choose to ignore this
+ this.unzonedRange = unzonedRange;
+ this.isAllDay = isAllDay;
+ }
+ /*
+ Only works for non-open-ended ranges.
+ */
+ ComponentFootprint.prototype.toLegacy = function (calendar) {
+ return {
+ start: calendar.msToMoment(this.unzonedRange.startMs, this.isAllDay),
+ end: calendar.msToMoment(this.unzonedRange.endMs, this.isAllDay)
+ };
+ };
+ return ComponentFootprint;
+}());
+exports.default = ComponentFootprint;
+
+
+/***/ }),
+/* 11 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var EventDef_1 = __webpack_require__(22);
+var EventInstance_1 = __webpack_require__(51);
+var EventDateProfile_1 = __webpack_require__(15);
+var SingleEventDef = /** @class */ (function (_super) {
+ tslib_1.__extends(SingleEventDef, _super);
+ function SingleEventDef() {
+ return _super !== null && _super.apply(this, arguments) || this;
+ }
+ /*
+ Will receive start/end params, but will be ignored.
+ */
+ SingleEventDef.prototype.buildInstances = function () {
+ return [this.buildInstance()];
+ };
+ SingleEventDef.prototype.buildInstance = function () {
+ return new EventInstance_1.default(this, // definition
+ this.dateProfile);
+ };
+ SingleEventDef.prototype.isAllDay = function () {
+ return this.dateProfile.isAllDay();
+ };
+ SingleEventDef.prototype.clone = function () {
+ var def = _super.prototype.clone.call(this);
+ def.dateProfile = this.dateProfile;
+ return def;
+ };
+ SingleEventDef.prototype.rezone = function () {
+ var calendar = this.source.calendar;
+ var dateProfile = this.dateProfile;
+ this.dateProfile = new EventDateProfile_1.default(calendar.moment(dateProfile.start), dateProfile.end ? calendar.moment(dateProfile.end) : null, calendar);
+ };
+ /*
+ NOTE: if super-method fails, should still attempt to apply
+ */
+ SingleEventDef.prototype.applyManualStandardProps = function (rawProps) {
+ var superSuccess = _super.prototype.applyManualStandardProps.call(this, rawProps);
+ var dateProfile = EventDateProfile_1.default.parse(rawProps, this.source); // returns null on failure
+ if (dateProfile) {
+ this.dateProfile = dateProfile;
+ // make sure `date` shows up in the legacy event objects as-is
+ if (rawProps.date != null) {
+ this.miscProps.date = rawProps.date;
+ }
+ return superSuccess;
+ }
+ else {
+ return false;
+ }
+ };
+ return SingleEventDef;
+}(EventDef_1.default));
+exports.default = SingleEventDef;
+// Parsing
+// ---------------------------------------------------------------------------------------------------------------------
+SingleEventDef.defineStandardProps({
+ start: false,
+ date: false,
+ end: false,
+ allDay: false
+});
+
+
+/***/ }),
+/* 12 */
+/***/ (function(module, exports) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var Mixin = /** @class */ (function () {
+ function Mixin() {
+ }
+ Mixin.mixInto = function (destClass) {
+ var _this = this;
+ Object.getOwnPropertyNames(this.prototype).forEach(function (name) {
+ if (!destClass.prototype[name]) {
+ destClass.prototype[name] = _this.prototype[name];
+ }
+ });
+ };
+ /*
+ will override existing methods
+ */
+ Mixin.mixOver = function (destClass) {
+ var _this = this;
+ Object.getOwnPropertyNames(this.prototype).forEach(function (name) {
+ destClass.prototype[name] = _this.prototype[name];
+ });
+ };
+ return Mixin;
+}());
+exports.default = Mixin;
+
+
+/***/ }),
+/* 13 */
+/***/ (function(module, exports) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var Interaction = /** @class */ (function () {
+ function Interaction(component) {
+ this.view = component._getView();
+ this.component = component;
+ }
+ Interaction.prototype.opt = function (name) {
+ return this.view.opt(name);
+ };
+ Interaction.prototype.end = function () {
+ // subclasses can implement
+ };
+ return Interaction;
+}());
+exports.default = Interaction;
+
+
+/***/ }),
+/* 14 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var $ = __webpack_require__(1);
+var namespace_hooks_1 = __webpack_require__(7);
+var EmitterMixin_1 = __webpack_require__(8);
+var ListenerMixin_1 = __webpack_require__(6);
+var globalEmitter = null;
+var neededCount = 0;
+/*
+Listens to document and window-level user-interaction events, like touch events and mouse events,
+and fires these events as-is to whoever is observing a GlobalEmitter.
+Best when used as a singleton via GlobalEmitter.get()
+
+Normalizes mouse/touch events. For examples:
+- ignores the the simulated mouse events that happen after a quick tap: mousemove+mousedown+mouseup+click
+- compensates for various buggy scenarios where a touchend does not fire
+*/
+var GlobalEmitter = /** @class */ (function () {
+ function GlobalEmitter() {
+ this.isTouching = false;
+ this.mouseIgnoreDepth = 0;
+ }
+ GlobalEmitter.prototype.bind = function () {
+ var _this = this;
+ this.listenTo($(document), {
+ touchstart: this.handleTouchStart,
+ touchcancel: this.handleTouchCancel,
+ touchend: this.handleTouchEnd,
+ mousedown: this.handleMouseDown,
+ mousemove: this.handleMouseMove,
+ mouseup: this.handleMouseUp,
+ click: this.handleClick,
+ selectstart: this.handleSelectStart,
+ contextmenu: this.handleContextMenu
+ });
+ // because we need to call preventDefault
+ // because https://www.chromestatus.com/features/5093566007214080
+ // TODO: investigate performance because this is a global handler
+ window.addEventListener('touchmove', this.handleTouchMoveProxy = function (ev) {
+ _this.handleTouchMove($.Event(ev));
+ }, { passive: false } // allows preventDefault()
+ );
+ // attach a handler to get called when ANY scroll action happens on the page.
+ // this was impossible to do with normal on/off because 'scroll' doesn't bubble.
+ // http://stackoverflow.com/a/32954565/96342
+ window.addEventListener('scroll', this.handleScrollProxy = function (ev) {
+ _this.handleScroll($.Event(ev));
+ }, true // useCapture
+ );
+ };
+ GlobalEmitter.prototype.unbind = function () {
+ this.stopListeningTo($(document));
+ window.removeEventListener('touchmove', this.handleTouchMoveProxy);
+ window.removeEventListener('scroll', this.handleScrollProxy, true // useCapture
+ );
+ };
+ // Touch Handlers
+ // -----------------------------------------------------------------------------------------------------------------
+ GlobalEmitter.prototype.handleTouchStart = function (ev) {
+ // if a previous touch interaction never ended with a touchend, then implicitly end it,
+ // but since a new touch interaction is about to begin, don't start the mouse ignore period.
+ this.stopTouch(ev, true); // skipMouseIgnore=true
+ this.isTouching = true;
+ this.trigger('touchstart', ev);
+ };
+ GlobalEmitter.prototype.handleTouchMove = function (ev) {
+ if (this.isTouching) {
+ this.trigger('touchmove', ev);
+ }
+ };
+ GlobalEmitter.prototype.handleTouchCancel = function (ev) {
+ if (this.isTouching) {
+ this.trigger('touchcancel', ev);
+ // Have touchcancel fire an artificial touchend. That way, handlers won't need to listen to both.
+ // If touchend fires later, it won't have any effect b/c isTouching will be false.
+ this.stopTouch(ev);
+ }
+ };
+ GlobalEmitter.prototype.handleTouchEnd = function (ev) {
+ this.stopTouch(ev);
+ };
+ // Mouse Handlers
+ // -----------------------------------------------------------------------------------------------------------------
+ GlobalEmitter.prototype.handleMouseDown = function (ev) {
+ if (!this.shouldIgnoreMouse()) {
+ this.trigger('mousedown', ev);
+ }
+ };
+ GlobalEmitter.prototype.handleMouseMove = function (ev) {
+ if (!this.shouldIgnoreMouse()) {
+ this.trigger('mousemove', ev);
+ }
+ };
+ GlobalEmitter.prototype.handleMouseUp = function (ev) {
+ if (!this.shouldIgnoreMouse()) {
+ this.trigger('mouseup', ev);
+ }
+ };
+ GlobalEmitter.prototype.handleClick = function (ev) {
+ if (!this.shouldIgnoreMouse()) {
+ this.trigger('click', ev);
+ }
+ };
+ // Misc Handlers
+ // -----------------------------------------------------------------------------------------------------------------
+ GlobalEmitter.prototype.handleSelectStart = function (ev) {
+ this.trigger('selectstart', ev);
+ };
+ GlobalEmitter.prototype.handleContextMenu = function (ev) {
+ this.trigger('contextmenu', ev);
+ };
+ GlobalEmitter.prototype.handleScroll = function (ev) {
+ this.trigger('scroll', ev);
+ };
+ // Utils
+ // -----------------------------------------------------------------------------------------------------------------
+ GlobalEmitter.prototype.stopTouch = function (ev, skipMouseIgnore) {
+ if (skipMouseIgnore === void 0) { skipMouseIgnore = false; }
+ if (this.isTouching) {
+ this.isTouching = false;
+ this.trigger('touchend', ev);
+ if (!skipMouseIgnore) {
+ this.startTouchMouseIgnore();
+ }
+ }
+ };
+ GlobalEmitter.prototype.startTouchMouseIgnore = function () {
+ var _this = this;
+ var wait = namespace_hooks_1.default.touchMouseIgnoreWait;
+ if (wait) {
+ this.mouseIgnoreDepth++;
+ setTimeout(function () {
+ _this.mouseIgnoreDepth--;
+ }, wait);
+ }
+ };
+ GlobalEmitter.prototype.shouldIgnoreMouse = function () {
+ return this.isTouching || Boolean(this.mouseIgnoreDepth);
+ };
+ // Singleton
+ // -----------------------------------------------------------------------------------------------------------------
+ // gets the singleton
+ GlobalEmitter.get = function () {
+ if (!globalEmitter) {
+ globalEmitter = new GlobalEmitter();
+ globalEmitter.bind();
+ }
+ return globalEmitter;
+ };
+ // called when an object knows it will need a GlobalEmitter in the near future.
+ GlobalEmitter.needed = function () {
+ GlobalEmitter.get(); // ensures globalEmitter
+ neededCount++;
+ };
+ // called when the object that originally called needed() doesn't need a GlobalEmitter anymore.
+ GlobalEmitter.unneeded = function () {
+ neededCount--;
+ if (!neededCount) {
+ globalEmitter.unbind();
+ globalEmitter = null;
+ }
+ };
+ return GlobalEmitter;
+}());
+exports.default = GlobalEmitter;
+ListenerMixin_1.default.mixInto(GlobalEmitter);
+EmitterMixin_1.default.mixInto(GlobalEmitter);
+
+
+/***/ }),
+/* 15 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var UnzonedRange_1 = __webpack_require__(4);
+/*
+Meant to be immutable
+*/
+var EventDateProfile = /** @class */ (function () {
+ function EventDateProfile(start, end, calendar) {
+ this.start = start;
+ this.end = end || null;
+ this.unzonedRange = this.buildUnzonedRange(calendar);
+ }
+ EventDateProfile.prototype.isAllDay = function () {
+ return !(this.start.hasTime() || (this.end && this.end.hasTime()));
+ };
+ /*
+ Needs a Calendar object
+ */
+ EventDateProfile.prototype.buildUnzonedRange = function (calendar) {
+ var startMs = this.start.clone().stripZone().valueOf();
+ var endMs = this.getEnd(calendar).stripZone().valueOf();
+ return new UnzonedRange_1.default(startMs, endMs);
+ };
+ /*
+ Needs a Calendar object
+ */
+ EventDateProfile.prototype.getEnd = function (calendar) {
+ return this.end ?
+ this.end.clone() :
+ // derive the end from the start and allDay. compute allDay if necessary
+ calendar.getDefaultEventEnd(this.isAllDay(), this.start);
+ };
+ EventDateProfile.isStandardProp = function (propName) {
+ return propName === 'start' || propName === 'date' || propName === 'end' || propName === 'allDay';
+ };
+ /*
+ Needs an EventSource object
+ */
+ EventDateProfile.parse = function (rawProps, source) {
+ var startInput = rawProps.start || rawProps.date;
+ var endInput = rawProps.end;
+ if (!startInput) {
+ return false;
+ }
+ var calendar = source.calendar;
+ var start = calendar.moment(startInput);
+ var end = endInput ? calendar.moment(endInput) : null;
+ var forcedAllDay = rawProps.allDay;
+ var forceEventDuration = calendar.opt('forceEventDuration');
+ if (!start.isValid()) {
+ return false;
+ }
+ if (end && (!end.isValid() || !end.isAfter(start))) {
+ end = null;
+ }
+ if (forcedAllDay == null) {
+ forcedAllDay = source.allDayDefault;
+ if (forcedAllDay == null) {
+ forcedAllDay = calendar.opt('allDayDefault');
+ }
+ }
+ if (forcedAllDay === true) {
+ start.stripTime();
+ if (end) {
+ end.stripTime();
+ }
+ }
+ else if (forcedAllDay === false) {
+ if (!start.hasTime()) {
+ start.time(0);
+ }
+ if (end && !end.hasTime()) {
+ end.time(0);
+ }
+ }
+ if (!end && forceEventDuration) {
+ end = calendar.getDefaultEventEnd(!start.hasTime(), start);
+ }
+ return new EventDateProfile(start, end, calendar);
+ };
+ return EventDateProfile;
+}());
+exports.default = EventDateProfile;
+
+
+/***/ }),
+/* 16 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var $ = __webpack_require__(1);
+var PromiseStub = {
+ construct: function (executor) {
+ var deferred = $.Deferred();
+ var promise = deferred.promise();
+ if (typeof executor === 'function') {
+ executor(function (val) {
+ deferred.resolve(val);
+ attachImmediatelyResolvingThen(promise, val);
+ }, function () {
+ deferred.reject();
+ attachImmediatelyRejectingThen(promise);
+ });
+ }
+ return promise;
+ },
+ resolve: function (val) {
+ var deferred = $.Deferred().resolve(val);
+ var promise = deferred.promise();
+ attachImmediatelyResolvingThen(promise, val);
+ return promise;
+ },
+ reject: function () {
+ var deferred = $.Deferred().reject();
+ var promise = deferred.promise();
+ attachImmediatelyRejectingThen(promise);
+ return promise;
+ }
+};
+exports.default = PromiseStub;
+function attachImmediatelyResolvingThen(promise, val) {
+ promise.then = function (onResolve) {
+ if (typeof onResolve === 'function') {
+ return PromiseStub.resolve(onResolve(val));
+ }
+ return promise;
+ };
+}
+function attachImmediatelyRejectingThen(promise) {
+ promise.then = function (onResolve, onReject) {
+ if (typeof onReject === 'function') {
+ onReject();
+ }
+ return promise;
+ };
+}
+
+
+/***/ }),
+/* 17 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var UnzonedRange_1 = __webpack_require__(4);
+var util_1 = __webpack_require__(23);
+var EventRange_1 = __webpack_require__(53);
+/*
+It's expected that there will be at least one EventInstance,
+OR that an explicitEventDef is assigned.
+*/
+var EventInstanceGroup = /** @class */ (function () {
+ function EventInstanceGroup(eventInstances) {
+ this.eventInstances = eventInstances || [];
+ }
+ EventInstanceGroup.prototype.getAllEventRanges = function (constraintRange) {
+ if (constraintRange) {
+ return this.sliceNormalRenderRanges(constraintRange);
+ }
+ else {
+ return this.eventInstances.map(util_1.eventInstanceToEventRange);
+ }
+ };
+ EventInstanceGroup.prototype.sliceRenderRanges = function (constraintRange) {
+ if (this.isInverse()) {
+ return this.sliceInverseRenderRanges(constraintRange);
+ }
+ else {
+ return this.sliceNormalRenderRanges(constraintRange);
+ }
+ };
+ EventInstanceGroup.prototype.sliceNormalRenderRanges = function (constraintRange) {
+ var eventInstances = this.eventInstances;
+ var i, eventInstance;
+ var slicedRange;
+ var slicedEventRanges = [];
+ for (i = 0; i < eventInstances.length; i++) {
+ eventInstance = eventInstances[i];
+ slicedRange = eventInstance.dateProfile.unzonedRange.intersect(constraintRange);
+ if (slicedRange) {
+ slicedEventRanges.push(new EventRange_1.default(slicedRange, eventInstance.def, eventInstance));
+ }
+ }
+ return slicedEventRanges;
+ };
+ EventInstanceGroup.prototype.sliceInverseRenderRanges = function (constraintRange) {
+ var unzonedRanges = this.eventInstances.map(util_1.eventInstanceToUnzonedRange);
+ var ownerDef = this.getEventDef();
+ unzonedRanges = UnzonedRange_1.default.invertRanges(unzonedRanges, constraintRange);
+ return unzonedRanges.map(function (unzonedRange) {
+ return new EventRange_1.default(unzonedRange, ownerDef); // don't give an EventInstance
+ });
+ };
+ EventInstanceGroup.prototype.isInverse = function () {
+ return this.getEventDef().hasInverseRendering();
+ };
+ EventInstanceGroup.prototype.getEventDef = function () {
+ return this.explicitEventDef || this.eventInstances[0].def;
+ };
+ return EventInstanceGroup;
+}());
+exports.default = EventInstanceGroup;
+
+
+/***/ }),
+/* 18 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var util_1 = __webpack_require__(2);
+var DragListener_1 = __webpack_require__(39);
+/* Tracks mouse movements over a component and raises events about which hit the mouse is over.
+------------------------------------------------------------------------------------------------------------------------
+options:
+- subjectEl
+- subjectCenter
+*/
+var HitDragListener = /** @class */ (function (_super) {
+ tslib_1.__extends(HitDragListener, _super);
+ function HitDragListener(component, options) {
+ var _this = _super.call(this, options) || this;
+ _this.component = component;
+ return _this;
+ }
+ // Called when drag listening starts (but a real drag has not necessarily began).
+ // ev might be undefined if dragging was started manually.
+ HitDragListener.prototype.handleInteractionStart = function (ev) {
+ var subjectEl = this.subjectEl;
+ var subjectRect;
+ var origPoint;
+ var point;
+ this.component.hitsNeeded();
+ this.computeScrollBounds(); // for autoscroll
+ if (ev) {
+ origPoint = { left: util_1.getEvX(ev), top: util_1.getEvY(ev) };
+ point = origPoint;
+ // constrain the point to bounds of the element being dragged
+ if (subjectEl) {
+ subjectRect = util_1.getOuterRect(subjectEl); // used for centering as well
+ point = util_1.constrainPoint(point, subjectRect);
+ }
+ this.origHit = this.queryHit(point.left, point.top);
+ // treat the center of the subject as the collision point?
+ if (subjectEl && this.options.subjectCenter) {
+ // only consider the area the subject overlaps the hit. best for large subjects.
+ // TODO: skip this if hit didn't supply left/right/top/bottom
+ if (this.origHit) {
+ subjectRect = util_1.intersectRects(this.origHit, subjectRect) ||
+ subjectRect; // in case there is no intersection
+ }
+ point = util_1.getRectCenter(subjectRect);
+ }
+ this.coordAdjust = util_1.diffPoints(point, origPoint); // point - origPoint
+ }
+ else {
+ this.origHit = null;
+ this.coordAdjust = null;
+ }
+ // call the super-method. do it after origHit has been computed
+ _super.prototype.handleInteractionStart.call(this, ev);
+ };
+ // Called when the actual drag has started
+ HitDragListener.prototype.handleDragStart = function (ev) {
+ var hit;
+ _super.prototype.handleDragStart.call(this, ev);
+ // might be different from this.origHit if the min-distance is large
+ hit = this.queryHit(util_1.getEvX(ev), util_1.getEvY(ev));
+ // report the initial hit the mouse is over
+ // especially important if no min-distance and drag starts immediately
+ if (hit) {
+ this.handleHitOver(hit);
+ }
+ };
+ // Called when the drag moves
+ HitDragListener.prototype.handleDrag = function (dx, dy, ev) {
+ var hit;
+ _super.prototype.handleDrag.call(this, dx, dy, ev);
+ hit = this.queryHit(util_1.getEvX(ev), util_1.getEvY(ev));
+ if (!isHitsEqual(hit, this.hit)) {
+ if (this.hit) {
+ this.handleHitOut();
+ }
+ if (hit) {
+ this.handleHitOver(hit);
+ }
+ }
+ };
+ // Called when dragging has been stopped
+ HitDragListener.prototype.handleDragEnd = function (ev) {
+ this.handleHitDone();
+ _super.prototype.handleDragEnd.call(this, ev);
+ };
+ // Called when a the mouse has just moved over a new hit
+ HitDragListener.prototype.handleHitOver = function (hit) {
+ var isOrig = isHitsEqual(hit, this.origHit);
+ this.hit = hit;
+ this.trigger('hitOver', this.hit, isOrig, this.origHit);
+ };
+ // Called when the mouse has just moved out of a hit
+ HitDragListener.prototype.handleHitOut = function () {
+ if (this.hit) {
+ this.trigger('hitOut', this.hit);
+ this.handleHitDone();
+ this.hit = null;
+ }
+ };
+ // Called after a hitOut. Also called before a dragStop
+ HitDragListener.prototype.handleHitDone = function () {
+ if (this.hit) {
+ this.trigger('hitDone', this.hit);
+ }
+ };
+ // Called when the interaction ends, whether there was a real drag or not
+ HitDragListener.prototype.handleInteractionEnd = function (ev, isCancelled) {
+ _super.prototype.handleInteractionEnd.call(this, ev, isCancelled);
+ this.origHit = null;
+ this.hit = null;
+ this.component.hitsNotNeeded();
+ };
+ // Called when scrolling has stopped, whether through auto scroll, or the user scrolling
+ HitDragListener.prototype.handleScrollEnd = function () {
+ _super.prototype.handleScrollEnd.call(this);
+ // hits' absolute positions will be in new places after a user's scroll.
+ // HACK for recomputing.
+ if (this.isDragging) {
+ this.component.releaseHits();
+ this.component.prepareHits();
+ }
+ };
+ // Gets the hit underneath the coordinates for the given mouse event
+ HitDragListener.prototype.queryHit = function (left, top) {
+ if (this.coordAdjust) {
+ left += this.coordAdjust.left;
+ top += this.coordAdjust.top;
+ }
+ return this.component.queryHit(left, top);
+ };
+ return HitDragListener;
+}(DragListener_1.default));
+exports.default = HitDragListener;
+// Returns `true` if the hits are identically equal. `false` otherwise. Must be from the same component.
+// Two null values will be considered equal, as two "out of the component" states are the same.
+function isHitsEqual(hit0, hit1) {
+ if (!hit0 && !hit1) {
+ return true;
+ }
+ if (hit0 && hit1) {
+ return hit0.component === hit1.component &&
+ isHitPropsWithin(hit0, hit1) &&
+ isHitPropsWithin(hit1, hit0); // ensures all props are identical
+ }
+ return false;
+}
+// Returns true if all of subHit's non-standard properties are within superHit
+function isHitPropsWithin(subHit, superHit) {
+ for (var propName in subHit) {
+ if (!/^(component|left|right|top|bottom)$/.test(propName)) {
+ if (subHit[propName] !== superHit[propName]) {
+ return false;
+ }
+ }
+ }
+ return true;
+}
+
+
+/***/ }),
+/* 19 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var util_1 = __webpack_require__(2);
+exports.globalDefaults = {
+ titleRangeSeparator: ' \u2013 ',
+ monthYearFormat: 'MMMM YYYY',
+ defaultTimedEventDuration: '02:00:00',
+ defaultAllDayEventDuration: { days: 1 },
+ forceEventDuration: false,
+ nextDayThreshold: '09:00:00',
+ // display
+ columnHeader: true,
+ defaultView: 'month',
+ aspectRatio: 1.35,
+ header: {
+ left: 'title',
+ center: '',
+ right: 'today prev,next'
+ },
+ weekends: true,
+ weekNumbers: false,
+ weekNumberTitle: 'W',
+ weekNumberCalculation: 'local',
+ //editable: false,
+ //nowIndicator: false,
+ scrollTime: '06:00:00',
+ minTime: '00:00:00',
+ maxTime: '24:00:00',
+ showNonCurrentDates: true,
+ // event ajax
+ lazyFetching: true,
+ startParam: 'start',
+ endParam: 'end',
+ timezoneParam: 'timezone',
+ timezone: false,
+ //allDayDefault: undefined,
+ // locale
+ locale: null,
+ isRTL: false,
+ buttonText: {
+ prev: "prev",
+ next: "next",
+ prevYear: "prev year",
+ nextYear: "next year",
+ year: 'year',
+ today: 'today',
+ month: 'month',
+ week: 'week',
+ day: 'day'
+ },
+ //buttonIcons: null,
+ allDayText: 'all-day',
+ // allows setting a min-height to the event segment to prevent short events overlapping each other
+ agendaEventMinHeight: 0,
+ // jquery-ui theming
+ theme: false,
+ //themeButtonIcons: null,
+ //eventResizableFromStart: false,
+ dragOpacity: .75,
+ dragRevertDuration: 500,
+ dragScroll: true,
+ //selectable: false,
+ unselectAuto: true,
+ //selectMinDistance: 0,
+ dropAccept: '*',
+ eventOrder: 'title',
+ //eventRenderWait: null,
+ eventLimit: false,
+ eventLimitText: 'more',
+ eventLimitClick: 'popover',
+ dayPopoverFormat: 'LL',
+ handleWindowResize: true,
+ windowResizeDelay: 100,
+ longPressDelay: 1000
+};
+exports.englishDefaults = {
+ dayPopoverFormat: 'dddd, MMMM D'
+};
+exports.rtlDefaults = {
+ header: {
+ left: 'next,prev today',
+ center: '',
+ right: 'title'
+ },
+ buttonIcons: {
+ prev: 'right-single-arrow',
+ next: 'left-single-arrow',
+ prevYear: 'right-double-arrow',
+ nextYear: 'left-double-arrow'
+ },
+ themeButtonIcons: {
+ prev: 'circle-triangle-e',
+ next: 'circle-triangle-w',
+ nextYear: 'seek-prev',
+ prevYear: 'seek-next'
+ }
+};
+var complexOptions = [
+ 'header',
+ 'footer',
+ 'buttonText',
+ 'buttonIcons',
+ 'themeButtonIcons'
+];
+// Merges an array of option objects into a single object
+function mergeOptions(optionObjs) {
+ return util_1.mergeProps(optionObjs, complexOptions);
+}
+exports.mergeOptions = mergeOptions;
+
+
+/***/ }),
+/* 20 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var $ = __webpack_require__(1);
+var moment = __webpack_require__(3);
+var namespace_hooks_1 = __webpack_require__(7);
+var options_1 = __webpack_require__(19);
+var util_1 = __webpack_require__(2);
+var localeOptionHash = namespace_hooks_1.default.locales;
+exports.localeOptionHash = localeOptionHash;
+// Initialize jQuery UI datepicker translations while using some of the translations
+// Will set this as the default locales for datepicker.
+function datepickerLocale(localeCode, dpLocaleCode, dpOptions) {
+ // get the FullCalendar internal option hash for this locale. create if necessary
+ var fcOptions = localeOptionHash[localeCode] || (localeOptionHash[localeCode] = {});
+ // transfer some simple options from datepicker to fc
+ fcOptions.isRTL = dpOptions.isRTL;
+ fcOptions.weekNumberTitle = dpOptions.weekHeader;
+ // compute some more complex options from datepicker
+ $.each(dpComputableOptions, function (name, func) {
+ fcOptions[name] = func(dpOptions);
+ });
+ var jqDatePicker = $.datepicker;
+ // is jQuery UI Datepicker is on the page?
+ if (jqDatePicker) {
+ // Register the locale data.
+ // FullCalendar and MomentJS use locale codes like "pt-br" but Datepicker
+ // does it like "pt-BR" or if it doesn't have the locale, maybe just "pt".
+ // Make an alias so the locale can be referenced either way.
+ jqDatePicker.regional[dpLocaleCode] =
+ jqDatePicker.regional[localeCode] = // alias
+ dpOptions;
+ // Alias 'en' to the default locale data. Do this every time.
+ jqDatePicker.regional.en = jqDatePicker.regional[''];
+ // Set as Datepicker's global defaults.
+ jqDatePicker.setDefaults(dpOptions);
+ }
+}
+exports.datepickerLocale = datepickerLocale;
+// Sets FullCalendar-specific translations. Will set the locales as the global default.
+function locale(localeCode, newFcOptions) {
+ var fcOptions;
+ var momOptions;
+ // get the FullCalendar internal option hash for this locale. create if necessary
+ fcOptions = localeOptionHash[localeCode] || (localeOptionHash[localeCode] = {});
+ // provided new options for this locales? merge them in
+ if (newFcOptions) {
+ fcOptions = localeOptionHash[localeCode] = options_1.mergeOptions([fcOptions, newFcOptions]);
+ }
+ // compute locale options that weren't defined.
+ // always do this. newFcOptions can be undefined when initializing from i18n file,
+ // so no way to tell if this is an initialization or a default-setting.
+ momOptions = getMomentLocaleData(localeCode); // will fall back to en
+ $.each(momComputableOptions, function (name, func) {
+ if (fcOptions[name] == null) {
+ fcOptions[name] = func(momOptions, fcOptions);
+ }
+ });
+ // set it as the default locale for FullCalendar
+ options_1.globalDefaults.locale = localeCode;
+}
+exports.locale = locale;
+;
+// NOTE: can't guarantee any of these computations will run because not every locale has datepicker
+// configs, so make sure there are English fallbacks for these in the defaults file.
+var dpComputableOptions = {
+ buttonText: function (dpOptions) {
+ return {
+ // the translations sometimes wrongly contain HTML entities
+ prev: util_1.stripHtmlEntities(dpOptions.prevText),
+ next: util_1.stripHtmlEntities(dpOptions.nextText),
+ today: util_1.stripHtmlEntities(dpOptions.currentText)
+ };
+ },
+ // Produces format strings like "MMMM YYYY" -> "September 2014"
+ monthYearFormat: function (dpOptions) {
+ return dpOptions.showMonthAfterYear ?
+ 'YYYY[' + dpOptions.yearSuffix + '] MMMM' :
+ 'MMMM YYYY[' + dpOptions.yearSuffix + ']';
+ }
+};
+var momComputableOptions = {
+ // Produces format strings like "ddd M/D" -> "Fri 9/15"
+ dayOfMonthFormat: function (momOptions, fcOptions) {
+ var format = momOptions.longDateFormat('l'); // for the format like "M/D/YYYY"
+ // strip the year off the edge, as well as other misc non-whitespace chars
+ format = format.replace(/^Y+[^\w\s]*|[^\w\s]*Y+$/g, '');
+ if (fcOptions.isRTL) {
+ format += ' ddd'; // for RTL, add day-of-week to end
+ }
+ else {
+ format = 'ddd ' + format; // for LTR, add day-of-week to beginning
+ }
+ return format;
+ },
+ // Produces format strings like "h:mma" -> "6:00pm"
+ mediumTimeFormat: function (momOptions) {
+ return momOptions.longDateFormat('LT')
+ .replace(/\s*a$/i, 'a'); // convert AM/PM/am/pm to lowercase. remove any spaces beforehand
+ },
+ // Produces format strings like "h(:mm)a" -> "6pm" / "6:30pm"
+ smallTimeFormat: function (momOptions) {
+ return momOptions.longDateFormat('LT')
+ .replace(':mm', '(:mm)')
+ .replace(/(\Wmm)$/, '($1)') // like above, but for foreign locales
+ .replace(/\s*a$/i, 'a'); // convert AM/PM/am/pm to lowercase. remove any spaces beforehand
+ },
+ // Produces format strings like "h(:mm)t" -> "6p" / "6:30p"
+ extraSmallTimeFormat: function (momOptions) {
+ return momOptions.longDateFormat('LT')
+ .replace(':mm', '(:mm)')
+ .replace(/(\Wmm)$/, '($1)') // like above, but for foreign locales
+ .replace(/\s*a$/i, 't'); // convert to AM/PM/am/pm to lowercase one-letter. remove any spaces beforehand
+ },
+ // Produces format strings like "ha" / "H" -> "6pm" / "18"
+ hourFormat: function (momOptions) {
+ return momOptions.longDateFormat('LT')
+ .replace(':mm', '')
+ .replace(/(\Wmm)$/, '') // like above, but for foreign locales
+ .replace(/\s*a$/i, 'a'); // convert AM/PM/am/pm to lowercase. remove any spaces beforehand
+ },
+ // Produces format strings like "h:mm" -> "6:30" (with no AM/PM)
+ noMeridiemTimeFormat: function (momOptions) {
+ return momOptions.longDateFormat('LT')
+ .replace(/\s*a$/i, ''); // remove trailing AM/PM
+ }
+};
+// options that should be computed off live calendar options (considers override options)
+// TODO: best place for this? related to locale?
+// TODO: flipping text based on isRTL is a bad idea because the CSS `direction` might want to handle it
+var instanceComputableOptions = {
+ // Produces format strings for results like "Mo 16"
+ smallDayDateFormat: function (options) {
+ return options.isRTL ?
+ 'D dd' :
+ 'dd D';
+ },
+ // Produces format strings for results like "Wk 5"
+ weekFormat: function (options) {
+ return options.isRTL ?
+ 'w[ ' + options.weekNumberTitle + ']' :
+ '[' + options.weekNumberTitle + ' ]w';
+ },
+ // Produces format strings for results like "Wk5"
+ smallWeekFormat: function (options) {
+ return options.isRTL ?
+ 'w[' + options.weekNumberTitle + ']' :
+ '[' + options.weekNumberTitle + ']w';
+ }
+};
+// TODO: make these computable properties in optionsManager
+function populateInstanceComputableOptions(options) {
+ $.each(instanceComputableOptions, function (name, func) {
+ if (options[name] == null) {
+ options[name] = func(options);
+ }
+ });
+}
+exports.populateInstanceComputableOptions = populateInstanceComputableOptions;
+// Returns moment's internal locale data. If doesn't exist, returns English.
+function getMomentLocaleData(localeCode) {
+ return moment.localeData(localeCode) || moment.localeData('en');
+}
+exports.getMomentLocaleData = getMomentLocaleData;
+// Initialize English by forcing computation of moment-derived options.
+// Also, sets it as the default.
+locale('en', options_1.englishDefaults);
+
+
+/***/ }),
+/* 21 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var util_1 = __webpack_require__(2);
+// Class that all other classes will inherit from
+var Class = /** @class */ (function () {
+ function Class() {
+ }
+ // Called on a class to create a subclass.
+ // LIMITATION: cannot provide a constructor!
+ Class.extend = function (members) {
+ var SubClass = /** @class */ (function (_super) {
+ tslib_1.__extends(SubClass, _super);
+ function SubClass() {
+ return _super !== null && _super.apply(this, arguments) || this;
+ }
+ return SubClass;
+ }(this));
+ ;
+ util_1.copyOwnProps(members, SubClass.prototype);
+ return SubClass;
+ };
+ // Adds new member variables/methods to the class's prototype.
+ // Can be called with another class, or a plain object hash containing new members.
+ Class.mixin = function (members) {
+ util_1.copyOwnProps(members, this.prototype);
+ };
+ return Class;
+}());
+exports.default = Class;
+
+
+/***/ }),
+/* 22 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var $ = __webpack_require__(1);
+var ParsableModelMixin_1 = __webpack_require__(50);
+var EventDef = /** @class */ (function () {
+ function EventDef(source) {
+ this.source = source;
+ this.className = [];
+ this.miscProps = {};
+ }
+ EventDef.prototype.clone = function () {
+ var copy = new this.constructor(this.source);
+ copy.id = this.id;
+ copy.rawId = this.rawId;
+ copy.uid = this.uid; // not really unique anymore :(
+ EventDef.copyVerbatimStandardProps(this, copy);
+ copy.className = this.className.slice(); // copy
+ copy.miscProps = $.extend({}, this.miscProps);
+ return copy;
+ };
+ EventDef.prototype.hasInverseRendering = function () {
+ return this.getRendering() === 'inverse-background';
+ };
+ EventDef.prototype.hasBgRendering = function () {
+ var rendering = this.getRendering();
+ return rendering === 'inverse-background' || rendering === 'background';
+ };
+ EventDef.prototype.getRendering = function () {
+ if (this.rendering != null) {
+ return this.rendering;
+ }
+ return this.source.rendering;
+ };
+ EventDef.prototype.getConstraint = function () {
+ if (this.constraint != null) {
+ return this.constraint;
+ }
+ if (this.source.constraint != null) {
+ return this.source.constraint;
+ }
+ return this.source.calendar.opt('eventConstraint'); // what about View option?
+ };
+ EventDef.prototype.getOverlap = function () {
+ if (this.overlap != null) {
+ return this.overlap;
+ }
+ if (this.source.overlap != null) {
+ return this.source.overlap;
+ }
+ return this.source.calendar.opt('eventOverlap'); // what about View option?
+ };
+ EventDef.prototype.isStartExplicitlyEditable = function () {
+ if (this.startEditable != null) {
+ return this.startEditable;
+ }
+ return this.source.startEditable;
+ };
+ EventDef.prototype.isDurationExplicitlyEditable = function () {
+ if (this.durationEditable != null) {
+ return this.durationEditable;
+ }
+ return this.source.durationEditable;
+ };
+ EventDef.prototype.isExplicitlyEditable = function () {
+ if (this.editable != null) {
+ return this.editable;
+ }
+ return this.source.editable;
+ };
+ EventDef.prototype.toLegacy = function () {
+ var obj = $.extend({}, this.miscProps);
+ obj._id = this.uid;
+ obj.source = this.source;
+ obj.className = this.className.slice(); // copy
+ obj.allDay = this.isAllDay();
+ if (this.rawId != null) {
+ obj.id = this.rawId;
+ }
+ EventDef.copyVerbatimStandardProps(this, obj);
+ return obj;
+ };
+ EventDef.prototype.applyManualStandardProps = function (rawProps) {
+ if (rawProps.id != null) {
+ this.id = EventDef.normalizeId((this.rawId = rawProps.id));
+ }
+ else {
+ this.id = EventDef.generateId();
+ }
+ if (rawProps._id != null) {
+ this.uid = String(rawProps._id);
+ }
+ else {
+ this.uid = EventDef.generateId();
+ }
+ // TODO: converge with EventSource
+ if ($.isArray(rawProps.className)) {
+ this.className = rawProps.className;
+ }
+ if (typeof rawProps.className === 'string') {
+ this.className = rawProps.className.split(/\s+/);
+ }
+ return true;
+ };
+ EventDef.prototype.applyMiscProps = function (rawProps) {
+ $.extend(this.miscProps, rawProps);
+ };
+ EventDef.parse = function (rawInput, source) {
+ var def = new this(source);
+ if (def.applyProps(rawInput)) {
+ return def;
+ }
+ return false;
+ };
+ EventDef.normalizeId = function (id) {
+ return String(id);
+ };
+ EventDef.generateId = function () {
+ return '_fc' + (EventDef.uuid++);
+ };
+ EventDef.defineStandardProps = ParsableModelMixin_1.default.defineStandardProps;
+ EventDef.copyVerbatimStandardProps = ParsableModelMixin_1.default.copyVerbatimStandardProps;
+ // IDs
+ // ---------------------------------------------------------------------------------------------------------------------
+ // TODO: converge with EventSource
+ EventDef.uuid = 0;
+ return EventDef;
+}());
+exports.default = EventDef;
+ParsableModelMixin_1.default.mixInto(EventDef);
+// Parsing
+// ---------------------------------------------------------------------------------------------------------------------
+EventDef.defineStandardProps({
+ // not automatically assigned (`false`)
+ _id: false,
+ id: false,
+ className: false,
+ source: false,
+ // automatically assigned (`true`)
+ title: true,
+ url: true,
+ rendering: true,
+ constraint: true,
+ overlap: true,
+ editable: true,
+ startEditable: true,
+ durationEditable: true,
+ color: true,
+ backgroundColor: true,
+ borderColor: true,
+ textColor: true
+});
+
+
+/***/ }),
+/* 23 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var EventRange_1 = __webpack_require__(53);
+var EventFootprint_1 = __webpack_require__(24);
+var ComponentFootprint_1 = __webpack_require__(10);
+function eventDefsToEventInstances(eventDefs, unzonedRange) {
+ var eventInstances = [];
+ var i;
+ for (i = 0; i < eventDefs.length; i++) {
+ eventInstances.push.apply(eventInstances, // append
+ eventDefs[i].buildInstances(unzonedRange));
+ }
+ return eventInstances;
+}
+exports.eventDefsToEventInstances = eventDefsToEventInstances;
+function eventInstanceToEventRange(eventInstance) {
+ return new EventRange_1.default(eventInstance.dateProfile.unzonedRange, eventInstance.def, eventInstance);
+}
+exports.eventInstanceToEventRange = eventInstanceToEventRange;
+function eventRangeToEventFootprint(eventRange) {
+ return new EventFootprint_1.default(new ComponentFootprint_1.default(eventRange.unzonedRange, eventRange.eventDef.isAllDay()), eventRange.eventDef, eventRange.eventInstance // might not exist
+ );
+}
+exports.eventRangeToEventFootprint = eventRangeToEventFootprint;
+function eventInstanceToUnzonedRange(eventInstance) {
+ return eventInstance.dateProfile.unzonedRange;
+}
+exports.eventInstanceToUnzonedRange = eventInstanceToUnzonedRange;
+function eventFootprintToComponentFootprint(eventFootprint) {
+ return eventFootprint.componentFootprint;
+}
+exports.eventFootprintToComponentFootprint = eventFootprintToComponentFootprint;
+
+
+/***/ }),
+/* 24 */
+/***/ (function(module, exports) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var EventFootprint = /** @class */ (function () {
+ function EventFootprint(componentFootprint, eventDef, eventInstance) {
+ this.componentFootprint = componentFootprint;
+ this.eventDef = eventDef;
+ if (eventInstance) {
+ this.eventInstance = eventInstance;
+ }
+ }
+ EventFootprint.prototype.getEventLegacy = function () {
+ return (this.eventInstance || this.eventDef).toLegacy();
+ };
+ return EventFootprint;
+}());
+exports.default = EventFootprint;
+
+
+/***/ }),
+/* 25 */
+/***/ (function(module, exports) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+exports.default = {
+ sourceClasses: [],
+ registerClass: function (EventSourceClass) {
+ this.sourceClasses.unshift(EventSourceClass); // give highest priority
+ },
+ parse: function (rawInput, calendar) {
+ var sourceClasses = this.sourceClasses;
+ var i;
+ var eventSource;
+ for (i = 0; i < sourceClasses.length; i++) {
+ eventSource = sourceClasses[i].parse(rawInput, calendar);
+ if (eventSource) {
+ return eventSource;
+ }
+ }
+ }
+};
+
+
+/***/ }),
+/* 26 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var util_1 = __webpack_require__(2);
+var EventDateProfile_1 = __webpack_require__(15);
+var EventDef_1 = __webpack_require__(22);
+var EventDefDateMutation_1 = __webpack_require__(35);
+var SingleEventDef_1 = __webpack_require__(11);
+var EventDefMutation = /** @class */ (function () {
+ function EventDefMutation() {
+ }
+ /*
+ eventDef assumed to be a SingleEventDef.
+ returns an undo function.
+ */
+ EventDefMutation.prototype.mutateSingle = function (eventDef) {
+ var origDateProfile;
+ if (this.dateMutation) {
+ origDateProfile = eventDef.dateProfile;
+ eventDef.dateProfile = this.dateMutation.buildNewDateProfile(origDateProfile, eventDef.source.calendar);
+ }
+ // can't undo
+ // TODO: more DRY with EventDef::applyManualStandardProps
+ if (this.eventDefId != null) {
+ eventDef.id = EventDef_1.default.normalizeId((eventDef.rawId = this.eventDefId));
+ }
+ // can't undo
+ // TODO: more DRY with EventDef::applyManualStandardProps
+ if (this.className) {
+ eventDef.className = this.className;
+ }
+ // can't undo
+ if (this.verbatimStandardProps) {
+ SingleEventDef_1.default.copyVerbatimStandardProps(this.verbatimStandardProps, // src
+ eventDef // dest
+ );
+ }
+ // can't undo
+ if (this.miscProps) {
+ eventDef.applyMiscProps(this.miscProps);
+ }
+ if (origDateProfile) {
+ return function () {
+ eventDef.dateProfile = origDateProfile;
+ };
+ }
+ else {
+ return function () { };
+ }
+ };
+ EventDefMutation.prototype.setDateMutation = function (dateMutation) {
+ if (dateMutation && !dateMutation.isEmpty()) {
+ this.dateMutation = dateMutation;
+ }
+ else {
+ this.dateMutation = null;
+ }
+ };
+ EventDefMutation.prototype.isEmpty = function () {
+ return !this.dateMutation;
+ };
+ EventDefMutation.createFromRawProps = function (eventInstance, rawProps, largeUnit) {
+ var eventDef = eventInstance.def;
+ var dateProps = {};
+ var standardProps = {};
+ var miscProps = {};
+ var verbatimStandardProps = {};
+ var eventDefId = null;
+ var className = null;
+ var propName;
+ var dateProfile;
+ var dateMutation;
+ var defMutation;
+ for (propName in rawProps) {
+ if (EventDateProfile_1.default.isStandardProp(propName)) {
+ dateProps[propName] = rawProps[propName];
+ }
+ else if (eventDef.isStandardProp(propName)) {
+ standardProps[propName] = rawProps[propName];
+ }
+ else if (eventDef.miscProps[propName] !== rawProps[propName]) {
+ miscProps[propName] = rawProps[propName];
+ }
+ }
+ dateProfile = EventDateProfile_1.default.parse(dateProps, eventDef.source);
+ if (dateProfile) {
+ dateMutation = EventDefDateMutation_1.default.createFromDiff(eventInstance.dateProfile, dateProfile, largeUnit);
+ }
+ if (standardProps.id !== eventDef.id) {
+ eventDefId = standardProps.id; // only apply if there's a change
+ }
+ if (!util_1.isArraysEqual(standardProps.className, eventDef.className)) {
+ className = standardProps.className; // only apply if there's a change
+ }
+ EventDef_1.default.copyVerbatimStandardProps(standardProps, // src
+ verbatimStandardProps // dest
+ );
+ defMutation = new EventDefMutation();
+ defMutation.eventDefId = eventDefId;
+ defMutation.className = className;
+ defMutation.verbatimStandardProps = verbatimStandardProps;
+ defMutation.miscProps = miscProps;
+ if (dateMutation) {
+ defMutation.dateMutation = dateMutation;
+ }
+ return defMutation;
+ };
+ return EventDefMutation;
+}());
+exports.default = EventDefMutation;
+
+
+/***/ }),
+/* 27 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var $ = __webpack_require__(1);
+var Theme = /** @class */ (function () {
+ function Theme(optionsManager) {
+ this.optionsManager = optionsManager;
+ this.processIconOverride();
+ }
+ Theme.prototype.processIconOverride = function () {
+ if (this.iconOverrideOption) {
+ this.setIconOverride(this.optionsManager.get(this.iconOverrideOption));
+ }
+ };
+ Theme.prototype.setIconOverride = function (iconOverrideHash) {
+ var iconClassesCopy;
+ var buttonName;
+ if ($.isPlainObject(iconOverrideHash)) {
+ iconClassesCopy = $.extend({}, this.iconClasses);
+ for (buttonName in iconOverrideHash) {
+ iconClassesCopy[buttonName] = this.applyIconOverridePrefix(iconOverrideHash[buttonName]);
+ }
+ this.iconClasses = iconClassesCopy;
+ }
+ else if (iconOverrideHash === false) {
+ this.iconClasses = {};
+ }
+ };
+ Theme.prototype.applyIconOverridePrefix = function (className) {
+ var prefix = this.iconOverridePrefix;
+ if (prefix && className.indexOf(prefix) !== 0) {
+ className = prefix + className;
+ }
+ return className;
+ };
+ Theme.prototype.getClass = function (key) {
+ return this.classes[key] || '';
+ };
+ Theme.prototype.getIconClass = function (buttonName) {
+ var className = this.iconClasses[buttonName];
+ if (className) {
+ return this.baseIconClass + ' ' + className;
+ }
+ return '';
+ };
+ Theme.prototype.getCustomButtonIconClass = function (customButtonProps) {
+ var className;
+ if (this.iconOverrideCustomButtonOption) {
+ className = customButtonProps[this.iconOverrideCustomButtonOption];
+ if (className) {
+ return this.baseIconClass + ' ' + this.applyIconOverridePrefix(className);
+ }
+ }
+ return '';
+ };
+ return Theme;
+}());
+exports.default = Theme;
+Theme.prototype.classes = {};
+Theme.prototype.iconClasses = {};
+Theme.prototype.baseIconClass = '';
+Theme.prototype.iconOverridePrefix = '';
+
+
+/***/ }),
+/* 28 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var $ = __webpack_require__(1);
+var util_1 = __webpack_require__(2);
+var Class_1 = __webpack_require__(21);
+/*
+Embodies a div that has potential scrollbars
+*/
+var Scroller = /** @class */ (function (_super) {
+ tslib_1.__extends(Scroller, _super);
+ function Scroller(options) {
+ var _this = _super.call(this) || this;
+ options = options || {};
+ _this.overflowX = options.overflowX || options.overflow || 'auto';
+ _this.overflowY = options.overflowY || options.overflow || 'auto';
+ return _this;
+ }
+ Scroller.prototype.render = function () {
+ this.el = this.renderEl();
+ this.applyOverflow();
+ };
+ Scroller.prototype.renderEl = function () {
+ return (this.scrollEl = $(''));
+ };
+ // sets to natural height, unlocks overflow
+ Scroller.prototype.clear = function () {
+ this.setHeight('auto');
+ this.applyOverflow();
+ };
+ Scroller.prototype.destroy = function () {
+ this.el.remove();
+ };
+ // Overflow
+ // -----------------------------------------------------------------------------------------------------------------
+ Scroller.prototype.applyOverflow = function () {
+ this.scrollEl.css({
+ 'overflow-x': this.overflowX,
+ 'overflow-y': this.overflowY
+ });
+ };
+ // Causes any 'auto' overflow values to resolves to 'scroll' or 'hidden'.
+ // Useful for preserving scrollbar widths regardless of future resizes.
+ // Can pass in scrollbarWidths for optimization.
+ Scroller.prototype.lockOverflow = function (scrollbarWidths) {
+ var overflowX = this.overflowX;
+ var overflowY = this.overflowY;
+ scrollbarWidths = scrollbarWidths || this.getScrollbarWidths();
+ if (overflowX === 'auto') {
+ overflowX = (scrollbarWidths.top || scrollbarWidths.bottom || // horizontal scrollbars?
+ // OR scrolling pane with massless scrollbars?
+ this.scrollEl[0].scrollWidth - 1 > this.scrollEl[0].clientWidth) ? 'scroll' : 'hidden';
+ }
+ if (overflowY === 'auto') {
+ overflowY = (scrollbarWidths.left || scrollbarWidths.right || // vertical scrollbars?
+ // OR scrolling pane with massless scrollbars?
+ this.scrollEl[0].scrollHeight - 1 > this.scrollEl[0].clientHeight) ? 'scroll' : 'hidden';
+ }
+ this.scrollEl.css({ 'overflow-x': overflowX, 'overflow-y': overflowY });
+ };
+ // Getters / Setters
+ // -----------------------------------------------------------------------------------------------------------------
+ Scroller.prototype.setHeight = function (height) {
+ this.scrollEl.height(height);
+ };
+ Scroller.prototype.getScrollTop = function () {
+ return this.scrollEl.scrollTop();
+ };
+ Scroller.prototype.setScrollTop = function (top) {
+ this.scrollEl.scrollTop(top);
+ };
+ Scroller.prototype.getClientWidth = function () {
+ return this.scrollEl[0].clientWidth;
+ };
+ Scroller.prototype.getClientHeight = function () {
+ return this.scrollEl[0].clientHeight;
+ };
+ Scroller.prototype.getScrollbarWidths = function () {
+ return util_1.getScrollbarWidths(this.scrollEl);
+ };
+ return Scroller;
+}(Class_1.default));
+exports.default = Scroller;
+
+
+/***/ }),
+/* 29 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var $ = __webpack_require__(1);
+var util_1 = __webpack_require__(2);
+var DateComponent_1 = __webpack_require__(61);
+var GlobalEmitter_1 = __webpack_require__(14);
+var InteractiveDateComponent = /** @class */ (function (_super) {
+ tslib_1.__extends(InteractiveDateComponent, _super);
+ function InteractiveDateComponent(_view, _options) {
+ var _this = _super.call(this, _view, _options) || this;
+ // self-config, overridable by subclasses
+ _this.segSelector = '.fc-event-container > *'; // what constitutes an event element?
+ if (_this.dateSelectingClass) {
+ _this.dateClicking = new _this.dateClickingClass(_this);
+ }
+ if (_this.dateSelectingClass) {
+ _this.dateSelecting = new _this.dateSelectingClass(_this);
+ }
+ if (_this.eventPointingClass) {
+ _this.eventPointing = new _this.eventPointingClass(_this);
+ }
+ if (_this.eventDraggingClass && _this.eventPointing) {
+ _this.eventDragging = new _this.eventDraggingClass(_this, _this.eventPointing);
+ }
+ if (_this.eventResizingClass && _this.eventPointing) {
+ _this.eventResizing = new _this.eventResizingClass(_this, _this.eventPointing);
+ }
+ if (_this.externalDroppingClass) {
+ _this.externalDropping = new _this.externalDroppingClass(_this);
+ }
+ return _this;
+ }
+ // Sets the container element that the view should render inside of, does global DOM-related initializations,
+ // and renders all the non-date-related content inside.
+ InteractiveDateComponent.prototype.setElement = function (el) {
+ _super.prototype.setElement.call(this, el);
+ if (this.dateClicking) {
+ this.dateClicking.bindToEl(el);
+ }
+ if (this.dateSelecting) {
+ this.dateSelecting.bindToEl(el);
+ }
+ this.bindAllSegHandlersToEl(el);
+ };
+ InteractiveDateComponent.prototype.removeElement = function () {
+ this.endInteractions();
+ _super.prototype.removeElement.call(this);
+ };
+ InteractiveDateComponent.prototype.executeEventUnrender = function () {
+ this.endInteractions();
+ _super.prototype.executeEventUnrender.call(this);
+ };
+ InteractiveDateComponent.prototype.bindGlobalHandlers = function () {
+ _super.prototype.bindGlobalHandlers.call(this);
+ if (this.externalDropping) {
+ this.externalDropping.bindToDocument();
+ }
+ };
+ InteractiveDateComponent.prototype.unbindGlobalHandlers = function () {
+ _super.prototype.unbindGlobalHandlers.call(this);
+ if (this.externalDropping) {
+ this.externalDropping.unbindFromDocument();
+ }
+ };
+ InteractiveDateComponent.prototype.bindDateHandlerToEl = function (el, name, handler) {
+ var _this = this;
+ // attach a handler to the grid's root element.
+ // jQuery will take care of unregistering them when removeElement gets called.
+ this.el.on(name, function (ev) {
+ if (!$(ev.target).is(_this.segSelector + ',' + // directly on an event element
+ _this.segSelector + ' *,' + // within an event element
+ '.fc-more,' + // a "more.." link
+ 'a[data-goto]' // a clickable nav link
+ )) {
+ return handler.call(_this, ev);
+ }
+ });
+ };
+ InteractiveDateComponent.prototype.bindAllSegHandlersToEl = function (el) {
+ [
+ this.eventPointing,
+ this.eventDragging,
+ this.eventResizing
+ ].forEach(function (eventInteraction) {
+ if (eventInteraction) {
+ eventInteraction.bindToEl(el);
+ }
+ });
+ };
+ InteractiveDateComponent.prototype.bindSegHandlerToEl = function (el, name, handler) {
+ var _this = this;
+ el.on(name, this.segSelector, function (ev) {
+ var seg = $(ev.currentTarget).data('fc-seg'); // grab segment data. put there by View::renderEventsPayload
+ if (seg && !_this.shouldIgnoreEventPointing()) {
+ return handler.call(_this, seg, ev); // context will be the Grid
+ }
+ });
+ };
+ InteractiveDateComponent.prototype.shouldIgnoreMouse = function () {
+ // HACK
+ // This will still work even though bindDateHandlerToEl doesn't use GlobalEmitter.
+ return GlobalEmitter_1.default.get().shouldIgnoreMouse();
+ };
+ InteractiveDateComponent.prototype.shouldIgnoreTouch = function () {
+ var view = this._getView();
+ // On iOS (and Android?) when a new selection is initiated overtop another selection,
+ // the touchend never fires because the elements gets removed mid-touch-interaction (my theory).
+ // HACK: simply don't allow this to happen.
+ // ALSO: prevent selection when an *event* is already raised.
+ return view.isSelected || view.selectedEvent;
+ };
+ InteractiveDateComponent.prototype.shouldIgnoreEventPointing = function () {
+ // only call the handlers if there is not a drag/resize in progress
+ return (this.eventDragging && this.eventDragging.isDragging) ||
+ (this.eventResizing && this.eventResizing.isResizing);
+ };
+ InteractiveDateComponent.prototype.canStartSelection = function (seg, ev) {
+ return util_1.getEvIsTouch(ev) &&
+ !this.canStartResize(seg, ev) &&
+ (this.isEventDefDraggable(seg.footprint.eventDef) ||
+ this.isEventDefResizable(seg.footprint.eventDef));
+ };
+ InteractiveDateComponent.prototype.canStartDrag = function (seg, ev) {
+ return !this.canStartResize(seg, ev) &&
+ this.isEventDefDraggable(seg.footprint.eventDef);
+ };
+ InteractiveDateComponent.prototype.canStartResize = function (seg, ev) {
+ var view = this._getView();
+ var eventDef = seg.footprint.eventDef;
+ return (!util_1.getEvIsTouch(ev) || view.isEventDefSelected(eventDef)) &&
+ this.isEventDefResizable(eventDef) &&
+ $(ev.target).is('.fc-resizer');
+ };
+ // Kills all in-progress dragging.
+ // Useful for when public API methods that result in re-rendering are invoked during a drag.
+ // Also useful for when touch devices misbehave and don't fire their touchend.
+ InteractiveDateComponent.prototype.endInteractions = function () {
+ [
+ this.dateClicking,
+ this.dateSelecting,
+ this.eventPointing,
+ this.eventDragging,
+ this.eventResizing
+ ].forEach(function (interaction) {
+ if (interaction) {
+ interaction.end();
+ }
+ });
+ };
+ // Event Drag-n-Drop
+ // ---------------------------------------------------------------------------------------------------------------
+ // Computes if the given event is allowed to be dragged by the user
+ InteractiveDateComponent.prototype.isEventDefDraggable = function (eventDef) {
+ return this.isEventDefStartEditable(eventDef);
+ };
+ InteractiveDateComponent.prototype.isEventDefStartEditable = function (eventDef) {
+ var isEditable = eventDef.isStartExplicitlyEditable();
+ if (isEditable == null) {
+ isEditable = this.opt('eventStartEditable');
+ if (isEditable == null) {
+ isEditable = this.isEventDefGenerallyEditable(eventDef);
+ }
+ }
+ return isEditable;
+ };
+ InteractiveDateComponent.prototype.isEventDefGenerallyEditable = function (eventDef) {
+ var isEditable = eventDef.isExplicitlyEditable();
+ if (isEditable == null) {
+ isEditable = this.opt('editable');
+ }
+ return isEditable;
+ };
+ // Event Resizing
+ // ---------------------------------------------------------------------------------------------------------------
+ // Computes if the given event is allowed to be resized from its starting edge
+ InteractiveDateComponent.prototype.isEventDefResizableFromStart = function (eventDef) {
+ return this.opt('eventResizableFromStart') && this.isEventDefResizable(eventDef);
+ };
+ // Computes if the given event is allowed to be resized from its ending edge
+ InteractiveDateComponent.prototype.isEventDefResizableFromEnd = function (eventDef) {
+ return this.isEventDefResizable(eventDef);
+ };
+ // Computes if the given event is allowed to be resized by the user at all
+ InteractiveDateComponent.prototype.isEventDefResizable = function (eventDef) {
+ var isResizable = eventDef.isDurationExplicitlyEditable();
+ if (isResizable == null) {
+ isResizable = this.opt('eventDurationEditable');
+ if (isResizable == null) {
+ isResizable = this.isEventDefGenerallyEditable(eventDef);
+ }
+ }
+ return isResizable;
+ };
+ // Event Mutation / Constraints
+ // ---------------------------------------------------------------------------------------------------------------
+ // Diffs the two dates, returning a duration, based on granularity of the grid
+ // TODO: port isTimeScale into this system?
+ InteractiveDateComponent.prototype.diffDates = function (a, b) {
+ if (this.largeUnit) {
+ return util_1.diffByUnit(a, b, this.largeUnit);
+ }
+ else {
+ return util_1.diffDayTime(a, b);
+ }
+ };
+ // is it allowed, in relation to the view's validRange?
+ // NOTE: very similar to isExternalInstanceGroupAllowed
+ InteractiveDateComponent.prototype.isEventInstanceGroupAllowed = function (eventInstanceGroup) {
+ var view = this._getView();
+ var dateProfile = this.dateProfile;
+ var eventFootprints = this.eventRangesToEventFootprints(eventInstanceGroup.getAllEventRanges());
+ var i;
+ for (i = 0; i < eventFootprints.length; i++) {
+ // TODO: just use getAllEventRanges directly
+ if (!dateProfile.validUnzonedRange.containsRange(eventFootprints[i].componentFootprint.unzonedRange)) {
+ return false;
+ }
+ }
+ return view.calendar.constraints.isEventInstanceGroupAllowed(eventInstanceGroup);
+ };
+ // NOTE: very similar to isEventInstanceGroupAllowed
+ // when it's a completely anonymous external drag, no event.
+ InteractiveDateComponent.prototype.isExternalInstanceGroupAllowed = function (eventInstanceGroup) {
+ var view = this._getView();
+ var dateProfile = this.dateProfile;
+ var eventFootprints = this.eventRangesToEventFootprints(eventInstanceGroup.getAllEventRanges());
+ var i;
+ for (i = 0; i < eventFootprints.length; i++) {
+ if (!dateProfile.validUnzonedRange.containsRange(eventFootprints[i].componentFootprint.unzonedRange)) {
+ return false;
+ }
+ }
+ for (i = 0; i < eventFootprints.length; i++) {
+ // treat it as a selection
+ // TODO: pass in eventInstanceGroup instead
+ // because we don't want calendar's constraint system to depend on a component's
+ // determination of footprints.
+ if (!view.calendar.constraints.isSelectionFootprintAllowed(eventFootprints[i].componentFootprint)) {
+ return false;
+ }
+ }
+ return true;
+ };
+ return InteractiveDateComponent;
+}(DateComponent_1.default));
+exports.default = InteractiveDateComponent;
+
+
+/***/ }),
+/* 30 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var $ = __webpack_require__(1);
+var moment = __webpack_require__(3);
+var util_1 = __webpack_require__(2);
+var RenderQueue_1 = __webpack_require__(60);
+var DateProfileGenerator_1 = __webpack_require__(62);
+var InteractiveDateComponent_1 = __webpack_require__(29);
+var GlobalEmitter_1 = __webpack_require__(14);
+var UnzonedRange_1 = __webpack_require__(4);
+/* An abstract class from which other views inherit from
+----------------------------------------------------------------------------------------------------------------------*/
+var View = /** @class */ (function (_super) {
+ tslib_1.__extends(View, _super);
+ function View(calendar, viewSpec) {
+ var _this = _super.call(this, null, viewSpec.options) || this;
+ _this.batchRenderDepth = 0;
+ _this.isSelected = false; // boolean whether a range of time is user-selected or not
+ _this.calendar = calendar;
+ _this.viewSpec = viewSpec;
+ // shortcuts
+ _this.type = viewSpec.type;
+ // .name is deprecated
+ _this.name = _this.type;
+ _this.initRenderQueue();
+ _this.initHiddenDays();
+ _this.dateProfileGenerator = new _this.dateProfileGeneratorClass(_this);
+ _this.bindBaseRenderHandlers();
+ _this.eventOrderSpecs = util_1.parseFieldSpecs(_this.opt('eventOrder'));
+ // legacy
+ if (_this['initialize']) {
+ _this['initialize']();
+ }
+ return _this;
+ }
+ View.prototype._getView = function () {
+ return this;
+ };
+ // Retrieves an option with the given name
+ View.prototype.opt = function (name) {
+ return this.options[name];
+ };
+ /* Render Queue
+ ------------------------------------------------------------------------------------------------------------------*/
+ View.prototype.initRenderQueue = function () {
+ this.renderQueue = new RenderQueue_1.default({
+ event: this.opt('eventRenderWait')
+ });
+ this.renderQueue.on('start', this.onRenderQueueStart.bind(this));
+ this.renderQueue.on('stop', this.onRenderQueueStop.bind(this));
+ this.on('before:change', this.startBatchRender);
+ this.on('change', this.stopBatchRender);
+ };
+ View.prototype.onRenderQueueStart = function () {
+ this.calendar.freezeContentHeight();
+ this.addScroll(this.queryScroll());
+ };
+ View.prototype.onRenderQueueStop = function () {
+ if (this.calendar.updateViewSize()) {
+ this.popScroll();
+ }
+ this.calendar.thawContentHeight();
+ };
+ View.prototype.startBatchRender = function () {
+ if (!(this.batchRenderDepth++)) {
+ this.renderQueue.pause();
+ }
+ };
+ View.prototype.stopBatchRender = function () {
+ if (!(--this.batchRenderDepth)) {
+ this.renderQueue.resume();
+ }
+ };
+ View.prototype.requestRender = function (func, namespace, actionType) {
+ this.renderQueue.queue(func, namespace, actionType);
+ };
+ // given func will auto-bind to `this`
+ View.prototype.whenSizeUpdated = function (func) {
+ if (this.renderQueue.isRunning) {
+ this.renderQueue.one('stop', func.bind(this));
+ }
+ else {
+ func.call(this);
+ }
+ };
+ /* Title and Date Formatting
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Computes what the title at the top of the calendar should be for this view
+ View.prototype.computeTitle = function (dateProfile) {
+ var unzonedRange;
+ // for views that span a large unit of time, show the proper interval, ignoring stray days before and after
+ if (/^(year|month)$/.test(dateProfile.currentRangeUnit)) {
+ unzonedRange = dateProfile.currentUnzonedRange;
+ }
+ else {
+ unzonedRange = dateProfile.activeUnzonedRange;
+ }
+ return this.formatRange({
+ start: this.calendar.msToMoment(unzonedRange.startMs, dateProfile.isRangeAllDay),
+ end: this.calendar.msToMoment(unzonedRange.endMs, dateProfile.isRangeAllDay)
+ }, dateProfile.isRangeAllDay, this.opt('titleFormat') || this.computeTitleFormat(dateProfile), this.opt('titleRangeSeparator'));
+ };
+ // Generates the format string that should be used to generate the title for the current date range.
+ // Attempts to compute the most appropriate format if not explicitly specified with `titleFormat`.
+ View.prototype.computeTitleFormat = function (dateProfile) {
+ var currentRangeUnit = dateProfile.currentRangeUnit;
+ if (currentRangeUnit == 'year') {
+ return 'YYYY';
+ }
+ else if (currentRangeUnit == 'month') {
+ return this.opt('monthYearFormat'); // like "September 2014"
+ }
+ else if (dateProfile.currentUnzonedRange.as('days') > 1) {
+ return 'll'; // multi-day range. shorter, like "Sep 9 - 10 2014"
+ }
+ else {
+ return 'LL'; // one day. longer, like "September 9 2014"
+ }
+ };
+ // Date Setting/Unsetting
+ // -----------------------------------------------------------------------------------------------------------------
+ View.prototype.setDate = function (date) {
+ var currentDateProfile = this.get('dateProfile');
+ var newDateProfile = this.dateProfileGenerator.build(date, undefined, true); // forceToValid=true
+ if (!currentDateProfile ||
+ !currentDateProfile.activeUnzonedRange.equals(newDateProfile.activeUnzonedRange)) {
+ this.set('dateProfile', newDateProfile);
+ }
+ };
+ View.prototype.unsetDate = function () {
+ this.unset('dateProfile');
+ };
+ // Event Data
+ // -----------------------------------------------------------------------------------------------------------------
+ View.prototype.fetchInitialEvents = function (dateProfile) {
+ var calendar = this.calendar;
+ var forceAllDay = dateProfile.isRangeAllDay && !this.usesMinMaxTime;
+ return calendar.requestEvents(calendar.msToMoment(dateProfile.activeUnzonedRange.startMs, forceAllDay), calendar.msToMoment(dateProfile.activeUnzonedRange.endMs, forceAllDay));
+ };
+ View.prototype.bindEventChanges = function () {
+ this.listenTo(this.calendar, 'eventsReset', this.resetEvents); // TODO: make this a real event
+ };
+ View.prototype.unbindEventChanges = function () {
+ this.stopListeningTo(this.calendar, 'eventsReset');
+ };
+ View.prototype.setEvents = function (eventsPayload) {
+ this.set('currentEvents', eventsPayload);
+ this.set('hasEvents', true);
+ };
+ View.prototype.unsetEvents = function () {
+ this.unset('currentEvents');
+ this.unset('hasEvents');
+ };
+ View.prototype.resetEvents = function (eventsPayload) {
+ this.startBatchRender();
+ this.unsetEvents();
+ this.setEvents(eventsPayload);
+ this.stopBatchRender();
+ };
+ // Date High-level Rendering
+ // -----------------------------------------------------------------------------------------------------------------
+ View.prototype.requestDateRender = function (dateProfile) {
+ var _this = this;
+ this.requestRender(function () {
+ _this.executeDateRender(dateProfile);
+ }, 'date', 'init');
+ };
+ View.prototype.requestDateUnrender = function () {
+ var _this = this;
+ this.requestRender(function () {
+ _this.executeDateUnrender();
+ }, 'date', 'destroy');
+ };
+ // if dateProfile not specified, uses current
+ View.prototype.executeDateRender = function (dateProfile) {
+ _super.prototype.executeDateRender.call(this, dateProfile);
+ if (this['render']) {
+ this['render'](); // TODO: deprecate
+ }
+ this.trigger('datesRendered');
+ this.addScroll({ isDateInit: true });
+ this.startNowIndicator(); // shouldn't render yet because updateSize will be called soon
+ };
+ View.prototype.executeDateUnrender = function () {
+ this.unselect();
+ this.stopNowIndicator();
+ this.trigger('before:datesUnrendered');
+ if (this['destroy']) {
+ this['destroy'](); // TODO: deprecate
+ }
+ _super.prototype.executeDateUnrender.call(this);
+ };
+ // "Base" rendering
+ // -----------------------------------------------------------------------------------------------------------------
+ View.prototype.bindBaseRenderHandlers = function () {
+ var _this = this;
+ this.on('datesRendered', function () {
+ _this.whenSizeUpdated(_this.triggerViewRender);
+ });
+ this.on('before:datesUnrendered', function () {
+ _this.triggerViewDestroy();
+ });
+ };
+ View.prototype.triggerViewRender = function () {
+ this.publiclyTrigger('viewRender', {
+ context: this,
+ args: [this, this.el]
+ });
+ };
+ View.prototype.triggerViewDestroy = function () {
+ this.publiclyTrigger('viewDestroy', {
+ context: this,
+ args: [this, this.el]
+ });
+ };
+ // Event High-level Rendering
+ // -----------------------------------------------------------------------------------------------------------------
+ View.prototype.requestEventsRender = function (eventsPayload) {
+ var _this = this;
+ this.requestRender(function () {
+ _this.executeEventRender(eventsPayload);
+ _this.whenSizeUpdated(_this.triggerAfterEventsRendered);
+ }, 'event', 'init');
+ };
+ View.prototype.requestEventsUnrender = function () {
+ var _this = this;
+ this.requestRender(function () {
+ _this.triggerBeforeEventsDestroyed();
+ _this.executeEventUnrender();
+ }, 'event', 'destroy');
+ };
+ // Business Hour High-level Rendering
+ // -----------------------------------------------------------------------------------------------------------------
+ View.prototype.requestBusinessHoursRender = function (businessHourGenerator) {
+ var _this = this;
+ this.requestRender(function () {
+ _this.renderBusinessHours(businessHourGenerator);
+ }, 'businessHours', 'init');
+ };
+ View.prototype.requestBusinessHoursUnrender = function () {
+ var _this = this;
+ this.requestRender(function () {
+ _this.unrenderBusinessHours();
+ }, 'businessHours', 'destroy');
+ };
+ // Misc view rendering utils
+ // -----------------------------------------------------------------------------------------------------------------
+ // Binds DOM handlers to elements that reside outside the view container, such as the document
+ View.prototype.bindGlobalHandlers = function () {
+ _super.prototype.bindGlobalHandlers.call(this);
+ this.listenTo(GlobalEmitter_1.default.get(), {
+ touchstart: this.processUnselect,
+ mousedown: this.handleDocumentMousedown
+ });
+ };
+ // Unbinds DOM handlers from elements that reside outside the view container
+ View.prototype.unbindGlobalHandlers = function () {
+ _super.prototype.unbindGlobalHandlers.call(this);
+ this.stopListeningTo(GlobalEmitter_1.default.get());
+ };
+ /* Now Indicator
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Immediately render the current time indicator and begins re-rendering it at an interval,
+ // which is defined by this.getNowIndicatorUnit().
+ // TODO: somehow do this for the current whole day's background too
+ View.prototype.startNowIndicator = function () {
+ var _this = this;
+ var unit;
+ var update;
+ var delay; // ms wait value
+ if (this.opt('nowIndicator')) {
+ unit = this.getNowIndicatorUnit();
+ if (unit) {
+ update = util_1.proxy(this, 'updateNowIndicator'); // bind to `this`
+ this.initialNowDate = this.calendar.getNow();
+ this.initialNowQueriedMs = +new Date();
+ // wait until the beginning of the next interval
+ delay = this.initialNowDate.clone().startOf(unit).add(1, unit) - this.initialNowDate;
+ this.nowIndicatorTimeoutID = setTimeout(function () {
+ _this.nowIndicatorTimeoutID = null;
+ update();
+ delay = +moment.duration(1, unit);
+ delay = Math.max(100, delay); // prevent too frequent
+ _this.nowIndicatorIntervalID = setInterval(update, delay); // update every interval
+ }, delay);
+ }
+ // rendering will be initiated in updateSize
+ }
+ };
+ // rerenders the now indicator, computing the new current time from the amount of time that has passed
+ // since the initial getNow call.
+ View.prototype.updateNowIndicator = function () {
+ if (this.isDatesRendered &&
+ this.initialNowDate // activated before?
+ ) {
+ this.unrenderNowIndicator(); // won't unrender if unnecessary
+ this.renderNowIndicator(this.initialNowDate.clone().add(new Date().valueOf() - this.initialNowQueriedMs) // add ms
+ );
+ this.isNowIndicatorRendered = true;
+ }
+ };
+ // Immediately unrenders the view's current time indicator and stops any re-rendering timers.
+ // Won't cause side effects if indicator isn't rendered.
+ View.prototype.stopNowIndicator = function () {
+ if (this.isNowIndicatorRendered) {
+ if (this.nowIndicatorTimeoutID) {
+ clearTimeout(this.nowIndicatorTimeoutID);
+ this.nowIndicatorTimeoutID = null;
+ }
+ if (this.nowIndicatorIntervalID) {
+ clearInterval(this.nowIndicatorIntervalID);
+ this.nowIndicatorIntervalID = null;
+ }
+ this.unrenderNowIndicator();
+ this.isNowIndicatorRendered = false;
+ }
+ };
+ /* Dimensions
+ ------------------------------------------------------------------------------------------------------------------*/
+ View.prototype.updateSize = function (totalHeight, isAuto, isResize) {
+ if (this['setHeight']) {
+ this['setHeight'](totalHeight, isAuto);
+ }
+ else {
+ _super.prototype.updateSize.call(this, totalHeight, isAuto, isResize);
+ }
+ this.updateNowIndicator();
+ };
+ /* Scroller
+ ------------------------------------------------------------------------------------------------------------------*/
+ View.prototype.addScroll = function (scroll) {
+ var queuedScroll = this.queuedScroll || (this.queuedScroll = {});
+ $.extend(queuedScroll, scroll);
+ };
+ View.prototype.popScroll = function () {
+ this.applyQueuedScroll();
+ this.queuedScroll = null;
+ };
+ View.prototype.applyQueuedScroll = function () {
+ if (this.queuedScroll) {
+ this.applyScroll(this.queuedScroll);
+ }
+ };
+ View.prototype.queryScroll = function () {
+ var scroll = {};
+ if (this.isDatesRendered) {
+ $.extend(scroll, this.queryDateScroll());
+ }
+ return scroll;
+ };
+ View.prototype.applyScroll = function (scroll) {
+ if (scroll.isDateInit && this.isDatesRendered) {
+ $.extend(scroll, this.computeInitialDateScroll());
+ }
+ if (this.isDatesRendered) {
+ this.applyDateScroll(scroll);
+ }
+ };
+ View.prototype.computeInitialDateScroll = function () {
+ return {}; // subclasses must implement
+ };
+ View.prototype.queryDateScroll = function () {
+ return {}; // subclasses must implement
+ };
+ View.prototype.applyDateScroll = function (scroll) {
+ ; // subclasses must implement
+ };
+ /* Event Drag-n-Drop
+ ------------------------------------------------------------------------------------------------------------------*/
+ View.prototype.reportEventDrop = function (eventInstance, eventMutation, el, ev) {
+ var eventManager = this.calendar.eventManager;
+ var undoFunc = eventManager.mutateEventsWithId(eventInstance.def.id, eventMutation, this.calendar);
+ var dateMutation = eventMutation.dateMutation;
+ // update the EventInstance, for handlers
+ if (dateMutation) {
+ eventInstance.dateProfile = dateMutation.buildNewDateProfile(eventInstance.dateProfile, this.calendar);
+ }
+ this.triggerEventDrop(eventInstance,
+ // a drop doesn't necessarily mean a date mutation (ex: resource change)
+ (dateMutation && dateMutation.dateDelta) || moment.duration(), undoFunc, el, ev);
+ };
+ // Triggers event-drop handlers that have subscribed via the API
+ View.prototype.triggerEventDrop = function (eventInstance, dateDelta, undoFunc, el, ev) {
+ this.publiclyTrigger('eventDrop', {
+ context: el[0],
+ args: [
+ eventInstance.toLegacy(),
+ dateDelta,
+ undoFunc,
+ ev,
+ {},
+ this
+ ]
+ });
+ };
+ /* External Element Drag-n-Drop
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Must be called when an external element, via jQuery UI, has been dropped onto the calendar.
+ // `meta` is the parsed data that has been embedded into the dragging event.
+ // `dropLocation` is an object that contains the new zoned start/end/allDay values for the event.
+ View.prototype.reportExternalDrop = function (singleEventDef, isEvent, isSticky, el, ev, ui) {
+ if (isEvent) {
+ this.calendar.eventManager.addEventDef(singleEventDef, isSticky);
+ }
+ this.triggerExternalDrop(singleEventDef, isEvent, el, ev, ui);
+ };
+ // Triggers external-drop handlers that have subscribed via the API
+ View.prototype.triggerExternalDrop = function (singleEventDef, isEvent, el, ev, ui) {
+ // trigger 'drop' regardless of whether element represents an event
+ this.publiclyTrigger('drop', {
+ context: el[0],
+ args: [
+ singleEventDef.dateProfile.start.clone(),
+ ev,
+ ui,
+ this
+ ]
+ });
+ if (isEvent) {
+ // signal an external event landed
+ this.publiclyTrigger('eventReceive', {
+ context: this,
+ args: [
+ singleEventDef.buildInstance().toLegacy(),
+ this
+ ]
+ });
+ }
+ };
+ /* Event Resizing
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Must be called when an event in the view has been resized to a new length
+ View.prototype.reportEventResize = function (eventInstance, eventMutation, el, ev) {
+ var eventManager = this.calendar.eventManager;
+ var undoFunc = eventManager.mutateEventsWithId(eventInstance.def.id, eventMutation, this.calendar);
+ // update the EventInstance, for handlers
+ eventInstance.dateProfile = eventMutation.dateMutation.buildNewDateProfile(eventInstance.dateProfile, this.calendar);
+ this.triggerEventResize(eventInstance, eventMutation.dateMutation.endDelta, undoFunc, el, ev);
+ };
+ // Triggers event-resize handlers that have subscribed via the API
+ View.prototype.triggerEventResize = function (eventInstance, durationDelta, undoFunc, el, ev) {
+ this.publiclyTrigger('eventResize', {
+ context: el[0],
+ args: [
+ eventInstance.toLegacy(),
+ durationDelta,
+ undoFunc,
+ ev,
+ {},
+ this
+ ]
+ });
+ };
+ /* Selection (time range)
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Selects a date span on the view. `start` and `end` are both Moments.
+ // `ev` is the native mouse event that begin the interaction.
+ View.prototype.select = function (footprint, ev) {
+ this.unselect(ev);
+ this.renderSelectionFootprint(footprint);
+ this.reportSelection(footprint, ev);
+ };
+ View.prototype.renderSelectionFootprint = function (footprint) {
+ if (this['renderSelection']) {
+ this['renderSelection'](footprint.toLegacy(this.calendar));
+ }
+ else {
+ _super.prototype.renderSelectionFootprint.call(this, footprint);
+ }
+ };
+ // Called when a new selection is made. Updates internal state and triggers handlers.
+ View.prototype.reportSelection = function (footprint, ev) {
+ this.isSelected = true;
+ this.triggerSelect(footprint, ev);
+ };
+ // Triggers handlers to 'select'
+ View.prototype.triggerSelect = function (footprint, ev) {
+ var dateProfile = this.calendar.footprintToDateProfile(footprint); // abuse of "Event"DateProfile?
+ this.publiclyTrigger('select', {
+ context: this,
+ args: [
+ dateProfile.start,
+ dateProfile.end,
+ ev,
+ this
+ ]
+ });
+ };
+ // Undoes a selection. updates in the internal state and triggers handlers.
+ // `ev` is the native mouse event that began the interaction.
+ View.prototype.unselect = function (ev) {
+ if (ev === void 0) { ev = null; }
+ if (this.isSelected) {
+ this.isSelected = false;
+ if (this['destroySelection']) {
+ this['destroySelection'](); // TODO: deprecate
+ }
+ this.unrenderSelection();
+ this.publiclyTrigger('unselect', {
+ context: this,
+ args: [ev, this]
+ });
+ }
+ };
+ /* Event Selection
+ ------------------------------------------------------------------------------------------------------------------*/
+ View.prototype.selectEventInstance = function (eventInstance) {
+ if (!this.selectedEventInstance ||
+ this.selectedEventInstance !== eventInstance) {
+ this.unselectEventInstance();
+ this.getEventSegs().forEach(function (seg) {
+ if (seg.footprint.eventInstance === eventInstance &&
+ seg.el // necessary?
+ ) {
+ seg.el.addClass('fc-selected');
+ }
+ });
+ this.selectedEventInstance = eventInstance;
+ }
+ };
+ View.prototype.unselectEventInstance = function () {
+ if (this.selectedEventInstance) {
+ this.getEventSegs().forEach(function (seg) {
+ if (seg.el) {
+ seg.el.removeClass('fc-selected');
+ }
+ });
+ this.selectedEventInstance = null;
+ }
+ };
+ View.prototype.isEventDefSelected = function (eventDef) {
+ // event references might change on refetchEvents(), while selectedEventInstance doesn't,
+ // so compare IDs
+ return this.selectedEventInstance && this.selectedEventInstance.def.id === eventDef.id;
+ };
+ /* Mouse / Touch Unselecting (time range & event unselection)
+ ------------------------------------------------------------------------------------------------------------------*/
+ // TODO: move consistently to down/start or up/end?
+ // TODO: don't kill previous selection if touch scrolling
+ View.prototype.handleDocumentMousedown = function (ev) {
+ if (util_1.isPrimaryMouseButton(ev)) {
+ this.processUnselect(ev);
+ }
+ };
+ View.prototype.processUnselect = function (ev) {
+ this.processRangeUnselect(ev);
+ this.processEventUnselect(ev);
+ };
+ View.prototype.processRangeUnselect = function (ev) {
+ var ignore;
+ // is there a time-range selection?
+ if (this.isSelected && this.opt('unselectAuto')) {
+ // only unselect if the clicked element is not identical to or inside of an 'unselectCancel' element
+ ignore = this.opt('unselectCancel');
+ if (!ignore || !$(ev.target).closest(ignore).length) {
+ this.unselect(ev);
+ }
+ }
+ };
+ View.prototype.processEventUnselect = function (ev) {
+ if (this.selectedEventInstance) {
+ if (!$(ev.target).closest('.fc-selected').length) {
+ this.unselectEventInstance();
+ }
+ }
+ };
+ /* Triggers
+ ------------------------------------------------------------------------------------------------------------------*/
+ View.prototype.triggerBaseRendered = function () {
+ this.publiclyTrigger('viewRender', {
+ context: this,
+ args: [this, this.el]
+ });
+ };
+ View.prototype.triggerBaseUnrendered = function () {
+ this.publiclyTrigger('viewDestroy', {
+ context: this,
+ args: [this, this.el]
+ });
+ };
+ // Triggers handlers to 'dayClick'
+ // Span has start/end of the clicked area. Only the start is useful.
+ View.prototype.triggerDayClick = function (footprint, dayEl, ev) {
+ var dateProfile = this.calendar.footprintToDateProfile(footprint); // abuse of "Event"DateProfile?
+ this.publiclyTrigger('dayClick', {
+ context: dayEl,
+ args: [dateProfile.start, ev, this]
+ });
+ };
+ /* Date Utils
+ ------------------------------------------------------------------------------------------------------------------*/
+ // For DateComponent::getDayClasses
+ View.prototype.isDateInOtherMonth = function (date, dateProfile) {
+ return false;
+ };
+ // Arguments after name will be forwarded to a hypothetical function value
+ // WARNING: passed-in arguments will be given to generator functions as-is and can cause side-effects.
+ // Always clone your objects if you fear mutation.
+ View.prototype.getUnzonedRangeOption = function (name) {
+ var val = this.opt(name);
+ if (typeof val === 'function') {
+ val = val.apply(null, Array.prototype.slice.call(arguments, 1));
+ }
+ if (val) {
+ return this.calendar.parseUnzonedRange(val);
+ }
+ };
+ /* Hidden Days
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Initializes internal variables related to calculating hidden days-of-week
+ View.prototype.initHiddenDays = function () {
+ var hiddenDays = this.opt('hiddenDays') || []; // array of day-of-week indices that are hidden
+ var isHiddenDayHash = []; // is the day-of-week hidden? (hash with day-of-week-index -> bool)
+ var dayCnt = 0;
+ var i;
+ if (this.opt('weekends') === false) {
+ hiddenDays.push(0, 6); // 0=sunday, 6=saturday
+ }
+ for (i = 0; i < 7; i++) {
+ if (!(isHiddenDayHash[i] = $.inArray(i, hiddenDays) !== -1)) {
+ dayCnt++;
+ }
+ }
+ if (!dayCnt) {
+ throw 'invalid hiddenDays'; // all days were hidden? bad.
+ }
+ this.isHiddenDayHash = isHiddenDayHash;
+ };
+ // Remove days from the beginning and end of the range that are computed as hidden.
+ // If the whole range is trimmed off, returns null
+ View.prototype.trimHiddenDays = function (inputUnzonedRange) {
+ var start = inputUnzonedRange.getStart();
+ var end = inputUnzonedRange.getEnd();
+ if (start) {
+ start = this.skipHiddenDays(start);
+ }
+ if (end) {
+ end = this.skipHiddenDays(end, -1, true);
+ }
+ if (start === null || end === null || start < end) {
+ return new UnzonedRange_1.default(start, end);
+ }
+ return null;
+ };
+ // Is the current day hidden?
+ // `day` is a day-of-week index (0-6), or a Moment
+ View.prototype.isHiddenDay = function (day) {
+ if (moment.isMoment(day)) {
+ day = day.day();
+ }
+ return this.isHiddenDayHash[day];
+ };
+ // Incrementing the current day until it is no longer a hidden day, returning a copy.
+ // DOES NOT CONSIDER validUnzonedRange!
+ // If the initial value of `date` is not a hidden day, don't do anything.
+ // Pass `isExclusive` as `true` if you are dealing with an end date.
+ // `inc` defaults to `1` (increment one day forward each time)
+ View.prototype.skipHiddenDays = function (date, inc, isExclusive) {
+ if (inc === void 0) { inc = 1; }
+ if (isExclusive === void 0) { isExclusive = false; }
+ var out = date.clone();
+ while (this.isHiddenDayHash[(out.day() + (isExclusive ? inc : 0) + 7) % 7]) {
+ out.add(inc, 'days');
+ }
+ return out;
+ };
+ return View;
+}(InteractiveDateComponent_1.default));
+exports.default = View;
+View.prototype.usesMinMaxTime = false;
+View.prototype.dateProfileGeneratorClass = DateProfileGenerator_1.default;
+View.watch('displayingDates', ['isInDom', 'dateProfile'], function (deps) {
+ this.requestDateRender(deps.dateProfile);
+}, function () {
+ this.requestDateUnrender();
+});
+View.watch('displayingBusinessHours', ['displayingDates', 'businessHourGenerator'], function (deps) {
+ this.requestBusinessHoursRender(deps.businessHourGenerator);
+}, function () {
+ this.requestBusinessHoursUnrender();
+});
+View.watch('initialEvents', ['dateProfile'], function (deps) {
+ return this.fetchInitialEvents(deps.dateProfile);
+});
+View.watch('bindingEvents', ['initialEvents'], function (deps) {
+ this.setEvents(deps.initialEvents);
+ this.bindEventChanges();
+}, function () {
+ this.unbindEventChanges();
+ this.unsetEvents();
+});
+View.watch('displayingEvents', ['displayingDates', 'hasEvents'], function () {
+ this.requestEventsRender(this.get('currentEvents'));
+}, function () {
+ this.requestEventsUnrender();
+});
+View.watch('title', ['dateProfile'], function (deps) {
+ return (this.title = this.computeTitle(deps.dateProfile)); // assign to View for legacy reasons
+});
+View.watch('legacyDateProps', ['dateProfile'], function (deps) {
+ var calendar = this.calendar;
+ var dateProfile = deps.dateProfile;
+ // DEPRECATED, but we need to keep it updated...
+ this.start = calendar.msToMoment(dateProfile.activeUnzonedRange.startMs, dateProfile.isRangeAllDay);
+ this.end = calendar.msToMoment(dateProfile.activeUnzonedRange.endMs, dateProfile.isRangeAllDay);
+ this.intervalStart = calendar.msToMoment(dateProfile.currentUnzonedRange.startMs, dateProfile.isRangeAllDay);
+ this.intervalEnd = calendar.msToMoment(dateProfile.currentUnzonedRange.endMs, dateProfile.isRangeAllDay);
+});
+
+
+/***/ }),
+/* 31 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var $ = __webpack_require__(1);
+var util_1 = __webpack_require__(2);
+var EventRenderer = /** @class */ (function () {
+ function EventRenderer(component, fillRenderer) {
+ this.view = component._getView();
+ this.component = component;
+ this.fillRenderer = fillRenderer;
+ }
+ EventRenderer.prototype.opt = function (name) {
+ return this.view.opt(name);
+ };
+ // Updates values that rely on options and also relate to range
+ EventRenderer.prototype.rangeUpdated = function () {
+ var displayEventTime;
+ var displayEventEnd;
+ this.eventTimeFormat =
+ this.opt('eventTimeFormat') ||
+ this.opt('timeFormat') || // deprecated
+ this.computeEventTimeFormat();
+ displayEventTime = this.opt('displayEventTime');
+ if (displayEventTime == null) {
+ displayEventTime = this.computeDisplayEventTime(); // might be based off of range
+ }
+ displayEventEnd = this.opt('displayEventEnd');
+ if (displayEventEnd == null) {
+ displayEventEnd = this.computeDisplayEventEnd(); // might be based off of range
+ }
+ this.displayEventTime = displayEventTime;
+ this.displayEventEnd = displayEventEnd;
+ };
+ EventRenderer.prototype.render = function (eventsPayload) {
+ var dateProfile = this.component._getDateProfile();
+ var eventDefId;
+ var instanceGroup;
+ var eventRanges;
+ var bgRanges = [];
+ var fgRanges = [];
+ for (eventDefId in eventsPayload) {
+ instanceGroup = eventsPayload[eventDefId];
+ eventRanges = instanceGroup.sliceRenderRanges(dateProfile.activeUnzonedRange);
+ if (instanceGroup.getEventDef().hasBgRendering()) {
+ bgRanges.push.apply(bgRanges, eventRanges);
+ }
+ else {
+ fgRanges.push.apply(fgRanges, eventRanges);
+ }
+ }
+ this.renderBgRanges(bgRanges);
+ this.renderFgRanges(fgRanges);
+ };
+ EventRenderer.prototype.unrender = function () {
+ this.unrenderBgRanges();
+ this.unrenderFgRanges();
+ };
+ EventRenderer.prototype.renderFgRanges = function (eventRanges) {
+ var eventFootprints = this.component.eventRangesToEventFootprints(eventRanges);
+ var segs = this.component.eventFootprintsToSegs(eventFootprints);
+ // render an `.el` on each seg
+ // returns a subset of the segs. segs that were actually rendered
+ segs = this.renderFgSegEls(segs);
+ if (this.renderFgSegs(segs) !== false) {
+ this.fgSegs = segs;
+ }
+ };
+ EventRenderer.prototype.unrenderFgRanges = function () {
+ this.unrenderFgSegs(this.fgSegs || []);
+ this.fgSegs = null;
+ };
+ EventRenderer.prototype.renderBgRanges = function (eventRanges) {
+ var eventFootprints = this.component.eventRangesToEventFootprints(eventRanges);
+ var segs = this.component.eventFootprintsToSegs(eventFootprints);
+ if (this.renderBgSegs(segs) !== false) {
+ this.bgSegs = segs;
+ }
+ };
+ EventRenderer.prototype.unrenderBgRanges = function () {
+ this.unrenderBgSegs();
+ this.bgSegs = null;
+ };
+ EventRenderer.prototype.getSegs = function () {
+ return (this.bgSegs || []).concat(this.fgSegs || []);
+ };
+ // Renders foreground event segments onto the grid
+ EventRenderer.prototype.renderFgSegs = function (segs) {
+ // subclasses must implement
+ // segs already has rendered els, and has been filtered.
+ return false; // signal failure if not implemented
+ };
+ // Unrenders all currently rendered foreground segments
+ EventRenderer.prototype.unrenderFgSegs = function (segs) {
+ // subclasses must implement
+ };
+ EventRenderer.prototype.renderBgSegs = function (segs) {
+ var _this = this;
+ if (this.fillRenderer) {
+ this.fillRenderer.renderSegs('bgEvent', segs, {
+ getClasses: function (seg) {
+ return _this.getBgClasses(seg.footprint.eventDef);
+ },
+ getCss: function (seg) {
+ return {
+ 'background-color': _this.getBgColor(seg.footprint.eventDef)
+ };
+ },
+ filterEl: function (seg, el) {
+ return _this.filterEventRenderEl(seg.footprint, el);
+ }
+ });
+ }
+ else {
+ return false; // signal failure if no fillRenderer
+ }
+ };
+ EventRenderer.prototype.unrenderBgSegs = function () {
+ if (this.fillRenderer) {
+ this.fillRenderer.unrender('bgEvent');
+ }
+ };
+ // Renders and assigns an `el` property for each foreground event segment.
+ // Only returns segments that successfully rendered.
+ EventRenderer.prototype.renderFgSegEls = function (segs, disableResizing) {
+ var _this = this;
+ if (disableResizing === void 0) { disableResizing = false; }
+ var hasEventRenderHandlers = this.view.hasPublicHandlers('eventRender');
+ var html = '';
+ var renderedSegs = [];
+ var i;
+ if (segs.length) {
+ // build a large concatenation of event segment HTML
+ for (i = 0; i < segs.length; i++) {
+ this.beforeFgSegHtml(segs[i]);
+ html += this.fgSegHtml(segs[i], disableResizing);
+ }
+ // Grab individual elements from the combined HTML string. Use each as the default rendering.
+ // Then, compute the 'el' for each segment. An el might be null if the eventRender callback returned false.
+ $(html).each(function (i, node) {
+ var seg = segs[i];
+ var el = $(node);
+ if (hasEventRenderHandlers) {
+ el = _this.filterEventRenderEl(seg.footprint, el);
+ }
+ if (el) {
+ el.data('fc-seg', seg); // used by handlers
+ seg.el = el;
+ renderedSegs.push(seg);
+ }
+ });
+ }
+ return renderedSegs;
+ };
+ EventRenderer.prototype.beforeFgSegHtml = function (seg) {
+ };
+ // Generates the HTML for the default rendering of a foreground event segment. Used by renderFgSegEls()
+ EventRenderer.prototype.fgSegHtml = function (seg, disableResizing) {
+ // subclasses should implement
+ };
+ // Generic utility for generating the HTML classNames for an event segment's element
+ EventRenderer.prototype.getSegClasses = function (seg, isDraggable, isResizable) {
+ var classes = [
+ 'fc-event',
+ seg.isStart ? 'fc-start' : 'fc-not-start',
+ seg.isEnd ? 'fc-end' : 'fc-not-end'
+ ].concat(this.getClasses(seg.footprint.eventDef));
+ if (isDraggable) {
+ classes.push('fc-draggable');
+ }
+ if (isResizable) {
+ classes.push('fc-resizable');
+ }
+ // event is currently selected? attach a className.
+ if (this.view.isEventDefSelected(seg.footprint.eventDef)) {
+ classes.push('fc-selected');
+ }
+ return classes;
+ };
+ // Given an event and the default element used for rendering, returns the element that should actually be used.
+ // Basically runs events and elements through the eventRender hook.
+ EventRenderer.prototype.filterEventRenderEl = function (eventFootprint, el) {
+ var legacy = eventFootprint.getEventLegacy();
+ var custom = this.view.publiclyTrigger('eventRender', {
+ context: legacy,
+ args: [legacy, el, this.view]
+ });
+ if (custom === false) {
+ el = null;
+ }
+ else if (custom && custom !== true) {
+ el = $(custom);
+ }
+ return el;
+ };
+ // Compute the text that should be displayed on an event's element.
+ // `range` can be the Event object itself, or something range-like, with at least a `start`.
+ // If event times are disabled, or the event has no time, will return a blank string.
+ // If not specified, formatStr will default to the eventTimeFormat setting,
+ // and displayEnd will default to the displayEventEnd setting.
+ EventRenderer.prototype.getTimeText = function (eventFootprint, formatStr, displayEnd) {
+ return this._getTimeText(eventFootprint.eventInstance.dateProfile.start, eventFootprint.eventInstance.dateProfile.end, eventFootprint.componentFootprint.isAllDay, formatStr, displayEnd);
+ };
+ EventRenderer.prototype._getTimeText = function (start, end, isAllDay, formatStr, displayEnd) {
+ if (formatStr == null) {
+ formatStr = this.eventTimeFormat;
+ }
+ if (displayEnd == null) {
+ displayEnd = this.displayEventEnd;
+ }
+ if (this.displayEventTime && !isAllDay) {
+ if (displayEnd && end) {
+ return this.view.formatRange({ start: start, end: end }, false, // allDay
+ formatStr);
+ }
+ else {
+ return start.format(formatStr);
+ }
+ }
+ return '';
+ };
+ EventRenderer.prototype.computeEventTimeFormat = function () {
+ return this.opt('smallTimeFormat');
+ };
+ EventRenderer.prototype.computeDisplayEventTime = function () {
+ return true;
+ };
+ EventRenderer.prototype.computeDisplayEventEnd = function () {
+ return true;
+ };
+ EventRenderer.prototype.getBgClasses = function (eventDef) {
+ var classNames = this.getClasses(eventDef);
+ classNames.push('fc-bgevent');
+ return classNames;
+ };
+ EventRenderer.prototype.getClasses = function (eventDef) {
+ var objs = this.getStylingObjs(eventDef);
+ var i;
+ var classNames = [];
+ for (i = 0; i < objs.length; i++) {
+ classNames.push.apply(// append
+ classNames, objs[i].eventClassName || objs[i].className || []);
+ }
+ return classNames;
+ };
+ // Utility for generating event skin-related CSS properties
+ EventRenderer.prototype.getSkinCss = function (eventDef) {
+ return {
+ 'background-color': this.getBgColor(eventDef),
+ 'border-color': this.getBorderColor(eventDef),
+ color: this.getTextColor(eventDef)
+ };
+ };
+ // Queries for caller-specified color, then falls back to default
+ EventRenderer.prototype.getBgColor = function (eventDef) {
+ var objs = this.getStylingObjs(eventDef);
+ var i;
+ var val;
+ for (i = 0; i < objs.length && !val; i++) {
+ val = objs[i].eventBackgroundColor || objs[i].eventColor ||
+ objs[i].backgroundColor || objs[i].color;
+ }
+ if (!val) {
+ val = this.opt('eventBackgroundColor') || this.opt('eventColor');
+ }
+ return val;
+ };
+ // Queries for caller-specified color, then falls back to default
+ EventRenderer.prototype.getBorderColor = function (eventDef) {
+ var objs = this.getStylingObjs(eventDef);
+ var i;
+ var val;
+ for (i = 0; i < objs.length && !val; i++) {
+ val = objs[i].eventBorderColor || objs[i].eventColor ||
+ objs[i].borderColor || objs[i].color;
+ }
+ if (!val) {
+ val = this.opt('eventBorderColor') || this.opt('eventColor');
+ }
+ return val;
+ };
+ // Queries for caller-specified color, then falls back to default
+ EventRenderer.prototype.getTextColor = function (eventDef) {
+ var objs = this.getStylingObjs(eventDef);
+ var i;
+ var val;
+ for (i = 0; i < objs.length && !val; i++) {
+ val = objs[i].eventTextColor ||
+ objs[i].textColor;
+ }
+ if (!val) {
+ val = this.opt('eventTextColor');
+ }
+ return val;
+ };
+ EventRenderer.prototype.getStylingObjs = function (eventDef) {
+ var objs = this.getFallbackStylingObjs(eventDef);
+ objs.unshift(eventDef);
+ return objs;
+ };
+ EventRenderer.prototype.getFallbackStylingObjs = function (eventDef) {
+ return [eventDef.source];
+ };
+ EventRenderer.prototype.sortEventSegs = function (segs) {
+ segs.sort(util_1.proxy(this, 'compareEventSegs'));
+ };
+ // A cmp function for determining which segments should take visual priority
+ EventRenderer.prototype.compareEventSegs = function (seg1, seg2) {
+ var f1 = seg1.footprint.componentFootprint;
+ var r1 = f1.unzonedRange;
+ var f2 = seg2.footprint.componentFootprint;
+ var r2 = f2.unzonedRange;
+ return r1.startMs - r2.startMs || // earlier events go first
+ (r2.endMs - r2.startMs) - (r1.endMs - r1.startMs) || // tie? longer events go first
+ f2.isAllDay - f1.isAllDay || // tie? put all-day events first (booleans cast to 0/1)
+ util_1.compareByFieldSpecs(seg1.footprint.eventDef, seg2.footprint.eventDef, this.view.eventOrderSpecs);
+ };
+ return EventRenderer;
+}());
+exports.default = EventRenderer;
+
+
+/***/ }),
+/* 32 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var Class_1 = __webpack_require__(21);
+var EmitterMixin_1 = __webpack_require__(8);
+var ListenerMixin_1 = __webpack_require__(6);
+var Model = /** @class */ (function (_super) {
+ tslib_1.__extends(Model, _super);
+ function Model() {
+ var _this = _super.call(this) || this;
+ _this._watchers = {};
+ _this._props = {};
+ _this.applyGlobalWatchers();
+ _this.constructed();
+ return _this;
+ }
+ // useful for monkeypatching. TODO: BaseClass?
+ Model.prototype.constructed = function () {
+ };
+ Model.prototype.applyGlobalWatchers = function () {
+ var map = this._globalWatchArgs;
+ var name;
+ for (name in map) {
+ this.watch.apply(this, [name].concat(map[name]));
+ }
+ };
+ Model.prototype.has = function (name) {
+ return name in this._props;
+ };
+ Model.prototype.get = function (name) {
+ if (name === undefined) {
+ return this._props;
+ }
+ return this._props[name];
+ };
+ Model.prototype.set = function (name, val) {
+ var newProps;
+ if (typeof name === 'string') {
+ newProps = {};
+ newProps[name] = val === undefined ? null : val;
+ }
+ else {
+ newProps = name;
+ }
+ this.setProps(newProps);
+ };
+ Model.prototype.reset = function (newProps) {
+ var oldProps = this._props;
+ var changeset = {}; // will have undefined's to signal unsets
+ var name;
+ for (name in oldProps) {
+ changeset[name] = undefined;
+ }
+ for (name in newProps) {
+ changeset[name] = newProps[name];
+ }
+ this.setProps(changeset);
+ };
+ Model.prototype.unset = function (name) {
+ var newProps = {};
+ var names;
+ var i;
+ if (typeof name === 'string') {
+ names = [name];
+ }
+ else {
+ names = name;
+ }
+ for (i = 0; i < names.length; i++) {
+ newProps[names[i]] = undefined;
+ }
+ this.setProps(newProps);
+ };
+ Model.prototype.setProps = function (newProps) {
+ var changedProps = {};
+ var changedCnt = 0;
+ var name, val;
+ for (name in newProps) {
+ val = newProps[name];
+ // a change in value?
+ // if an object, don't check equality, because might have been mutated internally.
+ // TODO: eventually enforce immutability.
+ if (typeof val === 'object' ||
+ val !== this._props[name]) {
+ changedProps[name] = val;
+ changedCnt++;
+ }
+ }
+ if (changedCnt) {
+ this.trigger('before:batchChange', changedProps);
+ for (name in changedProps) {
+ val = changedProps[name];
+ this.trigger('before:change', name, val);
+ this.trigger('before:change:' + name, val);
+ }
+ for (name in changedProps) {
+ val = changedProps[name];
+ if (val === undefined) {
+ delete this._props[name];
+ }
+ else {
+ this._props[name] = val;
+ }
+ this.trigger('change:' + name, val);
+ this.trigger('change', name, val);
+ }
+ this.trigger('batchChange', changedProps);
+ }
+ };
+ Model.prototype.watch = function (name, depList, startFunc, stopFunc) {
+ var _this = this;
+ this.unwatch(name);
+ this._watchers[name] = this._watchDeps(depList, function (deps) {
+ var res = startFunc.call(_this, deps);
+ if (res && res.then) {
+ _this.unset(name); // put in an unset state while resolving
+ res.then(function (val) {
+ _this.set(name, val);
+ });
+ }
+ else {
+ _this.set(name, res);
+ }
+ }, function (deps) {
+ _this.unset(name);
+ if (stopFunc) {
+ stopFunc.call(_this, deps);
+ }
+ });
+ };
+ Model.prototype.unwatch = function (name) {
+ var watcher = this._watchers[name];
+ if (watcher) {
+ delete this._watchers[name];
+ watcher.teardown();
+ }
+ };
+ Model.prototype._watchDeps = function (depList, startFunc, stopFunc) {
+ var _this = this;
+ var queuedChangeCnt = 0;
+ var depCnt = depList.length;
+ var satisfyCnt = 0;
+ var values = {}; // what's passed as the `deps` arguments
+ var bindTuples = []; // array of [ eventName, handlerFunc ] arrays
+ var isCallingStop = false;
+ var onBeforeDepChange = function (depName, val, isOptional) {
+ queuedChangeCnt++;
+ if (queuedChangeCnt === 1) {
+ if (satisfyCnt === depCnt) {
+ isCallingStop = true;
+ stopFunc(values);
+ isCallingStop = false;
+ }
+ }
+ };
+ var onDepChange = function (depName, val, isOptional) {
+ if (val === undefined) {
+ // required dependency that was previously set?
+ if (!isOptional && values[depName] !== undefined) {
+ satisfyCnt--;
+ }
+ delete values[depName];
+ }
+ else {
+ // required dependency that was previously unset?
+ if (!isOptional && values[depName] === undefined) {
+ satisfyCnt++;
+ }
+ values[depName] = val;
+ }
+ queuedChangeCnt--;
+ if (!queuedChangeCnt) {
+ // now finally satisfied or satisfied all along?
+ if (satisfyCnt === depCnt) {
+ // if the stopFunc initiated another value change, ignore it.
+ // it will be processed by another change event anyway.
+ if (!isCallingStop) {
+ startFunc(values);
+ }
+ }
+ }
+ };
+ // intercept for .on() that remembers handlers
+ var bind = function (eventName, handler) {
+ _this.on(eventName, handler);
+ bindTuples.push([eventName, handler]);
+ };
+ // listen to dependency changes
+ depList.forEach(function (depName) {
+ var isOptional = false;
+ if (depName.charAt(0) === '?') {
+ depName = depName.substring(1);
+ isOptional = true;
+ }
+ bind('before:change:' + depName, function (val) {
+ onBeforeDepChange(depName, val, isOptional);
+ });
+ bind('change:' + depName, function (val) {
+ onDepChange(depName, val, isOptional);
+ });
+ });
+ // process current dependency values
+ depList.forEach(function (depName) {
+ var isOptional = false;
+ if (depName.charAt(0) === '?') {
+ depName = depName.substring(1);
+ isOptional = true;
+ }
+ if (_this.has(depName)) {
+ values[depName] = _this.get(depName);
+ satisfyCnt++;
+ }
+ else if (isOptional) {
+ satisfyCnt++;
+ }
+ });
+ // initially satisfied
+ if (satisfyCnt === depCnt) {
+ startFunc(values);
+ }
+ return {
+ teardown: function () {
+ // remove all handlers
+ for (var i = 0; i < bindTuples.length; i++) {
+ _this.off(bindTuples[i][0], bindTuples[i][1]);
+ }
+ bindTuples = null;
+ // was satisfied, so call stopFunc
+ if (satisfyCnt === depCnt) {
+ stopFunc();
+ }
+ },
+ flash: function () {
+ if (satisfyCnt === depCnt) {
+ stopFunc();
+ startFunc(values);
+ }
+ }
+ };
+ };
+ Model.prototype.flash = function (name) {
+ var watcher = this._watchers[name];
+ if (watcher) {
+ watcher.flash();
+ }
+ };
+ Model.watch = function (name) {
+ var args = [];
+ for (var _i = 1; _i < arguments.length; _i++) {
+ args[_i - 1] = arguments[_i];
+ }
+ // subclasses should make a masked-copy of the superclass's map
+ // TODO: write test
+ if (!this.prototype.hasOwnProperty('_globalWatchArgs')) {
+ this.prototype._globalWatchArgs = Object.create(this.prototype._globalWatchArgs);
+ }
+ this.prototype._globalWatchArgs[name] = args;
+ };
+ return Model;
+}(Class_1.default));
+exports.default = Model;
+Model.prototype._globalWatchArgs = {}; // mutation protection in Model.watch
+EmitterMixin_1.default.mixInto(Model);
+ListenerMixin_1.default.mixInto(Model);
+
+
+/***/ }),
+/* 33 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var moment = __webpack_require__(3);
+var util_1 = __webpack_require__(2);
+var SingleEventDef_1 = __webpack_require__(11);
+var RecurringEventDef_1 = __webpack_require__(52);
+exports.default = {
+ parse: function (eventInput, source) {
+ if (util_1.isTimeString(eventInput.start) || moment.isDuration(eventInput.start) ||
+ util_1.isTimeString(eventInput.end) || moment.isDuration(eventInput.end)) {
+ return RecurringEventDef_1.default.parse(eventInput, source);
+ }
+ else {
+ return SingleEventDef_1.default.parse(eventInput, source);
+ }
+ }
+};
+
+
+/***/ }),
+/* 34 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var $ = __webpack_require__(1);
+var util_1 = __webpack_require__(2);
+var Promise_1 = __webpack_require__(16);
+var EventSource_1 = __webpack_require__(5);
+var SingleEventDef_1 = __webpack_require__(11);
+var ArrayEventSource = /** @class */ (function (_super) {
+ tslib_1.__extends(ArrayEventSource, _super);
+ function ArrayEventSource(calendar) {
+ var _this = _super.call(this, calendar) || this;
+ _this.eventDefs = []; // for if setRawEventDefs is never called
+ return _this;
+ }
+ ArrayEventSource.prototype.setRawEventDefs = function (rawEventDefs) {
+ this.rawEventDefs = rawEventDefs;
+ this.eventDefs = this.parseEventDefs(rawEventDefs);
+ };
+ ArrayEventSource.prototype.fetch = function (start, end, timezone) {
+ var eventDefs = this.eventDefs;
+ var i;
+ if (this.currentTimezone != null &&
+ this.currentTimezone !== timezone) {
+ for (i = 0; i < eventDefs.length; i++) {
+ if (eventDefs[i] instanceof SingleEventDef_1.default) {
+ eventDefs[i].rezone();
+ }
+ }
+ }
+ this.currentTimezone = timezone;
+ return Promise_1.default.resolve(eventDefs);
+ };
+ ArrayEventSource.prototype.addEventDef = function (eventDef) {
+ this.eventDefs.push(eventDef);
+ };
+ /*
+ eventDefId already normalized to a string
+ */
+ ArrayEventSource.prototype.removeEventDefsById = function (eventDefId) {
+ return util_1.removeMatching(this.eventDefs, function (eventDef) {
+ return eventDef.id === eventDefId;
+ });
+ };
+ ArrayEventSource.prototype.removeAllEventDefs = function () {
+ this.eventDefs = [];
+ };
+ ArrayEventSource.prototype.getPrimitive = function () {
+ return this.rawEventDefs;
+ };
+ ArrayEventSource.prototype.applyManualStandardProps = function (rawProps) {
+ var superSuccess = _super.prototype.applyManualStandardProps.call(this, rawProps);
+ this.setRawEventDefs(rawProps.events);
+ return superSuccess;
+ };
+ ArrayEventSource.parse = function (rawInput, calendar) {
+ var rawProps;
+ // normalize raw input
+ if ($.isArray(rawInput.events)) {
+ rawProps = rawInput;
+ }
+ else if ($.isArray(rawInput)) {
+ rawProps = { events: rawInput };
+ }
+ if (rawProps) {
+ return EventSource_1.default.parse.call(this, rawProps, calendar);
+ }
+ return false;
+ };
+ return ArrayEventSource;
+}(EventSource_1.default));
+exports.default = ArrayEventSource;
+ArrayEventSource.defineStandardProps({
+ events: false // don't automatically transfer
+});
+
+
+/***/ }),
+/* 35 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var util_1 = __webpack_require__(2);
+var EventDateProfile_1 = __webpack_require__(15);
+var EventDefDateMutation = /** @class */ (function () {
+ function EventDefDateMutation() {
+ this.clearEnd = false;
+ this.forceTimed = false;
+ this.forceAllDay = false;
+ }
+ /*
+ returns an undo function.
+ */
+ EventDefDateMutation.prototype.buildNewDateProfile = function (eventDateProfile, calendar) {
+ var start = eventDateProfile.start.clone();
+ var end = null;
+ var shouldRezone = false;
+ if (eventDateProfile.end && !this.clearEnd) {
+ end = eventDateProfile.end.clone();
+ }
+ else if (this.endDelta && !end) {
+ end = calendar.getDefaultEventEnd(eventDateProfile.isAllDay(), start);
+ }
+ if (this.forceTimed) {
+ shouldRezone = true;
+ if (!start.hasTime()) {
+ start.time(0);
+ }
+ if (end && !end.hasTime()) {
+ end.time(0);
+ }
+ }
+ else if (this.forceAllDay) {
+ if (start.hasTime()) {
+ start.stripTime();
+ }
+ if (end && end.hasTime()) {
+ end.stripTime();
+ }
+ }
+ if (this.dateDelta) {
+ shouldRezone = true;
+ start.add(this.dateDelta);
+ if (end) {
+ end.add(this.dateDelta);
+ }
+ }
+ // do this before adding startDelta to start, so we can work off of start
+ if (this.endDelta) {
+ shouldRezone = true;
+ end.add(this.endDelta);
+ }
+ if (this.startDelta) {
+ shouldRezone = true;
+ start.add(this.startDelta);
+ }
+ if (shouldRezone) {
+ start = calendar.applyTimezone(start);
+ if (end) {
+ end = calendar.applyTimezone(end);
+ }
+ }
+ // TODO: okay to access calendar option?
+ if (!end && calendar.opt('forceEventDuration')) {
+ end = calendar.getDefaultEventEnd(eventDateProfile.isAllDay(), start);
+ }
+ return new EventDateProfile_1.default(start, end, calendar);
+ };
+ EventDefDateMutation.prototype.setDateDelta = function (dateDelta) {
+ if (dateDelta && dateDelta.valueOf()) {
+ this.dateDelta = dateDelta;
+ }
+ else {
+ this.dateDelta = null;
+ }
+ };
+ EventDefDateMutation.prototype.setStartDelta = function (startDelta) {
+ if (startDelta && startDelta.valueOf()) {
+ this.startDelta = startDelta;
+ }
+ else {
+ this.startDelta = null;
+ }
+ };
+ EventDefDateMutation.prototype.setEndDelta = function (endDelta) {
+ if (endDelta && endDelta.valueOf()) {
+ this.endDelta = endDelta;
+ }
+ else {
+ this.endDelta = null;
+ }
+ };
+ EventDefDateMutation.prototype.isEmpty = function () {
+ return !this.clearEnd && !this.forceTimed && !this.forceAllDay &&
+ !this.dateDelta && !this.startDelta && !this.endDelta;
+ };
+ EventDefDateMutation.createFromDiff = function (dateProfile0, dateProfile1, largeUnit) {
+ var clearEnd = dateProfile0.end && !dateProfile1.end;
+ var forceTimed = dateProfile0.isAllDay() && !dateProfile1.isAllDay();
+ var forceAllDay = !dateProfile0.isAllDay() && dateProfile1.isAllDay();
+ var dateDelta;
+ var endDiff;
+ var endDelta;
+ var mutation;
+ // subtracts the dates in the appropriate way, returning a duration
+ function subtractDates(date1, date0) {
+ if (largeUnit) {
+ return util_1.diffByUnit(date1, date0, largeUnit); // poorly named
+ }
+ else if (dateProfile1.isAllDay()) {
+ return util_1.diffDay(date1, date0); // poorly named
+ }
+ else {
+ return util_1.diffDayTime(date1, date0); // poorly named
+ }
+ }
+ dateDelta = subtractDates(dateProfile1.start, dateProfile0.start);
+ if (dateProfile1.end) {
+ // use unzonedRanges because dateProfile0.end might be null
+ endDiff = subtractDates(dateProfile1.unzonedRange.getEnd(), dateProfile0.unzonedRange.getEnd());
+ endDelta = endDiff.subtract(dateDelta);
+ }
+ mutation = new EventDefDateMutation();
+ mutation.clearEnd = clearEnd;
+ mutation.forceTimed = forceTimed;
+ mutation.forceAllDay = forceAllDay;
+ mutation.setDateDelta(dateDelta);
+ mutation.setEndDelta(endDelta);
+ return mutation;
+ };
+ return EventDefDateMutation;
+}());
+exports.default = EventDefDateMutation;
+
+
+/***/ }),
+/* 36 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var StandardTheme_1 = __webpack_require__(55);
+var JqueryUiTheme_1 = __webpack_require__(56);
+exports.default = {
+ themeClassHash: {},
+ register: function (themeName, themeClass) {
+ this.themeClassHash[themeName] = themeClass;
+ },
+ getThemeClass: function (themeSetting) {
+ if (!themeSetting) {
+ return StandardTheme_1.default;
+ }
+ else if (themeSetting === true) {
+ return JqueryUiTheme_1.default;
+ }
+ else {
+ return this.themeClassHash[themeSetting];
+ }
+ }
+};
+
+
+/***/ }),
+/* 37 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var moment_ext_1 = __webpack_require__(9);
+// Plugin
+// -------------------------------------------------------------------------------------------------
+moment_ext_1.newMomentProto.format = function () {
+ if (this._fullCalendar && arguments[0]) {
+ return formatDate(this, arguments[0]); // our extended formatting
+ }
+ if (this._ambigTime) {
+ return moment_ext_1.oldMomentFormat(englishMoment(this), 'YYYY-MM-DD');
+ }
+ if (this._ambigZone) {
+ return moment_ext_1.oldMomentFormat(englishMoment(this), 'YYYY-MM-DD[T]HH:mm:ss');
+ }
+ if (this._fullCalendar) {
+ // moment.format() doesn't ensure english, but we want to.
+ return moment_ext_1.oldMomentFormat(englishMoment(this));
+ }
+ return moment_ext_1.oldMomentProto.format.apply(this, arguments);
+};
+moment_ext_1.newMomentProto.toISOString = function () {
+ if (this._ambigTime) {
+ return moment_ext_1.oldMomentFormat(englishMoment(this), 'YYYY-MM-DD');
+ }
+ if (this._ambigZone) {
+ return moment_ext_1.oldMomentFormat(englishMoment(this), 'YYYY-MM-DD[T]HH:mm:ss');
+ }
+ if (this._fullCalendar) {
+ // depending on browser, moment might not output english. ensure english.
+ // https://github.com/moment/moment/blob/2.18.1/src/lib/moment/format.js#L22
+ return moment_ext_1.oldMomentProto.toISOString.apply(englishMoment(this), arguments);
+ }
+ return moment_ext_1.oldMomentProto.toISOString.apply(this, arguments);
+};
+function englishMoment(mom) {
+ if (mom.locale() !== 'en') {
+ return mom.clone().locale('en');
+ }
+ return mom;
+}
+// Config
+// ---------------------------------------------------------------------------------------------------------------------
+/*
+Inserted between chunks in the fake ("intermediate") formatting string.
+Important that it passes as whitespace (\s) because moment often identifies non-standalone months
+via a regexp with an \s.
+*/
+var PART_SEPARATOR = '\u000b'; // vertical tab
+/*
+Inserted as the first character of a literal-text chunk to indicate that the literal text is not actually literal text,
+but rather, a "special" token that has custom rendering (see specialTokens map).
+*/
+var SPECIAL_TOKEN_MARKER = '\u001f'; // information separator 1
+/*
+Inserted at the beginning and end of a span of text that must have non-zero numeric characters.
+Handling of these markers is done in a post-processing step at the very end of text rendering.
+*/
+var MAYBE_MARKER = '\u001e'; // information separator 2
+var MAYBE_REGEXP = new RegExp(MAYBE_MARKER + '([^' + MAYBE_MARKER + ']*)' + MAYBE_MARKER, 'g'); // must be global
+/*
+Addition formatting tokens we want recognized
+*/
+var specialTokens = {
+ t: function (date) {
+ return moment_ext_1.oldMomentFormat(date, 'a').charAt(0);
+ },
+ T: function (date) {
+ return moment_ext_1.oldMomentFormat(date, 'A').charAt(0);
+ }
+};
+/*
+The first characters of formatting tokens for units that are 1 day or larger.
+`value` is for ranking relative size (lower means bigger).
+`unit` is a normalized unit, used for comparing moments.
+*/
+var largeTokenMap = {
+ Y: { value: 1, unit: 'year' },
+ M: { value: 2, unit: 'month' },
+ W: { value: 3, unit: 'week' },
+ w: { value: 3, unit: 'week' },
+ D: { value: 4, unit: 'day' },
+ d: { value: 4, unit: 'day' } // day of week
+};
+// Single Date Formatting
+// ---------------------------------------------------------------------------------------------------------------------
+/*
+Formats `date` with a Moment formatting string, but allow our non-zero areas and special token
+*/
+function formatDate(date, formatStr) {
+ return renderFakeFormatString(getParsedFormatString(formatStr).fakeFormatString, date);
+}
+exports.formatDate = formatDate;
+// Date Range Formatting
+// -------------------------------------------------------------------------------------------------
+// TODO: make it work with timezone offset
+/*
+Using a formatting string meant for a single date, generate a range string, like
+"Sep 2 - 9 2013", that intelligently inserts a separator where the dates differ.
+If the dates are the same as far as the format string is concerned, just return a single
+rendering of one date, without any separator.
+*/
+function formatRange(date1, date2, formatStr, separator, isRTL) {
+ var localeData;
+ date1 = moment_ext_1.default.parseZone(date1);
+ date2 = moment_ext_1.default.parseZone(date2);
+ localeData = date1.localeData();
+ // Expand localized format strings, like "LL" -> "MMMM D YYYY".
+ // BTW, this is not important for `formatDate` because it is impossible to put custom tokens
+ // or non-zero areas in Moment's localized format strings.
+ formatStr = localeData.longDateFormat(formatStr) || formatStr;
+ return renderParsedFormat(getParsedFormatString(formatStr), date1, date2, separator || ' - ', isRTL);
+}
+exports.formatRange = formatRange;
+/*
+Renders a range with an already-parsed format string.
+*/
+function renderParsedFormat(parsedFormat, date1, date2, separator, isRTL) {
+ var sameUnits = parsedFormat.sameUnits;
+ var unzonedDate1 = date1.clone().stripZone(); // for same-unit comparisons
+ var unzonedDate2 = date2.clone().stripZone(); // "
+ var renderedParts1 = renderFakeFormatStringParts(parsedFormat.fakeFormatString, date1);
+ var renderedParts2 = renderFakeFormatStringParts(parsedFormat.fakeFormatString, date2);
+ var leftI;
+ var leftStr = '';
+ var rightI;
+ var rightStr = '';
+ var middleI;
+ var middleStr1 = '';
+ var middleStr2 = '';
+ var middleStr = '';
+ // Start at the leftmost side of the formatting string and continue until you hit a token
+ // that is not the same between dates.
+ for (leftI = 0; leftI < sameUnits.length && (!sameUnits[leftI] || unzonedDate1.isSame(unzonedDate2, sameUnits[leftI])); leftI++) {
+ leftStr += renderedParts1[leftI];
+ }
+ // Similarly, start at the rightmost side of the formatting string and move left
+ for (rightI = sameUnits.length - 1; rightI > leftI && (!sameUnits[rightI] || unzonedDate1.isSame(unzonedDate2, sameUnits[rightI])); rightI--) {
+ // If current chunk is on the boundary of unique date-content, and is a special-case
+ // date-formatting postfix character, then don't consume it. Consider it unique date-content.
+ // TODO: make configurable
+ if (rightI - 1 === leftI && renderedParts1[rightI] === '.') {
+ break;
+ }
+ rightStr = renderedParts1[rightI] + rightStr;
+ }
+ // The area in the middle is different for both of the dates.
+ // Collect them distinctly so we can jam them together later.
+ for (middleI = leftI; middleI <= rightI; middleI++) {
+ middleStr1 += renderedParts1[middleI];
+ middleStr2 += renderedParts2[middleI];
+ }
+ if (middleStr1 || middleStr2) {
+ if (isRTL) {
+ middleStr = middleStr2 + separator + middleStr1;
+ }
+ else {
+ middleStr = middleStr1 + separator + middleStr2;
+ }
+ }
+ return processMaybeMarkers(leftStr + middleStr + rightStr);
+}
+// Format String Parsing
+// ---------------------------------------------------------------------------------------------------------------------
+var parsedFormatStrCache = {};
+/*
+Returns a parsed format string, leveraging a cache.
+*/
+function getParsedFormatString(formatStr) {
+ return parsedFormatStrCache[formatStr] ||
+ (parsedFormatStrCache[formatStr] = parseFormatString(formatStr));
+}
+/*
+Parses a format string into the following:
+- fakeFormatString: a momentJS formatting string, littered with special control characters that get post-processed.
+- sameUnits: for every part in fakeFormatString, if the part is a token, the value will be a unit string (like "day"),
+ that indicates how similar a range's start & end must be in order to share the same formatted text.
+ If not a token, then the value is null.
+ Always a flat array (not nested liked "chunks").
+*/
+function parseFormatString(formatStr) {
+ var chunks = chunkFormatString(formatStr);
+ return {
+ fakeFormatString: buildFakeFormatString(chunks),
+ sameUnits: buildSameUnits(chunks)
+ };
+}
+/*
+Break the formatting string into an array of chunks.
+A 'maybe' chunk will have nested chunks.
+*/
+function chunkFormatString(formatStr) {
+ var chunks = [];
+ var match;
+ // TODO: more descrimination
+ // \4 is a backreference to the first character of a multi-character set.
+ var chunker = /\[([^\]]*)\]|\(([^\)]*)\)|(LTS|LT|(\w)\4*o?)|([^\w\[\(]+)/g;
+ while ((match = chunker.exec(formatStr))) {
+ if (match[1]) {
+ chunks.push.apply(chunks, // append
+ splitStringLiteral(match[1]));
+ }
+ else if (match[2]) {
+ chunks.push({ maybe: chunkFormatString(match[2]) });
+ }
+ else if (match[3]) {
+ chunks.push({ token: match[3] });
+ }
+ else if (match[5]) {
+ chunks.push.apply(chunks, // append
+ splitStringLiteral(match[5]));
+ }
+ }
+ return chunks;
+}
+/*
+Potentially splits a literal-text string into multiple parts. For special cases.
+*/
+function splitStringLiteral(s) {
+ if (s === '. ') {
+ return ['.', ' ']; // for locales with periods bound to the end of each year/month/date
+ }
+ else {
+ return [s];
+ }
+}
+/*
+Given chunks parsed from a real format string, generate a fake (aka "intermediate") format string with special control
+characters that will eventually be given to moment for formatting, and then post-processed.
+*/
+function buildFakeFormatString(chunks) {
+ var parts = [];
+ var i, chunk;
+ for (i = 0; i < chunks.length; i++) {
+ chunk = chunks[i];
+ if (typeof chunk === 'string') {
+ parts.push('[' + chunk + ']');
+ }
+ else if (chunk.token) {
+ if (chunk.token in specialTokens) {
+ parts.push(SPECIAL_TOKEN_MARKER + // useful during post-processing
+ '[' + chunk.token + ']' // preserve as literal text
+ );
+ }
+ else {
+ parts.push(chunk.token); // unprotected text implies a format string
+ }
+ }
+ else if (chunk.maybe) {
+ parts.push(MAYBE_MARKER + // useful during post-processing
+ buildFakeFormatString(chunk.maybe) +
+ MAYBE_MARKER);
+ }
+ }
+ return parts.join(PART_SEPARATOR);
+}
+/*
+Given parsed chunks from a real formatting string, generates an array of unit strings (like "day") that indicate
+in which regard two dates must be similar in order to share range formatting text.
+The `chunks` can be nested (because of "maybe" chunks), however, the returned array will be flat.
+*/
+function buildSameUnits(chunks) {
+ var units = [];
+ var i, chunk;
+ var tokenInfo;
+ for (i = 0; i < chunks.length; i++) {
+ chunk = chunks[i];
+ if (chunk.token) {
+ tokenInfo = largeTokenMap[chunk.token.charAt(0)];
+ units.push(tokenInfo ? tokenInfo.unit : 'second'); // default to a very strict same-second
+ }
+ else if (chunk.maybe) {
+ units.push.apply(units, // append
+ buildSameUnits(chunk.maybe));
+ }
+ else {
+ units.push(null);
+ }
+ }
+ return units;
+}
+// Rendering to text
+// ---------------------------------------------------------------------------------------------------------------------
+/*
+Formats a date with a fake format string, post-processes the control characters, then returns.
+*/
+function renderFakeFormatString(fakeFormatString, date) {
+ return processMaybeMarkers(renderFakeFormatStringParts(fakeFormatString, date).join(''));
+}
+/*
+Formats a date into parts that will have been post-processed, EXCEPT for the "maybe" markers.
+*/
+function renderFakeFormatStringParts(fakeFormatString, date) {
+ var parts = [];
+ var fakeRender = moment_ext_1.oldMomentFormat(date, fakeFormatString);
+ var fakeParts = fakeRender.split(PART_SEPARATOR);
+ var i, fakePart;
+ for (i = 0; i < fakeParts.length; i++) {
+ fakePart = fakeParts[i];
+ if (fakePart.charAt(0) === SPECIAL_TOKEN_MARKER) {
+ parts.push(
+ // the literal string IS the token's name.
+ // call special token's registered function.
+ specialTokens[fakePart.substring(1)](date));
+ }
+ else {
+ parts.push(fakePart);
+ }
+ }
+ return parts;
+}
+/*
+Accepts an almost-finally-formatted string and processes the "maybe" control characters, returning a new string.
+*/
+function processMaybeMarkers(s) {
+ return s.replace(MAYBE_REGEXP, function (m0, m1) {
+ if (m1.match(/[1-9]/)) {
+ return m1;
+ }
+ else {
+ return '';
+ }
+ });
+}
+// Misc Utils
+// -------------------------------------------------------------------------------------------------
+/*
+Returns a unit string, either 'year', 'month', 'day', or null for the most granular formatting token in the string.
+*/
+function queryMostGranularFormatUnit(formatStr) {
+ var chunks = chunkFormatString(formatStr);
+ var i, chunk;
+ var candidate;
+ var best;
+ for (i = 0; i < chunks.length; i++) {
+ chunk = chunks[i];
+ if (chunk.token) {
+ candidate = largeTokenMap[chunk.token.charAt(0)];
+ if (candidate) {
+ if (!best || candidate.value > best.value) {
+ best = candidate;
+ }
+ }
+ }
+ }
+ if (best) {
+ return best.unit;
+ }
+ return null;
+}
+exports.queryMostGranularFormatUnit = queryMostGranularFormatUnit;
+
+
+/***/ }),
+/* 38 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var $ = __webpack_require__(1);
+var util_1 = __webpack_require__(2);
+/*
+A cache for the left/right/top/bottom/width/height values for one or more elements.
+Works with both offset (from topleft document) and position (from offsetParent).
+
+options:
+- els
+- isHorizontal
+- isVertical
+*/
+var CoordCache = /** @class */ (function () {
+ function CoordCache(options) {
+ this.isHorizontal = false; // whether to query for left/right/width
+ this.isVertical = false; // whether to query for top/bottom/height
+ this.els = $(options.els);
+ this.isHorizontal = options.isHorizontal;
+ this.isVertical = options.isVertical;
+ this.forcedOffsetParentEl = options.offsetParent ? $(options.offsetParent) : null;
+ }
+ // Queries the els for coordinates and stores them.
+ // Call this method before using and of the get* methods below.
+ CoordCache.prototype.build = function () {
+ var offsetParentEl = this.forcedOffsetParentEl;
+ if (!offsetParentEl && this.els.length > 0) {
+ offsetParentEl = this.els.eq(0).offsetParent();
+ }
+ this.origin = offsetParentEl ?
+ offsetParentEl.offset() :
+ null;
+ this.boundingRect = this.queryBoundingRect();
+ if (this.isHorizontal) {
+ this.buildElHorizontals();
+ }
+ if (this.isVertical) {
+ this.buildElVerticals();
+ }
+ };
+ // Destroys all internal data about coordinates, freeing memory
+ CoordCache.prototype.clear = function () {
+ this.origin = null;
+ this.boundingRect = null;
+ this.lefts = null;
+ this.rights = null;
+ this.tops = null;
+ this.bottoms = null;
+ };
+ // When called, if coord caches aren't built, builds them
+ CoordCache.prototype.ensureBuilt = function () {
+ if (!this.origin) {
+ this.build();
+ }
+ };
+ // Populates the left/right internal coordinate arrays
+ CoordCache.prototype.buildElHorizontals = function () {
+ var lefts = [];
+ var rights = [];
+ this.els.each(function (i, node) {
+ var el = $(node);
+ var left = el.offset().left;
+ var width = el.outerWidth();
+ lefts.push(left);
+ rights.push(left + width);
+ });
+ this.lefts = lefts;
+ this.rights = rights;
+ };
+ // Populates the top/bottom internal coordinate arrays
+ CoordCache.prototype.buildElVerticals = function () {
+ var tops = [];
+ var bottoms = [];
+ this.els.each(function (i, node) {
+ var el = $(node);
+ var top = el.offset().top;
+ var height = el.outerHeight();
+ tops.push(top);
+ bottoms.push(top + height);
+ });
+ this.tops = tops;
+ this.bottoms = bottoms;
+ };
+ // Given a left offset (from document left), returns the index of the el that it horizontally intersects.
+ // If no intersection is made, returns undefined.
+ CoordCache.prototype.getHorizontalIndex = function (leftOffset) {
+ this.ensureBuilt();
+ var lefts = this.lefts;
+ var rights = this.rights;
+ var len = lefts.length;
+ var i;
+ for (i = 0; i < len; i++) {
+ if (leftOffset >= lefts[i] && leftOffset < rights[i]) {
+ return i;
+ }
+ }
+ };
+ // Given a top offset (from document top), returns the index of the el that it vertically intersects.
+ // If no intersection is made, returns undefined.
+ CoordCache.prototype.getVerticalIndex = function (topOffset) {
+ this.ensureBuilt();
+ var tops = this.tops;
+ var bottoms = this.bottoms;
+ var len = tops.length;
+ var i;
+ for (i = 0; i < len; i++) {
+ if (topOffset >= tops[i] && topOffset < bottoms[i]) {
+ return i;
+ }
+ }
+ };
+ // Gets the left offset (from document left) of the element at the given index
+ CoordCache.prototype.getLeftOffset = function (leftIndex) {
+ this.ensureBuilt();
+ return this.lefts[leftIndex];
+ };
+ // Gets the left position (from offsetParent left) of the element at the given index
+ CoordCache.prototype.getLeftPosition = function (leftIndex) {
+ this.ensureBuilt();
+ return this.lefts[leftIndex] - this.origin.left;
+ };
+ // Gets the right offset (from document left) of the element at the given index.
+ // This value is NOT relative to the document's right edge, like the CSS concept of "right" would be.
+ CoordCache.prototype.getRightOffset = function (leftIndex) {
+ this.ensureBuilt();
+ return this.rights[leftIndex];
+ };
+ // Gets the right position (from offsetParent left) of the element at the given index.
+ // This value is NOT relative to the offsetParent's right edge, like the CSS concept of "right" would be.
+ CoordCache.prototype.getRightPosition = function (leftIndex) {
+ this.ensureBuilt();
+ return this.rights[leftIndex] - this.origin.left;
+ };
+ // Gets the width of the element at the given index
+ CoordCache.prototype.getWidth = function (leftIndex) {
+ this.ensureBuilt();
+ return this.rights[leftIndex] - this.lefts[leftIndex];
+ };
+ // Gets the top offset (from document top) of the element at the given index
+ CoordCache.prototype.getTopOffset = function (topIndex) {
+ this.ensureBuilt();
+ return this.tops[topIndex];
+ };
+ // Gets the top position (from offsetParent top) of the element at the given position
+ CoordCache.prototype.getTopPosition = function (topIndex) {
+ this.ensureBuilt();
+ return this.tops[topIndex] - this.origin.top;
+ };
+ // Gets the bottom offset (from the document top) of the element at the given index.
+ // This value is NOT relative to the offsetParent's bottom edge, like the CSS concept of "bottom" would be.
+ CoordCache.prototype.getBottomOffset = function (topIndex) {
+ this.ensureBuilt();
+ return this.bottoms[topIndex];
+ };
+ // Gets the bottom position (from the offsetParent top) of the element at the given index.
+ // This value is NOT relative to the offsetParent's bottom edge, like the CSS concept of "bottom" would be.
+ CoordCache.prototype.getBottomPosition = function (topIndex) {
+ this.ensureBuilt();
+ return this.bottoms[topIndex] - this.origin.top;
+ };
+ // Gets the height of the element at the given index
+ CoordCache.prototype.getHeight = function (topIndex) {
+ this.ensureBuilt();
+ return this.bottoms[topIndex] - this.tops[topIndex];
+ };
+ // Bounding Rect
+ // TODO: decouple this from CoordCache
+ // Compute and return what the elements' bounding rectangle is, from the user's perspective.
+ // Right now, only returns a rectangle if constrained by an overflow:scroll element.
+ // Returns null if there are no elements
+ CoordCache.prototype.queryBoundingRect = function () {
+ var scrollParentEl;
+ if (this.els.length > 0) {
+ scrollParentEl = util_1.getScrollParent(this.els.eq(0));
+ if (!scrollParentEl.is(document)) {
+ return util_1.getClientRect(scrollParentEl);
+ }
+ }
+ return null;
+ };
+ CoordCache.prototype.isPointInBounds = function (leftOffset, topOffset) {
+ return this.isLeftInBounds(leftOffset) && this.isTopInBounds(topOffset);
+ };
+ CoordCache.prototype.isLeftInBounds = function (leftOffset) {
+ return !this.boundingRect || (leftOffset >= this.boundingRect.left && leftOffset < this.boundingRect.right);
+ };
+ CoordCache.prototype.isTopInBounds = function (topOffset) {
+ return !this.boundingRect || (topOffset >= this.boundingRect.top && topOffset < this.boundingRect.bottom);
+ };
+ return CoordCache;
+}());
+exports.default = CoordCache;
+
+
+/***/ }),
+/* 39 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var $ = __webpack_require__(1);
+var util_1 = __webpack_require__(2);
+var ListenerMixin_1 = __webpack_require__(6);
+var GlobalEmitter_1 = __webpack_require__(14);
+/* Tracks a drag's mouse movement, firing various handlers
+----------------------------------------------------------------------------------------------------------------------*/
+// TODO: use Emitter
+var DragListener = /** @class */ (function () {
+ function DragListener(options) {
+ this.isInteracting = false;
+ this.isDistanceSurpassed = false;
+ this.isDelayEnded = false;
+ this.isDragging = false;
+ this.isTouch = false;
+ this.isGeneric = false; // initiated by 'dragstart' (jqui)
+ this.shouldCancelTouchScroll = true;
+ this.scrollAlwaysKills = false;
+ this.isAutoScroll = false;
+ // defaults
+ this.scrollSensitivity = 30; // pixels from edge for scrolling to start
+ this.scrollSpeed = 200; // pixels per second, at maximum speed
+ this.scrollIntervalMs = 50; // millisecond wait between scroll increment
+ this.options = options || {};
+ }
+ // Interaction (high-level)
+ // -----------------------------------------------------------------------------------------------------------------
+ DragListener.prototype.startInteraction = function (ev, extraOptions) {
+ if (extraOptions === void 0) { extraOptions = {}; }
+ if (ev.type === 'mousedown') {
+ if (GlobalEmitter_1.default.get().shouldIgnoreMouse()) {
+ return;
+ }
+ else if (!util_1.isPrimaryMouseButton(ev)) {
+ return;
+ }
+ else {
+ ev.preventDefault(); // prevents native selection in most browsers
+ }
+ }
+ if (!this.isInteracting) {
+ // process options
+ this.delay = util_1.firstDefined(extraOptions.delay, this.options.delay, 0);
+ this.minDistance = util_1.firstDefined(extraOptions.distance, this.options.distance, 0);
+ this.subjectEl = this.options.subjectEl;
+ util_1.preventSelection($('body'));
+ this.isInteracting = true;
+ this.isTouch = util_1.getEvIsTouch(ev);
+ this.isGeneric = ev.type === 'dragstart';
+ this.isDelayEnded = false;
+ this.isDistanceSurpassed = false;
+ this.originX = util_1.getEvX(ev);
+ this.originY = util_1.getEvY(ev);
+ this.scrollEl = util_1.getScrollParent($(ev.target));
+ this.bindHandlers();
+ this.initAutoScroll();
+ this.handleInteractionStart(ev);
+ this.startDelay(ev);
+ if (!this.minDistance) {
+ this.handleDistanceSurpassed(ev);
+ }
+ }
+ };
+ DragListener.prototype.handleInteractionStart = function (ev) {
+ this.trigger('interactionStart', ev);
+ };
+ DragListener.prototype.endInteraction = function (ev, isCancelled) {
+ if (this.isInteracting) {
+ this.endDrag(ev);
+ if (this.delayTimeoutId) {
+ clearTimeout(this.delayTimeoutId);
+ this.delayTimeoutId = null;
+ }
+ this.destroyAutoScroll();
+ this.unbindHandlers();
+ this.isInteracting = false;
+ this.handleInteractionEnd(ev, isCancelled);
+ util_1.allowSelection($('body'));
+ }
+ };
+ DragListener.prototype.handleInteractionEnd = function (ev, isCancelled) {
+ this.trigger('interactionEnd', ev, isCancelled || false);
+ };
+ // Binding To DOM
+ // -----------------------------------------------------------------------------------------------------------------
+ DragListener.prototype.bindHandlers = function () {
+ // some browsers (Safari in iOS 10) don't allow preventDefault on touch events that are bound after touchstart,
+ // so listen to the GlobalEmitter singleton, which is always bound, instead of the document directly.
+ var globalEmitter = GlobalEmitter_1.default.get();
+ if (this.isGeneric) {
+ this.listenTo($(document), {
+ drag: this.handleMove,
+ dragstop: this.endInteraction
+ });
+ }
+ else if (this.isTouch) {
+ this.listenTo(globalEmitter, {
+ touchmove: this.handleTouchMove,
+ touchend: this.endInteraction,
+ scroll: this.handleTouchScroll
+ });
+ }
+ else {
+ this.listenTo(globalEmitter, {
+ mousemove: this.handleMouseMove,
+ mouseup: this.endInteraction
+ });
+ }
+ this.listenTo(globalEmitter, {
+ selectstart: util_1.preventDefault,
+ contextmenu: util_1.preventDefault // long taps would open menu on Chrome dev tools
+ });
+ };
+ DragListener.prototype.unbindHandlers = function () {
+ this.stopListeningTo(GlobalEmitter_1.default.get());
+ this.stopListeningTo($(document)); // for isGeneric
+ };
+ // Drag (high-level)
+ // -----------------------------------------------------------------------------------------------------------------
+ // extraOptions ignored if drag already started
+ DragListener.prototype.startDrag = function (ev, extraOptions) {
+ this.startInteraction(ev, extraOptions); // ensure interaction began
+ if (!this.isDragging) {
+ this.isDragging = true;
+ this.handleDragStart(ev);
+ }
+ };
+ DragListener.prototype.handleDragStart = function (ev) {
+ this.trigger('dragStart', ev);
+ };
+ DragListener.prototype.handleMove = function (ev) {
+ var dx = util_1.getEvX(ev) - this.originX;
+ var dy = util_1.getEvY(ev) - this.originY;
+ var minDistance = this.minDistance;
+ var distanceSq; // current distance from the origin, squared
+ if (!this.isDistanceSurpassed) {
+ distanceSq = dx * dx + dy * dy;
+ if (distanceSq >= minDistance * minDistance) {
+ this.handleDistanceSurpassed(ev);
+ }
+ }
+ if (this.isDragging) {
+ this.handleDrag(dx, dy, ev);
+ }
+ };
+ // Called while the mouse is being moved and when we know a legitimate drag is taking place
+ DragListener.prototype.handleDrag = function (dx, dy, ev) {
+ this.trigger('drag', dx, dy, ev);
+ this.updateAutoScroll(ev); // will possibly cause scrolling
+ };
+ DragListener.prototype.endDrag = function (ev) {
+ if (this.isDragging) {
+ this.isDragging = false;
+ this.handleDragEnd(ev);
+ }
+ };
+ DragListener.prototype.handleDragEnd = function (ev) {
+ this.trigger('dragEnd', ev);
+ };
+ // Delay
+ // -----------------------------------------------------------------------------------------------------------------
+ DragListener.prototype.startDelay = function (initialEv) {
+ var _this = this;
+ if (this.delay) {
+ this.delayTimeoutId = setTimeout(function () {
+ _this.handleDelayEnd(initialEv);
+ }, this.delay);
+ }
+ else {
+ this.handleDelayEnd(initialEv);
+ }
+ };
+ DragListener.prototype.handleDelayEnd = function (initialEv) {
+ this.isDelayEnded = true;
+ if (this.isDistanceSurpassed) {
+ this.startDrag(initialEv);
+ }
+ };
+ // Distance
+ // -----------------------------------------------------------------------------------------------------------------
+ DragListener.prototype.handleDistanceSurpassed = function (ev) {
+ this.isDistanceSurpassed = true;
+ if (this.isDelayEnded) {
+ this.startDrag(ev);
+ }
+ };
+ // Mouse / Touch
+ // -----------------------------------------------------------------------------------------------------------------
+ DragListener.prototype.handleTouchMove = function (ev) {
+ // prevent inertia and touchmove-scrolling while dragging
+ if (this.isDragging && this.shouldCancelTouchScroll) {
+ ev.preventDefault();
+ }
+ this.handleMove(ev);
+ };
+ DragListener.prototype.handleMouseMove = function (ev) {
+ this.handleMove(ev);
+ };
+ // Scrolling (unrelated to auto-scroll)
+ // -----------------------------------------------------------------------------------------------------------------
+ DragListener.prototype.handleTouchScroll = function (ev) {
+ // if the drag is being initiated by touch, but a scroll happens before
+ // the drag-initiating delay is over, cancel the drag
+ if (!this.isDragging || this.scrollAlwaysKills) {
+ this.endInteraction(ev, true); // isCancelled=true
+ }
+ };
+ // Utils
+ // -----------------------------------------------------------------------------------------------------------------
+ // Triggers a callback. Calls a function in the option hash of the same name.
+ // Arguments beyond the first `name` are forwarded on.
+ DragListener.prototype.trigger = function (name) {
+ var args = [];
+ for (var _i = 1; _i < arguments.length; _i++) {
+ args[_i - 1] = arguments[_i];
+ }
+ if (this.options[name]) {
+ this.options[name].apply(this, args);
+ }
+ // makes _methods callable by event name. TODO: kill this
+ if (this['_' + name]) {
+ this['_' + name].apply(this, args);
+ }
+ };
+ // Auto-scroll
+ // -----------------------------------------------------------------------------------------------------------------
+ DragListener.prototype.initAutoScroll = function () {
+ var scrollEl = this.scrollEl;
+ this.isAutoScroll =
+ this.options.scroll &&
+ scrollEl &&
+ !scrollEl.is(window) &&
+ !scrollEl.is(document);
+ if (this.isAutoScroll) {
+ // debounce makes sure rapid calls don't happen
+ this.listenTo(scrollEl, 'scroll', util_1.debounce(this.handleDebouncedScroll, 100));
+ }
+ };
+ DragListener.prototype.destroyAutoScroll = function () {
+ this.endAutoScroll(); // kill any animation loop
+ // remove the scroll handler if there is a scrollEl
+ if (this.isAutoScroll) {
+ this.stopListeningTo(this.scrollEl, 'scroll'); // will probably get removed by unbindHandlers too :(
+ }
+ };
+ // Computes and stores the bounding rectangle of scrollEl
+ DragListener.prototype.computeScrollBounds = function () {
+ if (this.isAutoScroll) {
+ this.scrollBounds = util_1.getOuterRect(this.scrollEl);
+ // TODO: use getClientRect in future. but prevents auto scrolling when on top of scrollbars
+ }
+ };
+ // Called when the dragging is in progress and scrolling should be updated
+ DragListener.prototype.updateAutoScroll = function (ev) {
+ var sensitivity = this.scrollSensitivity;
+ var bounds = this.scrollBounds;
+ var topCloseness, bottomCloseness;
+ var leftCloseness, rightCloseness;
+ var topVel = 0;
+ var leftVel = 0;
+ if (bounds) {
+ // compute closeness to edges. valid range is from 0.0 - 1.0
+ topCloseness = (sensitivity - (util_1.getEvY(ev) - bounds.top)) / sensitivity;
+ bottomCloseness = (sensitivity - (bounds.bottom - util_1.getEvY(ev))) / sensitivity;
+ leftCloseness = (sensitivity - (util_1.getEvX(ev) - bounds.left)) / sensitivity;
+ rightCloseness = (sensitivity - (bounds.right - util_1.getEvX(ev))) / sensitivity;
+ // translate vertical closeness into velocity.
+ // mouse must be completely in bounds for velocity to happen.
+ if (topCloseness >= 0 && topCloseness <= 1) {
+ topVel = topCloseness * this.scrollSpeed * -1; // negative. for scrolling up
+ }
+ else if (bottomCloseness >= 0 && bottomCloseness <= 1) {
+ topVel = bottomCloseness * this.scrollSpeed;
+ }
+ // translate horizontal closeness into velocity
+ if (leftCloseness >= 0 && leftCloseness <= 1) {
+ leftVel = leftCloseness * this.scrollSpeed * -1; // negative. for scrolling left
+ }
+ else if (rightCloseness >= 0 && rightCloseness <= 1) {
+ leftVel = rightCloseness * this.scrollSpeed;
+ }
+ }
+ this.setScrollVel(topVel, leftVel);
+ };
+ // Sets the speed-of-scrolling for the scrollEl
+ DragListener.prototype.setScrollVel = function (topVel, leftVel) {
+ this.scrollTopVel = topVel;
+ this.scrollLeftVel = leftVel;
+ this.constrainScrollVel(); // massages into realistic values
+ // if there is non-zero velocity, and an animation loop hasn't already started, then START
+ if ((this.scrollTopVel || this.scrollLeftVel) && !this.scrollIntervalId) {
+ this.scrollIntervalId = setInterval(util_1.proxy(this, 'scrollIntervalFunc'), // scope to `this`
+ this.scrollIntervalMs);
+ }
+ };
+ // Forces scrollTopVel and scrollLeftVel to be zero if scrolling has already gone all the way
+ DragListener.prototype.constrainScrollVel = function () {
+ var el = this.scrollEl;
+ if (this.scrollTopVel < 0) {
+ if (el.scrollTop() <= 0) {
+ this.scrollTopVel = 0;
+ }
+ }
+ else if (this.scrollTopVel > 0) {
+ if (el.scrollTop() + el[0].clientHeight >= el[0].scrollHeight) {
+ this.scrollTopVel = 0;
+ }
+ }
+ if (this.scrollLeftVel < 0) {
+ if (el.scrollLeft() <= 0) {
+ this.scrollLeftVel = 0;
+ }
+ }
+ else if (this.scrollLeftVel > 0) {
+ if (el.scrollLeft() + el[0].clientWidth >= el[0].scrollWidth) {
+ this.scrollLeftVel = 0;
+ }
+ }
+ };
+ // This function gets called during every iteration of the scrolling animation loop
+ DragListener.prototype.scrollIntervalFunc = function () {
+ var el = this.scrollEl;
+ var frac = this.scrollIntervalMs / 1000; // considering animation frequency, what the vel should be mult'd by
+ // change the value of scrollEl's scroll
+ if (this.scrollTopVel) {
+ el.scrollTop(el.scrollTop() + this.scrollTopVel * frac);
+ }
+ if (this.scrollLeftVel) {
+ el.scrollLeft(el.scrollLeft() + this.scrollLeftVel * frac);
+ }
+ this.constrainScrollVel(); // since the scroll values changed, recompute the velocities
+ // if scrolled all the way, which causes the vels to be zero, stop the animation loop
+ if (!this.scrollTopVel && !this.scrollLeftVel) {
+ this.endAutoScroll();
+ }
+ };
+ // Kills any existing scrolling animation loop
+ DragListener.prototype.endAutoScroll = function () {
+ if (this.scrollIntervalId) {
+ clearInterval(this.scrollIntervalId);
+ this.scrollIntervalId = null;
+ this.handleScrollEnd();
+ }
+ };
+ // Get called when the scrollEl is scrolled (NOTE: this is delayed via debounce)
+ DragListener.prototype.handleDebouncedScroll = function () {
+ // recompute all coordinates, but *only* if this is *not* part of our scrolling animation
+ if (!this.scrollIntervalId) {
+ this.handleScrollEnd();
+ }
+ };
+ // Called when scrolling has stopped, whether through auto scroll, or the user scrolling
+ DragListener.prototype.handleScrollEnd = function () {
+ };
+ return DragListener;
+}());
+exports.default = DragListener;
+ListenerMixin_1.default.mixInto(DragListener);
+
+
+/***/ }),
+/* 40 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var util_1 = __webpack_require__(2);
+var Mixin_1 = __webpack_require__(12);
+/*
+A set of rendering and date-related methods for a visual component comprised of one or more rows of day columns.
+Prerequisite: the object being mixed into needs to be a *Grid*
+*/
+var DayTableMixin = /** @class */ (function (_super) {
+ tslib_1.__extends(DayTableMixin, _super);
+ function DayTableMixin() {
+ var _this = _super !== null && _super.apply(this, arguments) || this;
+ _this.breakOnWeeks = false; // should create a new row for each week?
+ return _this;
+ }
+ // Populates internal variables used for date calculation and rendering
+ DayTableMixin.prototype.updateDayTable = function () {
+ var t = this;
+ var view = t.view;
+ var calendar = view.calendar;
+ var date = calendar.msToUtcMoment(t.dateProfile.renderUnzonedRange.startMs, true);
+ var end = calendar.msToUtcMoment(t.dateProfile.renderUnzonedRange.endMs, true);
+ var dayIndex = -1;
+ var dayIndices = [];
+ var dayDates = [];
+ var daysPerRow;
+ var firstDay;
+ var rowCnt;
+ while (date.isBefore(end)) {
+ if (view.isHiddenDay(date)) {
+ dayIndices.push(dayIndex + 0.5); // mark that it's between indices
+ }
+ else {
+ dayIndex++;
+ dayIndices.push(dayIndex);
+ dayDates.push(date.clone());
+ }
+ date.add(1, 'days');
+ }
+ if (this.breakOnWeeks) {
+ // count columns until the day-of-week repeats
+ firstDay = dayDates[0].day();
+ for (daysPerRow = 1; daysPerRow < dayDates.length; daysPerRow++) {
+ if (dayDates[daysPerRow].day() == firstDay) {
+ break;
+ }
+ }
+ rowCnt = Math.ceil(dayDates.length / daysPerRow);
+ }
+ else {
+ rowCnt = 1;
+ daysPerRow = dayDates.length;
+ }
+ this.dayDates = dayDates;
+ this.dayIndices = dayIndices;
+ this.daysPerRow = daysPerRow;
+ this.rowCnt = rowCnt;
+ this.updateDayTableCols();
+ };
+ // Computes and assigned the colCnt property and updates any options that may be computed from it
+ DayTableMixin.prototype.updateDayTableCols = function () {
+ this.colCnt = this.computeColCnt();
+ this.colHeadFormat = this.opt('columnFormat') || this.computeColHeadFormat();
+ };
+ // Determines how many columns there should be in the table
+ DayTableMixin.prototype.computeColCnt = function () {
+ return this.daysPerRow;
+ };
+ // Computes the ambiguously-timed moment for the given cell
+ DayTableMixin.prototype.getCellDate = function (row, col) {
+ return this.dayDates[this.getCellDayIndex(row, col)].clone();
+ };
+ // Computes the ambiguously-timed date range for the given cell
+ DayTableMixin.prototype.getCellRange = function (row, col) {
+ var start = this.getCellDate(row, col);
+ var end = start.clone().add(1, 'days');
+ return { start: start, end: end };
+ };
+ // Returns the number of day cells, chronologically, from the first of the grid (0-based)
+ DayTableMixin.prototype.getCellDayIndex = function (row, col) {
+ return row * this.daysPerRow + this.getColDayIndex(col);
+ };
+ // Returns the numner of day cells, chronologically, from the first cell in *any given row*
+ DayTableMixin.prototype.getColDayIndex = function (col) {
+ if (this.isRTL) {
+ return this.colCnt - 1 - col;
+ }
+ else {
+ return col;
+ }
+ };
+ // Given a date, returns its chronolocial cell-index from the first cell of the grid.
+ // If the date lies between cells (because of hiddenDays), returns a floating-point value between offsets.
+ // If before the first offset, returns a negative number.
+ // If after the last offset, returns an offset past the last cell offset.
+ // Only works for *start* dates of cells. Will not work for exclusive end dates for cells.
+ DayTableMixin.prototype.getDateDayIndex = function (date) {
+ var dayIndices = this.dayIndices;
+ var dayOffset = date.diff(this.dayDates[0], 'days');
+ if (dayOffset < 0) {
+ return dayIndices[0] - 1;
+ }
+ else if (dayOffset >= dayIndices.length) {
+ return dayIndices[dayIndices.length - 1] + 1;
+ }
+ else {
+ return dayIndices[dayOffset];
+ }
+ };
+ /* Options
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Computes a default column header formatting string if `colFormat` is not explicitly defined
+ DayTableMixin.prototype.computeColHeadFormat = function () {
+ // if more than one week row, or if there are a lot of columns with not much space,
+ // put just the day numbers will be in each cell
+ if (this.rowCnt > 1 || this.colCnt > 10) {
+ return 'ddd'; // "Sat"
+ }
+ else if (this.colCnt > 1) {
+ return this.opt('dayOfMonthFormat'); // "Sat 12/10"
+ }
+ else {
+ return 'dddd'; // "Saturday"
+ }
+ };
+ /* Slicing
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Slices up a date range into a segment for every week-row it intersects with
+ DayTableMixin.prototype.sliceRangeByRow = function (unzonedRange) {
+ var daysPerRow = this.daysPerRow;
+ var normalRange = this.view.computeDayRange(unzonedRange); // make whole-day range, considering nextDayThreshold
+ var rangeFirst = this.getDateDayIndex(normalRange.start); // inclusive first index
+ var rangeLast = this.getDateDayIndex(normalRange.end.clone().subtract(1, 'days')); // inclusive last index
+ var segs = [];
+ var row;
+ var rowFirst, rowLast; // inclusive day-index range for current row
+ var segFirst, segLast; // inclusive day-index range for segment
+ for (row = 0; row < this.rowCnt; row++) {
+ rowFirst = row * daysPerRow;
+ rowLast = rowFirst + daysPerRow - 1;
+ // intersect segment's offset range with the row's
+ segFirst = Math.max(rangeFirst, rowFirst);
+ segLast = Math.min(rangeLast, rowLast);
+ // deal with in-between indices
+ segFirst = Math.ceil(segFirst); // in-between starts round to next cell
+ segLast = Math.floor(segLast); // in-between ends round to prev cell
+ if (segFirst <= segLast) {
+ segs.push({
+ row: row,
+ // normalize to start of row
+ firstRowDayIndex: segFirst - rowFirst,
+ lastRowDayIndex: segLast - rowFirst,
+ // must be matching integers to be the segment's start/end
+ isStart: segFirst === rangeFirst,
+ isEnd: segLast === rangeLast
+ });
+ }
+ }
+ return segs;
+ };
+ // Slices up a date range into a segment for every day-cell it intersects with.
+ // TODO: make more DRY with sliceRangeByRow somehow.
+ DayTableMixin.prototype.sliceRangeByDay = function (unzonedRange) {
+ var daysPerRow = this.daysPerRow;
+ var normalRange = this.view.computeDayRange(unzonedRange); // make whole-day range, considering nextDayThreshold
+ var rangeFirst = this.getDateDayIndex(normalRange.start); // inclusive first index
+ var rangeLast = this.getDateDayIndex(normalRange.end.clone().subtract(1, 'days')); // inclusive last index
+ var segs = [];
+ var row;
+ var rowFirst, rowLast; // inclusive day-index range for current row
+ var i;
+ var segFirst, segLast; // inclusive day-index range for segment
+ for (row = 0; row < this.rowCnt; row++) {
+ rowFirst = row * daysPerRow;
+ rowLast = rowFirst + daysPerRow - 1;
+ for (i = rowFirst; i <= rowLast; i++) {
+ // intersect segment's offset range with the row's
+ segFirst = Math.max(rangeFirst, i);
+ segLast = Math.min(rangeLast, i);
+ // deal with in-between indices
+ segFirst = Math.ceil(segFirst); // in-between starts round to next cell
+ segLast = Math.floor(segLast); // in-between ends round to prev cell
+ if (segFirst <= segLast) {
+ segs.push({
+ row: row,
+ // normalize to start of row
+ firstRowDayIndex: segFirst - rowFirst,
+ lastRowDayIndex: segLast - rowFirst,
+ // must be matching integers to be the segment's start/end
+ isStart: segFirst === rangeFirst,
+ isEnd: segLast === rangeLast
+ });
+ }
+ }
+ }
+ return segs;
+ };
+ /* Header Rendering
+ ------------------------------------------------------------------------------------------------------------------*/
+ DayTableMixin.prototype.renderHeadHtml = function () {
+ var theme = this.view.calendar.theme;
+ return '' +
+ '';
+ };
+ DayTableMixin.prototype.renderHeadIntroHtml = function () {
+ return this.renderIntroHtml(); // fall back to generic
+ };
+ DayTableMixin.prototype.renderHeadTrHtml = function () {
+ return '' +
+ ' | ' +
+ (this.isRTL ? '' : this.renderHeadIntroHtml()) +
+ this.renderHeadDateCellsHtml() +
+ (this.isRTL ? this.renderHeadIntroHtml() : '') +
+ '
';
+ };
+ DayTableMixin.prototype.renderHeadDateCellsHtml = function () {
+ var htmls = [];
+ var col, date;
+ for (col = 0; col < this.colCnt; col++) {
+ date = this.getCellDate(0, col);
+ htmls.push(this.renderHeadDateCellHtml(date));
+ }
+ return htmls.join('');
+ };
+ // TODO: when internalApiVersion, accept an object for HTML attributes
+ // (colspan should be no different)
+ DayTableMixin.prototype.renderHeadDateCellHtml = function (date, colspan, otherAttrs) {
+ var t = this;
+ var view = t.view;
+ var isDateValid = t.dateProfile.activeUnzonedRange.containsDate(date); // TODO: called too frequently. cache somehow.
+ var classNames = [
+ 'fc-day-header',
+ view.calendar.theme.getClass('widgetHeader')
+ ];
+ var innerHtml = util_1.htmlEscape(date.format(t.colHeadFormat));
+ // if only one row of days, the classNames on the header can represent the specific days beneath
+ if (t.rowCnt === 1) {
+ classNames = classNames.concat(
+ // includes the day-of-week class
+ // noThemeHighlight=true (don't highlight the header)
+ t.getDayClasses(date, true));
+ }
+ else {
+ classNames.push('fc-' + util_1.dayIDs[date.day()]); // only add the day-of-week class
+ }
+ return '' +
+ ' 1 ?
+ ' colspan="' + colspan + '"' :
+ '') +
+ (otherAttrs ?
+ ' ' + otherAttrs :
+ '') +
+ '>' +
+ (isDateValid ?
+ // don't make a link if the heading could represent multiple days, or if there's only one day (forceOff)
+ view.buildGotoAnchorHtml({ date: date, forceOff: t.rowCnt > 1 || t.colCnt === 1 }, innerHtml) :
+ // if not valid, display text, but no link
+ innerHtml) +
+ ' | ';
+ };
+ /* Background Rendering
+ ------------------------------------------------------------------------------------------------------------------*/
+ DayTableMixin.prototype.renderBgTrHtml = function (row) {
+ return '' +
+ '' +
+ (this.isRTL ? '' : this.renderBgIntroHtml(row)) +
+ this.renderBgCellsHtml(row) +
+ (this.isRTL ? this.renderBgIntroHtml(row) : '') +
+ '
';
+ };
+ DayTableMixin.prototype.renderBgIntroHtml = function (row) {
+ return this.renderIntroHtml(); // fall back to generic
+ };
+ DayTableMixin.prototype.renderBgCellsHtml = function (row) {
+ var htmls = [];
+ var col, date;
+ for (col = 0; col < this.colCnt; col++) {
+ date = this.getCellDate(row, col);
+ htmls.push(this.renderBgCellHtml(date));
+ }
+ return htmls.join('');
+ };
+ DayTableMixin.prototype.renderBgCellHtml = function (date, otherAttrs) {
+ var t = this;
+ var view = t.view;
+ var isDateValid = t.dateProfile.activeUnzonedRange.containsDate(date); // TODO: called too frequently. cache somehow.
+ var classes = t.getDayClasses(date);
+ classes.unshift('fc-day', view.calendar.theme.getClass('widgetContent'));
+ return ' | ';
+ };
+ /* Generic
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Generates the default HTML intro for any row. User classes should override
+ DayTableMixin.prototype.renderIntroHtml = function () {
+ };
+ // TODO: a generic method for dealing with , RTL, intro
+ // when increment internalApiVersion
+ // wrapTr (scheduler)
+ /* Utils
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Applies the generic "intro" and "outro" HTML to the given cells.
+ // Intro means the leftmost cell when the calendar is LTR and the rightmost cell when RTL. Vice-versa for outro.
+ DayTableMixin.prototype.bookendCells = function (trEl) {
+ var introHtml = this.renderIntroHtml();
+ if (introHtml) {
+ if (this.isRTL) {
+ trEl.append(introHtml);
+ }
+ else {
+ trEl.prepend(introHtml);
+ }
+ }
+ };
+ return DayTableMixin;
+}(Mixin_1.default));
+exports.default = DayTableMixin;
+
+
+/***/ }),
+/* 41 */
+/***/ (function(module, exports) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var BusinessHourRenderer = /** @class */ (function () {
+ /*
+ component implements:
+ - eventRangesToEventFootprints
+ - eventFootprintsToSegs
+ */
+ function BusinessHourRenderer(component, fillRenderer) {
+ this.component = component;
+ this.fillRenderer = fillRenderer;
+ }
+ BusinessHourRenderer.prototype.render = function (businessHourGenerator) {
+ var component = this.component;
+ var unzonedRange = component._getDateProfile().activeUnzonedRange;
+ var eventInstanceGroup = businessHourGenerator.buildEventInstanceGroup(component.hasAllDayBusinessHours, unzonedRange);
+ var eventFootprints = eventInstanceGroup ?
+ component.eventRangesToEventFootprints(eventInstanceGroup.sliceRenderRanges(unzonedRange)) :
+ [];
+ this.renderEventFootprints(eventFootprints);
+ };
+ BusinessHourRenderer.prototype.renderEventFootprints = function (eventFootprints) {
+ var segs = this.component.eventFootprintsToSegs(eventFootprints);
+ this.renderSegs(segs);
+ this.segs = segs;
+ };
+ BusinessHourRenderer.prototype.renderSegs = function (segs) {
+ if (this.fillRenderer) {
+ this.fillRenderer.renderSegs('businessHours', segs, {
+ getClasses: function (seg) {
+ return ['fc-nonbusiness', 'fc-bgevent'];
+ }
+ });
+ }
+ };
+ BusinessHourRenderer.prototype.unrender = function () {
+ if (this.fillRenderer) {
+ this.fillRenderer.unrender('businessHours');
+ }
+ this.segs = null;
+ };
+ BusinessHourRenderer.prototype.getSegs = function () {
+ return this.segs || [];
+ };
+ return BusinessHourRenderer;
+}());
+exports.default = BusinessHourRenderer;
+
+
+/***/ }),
+/* 42 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var $ = __webpack_require__(1);
+var util_1 = __webpack_require__(2);
+var FillRenderer = /** @class */ (function () {
+ function FillRenderer(component) {
+ this.fillSegTag = 'div';
+ this.component = component;
+ this.elsByFill = {};
+ }
+ FillRenderer.prototype.renderFootprint = function (type, componentFootprint, props) {
+ this.renderSegs(type, this.component.componentFootprintToSegs(componentFootprint), props);
+ };
+ FillRenderer.prototype.renderSegs = function (type, segs, props) {
+ var els;
+ segs = this.buildSegEls(type, segs, props); // assignes `.el` to each seg. returns successfully rendered segs
+ els = this.attachSegEls(type, segs);
+ if (els) {
+ this.reportEls(type, els);
+ }
+ return segs;
+ };
+ // Unrenders a specific type of fill that is currently rendered on the grid
+ FillRenderer.prototype.unrender = function (type) {
+ var el = this.elsByFill[type];
+ if (el) {
+ el.remove();
+ delete this.elsByFill[type];
+ }
+ };
+ // Renders and assigns an `el` property for each fill segment. Generic enough to work with different types.
+ // Only returns segments that successfully rendered.
+ FillRenderer.prototype.buildSegEls = function (type, segs, props) {
+ var _this = this;
+ var html = '';
+ var renderedSegs = [];
+ var i;
+ if (segs.length) {
+ // build a large concatenation of segment HTML
+ for (i = 0; i < segs.length; i++) {
+ html += this.buildSegHtml(type, segs[i], props);
+ }
+ // Grab individual elements from the combined HTML string. Use each as the default rendering.
+ // Then, compute the 'el' for each segment.
+ $(html).each(function (i, node) {
+ var seg = segs[i];
+ var el = $(node);
+ // allow custom filter methods per-type
+ if (props.filterEl) {
+ el = props.filterEl(seg, el);
+ }
+ if (el) {
+ el = $(el); // allow custom filter to return raw DOM node
+ // correct element type? (would be bad if a non-TD were inserted into a table for example)
+ if (el.is(_this.fillSegTag)) {
+ seg.el = el;
+ renderedSegs.push(seg);
+ }
+ }
+ });
+ }
+ return renderedSegs;
+ };
+ // Builds the HTML needed for one fill segment. Generic enough to work with different types.
+ FillRenderer.prototype.buildSegHtml = function (type, seg, props) {
+ // custom hooks per-type
+ var classes = props.getClasses ? props.getClasses(seg) : [];
+ var css = util_1.cssToStr(props.getCss ? props.getCss(seg) : {});
+ return '<' + this.fillSegTag +
+ (classes.length ? ' class="' + classes.join(' ') + '"' : '') +
+ (css ? ' style="' + css + '"' : '') +
+ ' />';
+ };
+ // Should return wrapping DOM structure
+ FillRenderer.prototype.attachSegEls = function (type, segs) {
+ // subclasses must implement
+ };
+ FillRenderer.prototype.reportEls = function (type, nodes) {
+ if (this.elsByFill[type]) {
+ this.elsByFill[type] = this.elsByFill[type].add(nodes);
+ }
+ else {
+ this.elsByFill[type] = $(nodes);
+ }
+ };
+ return FillRenderer;
+}());
+exports.default = FillRenderer;
+
+
+/***/ }),
+/* 43 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var SingleEventDef_1 = __webpack_require__(11);
+var EventFootprint_1 = __webpack_require__(24);
+var EventSource_1 = __webpack_require__(5);
+var HelperRenderer = /** @class */ (function () {
+ function HelperRenderer(component, eventRenderer) {
+ this.view = component._getView();
+ this.component = component;
+ this.eventRenderer = eventRenderer;
+ }
+ HelperRenderer.prototype.renderComponentFootprint = function (componentFootprint) {
+ this.renderEventFootprints([
+ this.fabricateEventFootprint(componentFootprint)
+ ]);
+ };
+ HelperRenderer.prototype.renderEventDraggingFootprints = function (eventFootprints, sourceSeg, isTouch) {
+ this.renderEventFootprints(eventFootprints, sourceSeg, 'fc-dragging', isTouch ? null : this.view.opt('dragOpacity'));
+ };
+ HelperRenderer.prototype.renderEventResizingFootprints = function (eventFootprints, sourceSeg, isTouch) {
+ this.renderEventFootprints(eventFootprints, sourceSeg, 'fc-resizing');
+ };
+ HelperRenderer.prototype.renderEventFootprints = function (eventFootprints, sourceSeg, extraClassNames, opacity) {
+ var segs = this.component.eventFootprintsToSegs(eventFootprints);
+ var classNames = 'fc-helper ' + (extraClassNames || '');
+ var i;
+ // assigns each seg's el and returns a subset of segs that were rendered
+ segs = this.eventRenderer.renderFgSegEls(segs);
+ for (i = 0; i < segs.length; i++) {
+ segs[i].el.addClass(classNames);
+ }
+ if (opacity != null) {
+ for (i = 0; i < segs.length; i++) {
+ segs[i].el.css('opacity', opacity);
+ }
+ }
+ this.helperEls = this.renderSegs(segs, sourceSeg);
+ };
+ /*
+ Must return all mock event elements
+ */
+ HelperRenderer.prototype.renderSegs = function (segs, sourceSeg) {
+ // Subclasses must implement
+ };
+ HelperRenderer.prototype.unrender = function () {
+ if (this.helperEls) {
+ this.helperEls.remove();
+ this.helperEls = null;
+ }
+ };
+ HelperRenderer.prototype.fabricateEventFootprint = function (componentFootprint) {
+ var calendar = this.view.calendar;
+ var eventDateProfile = calendar.footprintToDateProfile(componentFootprint);
+ var dummyEvent = new SingleEventDef_1.default(new EventSource_1.default(calendar));
+ var dummyInstance;
+ dummyEvent.dateProfile = eventDateProfile;
+ dummyInstance = dummyEvent.buildInstance();
+ return new EventFootprint_1.default(componentFootprint, dummyEvent, dummyInstance);
+ };
+ return HelperRenderer;
+}());
+exports.default = HelperRenderer;
+
+
+/***/ }),
+/* 44 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var GlobalEmitter_1 = __webpack_require__(14);
+var Interaction_1 = __webpack_require__(13);
+var EventPointing = /** @class */ (function (_super) {
+ tslib_1.__extends(EventPointing, _super);
+ function EventPointing() {
+ return _super !== null && _super.apply(this, arguments) || this;
+ }
+ /*
+ component must implement:
+ - publiclyTrigger
+ */
+ EventPointing.prototype.bindToEl = function (el) {
+ var component = this.component;
+ component.bindSegHandlerToEl(el, 'click', this.handleClick.bind(this));
+ component.bindSegHandlerToEl(el, 'mouseenter', this.handleMouseover.bind(this));
+ component.bindSegHandlerToEl(el, 'mouseleave', this.handleMouseout.bind(this));
+ };
+ EventPointing.prototype.handleClick = function (seg, ev) {
+ var res = this.component.publiclyTrigger('eventClick', {
+ context: seg.el[0],
+ args: [seg.footprint.getEventLegacy(), ev, this.view]
+ });
+ if (res === false) {
+ ev.preventDefault();
+ }
+ };
+ // Updates internal state and triggers handlers for when an event element is moused over
+ EventPointing.prototype.handleMouseover = function (seg, ev) {
+ if (!GlobalEmitter_1.default.get().shouldIgnoreMouse() &&
+ !this.mousedOverSeg) {
+ this.mousedOverSeg = seg;
+ // TODO: move to EventSelecting's responsibility
+ if (this.view.isEventDefResizable(seg.footprint.eventDef)) {
+ seg.el.addClass('fc-allow-mouse-resize');
+ }
+ this.component.publiclyTrigger('eventMouseover', {
+ context: seg.el[0],
+ args: [seg.footprint.getEventLegacy(), ev, this.view]
+ });
+ }
+ };
+ // Updates internal state and triggers handlers for when an event element is moused out.
+ // Can be given no arguments, in which case it will mouseout the segment that was previously moused over.
+ EventPointing.prototype.handleMouseout = function (seg, ev) {
+ if (this.mousedOverSeg) {
+ this.mousedOverSeg = null;
+ // TODO: move to EventSelecting's responsibility
+ if (this.view.isEventDefResizable(seg.footprint.eventDef)) {
+ seg.el.removeClass('fc-allow-mouse-resize');
+ }
+ this.component.publiclyTrigger('eventMouseout', {
+ context: seg.el[0],
+ args: [
+ seg.footprint.getEventLegacy(),
+ ev || {},
+ this.view
+ ]
+ });
+ }
+ };
+ EventPointing.prototype.end = function () {
+ if (this.mousedOverSeg) {
+ this.handleMouseout(this.mousedOverSeg);
+ }
+ };
+ return EventPointing;
+}(Interaction_1.default));
+exports.default = EventPointing;
+
+
+/***/ }),
+/* 45 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var Mixin_1 = __webpack_require__(12);
+var DateClicking_1 = __webpack_require__(82);
+var DateSelecting_1 = __webpack_require__(66);
+var EventPointing_1 = __webpack_require__(44);
+var EventDragging_1 = __webpack_require__(65);
+var EventResizing_1 = __webpack_require__(64);
+var ExternalDropping_1 = __webpack_require__(63);
+var StandardInteractionsMixin = /** @class */ (function (_super) {
+ tslib_1.__extends(StandardInteractionsMixin, _super);
+ function StandardInteractionsMixin() {
+ return _super !== null && _super.apply(this, arguments) || this;
+ }
+ return StandardInteractionsMixin;
+}(Mixin_1.default));
+exports.default = StandardInteractionsMixin;
+StandardInteractionsMixin.prototype.dateClickingClass = DateClicking_1.default;
+StandardInteractionsMixin.prototype.dateSelectingClass = DateSelecting_1.default;
+StandardInteractionsMixin.prototype.eventPointingClass = EventPointing_1.default;
+StandardInteractionsMixin.prototype.eventDraggingClass = EventDragging_1.default;
+StandardInteractionsMixin.prototype.eventResizingClass = EventResizing_1.default;
+StandardInteractionsMixin.prototype.externalDroppingClass = ExternalDropping_1.default;
+
+
+/***/ }),
+/* 46 */
+/***/ (function(module, exports, __webpack_require__) {
+
+Object.defineProperty(exports, "__esModule", { value: true });
+var tslib_1 = __webpack_require__(0);
+var $ = __webpack_require__(1);
+var util_1 = __webpack_require__(2);
+var CoordCache_1 = __webpack_require__(38);
+var Popover_1 = __webpack_require__(86);
+var UnzonedRange_1 = __webpack_require__(4);
+var ComponentFootprint_1 = __webpack_require__(10);
+var EventFootprint_1 = __webpack_require__(24);
+var BusinessHourRenderer_1 = __webpack_require__(41);
+var StandardInteractionsMixin_1 = __webpack_require__(45);
+var InteractiveDateComponent_1 = __webpack_require__(29);
+var DayTableMixin_1 = __webpack_require__(40);
+var DayGridEventRenderer_1 = __webpack_require__(87);
+var DayGridHelperRenderer_1 = __webpack_require__(88);
+var DayGridFillRenderer_1 = __webpack_require__(89);
+/* A component that renders a grid of whole-days that runs horizontally. There can be multiple rows, one per week.
+----------------------------------------------------------------------------------------------------------------------*/
+var DayGrid = /** @class */ (function (_super) {
+ tslib_1.__extends(DayGrid, _super);
+ function DayGrid(view) {
+ var _this = _super.call(this, view) || this;
+ _this.cellWeekNumbersVisible = false; // display week numbers in day cell?
+ _this.bottomCoordPadding = 0; // hack for extending the hit area for the last row of the coordinate grid
+ // isRigid determines whether the individual rows should ignore the contents and be a constant height.
+ // Relies on the view's colCnt and rowCnt. In the future, this component should probably be self-sufficient.
+ _this.isRigid = false;
+ _this.hasAllDayBusinessHours = true;
+ return _this;
+ }
+ // Slices up the given span (unzoned start/end with other misc data) into an array of segments
+ DayGrid.prototype.componentFootprintToSegs = function (componentFootprint) {
+ var segs = this.sliceRangeByRow(componentFootprint.unzonedRange);
+ var i, seg;
+ for (i = 0; i < segs.length; i++) {
+ seg = segs[i];
+ if (this.isRTL) {
+ seg.leftCol = this.daysPerRow - 1 - seg.lastRowDayIndex;
+ seg.rightCol = this.daysPerRow - 1 - seg.firstRowDayIndex;
+ }
+ else {
+ seg.leftCol = seg.firstRowDayIndex;
+ seg.rightCol = seg.lastRowDayIndex;
+ }
+ }
+ return segs;
+ };
+ /* Date Rendering
+ ------------------------------------------------------------------------------------------------------------------*/
+ DayGrid.prototype.renderDates = function (dateProfile) {
+ this.dateProfile = dateProfile;
+ this.updateDayTable();
+ this.renderGrid();
+ };
+ DayGrid.prototype.unrenderDates = function () {
+ this.removeSegPopover();
+ };
+ // Renders the rows and columns into the component's `this.el`, which should already be assigned.
+ DayGrid.prototype.renderGrid = function () {
+ var view = this.view;
+ var rowCnt = this.rowCnt;
+ var colCnt = this.colCnt;
+ var html = '';
+ var row;
+ var col;
+ if (this.headContainerEl) {
+ this.headContainerEl.html(this.renderHeadHtml());
+ }
+ for (row = 0; row < rowCnt; row++) {
+ html += this.renderDayRowHtml(row, this.isRigid);
+ }
+ this.el.html(html);
+ this.rowEls = this.el.find('.fc-row');
+ this.cellEls = this.el.find('.fc-day, .fc-disabled-day');
+ this.rowCoordCache = new CoordCache_1.default({
+ els: this.rowEls,
+ isVertical: true
+ });
+ this.colCoordCache = new CoordCache_1.default({
+ els: this.cellEls.slice(0, this.colCnt),
+ isHorizontal: true
+ });
+ // trigger dayRender with each cell's element
+ for (row = 0; row < rowCnt; row++) {
+ for (col = 0; col < colCnt; col++) {
+ this.publiclyTrigger('dayRender', {
+ context: view,
+ args: [
+ this.getCellDate(row, col),
+ this.getCellEl(row, col),
+ view
+ ]
+ });
+ }
+ }
+ };
+ // Generates the HTML for a single row, which is a div that wraps a table.
+ // `row` is the row number.
+ DayGrid.prototype.renderDayRowHtml = function (row, isRigid) {
+ var theme = this.view.calendar.theme;
+ var classes = ['fc-row', 'fc-week', theme.getClass('dayRow')];
+ if (isRigid) {
+ classes.push('fc-rigid');
+ }
+ return '' +
+ '' +
+ '
' +
+ '
' +
+ this.renderBgTrHtml(row) +
+ '
' +
+ '
' +
+ '
' +
+ '
' +
+ (this.getIsNumbersVisible() ?
+ '' +
+ this.renderNumberTrHtml(row) +
+ '' :
+ '') +
+ '
' +
+ '
' +
+ '
';
+ };
+ DayGrid.prototype.getIsNumbersVisible = function () {
+ return this.getIsDayNumbersVisible() || this.cellWeekNumbersVisible;
+ };
+ DayGrid.prototype.getIsDayNumbersVisible = function () {
+ return this.rowCnt > 1;
+ };
+ /* Grid Number Rendering
+ ------------------------------------------------------------------------------------------------------------------*/
+ DayGrid.prototype.renderNumberTrHtml = function (row) {
+ return '' +
+ '
' +
+ (this.isRTL ? '' : this.renderNumberIntroHtml(row)) +
+ this.renderNumberCellsHtml(row) +
+ (this.isRTL ? this.renderNumberIntroHtml(row) : '') +
+ '
';
+ };
+ DayGrid.prototype.renderNumberIntroHtml = function (row) {
+ return this.renderIntroHtml();
+ };
+ DayGrid.prototype.renderNumberCellsHtml = function (row) {
+ var htmls = [];
+ var col, date;
+ for (col = 0; col < this.colCnt; col++) {
+ date = this.getCellDate(row, col);
+ htmls.push(this.renderNumberCellHtml(date));
+ }
+ return htmls.join('');
+ };
+ // Generates the HTML for the s of the "number" row in the DayGrid's content skeleton.
+ // The number row will only exist if either day numbers or week numbers are turned on.
+ DayGrid.prototype.renderNumberCellHtml = function (date) {
+ var view = this.view;
+ var html = '';
+ var isDateValid = this.dateProfile.activeUnzonedRange.containsDate(date); // TODO: called too frequently. cache somehow.
+ var isDayNumberVisible = this.getIsDayNumbersVisible() && isDateValid;
+ var classes;
+ var weekCalcFirstDoW;
+ if (!isDayNumberVisible && !this.cellWeekNumbersVisible) {
+ // no numbers in day cell (week number must be along the side)
+ return ' | | '; // will create an empty space above events :(
+ }
+ classes = this.getDayClasses(date);
+ classes.unshift('fc-day-top');
+ if (this.cellWeekNumbersVisible) {
+ // To determine the day of week number change under ISO, we cannot
+ // rely on moment.js methods such as firstDayOfWeek() or weekday(),
+ // because they rely on the locale's dow (possibly overridden by
+ // our firstDay option), which may not be Monday. We cannot change
+ // dow, because that would affect the calendar start day as well.
+ if (date._locale._fullCalendar_weekCalc === 'ISO') {
+ weekCalcFirstDoW = 1; // Monday by ISO 8601 definition
+ }
+ else {
+ weekCalcFirstDoW = date._locale.firstDayOfWeek();
+ }
+ }
+ html += '';
+ if (this.cellWeekNumbersVisible && (date.day() == weekCalcFirstDoW)) {
+ html += view.buildGotoAnchorHtml({ date: date, type: 'week' }, { 'class': 'fc-week-number' }, date.format('w') // inner HTML
+ );
+ }
+ if (isDayNumberVisible) {
+ html += view.buildGotoAnchorHtml(date, { 'class': 'fc-day-number' }, date.date() // inner HTML
+ );
+ }
+ html += ' | ';
+ return html;
+ };
+ /* Hit System
+ ------------------------------------------------------------------------------------------------------------------*/
+ DayGrid.prototype.prepareHits = function () {
+ this.colCoordCache.build();
+ this.rowCoordCache.build();
+ this.rowCoordCache.bottoms[this.rowCnt - 1] += this.bottomCoordPadding; // hack
+ };
+ DayGrid.prototype.releaseHits = function () {
+ this.colCoordCache.clear();
+ this.rowCoordCache.clear();
+ };
+ DayGrid.prototype.queryHit = function (leftOffset, topOffset) {
+ if (this.colCoordCache.isLeftInBounds(leftOffset) && this.rowCoordCache.isTopInBounds(topOffset)) {
+ var col = this.colCoordCache.getHorizontalIndex(leftOffset);
+ var row = this.rowCoordCache.getVerticalIndex(topOffset);
+ if (row != null && col != null) {
+ return this.getCellHit(row, col);
+ }
+ }
+ };
+ DayGrid.prototype.getHitFootprint = function (hit) {
+ var range = this.getCellRange(hit.row, hit.col);
+ return new ComponentFootprint_1.default(new UnzonedRange_1.default(range.start, range.end), true // all-day?
+ );
+ };
+ DayGrid.prototype.getHitEl = function (hit) {
+ return this.getCellEl(hit.row, hit.col);
+ };
+ /* Cell System
+ ------------------------------------------------------------------------------------------------------------------*/
+ // FYI: the first column is the leftmost column, regardless of date
+ DayGrid.prototype.getCellHit = function (row, col) {
+ return {
+ row: row,
+ col: col,
+ component: this,
+ left: this.colCoordCache.getLeftOffset(col),
+ right: this.colCoordCache.getRightOffset(col),
+ top: this.rowCoordCache.getTopOffset(row),
+ bottom: this.rowCoordCache.getBottomOffset(row)
+ };
+ };
+ DayGrid.prototype.getCellEl = function (row, col) {
+ return this.cellEls.eq(row * this.colCnt + col);
+ };
+ /* Event Rendering
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Unrenders all events currently rendered on the grid
+ DayGrid.prototype.executeEventUnrender = function () {
+ this.removeSegPopover(); // removes the "more.." events popover
+ _super.prototype.executeEventUnrender.call(this);
+ };
+ // Retrieves all rendered segment objects currently rendered on the grid
+ DayGrid.prototype.getOwnEventSegs = function () {
+ // append the segments from the "more..." popover
+ return _super.prototype.getOwnEventSegs.call(this).concat(this.popoverSegs || []);
+ };
+ /* Event Drag Visualization
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Renders a visual indication of an event or external element being dragged.
+ // `eventLocation` has zoned start and end (optional)
+ DayGrid.prototype.renderDrag = function (eventFootprints, seg, isTouch) {
+ var i;
+ for (i = 0; i < eventFootprints.length; i++) {
+ this.renderHighlight(eventFootprints[i].componentFootprint);
+ }
+ // render drags from OTHER components as helpers
+ if (eventFootprints.length && seg && seg.component !== this) {
+ this.helperRenderer.renderEventDraggingFootprints(eventFootprints, seg, isTouch);
+ return true; // signal helpers rendered
+ }
+ };
+ // Unrenders any visual indication of a hovering event
+ DayGrid.prototype.unrenderDrag = function () {
+ this.unrenderHighlight();
+ this.helperRenderer.unrender();
+ };
+ /* Event Resize Visualization
+ ------------------------------------------------------------------------------------------------------------------*/
+ // Renders a visual indication of an event being resized
+ DayGrid.prototype.renderEventResize = function (eventFootprints, seg, isTouch) {
+ var i;
+ for (i = 0; i < eventFootprints.length; i++) {
+ this.renderHighlight(eventFootprints[i].componentFootprint);
+ }
+ this.helperRenderer.renderEventResizingFootprints(eventFootprints, seg, isTouch);
+ };
+ // Unrenders a visual indication of an event being resized
+ DayGrid.prototype.unrenderEventResize = function () {
+ this.unrenderHighlight();
+ this.helperRenderer.unrender();
+ };
+ /* More+ Link Popover
+ ------------------------------------------------------------------------------------------------------------------*/
+ DayGrid.prototype.removeSegPopover = function () {
+ if (this.segPopover) {
+ this.segPopover.hide(); // in handler, will call segPopover's removeElement
+ }
+ };
+ // Limits the number of "levels" (vertically stacking layers of events) for each row of the grid.
+ // `levelLimit` can be false (don't limit), a number, or true (should be computed).
+ DayGrid.prototype.limitRows = function (levelLimit) {
+ var rowStructs = this.eventRenderer.rowStructs || [];
+ var row; // row #
+ var rowLevelLimit;
+ for (row = 0; row < rowStructs.length; row++) {
+ this.unlimitRow(row);
+ if (!levelLimit) {
+ rowLevelLimit = false;
+ }
+ else if (typeof levelLimit === 'number') {
+ rowLevelLimit = levelLimit;
+ }
+ else {
+ rowLevelLimit = this.computeRowLevelLimit(row);
+ }
+ if (rowLevelLimit !== false) {
+ this.limitRow(row, rowLevelLimit);
+ }
+ }
+ };
+ // Computes the number of levels a row will accomodate without going outside its bounds.
+ // Assumes the row is "rigid" (maintains a constant height regardless of what is inside).
+ // `row` is the row number.
+ DayGrid.prototype.computeRowLevelLimit = function (row) {
+ var rowEl = this.rowEls.eq(row); // the containing "fake" row div
+ var rowHeight = rowEl.height(); // TODO: cache somehow?
+ var trEls = this.eventRenderer.rowStructs[row].tbodyEl.children();
+ var i, trEl;
+ var trHeight;
+ function iterInnerHeights(i, childNode) {
+ trHeight = Math.max(trHeight, $(childNode).outerHeight());
+ }
+ // Reveal one level at a time and stop when we find one out of bounds
+ for (i = 0; i < trEls.length; i++) {
+ trEl = trEls.eq(i).removeClass('fc-limited'); // reset to original state (reveal)
+ // with rowspans>1 and IE8, trEl.outerHeight() would return the height of the largest cell,
+ // so instead, find the tallest inner content element.
+ trHeight = 0;
+ trEl.find('> td > :first-child').each(iterInnerHeights);
+ if (trEl.position().top + trHeight > rowHeight) {
+ return i;
+ }
+ }
+ return false; // should not limit at all
+ };
+ // Limits the given grid row to the maximum number of levels and injects "more" links if necessary.
+ // `row` is the row number.
+ // `levelLimit` is a number for the maximum (inclusive) number of levels allowed.
+ DayGrid.prototype.limitRow = function (row, levelLimit) {
+ var _this = this;
+ var rowStruct = this.eventRenderer.rowStructs[row];
+ var moreNodes = []; // array of "more" links and DOM nodes
+ var col = 0; // col #, left-to-right (not chronologically)
+ var levelSegs; // array of segment objects in the last allowable level, ordered left-to-right
+ var cellMatrix; // a matrix (by level, then column) of all | jQuery elements in the row
+ var limitedNodes; // array of temporarily hidden level |
and segment DOM nodes
+ var i, seg;
+ var segsBelow; // array of segment objects below `seg` in the current `col`
+ var totalSegsBelow; // total number of segments below `seg` in any of the columns `seg` occupies
+ var colSegsBelow; // array of segment arrays, below seg, one for each column (offset from segs's first column)
+ var td, rowspan;
+ var segMoreNodes; // array of "more" | cells that will stand-in for the current seg's cell
+ var j;
+ var moreTd, moreWrap, moreLink;
+ // Iterates through empty level cells and places "more" links inside if need be
+ var emptyCellsUntil = function (endCol) {
+ while (col < endCol) {
+ segsBelow = _this.getCellSegs(row, col, levelLimit);
+ if (segsBelow.length) {
+ td = cellMatrix[levelLimit - 1][col];
+ moreLink = _this.renderMoreLink(row, col, segsBelow);
+ moreWrap = $('').append(moreLink);
+ td.append(moreWrap);
+ moreNodes.push(moreWrap[0]);
+ }
+ col++;
+ }
+ };
+ if (levelLimit && levelLimit < rowStruct.segLevels.length) {
+ levelSegs = rowStruct.segLevels[levelLimit - 1];
+ cellMatrix = rowStruct.cellMatrix;
+ limitedNodes = rowStruct.tbodyEl.children().slice(levelLimit) // get level |
elements past the limit
+ .addClass('fc-limited').get(); // hide elements and get a simple DOM-nodes array
+ // iterate though segments in the last allowable level
+ for (i = 0; i < levelSegs.length; i++) {
+ seg = levelSegs[i];
+ emptyCellsUntil(seg.leftCol); // process empty cells before the segment
+ // determine *all* segments below `seg` that occupy the same columns
+ colSegsBelow = [];
+ totalSegsBelow = 0;
+ while (col <= seg.rightCol) {
+ segsBelow = this.getCellSegs(row, col, levelLimit);
+ colSegsBelow.push(segsBelow);
+ totalSegsBelow += segsBelow.length;
+ col++;
+ }
+ if (totalSegsBelow) {
+ td = cellMatrix[levelLimit - 1][seg.leftCol]; // the segment's parent cell
+ rowspan = td.attr('rowspan') || 1;
+ segMoreNodes = [];
+ // make a replacement for each column the segment occupies. will be one for each colspan
+ for (j = 0; j < colSegsBelow.length; j++) {
+ moreTd = $(' | | ').attr('rowspan', rowspan);
+ segsBelow = colSegsBelow[j];
+ moreLink = this.renderMoreLink(row, seg.leftCol + j, [seg].concat(segsBelow) // count seg as hidden too
+ );
+ moreWrap = $('').append(moreLink);
+ moreTd.append(moreWrap);
+ segMoreNodes.push(moreTd[0]);
+ moreNodes.push(moreTd[0]);
+ }
+ td.addClass('fc-limited').after($(segMoreNodes)); // hide original and inject replacements
+ limitedNodes.push(td[0]);
+ }
+ }
+ emptyCellsUntil(this.colCnt); // finish off the level
+ rowStruct.moreEls = $(moreNodes); // for easy undoing later
+ rowStruct.limitedEls = $(limitedNodes); // for easy undoing later
+ }
+ };
+ // Reveals all levels and removes all "more"-related elements for a grid's row.
+ // `row` is a row number.
+ DayGrid.prototype.unlimitRow = function (row) {
+ var rowStruct = this.eventRenderer.rowStructs[row];
+ if (rowStruct.moreEls) {
+ rowStruct.moreEls.remove();
+ rowStruct.moreEls = null;
+ }
+ if (rowStruct.limitedEls) {
+ rowStruct.limitedEls.removeClass('fc-limited');
+ rowStruct.limitedEls = null;
+ }
+ };
+ // Renders an element that represents hidden event element for a cell.
+ // Responsible for attaching click handler as well.
+ DayGrid.prototype.renderMoreLink = function (row, col, hiddenSegs) {
+ var _this = this;
+ var view = this.view;
+ return $('')
+ .text(this.getMoreLinkText(hiddenSegs.length))
+ .on('click', function (ev) {
+ var clickOption = _this.opt('eventLimitClick');
+ var date = _this.getCellDate(row, col);
+ var moreEl = $(ev.currentTarget);
+ var dayEl = _this.getCellEl(row, col);
+ var allSegs = _this.getCellSegs(row, col);
+ // rescope the segments to be within the cell's date
+ var reslicedAllSegs = _this.resliceDaySegs(allSegs, date);
+ var reslicedHiddenSegs = _this.resliceDaySegs(hiddenSegs, date);
+ if (typeof clickOption === 'function') {
+ // the returned value can be an atomic option
+ clickOption = _this.publiclyTrigger('eventLimitClick', {
+ context: view,
+ args: [
+ {
+ date: date.clone(),
+ dayEl: dayEl,
+ moreEl: moreEl,
+ segs: reslicedAllSegs,
+ hiddenSegs: reslicedHiddenSegs
+ },
+ ev,
+ view
+ ]
+ });
+ }
+ if (clickOption === 'popover') {
+ _this.showSegPopover(row, col, moreEl, reslicedAllSegs);
+ }
+ else if (typeof clickOption === 'string') {
+ view.calendar.zoomTo(date, clickOption);
+ }
+ });
+ };
+ // Reveals the popover that displays all events within a cell
+ DayGrid.prototype.showSegPopover = function (row, col, moreLink, segs) {
+ var _this = this;
+ var view = this.view;
+ var moreWrap = moreLink.parent(); // the |