mirror of
https://github.com/idanoo/GoScrobble.git
synced 2024-11-22 16:35:14 +00:00
4506 lines
109 KiB
JavaScript
4506 lines
109 KiB
JavaScript
/* spencermountain/spacetime 6.16.3 Apache 2.0 */
|
|
(function (global, factory) {
|
|
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
|
|
typeof define === 'function' && define.amd ? define(factory) :
|
|
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.spacetime = factory());
|
|
}(this, (function () { 'use strict';
|
|
|
|
const MSEC_IN_HOUR = 60 * 60 * 1000; //convert our local date syntax a javascript UTC date
|
|
|
|
const toUtc = (dstChange, offset, year) => {
|
|
const [month, rest] = dstChange.split('/');
|
|
const [day, hour] = rest.split(':');
|
|
return Date.UTC(year, month - 1, day, hour) - offset * MSEC_IN_HOUR;
|
|
}; // compare epoch with dst change events (in utc)
|
|
|
|
|
|
const inSummerTime$1 = (epoch, start, end, summerOffset, winterOffset) => {
|
|
const year = new Date(epoch).getUTCFullYear();
|
|
const startUtc = toUtc(start, winterOffset, year);
|
|
const endUtc = toUtc(end, summerOffset, year); // simple number comparison now
|
|
|
|
return epoch >= startUtc && epoch < endUtc;
|
|
};
|
|
|
|
var summerTime = inSummerTime$1;
|
|
|
|
var inSummerTime = summerTime;
|
|
|
|
// it reproduces some things in ./index.js, but speeds up spacetime considerably
|
|
|
|
const quickOffset$1 = s => {
|
|
let zones = s.timezones;
|
|
let obj = zones[s.tz];
|
|
|
|
if (obj === undefined) {
|
|
console.warn("Warning: couldn't find timezone " + s.tz);
|
|
return 0;
|
|
}
|
|
|
|
if (obj.dst === undefined) {
|
|
return obj.offset;
|
|
} //get our two possible offsets
|
|
|
|
|
|
let jul = obj.offset;
|
|
let dec = obj.offset + 1; // assume it's the same for now
|
|
|
|
if (obj.hem === 'n') {
|
|
dec = jul - 1;
|
|
}
|
|
|
|
let split = obj.dst.split('->');
|
|
let inSummer = inSummerTime(s.epoch, split[0], split[1], jul, dec);
|
|
|
|
if (inSummer === true) {
|
|
return jul;
|
|
}
|
|
|
|
return dec;
|
|
};
|
|
|
|
var quick = quickOffset$1;
|
|
|
|
var _build = {
|
|
"9|s": "2/dili,2/jayapura",
|
|
"9|n": "2/chita,2/khandyga,2/pyongyang,2/seoul,2/tokyo,11/palau",
|
|
"9.5|s|04/04:03->10/03:02": "4/adelaide,4/broken_hill,4/south,4/yancowinna",
|
|
"9.5|s": "4/darwin,4/north",
|
|
"8|s|03/08:01->10/04:00": "12/casey",
|
|
"8|s": "2/kuala_lumpur,2/makassar,2/singapore,4/perth,2/ujung_pandang,4/west",
|
|
"8|n": "2/brunei,2/choibalsan,2/hong_kong,2/irkutsk,2/kuching,2/macau,2/manila,2/shanghai,2/taipei,2/ulaanbaatar,2/chongqing,2/chungking,2/harbin,2/macao,2/ulan_bator",
|
|
"8.75|s": "4/eucla",
|
|
"7|s": "12/davis,2/jakarta,9/christmas",
|
|
"7|n": "2/bangkok,2/barnaul,2/hovd,2/krasnoyarsk,2/novokuznetsk,2/novosibirsk,2/phnom_penh,2/pontianak,2/saigon,2/tomsk,2/vientiane,2/ho_chi_minh",
|
|
"6|s": "12/vostok",
|
|
"6|n": "2/almaty,2/bishkek,2/dhaka,2/omsk,2/qyzylorda,2/qostanay,2/thimphu,2/urumqi,9/chagos,2/dacca,2/kashgar,2/thimbu",
|
|
"6.5|n": "2/rangoon,9/cocos,2/yangon",
|
|
"5|s": "12/mawson,9/kerguelen",
|
|
"5|n": "2/aqtau,2/aqtobe,2/ashgabat,2/atyrau,2/baku,2/dushanbe,2/karachi,2/oral,2/samarkand,2/tashkent,2/yekaterinburg,9/maldives,2/ashkhabad",
|
|
"5.75|n": "2/katmandu,2/kathmandu",
|
|
"5.5|n": "2/calcutta,2/colombo,2/kolkata",
|
|
"4|s": "9/reunion",
|
|
"4|n": "2/dubai,2/muscat,2/tbilisi,2/yerevan,8/astrakhan,8/samara,8/saratov,8/ulyanovsk,8/volgograd,2/volgograd,9/mahe,9/mauritius",
|
|
"4.5|n|03/22:00->09/21:24": "2/tehran",
|
|
"4.5|n": "2/kabul",
|
|
"3|s": "12/syowa,9/antananarivo",
|
|
"3|n|03/28:03->10/31:04": "2/famagusta,2/nicosia,8/athens,8/bucharest,8/helsinki,8/kiev,8/mariehamn,8/riga,8/sofia,8/tallinn,8/uzhgorod,8/vilnius,8/zaporozhye,8/nicosia",
|
|
"3|n|03/28:02->10/31:03": "8/chisinau,8/tiraspol",
|
|
"3|n|03/28:00->10/30:24": "2/beirut",
|
|
"3|n|03/27:00->10/30:01": "2/gaza,2/hebron",
|
|
"3|n|03/26:02->10/31:02": "2/jerusalem,2/tel_aviv",
|
|
"3|n|03/26:00->10/29:01": "2/amman",
|
|
"3|n|03/26:00->10/28:24": "2/damascus",
|
|
"3|n": "0/addis_ababa,0/asmera,0/dar_es_salaam,0/djibouti,0/juba,0/kampala,0/mogadishu,0/nairobi,2/aden,2/baghdad,2/bahrain,2/kuwait,2/qatar,2/riyadh,8/istanbul,8/kirov,8/minsk,8/moscow,8/simferopol,9/comoro,9/mayotte,0/asmara,2/istanbul",
|
|
"2|s|03/28:02->10/31:02": "12/troll",
|
|
"2|s": "0/gaborone,0/harare,0/johannesburg,0/lubumbashi,0/lusaka,0/maputo,0/maseru,0/mbabane",
|
|
"2|n|03/28:02->10/31:03": "0/ceuta,arctic/longyearbyen,8/amsterdam,8/andorra,8/belgrade,8/berlin,8/bratislava,8/brussels,8/budapest,8/busingen,8/copenhagen,8/gibraltar,8/ljubljana,8/luxembourg,8/madrid,8/malta,8/monaco,8/oslo,8/paris,8/podgorica,8/prague,8/rome,8/san_marino,8/sarajevo,8/skopje,8/stockholm,8/tirane,8/vaduz,8/vatican,8/vienna,8/warsaw,8/zagreb,8/zurich,3/jan_mayen",
|
|
"2|n": "0/blantyre,0/bujumbura,0/cairo,0/khartoum,0/kigali,0/tripoli,8/kaliningrad",
|
|
"1|s": "0/brazzaville,0/kinshasa,0/luanda,0/windhoek",
|
|
"1|n|04/11:03->05/16:02": "0/casablanca,0/el_aaiun",
|
|
"1|n|03/28:01->10/31:02": "3/canary,3/faeroe,3/madeira,8/dublin,8/guernsey,8/isle_of_man,8/jersey,8/lisbon,8/london,3/faroe,8/belfast",
|
|
"1|n": "0/algiers,0/bangui,0/douala,0/lagos,0/libreville,0/malabo,0/ndjamena,0/niamey,0/porto-novo,0/tunis",
|
|
"14|n": "11/kiritimati",
|
|
"13|s|04/04:04->09/26:03": "11/apia",
|
|
"13|s|01/15:02->11/05:03": "11/tongatapu",
|
|
"13|n": "11/enderbury,11/fakaofo",
|
|
"12|s|04/04:03->09/26:02": "12/mcmurdo,11/auckland,12/south_pole",
|
|
"12|s|01/17:03->11/14:02": "11/fiji",
|
|
"12|n": "2/anadyr,2/kamchatka,2/srednekolymsk,11/funafuti,11/kwajalein,11/majuro,11/nauru,11/tarawa,11/wake,11/wallis",
|
|
"12.75|s|04/04:03->09/26:02": "11/chatham",
|
|
"11|s|04/04:03->10/03:02": "12/macquarie",
|
|
"11|s": "11/bougainville",
|
|
"11|n": "2/magadan,2/sakhalin,11/efate,11/guadalcanal,11/kosrae,11/noumea,11/ponape,11/pohnpei",
|
|
"11.5|n|04/04:03->10/03:02": "11/norfolk",
|
|
"10|s|04/04:03->10/03:02": "4/currie,4/hobart,4/melbourne,4/sydney,4/act,4/canberra,4/nsw,4/tasmania,4/victoria",
|
|
"10|s": "12/dumontdurville,4/brisbane,4/lindeman,11/port_moresby,4/queensland",
|
|
"10|n": "2/ust-nera,2/vladivostok,2/yakutsk,11/guam,11/saipan,11/truk,11/chuuk,11/yap",
|
|
"10.5|s|04/04:01->10/03:02": "4/lord_howe,4/lhi",
|
|
"0|n|03/28:00->10/31:01": "1/scoresbysund,3/azores",
|
|
"0|n": "0/abidjan,0/accra,0/bamako,0/banjul,0/bissau,0/conakry,0/dakar,0/freetown,0/lome,0/monrovia,0/nouakchott,0/ouagadougou,0/sao_tome,1/danmarkshavn,3/reykjavik,3/st_helena,13/gmt,13/utc,0/timbuktu,13/gmt-0,13/gmt+0,13/gmt0,13/greenwich,13/uct,13/universal,13/zulu",
|
|
"-9|n|03/14:02->11/07:02": "1/adak,1/atka,us/aleutian",
|
|
"-9|n": "11/gambier",
|
|
"-9.5|n": "11/marquesas",
|
|
"-8|n|03/14:02->11/07:02": "1/anchorage,1/juneau,1/metlakatla,1/nome,1/sitka,1/yakutat,us/alaska",
|
|
"-8|n": "11/pitcairn",
|
|
"-7|n|03/14:02->11/07:02": "1/los_angeles,1/santa_isabel,1/tijuana,1/vancouver,1/ensenada,6/pacific,10/bajanorte,us/pacific-new,us/pacific",
|
|
"-7|n|03/08:02->11/01:01": "1/dawson,1/whitehorse,6/yukon",
|
|
"-7|n": "1/creston,1/dawson_creek,1/fort_nelson,1/hermosillo,1/phoenix,us/arizona",
|
|
"-6|s|04/03:22->09/04:22": "11/easter,7/easterisland",
|
|
"-6|n|04/04:02->10/31:02": "1/chihuahua,1/mazatlan,10/bajasur",
|
|
"-6|n|03/14:02->11/07:02": "1/boise,1/cambridge_bay,1/denver,1/edmonton,1/inuvik,1/ojinaga,1/yellowknife,1/shiprock,6/mountain,us/mountain",
|
|
"-6|n": "1/belize,1/costa_rica,1/el_salvador,1/guatemala,1/managua,1/regina,1/swift_current,1/tegucigalpa,11/galapagos,6/east-saskatchewan,6/saskatchewan",
|
|
"-5|s": "1/lima,1/rio_branco,1/porto_acre,5/acre",
|
|
"-5|n|04/04:02->10/31:02": "1/bahia_banderas,1/merida,1/mexico_city,1/monterrey,10/general",
|
|
"-5|n|03/14:02->11/07:02": "1/chicago,1/matamoros,1/menominee,1/rainy_river,1/rankin_inlet,1/resolute,1/winnipeg,1/indiana/knox,1/indiana/tell_city,1/north_dakota/beulah,1/north_dakota/center,1/north_dakota/new_salem,1/knox_in,6/central,us/central,us/indiana-starke",
|
|
"-5|n|03/12:03->11/05:01": "1/north_dakota",
|
|
"-5|n": "1/bogota,1/cancun,1/cayman,1/coral_harbour,1/eirunepe,1/guayaquil,1/jamaica,1/panama,1/atikokan",
|
|
"-4|s|05/13:23->08/13:01": "12/palmer",
|
|
"-4|s|04/03:24->09/05:00": "1/santiago,7/continental",
|
|
"-4|s|03/27:24->10/03:00": "1/asuncion",
|
|
"-4|s|02/16:24->11/03:00": "1/campo_grande,1/cuiaba",
|
|
"-4|s": "1/la_paz,1/manaus,5/west",
|
|
"-4|n|03/14:02->11/07:02": "1/detroit,1/grand_turk,1/indianapolis,1/iqaluit,1/louisville,1/montreal,1/nassau,1/new_york,1/nipigon,1/pangnirtung,1/port-au-prince,1/thunder_bay,1/toronto,1/indiana/marengo,1/indiana/petersburg,1/indiana/vevay,1/indiana/vincennes,1/indiana/winamac,1/kentucky/monticello,1/fort_wayne,1/indiana/indianapolis,1/kentucky/louisville,6/eastern,us/east-indiana,us/eastern,us/michigan",
|
|
"-4|n|03/14:00->11/07:01": "1/havana",
|
|
"-4|n|03/12:03->11/05:01": "1/indiana,1/kentucky",
|
|
"-4|n": "1/anguilla,1/antigua,1/aruba,1/barbados,1/blanc-sablon,1/boa_vista,1/caracas,1/curacao,1/dominica,1/grenada,1/guadeloupe,1/guyana,1/kralendijk,1/lower_princes,1/marigot,1/martinique,1/montserrat,1/port_of_spain,1/porto_velho,1/puerto_rico,1/santo_domingo,1/st_barthelemy,1/st_kitts,1/st_lucia,1/st_thomas,1/st_vincent,1/tortola,1/virgin",
|
|
"-3|s": "1/argentina,1/buenos_aires,1/catamarca,1/cordoba,1/fortaleza,1/jujuy,1/mendoza,1/montevideo,1/punta_arenas,1/sao_paulo,12/rothera,3/stanley,1/argentina/la_rioja,1/argentina/rio_gallegos,1/argentina/salta,1/argentina/san_juan,1/argentina/san_luis,1/argentina/tucuman,1/argentina/ushuaia,1/argentina/buenos_aires,1/argentina/catamarca,1/argentina/comodrivadavia,1/argentina/cordoba,1/argentina/jujuy,1/argentina/mendoza,1/rosario,5/east",
|
|
"-3|n|03/14:02->11/07:02": "1/glace_bay,1/goose_bay,1/halifax,1/moncton,1/thule,3/bermuda,6/atlantic",
|
|
"-3|n": "1/araguaina,1/bahia,1/belem,1/cayenne,1/maceio,1/paramaribo,1/recife,1/santarem",
|
|
"-2|n|03/27:22->10/30:23": "1/godthab,1/nuuk",
|
|
"-2|n|03/14:02->11/07:02": "1/miquelon",
|
|
"-2|n": "1/noronha,3/south_georgia,5/denoronha",
|
|
"-2.5|n|03/14:02->11/07:02": "1/st_johns,6/newfoundland",
|
|
"-1|n": "3/cape_verde",
|
|
"-11|n": "11/midway,11/niue,11/pago_pago,11/samoa,us/samoa",
|
|
"-10|n": "11/honolulu,11/johnston,11/rarotonga,11/tahiti,us/hawaii"
|
|
};
|
|
|
|
var _build$1 = /*#__PURE__*/Object.freeze({
|
|
__proto__: null,
|
|
'default': _build
|
|
});
|
|
|
|
//prefixes for iana names..
|
|
var _prefixes = ['africa', 'america', 'asia', 'atlantic', 'australia', 'brazil', 'canada', 'chile', 'europe', 'indian', 'mexico', 'pacific', 'antarctica', 'etc'];
|
|
|
|
function createCommonjsModule(fn, module) {
|
|
return module = { exports: {} }, fn(module, module.exports), module.exports;
|
|
}
|
|
|
|
function getCjsExportFromNamespace (n) {
|
|
return n && n['default'] || n;
|
|
}
|
|
|
|
var data = getCjsExportFromNamespace(_build$1);
|
|
|
|
var prefixes = _prefixes;
|
|
|
|
let all = {};
|
|
Object.keys(data).forEach(k => {
|
|
let split = k.split('|');
|
|
let obj = {
|
|
offset: Number(split[0]),
|
|
hem: split[1]
|
|
};
|
|
|
|
if (split[2]) {
|
|
obj.dst = split[2];
|
|
}
|
|
|
|
let names = data[k].split(',');
|
|
names.forEach(str => {
|
|
str = str.replace(/(^[0-9]+)\//, (before, num) => {
|
|
num = Number(num);
|
|
return prefixes[num] + '/';
|
|
});
|
|
all[str] = obj;
|
|
});
|
|
});
|
|
all['utc'] = {
|
|
offset: 0,
|
|
hem: 'n' //default to northern hemisphere - (sorry!)
|
|
|
|
}; //add etc/gmt+n
|
|
|
|
for (let i = -14; i <= 14; i += 0.5) {
|
|
let num = i;
|
|
|
|
if (num > 0) {
|
|
num = '+' + num;
|
|
}
|
|
|
|
let name = 'etc/gmt' + num;
|
|
all[name] = {
|
|
offset: i * -1,
|
|
//they're negative!
|
|
hem: 'n' //(sorry)
|
|
|
|
};
|
|
name = 'utc/gmt' + num; //this one too, why not.
|
|
|
|
all[name] = {
|
|
offset: i * -1,
|
|
hem: 'n'
|
|
};
|
|
}
|
|
|
|
var unpack = all;
|
|
|
|
//find the implicit iana code for this machine.
|
|
//safely query the Intl object
|
|
//based on - https://bitbucket.org/pellepim/jstimezonedetect/src
|
|
const fallbackTZ = 'utc'; //
|
|
//this Intl object is not supported often, yet
|
|
|
|
const safeIntl = () => {
|
|
if (typeof Intl === 'undefined' || typeof Intl.DateTimeFormat === 'undefined') {
|
|
return null;
|
|
}
|
|
|
|
let format = Intl.DateTimeFormat();
|
|
|
|
if (typeof format === 'undefined' || typeof format.resolvedOptions === 'undefined') {
|
|
return null;
|
|
}
|
|
|
|
let timezone = format.resolvedOptions().timeZone;
|
|
|
|
if (!timezone) {
|
|
return null;
|
|
}
|
|
|
|
return timezone.toLowerCase();
|
|
};
|
|
|
|
const guessTz$1 = () => {
|
|
let timezone = safeIntl();
|
|
|
|
if (timezone === null) {
|
|
return fallbackTZ;
|
|
}
|
|
|
|
return timezone;
|
|
}; //do it once per computer
|
|
|
|
|
|
var guessTz_1 = guessTz$1;
|
|
|
|
const isOffset = /(\-?[0-9]+)h(rs)?/i;
|
|
const isNumber = /(\-?[0-9]+)/;
|
|
const utcOffset = /utc([\-+]?[0-9]+)/i;
|
|
const gmtOffset = /gmt([\-+]?[0-9]+)/i;
|
|
|
|
const toIana = function (num) {
|
|
num = Number(num);
|
|
|
|
if (num >= -13 && num <= 13) {
|
|
num = num * -1; //it's opposite!
|
|
|
|
num = (num > 0 ? '+' : '') + num; //add plus sign
|
|
|
|
return 'etc/gmt' + num;
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
const parseOffset$3 = function (tz) {
|
|
// '+5hrs'
|
|
let m = tz.match(isOffset);
|
|
|
|
if (m !== null) {
|
|
return toIana(m[1]);
|
|
} // 'utc+5'
|
|
|
|
|
|
m = tz.match(utcOffset);
|
|
|
|
if (m !== null) {
|
|
return toIana(m[1]);
|
|
} // 'GMT-5' (not opposite)
|
|
|
|
|
|
m = tz.match(gmtOffset);
|
|
|
|
if (m !== null) {
|
|
let num = Number(m[1]) * -1;
|
|
return toIana(num);
|
|
} // '+5'
|
|
|
|
|
|
m = tz.match(isNumber);
|
|
|
|
if (m !== null) {
|
|
return toIana(m[1]);
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
var parseOffset_1$1 = parseOffset$3;
|
|
|
|
var require$$0$4 = unpack;
|
|
|
|
var guessTz = guessTz_1;
|
|
|
|
var parseOffset$2 = parseOffset_1$1;
|
|
|
|
const local = guessTz(); //add all the city names by themselves
|
|
|
|
const cities = Object.keys(require$$0$4).reduce((h, k) => {
|
|
let city = k.split('/')[1] || '';
|
|
city = city.replace(/_/g, ' ');
|
|
h[city] = k;
|
|
return h;
|
|
}, {}); //try to match these against iana form
|
|
|
|
const normalize$2 = tz => {
|
|
tz = tz.replace(/ time/g, '');
|
|
tz = tz.replace(/ (standard|daylight|summer)/g, '');
|
|
tz = tz.replace(/\b(east|west|north|south)ern/g, '$1');
|
|
tz = tz.replace(/\b(africa|america|australia)n/g, '$1');
|
|
tz = tz.replace(/\beuropean/g, 'europe');
|
|
tz = tz.replace(/\islands/g, 'island');
|
|
return tz;
|
|
}; // try our best to reconcile the timzone to this given string
|
|
|
|
|
|
const lookupTz = (str, zones) => {
|
|
if (!str) {
|
|
return local;
|
|
}
|
|
|
|
if (typeof str !== 'string') {
|
|
console.error("Timezone must be a string - recieved: '", str, "'\n");
|
|
}
|
|
|
|
let tz = str.trim(); // let split = str.split('/')
|
|
//support long timezones like 'America/Argentina/Rio_Gallegos'
|
|
// if (split.length > 2 && zones.hasOwnProperty(tz) === false) {
|
|
// tz = split[0] + '/' + split[1]
|
|
// }
|
|
|
|
tz = tz.toLowerCase();
|
|
|
|
if (zones.hasOwnProperty(tz) === true) {
|
|
return tz;
|
|
} //lookup more loosely..
|
|
|
|
|
|
tz = normalize$2(tz);
|
|
|
|
if (zones.hasOwnProperty(tz) === true) {
|
|
return tz;
|
|
} //try city-names
|
|
|
|
|
|
if (cities.hasOwnProperty(tz) === true) {
|
|
return cities[tz];
|
|
} // //try to parse '-5h'
|
|
|
|
|
|
if (/[0-9]/.test(tz) === true) {
|
|
let id = parseOffset$2(tz);
|
|
|
|
if (id) {
|
|
return id;
|
|
}
|
|
}
|
|
|
|
throw new Error("Spacetime: Cannot find timezone named: '" + str + "'. Please enter an IANA timezone id.");
|
|
};
|
|
|
|
var find = lookupTz;
|
|
|
|
var fns$1 = createCommonjsModule(function (module, exports) {
|
|
//git:blame @JuliasCaesar https://www.timeanddate.com/date/leapyear.html
|
|
exports.isLeapYear = year => year % 4 === 0 && year % 100 !== 0 || year % 400 === 0; // unsurprisingly-nasty `typeof date` call
|
|
|
|
|
|
exports.isDate = d => Object.prototype.toString.call(d) === '[object Date]' && !isNaN(d.valueOf());
|
|
|
|
exports.isArray = input => Object.prototype.toString.call(input) === '[object Array]';
|
|
|
|
exports.isObject = input => Object.prototype.toString.call(input) === '[object Object]';
|
|
|
|
exports.isBoolean = input => Object.prototype.toString.call(input) === '[object Boolean]';
|
|
|
|
exports.zeroPad = (str, len = 2) => {
|
|
let pad = '0';
|
|
str = str + '';
|
|
return str.length >= len ? str : new Array(len - str.length + 1).join(pad) + str;
|
|
};
|
|
|
|
exports.titleCase = str => {
|
|
if (!str) {
|
|
return '';
|
|
}
|
|
|
|
return str[0].toUpperCase() + str.substr(1);
|
|
};
|
|
|
|
exports.ordinal = i => {
|
|
let j = i % 10;
|
|
let k = i % 100;
|
|
|
|
if (j === 1 && k !== 11) {
|
|
return i + 'st';
|
|
}
|
|
|
|
if (j === 2 && k !== 12) {
|
|
return i + 'nd';
|
|
}
|
|
|
|
if (j === 3 && k !== 13) {
|
|
return i + 'rd';
|
|
}
|
|
|
|
return i + 'th';
|
|
}; //strip 'st' off '1st'..
|
|
|
|
|
|
exports.toCardinal = str => {
|
|
str = String(str);
|
|
str = str.replace(/([0-9])(st|nd|rd|th)$/i, '$1');
|
|
return parseInt(str, 10);
|
|
}; //used mostly for cleanup of unit names, like 'months'
|
|
|
|
|
|
exports.normalize = (str = '') => {
|
|
str = str.toLowerCase().trim();
|
|
str = str.replace(/ies$/, 'y'); //'centuries'
|
|
|
|
str = str.replace(/s$/, '');
|
|
str = str.replace(/-/g, '');
|
|
|
|
if (str === 'day' || str === 'days') {
|
|
return 'date';
|
|
}
|
|
|
|
if (str === 'min' || str === 'mins') {
|
|
return 'minute';
|
|
}
|
|
|
|
return str;
|
|
};
|
|
|
|
exports.getEpoch = tmp => {
|
|
//support epoch
|
|
if (typeof tmp === 'number') {
|
|
return tmp;
|
|
} //suport date objects
|
|
|
|
|
|
if (exports.isDate(tmp)) {
|
|
return tmp.getTime();
|
|
}
|
|
|
|
if (tmp.epoch) {
|
|
return tmp.epoch;
|
|
}
|
|
|
|
return null;
|
|
}; //make sure this input is a spacetime obj
|
|
|
|
|
|
exports.beADate = (d, s) => {
|
|
if (exports.isObject(d) === false) {
|
|
return s.clone().set(d);
|
|
}
|
|
|
|
return d;
|
|
};
|
|
|
|
exports.formatTimezone = (offset, delimiter = '') => {
|
|
const sign = offset > 0 ? '+' : '-';
|
|
const absOffset = Math.abs(offset);
|
|
const hours = exports.zeroPad(parseInt('' + absOffset, 10));
|
|
const minutes = exports.zeroPad(absOffset % 1 * 60);
|
|
return `${sign}${hours}${delimiter}${minutes}`;
|
|
};
|
|
});
|
|
fns$1.isLeapYear;
|
|
fns$1.isDate;
|
|
fns$1.isArray;
|
|
fns$1.isObject;
|
|
fns$1.isBoolean;
|
|
fns$1.zeroPad;
|
|
fns$1.titleCase;
|
|
fns$1.ordinal;
|
|
fns$1.toCardinal;
|
|
fns$1.normalize;
|
|
fns$1.getEpoch;
|
|
fns$1.beADate;
|
|
fns$1.formatTimezone;
|
|
|
|
const defaults$1 = {
|
|
year: new Date().getFullYear(),
|
|
month: 0,
|
|
date: 1
|
|
}; //support [2016, 03, 01] format
|
|
|
|
const parseArray$1 = (s, arr, today) => {
|
|
if (arr.length === 0) {
|
|
return s;
|
|
}
|
|
|
|
let order = ['year', 'month', 'date', 'hour', 'minute', 'second', 'millisecond'];
|
|
|
|
for (let i = 0; i < order.length; i++) {
|
|
let num = arr[i] || today[order[i]] || defaults$1[order[i]] || 0;
|
|
s = s[order[i]](num);
|
|
}
|
|
|
|
return s;
|
|
}; //support {year:2016, month:3} format
|
|
|
|
|
|
const parseObject$1 = (s, obj, today) => {
|
|
// if obj is empty, do nothing
|
|
if (Object.keys(obj).length === 0) {
|
|
return s;
|
|
}
|
|
|
|
obj = Object.assign({}, defaults$1, today, obj);
|
|
let keys = Object.keys(obj);
|
|
|
|
for (let i = 0; i < keys.length; i++) {
|
|
let unit = keys[i]; //make sure we have this method
|
|
|
|
if (s[unit] === undefined || typeof s[unit] !== 'function') {
|
|
continue;
|
|
} //make sure the value is a number
|
|
|
|
|
|
if (obj[unit] === null || obj[unit] === undefined || obj[unit] === '') {
|
|
continue;
|
|
}
|
|
|
|
let num = obj[unit] || today[unit] || defaults$1[unit] || 0;
|
|
s = s[unit](num);
|
|
}
|
|
|
|
return s;
|
|
}; // this may seem like an arbitrary number, but it's 'within jan 1970'
|
|
// this is only really ambiguous until 2054 or so
|
|
|
|
|
|
const parseNumber$1 = function (s, input) {
|
|
const minimumEpoch = 2500000000; // if the given epoch is really small, they've probably given seconds and not milliseconds
|
|
// anything below this number is likely (but not necessarily) a mistaken input.
|
|
|
|
if (input > 0 && input < minimumEpoch && s.silent === false) {
|
|
console.warn(' - Warning: You are setting the date to January 1970.');
|
|
console.warn(' - did input seconds instead of milliseconds?');
|
|
}
|
|
|
|
s.epoch = input;
|
|
return s;
|
|
};
|
|
|
|
var helpers = {
|
|
parseArray: parseArray$1,
|
|
parseObject: parseObject$1,
|
|
parseNumber: parseNumber$1
|
|
};
|
|
|
|
// pull in 'today' data for the baseline moment
|
|
const getNow = function (s) {
|
|
s.epoch = Date.now();
|
|
Object.keys(s._today || {}).forEach(k => {
|
|
if (typeof s[k] === 'function') {
|
|
s = s[k](s._today[k]);
|
|
}
|
|
});
|
|
return s;
|
|
};
|
|
|
|
const dates = {
|
|
now: s => {
|
|
return getNow(s);
|
|
},
|
|
today: s => {
|
|
return getNow(s);
|
|
},
|
|
tonight: s => {
|
|
s = getNow(s);
|
|
s = s.hour(18); //6pm
|
|
|
|
return s;
|
|
},
|
|
tomorrow: s => {
|
|
s = getNow(s);
|
|
s = s.add(1, 'day');
|
|
s = s.startOf('day');
|
|
return s;
|
|
},
|
|
yesterday: s => {
|
|
s = getNow(s);
|
|
s = s.subtract(1, 'day');
|
|
s = s.startOf('day');
|
|
return s;
|
|
},
|
|
christmas: s => {
|
|
let year = getNow(s).year();
|
|
s = s.set([year, 11, 25, 18, 0, 0]); // Dec 25
|
|
|
|
return s;
|
|
},
|
|
'new years': s => {
|
|
let year = getNow(s).year();
|
|
s = s.set([year, 11, 31, 18, 0, 0]); // Dec 31
|
|
|
|
return s;
|
|
}
|
|
};
|
|
dates['new years eve'] = dates['new years'];
|
|
var namedDates$1 = dates;
|
|
|
|
//little cleanup..
|
|
const normalize$1 = function (str) {
|
|
// remove all day-names
|
|
str = str.replace(/\b(mon|tues?|wed|wednes|thur?s?|fri|sat|satur|sun)(day)?\b/i, ''); //remove ordinal ending
|
|
|
|
str = str.replace(/([0-9])(th|rd|st|nd)/, '$1');
|
|
str = str.replace(/,/g, '');
|
|
str = str.replace(/ +/g, ' ').trim();
|
|
return str;
|
|
};
|
|
|
|
var normalize_1 = normalize$1;
|
|
|
|
let o = {
|
|
millisecond: 1
|
|
};
|
|
o.second = 1000;
|
|
o.minute = 60000;
|
|
o.hour = 3.6e6; // dst is supported post-hoc
|
|
|
|
o.day = 8.64e7; //
|
|
|
|
o.date = o.day;
|
|
o.month = 8.64e7 * 29.5; //(average)
|
|
|
|
o.week = 6.048e8;
|
|
o.year = 3.154e10; // leap-years are supported post-hoc
|
|
//add plurals
|
|
|
|
Object.keys(o).forEach(k => {
|
|
o[k + 's'] = o[k];
|
|
});
|
|
var milliseconds = o;
|
|
|
|
var ms = milliseconds;
|
|
|
|
const walk = (s, n, fn, unit, previous) => {
|
|
let current = s.d[fn]();
|
|
|
|
if (current === n) {
|
|
return; //already there
|
|
}
|
|
|
|
let startUnit = previous === null ? null : s.d[previous]();
|
|
let original = s.epoch; //try to get it as close as we can
|
|
|
|
let diff = n - current;
|
|
s.epoch += ms[unit] * diff; //DST edge-case: if we are going many days, be a little conservative
|
|
// console.log(unit, diff)
|
|
|
|
if (unit === 'day') {
|
|
// s.epoch -= ms.minute
|
|
//but don't push it over a month
|
|
if (Math.abs(diff) > 28 && n < 28) {
|
|
s.epoch += ms.hour;
|
|
}
|
|
} // 1st time: oops, did we change previous unit? revert it.
|
|
|
|
|
|
if (previous !== null && startUnit !== s.d[previous]()) {
|
|
// console.warn('spacetime warning: missed setting ' + unit)
|
|
s.epoch = original; // s.epoch += ms[unit] * diff * 0.89 // maybe try and make it close...?
|
|
} //repair it if we've gone too far or something
|
|
//(go by half-steps, just in case)
|
|
|
|
|
|
const halfStep = ms[unit] / 2;
|
|
|
|
while (s.d[fn]() < n) {
|
|
s.epoch += halfStep;
|
|
}
|
|
|
|
while (s.d[fn]() > n) {
|
|
s.epoch -= halfStep;
|
|
} // 2nd time: did we change previous unit? revert it.
|
|
|
|
|
|
if (previous !== null && startUnit !== s.d[previous]()) {
|
|
// console.warn('spacetime warning: missed setting ' + unit)
|
|
s.epoch = original;
|
|
}
|
|
}; //find the desired date by a increment/check while loop
|
|
|
|
|
|
const units$3 = {
|
|
year: {
|
|
valid: n => n > -4000 && n < 4000,
|
|
walkTo: (s, n) => walk(s, n, 'getFullYear', 'year', null)
|
|
},
|
|
month: {
|
|
valid: n => n >= 0 && n <= 11,
|
|
walkTo: (s, n) => {
|
|
let d = s.d;
|
|
let current = d.getMonth();
|
|
let original = s.epoch;
|
|
let startUnit = d.getFullYear();
|
|
|
|
if (current === n) {
|
|
return;
|
|
} //try to get it as close as we can..
|
|
|
|
|
|
let diff = n - current;
|
|
s.epoch += ms.day * (diff * 28); //special case
|
|
//oops, did we change the year? revert it.
|
|
|
|
if (startUnit !== s.d.getFullYear()) {
|
|
s.epoch = original;
|
|
} //increment by day
|
|
|
|
|
|
while (s.d.getMonth() < n) {
|
|
s.epoch += ms.day;
|
|
}
|
|
|
|
while (s.d.getMonth() > n) {
|
|
s.epoch -= ms.day;
|
|
}
|
|
}
|
|
},
|
|
date: {
|
|
valid: n => n > 0 && n <= 31,
|
|
walkTo: (s, n) => walk(s, n, 'getDate', 'day', 'getMonth')
|
|
},
|
|
hour: {
|
|
valid: n => n >= 0 && n < 24,
|
|
walkTo: (s, n) => walk(s, n, 'getHours', 'hour', 'getDate')
|
|
},
|
|
minute: {
|
|
valid: n => n >= 0 && n < 60,
|
|
walkTo: (s, n) => walk(s, n, 'getMinutes', 'minute', 'getHours')
|
|
},
|
|
second: {
|
|
valid: n => n >= 0 && n < 60,
|
|
walkTo: (s, n) => {
|
|
//do this one directly
|
|
s.epoch = s.seconds(n).epoch;
|
|
}
|
|
},
|
|
millisecond: {
|
|
valid: n => n >= 0 && n < 1000,
|
|
walkTo: (s, n) => {
|
|
//do this one directly
|
|
s.epoch = s.milliseconds(n).epoch;
|
|
}
|
|
}
|
|
};
|
|
|
|
const walkTo$1 = (s, wants) => {
|
|
let keys = Object.keys(units$3);
|
|
let old = s.clone();
|
|
|
|
for (let i = 0; i < keys.length; i++) {
|
|
let k = keys[i];
|
|
let n = wants[k];
|
|
|
|
if (n === undefined) {
|
|
n = old[k]();
|
|
}
|
|
|
|
if (typeof n === 'string') {
|
|
n = parseInt(n, 10);
|
|
} //make-sure it's valid
|
|
|
|
|
|
if (!units$3[k].valid(n)) {
|
|
s.epoch = null;
|
|
|
|
if (s.silent === false) {
|
|
console.warn('invalid ' + k + ': ' + n);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
units$3[k].walkTo(s, n);
|
|
}
|
|
|
|
return;
|
|
};
|
|
|
|
var walk_1 = walkTo$1;
|
|
|
|
const monthLengths = [31, // January - 31 days
|
|
28, // February - 28 days in a common year and 29 days in leap years
|
|
31, // March - 31 days
|
|
30, // April - 30 days
|
|
31, // May - 31 days
|
|
30, // June - 30 days
|
|
31, // July - 31 days
|
|
31, // August - 31 days
|
|
30, // September - 30 days
|
|
31, // October - 31 days
|
|
30, // November - 30 days
|
|
31 // December - 31 days
|
|
];
|
|
var monthLengths_1 = monthLengths; // 28 - feb
|
|
|
|
let shortMonths = ['jan', 'feb', 'mar', 'apr', 'may', 'jun', 'jul', 'aug', 'sept', 'oct', 'nov', 'dec'];
|
|
let longMonths = ['january', 'february', 'march', 'april', 'may', 'june', 'july', 'august', 'september', 'october', 'november', 'december'];
|
|
|
|
function buildMapping() {
|
|
const obj = {
|
|
sep: 8 //support this format
|
|
|
|
};
|
|
|
|
for (let i = 0; i < shortMonths.length; i++) {
|
|
obj[shortMonths[i]] = i;
|
|
}
|
|
|
|
for (let i = 0; i < longMonths.length; i++) {
|
|
obj[longMonths[i]] = i;
|
|
}
|
|
|
|
return obj;
|
|
}
|
|
|
|
var months$2 = {
|
|
short: () => shortMonths,
|
|
long: () => longMonths,
|
|
mapping: () => buildMapping(),
|
|
set: i18n => {
|
|
shortMonths = i18n.short || shortMonths;
|
|
longMonths = i18n.long || longMonths;
|
|
}
|
|
};
|
|
|
|
//pull-apart ISO offsets, like "+0100"
|
|
const parseOffset$1 = (s, offset) => {
|
|
if (!offset) {
|
|
return s;
|
|
} //this is a fancy-move
|
|
|
|
|
|
if (offset === 'Z' || offset === 'z') {
|
|
offset = '+0000';
|
|
} // according to ISO8601, tz could be hh:mm, hhmm or hh
|
|
// so need few more steps before the calculation.
|
|
|
|
|
|
let num = 0; // for (+-)hh:mm
|
|
|
|
if (/^[\+-]?[0-9]{2}:[0-9]{2}$/.test(offset)) {
|
|
//support "+01:00"
|
|
if (/:00/.test(offset) === true) {
|
|
offset = offset.replace(/:00/, '');
|
|
} //support "+01:30"
|
|
|
|
|
|
if (/:30/.test(offset) === true) {
|
|
offset = offset.replace(/:30/, '.5');
|
|
}
|
|
} // for (+-)hhmm
|
|
|
|
|
|
if (/^[\+-]?[0-9]{4}$/.test(offset)) {
|
|
offset = offset.replace(/30$/, '.5');
|
|
}
|
|
|
|
num = parseFloat(offset); //divide by 100 or 10 - , "+0100", "+01"
|
|
|
|
if (Math.abs(num) > 100) {
|
|
num = num / 100;
|
|
} //okay, try to match it to a utc timezone
|
|
//remember - this is opposite! a -5 offset maps to Etc/GMT+5 ¯\_(:/)_/¯
|
|
//https://askubuntu.com/questions/519550/why-is-the-8-timezone-called-gmt-8-in-the-filesystem
|
|
|
|
|
|
num *= -1;
|
|
|
|
if (num >= 0) {
|
|
num = '+' + num;
|
|
}
|
|
|
|
let tz = 'etc/gmt' + num;
|
|
let zones = s.timezones;
|
|
|
|
if (zones[tz]) {
|
|
// log a warning if we're over-writing a given timezone?
|
|
// console.log('changing timezone to: ' + tz)
|
|
s.tz = tz;
|
|
}
|
|
|
|
return s;
|
|
};
|
|
|
|
var parseOffset_1 = parseOffset$1;
|
|
|
|
const parseTime$4 = (s, str = '') => {
|
|
// remove all whitespace
|
|
str = str.replace(/^\s+/, '').toLowerCase(); //formal time format - 04:30.23
|
|
|
|
let arr = str.match(/([0-9]{1,2}):([0-9]{1,2}):?([0-9]{1,2})?[:\.]?([0-9]{1,4})?/);
|
|
|
|
if (arr !== null) {
|
|
//validate it a little
|
|
let h = Number(arr[1]);
|
|
|
|
if (h < 0 || h > 24) {
|
|
return s.startOf('day');
|
|
}
|
|
|
|
let m = Number(arr[2]); //don't accept '5:3pm'
|
|
|
|
if (arr[2].length < 2 || m < 0 || m > 59) {
|
|
return s.startOf('day');
|
|
}
|
|
|
|
if (arr[4] > 999) {
|
|
// fix overflow issue with milliseconds, if input is longer than standard (e.g. 2017-08-06T09:00:00.123456Z)
|
|
arr[4] = parseInt(`${arr[4]}`.substring(0, 3), 10);
|
|
}
|
|
|
|
s = s.hour(h);
|
|
s = s.minute(m);
|
|
s = s.seconds(arr[3] || 0);
|
|
s = s.millisecond(arr[4] || 0); //parse-out am/pm
|
|
|
|
let ampm = str.match(/[\b0-9] ?(am|pm)\b/);
|
|
|
|
if (ampm !== null && ampm[1]) {
|
|
s = s.ampm(ampm[1]);
|
|
}
|
|
|
|
return s;
|
|
} //try an informal form - 5pm (no minutes)
|
|
|
|
|
|
arr = str.match(/([0-9]+) ?(am|pm)/);
|
|
|
|
if (arr !== null && arr[1]) {
|
|
let h = Number(arr[1]); //validate it a little..
|
|
|
|
if (h > 12 || h < 1) {
|
|
return s.startOf('day');
|
|
}
|
|
|
|
s = s.hour(arr[1] || 0);
|
|
s = s.ampm(arr[2]);
|
|
s = s.startOf('hour');
|
|
return s;
|
|
} //no time info found, use start-of-day
|
|
|
|
|
|
s = s.startOf('day');
|
|
return s;
|
|
};
|
|
|
|
var parseTime_1 = parseTime$4;
|
|
|
|
var monthLength = monthLengths_1;
|
|
|
|
var fns = fns$1;
|
|
|
|
var months$1 = months$2;
|
|
|
|
var require$$2$3 = parseOffset_1;
|
|
|
|
var require$$3$2 = parseTime_1;
|
|
|
|
const isLeapYear$2 = fns.isLeapYear;
|
|
const months = months$1.mapping(); //given a month, return whether day number exists in it
|
|
|
|
const validate$5 = obj => {
|
|
//invalid values
|
|
if (monthLength.hasOwnProperty(obj.month) !== true) {
|
|
return false;
|
|
} //support leap-year in february
|
|
|
|
|
|
if (obj.month === 1) {
|
|
if (isLeapYear$2(obj.year) && obj.date <= 29) {
|
|
return true;
|
|
} else {
|
|
return obj.date <= 28;
|
|
}
|
|
} //is this date too-big for this month?
|
|
|
|
|
|
let max = monthLength[obj.month] || 0;
|
|
|
|
if (obj.date <= max) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
const parseYear$4 = (str = '', today) => {
|
|
str = str.trim(); // parse '86 shorthand
|
|
|
|
if (/^'[0-9][0-9]$/.test(str) === true) {
|
|
let num = Number(str.replace(/'/, ''));
|
|
|
|
if (num > 50) {
|
|
return 1900 + num;
|
|
}
|
|
|
|
return 2000 + num;
|
|
}
|
|
|
|
let year = parseInt(str, 10); // use a given year from options.today
|
|
|
|
if (!year && today) {
|
|
year = today.year;
|
|
} // fallback to this year
|
|
|
|
|
|
year = year || new Date().getFullYear();
|
|
return year;
|
|
};
|
|
|
|
const parseMonth$4 = function (str) {
|
|
str = str.toLowerCase().trim();
|
|
return months[str];
|
|
};
|
|
|
|
const parsers$1 = {
|
|
parseOffset: require$$2$3,
|
|
parseTime: require$$3$2,
|
|
parseYear: parseYear$4,
|
|
parseMonth: parseMonth$4,
|
|
validate: validate$5
|
|
};
|
|
var _parsers = parsers$1;
|
|
|
|
var walkTo = walk_1;
|
|
|
|
var require$$0$3 = _parsers;
|
|
|
|
const {
|
|
validate: validate$4,
|
|
parseTime: parseTime$3,
|
|
parseYear: parseYear$3,
|
|
parseMonth: parseMonth$3,
|
|
parseOffset
|
|
} = require$$0$3;
|
|
var _01Ymd = [// =====
|
|
// y-m-d
|
|
// =====
|
|
//iso-this 1998-05-30T22:00:00:000Z, iso-that 2017-04-03T08:00:00-0700
|
|
{
|
|
reg: /^(\-?0?0?[0-9]{3,4})-([0-9]{1,2})-([0-9]{1,2})[T| ]([0-9.:]+)(Z|[0-9\-\+:]+)?$/i,
|
|
parse: (s, m) => {
|
|
let obj = {
|
|
year: m[1],
|
|
month: parseInt(m[2], 10) - 1,
|
|
date: m[3]
|
|
};
|
|
|
|
if (validate$4(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
parseOffset(s, m[5]);
|
|
walkTo(s, obj);
|
|
s = parseTime$3(s, m[4]);
|
|
return s;
|
|
}
|
|
}, //short-iso "2015-03-25" or "2015/03/25" or "2015/03/25 12:26:14 PM"
|
|
{
|
|
reg: /^([0-9]{4})[\-\/\. ]([0-9]{1,2})[\-\/\. ]([0-9]{1,2})( [0-9]{1,2}(:[0-9]{0,2})?(:[0-9]{0,3})? ?(am|pm)?)?$/i,
|
|
parse: (s, m) => {
|
|
let obj = {
|
|
year: m[1],
|
|
month: parseInt(m[2], 10) - 1,
|
|
date: parseInt(m[3], 10)
|
|
};
|
|
|
|
if (obj.month >= 12) {
|
|
//support yyyy/dd/mm (weird, but ok)
|
|
obj.date = parseInt(m[2], 10);
|
|
obj.month = parseInt(m[3], 10) - 1;
|
|
}
|
|
|
|
if (validate$4(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime$3(s, m[4]);
|
|
return s;
|
|
}
|
|
}, //text-month "2015-feb-25"
|
|
{
|
|
reg: /^([0-9]{4})[\-\/\. ]([a-z]+)[\-\/\. ]([0-9]{1,2})( [0-9]{1,2}(:[0-9]{0,2})?(:[0-9]{0,3})? ?(am|pm)?)?$/i,
|
|
parse: (s, m) => {
|
|
let obj = {
|
|
year: parseYear$3(m[1], s._today),
|
|
month: parseMonth$3(m[2]),
|
|
date: fns.toCardinal(m[3] || '')
|
|
};
|
|
|
|
if (validate$4(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime$3(s, m[4]);
|
|
return s;
|
|
}
|
|
}];
|
|
|
|
const {
|
|
validate: validate$3,
|
|
parseTime: parseTime$2,
|
|
parseYear: parseYear$2,
|
|
parseMonth: parseMonth$2
|
|
} = require$$0$3;
|
|
var _02Mdy = [// =====
|
|
// m-d-y
|
|
// =====
|
|
//mm/dd/yyyy - uk/canada "6/28/2019, 12:26:14 PM"
|
|
{
|
|
reg: /^([0-9]{1,2})[\-\/.]([0-9]{1,2})[\-\/.]?([0-9]{4})?( [0-9]{1,2}:[0-9]{2}:?[0-9]{0,2}? ?(am|pm|gmt))?$/i,
|
|
parse: (s, arr) => {
|
|
let month = parseInt(arr[1], 10) - 1;
|
|
let date = parseInt(arr[2], 10); //support dd/mm/yyy
|
|
|
|
if (s.british || month >= 12) {
|
|
date = parseInt(arr[1], 10);
|
|
month = parseInt(arr[2], 10) - 1;
|
|
}
|
|
|
|
let obj = {
|
|
date,
|
|
month,
|
|
year: parseYear$2(arr[3], s._today) || new Date().getFullYear()
|
|
};
|
|
|
|
if (validate$3(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime$2(s, arr[4]);
|
|
return s;
|
|
}
|
|
}, //alt short format - "feb-25-2015"
|
|
{
|
|
reg: /^([a-z]+)[\-\/\. ]([0-9]{1,2})[\-\/\. ]?([0-9]{4}|'[0-9]{2})?( [0-9]{1,2}(:[0-9]{0,2})?(:[0-9]{0,3})? ?(am|pm)?)?$/i,
|
|
parse: (s, arr) => {
|
|
let obj = {
|
|
year: parseYear$2(arr[3], s._today),
|
|
month: parseMonth$2(arr[1]),
|
|
date: fns.toCardinal(arr[2] || '')
|
|
};
|
|
|
|
if (validate$3(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime$2(s, arr[4]);
|
|
return s;
|
|
}
|
|
}, //Long "Mar 25 2015"
|
|
//February 22, 2017 15:30:00
|
|
{
|
|
reg: /^([a-z]+) ([0-9]{1,2})( [0-9]{4})?( ([0-9:]+( ?am| ?pm| ?gmt)?))?$/i,
|
|
parse: (s, arr) => {
|
|
let obj = {
|
|
year: parseYear$2(arr[3], s._today),
|
|
month: parseMonth$2(arr[1]),
|
|
date: fns.toCardinal(arr[2] || '')
|
|
};
|
|
|
|
if (validate$3(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime$2(s, arr[4]);
|
|
return s;
|
|
}
|
|
}, // 'Sun Mar 14 15:09:48 +0000 2021'
|
|
{
|
|
reg: /^([a-z]+) ([0-9]{1,2})( [0-9:]+)?( \+[0-9]{4})?( [0-9]{4})?$/i,
|
|
parse: (s, arr) => {
|
|
let obj = {
|
|
year: parseYear$2(arr[5], s._today),
|
|
month: parseMonth$2(arr[1]),
|
|
date: fns.toCardinal(arr[2] || '')
|
|
};
|
|
|
|
if (validate$3(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime$2(s, arr[3]);
|
|
return s;
|
|
}
|
|
}];
|
|
|
|
const {
|
|
validate: validate$2,
|
|
parseTime: parseTime$1,
|
|
parseYear: parseYear$1,
|
|
parseMonth: parseMonth$1
|
|
} = require$$0$3;
|
|
var _03Dmy = [// =====
|
|
// d-m-y
|
|
// =====
|
|
//common british format - "25-feb-2015"
|
|
{
|
|
reg: /^([0-9]{1,2})[\-\/]([a-z]+)[\-\/]?([0-9]{4})?$/i,
|
|
parse: (s, m) => {
|
|
let obj = {
|
|
year: parseYear$1(m[3], s._today),
|
|
month: parseMonth$1(m[2]),
|
|
date: fns.toCardinal(m[1] || '')
|
|
};
|
|
|
|
if (validate$2(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime$1(s, m[4]);
|
|
return s;
|
|
}
|
|
}, // "25 Mar 2015"
|
|
{
|
|
reg: /^([0-9]{1,2})( [a-z]+)( [0-9]{4}| '[0-9]{2})? ?([0-9]{1,2}:[0-9]{2}:?[0-9]{0,2}? ?(am|pm|gmt))?$/i,
|
|
parse: (s, m) => {
|
|
let obj = {
|
|
year: parseYear$1(m[3], s._today),
|
|
month: parseMonth$1(m[2]),
|
|
date: fns.toCardinal(m[1])
|
|
};
|
|
|
|
if (!obj.month || validate$2(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime$1(s, m[4]);
|
|
return s;
|
|
}
|
|
}, // 01-jan-2020
|
|
{
|
|
reg: /^([0-9]{1,2})[\. -/]([a-z]+)[\. -/]([0-9]{4})?( [0-9]{1,2}(:[0-9]{0,2})?(:[0-9]{0,3})? ?(am|pm)?)?$/i,
|
|
parse: (s, m) => {
|
|
let obj = {
|
|
date: Number(m[1]),
|
|
month: parseMonth$1(m[2]),
|
|
year: Number(m[3])
|
|
};
|
|
|
|
if (validate$2(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = s.startOf('day');
|
|
s = parseTime$1(s, m[4]);
|
|
return s;
|
|
}
|
|
}];
|
|
|
|
const {
|
|
validate: validate$1,
|
|
parseTime,
|
|
parseYear,
|
|
parseMonth
|
|
} = require$$0$3;
|
|
var _04Misc = [// =====
|
|
// no dates
|
|
// =====
|
|
// '2012-06' month-only
|
|
{
|
|
reg: /^([0-9]{4})[\-\/]([0-9]{2})$/i,
|
|
parse: (s, m) => {
|
|
let obj = {
|
|
year: m[1],
|
|
month: parseInt(m[2], 10) - 1,
|
|
date: 1
|
|
};
|
|
|
|
if (validate$1(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime(s, m[4]);
|
|
return s;
|
|
}
|
|
}, //February 2017 (implied date)
|
|
{
|
|
reg: /^([a-z]+) ([0-9]{4})$/i,
|
|
parse: (s, arr) => {
|
|
let obj = {
|
|
year: parseYear(arr[2], s._today),
|
|
month: parseMonth(arr[1]),
|
|
date: s._today.date || 1
|
|
};
|
|
|
|
if (validate$1(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime(s, arr[4]);
|
|
return s;
|
|
}
|
|
}, {
|
|
// 'q2 2002'
|
|
reg: /^(q[0-9])( of)?( [0-9]{4})?/i,
|
|
parse: (s, arr) => {
|
|
let quarter = arr[1] || '';
|
|
s = s.quarter(quarter);
|
|
let year = arr[3] || '';
|
|
|
|
if (year) {
|
|
year = year.trim();
|
|
s = s.year(year);
|
|
}
|
|
|
|
return s;
|
|
}
|
|
}, {
|
|
// 'summer 2002'
|
|
reg: /^(spring|summer|winter|fall|autumn)( of)?( [0-9]{4})?/i,
|
|
parse: (s, arr) => {
|
|
let season = arr[1] || '';
|
|
s = s.season(season);
|
|
let year = arr[3] || '';
|
|
|
|
if (year) {
|
|
year = year.trim();
|
|
s = s.year(year);
|
|
}
|
|
|
|
return s;
|
|
}
|
|
}, {
|
|
// '200bc'
|
|
reg: /^[0-9,]+ ?b\.?c\.?$/i,
|
|
parse: (s, arr) => {
|
|
let str = arr[0] || ''; //make year-negative
|
|
|
|
str = str.replace(/^([0-9,]+) ?b\.?c\.?$/i, '-$1');
|
|
let d = new Date();
|
|
let obj = {
|
|
year: parseInt(str.trim(), 10),
|
|
month: d.getMonth(),
|
|
date: d.getDate()
|
|
};
|
|
|
|
if (validate$1(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime(s);
|
|
return s;
|
|
}
|
|
}, {
|
|
// '200ad'
|
|
reg: /^[0-9,]+ ?(a\.?d\.?|c\.?e\.?)$/i,
|
|
parse: (s, arr) => {
|
|
let str = arr[0] || ''; //remove commas
|
|
|
|
str = str.replace(/,/g, '');
|
|
let d = new Date();
|
|
let obj = {
|
|
year: parseInt(str.trim(), 10),
|
|
month: d.getMonth(),
|
|
date: d.getDate()
|
|
};
|
|
|
|
if (validate$1(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime(s);
|
|
return s;
|
|
}
|
|
}, {
|
|
// '1992'
|
|
reg: /^[0-9]{4}( ?a\.?d\.?)?$/i,
|
|
parse: (s, arr) => {
|
|
let today = s._today; // using today's date, but a new month is awkward.
|
|
|
|
if (today.month && !today.date) {
|
|
today.date = 1;
|
|
}
|
|
|
|
let d = new Date();
|
|
let obj = {
|
|
year: parseYear(arr[0], today),
|
|
month: today.month || d.getMonth(),
|
|
date: today.date || d.getDate()
|
|
};
|
|
|
|
if (validate$1(obj) === false) {
|
|
s.epoch = null;
|
|
return s;
|
|
}
|
|
|
|
walkTo(s, obj);
|
|
s = parseTime(s);
|
|
return s;
|
|
}
|
|
}];
|
|
|
|
var require$$0$2 = _01Ymd;
|
|
|
|
var require$$1$2 = _02Mdy;
|
|
|
|
var require$$2$2 = _03Dmy;
|
|
|
|
var require$$3$1 = _04Misc;
|
|
|
|
var formats = [].concat(require$$0$2, require$$1$2, require$$2$2, require$$3$1);
|
|
|
|
var parsers = formats;
|
|
|
|
const parseString$1 = function (s, input, givenTz) {
|
|
// let parsers = s.parsers || []
|
|
//try each text-parse template, use the first good result
|
|
for (let i = 0; i < parsers.length; i++) {
|
|
let m = input.match(parsers[i].reg);
|
|
|
|
if (m) {
|
|
// console.log(parsers[i].reg)
|
|
let res = parsers[i].parse(s, m, givenTz);
|
|
|
|
if (res !== null && res.isValid()) {
|
|
return res;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (s.silent === false) {
|
|
console.warn("Warning: couldn't parse date-string: '" + input + "'");
|
|
}
|
|
|
|
s.epoch = null;
|
|
return s;
|
|
};
|
|
|
|
var parse = parseString$1;
|
|
|
|
var require$$0$1 = helpers;
|
|
|
|
var namedDates = namedDates$1;
|
|
|
|
var normalize = normalize_1;
|
|
|
|
var parseString = parse;
|
|
|
|
const {
|
|
parseArray,
|
|
parseObject,
|
|
parseNumber
|
|
} = require$$0$1; //we have to actually parse these inputs ourselves
|
|
// - can't use built-in js parser ;(
|
|
//=========================================
|
|
// ISO Date "2015-03-25"
|
|
// Short Date "03/25/2015" or "2015/03/25"
|
|
// Long Date "Mar 25 2015" or "25 Mar 2015"
|
|
// Full Date "Wednesday March 25 2015"
|
|
//=========================================
|
|
|
|
const defaults = {
|
|
year: new Date().getFullYear(),
|
|
month: 0,
|
|
date: 1
|
|
}; //find the epoch from different input styles
|
|
|
|
const parseInput = (s, input) => {
|
|
let today = s._today || defaults; //if we've been given a epoch number, it's easy
|
|
|
|
if (typeof input === 'number') {
|
|
return parseNumber(s, input);
|
|
} //set tmp time
|
|
|
|
|
|
s.epoch = Date.now(); // overwrite tmp time with 'today' value, if exists
|
|
|
|
if (s._today && fns.isObject(s._today) && Object.keys(s._today).length > 0) {
|
|
let res = parseObject(s, today, defaults);
|
|
|
|
if (res.isValid()) {
|
|
s.epoch = res.epoch;
|
|
}
|
|
} // null input means 'now'
|
|
|
|
|
|
if (input === null || input === undefined || input === '') {
|
|
return s; //k, we're good.
|
|
} //support input of Date() object
|
|
|
|
|
|
if (fns.isDate(input) === true) {
|
|
s.epoch = input.getTime();
|
|
return s;
|
|
} //support [2016, 03, 01] format
|
|
|
|
|
|
if (fns.isArray(input) === true) {
|
|
s = parseArray(s, input, today);
|
|
return s;
|
|
} //support {year:2016, month:3} format
|
|
|
|
|
|
if (fns.isObject(input) === true) {
|
|
//support spacetime object as input
|
|
if (input.epoch) {
|
|
s.epoch = input.epoch;
|
|
s.tz = input.tz;
|
|
return s;
|
|
}
|
|
|
|
s = parseObject(s, input, today);
|
|
return s;
|
|
} //input as a string..
|
|
|
|
|
|
if (typeof input !== 'string') {
|
|
return s;
|
|
} //little cleanup..
|
|
|
|
|
|
input = normalize(input); //try some known-words, like 'now'
|
|
|
|
if (namedDates.hasOwnProperty(input) === true) {
|
|
s = namedDates[input](s);
|
|
return s;
|
|
} //try each text-parse template, use the first good result
|
|
|
|
|
|
return parseString(s, input);
|
|
};
|
|
|
|
var input = parseInput;
|
|
|
|
let shortDays = ['sun', 'mon', 'tue', 'wed', 'thu', 'fri', 'sat'];
|
|
let longDays = ['sunday', 'monday', 'tuesday', 'wednesday', 'thursday', 'friday', 'saturday'];
|
|
var days$1 = {
|
|
short: () => shortDays,
|
|
long: () => longDays,
|
|
set: i18n => {
|
|
shortDays = i18n.short || shortDays;
|
|
longDays = i18n.long || longDays;
|
|
},
|
|
aliases: {
|
|
mo: 1,
|
|
tu: 2,
|
|
we: 3,
|
|
th: 4,
|
|
fr: 5,
|
|
sa: 6,
|
|
su: 7,
|
|
tues: 2,
|
|
weds: 3,
|
|
wedn: 3,
|
|
thur: 4,
|
|
thurs: 4
|
|
}
|
|
};
|
|
|
|
let titleCaseEnabled = true;
|
|
var caseFormat$1 = {
|
|
useTitleCase: () => titleCaseEnabled,
|
|
set: useTitleCase => {
|
|
titleCaseEnabled = useTitleCase;
|
|
}
|
|
};
|
|
|
|
// it's kind of nuts how involved this is
|
|
// "+01:00", "+0100", or simply "+01"
|
|
|
|
const isoOffset$1 = s => {
|
|
let offset = s.timezone().current.offset;
|
|
return !offset ? 'Z' : fns.formatTimezone(offset, ':');
|
|
};
|
|
|
|
var _offset = isoOffset$1;
|
|
|
|
var days = days$1;
|
|
|
|
var caseFormat = caseFormat$1;
|
|
|
|
var isoOffset = _offset;
|
|
|
|
const applyCaseFormat = str => {
|
|
if (caseFormat.useTitleCase()) {
|
|
return fns.titleCase(str);
|
|
}
|
|
|
|
return str;
|
|
}; // iso-year padding
|
|
|
|
|
|
const padYear = num => {
|
|
if (num >= 0) {
|
|
return fns.zeroPad(num, 4);
|
|
} else {
|
|
num = Math.abs(num);
|
|
return '-' + fns.zeroPad(num, 4);
|
|
}
|
|
};
|
|
|
|
const format$1 = {
|
|
day: s => applyCaseFormat(s.dayName()),
|
|
'day-short': s => applyCaseFormat(days.short()[s.day()]),
|
|
'day-number': s => s.day(),
|
|
'day-ordinal': s => fns.ordinal(s.day()),
|
|
'day-pad': s => fns.zeroPad(s.day()),
|
|
date: s => s.date(),
|
|
'date-ordinal': s => fns.ordinal(s.date()),
|
|
'date-pad': s => fns.zeroPad(s.date()),
|
|
month: s => applyCaseFormat(s.monthName()),
|
|
'month-short': s => applyCaseFormat(months$1.short()[s.month()]),
|
|
'month-number': s => s.month(),
|
|
'month-ordinal': s => fns.ordinal(s.month()),
|
|
'month-pad': s => fns.zeroPad(s.month()),
|
|
'iso-month': s => fns.zeroPad(s.month() + 1),
|
|
//1-based months
|
|
year: s => {
|
|
let year = s.year();
|
|
|
|
if (year > 0) {
|
|
return year;
|
|
}
|
|
|
|
year = Math.abs(year);
|
|
return year + ' BC';
|
|
},
|
|
'year-short': s => {
|
|
let year = s.year();
|
|
|
|
if (year > 0) {
|
|
return `'${String(s.year()).substr(2, 4)}`;
|
|
}
|
|
|
|
year = Math.abs(year);
|
|
return year + ' BC';
|
|
},
|
|
'iso-year': s => {
|
|
let year = s.year();
|
|
let isNegative = year < 0;
|
|
let str = fns.zeroPad(Math.abs(year), 4); //0-padded
|
|
|
|
if (isNegative) {
|
|
//negative years are for some reason 6-digits ('-00008')
|
|
str = fns.zeroPad(str, 6);
|
|
str = '-' + str;
|
|
}
|
|
|
|
return str;
|
|
},
|
|
time: s => s.time(),
|
|
'time-24': s => `${s.hour24()}:${fns.zeroPad(s.minute())}`,
|
|
hour: s => s.hour12(),
|
|
'hour-pad': s => fns.zeroPad(s.hour12()),
|
|
'hour-24': s => s.hour24(),
|
|
'hour-24-pad': s => fns.zeroPad(s.hour24()),
|
|
minute: s => s.minute(),
|
|
'minute-pad': s => fns.zeroPad(s.minute()),
|
|
second: s => s.second(),
|
|
'second-pad': s => fns.zeroPad(s.second()),
|
|
millisecond: s => s.millisecond(),
|
|
'millisecond-pad': s => fns.zeroPad(s.millisecond(), 3),
|
|
ampm: s => s.ampm(),
|
|
quarter: s => 'Q' + s.quarter(),
|
|
season: s => s.season(),
|
|
era: s => s.era(),
|
|
json: s => s.json(),
|
|
timezone: s => s.timezone().name,
|
|
offset: s => isoOffset(s),
|
|
numeric: s => `${s.year()}/${fns.zeroPad(s.month() + 1)}/${fns.zeroPad(s.date())}`,
|
|
// yyyy/mm/dd
|
|
'numeric-us': s => `${fns.zeroPad(s.month() + 1)}/${fns.zeroPad(s.date())}/${s.year()}`,
|
|
// mm/dd/yyyy
|
|
'numeric-uk': s => `${fns.zeroPad(s.date())}/${fns.zeroPad(s.month() + 1)}/${s.year()}`,
|
|
//dd/mm/yyyy
|
|
'mm/dd': s => `${fns.zeroPad(s.month() + 1)}/${fns.zeroPad(s.date())}`,
|
|
//mm/dd
|
|
// ... https://en.wikipedia.org/wiki/ISO_8601 ;(((
|
|
iso: s => {
|
|
let year = s.format('iso-year');
|
|
let month = fns.zeroPad(s.month() + 1); //1-based months
|
|
|
|
let date = fns.zeroPad(s.date());
|
|
let hour = fns.zeroPad(s.h24());
|
|
let minute = fns.zeroPad(s.minute());
|
|
let second = fns.zeroPad(s.second());
|
|
let ms = fns.zeroPad(s.millisecond(), 3);
|
|
let offset = isoOffset(s);
|
|
return `${year}-${month}-${date}T${hour}:${minute}:${second}.${ms}${offset}`; //2018-03-09T08:50:00.000-05:00
|
|
},
|
|
'iso-short': s => {
|
|
let month = fns.zeroPad(s.month() + 1); //1-based months
|
|
|
|
let date = fns.zeroPad(s.date());
|
|
let year = padYear(s.year());
|
|
return `${year}-${month}-${date}`; //2017-02-15
|
|
},
|
|
'iso-utc': s => {
|
|
return new Date(s.epoch).toISOString(); //2017-03-08T19:45:28.367Z
|
|
},
|
|
//i made these up
|
|
nice: s => `${months$1.short()[s.month()]} ${fns.ordinal(s.date())}, ${s.time()}`,
|
|
'nice-24': s => `${months$1.short()[s.month()]} ${fns.ordinal(s.date())}, ${s.hour24()}:${fns.zeroPad(s.minute())}`,
|
|
'nice-year': s => `${months$1.short()[s.month()]} ${fns.ordinal(s.date())}, ${s.year()}`,
|
|
'nice-day': s => `${days.short()[s.day()]} ${applyCaseFormat(months$1.short()[s.month()])} ${fns.ordinal(s.date())}`,
|
|
'nice-full': s => `${s.dayName()} ${applyCaseFormat(s.monthName())} ${fns.ordinal(s.date())}, ${s.time()}`,
|
|
'nice-full-24': s => `${s.dayName()} ${applyCaseFormat(s.monthName())} ${fns.ordinal(s.date())}, ${s.hour24()}:${fns.zeroPad(s.minute())}`
|
|
}; //aliases
|
|
|
|
const aliases = {
|
|
'day-name': 'day',
|
|
'month-name': 'month',
|
|
'iso 8601': 'iso',
|
|
'time-h24': 'time-24',
|
|
'time-12': 'time',
|
|
'time-h12': 'time',
|
|
tz: 'timezone',
|
|
'day-num': 'day-number',
|
|
'month-num': 'month-number',
|
|
'month-iso': 'iso-month',
|
|
'year-iso': 'iso-year',
|
|
'nice-short': 'nice',
|
|
'nice-short-24': 'nice-24',
|
|
mdy: 'numeric-us',
|
|
dmy: 'numeric-uk',
|
|
ymd: 'numeric',
|
|
'yyyy/mm/dd': 'numeric',
|
|
'mm/dd/yyyy': 'numeric-us',
|
|
'dd/mm/yyyy': 'numeric-us',
|
|
'little-endian': 'numeric-uk',
|
|
'big-endian': 'numeric',
|
|
'day-nice': 'nice-day'
|
|
};
|
|
Object.keys(aliases).forEach(k => format$1[k] = format$1[aliases[k]]);
|
|
|
|
const printFormat = (s, str = '') => {
|
|
//don't print anything if it's an invalid date
|
|
if (s.isValid() !== true) {
|
|
return '';
|
|
} //support .format('month')
|
|
|
|
|
|
if (format$1.hasOwnProperty(str)) {
|
|
let out = format$1[str](s) || '';
|
|
|
|
if (str !== 'json') {
|
|
out = String(out);
|
|
|
|
if (str !== 'ampm') {
|
|
out = applyCaseFormat(out);
|
|
}
|
|
}
|
|
|
|
return out;
|
|
} //support '{hour}:{minute}' notation
|
|
|
|
|
|
if (str.indexOf('{') !== -1) {
|
|
let sections = /\{(.+?)\}/g;
|
|
str = str.replace(sections, (_, fmt) => {
|
|
fmt = fmt.toLowerCase().trim();
|
|
|
|
if (format$1.hasOwnProperty(fmt)) {
|
|
let out = String(format$1[fmt](s));
|
|
|
|
if (fmt !== 'ampm') {
|
|
return applyCaseFormat(out);
|
|
}
|
|
|
|
return out;
|
|
}
|
|
|
|
return '';
|
|
});
|
|
return str;
|
|
}
|
|
|
|
return s.format('iso-short');
|
|
};
|
|
|
|
var format_1 = printFormat;
|
|
|
|
const pad = fns.zeroPad;
|
|
const formatTimezone = fns.formatTimezone; //parse this insane unix-time-templating thing, from the 19th century
|
|
//http://unicode.org/reports/tr35/tr35-25.html#Date_Format_Patterns
|
|
//time-symbols we support
|
|
|
|
const mapping = {
|
|
G: s => s.era(),
|
|
GG: s => s.era(),
|
|
GGG: s => s.era(),
|
|
GGGG: s => s.era() === 'AD' ? 'Anno Domini' : 'Before Christ',
|
|
//year
|
|
y: s => s.year(),
|
|
yy: s => {
|
|
//last two chars
|
|
return parseInt(String(s.year()).substr(2, 4), 10);
|
|
},
|
|
yyy: s => s.year(),
|
|
yyyy: s => s.year(),
|
|
yyyyy: s => '0' + s.year(),
|
|
// u: (s) => {},//extended non-gregorian years
|
|
//quarter
|
|
Q: s => s.quarter(),
|
|
QQ: s => s.quarter(),
|
|
QQQ: s => s.quarter(),
|
|
QQQQ: s => s.quarter(),
|
|
//month
|
|
M: s => s.month() + 1,
|
|
MM: s => pad(s.month() + 1),
|
|
MMM: s => s.format('month-short'),
|
|
MMMM: s => s.format('month'),
|
|
//week
|
|
w: s => s.week(),
|
|
ww: s => pad(s.week()),
|
|
//week of month
|
|
// W: (s) => s.week(),
|
|
//date of month
|
|
d: s => s.date(),
|
|
dd: s => pad(s.date()),
|
|
//date of year
|
|
D: s => s.dayOfYear(),
|
|
DD: s => pad(s.dayOfYear()),
|
|
DDD: s => pad(s.dayOfYear(), 3),
|
|
// F: (s) => {},//date of week in month
|
|
// g: (s) => {},//modified julian day
|
|
//day
|
|
E: s => s.format('day-short'),
|
|
EE: s => s.format('day-short'),
|
|
EEE: s => s.format('day-short'),
|
|
EEEE: s => s.format('day'),
|
|
EEEEE: s => s.format('day')[0],
|
|
e: s => s.day(),
|
|
ee: s => s.day(),
|
|
eee: s => s.format('day-short'),
|
|
eeee: s => s.format('day'),
|
|
eeeee: s => s.format('day')[0],
|
|
//am/pm
|
|
a: s => s.ampm().toUpperCase(),
|
|
aa: s => s.ampm().toUpperCase(),
|
|
aaa: s => s.ampm().toUpperCase(),
|
|
aaaa: s => s.ampm().toUpperCase(),
|
|
//hour
|
|
h: s => s.h12(),
|
|
hh: s => pad(s.h12()),
|
|
H: s => s.hour(),
|
|
HH: s => pad(s.hour()),
|
|
// j: (s) => {},//weird hour format
|
|
m: s => s.minute(),
|
|
mm: s => pad(s.minute()),
|
|
s: s => s.second(),
|
|
ss: s => pad(s.second()),
|
|
//milliseconds
|
|
SSS: s => pad(s.millisecond(), 3),
|
|
//milliseconds in the day
|
|
A: s => s.epoch - s.startOf('day').epoch,
|
|
//timezone
|
|
z: s => s.timezone().name,
|
|
zz: s => s.timezone().name,
|
|
zzz: s => s.timezone().name,
|
|
zzzz: s => s.timezone().name,
|
|
Z: s => formatTimezone(s.timezone().current.offset),
|
|
ZZ: s => formatTimezone(s.timezone().current.offset),
|
|
ZZZ: s => formatTimezone(s.timezone().current.offset),
|
|
ZZZZ: s => formatTimezone(s.timezone().current.offset, ':')
|
|
};
|
|
|
|
const addAlias = (char, to, n) => {
|
|
let name = char;
|
|
let toName = to;
|
|
|
|
for (let i = 0; i < n; i += 1) {
|
|
mapping[name] = mapping[toName];
|
|
name += char;
|
|
toName += to;
|
|
}
|
|
};
|
|
|
|
addAlias('q', 'Q', 4);
|
|
addAlias('L', 'M', 4);
|
|
addAlias('Y', 'y', 4);
|
|
addAlias('c', 'e', 4);
|
|
addAlias('k', 'H', 2);
|
|
addAlias('K', 'h', 2);
|
|
addAlias('S', 's', 2);
|
|
addAlias('v', 'z', 4);
|
|
addAlias('V', 'Z', 4); // support unix-style escaping with ' character
|
|
|
|
const escapeChars = function (arr) {
|
|
for (let i = 0; i < arr.length; i += 1) {
|
|
if (arr[i] === `'`) {
|
|
// greedy-search for next apostrophe
|
|
for (let o = i + 1; o < arr.length; o += 1) {
|
|
if (arr[o]) {
|
|
arr[i] += arr[o];
|
|
}
|
|
|
|
if (arr[o] === `'`) {
|
|
arr[o] = null;
|
|
break;
|
|
}
|
|
|
|
arr[o] = null;
|
|
}
|
|
}
|
|
}
|
|
|
|
return arr.filter(ch => ch);
|
|
}; //combine consecutive chars, like 'yyyy' as one.
|
|
|
|
|
|
const combineRepeated = function (arr) {
|
|
for (let i = 0; i < arr.length; i += 1) {
|
|
let c = arr[i]; // greedy-forward
|
|
|
|
for (let o = i + 1; o < arr.length; o += 1) {
|
|
if (arr[o] === c) {
|
|
arr[i] += arr[o];
|
|
arr[o] = null;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
} // '' means one apostrophe
|
|
|
|
|
|
arr = arr.filter(ch => ch);
|
|
arr = arr.map(str => {
|
|
if (str === `''`) {
|
|
str = `'`;
|
|
}
|
|
|
|
return str;
|
|
});
|
|
return arr;
|
|
};
|
|
|
|
const unixFmt$1 = (s, str) => {
|
|
let arr = str.split(''); // support character escaping
|
|
|
|
arr = escapeChars(arr); //combine 'yyyy' as string.
|
|
|
|
arr = combineRepeated(arr);
|
|
return arr.reduce((txt, c) => {
|
|
if (mapping[c] !== undefined) {
|
|
txt += mapping[c](s) || '';
|
|
} else {
|
|
// 'unescape'
|
|
if (/^'.{1,}'$/.test(c)) {
|
|
c = c.replace(/'/g, '');
|
|
}
|
|
|
|
txt += c;
|
|
}
|
|
|
|
return txt;
|
|
}, '');
|
|
};
|
|
|
|
var unixFmt_1 = unixFmt$1;
|
|
|
|
const units$2 = ['year', 'season', 'quarter', 'month', 'week', 'day', 'quarterHour', 'hour', 'minute'];
|
|
|
|
const doUnit = function (s, k) {
|
|
let start = s.clone().startOf(k);
|
|
let end = s.clone().endOf(k);
|
|
let duration = end.epoch - start.epoch;
|
|
let percent = (s.epoch - start.epoch) / duration;
|
|
return parseFloat(percent.toFixed(2));
|
|
}; //how far it is along, from 0-1
|
|
|
|
|
|
const progress$1 = (s, unit) => {
|
|
if (unit) {
|
|
unit = fns.normalize(unit);
|
|
return doUnit(s, unit);
|
|
}
|
|
|
|
let obj = {};
|
|
units$2.forEach(k => {
|
|
obj[k] = doUnit(s, k);
|
|
});
|
|
return obj;
|
|
};
|
|
|
|
var progress_1 = progress$1;
|
|
|
|
const nearest$1 = (s, unit) => {
|
|
//how far have we gone?
|
|
let prog = s.progress();
|
|
unit = fns.normalize(unit); //fix camel-case for this one
|
|
|
|
if (unit === 'quarterhour') {
|
|
unit = 'quarterHour';
|
|
}
|
|
|
|
if (prog[unit] !== undefined) {
|
|
// go forward one?
|
|
if (prog[unit] > 0.5) {
|
|
s = s.add(1, unit);
|
|
} // go to start
|
|
|
|
|
|
s = s.startOf(unit);
|
|
} else if (s.silent === false) {
|
|
console.warn("no known unit '" + unit + "'");
|
|
}
|
|
|
|
return s;
|
|
};
|
|
|
|
var nearest_1 = nearest$1;
|
|
|
|
//increment until dates are the same
|
|
const climb = (a, b, unit) => {
|
|
let i = 0;
|
|
a = a.clone();
|
|
|
|
while (a.isBefore(b)) {
|
|
//do proper, expensive increment to catch all-the-tricks
|
|
a = a.add(1, unit);
|
|
i += 1;
|
|
} //oops, we went too-far..
|
|
|
|
|
|
if (a.isAfter(b, unit)) {
|
|
i -= 1;
|
|
}
|
|
|
|
return i;
|
|
}; // do a thurough +=1 on the unit, until they match
|
|
// for speed-reasons, only used on day, month, week.
|
|
|
|
|
|
const diffOne$1 = (a, b, unit) => {
|
|
if (a.isBefore(b)) {
|
|
return climb(a, b, unit);
|
|
} else {
|
|
return climb(b, a, unit) * -1; //reverse it
|
|
}
|
|
};
|
|
|
|
var one = diffOne$1;
|
|
|
|
var diffOne = one;
|
|
|
|
// 2020 - 2019 may be 1 year, or 0 years
|
|
// - '1 year difference' means 366 days during a leap year
|
|
|
|
const fastYear = (a, b) => {
|
|
let years = b.year() - a.year(); // should we decrement it by 1?
|
|
|
|
a = a.year(b.year());
|
|
|
|
if (a.isAfter(b)) {
|
|
years -= 1;
|
|
}
|
|
|
|
return years;
|
|
}; // use a waterfall-method for computing a diff of any 'pre-knowable' units
|
|
// compute years, then compute months, etc..
|
|
// ... then ms-math for any very-small units
|
|
|
|
|
|
const diff$2 = function (a, b) {
|
|
// an hour is always the same # of milliseconds
|
|
// so these units can be 'pre-calculated'
|
|
let msDiff = b.epoch - a.epoch;
|
|
let obj = {
|
|
milliseconds: msDiff,
|
|
seconds: parseInt(msDiff / 1000, 10)
|
|
};
|
|
obj.minutes = parseInt(obj.seconds / 60, 10);
|
|
obj.hours = parseInt(obj.minutes / 60, 10); //do the year
|
|
|
|
let tmp = a.clone();
|
|
obj.years = fastYear(tmp, b);
|
|
tmp = a.add(obj.years, 'year'); //there's always 12 months in a year...
|
|
|
|
obj.months = obj.years * 12;
|
|
tmp = a.add(obj.months, 'month');
|
|
obj.months += diffOne(tmp, b, 'month'); // there's always atleast 52 weeks in a year..
|
|
// (month * 4) isn't as close
|
|
|
|
obj.weeks = obj.years * 52;
|
|
tmp = a.add(obj.weeks, 'week');
|
|
obj.weeks += diffOne(tmp, b, 'week'); // there's always atleast 7 days in a week
|
|
|
|
obj.days = obj.weeks * 7;
|
|
tmp = a.add(obj.days, 'day');
|
|
obj.days += diffOne(tmp, b, 'day');
|
|
return obj;
|
|
};
|
|
|
|
var waterfall$1 = diff$2;
|
|
|
|
var waterfall = waterfall$1;
|
|
|
|
const reverseDiff = function (obj) {
|
|
Object.keys(obj).forEach(k => {
|
|
obj[k] *= -1;
|
|
});
|
|
return obj;
|
|
}; // this method counts a total # of each unit, between a, b.
|
|
// '1 month' means 28 days in february
|
|
// '1 year' means 366 days in a leap year
|
|
|
|
|
|
const main$1 = function (a, b, unit) {
|
|
b = fns.beADate(b, a); //reverse values, if necessary
|
|
|
|
let reversed = false;
|
|
|
|
if (a.isAfter(b)) {
|
|
let tmp = a;
|
|
a = b;
|
|
b = tmp;
|
|
reversed = true;
|
|
} //compute them all (i know!)
|
|
|
|
|
|
let obj = waterfall(a, b);
|
|
|
|
if (reversed) {
|
|
obj = reverseDiff(obj);
|
|
} //return just the requested unit
|
|
|
|
|
|
if (unit) {
|
|
//make sure it's plural-form
|
|
unit = fns.normalize(unit);
|
|
|
|
if (/s$/.test(unit) !== true) {
|
|
unit += 's';
|
|
}
|
|
|
|
if (unit === 'dates') {
|
|
unit = 'days';
|
|
}
|
|
|
|
return obj[unit];
|
|
}
|
|
|
|
return obj;
|
|
};
|
|
|
|
var diff$1 = main$1;
|
|
|
|
//our conceptual 'break-points' for each unit
|
|
|
|
const qualifiers = {
|
|
months: {
|
|
almost: 10,
|
|
over: 4
|
|
},
|
|
days: {
|
|
almost: 25,
|
|
over: 10
|
|
},
|
|
hours: {
|
|
almost: 20,
|
|
over: 8
|
|
},
|
|
minutes: {
|
|
almost: 50,
|
|
over: 20
|
|
},
|
|
seconds: {
|
|
almost: 50,
|
|
over: 20
|
|
}
|
|
}; //get number of hours/minutes... between the two dates
|
|
|
|
function getDiff(a, b) {
|
|
const isBefore = a.isBefore(b);
|
|
const later = isBefore ? b : a;
|
|
let earlier = isBefore ? a : b;
|
|
earlier = earlier.clone();
|
|
const diff = {
|
|
years: 0,
|
|
months: 0,
|
|
days: 0,
|
|
hours: 0,
|
|
minutes: 0,
|
|
seconds: 0
|
|
};
|
|
Object.keys(diff).forEach(unit => {
|
|
if (earlier.isSame(later, unit)) {
|
|
return;
|
|
}
|
|
|
|
let max = earlier.diff(later, unit);
|
|
earlier = earlier.add(max, unit);
|
|
diff[unit] = max;
|
|
}); //reverse it, if necessary
|
|
|
|
if (isBefore) {
|
|
Object.keys(diff).forEach(u => {
|
|
if (diff[u] !== 0) {
|
|
diff[u] *= -1;
|
|
}
|
|
});
|
|
}
|
|
|
|
return diff;
|
|
} // Expects a plural unit arg
|
|
|
|
|
|
function pluralize(value, unit) {
|
|
if (value === 1) {
|
|
unit = unit.slice(0, -1);
|
|
}
|
|
|
|
return value + ' ' + unit;
|
|
} //create the human-readable diff between the two dates
|
|
|
|
|
|
const since$1 = (start, end) => {
|
|
end = fns.beADate(end, start);
|
|
const diff = getDiff(start, end);
|
|
const isNow = Object.keys(diff).every(u => !diff[u]);
|
|
|
|
if (isNow === true) {
|
|
return {
|
|
diff,
|
|
rounded: 'now',
|
|
qualified: 'now',
|
|
precise: 'now'
|
|
};
|
|
}
|
|
|
|
let rounded;
|
|
let qualified;
|
|
let precise;
|
|
let englishValues = []; //go through each value and create its text-representation
|
|
|
|
Object.keys(diff).forEach((unit, i, units) => {
|
|
const value = Math.abs(diff[unit]);
|
|
|
|
if (value === 0) {
|
|
return;
|
|
}
|
|
|
|
const englishValue = pluralize(value, unit);
|
|
englishValues.push(englishValue);
|
|
|
|
if (!rounded) {
|
|
rounded = qualified = englishValue;
|
|
|
|
if (i > 4) {
|
|
return;
|
|
} //is it a 'almost' something, etc?
|
|
|
|
|
|
const nextUnit = units[i + 1];
|
|
const nextValue = Math.abs(diff[nextUnit]);
|
|
|
|
if (nextValue > qualifiers[nextUnit].almost) {
|
|
rounded = pluralize(value + 1, unit);
|
|
qualified = 'almost ' + rounded;
|
|
} else if (nextValue > qualifiers[nextUnit].over) qualified = 'over ' + englishValue;
|
|
}
|
|
}); //make them into a string
|
|
|
|
precise = englishValues.splice(0, 2).join(', '); //handle before/after logic
|
|
|
|
if (start.isAfter(end) === true) {
|
|
rounded += ' ago';
|
|
qualified += ' ago';
|
|
precise += ' ago';
|
|
} else {
|
|
rounded = 'in ' + rounded;
|
|
qualified = 'in ' + qualified;
|
|
precise = 'in ' + precise;
|
|
}
|
|
|
|
return {
|
|
diff,
|
|
rounded,
|
|
qualified,
|
|
precise
|
|
};
|
|
};
|
|
|
|
var since_1 = since$1;
|
|
|
|
//https://www.timeanddate.com/calendar/aboutseasons.html
|
|
// Spring - from March 1 to May 31;
|
|
// Summer - from June 1 to August 31;
|
|
// Fall (autumn) - from September 1 to November 30; and,
|
|
// Winter - from December 1 to February 28 (February 29 in a leap year).
|
|
var seasons$1 = {
|
|
north: [['spring', 2, 1], //spring march 1
|
|
['summer', 5, 1], //june 1
|
|
['fall', 8, 1], //sept 1
|
|
['autumn', 8, 1], //sept 1
|
|
['winter', 11, 1] //dec 1
|
|
],
|
|
south: [['fall', 2, 1], //march 1
|
|
['autumn', 2, 1], //march 1
|
|
['winter', 5, 1], //june 1
|
|
['spring', 8, 1], //sept 1
|
|
['summer', 11, 1] //dec 1
|
|
]
|
|
};
|
|
|
|
var quarters$1 = [null, [0, 1], //jan 1
|
|
[3, 1], //apr 1
|
|
[6, 1], //july 1
|
|
[9, 1] //oct 1
|
|
];
|
|
|
|
var seasons = seasons$1;
|
|
|
|
var quarters = quarters$1;
|
|
|
|
const units$1 = {
|
|
minute: s => {
|
|
walkTo(s, {
|
|
second: 0,
|
|
millisecond: 0
|
|
});
|
|
return s;
|
|
},
|
|
quarterhour: s => {
|
|
let minute = s.minutes();
|
|
|
|
if (minute >= 45) {
|
|
s = s.minutes(45);
|
|
} else if (minute >= 30) {
|
|
s = s.minutes(30);
|
|
} else if (minute >= 15) {
|
|
s = s.minutes(15);
|
|
} else {
|
|
s = s.minutes(0);
|
|
}
|
|
|
|
walkTo(s, {
|
|
second: 0,
|
|
millisecond: 0
|
|
});
|
|
return s;
|
|
},
|
|
hour: s => {
|
|
walkTo(s, {
|
|
minute: 0,
|
|
second: 0,
|
|
millisecond: 0
|
|
});
|
|
return s;
|
|
},
|
|
day: s => {
|
|
walkTo(s, {
|
|
hour: 0,
|
|
minute: 0,
|
|
second: 0,
|
|
millisecond: 0
|
|
});
|
|
return s;
|
|
},
|
|
week: s => {
|
|
let original = s.clone();
|
|
s = s.day(s._weekStart); //monday
|
|
|
|
if (s.isAfter(original)) {
|
|
s = s.subtract(1, 'week');
|
|
}
|
|
|
|
walkTo(s, {
|
|
hour: 0,
|
|
minute: 0,
|
|
second: 0,
|
|
millisecond: 0
|
|
});
|
|
return s;
|
|
},
|
|
month: s => {
|
|
walkTo(s, {
|
|
date: 1,
|
|
hour: 0,
|
|
minute: 0,
|
|
second: 0,
|
|
millisecond: 0
|
|
});
|
|
return s;
|
|
},
|
|
quarter: s => {
|
|
let q = s.quarter();
|
|
|
|
if (quarters[q]) {
|
|
walkTo(s, {
|
|
month: quarters[q][0],
|
|
date: quarters[q][1],
|
|
hour: 0,
|
|
minute: 0,
|
|
second: 0,
|
|
millisecond: 0
|
|
});
|
|
}
|
|
|
|
return s;
|
|
},
|
|
season: s => {
|
|
let current = s.season();
|
|
let hem = 'north';
|
|
|
|
if (s.hemisphere() === 'South') {
|
|
hem = 'south';
|
|
}
|
|
|
|
for (let i = 0; i < seasons[hem].length; i++) {
|
|
if (seasons[hem][i][0] === current) {
|
|
//winter goes between years
|
|
let year = s.year();
|
|
|
|
if (current === 'winter' && s.month() < 3) {
|
|
year -= 1;
|
|
}
|
|
|
|
walkTo(s, {
|
|
year,
|
|
month: seasons[hem][i][1],
|
|
date: seasons[hem][i][2],
|
|
hour: 0,
|
|
minute: 0,
|
|
second: 0,
|
|
millisecond: 0
|
|
});
|
|
return s;
|
|
}
|
|
}
|
|
|
|
return s;
|
|
},
|
|
year: s => {
|
|
walkTo(s, {
|
|
month: 0,
|
|
date: 1,
|
|
hour: 0,
|
|
minute: 0,
|
|
second: 0,
|
|
millisecond: 0
|
|
});
|
|
return s;
|
|
},
|
|
decade: s => {
|
|
s = s.startOf('year');
|
|
let year = s.year();
|
|
let decade = parseInt(year / 10, 10) * 10;
|
|
s = s.year(decade);
|
|
return s;
|
|
},
|
|
century: s => {
|
|
s = s.startOf('year');
|
|
let year = s.year(); // near 0AD goes '-1 | +1'
|
|
|
|
let decade = parseInt(year / 100, 10) * 100;
|
|
s = s.year(decade);
|
|
return s;
|
|
}
|
|
};
|
|
units$1.date = units$1.day;
|
|
|
|
const startOf = (a, unit) => {
|
|
let s = a.clone();
|
|
unit = fns.normalize(unit);
|
|
|
|
if (units$1[unit]) {
|
|
return units$1[unit](s);
|
|
}
|
|
|
|
if (unit === 'summer' || unit === 'winter') {
|
|
s = s.season(unit);
|
|
return units$1.season(s);
|
|
}
|
|
|
|
return s;
|
|
}; //piggy-backs off startOf
|
|
|
|
|
|
const endOf = (a, unit) => {
|
|
let s = a.clone();
|
|
unit = fns.normalize(unit);
|
|
|
|
if (units$1[unit]) {
|
|
// go to beginning, go to next one, step back 1ms
|
|
s = units$1[unit](s); // startof
|
|
|
|
s = s.add(1, unit);
|
|
s = s.subtract(1, 'millisecond');
|
|
return s;
|
|
}
|
|
|
|
return s;
|
|
};
|
|
|
|
var startOf_1 = {
|
|
startOf,
|
|
endOf
|
|
};
|
|
|
|
const isDay = function (unit) {
|
|
if (days.short().find(s => s === unit)) {
|
|
return true;
|
|
}
|
|
|
|
if (days.long().find(s => s === unit)) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}; // return a list of the weeks/months/days between a -> b
|
|
// returns spacetime objects in the timezone of the input
|
|
|
|
|
|
const every$1 = function (start, unit = '', end) {
|
|
if (!unit || !end) {
|
|
return [];
|
|
} //cleanup unit param
|
|
|
|
|
|
unit = fns.normalize(unit); //cleanup to param
|
|
|
|
end = start.clone().set(end); //swap them, if they're backwards
|
|
|
|
if (start.isAfter(end)) {
|
|
let tmp = start;
|
|
start = end;
|
|
end = tmp;
|
|
} //support 'every wednesday'
|
|
|
|
|
|
let d = start.clone();
|
|
|
|
if (isDay(unit)) {
|
|
d = d.next(unit);
|
|
unit = 'week';
|
|
} else {
|
|
d = d.next(unit);
|
|
} //okay, actually start doing it
|
|
|
|
|
|
let result = [];
|
|
|
|
while (d.isBefore(end)) {
|
|
result.push(d);
|
|
d = d.add(1, unit);
|
|
}
|
|
|
|
return result;
|
|
};
|
|
|
|
var every_1 = every$1;
|
|
|
|
var findTz = find;
|
|
|
|
const parseDst = dst => {
|
|
if (!dst) {
|
|
return [];
|
|
}
|
|
|
|
return dst.split('->');
|
|
};
|
|
|
|
const titleCase = str => {
|
|
str = str[0].toUpperCase() + str.substr(1);
|
|
str = str.replace(/\/gmt/, '/GMT');
|
|
str = str.replace(/[\/_]([a-z])/gi, s => {
|
|
return s.toUpperCase();
|
|
});
|
|
return str;
|
|
}; //get metadata about this timezone
|
|
|
|
|
|
const timezone$1 = s => {
|
|
let zones = s.timezones;
|
|
let tz = s.tz;
|
|
|
|
if (zones.hasOwnProperty(tz) === false) {
|
|
tz = findTz(s.tz, zones);
|
|
}
|
|
|
|
if (tz === null) {
|
|
if (s.silent === false) {
|
|
console.warn("Warn: could not find given or local timezone - '" + s.tz + "'");
|
|
}
|
|
|
|
return {
|
|
current: {
|
|
epochShift: 0
|
|
}
|
|
};
|
|
}
|
|
|
|
let found = zones[tz];
|
|
let result = {
|
|
name: titleCase(tz),
|
|
hasDst: Boolean(found.dst),
|
|
default_offset: found.offset,
|
|
//do north-hemisphere version as default (sorry!)
|
|
hemisphere: found.hem === 's' ? 'South' : 'North',
|
|
current: {}
|
|
};
|
|
|
|
if (result.hasDst) {
|
|
let arr = parseDst(found.dst);
|
|
result.change = {
|
|
start: arr[0],
|
|
back: arr[1]
|
|
};
|
|
} //find the offsets for summer/winter times
|
|
//(these variable names are north-centric)
|
|
|
|
|
|
let summer = found.offset; // (july)
|
|
|
|
let winter = summer; // (january) assume it's the same for now
|
|
|
|
if (result.hasDst === true) {
|
|
if (result.hemisphere === 'North') {
|
|
winter = summer - 1;
|
|
} else {
|
|
//southern hemisphere
|
|
winter = found.offset + 1;
|
|
}
|
|
} //find out which offset to use right now
|
|
//use 'summer' time july-time
|
|
|
|
|
|
if (result.hasDst === false) {
|
|
result.current.offset = summer;
|
|
result.current.isDST = false;
|
|
} else if (inSummerTime(s.epoch, result.change.start, result.change.back, summer, winter) === true) {
|
|
result.current.offset = summer;
|
|
result.current.isDST = result.hemisphere === 'North'; //dst 'on' in winter in north
|
|
} else {
|
|
//use 'winter' january-time
|
|
result.current.offset = winter;
|
|
result.current.isDST = result.hemisphere === 'South'; //dst 'on' in summer in south
|
|
}
|
|
|
|
return result;
|
|
};
|
|
|
|
var timezone_1 = timezone$1;
|
|
|
|
var format = format_1;
|
|
|
|
var unixFmt = unixFmt_1;
|
|
|
|
var progress = progress_1;
|
|
|
|
var nearest = nearest_1;
|
|
|
|
var diff = diff$1;
|
|
|
|
var since = since_1;
|
|
|
|
var ends = startOf_1;
|
|
|
|
var every = every_1;
|
|
|
|
var timezone = timezone_1;
|
|
|
|
var handleInput = input;
|
|
|
|
const units = ['century', 'decade', 'year', 'month', 'date', 'day', 'hour', 'minute', 'second', 'millisecond']; //the spacetime instance methods (also, the API)
|
|
|
|
const methods$5 = {
|
|
set: function (input, tz) {
|
|
let s = this.clone();
|
|
s = handleInput(s, input, null);
|
|
|
|
if (tz) {
|
|
this.tz = findTz(tz);
|
|
}
|
|
|
|
return s;
|
|
},
|
|
timezone: function () {
|
|
return timezone(this);
|
|
},
|
|
isDST: function () {
|
|
return timezone(this).current.isDST;
|
|
},
|
|
hasDST: function () {
|
|
return timezone(this).hasDst;
|
|
},
|
|
offset: function () {
|
|
return timezone(this).current.offset * 60;
|
|
},
|
|
hemisphere: function () {
|
|
return timezone(this).hemisphere;
|
|
},
|
|
format: function (fmt) {
|
|
return format(this, fmt);
|
|
},
|
|
unixFmt: function (fmt) {
|
|
return unixFmt(this, fmt);
|
|
},
|
|
startOf: function (unit) {
|
|
return ends.startOf(this, unit);
|
|
},
|
|
endOf: function (unit) {
|
|
return ends.endOf(this, unit);
|
|
},
|
|
leapYear: function () {
|
|
let year = this.year();
|
|
return fns.isLeapYear(year);
|
|
},
|
|
progress: function (unit) {
|
|
return progress(this, unit);
|
|
},
|
|
nearest: function (unit) {
|
|
return nearest(this, unit);
|
|
},
|
|
diff: function (d, unit) {
|
|
return diff(this, d, unit);
|
|
},
|
|
since: function (d) {
|
|
if (!d) {
|
|
d = this.clone().set();
|
|
}
|
|
|
|
return since(this, d);
|
|
},
|
|
next: function (unit) {
|
|
let s = this.add(1, unit);
|
|
return s.startOf(unit);
|
|
},
|
|
//the start of the previous year/week/century
|
|
last: function (unit) {
|
|
let s = this.subtract(1, unit);
|
|
return s.startOf(unit);
|
|
},
|
|
isValid: function () {
|
|
//null/undefined epochs
|
|
if (!this.epoch && this.epoch !== 0) {
|
|
return false;
|
|
}
|
|
|
|
return !isNaN(this.d.getTime());
|
|
},
|
|
//travel to this timezone
|
|
goto: function (tz) {
|
|
let s = this.clone();
|
|
s.tz = findTz(tz, s.timezones); //science!
|
|
|
|
return s;
|
|
},
|
|
//get each week/month/day between a -> b
|
|
every: function (unit, to) {
|
|
// allow swapping these params:
|
|
if (typeof unit === 'object' && typeof to === 'string') {
|
|
let tmp = to;
|
|
to = unit;
|
|
unit = tmp;
|
|
}
|
|
|
|
return every(this, unit, to);
|
|
},
|
|
isAwake: function () {
|
|
let hour = this.hour(); //10pm -> 8am
|
|
|
|
if (hour < 8 || hour > 22) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
},
|
|
isAsleep: function () {
|
|
return !this.isAwake();
|
|
},
|
|
daysInMonth: function () {
|
|
switch (this.month()) {
|
|
case 0:
|
|
return 31;
|
|
|
|
case 1:
|
|
return this.leapYear() ? 29 : 28;
|
|
|
|
case 2:
|
|
return 31;
|
|
|
|
case 3:
|
|
return 30;
|
|
|
|
case 4:
|
|
return 31;
|
|
|
|
case 5:
|
|
return 30;
|
|
|
|
case 6:
|
|
return 31;
|
|
|
|
case 7:
|
|
return 31;
|
|
|
|
case 8:
|
|
return 30;
|
|
|
|
case 9:
|
|
return 31;
|
|
|
|
case 10:
|
|
return 30;
|
|
|
|
case 11:
|
|
return 31;
|
|
|
|
default:
|
|
throw new Error('Invalid Month state.');
|
|
}
|
|
},
|
|
//pretty-printing
|
|
log: function () {
|
|
console.log('');
|
|
console.log(format(this, 'nice-short'));
|
|
return this;
|
|
},
|
|
logYear: function () {
|
|
console.log('');
|
|
console.log(format(this, 'full-short'));
|
|
return this;
|
|
},
|
|
json: function () {
|
|
return units.reduce((h, unit) => {
|
|
h[unit] = this[unit]();
|
|
return h;
|
|
}, {});
|
|
},
|
|
debug: function () {
|
|
let tz = this.timezone();
|
|
let date = this.format('MM') + ' ' + this.format('date-ordinal') + ' ' + this.year();
|
|
date += '\n - ' + this.format('time');
|
|
console.log('\n\n', date + '\n - ' + tz.name + ' (' + tz.current.offset + ')');
|
|
return this;
|
|
},
|
|
//alias of 'since' but opposite - like moment.js
|
|
from: function (d) {
|
|
d = this.clone().set(d);
|
|
return d.since(this);
|
|
},
|
|
fromNow: function () {
|
|
let d = this.clone().set(Date.now());
|
|
return d.since(this);
|
|
},
|
|
weekStart: function (input) {
|
|
//accept a number directly
|
|
if (typeof input === 'number') {
|
|
this._weekStart = input;
|
|
return this;
|
|
}
|
|
|
|
if (typeof input === 'string') {
|
|
// accept 'wednesday'
|
|
input = input.toLowerCase().trim();
|
|
let num = days.short().indexOf(input);
|
|
|
|
if (num === -1) {
|
|
num = days.long().indexOf(input);
|
|
}
|
|
|
|
if (num === -1) {
|
|
num = 1; //go back to default
|
|
}
|
|
|
|
this._weekStart = num;
|
|
} else {
|
|
console.warn('Spacetime Error: Cannot understand .weekStart() input:', input);
|
|
}
|
|
|
|
return this;
|
|
}
|
|
}; // aliases
|
|
|
|
methods$5.inDST = methods$5.isDST;
|
|
methods$5.round = methods$5.nearest;
|
|
methods$5.each = methods$5.every;
|
|
var methods_1 = methods$5;
|
|
|
|
//these methods wrap around them.
|
|
|
|
const isLeapYear$1 = fns.isLeapYear;
|
|
|
|
const validate = n => {
|
|
//handle number as a string
|
|
if (typeof n === 'string') {
|
|
n = parseInt(n, 10);
|
|
}
|
|
|
|
return n;
|
|
};
|
|
|
|
const order$1 = ['year', 'month', 'date', 'hour', 'minute', 'second', 'millisecond']; //reduce hostile micro-changes when moving dates by millisecond
|
|
|
|
const confirm = (s, tmp, unit) => {
|
|
let n = order$1.indexOf(unit);
|
|
let arr = order$1.slice(n, order$1.length);
|
|
|
|
for (let i = 0; i < arr.length; i++) {
|
|
let want = tmp[arr[i]]();
|
|
s[arr[i]](want);
|
|
}
|
|
|
|
return s;
|
|
}; // allow specifying setter direction
|
|
|
|
|
|
const fwdBkwd = function (s, old, goFwd, unit) {
|
|
if (goFwd === true && s.isBefore(old)) {
|
|
s = s.add(1, unit);
|
|
} else if (goFwd === false && s.isAfter(old)) {
|
|
s = s.minus(1, unit);
|
|
}
|
|
|
|
return s;
|
|
};
|
|
|
|
var set$1 = {
|
|
milliseconds: (s, n) => {
|
|
n = validate(n);
|
|
let current = s.millisecond();
|
|
let diff = current - n; //milliseconds to shift by
|
|
|
|
return s.epoch - diff;
|
|
},
|
|
seconds: (s, n, goFwd) => {
|
|
n = validate(n);
|
|
let old = s.clone();
|
|
let diff = s.second() - n;
|
|
let shift = diff * ms.second;
|
|
s.epoch = s.epoch - shift;
|
|
s = fwdBkwd(s, old, goFwd, 'minute'); // specify direction
|
|
|
|
return s.epoch;
|
|
},
|
|
minutes: (s, n, goFwd) => {
|
|
n = validate(n);
|
|
let old = s.clone();
|
|
let diff = s.minute() - n;
|
|
let shift = diff * ms.minute;
|
|
s.epoch -= shift;
|
|
confirm(s, old, 'second');
|
|
s = fwdBkwd(s, old, goFwd, 'hour'); // specify direction
|
|
|
|
return s.epoch;
|
|
},
|
|
hours: (s, n, goFwd) => {
|
|
n = validate(n);
|
|
|
|
if (n >= 24) {
|
|
n = 24;
|
|
} else if (n < 0) {
|
|
n = 0;
|
|
}
|
|
|
|
let old = s.clone();
|
|
let diff = s.hour() - n;
|
|
let shift = diff * ms.hour;
|
|
s.epoch -= shift; // oops, did we change the day?
|
|
|
|
if (s.date() !== old.date()) {
|
|
s = old.clone();
|
|
|
|
if (diff > 1) {
|
|
diff -= 1;
|
|
}
|
|
|
|
if (diff < 1) {
|
|
diff += 1;
|
|
}
|
|
|
|
shift = diff * ms.hour;
|
|
s.epoch -= shift;
|
|
}
|
|
|
|
walkTo(s, {
|
|
hour: n
|
|
});
|
|
confirm(s, old, 'minute');
|
|
s = fwdBkwd(s, old, goFwd, 'day'); // specify direction
|
|
|
|
return s.epoch;
|
|
},
|
|
//support setting time by '4:25pm'
|
|
time: (s, str, goFwd) => {
|
|
let m = str.match(/([0-9]{1,2})[:h]([0-9]{1,2})(:[0-9]{1,2})? ?(am|pm)?/);
|
|
|
|
if (!m) {
|
|
//fallback to support just '2am'
|
|
m = str.match(/([0-9]{1,2}) ?(am|pm)/);
|
|
|
|
if (!m) {
|
|
return s.epoch;
|
|
}
|
|
|
|
m.splice(2, 0, '0'); //add implicit 0 minutes
|
|
|
|
m.splice(3, 0, ''); //add implicit seconds
|
|
}
|
|
|
|
let h24 = false;
|
|
let hour = parseInt(m[1], 10);
|
|
let minute = parseInt(m[2], 10);
|
|
|
|
if (minute >= 60) {
|
|
minute = 59;
|
|
}
|
|
|
|
if (hour > 12) {
|
|
h24 = true;
|
|
} //make the hour into proper 24h time
|
|
|
|
|
|
if (h24 === false) {
|
|
if (m[4] === 'am' && hour === 12) {
|
|
//12am is midnight
|
|
hour = 0;
|
|
}
|
|
|
|
if (m[4] === 'pm' && hour < 12) {
|
|
//12pm is noon
|
|
hour += 12;
|
|
}
|
|
} // handle seconds
|
|
|
|
|
|
m[3] = m[3] || '';
|
|
m[3] = m[3].replace(/:/, '');
|
|
let sec = parseInt(m[3], 10) || 0;
|
|
let old = s.clone();
|
|
s = s.hour(hour);
|
|
s = s.minute(minute);
|
|
s = s.second(sec);
|
|
s = s.millisecond(0);
|
|
s = fwdBkwd(s, old, goFwd, 'day'); // specify direction
|
|
|
|
return s.epoch;
|
|
},
|
|
date: (s, n, goFwd) => {
|
|
n = validate(n); //avoid setting february 31st
|
|
|
|
if (n > 28) {
|
|
let month = s.month();
|
|
let max = monthLength[month]; // support leap day in february
|
|
|
|
if (month === 1 && n === 29 && isLeapYear$1(s.year())) {
|
|
max = 29;
|
|
}
|
|
|
|
if (n > max) {
|
|
n = max;
|
|
}
|
|
} //avoid setting < 0
|
|
|
|
|
|
if (n <= 0) {
|
|
n = 1;
|
|
}
|
|
|
|
let old = s.clone();
|
|
walkTo(s, {
|
|
date: n
|
|
});
|
|
s = fwdBkwd(s, old, goFwd, 'month'); // specify direction
|
|
|
|
return s.epoch;
|
|
},
|
|
//this one's tricky
|
|
month: (s, n, goFwd) => {
|
|
if (typeof n === 'string') {
|
|
n = months$1.mapping()[n.toLowerCase()];
|
|
}
|
|
|
|
n = validate(n); //don't go past december
|
|
|
|
if (n >= 12) {
|
|
n = 11;
|
|
}
|
|
|
|
if (n <= 0) {
|
|
n = 0;
|
|
}
|
|
|
|
let date = s.date(); //there's no 30th of february, etc.
|
|
|
|
if (date > monthLength[n]) {
|
|
//make it as close as we can..
|
|
date = monthLength[n];
|
|
}
|
|
|
|
let old = s.clone();
|
|
walkTo(s, {
|
|
month: n,
|
|
date
|
|
});
|
|
s = fwdBkwd(s, old, goFwd, 'year'); // specify direction
|
|
|
|
return s.epoch;
|
|
},
|
|
year: (s, n) => {
|
|
// support '97
|
|
if (typeof n === 'string' && /^'[0-9]{2}$/.test(n)) {
|
|
n = n.replace(/'/, '').trim();
|
|
n = Number(n); // '89 is 1989
|
|
|
|
if (n > 30) {
|
|
//change this in 10y
|
|
n = 1900 + n;
|
|
} else {
|
|
// '12 is 2012
|
|
n = 2000 + n;
|
|
}
|
|
}
|
|
|
|
n = validate(n);
|
|
walkTo(s, {
|
|
year: n
|
|
});
|
|
return s.epoch;
|
|
},
|
|
// go to the nth week of the year
|
|
week: (s, n, goFwd) => {
|
|
let old = s.clone();
|
|
n = validate(n);
|
|
s = s.month(0);
|
|
s = s.date(1);
|
|
s = s.day('monday'); //first week starts first Thurs in Jan
|
|
// so mon dec 28th is 1st week
|
|
// so mon dec 29th is not the week
|
|
|
|
if (s.monthName() === 'december' && s.date() >= 28) {
|
|
s = s.add(1, 'week');
|
|
}
|
|
|
|
n -= 1; //1-based
|
|
|
|
s = s.add(n, 'weeks');
|
|
s = fwdBkwd(s, old, goFwd, 'year'); // specify direction
|
|
|
|
return s.epoch;
|
|
},
|
|
// go to the nth day of the year
|
|
dayOfYear: (s, n, goFwd) => {
|
|
n = validate(n);
|
|
let old = s.clone();
|
|
n -= 1; //days are 1-based
|
|
|
|
if (n <= 0) {
|
|
n = 0;
|
|
} else if (n >= 365) {
|
|
n = 364;
|
|
}
|
|
|
|
s = s.startOf('year');
|
|
s = s.add(n, 'day');
|
|
confirm(s, old, 'hour');
|
|
s = fwdBkwd(s, old, goFwd, 'year'); // specify direction
|
|
|
|
return s.epoch;
|
|
}
|
|
};
|
|
|
|
var set = set$1;
|
|
|
|
const methods$4 = {
|
|
millisecond: function (num) {
|
|
if (num !== undefined) {
|
|
let s = this.clone();
|
|
s.epoch = set.milliseconds(s, num);
|
|
return s;
|
|
}
|
|
|
|
return this.d.getMilliseconds();
|
|
},
|
|
second: function (num, goFwd) {
|
|
if (num !== undefined) {
|
|
let s = this.clone();
|
|
s.epoch = set.seconds(s, num, goFwd);
|
|
return s;
|
|
}
|
|
|
|
return this.d.getSeconds();
|
|
},
|
|
minute: function (num, goFwd) {
|
|
if (num !== undefined) {
|
|
let s = this.clone();
|
|
s.epoch = set.minutes(s, num, goFwd);
|
|
return s;
|
|
}
|
|
|
|
return this.d.getMinutes();
|
|
},
|
|
hour: function (num, goFwd) {
|
|
let d = this.d;
|
|
|
|
if (num !== undefined) {
|
|
let s = this.clone();
|
|
s.epoch = set.hours(s, num, goFwd);
|
|
return s;
|
|
}
|
|
|
|
return d.getHours();
|
|
},
|
|
//'3:30' is 3.5
|
|
hourFloat: function (num, goFwd) {
|
|
if (num !== undefined) {
|
|
let s = this.clone();
|
|
let minute = num % 1;
|
|
minute = minute * 60;
|
|
let hour = parseInt(num, 10);
|
|
s.epoch = set.hours(s, hour, goFwd);
|
|
s.epoch = set.minutes(s, minute, goFwd);
|
|
return s;
|
|
}
|
|
|
|
let d = this.d;
|
|
let hour = d.getHours();
|
|
let minute = d.getMinutes();
|
|
minute = minute / 60;
|
|
return hour + minute;
|
|
},
|
|
// hour in 12h format
|
|
hour12: function (str, goFwd) {
|
|
let d = this.d;
|
|
|
|
if (str !== undefined) {
|
|
let s = this.clone();
|
|
str = '' + str;
|
|
let m = str.match(/^([0-9]+)(am|pm)$/);
|
|
|
|
if (m) {
|
|
let hour = parseInt(m[1], 10);
|
|
|
|
if (m[2] === 'pm') {
|
|
hour += 12;
|
|
}
|
|
|
|
s.epoch = set.hours(s, hour, goFwd);
|
|
}
|
|
|
|
return s;
|
|
} //get the hour
|
|
|
|
|
|
let hour12 = d.getHours();
|
|
|
|
if (hour12 > 12) {
|
|
hour12 = hour12 - 12;
|
|
}
|
|
|
|
if (hour12 === 0) {
|
|
hour12 = 12;
|
|
}
|
|
|
|
return hour12;
|
|
},
|
|
//some ambiguity here with 12/24h
|
|
time: function (str, goFwd) {
|
|
if (str !== undefined) {
|
|
let s = this.clone();
|
|
str = str.toLowerCase().trim();
|
|
s.epoch = set.time(s, str, goFwd);
|
|
return s;
|
|
}
|
|
|
|
return `${this.h12()}:${fns.zeroPad(this.minute())}${this.ampm()}`;
|
|
},
|
|
// either 'am' or 'pm'
|
|
ampm: function (input, goFwd) {
|
|
let which = 'am';
|
|
let hour = this.hour();
|
|
|
|
if (hour >= 12) {
|
|
which = 'pm';
|
|
}
|
|
|
|
if (typeof input !== 'string') {
|
|
return which;
|
|
} //okay, we're doing a setter
|
|
|
|
|
|
let s = this.clone();
|
|
input = input.toLowerCase().trim(); //ampm should never change the day
|
|
// - so use `.hour(n)` instead of `.minus(12,'hour')`
|
|
|
|
if (hour >= 12 && input === 'am') {
|
|
//noon is 12pm
|
|
hour -= 12;
|
|
return s.hour(hour, goFwd);
|
|
}
|
|
|
|
if (hour < 12 && input === 'pm') {
|
|
hour += 12;
|
|
return s.hour(hour, goFwd);
|
|
}
|
|
|
|
return s;
|
|
},
|
|
//some hard-coded times of day, like 'noon'
|
|
dayTime: function (str, goFwd) {
|
|
if (str !== undefined) {
|
|
const times = {
|
|
morning: '7:00am',
|
|
breakfast: '7:00am',
|
|
noon: '12:00am',
|
|
lunch: '12:00pm',
|
|
afternoon: '2:00pm',
|
|
evening: '6:00pm',
|
|
dinner: '6:00pm',
|
|
night: '11:00pm',
|
|
midnight: '23:59pm'
|
|
};
|
|
let s = this.clone();
|
|
str = str || '';
|
|
str = str.toLowerCase();
|
|
|
|
if (times.hasOwnProperty(str) === true) {
|
|
s = s.time(times[str], goFwd);
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
let h = this.hour();
|
|
|
|
if (h < 6) {
|
|
return 'night';
|
|
}
|
|
|
|
if (h < 12) {
|
|
//until noon
|
|
return 'morning';
|
|
}
|
|
|
|
if (h < 17) {
|
|
//until 5pm
|
|
return 'afternoon';
|
|
}
|
|
|
|
if (h < 22) {
|
|
//until 10pm
|
|
return 'evening';
|
|
}
|
|
|
|
return 'night';
|
|
},
|
|
//parse a proper iso string
|
|
iso: function (num) {
|
|
if (num !== undefined) {
|
|
return this.set(num);
|
|
}
|
|
|
|
return this.format('iso');
|
|
}
|
|
};
|
|
var _01Time = methods$4;
|
|
|
|
const methods$3 = {
|
|
// # day in the month
|
|
date: function (num, goFwd) {
|
|
if (num !== undefined) {
|
|
let s = this.clone();
|
|
num = parseInt(num, 10);
|
|
|
|
if (num) {
|
|
s.epoch = set.date(s, num, goFwd);
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
return this.d.getDate();
|
|
},
|
|
//like 'wednesday' (hard!)
|
|
day: function (input, goFwd) {
|
|
if (input === undefined) {
|
|
return this.d.getDay();
|
|
}
|
|
|
|
let original = this.clone();
|
|
let want = input; // accept 'wednesday'
|
|
|
|
if (typeof input === 'string') {
|
|
input = input.toLowerCase();
|
|
|
|
if (days.aliases.hasOwnProperty(input)) {
|
|
want = days.aliases[input];
|
|
} else {
|
|
want = days.short().indexOf(input);
|
|
|
|
if (want === -1) {
|
|
want = days.long().indexOf(input);
|
|
}
|
|
}
|
|
} //move approx
|
|
|
|
|
|
let day = this.d.getDay();
|
|
let diff = day - want;
|
|
|
|
if (goFwd === true && diff > 0) {
|
|
diff = diff - 7;
|
|
}
|
|
|
|
if (goFwd === false && diff < 0) {
|
|
diff = diff + 7;
|
|
}
|
|
|
|
let s = this.subtract(diff, 'days'); //tighten it back up
|
|
|
|
walkTo(s, {
|
|
hour: original.hour(),
|
|
minute: original.minute(),
|
|
second: original.second()
|
|
});
|
|
return s;
|
|
},
|
|
//these are helpful name-wrappers
|
|
dayName: function (input, goFwd) {
|
|
if (input === undefined) {
|
|
return days.long()[this.day()];
|
|
}
|
|
|
|
let s = this.clone();
|
|
s = s.day(input, goFwd);
|
|
return s;
|
|
}
|
|
};
|
|
var _02Date = methods$3;
|
|
|
|
const clearMinutes = s => {
|
|
s = s.minute(0);
|
|
s = s.second(0);
|
|
s = s.millisecond(1);
|
|
return s;
|
|
};
|
|
|
|
const methods$2 = {
|
|
// day 0-366
|
|
dayOfYear: function (num, goFwd) {
|
|
if (num !== undefined) {
|
|
let s = this.clone();
|
|
s.epoch = set.dayOfYear(s, num, goFwd);
|
|
return s;
|
|
} //days since newyears - jan 1st is 1, jan 2nd is 2...
|
|
|
|
|
|
let sum = 0;
|
|
let month = this.d.getMonth();
|
|
let tmp; //count the num days in each month
|
|
|
|
for (let i = 1; i <= month; i++) {
|
|
tmp = new Date();
|
|
tmp.setDate(1);
|
|
tmp.setFullYear(this.d.getFullYear()); //the year matters, because leap-years
|
|
|
|
tmp.setHours(1);
|
|
tmp.setMinutes(1);
|
|
tmp.setMonth(i);
|
|
tmp.setHours(-2); //the last day of the month
|
|
|
|
sum += tmp.getDate();
|
|
}
|
|
|
|
return sum + this.d.getDate();
|
|
},
|
|
//since the start of the year
|
|
week: function (num, goFwd) {
|
|
// week-setter
|
|
if (num !== undefined) {
|
|
let s = this.clone();
|
|
s.epoch = set.week(this, num, goFwd);
|
|
s = clearMinutes(s);
|
|
return s;
|
|
} //find-out which week it is
|
|
|
|
|
|
let tmp = this.clone();
|
|
tmp = tmp.month(0);
|
|
tmp = tmp.date(1);
|
|
tmp = clearMinutes(tmp);
|
|
tmp = tmp.day('monday'); //don't go into last-year
|
|
|
|
if (tmp.monthName() === 'december' && tmp.date() >= 28) {
|
|
tmp = tmp.add(1, 'week');
|
|
} // is first monday the 1st?
|
|
|
|
|
|
let toAdd = 1;
|
|
|
|
if (tmp.date() === 1) {
|
|
toAdd = 0;
|
|
}
|
|
|
|
tmp = tmp.minus(1, 'second');
|
|
const thisOne = this.epoch; //if the week technically hasn't started yet
|
|
|
|
if (tmp.epoch > thisOne) {
|
|
return 1;
|
|
} //speed it up, if we can
|
|
|
|
|
|
let i = 0;
|
|
let skipWeeks = this.month() * 4;
|
|
tmp.epoch += ms.week * skipWeeks;
|
|
i += skipWeeks;
|
|
|
|
for (; i <= 52; i++) {
|
|
if (tmp.epoch > thisOne) {
|
|
return i + toAdd;
|
|
}
|
|
|
|
tmp = tmp.add(1, 'week');
|
|
}
|
|
|
|
return 52;
|
|
},
|
|
//either name or number
|
|
month: function (input, goFwd) {
|
|
if (input !== undefined) {
|
|
let s = this.clone();
|
|
s.epoch = set.month(s, input, goFwd);
|
|
return s;
|
|
}
|
|
|
|
return this.d.getMonth();
|
|
},
|
|
//'january'
|
|
monthName: function (input, goFwd) {
|
|
if (input !== undefined) {
|
|
let s = this.clone();
|
|
s = s.month(input, goFwd);
|
|
return s;
|
|
}
|
|
|
|
return months$1.long()[this.month()];
|
|
},
|
|
//q1, q2, q3, q4
|
|
quarter: function (num, goFwd) {
|
|
if (num !== undefined) {
|
|
if (typeof num === 'string') {
|
|
num = num.replace(/^q/i, '');
|
|
num = parseInt(num, 10);
|
|
}
|
|
|
|
if (quarters[num]) {
|
|
let s = this.clone();
|
|
let month = quarters[num][0];
|
|
s = s.month(month, goFwd);
|
|
s = s.date(1, goFwd);
|
|
s = s.startOf('day');
|
|
return s;
|
|
}
|
|
}
|
|
|
|
let month = this.d.getMonth();
|
|
|
|
for (let i = 1; i < quarters.length; i++) {
|
|
if (month < quarters[i][0]) {
|
|
return i - 1;
|
|
}
|
|
}
|
|
|
|
return 4;
|
|
},
|
|
//spring, summer, winter, fall
|
|
season: function (input, goFwd) {
|
|
let hem = 'north';
|
|
|
|
if (this.hemisphere() === 'South') {
|
|
hem = 'south';
|
|
}
|
|
|
|
if (input !== undefined) {
|
|
let s = this.clone();
|
|
|
|
for (let i = 0; i < seasons[hem].length; i++) {
|
|
if (input === seasons[hem][i][0]) {
|
|
s = s.month(seasons[hem][i][1], goFwd);
|
|
s = s.date(1);
|
|
s = s.startOf('day');
|
|
}
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
let month = this.d.getMonth();
|
|
|
|
for (let i = 0; i < seasons[hem].length - 1; i++) {
|
|
if (month >= seasons[hem][i][1] && month < seasons[hem][i + 1][1]) {
|
|
return seasons[hem][i][0];
|
|
}
|
|
}
|
|
|
|
return 'winter';
|
|
},
|
|
//the year number
|
|
year: function (num) {
|
|
if (num !== undefined) {
|
|
let s = this.clone();
|
|
s.epoch = set.year(s, num);
|
|
return s;
|
|
}
|
|
|
|
return this.d.getFullYear();
|
|
},
|
|
//bc/ad years
|
|
era: function (str) {
|
|
if (str !== undefined) {
|
|
let s = this.clone();
|
|
str = str.toLowerCase(); //TODO: there is no year-0AD i think. may have off-by-1 error here
|
|
|
|
let year = s.d.getFullYear(); //make '1992' into 1992bc..
|
|
|
|
if (str === 'bc' && year > 0) {
|
|
s.epoch = set.year(s, year * -1);
|
|
} //make '1992bc' into '1992'
|
|
|
|
|
|
if (str === 'ad' && year < 0) {
|
|
s.epoch = set.year(s, year * -1);
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
if (this.d.getFullYear() < 0) {
|
|
return 'BC';
|
|
}
|
|
|
|
return 'AD';
|
|
},
|
|
// 2019 -> 2010
|
|
decade: function (input) {
|
|
if (input !== undefined) {
|
|
input = String(input);
|
|
input = input.replace(/([0-9])'?s$/, '$1'); //1950's
|
|
|
|
input = input.replace(/([0-9])(th|rd|st|nd)/, '$1'); //fix ordinals
|
|
|
|
if (!input) {
|
|
console.warn('Spacetime: Invalid decade input');
|
|
return this;
|
|
} // assume 20th century?? for '70s'.
|
|
|
|
|
|
if (input.length === 2 && /[0-9][0-9]/.test(input)) {
|
|
input = '19' + input;
|
|
}
|
|
|
|
let year = Number(input);
|
|
|
|
if (isNaN(year)) {
|
|
return this;
|
|
} // round it down to the decade
|
|
|
|
|
|
year = Math.floor(year / 10) * 10;
|
|
return this.year(year); //.startOf('decade')
|
|
}
|
|
|
|
return this.startOf('decade').year();
|
|
},
|
|
// 1950 -> 19+1
|
|
century: function (input) {
|
|
if (input !== undefined) {
|
|
if (typeof input === 'string') {
|
|
input = input.replace(/([0-9])(th|rd|st|nd)/, '$1'); //fix ordinals
|
|
|
|
input = input.replace(/([0-9]+) ?(b\.?c\.?|a\.?d\.?)/i, (a, b, c) => {
|
|
if (c.match(/b\.?c\.?/i)) {
|
|
b = '-' + b;
|
|
}
|
|
|
|
return b;
|
|
});
|
|
input = input.replace(/c$/, ''); //20thC
|
|
}
|
|
|
|
let year = Number(input);
|
|
|
|
if (isNaN(input)) {
|
|
console.warn('Spacetime: Invalid century input');
|
|
return this;
|
|
} // there is no century 0
|
|
|
|
|
|
if (year === 0) {
|
|
year = 1;
|
|
}
|
|
|
|
if (year >= 0) {
|
|
year = (year - 1) * 100;
|
|
} else {
|
|
year = (year + 1) * 100;
|
|
}
|
|
|
|
return this.year(year);
|
|
} // century getter
|
|
|
|
|
|
let num = this.startOf('century').year();
|
|
num = Math.floor(num / 100);
|
|
|
|
if (num < 0) {
|
|
return num - 1;
|
|
}
|
|
|
|
return num + 1;
|
|
},
|
|
// 2019 -> 2+1
|
|
millenium: function (input) {
|
|
if (input !== undefined) {
|
|
if (typeof input === 'string') {
|
|
input = input.replace(/([0-9])(th|rd|st|nd)/, '$1'); //fix ordinals
|
|
|
|
input = Number(input);
|
|
|
|
if (isNaN(input)) {
|
|
console.warn('Spacetime: Invalid millenium input');
|
|
return this;
|
|
}
|
|
}
|
|
|
|
if (input > 0) {
|
|
input -= 1;
|
|
}
|
|
|
|
let year = input * 1000; // there is no year 0
|
|
|
|
if (year === 0) {
|
|
year = 1;
|
|
}
|
|
|
|
return this.year(year);
|
|
} // get the current millenium
|
|
|
|
|
|
let num = Math.floor(this.year() / 1000);
|
|
|
|
if (num >= 0) {
|
|
num += 1;
|
|
}
|
|
|
|
return num;
|
|
}
|
|
};
|
|
var _03Year = methods$2;
|
|
|
|
var require$$0 = _01Time;
|
|
|
|
var require$$1$1 = _02Date;
|
|
|
|
var require$$2$1 = _03Year;
|
|
|
|
const methods$1 = Object.assign({}, require$$0, require$$1$1, require$$2$1); //aliases
|
|
|
|
methods$1.milliseconds = methods$1.millisecond;
|
|
methods$1.seconds = methods$1.second;
|
|
methods$1.minutes = methods$1.minute;
|
|
methods$1.hours = methods$1.hour;
|
|
methods$1.hour24 = methods$1.hour;
|
|
methods$1.h12 = methods$1.hour12;
|
|
methods$1.h24 = methods$1.hour24;
|
|
methods$1.days = methods$1.day;
|
|
|
|
const addMethods$4 = Space => {
|
|
//hook the methods into prototype
|
|
Object.keys(methods$1).forEach(k => {
|
|
Space.prototype[k] = methods$1[k];
|
|
});
|
|
};
|
|
|
|
var query = addMethods$4;
|
|
|
|
const isLeapYear = fns.isLeapYear;
|
|
|
|
const getMonthLength = function (month, year) {
|
|
if (month === 1 && isLeapYear(year)) {
|
|
return 29;
|
|
}
|
|
|
|
return monthLength[month];
|
|
}; //month is the one thing we 'model/compute'
|
|
//- because ms-shifting can be off by enough
|
|
|
|
|
|
const rollMonth = (want, old) => {
|
|
//increment year
|
|
if (want.month > 0) {
|
|
let years = parseInt(want.month / 12, 10);
|
|
want.year = old.year() + years;
|
|
want.month = want.month % 12;
|
|
} else if (want.month < 0) {
|
|
//decrement year
|
|
let years = Math.floor(Math.abs(want.month) / 13, 10);
|
|
years = Math.abs(years) + 1;
|
|
want.year = old.year() - years; //ignore extras
|
|
|
|
want.month = want.month % 12;
|
|
want.month = want.month + 12;
|
|
|
|
if (want.month === 12) {
|
|
want.month = 0;
|
|
}
|
|
}
|
|
|
|
return want;
|
|
}; // briefly support day=-2 (this does not need to be perfect.)
|
|
|
|
|
|
const rollDaysDown = (want, old, sum) => {
|
|
want.year = old.year();
|
|
want.month = old.month();
|
|
let date = old.date();
|
|
want.date = date - Math.abs(sum);
|
|
|
|
while (want.date < 1) {
|
|
want.month -= 1;
|
|
|
|
if (want.month < 0) {
|
|
want.month = 11;
|
|
want.year -= 1;
|
|
}
|
|
|
|
let max = getMonthLength(want.month, want.year);
|
|
want.date += max;
|
|
}
|
|
|
|
return want;
|
|
}; // briefly support day=33 (this does not need to be perfect.)
|
|
|
|
|
|
const rollDaysUp = (want, old, sum) => {
|
|
let year = old.year();
|
|
let month = old.month();
|
|
let max = getMonthLength(month, year);
|
|
|
|
while (sum > max) {
|
|
sum -= max;
|
|
month += 1;
|
|
|
|
if (month >= 12) {
|
|
month -= 12;
|
|
year += 1;
|
|
}
|
|
|
|
max = getMonthLength(month, year);
|
|
}
|
|
|
|
want.month = month;
|
|
want.date = sum;
|
|
return want;
|
|
};
|
|
|
|
var _model = {
|
|
months: rollMonth,
|
|
days: rollDaysUp,
|
|
daysBack: rollDaysDown
|
|
};
|
|
|
|
var model = _model;
|
|
|
|
// but briefly:
|
|
// millisecond-math, and some post-processing covers most-things
|
|
// we 'model' the calendar here only a little bit
|
|
// and that usually works-out...
|
|
|
|
const order = ['millisecond', 'second', 'minute', 'hour', 'date', 'month'];
|
|
let keep = {
|
|
second: order.slice(0, 1),
|
|
minute: order.slice(0, 2),
|
|
quarterhour: order.slice(0, 2),
|
|
hour: order.slice(0, 3),
|
|
date: order.slice(0, 4),
|
|
month: order.slice(0, 4),
|
|
quarter: order.slice(0, 4),
|
|
season: order.slice(0, 4),
|
|
year: order,
|
|
decade: order,
|
|
century: order
|
|
};
|
|
keep.week = keep.hour;
|
|
keep.season = keep.date;
|
|
keep.quarter = keep.date; // Units need to be dst adjuested
|
|
|
|
const dstAwareUnits = {
|
|
year: true,
|
|
quarter: true,
|
|
season: true,
|
|
month: true,
|
|
week: true,
|
|
date: true
|
|
};
|
|
const keepDate = {
|
|
month: true,
|
|
quarter: true,
|
|
season: true,
|
|
year: true
|
|
};
|
|
|
|
const addMethods$3 = SpaceTime => {
|
|
SpaceTime.prototype.add = function (num, unit) {
|
|
let s = this.clone();
|
|
|
|
if (!unit || num === 0) {
|
|
return s; //don't bother
|
|
}
|
|
|
|
let old = this.clone();
|
|
unit = fns.normalize(unit);
|
|
|
|
if (unit === 'millisecond') {
|
|
s.epoch += num;
|
|
return s;
|
|
} // support 'fortnight' alias
|
|
|
|
|
|
if (unit === 'fortnight') {
|
|
num *= 2;
|
|
unit = 'week';
|
|
} //move forward by the estimated milliseconds (rough)
|
|
|
|
|
|
if (ms[unit]) {
|
|
s.epoch += ms[unit] * num;
|
|
} else if (unit === 'week' || unit === 'weekend') {
|
|
s.epoch += ms.day * (num * 7);
|
|
} else if (unit === 'quarter' || unit === 'season') {
|
|
s.epoch += ms.month * (num * 3);
|
|
} else if (unit === 'quarterhour') {
|
|
s.epoch += ms.minute * 15 * num;
|
|
} //now ensure our milliseconds/etc are in-line
|
|
|
|
|
|
let want = {};
|
|
|
|
if (keep[unit]) {
|
|
keep[unit].forEach(u => {
|
|
want[u] = old[u]();
|
|
});
|
|
}
|
|
|
|
if (dstAwareUnits[unit]) {
|
|
const diff = old.timezone().current.offset - s.timezone().current.offset;
|
|
s.epoch += diff * 3600 * 1000;
|
|
} //ensure month/year has ticked-over
|
|
|
|
|
|
if (unit === 'month') {
|
|
want.month = old.month() + num; //month is the one unit we 'model' directly
|
|
|
|
want = model.months(want, old);
|
|
} //support coercing a week, too
|
|
|
|
|
|
if (unit === 'week') {
|
|
let sum = old.date() + num * 7;
|
|
|
|
if (sum <= 28 && sum > 1) {
|
|
want.date = sum;
|
|
}
|
|
}
|
|
|
|
if (unit === 'weekend' && s.dayName() !== 'saturday') {
|
|
s = s.day('saturday', true); //ensure it's saturday
|
|
} //support 25-hour day-changes on dst-changes
|
|
else if (unit === 'date') {
|
|
if (num < 0) {
|
|
want = model.daysBack(want, old, num);
|
|
} else {
|
|
//specify a naive date number, if it's easy to do...
|
|
let sum = old.date() + num; // ok, model this one too
|
|
|
|
want = model.days(want, old, sum);
|
|
} //manually punt it if we haven't moved at all..
|
|
|
|
|
|
if (num !== 0 && old.isSame(s, 'day')) {
|
|
want.date = old.date() + num;
|
|
}
|
|
} // ensure a quarter is 3 months over
|
|
else if (unit === 'quarter') {
|
|
want.month = old.month() + num * 3;
|
|
want.year = old.year(); // handle rollover
|
|
|
|
if (want.month < 0) {
|
|
let years = Math.floor(want.month / 12);
|
|
let remainder = want.month + Math.abs(years) * 12;
|
|
want.month = remainder;
|
|
want.year += years;
|
|
} else if (want.month >= 12) {
|
|
let years = Math.floor(want.month / 12);
|
|
want.month = want.month % 12;
|
|
want.year += years;
|
|
}
|
|
|
|
want.date = old.date();
|
|
} //ensure year has changed (leap-years)
|
|
else if (unit === 'year') {
|
|
let wantYear = old.year() + num;
|
|
let haveYear = s.year();
|
|
|
|
if (haveYear < wantYear) {
|
|
let toAdd = Math.floor(num / 4) || 1; //approx num of leap-days
|
|
|
|
s.epoch += Math.abs(ms.day * toAdd);
|
|
} else if (haveYear > wantYear) {
|
|
let toAdd = Math.floor(num / 4) || 1; //approx num of leap-days
|
|
|
|
s.epoch += ms.day * toAdd;
|
|
}
|
|
} //these are easier
|
|
else if (unit === 'decade') {
|
|
want.year = s.year() + 10;
|
|
} else if (unit === 'century') {
|
|
want.year = s.year() + 100;
|
|
} //keep current date, unless the month doesn't have it.
|
|
|
|
|
|
if (keepDate[unit]) {
|
|
let max = monthLength[want.month];
|
|
want.date = old.date();
|
|
|
|
if (want.date > max) {
|
|
want.date = max;
|
|
}
|
|
}
|
|
|
|
if (Object.keys(want).length > 1) {
|
|
walkTo(s, want);
|
|
}
|
|
|
|
return s;
|
|
}; //subtract is only add *-1
|
|
|
|
|
|
SpaceTime.prototype.subtract = function (num, unit) {
|
|
let s = this.clone();
|
|
return s.add(num * -1, unit);
|
|
}; //add aliases
|
|
|
|
|
|
SpaceTime.prototype.minus = SpaceTime.prototype.subtract;
|
|
SpaceTime.prototype.plus = SpaceTime.prototype.add;
|
|
};
|
|
|
|
var add = addMethods$3;
|
|
|
|
//make a string, for easy comparison between dates
|
|
const print = {
|
|
millisecond: s => {
|
|
return s.epoch;
|
|
},
|
|
second: s => {
|
|
return [s.year(), s.month(), s.date(), s.hour(), s.minute(), s.second()].join('-');
|
|
},
|
|
minute: s => {
|
|
return [s.year(), s.month(), s.date(), s.hour(), s.minute()].join('-');
|
|
},
|
|
hour: s => {
|
|
return [s.year(), s.month(), s.date(), s.hour()].join('-');
|
|
},
|
|
day: s => {
|
|
return [s.year(), s.month(), s.date()].join('-');
|
|
},
|
|
week: s => {
|
|
return [s.year(), s.week()].join('-');
|
|
},
|
|
month: s => {
|
|
return [s.year(), s.month()].join('-');
|
|
},
|
|
quarter: s => {
|
|
return [s.year(), s.quarter()].join('-');
|
|
},
|
|
year: s => {
|
|
return s.year();
|
|
}
|
|
};
|
|
print.date = print.day;
|
|
|
|
const addMethods$2 = SpaceTime => {
|
|
SpaceTime.prototype.isSame = function (b, unit, tzAware = true) {
|
|
let a = this;
|
|
|
|
if (!unit) {
|
|
return null;
|
|
} // support swapped params
|
|
|
|
|
|
if (typeof b === 'string' && typeof unit === 'object') {
|
|
let tmp = b;
|
|
b = unit;
|
|
unit = tmp;
|
|
}
|
|
|
|
if (typeof b === 'string' || typeof b === 'number') {
|
|
b = new SpaceTime(b, this.timezone.name);
|
|
} //support 'seconds' aswell as 'second'
|
|
|
|
|
|
unit = unit.replace(/s$/, ''); // make them the same timezone for proper comparison
|
|
|
|
if (tzAware === true && a.tz !== b.tz) {
|
|
b = b.clone();
|
|
b.tz = a.tz;
|
|
}
|
|
|
|
if (print[unit]) {
|
|
return print[unit](a) === print[unit](b);
|
|
}
|
|
|
|
return null;
|
|
};
|
|
};
|
|
|
|
var same = addMethods$2;
|
|
|
|
const addMethods$1 = SpaceTime => {
|
|
const methods = {
|
|
isAfter: function (d) {
|
|
d = fns.beADate(d, this);
|
|
let epoch = fns.getEpoch(d);
|
|
|
|
if (epoch === null) {
|
|
return null;
|
|
}
|
|
|
|
return this.epoch > epoch;
|
|
},
|
|
isBefore: function (d) {
|
|
d = fns.beADate(d, this);
|
|
let epoch = fns.getEpoch(d);
|
|
|
|
if (epoch === null) {
|
|
return null;
|
|
}
|
|
|
|
return this.epoch < epoch;
|
|
},
|
|
isEqual: function (d) {
|
|
d = fns.beADate(d, this);
|
|
let epoch = fns.getEpoch(d);
|
|
|
|
if (epoch === null) {
|
|
return null;
|
|
}
|
|
|
|
return this.epoch === epoch;
|
|
},
|
|
isBetween: function (start, end, isInclusive = false) {
|
|
start = fns.beADate(start, this);
|
|
end = fns.beADate(end, this);
|
|
let startEpoch = fns.getEpoch(start);
|
|
|
|
if (startEpoch === null) {
|
|
return null;
|
|
}
|
|
|
|
let endEpoch = fns.getEpoch(end);
|
|
|
|
if (endEpoch === null) {
|
|
return null;
|
|
}
|
|
|
|
if (isInclusive) {
|
|
return this.isBetween(start, end) || this.isEqual(start) || this.isEqual(end);
|
|
}
|
|
|
|
return startEpoch < this.epoch && this.epoch < endEpoch;
|
|
}
|
|
}; //hook them into proto
|
|
|
|
Object.keys(methods).forEach(k => {
|
|
SpaceTime.prototype[k] = methods[k];
|
|
});
|
|
};
|
|
|
|
var compare = addMethods$1;
|
|
|
|
const addMethods = SpaceTime => {
|
|
const methods = {
|
|
i18n: data => {
|
|
//change the day names
|
|
if (fns.isObject(data.days)) {
|
|
days.set(data.days);
|
|
} //change the month names
|
|
|
|
|
|
if (fns.isObject(data.months)) {
|
|
months$1.set(data.months);
|
|
} // change the the display style of the month / day names
|
|
|
|
|
|
if (fns.isBoolean(data.useTitleCase)) {
|
|
caseFormat.set(data.useTitleCase);
|
|
}
|
|
}
|
|
}; //hook them into proto
|
|
|
|
Object.keys(methods).forEach(k => {
|
|
SpaceTime.prototype[k] = methods[k];
|
|
});
|
|
};
|
|
|
|
var i18n = addMethods;
|
|
|
|
var quickOffset = quick;
|
|
|
|
var methods = methods_1;
|
|
|
|
var require$$1 = query;
|
|
|
|
var require$$2 = add;
|
|
|
|
var require$$3 = same;
|
|
|
|
var require$$4 = compare;
|
|
|
|
var require$$5 = i18n;
|
|
|
|
let timezones = require$$0$4; //fake timezone-support, for fakers (es5 class)
|
|
|
|
const SpaceTime = function (input, tz, options = {}) {
|
|
//the holy moment
|
|
this.epoch = null; //the shift for the given timezone
|
|
|
|
this.tz = findTz(tz, timezones); //whether to output warnings to console
|
|
|
|
this.silent = options.silent || true; // favour british interpretation of 02/02/2018, etc
|
|
|
|
this.british = options.dmy || options.british; //does the week start on sunday, or monday:
|
|
|
|
this._weekStart = 1; //default to monday
|
|
|
|
if (options.weekStart !== undefined) {
|
|
this._weekStart = options.weekStart;
|
|
} // the reference today date object, (for testing)
|
|
|
|
|
|
this._today = {};
|
|
|
|
if (options.today !== undefined) {
|
|
this._today = options.today;
|
|
} // dunno if this is a good idea, or not
|
|
// Object.defineProperty(this, 'parsers', {
|
|
// enumerable: false,
|
|
// writable: true,
|
|
// value: parsers
|
|
// })
|
|
//add getter/setters
|
|
|
|
|
|
Object.defineProperty(this, 'd', {
|
|
//return a js date object
|
|
get: function () {
|
|
let offset = quickOffset(this); //every computer is somewhere- get this computer's built-in offset
|
|
|
|
let bias = new Date(this.epoch).getTimezoneOffset() || 0; //movement
|
|
|
|
let shift = bias + offset * 60; //in minutes
|
|
|
|
shift = shift * 60 * 1000; //in ms
|
|
//remove this computer's offset
|
|
|
|
let epoch = this.epoch + shift;
|
|
let d = new Date(epoch);
|
|
return d;
|
|
}
|
|
}); //add this data on the object, to allow adding new timezones
|
|
|
|
Object.defineProperty(this, 'timezones', {
|
|
get: () => timezones,
|
|
set: obj => {
|
|
timezones = obj;
|
|
return obj;
|
|
}
|
|
}); //parse the various formats
|
|
|
|
let tmp = handleInput(this, input);
|
|
this.epoch = tmp.epoch;
|
|
}; //(add instance methods to prototype)
|
|
|
|
|
|
Object.keys(methods).forEach(k => {
|
|
SpaceTime.prototype[k] = methods[k];
|
|
}); // ¯\_(ツ)_/¯
|
|
|
|
SpaceTime.prototype.clone = function () {
|
|
return new SpaceTime(this.epoch, this.tz, {
|
|
silent: this.silent,
|
|
weekStart: this._weekStart,
|
|
today: this._today,
|
|
parsers: this.parsers
|
|
});
|
|
};
|
|
/**
|
|
* @deprecated use toNativeDate()
|
|
* @returns native date object at the same epoch
|
|
*/
|
|
|
|
|
|
SpaceTime.prototype.toLocalDate = function () {
|
|
return this.toNativeDate();
|
|
};
|
|
/**
|
|
* @returns native date object at the same epoch
|
|
*/
|
|
|
|
|
|
SpaceTime.prototype.toNativeDate = function () {
|
|
return new Date(this.epoch);
|
|
}; //append more methods
|
|
|
|
|
|
require$$1(SpaceTime);
|
|
require$$2(SpaceTime);
|
|
require$$3(SpaceTime);
|
|
require$$4(SpaceTime);
|
|
require$$5(SpaceTime);
|
|
var spacetime = SpaceTime;
|
|
|
|
var Spacetime = spacetime;
|
|
|
|
const whereIts$1 = (a, b) => {
|
|
let start = new Spacetime(null);
|
|
let end = new Spacetime(null);
|
|
start = start.time(a); //if b is undefined, use as 'within one hour'
|
|
|
|
if (b) {
|
|
end = end.time(b);
|
|
} else {
|
|
end = start.add(59, 'minutes');
|
|
}
|
|
|
|
let startHour = start.hour();
|
|
let endHour = end.hour();
|
|
let tzs = Object.keys(start.timezones).filter(tz => {
|
|
if (tz.indexOf('/') === -1) {
|
|
return false;
|
|
}
|
|
|
|
let m = new Spacetime(null, tz);
|
|
let hour = m.hour(); //do 'calendar-compare' not real-time-compare
|
|
|
|
if (hour >= startHour && hour <= endHour) {
|
|
//test minutes too, if applicable
|
|
if (hour === startHour && m.minute() < start.minute()) {
|
|
return false;
|
|
}
|
|
|
|
if (hour === endHour && m.minute() > end.minute()) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
});
|
|
return tzs;
|
|
};
|
|
|
|
var whereIts_1 = whereIts$1;
|
|
|
|
var _version = '6.16.3';
|
|
|
|
var whereIts = whereIts_1;
|
|
|
|
var version = _version;
|
|
|
|
const main = (input, tz, options) => new Spacetime(input, tz, options); // set all properties of a given 'today' object
|
|
|
|
|
|
const setToday = function (s) {
|
|
let today = s._today || {};
|
|
Object.keys(today).forEach(k => {
|
|
s = s[k](today[k]);
|
|
});
|
|
return s;
|
|
}; //some helper functions on the main method
|
|
|
|
|
|
main.now = (tz, options) => {
|
|
let s = new Spacetime(new Date().getTime(), tz, options);
|
|
s = setToday(s);
|
|
return s;
|
|
};
|
|
|
|
main.today = (tz, options) => {
|
|
let s = new Spacetime(new Date().getTime(), tz, options);
|
|
s = setToday(s);
|
|
return s.startOf('day');
|
|
};
|
|
|
|
main.tomorrow = (tz, options) => {
|
|
let s = new Spacetime(new Date().getTime(), tz, options);
|
|
s = setToday(s);
|
|
return s.add(1, 'day').startOf('day');
|
|
};
|
|
|
|
main.yesterday = (tz, options) => {
|
|
let s = new Spacetime(new Date().getTime(), tz, options);
|
|
s = setToday(s);
|
|
return s.subtract(1, 'day').startOf('day');
|
|
};
|
|
|
|
main.extend = function (obj = {}) {
|
|
Object.keys(obj).forEach(k => {
|
|
Spacetime.prototype[k] = obj[k];
|
|
});
|
|
return this;
|
|
};
|
|
|
|
main.timezones = function () {
|
|
let s = new Spacetime();
|
|
return s.timezones;
|
|
};
|
|
|
|
main.max = function (tz, options) {
|
|
let s = new Spacetime(null, tz, options);
|
|
s.epoch = 8640000000000000;
|
|
return s;
|
|
};
|
|
|
|
main.min = function (tz, options) {
|
|
let s = new Spacetime(null, tz, options);
|
|
s.epoch = -8640000000000000;
|
|
return s;
|
|
}; //find tz by time
|
|
|
|
|
|
main.whereIts = whereIts;
|
|
main.version = version; //aliases:
|
|
|
|
main.plugin = main.extend;
|
|
var src = main;
|
|
|
|
return src;
|
|
|
|
})));
|