@prairielearn/formatter 1.4.2 → 1.4.4
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.
- package/CHANGELOG.md +12 -0
- package/dist/date.d.ts +20 -10
- package/dist/date.js +22 -11
- package/dist/date.js.map +1 -1
- package/dist/interval.d.ts +2 -1
- package/dist/interval.js +2 -1
- package/dist/interval.js.map +1 -1
- package/package.json +3 -3
- package/src/date.ts +22 -11
- package/src/interval.ts +2 -1
package/CHANGELOG.md
CHANGED
package/dist/date.d.ts
CHANGED
|
@@ -4,8 +4,10 @@ type TimePrecision = 'hour' | 'minute' | 'second';
|
|
|
4
4
|
*
|
|
5
5
|
* @param date The date to format.
|
|
6
6
|
* @param timeZone The time zone to use for formatting.
|
|
7
|
-
* @param
|
|
8
|
-
* @param
|
|
7
|
+
* @param options
|
|
8
|
+
* @param options.includeMs Whether to include milliseconds in the output (default false).
|
|
9
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
10
|
+
* @param options.longTz Whether to use the long time zone name (default false).
|
|
9
11
|
* @returns Human-readable string representing the date.
|
|
10
12
|
*/
|
|
11
13
|
export declare function formatDate(date: Date, timeZone: string, { includeTz, longTz, includeMs, }?: {
|
|
@@ -41,7 +43,8 @@ export declare function formatTz(timeZone: string): string;
|
|
|
41
43
|
*
|
|
42
44
|
* @param date The date to format.
|
|
43
45
|
* @param timeZone The time zone to use for formatting.
|
|
44
|
-
* @param
|
|
46
|
+
* @param options
|
|
47
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
45
48
|
* @returns Human-readable string representing the date.
|
|
46
49
|
*/
|
|
47
50
|
export declare function formatDateHMS(date: Date, timeZone: string, { includeTz }?: {
|
|
@@ -74,13 +77,14 @@ export declare function formatDateWithinRange(date: Date, rangeStart: Date, rang
|
|
|
74
77
|
*
|
|
75
78
|
* @param date The date to format.
|
|
76
79
|
* @param timezone The time zone to use for formatting.
|
|
77
|
-
* @param
|
|
78
|
-
* @param
|
|
79
|
-
* @param
|
|
80
|
-
* @param
|
|
81
|
-
* @param
|
|
82
|
-
* @param
|
|
83
|
-
* @param
|
|
80
|
+
* @param options
|
|
81
|
+
* @param options.baseDate The base date to use for comparison (default is the current date).
|
|
82
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
83
|
+
* @param options.timeFirst If true, the time is shown before the date (default false).
|
|
84
|
+
* @param options.dateOnly If true, only the date is shown (default false).
|
|
85
|
+
* @param options.timeOnly If true, only the time is shown (default false).
|
|
86
|
+
* @param options.maxPrecision The maximum precision to show for time (default 'minute').
|
|
87
|
+
* @param options.minPrecision The minimum precision to always show for time (default 'hour').
|
|
84
88
|
* @returns Human-readable string representing the date and time.
|
|
85
89
|
*/
|
|
86
90
|
export declare function formatDateFriendly(date: Date, timezone: string, { baseDate, includeTz, timeFirst, dateOnly, timeOnly, maxPrecision, minPrecision, }?: {
|
|
@@ -107,6 +111,12 @@ export declare function formatDateFriendly(date: Date, timezone: string, { baseD
|
|
|
107
111
|
* @param end The end date and time.
|
|
108
112
|
* @param timezone The time zone to use for formatting.
|
|
109
113
|
* @param options Additional options for formatting the displayed date, taken from `formatDateFriendlyString()`.
|
|
114
|
+
* @param options.baseDate The base date to use for comparison (default is the current date).
|
|
115
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
116
|
+
* @param options.timeFirst If true, the time is shown before the date (default false).
|
|
117
|
+
* @param options.dateOnly If true, only the date is shown (default false).
|
|
118
|
+
* @param options.maxPrecision The maximum precision to show for time (default 'minute').
|
|
119
|
+
* @param options.minPrecision The minimum precision to always show for time (default 'hour').
|
|
110
120
|
* @returns Human-readable string representing the datetime range.
|
|
111
121
|
*/
|
|
112
122
|
export declare function formatDateRangeFriendly(start: Date, end: Date, timezone: string, { baseDate, includeTz, timeFirst, dateOnly, maxPrecision, minPrecision, }?: Parameters<typeof formatDateFriendly>[2]): string;
|
package/dist/date.js
CHANGED
|
@@ -5,8 +5,10 @@ import keyBy from 'lodash/keyBy.js';
|
|
|
5
5
|
*
|
|
6
6
|
* @param date The date to format.
|
|
7
7
|
* @param timeZone The time zone to use for formatting.
|
|
8
|
-
* @param
|
|
9
|
-
* @param
|
|
8
|
+
* @param options
|
|
9
|
+
* @param options.includeMs Whether to include milliseconds in the output (default false).
|
|
10
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
11
|
+
* @param options.longTz Whether to use the long time zone name (default false).
|
|
10
12
|
* @returns Human-readable string representing the date.
|
|
11
13
|
*/
|
|
12
14
|
export function formatDate(date, timeZone, { includeTz = true, longTz = false, includeMs = false, } = {}) {
|
|
@@ -90,7 +92,8 @@ export function formatTz(timeZone) {
|
|
|
90
92
|
*
|
|
91
93
|
* @param date The date to format.
|
|
92
94
|
* @param timeZone The time zone to use for formatting.
|
|
93
|
-
* @param
|
|
95
|
+
* @param options
|
|
96
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
94
97
|
* @returns Human-readable string representing the date.
|
|
95
98
|
*/
|
|
96
99
|
export function formatDateHMS(date, timeZone, { includeTz = true } = {}) {
|
|
@@ -245,7 +248,8 @@ function formatDateFriendlyParts(date, timezone, baseDate, maxPrecision = 'secon
|
|
|
245
248
|
if (maxIndex > minIndex) {
|
|
246
249
|
throw new Error('maxPrecision must be an equal or smaller unit than minPrecision.');
|
|
247
250
|
}
|
|
248
|
-
/**
|
|
251
|
+
/**
|
|
252
|
+
* Examples:
|
|
249
253
|
* min=h, max=h: 0:00:00AM -> 0AM, 0:00:01AM -> 0AM, 0:01:01AM -> 0AM
|
|
250
254
|
* min=h, max=m: 0:00:00AM -> 0AM, 0:00:01AM -> 0AM, 0:01:01AM -> 0:01AM
|
|
251
255
|
* min=h, max=s: 0:00:00AM -> 0AM, 0:00:01AM -> 0:00:01AM, 0:01:01AM -> 0:01:01AM
|
|
@@ -292,13 +296,14 @@ function formatDateFriendlyParts(date, timezone, baseDate, maxPrecision = 'secon
|
|
|
292
296
|
*
|
|
293
297
|
* @param date The date to format.
|
|
294
298
|
* @param timezone The time zone to use for formatting.
|
|
295
|
-
* @param
|
|
296
|
-
* @param
|
|
297
|
-
* @param
|
|
298
|
-
* @param
|
|
299
|
-
* @param
|
|
300
|
-
* @param
|
|
301
|
-
* @param
|
|
299
|
+
* @param options
|
|
300
|
+
* @param options.baseDate The base date to use for comparison (default is the current date).
|
|
301
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
302
|
+
* @param options.timeFirst If true, the time is shown before the date (default false).
|
|
303
|
+
* @param options.dateOnly If true, only the date is shown (default false).
|
|
304
|
+
* @param options.timeOnly If true, only the time is shown (default false).
|
|
305
|
+
* @param options.maxPrecision The maximum precision to show for time (default 'minute').
|
|
306
|
+
* @param options.minPrecision The minimum precision to always show for time (default 'hour').
|
|
302
307
|
* @returns Human-readable string representing the date and time.
|
|
303
308
|
*/
|
|
304
309
|
export function formatDateFriendly(date, timezone, { baseDate = new Date(), includeTz = true, timeFirst = false, dateOnly = false, timeOnly = false, maxPrecision = 'second', minPrecision = 'hour', } = {}) {
|
|
@@ -338,6 +343,12 @@ export function formatDateFriendly(date, timezone, { baseDate = new Date(), incl
|
|
|
338
343
|
* @param end The end date and time.
|
|
339
344
|
* @param timezone The time zone to use for formatting.
|
|
340
345
|
* @param options Additional options for formatting the displayed date, taken from `formatDateFriendlyString()`.
|
|
346
|
+
* @param options.baseDate The base date to use for comparison (default is the current date).
|
|
347
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
348
|
+
* @param options.timeFirst If true, the time is shown before the date (default false).
|
|
349
|
+
* @param options.dateOnly If true, only the date is shown (default false).
|
|
350
|
+
* @param options.maxPrecision The maximum precision to show for time (default 'minute').
|
|
351
|
+
* @param options.minPrecision The minimum precision to always show for time (default 'hour').
|
|
341
352
|
* @returns Human-readable string representing the datetime range.
|
|
342
353
|
*/
|
|
343
354
|
export function formatDateRangeFriendly(start, end, timezone, { baseDate = new Date(), includeTz = true, timeFirst = false, dateOnly = false, maxPrecision = 'second', minPrecision = 'hour', } = {}) {
|
package/dist/date.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"date.js","sourceRoot":"","sources":["../src/date.ts"],"names":[],"mappings":"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","sourcesContent":["import { toTemporalInstant } from '@js-temporal/polyfill';\nimport keyBy from 'lodash/keyBy.js';\n\ntype TimePrecision = 'hour' | 'minute' | 'second';\n\n/**\n * Format a date to a human-readable string like '2020-03-27T12:34:56 (CDT)'.\n *\n * @param date The date to format.\n * @param timeZone The time zone to use for formatting.\n * @param param2.includeTz Whether to include the time zone in the output (default true).\n * @param param2.longTz Whether to use the long time zone name (default false).\n * @returns Human-readable string representing the date.\n */\nexport function formatDate(\n date: Date,\n timeZone: string,\n {\n includeTz = true,\n longTz = false,\n includeMs = false,\n }: { includeTz?: boolean; longTz?: boolean; includeMs?: boolean } = {},\n): string {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n hourCycle: 'h23',\n year: 'numeric',\n month: '2-digit',\n day: '2-digit',\n hour: '2-digit',\n minute: '2-digit',\n second: '2-digit',\n fractionalSecondDigits: includeMs ? 3 : undefined,\n timeZoneName: longTz ? 'long' : 'short',\n };\n const parts = keyBy(new Intl.DateTimeFormat('en-US', options).formatToParts(date), (x) => x.type);\n let dateFormatted = `${parts.year.value}-${parts.month.value}-${parts.day.value} ${parts.hour.value}:${parts.minute.value}:${parts.second.value}`;\n if (includeMs) {\n dateFormatted = `${dateFormatted}.${parts.fractionalSecond.value}`;\n }\n if (includeTz) {\n dateFormatted = `${dateFormatted} (${parts.timeZoneName.value})`;\n }\n return dateFormatted;\n}\n\n/**\n * Format a date to a human-readable string like '2020-03-27'.\n *\n * @param date The date to format.\n * @param timeZone The time zone to use for formatting.\n * @returns Human-readable string representing the date.\n */\nexport function formatDateYMD(date: Date, timeZone: string): string {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n hourCycle: 'h23',\n year: 'numeric',\n month: '2-digit',\n day: '2-digit',\n };\n const parts = keyBy(new Intl.DateTimeFormat('en-US', options).formatToParts(date), (x) => x.type);\n return `${parts.year.value}-${parts.month.value}-${parts.day.value}`;\n}\n\n/**\n * Format a date to a human-readable string like '2020-03-27 14:27'.\n *\n * @param date The date to format.\n * @param timeZone The time zone to use for formatting.\n * @returns Human-readable string representing the date.\n */\nexport function formatDateYMDHM(date: Date, timeZone: string): string {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n hourCycle: 'h23',\n year: 'numeric',\n month: '2-digit',\n day: '2-digit',\n hour: '2-digit',\n minute: '2-digit',\n };\n const parts = keyBy(new Intl.DateTimeFormat('en-US', options).formatToParts(date), (x) => x.type);\n return `${parts.year.value}-${parts.month.value}-${parts.day.value} ${parts.hour.value}:${parts.minute.value}`;\n}\n\n/**\n * Format a time zone to a human-readable string like 'CDT'.\n *\n * @param timeZone The time zone to format.\n * @returns Human-readable string representing the time zone.\n */\nexport function formatTz(timeZone: string): string {\n const date = new Date();\n const parts = new Intl.DateTimeFormat('en-US', {\n timeZone,\n timeZoneName: 'short',\n }).formatToParts(date);\n const tz = parts.find((p) => p.type === 'timeZoneName');\n return tz ? tz.value : timeZone;\n}\n\n/**\n * Format a date to a human-readable string like '14:27:00 (CDT)'.\n *\n * @param date The date to format.\n * @param timeZone The time zone to use for formatting.\n * @param param2.includeTz Whether to include the time zone in the output (default true).\n * @returns Human-readable string representing the date.\n */\nexport function formatDateHMS(\n date: Date,\n timeZone: string,\n { includeTz = true }: { includeTz?: boolean } = {},\n): string {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n hourCycle: 'h23',\n hour: '2-digit',\n minute: '2-digit',\n second: '2-digit',\n timeZoneName: 'short',\n };\n const parts = keyBy(new Intl.DateTimeFormat('en-US', options).formatToParts(date), (x) => x.type);\n let dateFormatted = `${parts.hour.value}:${parts.minute.value}:${parts.second.value}`;\n if (includeTz) {\n dateFormatted = `${dateFormatted} (${parts.timeZoneName.value})`;\n }\n return dateFormatted;\n}\n\n/**\n * Format a date to a human-readable string like '18:23' or 'May 2, 07:12',\n * where the precision is determined by the range.\n *\n * @param date The date to format.\n * @param rangeStart The start of the range.\n * @param rangeEnd The end of the range.\n * @param timeZone The time zone to use for formatting.\n * @returns Human-readable string representing the date.\n */\nexport function formatDateWithinRange(\n date: Date,\n rangeStart: Date,\n rangeEnd: Date,\n timeZone: string,\n): string {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n hourCycle: 'h23',\n year: 'numeric',\n month: '2-digit',\n day: '2-digit',\n hour: '2-digit',\n minute: '2-digit',\n timeZoneName: 'short',\n };\n const dateParts = keyBy(\n new Intl.DateTimeFormat('en-US', options).formatToParts(date),\n (x) => x.type,\n );\n const startParts = keyBy(\n new Intl.DateTimeFormat('en-US', options).formatToParts(rangeStart),\n (x) => x.type,\n );\n const endParts = keyBy(\n new Intl.DateTimeFormat('en-US', options).formatToParts(rangeEnd),\n (x) => x.type,\n );\n\n // format the date (not time) parts\n const dateYMD = `${dateParts.year.value}-${dateParts.month.value}-${dateParts.day.value}`;\n const startYMD = `${startParts.year.value}-${startParts.month.value}-${startParts.day.value}`;\n const endYMD = `${endParts.year.value}-${endParts.month.value}-${endParts.day.value}`;\n\n if (dateYMD === startYMD && dateYMD === endYMD) {\n // only show the time if the date is the same for all three\n return `${dateParts.hour.value}:${dateParts.minute.value}`;\n }\n\n // format the year, but not the month or day\n const dateY = `${dateParts.year.value}`;\n const startY = `${startParts.year.value}`;\n const endY = `${endParts.year.value}`;\n\n // if the year is the same for all three, show the month, day, and time\n if (dateY === startY && dateY === endY) {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n month: 'short',\n day: 'numeric',\n hour: '2-digit',\n minute: '2-digit',\n };\n const dateParts = keyBy(\n new Intl.DateTimeFormat('en-US', options).formatToParts(date),\n (x) => x.type,\n );\n return `${dateParts.month.value} ${dateParts.day.value}, ${dateParts.hour.value}:${dateParts.minute.value}`;\n }\n\n // fall back to the full date\n return `${dateParts.year.value}-${dateParts.month.value}-${dateParts.day.value} ${dateParts.hour.value}:${dateParts.minute.value}`;\n}\n\n/**\n * Format a Date to date and time strings in the given time zone. The date is\n * formatted like\n * - 'today'\n * - 'Mon, Mar 20' (if within 180 days of the base date)\n * - 'Wed, Jan 1, 2020'\n *\n * The time format leaves off zero minutes and seconds, and uses 12-hour time,\n * giving strings like\n * - '3pm'\n * - '3:34pm'\n * - '3:34:17pm'\n *\n * maxPrecision must be an equal or smaller unit than minPrecision.\n *\n * @param date The date to format.\n * @param timezone The time zone to use for formatting.\n * @param baseDate The base date to use for comparison.\n * @param maxPrecision Only show units as large or larger than the max precision.\n * @param minPrecision Always show that unit and larger, potentially showing smaller units.\n *\n */\nfunction formatDateFriendlyParts(\n date: Date,\n timezone: string,\n baseDate: Date,\n maxPrecision: TimePrecision = 'second',\n minPrecision: TimePrecision = 'hour',\n): { dateFormatted: string; timeFormatted: string; timezoneFormatted: string } {\n // compute the number of days from the base date (0 = today, 1 = tomorrow, etc.)\n\n const baseZonedDateTime = toTemporalInstant.call(baseDate).toZonedDateTimeISO(timezone);\n const zonedDateTime = toTemporalInstant.call(date).toZonedDateTimeISO(timezone);\n\n const basePlainDate = baseZonedDateTime.toPlainDate();\n const plainDate = zonedDateTime.toPlainDate();\n\n const daysOffset = plainDate.since(basePlainDate, { largestUnit: 'day' }).days;\n\n // format the parts of the date and time\n\n const options: Intl.DateTimeFormatOptions = {\n timeZone: timezone,\n year: 'numeric',\n month: 'short',\n day: 'numeric',\n weekday: 'short',\n hourCycle: 'h12',\n hour: 'numeric',\n minute: '2-digit',\n second: '2-digit',\n timeZoneName: 'short',\n };\n const parts = keyBy(new Intl.DateTimeFormat('en-US', options).formatToParts(date), (x) => x.type);\n\n // format the date string\n\n let dateFormatted = '';\n if (daysOffset === 0) {\n dateFormatted = 'today';\n } else if (daysOffset === 1) {\n dateFormatted = 'tomorrow';\n } else if (daysOffset === -1) {\n dateFormatted = 'yesterday';\n } else if (Math.abs(daysOffset) <= 180) {\n // non-breaking-space (\\u00a0) is used between the month and day\n dateFormatted = `${parts.weekday.value}, ${parts.month.value}\\u00a0${parts.day.value}`;\n } else {\n dateFormatted = `${parts.weekday.value}, ${parts.month.value}\\u00a0${parts.day.value}, ${parts.year.value}`;\n }\n\n const precisionOrder: TimePrecision[] = ['second', 'minute', 'hour'];\n const maxIndex = precisionOrder.indexOf(maxPrecision);\n const minIndex = precisionOrder.indexOf(minPrecision);\n\n /**\n * The maximum precision must be a unit smaller than or equal to the minimum precision, otherwise the rules will contradict each other.\n *\n * If max is a larger unit than min, e.g. max = hour, min = minute, then by \"min\"\n * we must display minute and smaller but by \"max\" we can display hour and larger, which is a contradiction.\n *\n * If min is a larger unit than max, e.g. max = minute, min = hour, then by \"min\" we must display\n * hour and smaller and by \"max\" we can display minutes and larger. These do not contradict each other.\n *\n * V min/max > | h | m | s\n * h | X | X | X\n * m | I | X | X\n * s | I | I | X\n *\n * X - valid configuration\n * I - invalid configuration\n */\n\n // A higher index corresponds to a larger unit, so if maxIndex is larger than minIndex, then the rules contradict each other.\n if (maxIndex > minIndex) {\n throw new Error('maxPrecision must be an equal or smaller unit than minPrecision.');\n }\n\n /** Examples:\n * min=h, max=h: 0:00:00AM -> 0AM, 0:00:01AM -> 0AM, 0:01:01AM -> 0AM\n * min=h, max=m: 0:00:00AM -> 0AM, 0:00:01AM -> 0AM, 0:01:01AM -> 0:01AM\n * min=h, max=s: 0:00:00AM -> 0AM, 0:00:01AM -> 0:00:01AM, 0:01:01AM -> 0:01:01AM\n *\n * min=m, max=m: 0:00:00AM -> 0:00AM, 0:00:01AM -> 0:00AM, 0:01:01AM -> 0:00AM\n * min=m, max=s: 0:00:00AM -> 0:00AM, 0:00:01AM -> 0:00AM, 0:01:01AM -> 0:01:01AM\n *\n * min=s, max=s: 0:00:00AM -> 0:00:00AM, 0:00:01AM -> 0:00:01AM, 0:01:01AM -> 0:01:01AM\n */\n\n let timeFormatted = parts.hour.value;\n\n const shouldShowMinutes =\n ['minute', 'second'].includes(minPrecision) ||\n (maxPrecision === 'minute' && parts.minute.value !== '00') ||\n (maxPrecision === 'second' && (parts.minute.value !== '00' || parts.second.value !== '00'));\n\n if (shouldShowMinutes) {\n timeFormatted += `:${parts.minute.value}`;\n }\n\n const shouldShowSeconds =\n minPrecision === 'second' || (maxPrecision === 'second' && parts.second.value !== '00');\n\n if (shouldShowSeconds) {\n timeFormatted += `:${parts.second.value}`;\n }\n // add the am/pm part\n timeFormatted = `${timeFormatted}${parts.dayPeriod.value.toLowerCase()}`;\n\n // format the timezone\n\n const timezoneFormatted = `(${parts.timeZoneName.value})`;\n\n return {\n dateFormatted,\n timeFormatted,\n timezoneFormatted,\n };\n}\n\n/**\n * Format a date to a string like:\n * - 'today, 3pm'\n * - 'tomorrow, 10:30am'\n * - 'yesterday, 11:45pm'\n * - 'Mon, Mar 20, 8:15am' (if within 180 days of the base date)\n * - 'Wed, Jan 1, 2020, 12pm'\n * - `today, 3pm (CDT)` (if `includeTz` is true)\n * - `3pm today` (if `timeFirst` is true)\n * - 'today' (if `dateOnly` is true)\n *\n * If using this within a sentence like `... at ${formatDateFriendlyString()}`,\n * use `timeFirst: true` to improve readability.\n *\n * @param date The date to format.\n * @param timezone The time zone to use for formatting.\n * @param param.baseDate The base date to use for comparison (default is the current date).\n * @param param.includeTz Whether to include the time zone in the output (default true).\n * @param param.timeFirst If true, the time is shown before the date (default false).\n * @param param.dateOnly If true, only the date is shown (default false).\n * @param param.timeOnly If true, only the time is shown (default false).\n * @param param.maxPrecision The maximum precision to show for time (default 'minute').\n * @param param.minPrecision The minimum precision to always show for time (default 'hour').\n * @returns Human-readable string representing the date and time.\n */\nexport function formatDateFriendly(\n date: Date,\n timezone: string,\n {\n baseDate = new Date(),\n includeTz = true,\n timeFirst = false,\n dateOnly = false,\n timeOnly = false,\n maxPrecision = 'second',\n minPrecision = 'hour',\n }: {\n baseDate?: Date;\n includeTz?: boolean;\n timeFirst?: boolean;\n dateOnly?: boolean;\n timeOnly?: boolean;\n maxPrecision?: TimePrecision;\n minPrecision?: TimePrecision;\n } = {},\n): string {\n const { dateFormatted, timeFormatted, timezoneFormatted } = formatDateFriendlyParts(\n date,\n timezone,\n baseDate,\n maxPrecision,\n minPrecision,\n );\n\n let dateTimeFormatted = '';\n if (dateOnly) {\n dateTimeFormatted = dateFormatted;\n } else if (timeOnly) {\n dateTimeFormatted = timeFormatted;\n } else {\n if (timeFirst) {\n dateTimeFormatted = `${timeFormatted} ${dateFormatted}`;\n } else {\n dateTimeFormatted = `${dateFormatted}, ${timeFormatted}`;\n }\n }\n if (includeTz) {\n dateTimeFormatted = `${dateTimeFormatted} ${timezoneFormatted}`;\n }\n return dateTimeFormatted;\n}\n\n/**\n * Format a datetime range to a string like:\n * - 'today, 10am'\n * - 'today, 3pm to 5pm'\n * - 'today, 3pm to tomorrow, 5pm'\n * - 'today, 3pm to 5pm (CDT)' (if `includeTz` is true)\n * - '3pm today to 5pm tomorrow' (if `timeFirst` is true)\n * - 'today to tomorrow' (if `dateOnly` is true)\n *\n * This uses `formatDateFriendlyString()` to format the individual dates and times.\n *\n * @param start The start date and time.\n * @param end The end date and time.\n * @param timezone The time zone to use for formatting.\n * @param options Additional options for formatting the displayed date, taken from `formatDateFriendlyString()`.\n * @returns Human-readable string representing the datetime range.\n */\nexport function formatDateRangeFriendly(\n start: Date,\n end: Date,\n timezone: string,\n {\n baseDate = new Date(),\n includeTz = true,\n timeFirst = false,\n dateOnly = false,\n maxPrecision = 'second',\n minPrecision = 'hour',\n }: Parameters<typeof formatDateFriendly>[2] = {},\n): string {\n const {\n dateFormatted: startDateFormatted,\n timeFormatted: startTimeFormatted,\n timezoneFormatted,\n } = formatDateFriendlyParts(start, timezone, baseDate, maxPrecision, minPrecision);\n const { dateFormatted: endDateFormatted, timeFormatted: endTimeFormatted } =\n formatDateFriendlyParts(end, timezone, baseDate, maxPrecision, minPrecision);\n\n let result: string | undefined;\n if (dateOnly) {\n if (startDateFormatted === endDateFormatted) {\n result = startDateFormatted;\n } else {\n result = `${startDateFormatted} to ${endDateFormatted}`;\n }\n } else {\n if (startDateFormatted === endDateFormatted) {\n let timeRangeFormatted: string | undefined;\n if (startTimeFormatted === endTimeFormatted) {\n timeRangeFormatted = startTimeFormatted;\n } else {\n timeRangeFormatted = `${startTimeFormatted} to ${endTimeFormatted}`;\n }\n if (timeFirst) {\n result = `${timeRangeFormatted} ${startDateFormatted}`;\n } else {\n result = `${startDateFormatted}, ${timeRangeFormatted}`;\n }\n } else {\n if (timeFirst) {\n result = `${startTimeFormatted} ${startDateFormatted} to ${endTimeFormatted} ${endDateFormatted}`;\n } else {\n result = `${startDateFormatted}, ${startTimeFormatted} to ${endDateFormatted}, ${endTimeFormatted}`;\n }\n }\n }\n if (includeTz) {\n result = `${result} ${timezoneFormatted}`;\n }\n return result;\n}\n"]}
|
|
1
|
+
{"version":3,"file":"date.js","sourceRoot":"","sources":["../src/date.ts"],"names":[],"mappings":"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","sourcesContent":["import { toTemporalInstant } from '@js-temporal/polyfill';\nimport keyBy from 'lodash/keyBy.js';\n\ntype TimePrecision = 'hour' | 'minute' | 'second';\n\n/**\n * Format a date to a human-readable string like '2020-03-27T12:34:56 (CDT)'.\n *\n * @param date The date to format.\n * @param timeZone The time zone to use for formatting.\n * @param options\n * @param options.includeMs Whether to include milliseconds in the output (default false).\n * @param options.includeTz Whether to include the time zone in the output (default true).\n * @param options.longTz Whether to use the long time zone name (default false).\n * @returns Human-readable string representing the date.\n */\nexport function formatDate(\n date: Date,\n timeZone: string,\n {\n includeTz = true,\n longTz = false,\n includeMs = false,\n }: { includeTz?: boolean; longTz?: boolean; includeMs?: boolean } = {},\n): string {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n hourCycle: 'h23',\n year: 'numeric',\n month: '2-digit',\n day: '2-digit',\n hour: '2-digit',\n minute: '2-digit',\n second: '2-digit',\n fractionalSecondDigits: includeMs ? 3 : undefined,\n timeZoneName: longTz ? 'long' : 'short',\n };\n const parts = keyBy(new Intl.DateTimeFormat('en-US', options).formatToParts(date), (x) => x.type);\n let dateFormatted = `${parts.year.value}-${parts.month.value}-${parts.day.value} ${parts.hour.value}:${parts.minute.value}:${parts.second.value}`;\n if (includeMs) {\n dateFormatted = `${dateFormatted}.${parts.fractionalSecond.value}`;\n }\n if (includeTz) {\n dateFormatted = `${dateFormatted} (${parts.timeZoneName.value})`;\n }\n return dateFormatted;\n}\n\n/**\n * Format a date to a human-readable string like '2020-03-27'.\n *\n * @param date The date to format.\n * @param timeZone The time zone to use for formatting.\n * @returns Human-readable string representing the date.\n */\nexport function formatDateYMD(date: Date, timeZone: string): string {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n hourCycle: 'h23',\n year: 'numeric',\n month: '2-digit',\n day: '2-digit',\n };\n const parts = keyBy(new Intl.DateTimeFormat('en-US', options).formatToParts(date), (x) => x.type);\n return `${parts.year.value}-${parts.month.value}-${parts.day.value}`;\n}\n\n/**\n * Format a date to a human-readable string like '2020-03-27 14:27'.\n *\n * @param date The date to format.\n * @param timeZone The time zone to use for formatting.\n * @returns Human-readable string representing the date.\n */\nexport function formatDateYMDHM(date: Date, timeZone: string): string {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n hourCycle: 'h23',\n year: 'numeric',\n month: '2-digit',\n day: '2-digit',\n hour: '2-digit',\n minute: '2-digit',\n };\n const parts = keyBy(new Intl.DateTimeFormat('en-US', options).formatToParts(date), (x) => x.type);\n return `${parts.year.value}-${parts.month.value}-${parts.day.value} ${parts.hour.value}:${parts.minute.value}`;\n}\n\n/**\n * Format a time zone to a human-readable string like 'CDT'.\n *\n * @param timeZone The time zone to format.\n * @returns Human-readable string representing the time zone.\n */\nexport function formatTz(timeZone: string): string {\n const date = new Date();\n const parts = new Intl.DateTimeFormat('en-US', {\n timeZone,\n timeZoneName: 'short',\n }).formatToParts(date);\n const tz = parts.find((p) => p.type === 'timeZoneName');\n return tz ? tz.value : timeZone;\n}\n\n/**\n * Format a date to a human-readable string like '14:27:00 (CDT)'.\n *\n * @param date The date to format.\n * @param timeZone The time zone to use for formatting.\n * @param options\n * @param options.includeTz Whether to include the time zone in the output (default true).\n * @returns Human-readable string representing the date.\n */\nexport function formatDateHMS(\n date: Date,\n timeZone: string,\n { includeTz = true }: { includeTz?: boolean } = {},\n): string {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n hourCycle: 'h23',\n hour: '2-digit',\n minute: '2-digit',\n second: '2-digit',\n timeZoneName: 'short',\n };\n const parts = keyBy(new Intl.DateTimeFormat('en-US', options).formatToParts(date), (x) => x.type);\n let dateFormatted = `${parts.hour.value}:${parts.minute.value}:${parts.second.value}`;\n if (includeTz) {\n dateFormatted = `${dateFormatted} (${parts.timeZoneName.value})`;\n }\n return dateFormatted;\n}\n\n/**\n * Format a date to a human-readable string like '18:23' or 'May 2, 07:12',\n * where the precision is determined by the range.\n *\n * @param date The date to format.\n * @param rangeStart The start of the range.\n * @param rangeEnd The end of the range.\n * @param timeZone The time zone to use for formatting.\n * @returns Human-readable string representing the date.\n */\nexport function formatDateWithinRange(\n date: Date,\n rangeStart: Date,\n rangeEnd: Date,\n timeZone: string,\n): string {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n hourCycle: 'h23',\n year: 'numeric',\n month: '2-digit',\n day: '2-digit',\n hour: '2-digit',\n minute: '2-digit',\n timeZoneName: 'short',\n };\n const dateParts = keyBy(\n new Intl.DateTimeFormat('en-US', options).formatToParts(date),\n (x) => x.type,\n );\n const startParts = keyBy(\n new Intl.DateTimeFormat('en-US', options).formatToParts(rangeStart),\n (x) => x.type,\n );\n const endParts = keyBy(\n new Intl.DateTimeFormat('en-US', options).formatToParts(rangeEnd),\n (x) => x.type,\n );\n\n // format the date (not time) parts\n const dateYMD = `${dateParts.year.value}-${dateParts.month.value}-${dateParts.day.value}`;\n const startYMD = `${startParts.year.value}-${startParts.month.value}-${startParts.day.value}`;\n const endYMD = `${endParts.year.value}-${endParts.month.value}-${endParts.day.value}`;\n\n if (dateYMD === startYMD && dateYMD === endYMD) {\n // only show the time if the date is the same for all three\n return `${dateParts.hour.value}:${dateParts.minute.value}`;\n }\n\n // format the year, but not the month or day\n const dateY = `${dateParts.year.value}`;\n const startY = `${startParts.year.value}`;\n const endY = `${endParts.year.value}`;\n\n // if the year is the same for all three, show the month, day, and time\n if (dateY === startY && dateY === endY) {\n const options: Intl.DateTimeFormatOptions = {\n timeZone,\n month: 'short',\n day: 'numeric',\n hour: '2-digit',\n minute: '2-digit',\n };\n const dateParts = keyBy(\n new Intl.DateTimeFormat('en-US', options).formatToParts(date),\n (x) => x.type,\n );\n return `${dateParts.month.value} ${dateParts.day.value}, ${dateParts.hour.value}:${dateParts.minute.value}`;\n }\n\n // fall back to the full date\n return `${dateParts.year.value}-${dateParts.month.value}-${dateParts.day.value} ${dateParts.hour.value}:${dateParts.minute.value}`;\n}\n\n/**\n * Format a Date to date and time strings in the given time zone. The date is\n * formatted like\n * - 'today'\n * - 'Mon, Mar 20' (if within 180 days of the base date)\n * - 'Wed, Jan 1, 2020'\n *\n * The time format leaves off zero minutes and seconds, and uses 12-hour time,\n * giving strings like\n * - '3pm'\n * - '3:34pm'\n * - '3:34:17pm'\n *\n * maxPrecision must be an equal or smaller unit than minPrecision.\n *\n * @param date The date to format.\n * @param timezone The time zone to use for formatting.\n * @param baseDate The base date to use for comparison.\n * @param maxPrecision Only show units as large or larger than the max precision.\n * @param minPrecision Always show that unit and larger, potentially showing smaller units.\n *\n */\nfunction formatDateFriendlyParts(\n date: Date,\n timezone: string,\n baseDate: Date,\n maxPrecision: TimePrecision = 'second',\n minPrecision: TimePrecision = 'hour',\n): { dateFormatted: string; timeFormatted: string; timezoneFormatted: string } {\n // compute the number of days from the base date (0 = today, 1 = tomorrow, etc.)\n\n const baseZonedDateTime = toTemporalInstant.call(baseDate).toZonedDateTimeISO(timezone);\n const zonedDateTime = toTemporalInstant.call(date).toZonedDateTimeISO(timezone);\n\n const basePlainDate = baseZonedDateTime.toPlainDate();\n const plainDate = zonedDateTime.toPlainDate();\n\n const daysOffset = plainDate.since(basePlainDate, { largestUnit: 'day' }).days;\n\n // format the parts of the date and time\n\n const options: Intl.DateTimeFormatOptions = {\n timeZone: timezone,\n year: 'numeric',\n month: 'short',\n day: 'numeric',\n weekday: 'short',\n hourCycle: 'h12',\n hour: 'numeric',\n minute: '2-digit',\n second: '2-digit',\n timeZoneName: 'short',\n };\n const parts = keyBy(new Intl.DateTimeFormat('en-US', options).formatToParts(date), (x) => x.type);\n\n // format the date string\n\n let dateFormatted = '';\n if (daysOffset === 0) {\n dateFormatted = 'today';\n } else if (daysOffset === 1) {\n dateFormatted = 'tomorrow';\n } else if (daysOffset === -1) {\n dateFormatted = 'yesterday';\n } else if (Math.abs(daysOffset) <= 180) {\n // non-breaking-space (\\u00a0) is used between the month and day\n dateFormatted = `${parts.weekday.value}, ${parts.month.value}\\u00a0${parts.day.value}`;\n } else {\n dateFormatted = `${parts.weekday.value}, ${parts.month.value}\\u00a0${parts.day.value}, ${parts.year.value}`;\n }\n\n const precisionOrder: TimePrecision[] = ['second', 'minute', 'hour'];\n const maxIndex = precisionOrder.indexOf(maxPrecision);\n const minIndex = precisionOrder.indexOf(minPrecision);\n\n /**\n * The maximum precision must be a unit smaller than or equal to the minimum precision, otherwise the rules will contradict each other.\n *\n * If max is a larger unit than min, e.g. max = hour, min = minute, then by \"min\"\n * we must display minute and smaller but by \"max\" we can display hour and larger, which is a contradiction.\n *\n * If min is a larger unit than max, e.g. max = minute, min = hour, then by \"min\" we must display\n * hour and smaller and by \"max\" we can display minutes and larger. These do not contradict each other.\n *\n * V min/max > | h | m | s\n * h | X | X | X\n * m | I | X | X\n * s | I | I | X\n *\n * X - valid configuration\n * I - invalid configuration\n */\n\n // A higher index corresponds to a larger unit, so if maxIndex is larger than minIndex, then the rules contradict each other.\n if (maxIndex > minIndex) {\n throw new Error('maxPrecision must be an equal or smaller unit than minPrecision.');\n }\n\n /**\n * Examples:\n * min=h, max=h: 0:00:00AM -> 0AM, 0:00:01AM -> 0AM, 0:01:01AM -> 0AM\n * min=h, max=m: 0:00:00AM -> 0AM, 0:00:01AM -> 0AM, 0:01:01AM -> 0:01AM\n * min=h, max=s: 0:00:00AM -> 0AM, 0:00:01AM -> 0:00:01AM, 0:01:01AM -> 0:01:01AM\n *\n * min=m, max=m: 0:00:00AM -> 0:00AM, 0:00:01AM -> 0:00AM, 0:01:01AM -> 0:00AM\n * min=m, max=s: 0:00:00AM -> 0:00AM, 0:00:01AM -> 0:00AM, 0:01:01AM -> 0:01:01AM\n *\n * min=s, max=s: 0:00:00AM -> 0:00:00AM, 0:00:01AM -> 0:00:01AM, 0:01:01AM -> 0:01:01AM\n */\n\n let timeFormatted = parts.hour.value;\n\n const shouldShowMinutes =\n ['minute', 'second'].includes(minPrecision) ||\n (maxPrecision === 'minute' && parts.minute.value !== '00') ||\n (maxPrecision === 'second' && (parts.minute.value !== '00' || parts.second.value !== '00'));\n\n if (shouldShowMinutes) {\n timeFormatted += `:${parts.minute.value}`;\n }\n\n const shouldShowSeconds =\n minPrecision === 'second' || (maxPrecision === 'second' && parts.second.value !== '00');\n\n if (shouldShowSeconds) {\n timeFormatted += `:${parts.second.value}`;\n }\n // add the am/pm part\n timeFormatted = `${timeFormatted}${parts.dayPeriod.value.toLowerCase()}`;\n\n // format the timezone\n\n const timezoneFormatted = `(${parts.timeZoneName.value})`;\n\n return {\n dateFormatted,\n timeFormatted,\n timezoneFormatted,\n };\n}\n\n/**\n * Format a date to a string like:\n * - 'today, 3pm'\n * - 'tomorrow, 10:30am'\n * - 'yesterday, 11:45pm'\n * - 'Mon, Mar 20, 8:15am' (if within 180 days of the base date)\n * - 'Wed, Jan 1, 2020, 12pm'\n * - `today, 3pm (CDT)` (if `includeTz` is true)\n * - `3pm today` (if `timeFirst` is true)\n * - 'today' (if `dateOnly` is true)\n *\n * If using this within a sentence like `... at ${formatDateFriendlyString()}`,\n * use `timeFirst: true` to improve readability.\n *\n * @param date The date to format.\n * @param timezone The time zone to use for formatting.\n * @param options\n * @param options.baseDate The base date to use for comparison (default is the current date).\n * @param options.includeTz Whether to include the time zone in the output (default true).\n * @param options.timeFirst If true, the time is shown before the date (default false).\n * @param options.dateOnly If true, only the date is shown (default false).\n * @param options.timeOnly If true, only the time is shown (default false).\n * @param options.maxPrecision The maximum precision to show for time (default 'minute').\n * @param options.minPrecision The minimum precision to always show for time (default 'hour').\n * @returns Human-readable string representing the date and time.\n */\nexport function formatDateFriendly(\n date: Date,\n timezone: string,\n {\n baseDate = new Date(),\n includeTz = true,\n timeFirst = false,\n dateOnly = false,\n timeOnly = false,\n maxPrecision = 'second',\n minPrecision = 'hour',\n }: {\n baseDate?: Date;\n includeTz?: boolean;\n timeFirst?: boolean;\n dateOnly?: boolean;\n timeOnly?: boolean;\n maxPrecision?: TimePrecision;\n minPrecision?: TimePrecision;\n } = {},\n): string {\n const { dateFormatted, timeFormatted, timezoneFormatted } = formatDateFriendlyParts(\n date,\n timezone,\n baseDate,\n maxPrecision,\n minPrecision,\n );\n\n let dateTimeFormatted = '';\n if (dateOnly) {\n dateTimeFormatted = dateFormatted;\n } else if (timeOnly) {\n dateTimeFormatted = timeFormatted;\n } else {\n if (timeFirst) {\n dateTimeFormatted = `${timeFormatted} ${dateFormatted}`;\n } else {\n dateTimeFormatted = `${dateFormatted}, ${timeFormatted}`;\n }\n }\n if (includeTz) {\n dateTimeFormatted = `${dateTimeFormatted} ${timezoneFormatted}`;\n }\n return dateTimeFormatted;\n}\n\n/**\n * Format a datetime range to a string like:\n * - 'today, 10am'\n * - 'today, 3pm to 5pm'\n * - 'today, 3pm to tomorrow, 5pm'\n * - 'today, 3pm to 5pm (CDT)' (if `includeTz` is true)\n * - '3pm today to 5pm tomorrow' (if `timeFirst` is true)\n * - 'today to tomorrow' (if `dateOnly` is true)\n *\n * This uses `formatDateFriendlyString()` to format the individual dates and times.\n *\n * @param start The start date and time.\n * @param end The end date and time.\n * @param timezone The time zone to use for formatting.\n * @param options Additional options for formatting the displayed date, taken from `formatDateFriendlyString()`.\n * @param options.baseDate The base date to use for comparison (default is the current date).\n * @param options.includeTz Whether to include the time zone in the output (default true).\n * @param options.timeFirst If true, the time is shown before the date (default false).\n * @param options.dateOnly If true, only the date is shown (default false).\n * @param options.maxPrecision The maximum precision to show for time (default 'minute').\n * @param options.minPrecision The minimum precision to always show for time (default 'hour').\n * @returns Human-readable string representing the datetime range.\n */\nexport function formatDateRangeFriendly(\n start: Date,\n end: Date,\n timezone: string,\n {\n baseDate = new Date(),\n includeTz = true,\n timeFirst = false,\n dateOnly = false,\n maxPrecision = 'second',\n minPrecision = 'hour',\n }: Parameters<typeof formatDateFriendly>[2] = {},\n): string {\n const {\n dateFormatted: startDateFormatted,\n timeFormatted: startTimeFormatted,\n timezoneFormatted,\n } = formatDateFriendlyParts(start, timezone, baseDate, maxPrecision, minPrecision);\n const { dateFormatted: endDateFormatted, timeFormatted: endTimeFormatted } =\n formatDateFriendlyParts(end, timezone, baseDate, maxPrecision, minPrecision);\n\n let result: string | undefined;\n if (dateOnly) {\n if (startDateFormatted === endDateFormatted) {\n result = startDateFormatted;\n } else {\n result = `${startDateFormatted} to ${endDateFormatted}`;\n }\n } else {\n if (startDateFormatted === endDateFormatted) {\n let timeRangeFormatted: string | undefined;\n if (startTimeFormatted === endTimeFormatted) {\n timeRangeFormatted = startTimeFormatted;\n } else {\n timeRangeFormatted = `${startTimeFormatted} to ${endTimeFormatted}`;\n }\n if (timeFirst) {\n result = `${timeRangeFormatted} ${startDateFormatted}`;\n } else {\n result = `${startDateFormatted}, ${timeRangeFormatted}`;\n }\n } else {\n if (timeFirst) {\n result = `${startTimeFormatted} ${startDateFormatted} to ${endTimeFormatted} ${endDateFormatted}`;\n } else {\n result = `${startDateFormatted}, ${startTimeFormatted} to ${endDateFormatted}, ${endTimeFormatted}`;\n }\n }\n }\n if (includeTz) {\n result = `${result} ${timezoneFormatted}`;\n }\n return result;\n}\n"]}
|
package/dist/interval.d.ts
CHANGED
|
@@ -4,7 +4,7 @@ export declare const HOUR_IN_MILLISECONDS: number;
|
|
|
4
4
|
export declare const DAY_IN_MILLISECONDS: number;
|
|
5
5
|
/**
|
|
6
6
|
* Makes an interval (in milliseconds).
|
|
7
|
-
*
|
|
7
|
+
* @param param
|
|
8
8
|
* @param param.days The number of days in the interval.
|
|
9
9
|
* @param param.hours The number of hours in the interval.
|
|
10
10
|
* @param param.minutes The number of minutes in the interval.
|
|
@@ -37,6 +37,7 @@ export declare function formatIntervalRelative(interval: number, prefix: 'Until'
|
|
|
37
37
|
* Format an interval (in milliseconds) to a human-readable string like HH:MM or +HH:MM.
|
|
38
38
|
*
|
|
39
39
|
* @param interval Time interval in milliseconds.
|
|
40
|
+
* @param options
|
|
40
41
|
* @param options.signed Whether to include the sign in the output.
|
|
41
42
|
* @returns Human-readable string representing the interval in minutes.
|
|
42
43
|
*/
|
package/dist/interval.js
CHANGED
|
@@ -4,7 +4,7 @@ export const HOUR_IN_MILLISECONDS = 60 * MINUTE_IN_MILLISECONDS;
|
|
|
4
4
|
export const DAY_IN_MILLISECONDS = 24 * HOUR_IN_MILLISECONDS;
|
|
5
5
|
/**
|
|
6
6
|
* Makes an interval (in milliseconds).
|
|
7
|
-
*
|
|
7
|
+
* @param param
|
|
8
8
|
* @param param.days The number of days in the interval.
|
|
9
9
|
* @param param.hours The number of hours in the interval.
|
|
10
10
|
* @param param.minutes The number of minutes in the interval.
|
|
@@ -70,6 +70,7 @@ export function formatIntervalRelative(interval, prefix, reference) {
|
|
|
70
70
|
* Format an interval (in milliseconds) to a human-readable string like HH:MM or +HH:MM.
|
|
71
71
|
*
|
|
72
72
|
* @param interval Time interval in milliseconds.
|
|
73
|
+
* @param options
|
|
73
74
|
* @param options.signed Whether to include the sign in the output.
|
|
74
75
|
* @returns Human-readable string representing the interval in minutes.
|
|
75
76
|
*/
|
package/dist/interval.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"interval.js","sourceRoot":"","sources":["../src/interval.ts"],"names":[],"mappings":"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
|
|
1
|
+
{"version":3,"file":"interval.js","sourceRoot":"","sources":["../src/interval.ts"],"names":[],"mappings":"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","sourcesContent":["export const SECOND_IN_MILLISECONDS = 1000;\nexport const MINUTE_IN_MILLISECONDS = 60 * SECOND_IN_MILLISECONDS;\nexport const HOUR_IN_MILLISECONDS = 60 * MINUTE_IN_MILLISECONDS;\nexport const DAY_IN_MILLISECONDS = 24 * HOUR_IN_MILLISECONDS;\n\n/**\n * Makes an interval (in milliseconds).\n * @param param\n * @param param.days The number of days in the interval.\n * @param param.hours The number of hours in the interval.\n * @param param.minutes The number of minutes in the interval.\n * @param param.seconds The number of seconds in the interval.\n */\nexport function makeInterval({\n days = 0,\n hours = 0,\n minutes = 0,\n seconds = 0,\n}: {\n days?: number;\n hours?: number;\n minutes?: number;\n seconds?: number;\n}): number {\n return (((days * 24 + hours) * 60 + minutes) * 60 + seconds) * 1000;\n}\n\n/**\n * Format an interval (in milliseconds) to a human-readable string like '3 h 40 m'.\n *\n * @param interval Time interval in milliseconds.\n * @returns Human-readable string representing the interval.\n */\nexport function formatInterval(interval: number): string {\n const sign = interval < 0 ? '-' : '';\n\n const days = Math.floor(Math.abs(interval) / DAY_IN_MILLISECONDS);\n const hours = Math.floor(Math.abs(interval) / HOUR_IN_MILLISECONDS) % 24;\n const mins = Math.floor(Math.abs(interval) / MINUTE_IN_MILLISECONDS) % 60;\n const secs = Math.floor(Math.abs(interval) / SECOND_IN_MILLISECONDS) % 60;\n\n const parts: string[] = [];\n\n if (days > 0) {\n parts.push(`${sign}${days} d`);\n }\n if (hours > 0) {\n parts.push(`${sign}${hours} h`);\n }\n if (mins > 0) {\n parts.push(`${sign}${mins} min`);\n }\n if (secs > 0) {\n parts.push(`${sign}${secs} s`);\n }\n if (parts.length === 0) {\n parts.push('0 s');\n }\n\n return parts.join(' ');\n}\n\n/**\n * Format an interval (in milliseconds) to a human-readable string like 'Until 6\n * minutes before the session start time'.\n *\n * @param interval Time interval in milliseconds relative to `reference` (positive intervals are after `reference`).\n * @param prefix The prefix to use, must be 'Until' or 'From' (or lowercase versions of these).\n * @param reference The reference time, for example 'session start time'.\n * @returns Human-readable string representing the interval.\n */\nexport function formatIntervalRelative(\n interval: number,\n prefix: 'Until' | 'until' | 'From' | 'from',\n reference: string,\n): string {\n if (interval > 0) {\n return `${prefix} ${formatInterval(interval)} after ${reference}`;\n } else if (interval < 0) {\n return `${prefix} ${formatInterval(-interval)} before ${reference}`;\n } else if (interval === 0) {\n return `${prefix} ${reference}`;\n } else {\n return `Invalid interval: ${interval}`;\n }\n}\n\n/**\n * Format an interval (in milliseconds) to a human-readable string like HH:MM or +HH:MM.\n *\n * @param interval Time interval in milliseconds.\n * @param options\n * @param options.signed Whether to include the sign in the output.\n * @returns Human-readable string representing the interval in minutes.\n */\nexport function formatIntervalHM(\n interval: number,\n { signed = false }: { signed?: boolean } = { signed: false },\n): string {\n const sign = interval < 0 ? '-' : interval > 0 ? (signed ? '+' : '') : '';\n const hours = Math.floor(Math.abs(interval) / HOUR_IN_MILLISECONDS);\n const mins = Math.floor(Math.abs(interval) / MINUTE_IN_MILLISECONDS) % 60;\n return `${sign}${hours.toString().padStart(2, '0')}:${mins.toString().padStart(2, '0')}`;\n}\n\n/**\n * Format an interval (in milliseconds) to a human-readable string with the number of minutes, like '7 minutes' or '1 minute'.\n *\n * @param interval Time interval in milliseconds.\n * @returns Human-readable string representing the interval in minutes.\n */\nexport function formatIntervalMinutes(interval: number): string {\n const sign = interval < 0 ? '-' : '';\n const minutes = Math.ceil(Math.abs(interval / MINUTE_IN_MILLISECONDS));\n if (minutes === 1) {\n return `${sign}1 minute`;\n } else {\n return `${sign}${minutes} minutes`;\n }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@prairielearn/formatter",
|
|
3
|
-
"version": "1.4.
|
|
3
|
+
"version": "1.4.4",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"repository": {
|
|
6
6
|
"type": "git",
|
|
@@ -19,10 +19,10 @@
|
|
|
19
19
|
},
|
|
20
20
|
"devDependencies": {
|
|
21
21
|
"@prairielearn/tsconfig": "^0.0.0",
|
|
22
|
-
"@types/node": "^22.
|
|
22
|
+
"@types/node": "^22.17.0",
|
|
23
23
|
"@vitest/coverage-v8": "^3.2.4",
|
|
24
24
|
"tsx": "^4.20.3",
|
|
25
|
-
"typescript": "^5.
|
|
25
|
+
"typescript": "^5.9.2",
|
|
26
26
|
"vitest": "^3.2.4"
|
|
27
27
|
}
|
|
28
28
|
}
|
package/src/date.ts
CHANGED
|
@@ -8,8 +8,10 @@ type TimePrecision = 'hour' | 'minute' | 'second';
|
|
|
8
8
|
*
|
|
9
9
|
* @param date The date to format.
|
|
10
10
|
* @param timeZone The time zone to use for formatting.
|
|
11
|
-
* @param
|
|
12
|
-
* @param
|
|
11
|
+
* @param options
|
|
12
|
+
* @param options.includeMs Whether to include milliseconds in the output (default false).
|
|
13
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
14
|
+
* @param options.longTz Whether to use the long time zone name (default false).
|
|
13
15
|
* @returns Human-readable string representing the date.
|
|
14
16
|
*/
|
|
15
17
|
export function formatDate(
|
|
@@ -105,7 +107,8 @@ export function formatTz(timeZone: string): string {
|
|
|
105
107
|
*
|
|
106
108
|
* @param date The date to format.
|
|
107
109
|
* @param timeZone The time zone to use for formatting.
|
|
108
|
-
* @param
|
|
110
|
+
* @param options
|
|
111
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
109
112
|
* @returns Human-readable string representing the date.
|
|
110
113
|
*/
|
|
111
114
|
export function formatDateHMS(
|
|
@@ -301,7 +304,8 @@ function formatDateFriendlyParts(
|
|
|
301
304
|
throw new Error('maxPrecision must be an equal or smaller unit than minPrecision.');
|
|
302
305
|
}
|
|
303
306
|
|
|
304
|
-
/**
|
|
307
|
+
/**
|
|
308
|
+
* Examples:
|
|
305
309
|
* min=h, max=h: 0:00:00AM -> 0AM, 0:00:01AM -> 0AM, 0:01:01AM -> 0AM
|
|
306
310
|
* min=h, max=m: 0:00:00AM -> 0AM, 0:00:01AM -> 0AM, 0:01:01AM -> 0:01AM
|
|
307
311
|
* min=h, max=s: 0:00:00AM -> 0AM, 0:00:01AM -> 0:00:01AM, 0:01:01AM -> 0:01:01AM
|
|
@@ -359,13 +363,14 @@ function formatDateFriendlyParts(
|
|
|
359
363
|
*
|
|
360
364
|
* @param date The date to format.
|
|
361
365
|
* @param timezone The time zone to use for formatting.
|
|
362
|
-
* @param
|
|
363
|
-
* @param
|
|
364
|
-
* @param
|
|
365
|
-
* @param
|
|
366
|
-
* @param
|
|
367
|
-
* @param
|
|
368
|
-
* @param
|
|
366
|
+
* @param options
|
|
367
|
+
* @param options.baseDate The base date to use for comparison (default is the current date).
|
|
368
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
369
|
+
* @param options.timeFirst If true, the time is shown before the date (default false).
|
|
370
|
+
* @param options.dateOnly If true, only the date is shown (default false).
|
|
371
|
+
* @param options.timeOnly If true, only the time is shown (default false).
|
|
372
|
+
* @param options.maxPrecision The maximum precision to show for time (default 'minute').
|
|
373
|
+
* @param options.minPrecision The minimum precision to always show for time (default 'hour').
|
|
369
374
|
* @returns Human-readable string representing the date and time.
|
|
370
375
|
*/
|
|
371
376
|
export function formatDateFriendly(
|
|
@@ -430,6 +435,12 @@ export function formatDateFriendly(
|
|
|
430
435
|
* @param end The end date and time.
|
|
431
436
|
* @param timezone The time zone to use for formatting.
|
|
432
437
|
* @param options Additional options for formatting the displayed date, taken from `formatDateFriendlyString()`.
|
|
438
|
+
* @param options.baseDate The base date to use for comparison (default is the current date).
|
|
439
|
+
* @param options.includeTz Whether to include the time zone in the output (default true).
|
|
440
|
+
* @param options.timeFirst If true, the time is shown before the date (default false).
|
|
441
|
+
* @param options.dateOnly If true, only the date is shown (default false).
|
|
442
|
+
* @param options.maxPrecision The maximum precision to show for time (default 'minute').
|
|
443
|
+
* @param options.minPrecision The minimum precision to always show for time (default 'hour').
|
|
433
444
|
* @returns Human-readable string representing the datetime range.
|
|
434
445
|
*/
|
|
435
446
|
export function formatDateRangeFriendly(
|
package/src/interval.ts
CHANGED
|
@@ -5,7 +5,7 @@ export const DAY_IN_MILLISECONDS = 24 * HOUR_IN_MILLISECONDS;
|
|
|
5
5
|
|
|
6
6
|
/**
|
|
7
7
|
* Makes an interval (in milliseconds).
|
|
8
|
-
*
|
|
8
|
+
* @param param
|
|
9
9
|
* @param param.days The number of days in the interval.
|
|
10
10
|
* @param param.hours The number of hours in the interval.
|
|
11
11
|
* @param param.minutes The number of minutes in the interval.
|
|
@@ -89,6 +89,7 @@ export function formatIntervalRelative(
|
|
|
89
89
|
* Format an interval (in milliseconds) to a human-readable string like HH:MM or +HH:MM.
|
|
90
90
|
*
|
|
91
91
|
* @param interval Time interval in milliseconds.
|
|
92
|
+
* @param options
|
|
92
93
|
* @param options.signed Whether to include the sign in the output.
|
|
93
94
|
* @returns Human-readable string representing the interval in minutes.
|
|
94
95
|
*/
|