feelin 4.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.gitignore +6 -0
- data/.rspec +2 -0
- data/CHANGELOG.md +10 -0
- data/Gemfile +3 -0
- data/LICENSE.txt +21 -0
- data/README.md +43 -0
- data/feelin.gemspec +22 -0
- data/lib/feelin/js/node_modules/.package-lock.json +67 -0
- data/lib/feelin/js/node_modules/@lezer/common/LICENSE +21 -0
- data/lib/feelin/js/node_modules/@lezer/common/README.md +14 -0
- data/lib/feelin/js/node_modules/@lezer/common/dist/index.cjs +2181 -0
- data/lib/feelin/js/node_modules/@lezer/common/dist/index.d.cts +1137 -0
- data/lib/feelin/js/node_modules/@lezer/common/dist/index.d.ts +1137 -0
- data/lib/feelin/js/node_modules/@lezer/common/dist/index.js +2168 -0
- data/lib/feelin/js/node_modules/@lezer/common/package.json +32 -0
- data/lib/feelin/js/node_modules/@lezer/highlight/LICENSE +21 -0
- data/lib/feelin/js/node_modules/@lezer/highlight/README.md +14 -0
- data/lib/feelin/js/node_modules/@lezer/highlight/dist/index.cjs +915 -0
- data/lib/feelin/js/node_modules/@lezer/highlight/dist/index.d.cts +621 -0
- data/lib/feelin/js/node_modules/@lezer/highlight/dist/index.d.ts +623 -0
- data/lib/feelin/js/node_modules/@lezer/highlight/dist/index.js +904 -0
- data/lib/feelin/js/node_modules/@lezer/highlight/package.json +31 -0
- data/lib/feelin/js/node_modules/@lezer/lr/LICENSE +21 -0
- data/lib/feelin/js/node_modules/@lezer/lr/README.md +25 -0
- data/lib/feelin/js/node_modules/@lezer/lr/dist/constants.d.ts +45 -0
- data/lib/feelin/js/node_modules/@lezer/lr/dist/constants.js +5 -0
- data/lib/feelin/js/node_modules/@lezer/lr/dist/index.cjs +1890 -0
- data/lib/feelin/js/node_modules/@lezer/lr/dist/index.d.cts +303 -0
- data/lib/feelin/js/node_modules/@lezer/lr/dist/index.d.ts +303 -0
- data/lib/feelin/js/node_modules/@lezer/lr/dist/index.js +1883 -0
- data/lib/feelin/js/node_modules/@lezer/lr/package.json +32 -0
- data/lib/feelin/js/node_modules/feelin/LICENSE +21 -0
- data/lib/feelin/js/node_modules/feelin/README.md +65 -0
- data/lib/feelin/js/node_modules/feelin/dist/builtins.d.ts +355 -0
- data/lib/feelin/js/node_modules/feelin/dist/index.cjs +2072 -0
- data/lib/feelin/js/node_modules/feelin/dist/index.cjs.map +1 -0
- data/lib/feelin/js/node_modules/feelin/dist/index.d.ts +3 -0
- data/lib/feelin/js/node_modules/feelin/dist/index.esm.js +2063 -0
- data/lib/feelin/js/node_modules/feelin/dist/index.esm.js.map +1 -0
- data/lib/feelin/js/node_modules/feelin/dist/interpreter.d.ts +26 -0
- data/lib/feelin/js/node_modules/feelin/dist/parser.d.ts +4 -0
- data/lib/feelin/js/node_modules/feelin/dist/temporal.d.ts +6 -0
- data/lib/feelin/js/node_modules/feelin/dist/types.d.ts +35 -0
- data/lib/feelin/js/node_modules/feelin/dist/utils.d.ts +12 -0
- data/lib/feelin/js/node_modules/feelin/package.json +63 -0
- data/lib/feelin/js/node_modules/lezer-feel/LICENSE +21 -0
- data/lib/feelin/js/node_modules/lezer-feel/README.md +94 -0
- data/lib/feelin/js/node_modules/lezer-feel/dist/index.cjs +1328 -0
- data/lib/feelin/js/node_modules/lezer-feel/dist/index.cjs.map +1 -0
- data/lib/feelin/js/node_modules/lezer-feel/dist/index.d.ts +32 -0
- data/lib/feelin/js/node_modules/lezer-feel/dist/index.js +1323 -0
- data/lib/feelin/js/node_modules/lezer-feel/dist/index.js.map +1 -0
- data/lib/feelin/js/node_modules/lezer-feel/package.json +61 -0
- data/lib/feelin/js/node_modules/luxon/LICENSE.md +7 -0
- data/lib/feelin/js/node_modules/luxon/README.md +55 -0
- data/lib/feelin/js/node_modules/luxon/build/amd/luxon.js +8623 -0
- data/lib/feelin/js/node_modules/luxon/build/amd/luxon.js.map +1 -0
- data/lib/feelin/js/node_modules/luxon/build/cjs-browser/luxon.js +8621 -0
- data/lib/feelin/js/node_modules/luxon/build/cjs-browser/luxon.js.map +1 -0
- data/lib/feelin/js/node_modules/luxon/build/es6/luxon.js +8011 -0
- data/lib/feelin/js/node_modules/luxon/build/es6/luxon.js.map +1 -0
- data/lib/feelin/js/node_modules/luxon/build/global/luxon.js +8626 -0
- data/lib/feelin/js/node_modules/luxon/build/global/luxon.js.map +1 -0
- data/lib/feelin/js/node_modules/luxon/build/global/luxon.min.js +1 -0
- data/lib/feelin/js/node_modules/luxon/build/global/luxon.min.js.map +1 -0
- data/lib/feelin/js/node_modules/luxon/build/node/luxon.js +7679 -0
- data/lib/feelin/js/node_modules/luxon/build/node/luxon.js.map +1 -0
- data/lib/feelin/js/node_modules/luxon/package.json +87 -0
- data/lib/feelin/js/node_modules/luxon/src/datetime.js +2566 -0
- data/lib/feelin/js/node_modules/luxon/src/duration.js +990 -0
- data/lib/feelin/js/node_modules/luxon/src/errors.js +61 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/conversions.js +206 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/diff.js +95 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/digits.js +90 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/english.js +233 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/formats.js +176 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/formatter.js +409 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/invalid.js +14 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/locale.js +554 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/regexParser.js +335 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/tokenParser.js +505 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/util.js +316 -0
- data/lib/feelin/js/node_modules/luxon/src/impl/zoneUtil.js +34 -0
- data/lib/feelin/js/node_modules/luxon/src/info.js +205 -0
- data/lib/feelin/js/node_modules/luxon/src/interval.js +665 -0
- data/lib/feelin/js/node_modules/luxon/src/luxon.js +26 -0
- data/lib/feelin/js/node_modules/luxon/src/package.json +4 -0
- data/lib/feelin/js/node_modules/luxon/src/settings.js +180 -0
- data/lib/feelin/js/node_modules/luxon/src/zone.js +97 -0
- data/lib/feelin/js/node_modules/luxon/src/zones/IANAZone.js +231 -0
- data/lib/feelin/js/node_modules/luxon/src/zones/fixedOffsetZone.js +150 -0
- data/lib/feelin/js/node_modules/luxon/src/zones/invalidZone.js +53 -0
- data/lib/feelin/js/node_modules/luxon/src/zones/systemZone.js +61 -0
- data/lib/feelin/js/node_modules/min-dash/LICENSE +21 -0
- data/lib/feelin/js/node_modules/min-dash/README.md +38 -0
- data/lib/feelin/js/node_modules/min-dash/dist/array.d.ts +12 -0
- data/lib/feelin/js/node_modules/min-dash/dist/collection.d.ts +174 -0
- data/lib/feelin/js/node_modules/min-dash/dist/fn.d.ts +37 -0
- data/lib/feelin/js/node_modules/min-dash/dist/index.cjs +910 -0
- data/lib/feelin/js/node_modules/min-dash/dist/index.d.ts +5 -0
- data/lib/feelin/js/node_modules/min-dash/dist/index.esm.js +872 -0
- data/lib/feelin/js/node_modules/min-dash/dist/lang.d.ts +29 -0
- data/lib/feelin/js/node_modules/min-dash/dist/min-dash.js +916 -0
- data/lib/feelin/js/node_modules/min-dash/dist/min-dash.min.js +1 -0
- data/lib/feelin/js/node_modules/min-dash/dist/object.d.ts +112 -0
- data/lib/feelin/js/node_modules/min-dash/package.json +72 -0
- data/lib/feelin/js/package-lock.json +72 -0
- data/lib/feelin/js/package.json +5 -0
- data/lib/feelin/version.rb +3 -0
- data/lib/feelin.rb +63 -0
- data/spec/feelin/feelin_spec.rb +38 -0
- data/spec/spec_helper.rb +2 -0
- metadata +198 -0
@@ -0,0 +1,61 @@
|
|
1
|
+
// these aren't really private, but nor are they really useful to document
|
2
|
+
|
3
|
+
/**
|
4
|
+
* @private
|
5
|
+
*/
|
6
|
+
class LuxonError extends Error {}
|
7
|
+
|
8
|
+
/**
|
9
|
+
* @private
|
10
|
+
*/
|
11
|
+
export class InvalidDateTimeError extends LuxonError {
|
12
|
+
constructor(reason) {
|
13
|
+
super(`Invalid DateTime: ${reason.toMessage()}`);
|
14
|
+
}
|
15
|
+
}
|
16
|
+
|
17
|
+
/**
|
18
|
+
* @private
|
19
|
+
*/
|
20
|
+
export class InvalidIntervalError extends LuxonError {
|
21
|
+
constructor(reason) {
|
22
|
+
super(`Invalid Interval: ${reason.toMessage()}`);
|
23
|
+
}
|
24
|
+
}
|
25
|
+
|
26
|
+
/**
|
27
|
+
* @private
|
28
|
+
*/
|
29
|
+
export class InvalidDurationError extends LuxonError {
|
30
|
+
constructor(reason) {
|
31
|
+
super(`Invalid Duration: ${reason.toMessage()}`);
|
32
|
+
}
|
33
|
+
}
|
34
|
+
|
35
|
+
/**
|
36
|
+
* @private
|
37
|
+
*/
|
38
|
+
export class ConflictingSpecificationError extends LuxonError {}
|
39
|
+
|
40
|
+
/**
|
41
|
+
* @private
|
42
|
+
*/
|
43
|
+
export class InvalidUnitError extends LuxonError {
|
44
|
+
constructor(unit) {
|
45
|
+
super(`Invalid unit ${unit}`);
|
46
|
+
}
|
47
|
+
}
|
48
|
+
|
49
|
+
/**
|
50
|
+
* @private
|
51
|
+
*/
|
52
|
+
export class InvalidArgumentError extends LuxonError {}
|
53
|
+
|
54
|
+
/**
|
55
|
+
* @private
|
56
|
+
*/
|
57
|
+
export class ZoneIsAbstractError extends LuxonError {
|
58
|
+
constructor() {
|
59
|
+
super("Zone is an abstract class");
|
60
|
+
}
|
61
|
+
}
|
@@ -0,0 +1,206 @@
|
|
1
|
+
import {
|
2
|
+
integerBetween,
|
3
|
+
isLeapYear,
|
4
|
+
timeObject,
|
5
|
+
daysInYear,
|
6
|
+
daysInMonth,
|
7
|
+
weeksInWeekYear,
|
8
|
+
isInteger,
|
9
|
+
isUndefined,
|
10
|
+
} from "./util.js";
|
11
|
+
import Invalid from "./invalid.js";
|
12
|
+
import { ConflictingSpecificationError } from "../errors.js";
|
13
|
+
|
14
|
+
const nonLeapLadder = [0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334],
|
15
|
+
leapLadder = [0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335];
|
16
|
+
|
17
|
+
function unitOutOfRange(unit, value) {
|
18
|
+
return new Invalid(
|
19
|
+
"unit out of range",
|
20
|
+
`you specified ${value} (of type ${typeof value}) as a ${unit}, which is invalid`
|
21
|
+
);
|
22
|
+
}
|
23
|
+
|
24
|
+
export function dayOfWeek(year, month, day) {
|
25
|
+
const d = new Date(Date.UTC(year, month - 1, day));
|
26
|
+
|
27
|
+
if (year < 100 && year >= 0) {
|
28
|
+
d.setUTCFullYear(d.getUTCFullYear() - 1900);
|
29
|
+
}
|
30
|
+
|
31
|
+
const js = d.getUTCDay();
|
32
|
+
|
33
|
+
return js === 0 ? 7 : js;
|
34
|
+
}
|
35
|
+
|
36
|
+
function computeOrdinal(year, month, day) {
|
37
|
+
return day + (isLeapYear(year) ? leapLadder : nonLeapLadder)[month - 1];
|
38
|
+
}
|
39
|
+
|
40
|
+
function uncomputeOrdinal(year, ordinal) {
|
41
|
+
const table = isLeapYear(year) ? leapLadder : nonLeapLadder,
|
42
|
+
month0 = table.findIndex((i) => i < ordinal),
|
43
|
+
day = ordinal - table[month0];
|
44
|
+
return { month: month0 + 1, day };
|
45
|
+
}
|
46
|
+
|
47
|
+
export function isoWeekdayToLocal(isoWeekday, startOfWeek) {
|
48
|
+
return ((isoWeekday - startOfWeek + 7) % 7) + 1;
|
49
|
+
}
|
50
|
+
|
51
|
+
/**
|
52
|
+
* @private
|
53
|
+
*/
|
54
|
+
|
55
|
+
export function gregorianToWeek(gregObj, minDaysInFirstWeek = 4, startOfWeek = 1) {
|
56
|
+
const { year, month, day } = gregObj,
|
57
|
+
ordinal = computeOrdinal(year, month, day),
|
58
|
+
weekday = isoWeekdayToLocal(dayOfWeek(year, month, day), startOfWeek);
|
59
|
+
|
60
|
+
let weekNumber = Math.floor((ordinal - weekday + 14 - minDaysInFirstWeek) / 7),
|
61
|
+
weekYear;
|
62
|
+
|
63
|
+
if (weekNumber < 1) {
|
64
|
+
weekYear = year - 1;
|
65
|
+
weekNumber = weeksInWeekYear(weekYear, minDaysInFirstWeek, startOfWeek);
|
66
|
+
} else if (weekNumber > weeksInWeekYear(year, minDaysInFirstWeek, startOfWeek)) {
|
67
|
+
weekYear = year + 1;
|
68
|
+
weekNumber = 1;
|
69
|
+
} else {
|
70
|
+
weekYear = year;
|
71
|
+
}
|
72
|
+
|
73
|
+
return { weekYear, weekNumber, weekday, ...timeObject(gregObj) };
|
74
|
+
}
|
75
|
+
|
76
|
+
export function weekToGregorian(weekData, minDaysInFirstWeek = 4, startOfWeek = 1) {
|
77
|
+
const { weekYear, weekNumber, weekday } = weekData,
|
78
|
+
weekdayOfJan4 = isoWeekdayToLocal(dayOfWeek(weekYear, 1, minDaysInFirstWeek), startOfWeek),
|
79
|
+
yearInDays = daysInYear(weekYear);
|
80
|
+
|
81
|
+
let ordinal = weekNumber * 7 + weekday - weekdayOfJan4 - 7 + minDaysInFirstWeek,
|
82
|
+
year;
|
83
|
+
|
84
|
+
if (ordinal < 1) {
|
85
|
+
year = weekYear - 1;
|
86
|
+
ordinal += daysInYear(year);
|
87
|
+
} else if (ordinal > yearInDays) {
|
88
|
+
year = weekYear + 1;
|
89
|
+
ordinal -= daysInYear(weekYear);
|
90
|
+
} else {
|
91
|
+
year = weekYear;
|
92
|
+
}
|
93
|
+
|
94
|
+
const { month, day } = uncomputeOrdinal(year, ordinal);
|
95
|
+
return { year, month, day, ...timeObject(weekData) };
|
96
|
+
}
|
97
|
+
|
98
|
+
export function gregorianToOrdinal(gregData) {
|
99
|
+
const { year, month, day } = gregData;
|
100
|
+
const ordinal = computeOrdinal(year, month, day);
|
101
|
+
return { year, ordinal, ...timeObject(gregData) };
|
102
|
+
}
|
103
|
+
|
104
|
+
export function ordinalToGregorian(ordinalData) {
|
105
|
+
const { year, ordinal } = ordinalData;
|
106
|
+
const { month, day } = uncomputeOrdinal(year, ordinal);
|
107
|
+
return { year, month, day, ...timeObject(ordinalData) };
|
108
|
+
}
|
109
|
+
|
110
|
+
/**
|
111
|
+
* Check if local week units like localWeekday are used in obj.
|
112
|
+
* If so, validates that they are not mixed with ISO week units and then copies them to the normal week unit properties.
|
113
|
+
* Modifies obj in-place!
|
114
|
+
* @param obj the object values
|
115
|
+
*/
|
116
|
+
export function usesLocalWeekValues(obj, loc) {
|
117
|
+
const hasLocaleWeekData =
|
118
|
+
!isUndefined(obj.localWeekday) ||
|
119
|
+
!isUndefined(obj.localWeekNumber) ||
|
120
|
+
!isUndefined(obj.localWeekYear);
|
121
|
+
if (hasLocaleWeekData) {
|
122
|
+
const hasIsoWeekData =
|
123
|
+
!isUndefined(obj.weekday) || !isUndefined(obj.weekNumber) || !isUndefined(obj.weekYear);
|
124
|
+
|
125
|
+
if (hasIsoWeekData) {
|
126
|
+
throw new ConflictingSpecificationError(
|
127
|
+
"Cannot mix locale-based week fields with ISO-based week fields"
|
128
|
+
);
|
129
|
+
}
|
130
|
+
if (!isUndefined(obj.localWeekday)) obj.weekday = obj.localWeekday;
|
131
|
+
if (!isUndefined(obj.localWeekNumber)) obj.weekNumber = obj.localWeekNumber;
|
132
|
+
if (!isUndefined(obj.localWeekYear)) obj.weekYear = obj.localWeekYear;
|
133
|
+
delete obj.localWeekday;
|
134
|
+
delete obj.localWeekNumber;
|
135
|
+
delete obj.localWeekYear;
|
136
|
+
return {
|
137
|
+
minDaysInFirstWeek: loc.getMinDaysInFirstWeek(),
|
138
|
+
startOfWeek: loc.getStartOfWeek(),
|
139
|
+
};
|
140
|
+
} else {
|
141
|
+
return { minDaysInFirstWeek: 4, startOfWeek: 1 };
|
142
|
+
}
|
143
|
+
}
|
144
|
+
|
145
|
+
export function hasInvalidWeekData(obj, minDaysInFirstWeek = 4, startOfWeek = 1) {
|
146
|
+
const validYear = isInteger(obj.weekYear),
|
147
|
+
validWeek = integerBetween(
|
148
|
+
obj.weekNumber,
|
149
|
+
1,
|
150
|
+
weeksInWeekYear(obj.weekYear, minDaysInFirstWeek, startOfWeek)
|
151
|
+
),
|
152
|
+
validWeekday = integerBetween(obj.weekday, 1, 7);
|
153
|
+
|
154
|
+
if (!validYear) {
|
155
|
+
return unitOutOfRange("weekYear", obj.weekYear);
|
156
|
+
} else if (!validWeek) {
|
157
|
+
return unitOutOfRange("week", obj.weekNumber);
|
158
|
+
} else if (!validWeekday) {
|
159
|
+
return unitOutOfRange("weekday", obj.weekday);
|
160
|
+
} else return false;
|
161
|
+
}
|
162
|
+
|
163
|
+
export function hasInvalidOrdinalData(obj) {
|
164
|
+
const validYear = isInteger(obj.year),
|
165
|
+
validOrdinal = integerBetween(obj.ordinal, 1, daysInYear(obj.year));
|
166
|
+
|
167
|
+
if (!validYear) {
|
168
|
+
return unitOutOfRange("year", obj.year);
|
169
|
+
} else if (!validOrdinal) {
|
170
|
+
return unitOutOfRange("ordinal", obj.ordinal);
|
171
|
+
} else return false;
|
172
|
+
}
|
173
|
+
|
174
|
+
export function hasInvalidGregorianData(obj) {
|
175
|
+
const validYear = isInteger(obj.year),
|
176
|
+
validMonth = integerBetween(obj.month, 1, 12),
|
177
|
+
validDay = integerBetween(obj.day, 1, daysInMonth(obj.year, obj.month));
|
178
|
+
|
179
|
+
if (!validYear) {
|
180
|
+
return unitOutOfRange("year", obj.year);
|
181
|
+
} else if (!validMonth) {
|
182
|
+
return unitOutOfRange("month", obj.month);
|
183
|
+
} else if (!validDay) {
|
184
|
+
return unitOutOfRange("day", obj.day);
|
185
|
+
} else return false;
|
186
|
+
}
|
187
|
+
|
188
|
+
export function hasInvalidTimeData(obj) {
|
189
|
+
const { hour, minute, second, millisecond } = obj;
|
190
|
+
const validHour =
|
191
|
+
integerBetween(hour, 0, 23) ||
|
192
|
+
(hour === 24 && minute === 0 && second === 0 && millisecond === 0),
|
193
|
+
validMinute = integerBetween(minute, 0, 59),
|
194
|
+
validSecond = integerBetween(second, 0, 59),
|
195
|
+
validMillisecond = integerBetween(millisecond, 0, 999);
|
196
|
+
|
197
|
+
if (!validHour) {
|
198
|
+
return unitOutOfRange("hour", hour);
|
199
|
+
} else if (!validMinute) {
|
200
|
+
return unitOutOfRange("minute", minute);
|
201
|
+
} else if (!validSecond) {
|
202
|
+
return unitOutOfRange("second", second);
|
203
|
+
} else if (!validMillisecond) {
|
204
|
+
return unitOutOfRange("millisecond", millisecond);
|
205
|
+
} else return false;
|
206
|
+
}
|
@@ -0,0 +1,95 @@
|
|
1
|
+
import Duration from "../duration.js";
|
2
|
+
|
3
|
+
function dayDiff(earlier, later) {
|
4
|
+
const utcDayStart = (dt) => dt.toUTC(0, { keepLocalTime: true }).startOf("day").valueOf(),
|
5
|
+
ms = utcDayStart(later) - utcDayStart(earlier);
|
6
|
+
return Math.floor(Duration.fromMillis(ms).as("days"));
|
7
|
+
}
|
8
|
+
|
9
|
+
function highOrderDiffs(cursor, later, units) {
|
10
|
+
const differs = [
|
11
|
+
["years", (a, b) => b.year - a.year],
|
12
|
+
["quarters", (a, b) => b.quarter - a.quarter + (b.year - a.year) * 4],
|
13
|
+
["months", (a, b) => b.month - a.month + (b.year - a.year) * 12],
|
14
|
+
[
|
15
|
+
"weeks",
|
16
|
+
(a, b) => {
|
17
|
+
const days = dayDiff(a, b);
|
18
|
+
return (days - (days % 7)) / 7;
|
19
|
+
},
|
20
|
+
],
|
21
|
+
["days", dayDiff],
|
22
|
+
];
|
23
|
+
|
24
|
+
const results = {};
|
25
|
+
const earlier = cursor;
|
26
|
+
let lowestOrder, highWater;
|
27
|
+
|
28
|
+
/* This loop tries to diff using larger units first.
|
29
|
+
If we overshoot, we backtrack and try the next smaller unit.
|
30
|
+
"cursor" starts out at the earlier timestamp and moves closer and closer to "later"
|
31
|
+
as we use smaller and smaller units.
|
32
|
+
highWater keeps track of where we would be if we added one more of the smallest unit,
|
33
|
+
this is used later to potentially convert any difference smaller than the smallest higher order unit
|
34
|
+
into a fraction of that smallest higher order unit
|
35
|
+
*/
|
36
|
+
for (const [unit, differ] of differs) {
|
37
|
+
if (units.indexOf(unit) >= 0) {
|
38
|
+
lowestOrder = unit;
|
39
|
+
|
40
|
+
results[unit] = differ(cursor, later);
|
41
|
+
highWater = earlier.plus(results);
|
42
|
+
|
43
|
+
if (highWater > later) {
|
44
|
+
// we overshot the end point, backtrack cursor by 1
|
45
|
+
results[unit]--;
|
46
|
+
cursor = earlier.plus(results);
|
47
|
+
|
48
|
+
// if we are still overshooting now, we need to backtrack again
|
49
|
+
// this happens in certain situations when diffing times in different zones,
|
50
|
+
// because this calculation ignores time zones
|
51
|
+
if (cursor > later) {
|
52
|
+
// keep the "overshot by 1" around as highWater
|
53
|
+
highWater = cursor;
|
54
|
+
// backtrack cursor by 1
|
55
|
+
results[unit]--;
|
56
|
+
cursor = earlier.plus(results);
|
57
|
+
}
|
58
|
+
} else {
|
59
|
+
cursor = highWater;
|
60
|
+
}
|
61
|
+
}
|
62
|
+
}
|
63
|
+
|
64
|
+
return [cursor, results, highWater, lowestOrder];
|
65
|
+
}
|
66
|
+
|
67
|
+
export default function (earlier, later, units, opts) {
|
68
|
+
let [cursor, results, highWater, lowestOrder] = highOrderDiffs(earlier, later, units);
|
69
|
+
|
70
|
+
const remainingMillis = later - cursor;
|
71
|
+
|
72
|
+
const lowerOrderUnits = units.filter(
|
73
|
+
(u) => ["hours", "minutes", "seconds", "milliseconds"].indexOf(u) >= 0
|
74
|
+
);
|
75
|
+
|
76
|
+
if (lowerOrderUnits.length === 0) {
|
77
|
+
if (highWater < later) {
|
78
|
+
highWater = cursor.plus({ [lowestOrder]: 1 });
|
79
|
+
}
|
80
|
+
|
81
|
+
if (highWater !== cursor) {
|
82
|
+
results[lowestOrder] = (results[lowestOrder] || 0) + remainingMillis / (highWater - cursor);
|
83
|
+
}
|
84
|
+
}
|
85
|
+
|
86
|
+
const duration = Duration.fromObject(results, opts);
|
87
|
+
|
88
|
+
if (lowerOrderUnits.length > 0) {
|
89
|
+
return Duration.fromMillis(remainingMillis, opts)
|
90
|
+
.shiftTo(...lowerOrderUnits)
|
91
|
+
.plus(duration);
|
92
|
+
} else {
|
93
|
+
return duration;
|
94
|
+
}
|
95
|
+
}
|
@@ -0,0 +1,90 @@
|
|
1
|
+
const numberingSystems = {
|
2
|
+
arab: "[\u0660-\u0669]",
|
3
|
+
arabext: "[\u06F0-\u06F9]",
|
4
|
+
bali: "[\u1B50-\u1B59]",
|
5
|
+
beng: "[\u09E6-\u09EF]",
|
6
|
+
deva: "[\u0966-\u096F]",
|
7
|
+
fullwide: "[\uFF10-\uFF19]",
|
8
|
+
gujr: "[\u0AE6-\u0AEF]",
|
9
|
+
hanidec: "[〇|一|二|三|四|五|六|七|八|九]",
|
10
|
+
khmr: "[\u17E0-\u17E9]",
|
11
|
+
knda: "[\u0CE6-\u0CEF]",
|
12
|
+
laoo: "[\u0ED0-\u0ED9]",
|
13
|
+
limb: "[\u1946-\u194F]",
|
14
|
+
mlym: "[\u0D66-\u0D6F]",
|
15
|
+
mong: "[\u1810-\u1819]",
|
16
|
+
mymr: "[\u1040-\u1049]",
|
17
|
+
orya: "[\u0B66-\u0B6F]",
|
18
|
+
tamldec: "[\u0BE6-\u0BEF]",
|
19
|
+
telu: "[\u0C66-\u0C6F]",
|
20
|
+
thai: "[\u0E50-\u0E59]",
|
21
|
+
tibt: "[\u0F20-\u0F29]",
|
22
|
+
latn: "\\d",
|
23
|
+
};
|
24
|
+
|
25
|
+
const numberingSystemsUTF16 = {
|
26
|
+
arab: [1632, 1641],
|
27
|
+
arabext: [1776, 1785],
|
28
|
+
bali: [6992, 7001],
|
29
|
+
beng: [2534, 2543],
|
30
|
+
deva: [2406, 2415],
|
31
|
+
fullwide: [65296, 65303],
|
32
|
+
gujr: [2790, 2799],
|
33
|
+
khmr: [6112, 6121],
|
34
|
+
knda: [3302, 3311],
|
35
|
+
laoo: [3792, 3801],
|
36
|
+
limb: [6470, 6479],
|
37
|
+
mlym: [3430, 3439],
|
38
|
+
mong: [6160, 6169],
|
39
|
+
mymr: [4160, 4169],
|
40
|
+
orya: [2918, 2927],
|
41
|
+
tamldec: [3046, 3055],
|
42
|
+
telu: [3174, 3183],
|
43
|
+
thai: [3664, 3673],
|
44
|
+
tibt: [3872, 3881],
|
45
|
+
};
|
46
|
+
|
47
|
+
const hanidecChars = numberingSystems.hanidec.replace(/[\[|\]]/g, "").split("");
|
48
|
+
|
49
|
+
export function parseDigits(str) {
|
50
|
+
let value = parseInt(str, 10);
|
51
|
+
if (isNaN(value)) {
|
52
|
+
value = "";
|
53
|
+
for (let i = 0; i < str.length; i++) {
|
54
|
+
const code = str.charCodeAt(i);
|
55
|
+
|
56
|
+
if (str[i].search(numberingSystems.hanidec) !== -1) {
|
57
|
+
value += hanidecChars.indexOf(str[i]);
|
58
|
+
} else {
|
59
|
+
for (const key in numberingSystemsUTF16) {
|
60
|
+
const [min, max] = numberingSystemsUTF16[key];
|
61
|
+
if (code >= min && code <= max) {
|
62
|
+
value += code - min;
|
63
|
+
}
|
64
|
+
}
|
65
|
+
}
|
66
|
+
}
|
67
|
+
return parseInt(value, 10);
|
68
|
+
} else {
|
69
|
+
return value;
|
70
|
+
}
|
71
|
+
}
|
72
|
+
|
73
|
+
// cache of {numberingSystem: {append: regex}}
|
74
|
+
let digitRegexCache = {};
|
75
|
+
export function resetDigitRegexCache() {
|
76
|
+
digitRegexCache = {};
|
77
|
+
}
|
78
|
+
|
79
|
+
export function digitRegex({ numberingSystem }, append = "") {
|
80
|
+
const ns = numberingSystem || "latn";
|
81
|
+
|
82
|
+
if (!digitRegexCache[ns]) {
|
83
|
+
digitRegexCache[ns] = {};
|
84
|
+
}
|
85
|
+
if (!digitRegexCache[ns][append]) {
|
86
|
+
digitRegexCache[ns][append] = new RegExp(`${numberingSystems[ns]}${append}`);
|
87
|
+
}
|
88
|
+
|
89
|
+
return digitRegexCache[ns][append];
|
90
|
+
}
|
@@ -0,0 +1,233 @@
|
|
1
|
+
import * as Formats from "./formats.js";
|
2
|
+
import { pick } from "./util.js";
|
3
|
+
|
4
|
+
function stringify(obj) {
|
5
|
+
return JSON.stringify(obj, Object.keys(obj).sort());
|
6
|
+
}
|
7
|
+
|
8
|
+
/**
|
9
|
+
* @private
|
10
|
+
*/
|
11
|
+
|
12
|
+
export const monthsLong = [
|
13
|
+
"January",
|
14
|
+
"February",
|
15
|
+
"March",
|
16
|
+
"April",
|
17
|
+
"May",
|
18
|
+
"June",
|
19
|
+
"July",
|
20
|
+
"August",
|
21
|
+
"September",
|
22
|
+
"October",
|
23
|
+
"November",
|
24
|
+
"December",
|
25
|
+
];
|
26
|
+
|
27
|
+
export const monthsShort = [
|
28
|
+
"Jan",
|
29
|
+
"Feb",
|
30
|
+
"Mar",
|
31
|
+
"Apr",
|
32
|
+
"May",
|
33
|
+
"Jun",
|
34
|
+
"Jul",
|
35
|
+
"Aug",
|
36
|
+
"Sep",
|
37
|
+
"Oct",
|
38
|
+
"Nov",
|
39
|
+
"Dec",
|
40
|
+
];
|
41
|
+
|
42
|
+
export const monthsNarrow = ["J", "F", "M", "A", "M", "J", "J", "A", "S", "O", "N", "D"];
|
43
|
+
|
44
|
+
export function months(length) {
|
45
|
+
switch (length) {
|
46
|
+
case "narrow":
|
47
|
+
return [...monthsNarrow];
|
48
|
+
case "short":
|
49
|
+
return [...monthsShort];
|
50
|
+
case "long":
|
51
|
+
return [...monthsLong];
|
52
|
+
case "numeric":
|
53
|
+
return ["1", "2", "3", "4", "5", "6", "7", "8", "9", "10", "11", "12"];
|
54
|
+
case "2-digit":
|
55
|
+
return ["01", "02", "03", "04", "05", "06", "07", "08", "09", "10", "11", "12"];
|
56
|
+
default:
|
57
|
+
return null;
|
58
|
+
}
|
59
|
+
}
|
60
|
+
|
61
|
+
export const weekdaysLong = [
|
62
|
+
"Monday",
|
63
|
+
"Tuesday",
|
64
|
+
"Wednesday",
|
65
|
+
"Thursday",
|
66
|
+
"Friday",
|
67
|
+
"Saturday",
|
68
|
+
"Sunday",
|
69
|
+
];
|
70
|
+
|
71
|
+
export const weekdaysShort = ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"];
|
72
|
+
|
73
|
+
export const weekdaysNarrow = ["M", "T", "W", "T", "F", "S", "S"];
|
74
|
+
|
75
|
+
export function weekdays(length) {
|
76
|
+
switch (length) {
|
77
|
+
case "narrow":
|
78
|
+
return [...weekdaysNarrow];
|
79
|
+
case "short":
|
80
|
+
return [...weekdaysShort];
|
81
|
+
case "long":
|
82
|
+
return [...weekdaysLong];
|
83
|
+
case "numeric":
|
84
|
+
return ["1", "2", "3", "4", "5", "6", "7"];
|
85
|
+
default:
|
86
|
+
return null;
|
87
|
+
}
|
88
|
+
}
|
89
|
+
|
90
|
+
export const meridiems = ["AM", "PM"];
|
91
|
+
|
92
|
+
export const erasLong = ["Before Christ", "Anno Domini"];
|
93
|
+
|
94
|
+
export const erasShort = ["BC", "AD"];
|
95
|
+
|
96
|
+
export const erasNarrow = ["B", "A"];
|
97
|
+
|
98
|
+
export function eras(length) {
|
99
|
+
switch (length) {
|
100
|
+
case "narrow":
|
101
|
+
return [...erasNarrow];
|
102
|
+
case "short":
|
103
|
+
return [...erasShort];
|
104
|
+
case "long":
|
105
|
+
return [...erasLong];
|
106
|
+
default:
|
107
|
+
return null;
|
108
|
+
}
|
109
|
+
}
|
110
|
+
|
111
|
+
export function meridiemForDateTime(dt) {
|
112
|
+
return meridiems[dt.hour < 12 ? 0 : 1];
|
113
|
+
}
|
114
|
+
|
115
|
+
export function weekdayForDateTime(dt, length) {
|
116
|
+
return weekdays(length)[dt.weekday - 1];
|
117
|
+
}
|
118
|
+
|
119
|
+
export function monthForDateTime(dt, length) {
|
120
|
+
return months(length)[dt.month - 1];
|
121
|
+
}
|
122
|
+
|
123
|
+
export function eraForDateTime(dt, length) {
|
124
|
+
return eras(length)[dt.year < 0 ? 0 : 1];
|
125
|
+
}
|
126
|
+
|
127
|
+
export function formatRelativeTime(unit, count, numeric = "always", narrow = false) {
|
128
|
+
const units = {
|
129
|
+
years: ["year", "yr."],
|
130
|
+
quarters: ["quarter", "qtr."],
|
131
|
+
months: ["month", "mo."],
|
132
|
+
weeks: ["week", "wk."],
|
133
|
+
days: ["day", "day", "days"],
|
134
|
+
hours: ["hour", "hr."],
|
135
|
+
minutes: ["minute", "min."],
|
136
|
+
seconds: ["second", "sec."],
|
137
|
+
};
|
138
|
+
|
139
|
+
const lastable = ["hours", "minutes", "seconds"].indexOf(unit) === -1;
|
140
|
+
|
141
|
+
if (numeric === "auto" && lastable) {
|
142
|
+
const isDay = unit === "days";
|
143
|
+
switch (count) {
|
144
|
+
case 1:
|
145
|
+
return isDay ? "tomorrow" : `next ${units[unit][0]}`;
|
146
|
+
case -1:
|
147
|
+
return isDay ? "yesterday" : `last ${units[unit][0]}`;
|
148
|
+
case 0:
|
149
|
+
return isDay ? "today" : `this ${units[unit][0]}`;
|
150
|
+
default: // fall through
|
151
|
+
}
|
152
|
+
}
|
153
|
+
|
154
|
+
const isInPast = Object.is(count, -0) || count < 0,
|
155
|
+
fmtValue = Math.abs(count),
|
156
|
+
singular = fmtValue === 1,
|
157
|
+
lilUnits = units[unit],
|
158
|
+
fmtUnit = narrow
|
159
|
+
? singular
|
160
|
+
? lilUnits[1]
|
161
|
+
: lilUnits[2] || lilUnits[1]
|
162
|
+
: singular
|
163
|
+
? units[unit][0]
|
164
|
+
: unit;
|
165
|
+
return isInPast ? `${fmtValue} ${fmtUnit} ago` : `in ${fmtValue} ${fmtUnit}`;
|
166
|
+
}
|
167
|
+
|
168
|
+
export function formatString(knownFormat) {
|
169
|
+
// these all have the offsets removed because we don't have access to them
|
170
|
+
// without all the intl stuff this is backfilling
|
171
|
+
const filtered = pick(knownFormat, [
|
172
|
+
"weekday",
|
173
|
+
"era",
|
174
|
+
"year",
|
175
|
+
"month",
|
176
|
+
"day",
|
177
|
+
"hour",
|
178
|
+
"minute",
|
179
|
+
"second",
|
180
|
+
"timeZoneName",
|
181
|
+
"hourCycle",
|
182
|
+
]),
|
183
|
+
key = stringify(filtered),
|
184
|
+
dateTimeHuge = "EEEE, LLLL d, yyyy, h:mm a";
|
185
|
+
switch (key) {
|
186
|
+
case stringify(Formats.DATE_SHORT):
|
187
|
+
return "M/d/yyyy";
|
188
|
+
case stringify(Formats.DATE_MED):
|
189
|
+
return "LLL d, yyyy";
|
190
|
+
case stringify(Formats.DATE_MED_WITH_WEEKDAY):
|
191
|
+
return "EEE, LLL d, yyyy";
|
192
|
+
case stringify(Formats.DATE_FULL):
|
193
|
+
return "LLLL d, yyyy";
|
194
|
+
case stringify(Formats.DATE_HUGE):
|
195
|
+
return "EEEE, LLLL d, yyyy";
|
196
|
+
case stringify(Formats.TIME_SIMPLE):
|
197
|
+
return "h:mm a";
|
198
|
+
case stringify(Formats.TIME_WITH_SECONDS):
|
199
|
+
return "h:mm:ss a";
|
200
|
+
case stringify(Formats.TIME_WITH_SHORT_OFFSET):
|
201
|
+
return "h:mm a";
|
202
|
+
case stringify(Formats.TIME_WITH_LONG_OFFSET):
|
203
|
+
return "h:mm a";
|
204
|
+
case stringify(Formats.TIME_24_SIMPLE):
|
205
|
+
return "HH:mm";
|
206
|
+
case stringify(Formats.TIME_24_WITH_SECONDS):
|
207
|
+
return "HH:mm:ss";
|
208
|
+
case stringify(Formats.TIME_24_WITH_SHORT_OFFSET):
|
209
|
+
return "HH:mm";
|
210
|
+
case stringify(Formats.TIME_24_WITH_LONG_OFFSET):
|
211
|
+
return "HH:mm";
|
212
|
+
case stringify(Formats.DATETIME_SHORT):
|
213
|
+
return "M/d/yyyy, h:mm a";
|
214
|
+
case stringify(Formats.DATETIME_MED):
|
215
|
+
return "LLL d, yyyy, h:mm a";
|
216
|
+
case stringify(Formats.DATETIME_FULL):
|
217
|
+
return "LLLL d, yyyy, h:mm a";
|
218
|
+
case stringify(Formats.DATETIME_HUGE):
|
219
|
+
return dateTimeHuge;
|
220
|
+
case stringify(Formats.DATETIME_SHORT_WITH_SECONDS):
|
221
|
+
return "M/d/yyyy, h:mm:ss a";
|
222
|
+
case stringify(Formats.DATETIME_MED_WITH_SECONDS):
|
223
|
+
return "LLL d, yyyy, h:mm:ss a";
|
224
|
+
case stringify(Formats.DATETIME_MED_WITH_WEEKDAY):
|
225
|
+
return "EEE, d LLL yyyy, h:mm a";
|
226
|
+
case stringify(Formats.DATETIME_FULL_WITH_SECONDS):
|
227
|
+
return "LLLL d, yyyy, h:mm:ss a";
|
228
|
+
case stringify(Formats.DATETIME_HUGE_WITH_SECONDS):
|
229
|
+
return "EEEE, LLLL d, yyyy, h:mm:ss a";
|
230
|
+
default:
|
231
|
+
return dateTimeHuge;
|
232
|
+
}
|
233
|
+
}
|