2023-01-10 05:06:12 -08:00
|
|
|
/* eslint-disable @typescript-eslint/unbound-method */
|
|
|
|
/* eslint-disable @typescript-eslint/explicit-member-accessibility */
|
2023-01-27 05:56:56 -08:00
|
|
|
import { DateTime } from 'luxon';
|
2023-02-02 03:35:38 -08:00
|
|
|
import type { DateTimeUnit, DurationLike, DurationObjectUnits, LocaleOptions } from 'luxon';
|
2023-01-10 05:06:12 -08:00
|
|
|
import type { ExtensionMap } from './Extensions';
|
|
|
|
|
|
|
|
type DurationUnit =
|
|
|
|
| 'milliseconds'
|
|
|
|
| 'seconds'
|
|
|
|
| 'minutes'
|
|
|
|
| 'hours'
|
|
|
|
| 'days'
|
|
|
|
| 'weeks'
|
|
|
|
| 'months'
|
|
|
|
| 'quarter'
|
|
|
|
| 'years';
|
|
|
|
type DatePart =
|
|
|
|
| 'day'
|
2023-02-02 03:35:38 -08:00
|
|
|
| 'week'
|
2023-01-10 05:06:12 -08:00
|
|
|
| 'month'
|
|
|
|
| 'year'
|
|
|
|
| 'hour'
|
|
|
|
| 'minute'
|
|
|
|
| 'second'
|
|
|
|
| 'millisecond'
|
|
|
|
| 'weekNumber'
|
|
|
|
| 'yearDayNumber'
|
|
|
|
| 'weekday';
|
|
|
|
|
|
|
|
const DURATION_MAP: Record<string, DurationUnit> = {
|
|
|
|
day: 'days',
|
|
|
|
month: 'months',
|
|
|
|
year: 'years',
|
|
|
|
week: 'weeks',
|
|
|
|
hour: 'hours',
|
|
|
|
minute: 'minutes',
|
|
|
|
second: 'seconds',
|
|
|
|
millisecond: 'milliseconds',
|
|
|
|
ms: 'milliseconds',
|
|
|
|
sec: 'seconds',
|
|
|
|
secs: 'seconds',
|
|
|
|
hr: 'hours',
|
|
|
|
hrs: 'hours',
|
|
|
|
min: 'minutes',
|
|
|
|
mins: 'minutes',
|
|
|
|
};
|
|
|
|
|
|
|
|
const DATETIMEUNIT_MAP: Record<string, DateTimeUnit> = {
|
|
|
|
days: 'day',
|
|
|
|
months: 'month',
|
|
|
|
years: 'year',
|
|
|
|
hours: 'hour',
|
|
|
|
minutes: 'minute',
|
|
|
|
seconds: 'second',
|
|
|
|
milliseconds: 'millisecond',
|
|
|
|
hrs: 'hour',
|
|
|
|
hr: 'hour',
|
|
|
|
mins: 'minute',
|
|
|
|
min: 'minute',
|
|
|
|
secs: 'second',
|
|
|
|
sec: 'second',
|
|
|
|
ms: 'millisecond',
|
|
|
|
};
|
|
|
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
function isDateTime(date: any): date is DateTime {
|
|
|
|
if (date) {
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
|
|
|
|
return DateTime.isDateTime(date);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
function generateDurationObject(durationValue: number, unit: DurationUnit) {
|
|
|
|
const convertedUnit = DURATION_MAP[unit] || unit;
|
|
|
|
return { [`${convertedUnit}`]: durationValue } as DurationObjectUnits;
|
|
|
|
}
|
|
|
|
|
|
|
|
function beginningOf(date: Date | DateTime, extraArgs: DurationUnit[]): Date {
|
|
|
|
const [unit = 'week'] = extraArgs;
|
|
|
|
|
|
|
|
if (isDateTime(date)) {
|
|
|
|
return date.startOf(DATETIMEUNIT_MAP[unit] || unit).toJSDate();
|
|
|
|
}
|
|
|
|
let datetime = DateTime.fromJSDate(date);
|
|
|
|
if (date.getTimezoneOffset() === 0) {
|
|
|
|
datetime = datetime.setZone('UTC');
|
|
|
|
}
|
|
|
|
return datetime.startOf(DATETIMEUNIT_MAP[unit] || unit).toJSDate();
|
|
|
|
}
|
|
|
|
|
|
|
|
function endOfMonth(date: Date | DateTime): Date {
|
|
|
|
if (isDateTime(date)) {
|
|
|
|
return date.endOf('month').toJSDate();
|
|
|
|
}
|
|
|
|
return DateTime.fromJSDate(date).endOf('month').toJSDate();
|
|
|
|
}
|
|
|
|
|
|
|
|
function extract(inputDate: Date | DateTime, extraArgs: DatePart[]): number | Date {
|
2023-02-02 03:35:38 -08:00
|
|
|
let [part] = extraArgs;
|
2023-01-10 05:06:12 -08:00
|
|
|
let date = inputDate;
|
|
|
|
if (isDateTime(date)) {
|
|
|
|
date = date.toJSDate();
|
|
|
|
}
|
|
|
|
if (part === 'yearDayNumber') {
|
|
|
|
const firstDayOfTheYear = new Date(date.getFullYear(), 0, 0);
|
|
|
|
const diff =
|
|
|
|
date.getTime() -
|
|
|
|
firstDayOfTheYear.getTime() +
|
|
|
|
(firstDayOfTheYear.getTimezoneOffset() - date.getTimezoneOffset()) * 60 * 1000;
|
|
|
|
return Math.floor(diff / (1000 * 60 * 60 * 24));
|
|
|
|
}
|
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
if (part === 'week') {
|
|
|
|
part = 'weekNumber';
|
|
|
|
}
|
|
|
|
|
2023-01-10 05:06:12 -08:00
|
|
|
return DateTime.fromJSDate(date).get((DATETIMEUNIT_MAP[part] as keyof DateTime) || part);
|
|
|
|
}
|
|
|
|
|
|
|
|
function format(date: Date | DateTime, extraArgs: unknown[]): string {
|
|
|
|
const [dateFormat, localeOpts = {}] = extraArgs as [string, LocaleOptions];
|
|
|
|
if (isDateTime(date)) {
|
|
|
|
return date.toFormat(dateFormat, { ...localeOpts });
|
|
|
|
}
|
|
|
|
return DateTime.fromJSDate(date).toFormat(dateFormat, { ...localeOpts });
|
|
|
|
}
|
|
|
|
|
|
|
|
function isBetween(date: Date | DateTime, extraArgs: unknown[]): boolean {
|
|
|
|
const [first, second] = extraArgs as string[];
|
|
|
|
const firstDate = new Date(first);
|
|
|
|
const secondDate = new Date(second);
|
|
|
|
|
|
|
|
if (firstDate > secondDate) {
|
|
|
|
return secondDate < date && date < firstDate;
|
|
|
|
}
|
|
|
|
return secondDate > date && date > firstDate;
|
|
|
|
}
|
|
|
|
|
2023-01-23 05:28:17 -08:00
|
|
|
function isDst(date: Date | DateTime): boolean {
|
|
|
|
if (isDateTime(date)) {
|
|
|
|
return date.isInDST;
|
|
|
|
}
|
2023-01-10 05:06:12 -08:00
|
|
|
return DateTime.fromJSDate(date).isInDST;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isInLast(date: Date | DateTime, extraArgs: unknown[]): boolean {
|
|
|
|
const [durationValue = 0, unit = 'minutes'] = extraArgs as [number, DurationUnit];
|
|
|
|
|
|
|
|
const dateInThePast = DateTime.now().minus(generateDurationObject(durationValue, unit));
|
|
|
|
let thisDate = date;
|
|
|
|
if (!isDateTime(thisDate)) {
|
|
|
|
thisDate = DateTime.fromJSDate(thisDate);
|
|
|
|
}
|
|
|
|
return dateInThePast <= thisDate && thisDate <= DateTime.now();
|
|
|
|
}
|
|
|
|
|
2023-01-23 05:28:17 -08:00
|
|
|
function isWeekend(date: Date | DateTime): boolean {
|
2023-01-10 05:06:12 -08:00
|
|
|
enum DAYS {
|
|
|
|
saturday = 6,
|
|
|
|
sunday = 7,
|
|
|
|
}
|
2023-01-23 05:28:17 -08:00
|
|
|
if (isDateTime(date)) {
|
|
|
|
return [DAYS.saturday, DAYS.sunday].includes(date.weekday);
|
|
|
|
}
|
2023-01-10 05:06:12 -08:00
|
|
|
return [DAYS.saturday, DAYS.sunday].includes(DateTime.fromJSDate(date).weekday);
|
|
|
|
}
|
|
|
|
|
2023-01-16 05:01:58 -08:00
|
|
|
function minus(date: Date | DateTime, extraArgs: unknown[]): Date | DateTime {
|
|
|
|
if (isDateTime(date) && extraArgs.length === 1) {
|
|
|
|
return date.minus(extraArgs[0] as DurationLike);
|
|
|
|
}
|
|
|
|
|
2023-01-10 05:06:12 -08:00
|
|
|
const [durationValue = 0, unit = 'minutes'] = extraArgs as [number, DurationUnit];
|
|
|
|
|
|
|
|
if (isDateTime(date)) {
|
|
|
|
return date.minus(generateDurationObject(durationValue, unit)).toJSDate();
|
|
|
|
}
|
|
|
|
return DateTime.fromJSDate(date).minus(generateDurationObject(durationValue, unit)).toJSDate();
|
|
|
|
}
|
|
|
|
|
2023-01-16 05:01:58 -08:00
|
|
|
function plus(date: Date | DateTime, extraArgs: unknown[]): Date | DateTime {
|
|
|
|
if (isDateTime(date) && extraArgs.length === 1) {
|
|
|
|
return date.plus(extraArgs[0] as DurationLike);
|
|
|
|
}
|
|
|
|
|
2023-01-10 05:06:12 -08:00
|
|
|
const [durationValue = 0, unit = 'minutes'] = extraArgs as [number, DurationUnit];
|
|
|
|
|
|
|
|
if (isDateTime(date)) {
|
|
|
|
return date.plus(generateDurationObject(durationValue, unit)).toJSDate();
|
|
|
|
}
|
|
|
|
return DateTime.fromJSDate(date).plus(generateDurationObject(durationValue, unit)).toJSDate();
|
|
|
|
}
|
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
endOfMonth.doc = {
|
|
|
|
name: 'endOfMonth',
|
|
|
|
returnType: 'Date',
|
|
|
|
description: 'Transforms a date to the last possible moment that lies within the month',
|
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
isDst.doc = {
|
|
|
|
name: 'isDst',
|
|
|
|
returnType: 'boolean',
|
|
|
|
description: 'Checks if a Date is within Daylight Savings Time',
|
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
isWeekend.doc = {
|
|
|
|
name: 'isWeekend',
|
|
|
|
returnType: 'boolean',
|
|
|
|
description: 'Checks if the Date falls on a Saturday or Sunday',
|
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
// @TODO_NEXT_PHASE: Surface extensions below which take args
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
beginningOf.doc = {
|
|
|
|
name: 'beginningOf',
|
|
|
|
returnType: 'Date',
|
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
extract.doc = {
|
|
|
|
name: 'extract',
|
|
|
|
returnType: 'number',
|
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
2023-02-02 03:35:38 -08:00
|
|
|
format.doc = {
|
|
|
|
name: 'format',
|
|
|
|
returnType: '(?)',
|
|
|
|
};
|
|
|
|
|
|
|
|
isBetween.doc = {
|
|
|
|
name: 'isBetween',
|
|
|
|
returnType: 'boolean',
|
|
|
|
};
|
|
|
|
|
|
|
|
isInLast.doc = {
|
|
|
|
name: 'isInLast',
|
|
|
|
returnType: 'boolean',
|
|
|
|
};
|
|
|
|
|
|
|
|
minus.doc = {
|
|
|
|
name: 'minus',
|
|
|
|
returnType: 'Date',
|
|
|
|
};
|
|
|
|
|
|
|
|
plus.doc = {
|
|
|
|
name: 'plus',
|
|
|
|
returnType: 'Date',
|
|
|
|
};
|
2023-01-10 05:06:12 -08:00
|
|
|
|
|
|
|
export const dateExtensions: ExtensionMap = {
|
|
|
|
typeName: 'Date',
|
|
|
|
functions: {
|
|
|
|
beginningOf,
|
|
|
|
endOfMonth,
|
|
|
|
extract,
|
|
|
|
isBetween,
|
|
|
|
isDst,
|
|
|
|
isInLast,
|
|
|
|
isWeekend,
|
|
|
|
minus,
|
|
|
|
plus,
|
|
|
|
format,
|
|
|
|
},
|
|
|
|
};
|