|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let formatterCache = new Map<string, Intl.DateTimeFormat>(); |
|
|
|
interface ResolvedDateTimeFormatOptions extends Intl.ResolvedDateTimeFormatOptions { |
|
hourCycle?: Intl.DateTimeFormatOptions['hourCycle'] |
|
} |
|
|
|
interface DateRangeFormatPart extends Intl.DateTimeFormatPart { |
|
source: 'startRange' | 'endRange' | 'shared' |
|
} |
|
|
|
|
|
export class DateFormatter implements Intl.DateTimeFormat { |
|
private formatter: Intl.DateTimeFormat; |
|
private options: Intl.DateTimeFormatOptions; |
|
private resolvedHourCycle: Intl.DateTimeFormatOptions['hourCycle']; |
|
|
|
constructor(locale: string, options: Intl.DateTimeFormatOptions = {}) { |
|
this.formatter = getCachedDateFormatter(locale, options); |
|
this.options = options; |
|
} |
|
|
|
|
|
format(value: Date): string { |
|
return this.formatter.format(value); |
|
} |
|
|
|
|
|
formatToParts(value: Date): Intl.DateTimeFormatPart[] { |
|
return this.formatter.formatToParts(value); |
|
} |
|
|
|
|
|
formatRange(start: Date, end: Date): string { |
|
|
|
if (typeof this.formatter.formatRange === 'function') { |
|
|
|
return this.formatter.formatRange(start, end); |
|
} |
|
|
|
if (end < start) { |
|
throw new RangeError('End date must be >= start date'); |
|
} |
|
|
|
|
|
return `${this.formatter.format(start)} – ${this.formatter.format(end)}`; |
|
} |
|
|
|
|
|
formatRangeToParts(start: Date, end: Date): DateRangeFormatPart[] { |
|
|
|
if (typeof this.formatter.formatRangeToParts === 'function') { |
|
|
|
return this.formatter.formatRangeToParts(start, end); |
|
} |
|
|
|
if (end < start) { |
|
throw new RangeError('End date must be >= start date'); |
|
} |
|
|
|
let startParts = this.formatter.formatToParts(start); |
|
let endParts = this.formatter.formatToParts(end); |
|
return [ |
|
...startParts.map(p => ({...p, source: 'startRange'} as DateRangeFormatPart)), |
|
{type: 'literal', value: ' – ', source: 'shared'}, |
|
...endParts.map(p => ({...p, source: 'endRange'} as DateRangeFormatPart)) |
|
]; |
|
} |
|
|
|
|
|
resolvedOptions(): ResolvedDateTimeFormatOptions { |
|
let resolvedOptions = this.formatter.resolvedOptions() as ResolvedDateTimeFormatOptions; |
|
if (hasBuggyResolvedHourCycle()) { |
|
if (!this.resolvedHourCycle) { |
|
this.resolvedHourCycle = getResolvedHourCycle(resolvedOptions.locale, this.options); |
|
} |
|
resolvedOptions.hourCycle = this.resolvedHourCycle; |
|
resolvedOptions.hour12 = this.resolvedHourCycle === 'h11' || this.resolvedHourCycle === 'h12'; |
|
} |
|
|
|
|
|
|
|
if (resolvedOptions.calendar === 'ethiopic-amete-alem') { |
|
resolvedOptions.calendar = 'ethioaa'; |
|
} |
|
|
|
return resolvedOptions; |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const hour12Preferences = { |
|
true: { |
|
|
|
ja: 'h11' |
|
}, |
|
false: { |
|
|
|
} |
|
}; |
|
|
|
function getCachedDateFormatter(locale: string, options: Intl.DateTimeFormatOptions = {}): Intl.DateTimeFormat { |
|
|
|
|
|
if (typeof options.hour12 === 'boolean' && hasBuggyHour12Behavior()) { |
|
options = {...options}; |
|
let pref = hour12Preferences[String(options.hour12)][locale.split('-')[0]]; |
|
let defaultHourCycle = options.hour12 ? 'h12' : 'h23'; |
|
options.hourCycle = pref ?? defaultHourCycle; |
|
delete options.hour12; |
|
} |
|
|
|
let cacheKey = locale + (options ? Object.entries(options).sort((a, b) => a[0] < b[0] ? -1 : 1).join() : ''); |
|
if (formatterCache.has(cacheKey)) { |
|
return formatterCache.get(cacheKey)!; |
|
} |
|
|
|
let numberFormatter = new Intl.DateTimeFormat(locale, options); |
|
formatterCache.set(cacheKey, numberFormatter); |
|
return numberFormatter; |
|
} |
|
|
|
let _hasBuggyHour12Behavior: boolean | null = null; |
|
function hasBuggyHour12Behavior() { |
|
if (_hasBuggyHour12Behavior == null) { |
|
_hasBuggyHour12Behavior = new Intl.DateTimeFormat('en-US', { |
|
hour: 'numeric', |
|
hour12: false |
|
}).format(new Date(2020, 2, 3, 0)) === '24'; |
|
} |
|
|
|
return _hasBuggyHour12Behavior; |
|
} |
|
|
|
let _hasBuggyResolvedHourCycle: boolean | null = null; |
|
function hasBuggyResolvedHourCycle() { |
|
if (_hasBuggyResolvedHourCycle == null) { |
|
_hasBuggyResolvedHourCycle = (new Intl.DateTimeFormat('fr', { |
|
hour: 'numeric', |
|
hour12: false |
|
}).resolvedOptions() as ResolvedDateTimeFormatOptions).hourCycle === 'h12'; |
|
} |
|
|
|
return _hasBuggyResolvedHourCycle; |
|
} |
|
|
|
function getResolvedHourCycle(locale: string, options: Intl.DateTimeFormatOptions) { |
|
if (!options.timeStyle && !options.hour) { |
|
return undefined; |
|
} |
|
|
|
|
|
|
|
locale = locale.replace(/(-u-)?-nu-[a-zA-Z0-9]+/, ''); |
|
locale += (locale.includes('-u-') ? '' : '-u') + '-nu-latn'; |
|
let formatter = getCachedDateFormatter(locale, { |
|
...options, |
|
timeZone: undefined |
|
}); |
|
|
|
let min = parseInt(formatter.formatToParts(new Date(2020, 2, 3, 0)).find(p => p.type === 'hour')!.value, 10); |
|
let max = parseInt(formatter.formatToParts(new Date(2020, 2, 3, 23)).find(p => p.type === 'hour')!.value, 10); |
|
|
|
if (min === 0 && max === 23) { |
|
return 'h23'; |
|
} |
|
|
|
if (min === 24 && max === 23) { |
|
return 'h24'; |
|
} |
|
|
|
if (min === 0 && max === 11) { |
|
return 'h11'; |
|
} |
|
|
|
if (min === 12 && max === 11) { |
|
return 'h12'; |
|
} |
|
|
|
throw new Error('Unexpected hour cycle result'); |
|
} |
|
|