|
@ -1,9 +1,121 @@ |
|
|
'use strict'; |
|
|
'use strict'; |
|
|
import * as moment from 'moment'; |
|
|
|
|
|
|
|
|
import { distanceInWordsToNow as _fromNow, format as _format } from 'date-fns'; |
|
|
|
|
|
import * as en from 'date-fns/locale/en'; |
|
|
|
|
|
|
|
|
const MillisecondsPerMinute = 60000; // 60 * 1000
|
|
|
const MillisecondsPerMinute = 60000; // 60 * 1000
|
|
|
const MillisecondsPerDay = 86400000; // 24 * 60 * 60 * 1000
|
|
|
const MillisecondsPerDay = 86400000; // 24 * 60 * 60 * 1000
|
|
|
|
|
|
|
|
|
|
|
|
// Taken from https://github.com/date-fns/date-fns/blob/601bc8e5708cbaebee5389bdaf51c2b4b33b73c4/src/locale/en/build_distance_in_words_locale/index.js
|
|
|
|
|
|
function buildDistanceInWordsLocale() { |
|
|
|
|
|
const distanceInWordsLocale: { [key: string]: string | { one: string, other: string } } = { |
|
|
|
|
|
lessThanXSeconds: { |
|
|
|
|
|
one: 'less than a second', |
|
|
|
|
|
other: 'less than {{count}} seconds' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
xSeconds: { |
|
|
|
|
|
one: '1 second', |
|
|
|
|
|
other: '{{count}} seconds' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
halfAMinute: 'half a minute', |
|
|
|
|
|
|
|
|
|
|
|
lessThanXMinutes: { |
|
|
|
|
|
one: 'a few seconds', |
|
|
|
|
|
other: 'less than {{count}} minutes' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
xMinutes: { |
|
|
|
|
|
one: 'a minute', |
|
|
|
|
|
other: '{{count}} minutes' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
aboutXHours: { |
|
|
|
|
|
one: 'an hour', |
|
|
|
|
|
other: '{{count}} hours' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
xHours: { |
|
|
|
|
|
one: 'an hour', |
|
|
|
|
|
other: '{{count}} hours' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
xDays: { |
|
|
|
|
|
one: 'a day', |
|
|
|
|
|
other: '{{count}} days' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
aboutXMonths: { |
|
|
|
|
|
one: 'a month', |
|
|
|
|
|
other: '{{count}} months' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
xMonths: { |
|
|
|
|
|
one: 'a month', |
|
|
|
|
|
other: '{{count}} months' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
aboutXYears: { |
|
|
|
|
|
one: 'a year', |
|
|
|
|
|
other: '{{count}} years' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
xYears: { |
|
|
|
|
|
one: 'a year', |
|
|
|
|
|
other: '{{count}} years' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
overXYears: { |
|
|
|
|
|
one: 'a year', |
|
|
|
|
|
other: '{{count}} years' |
|
|
|
|
|
}, |
|
|
|
|
|
|
|
|
|
|
|
almostXYears: { |
|
|
|
|
|
one: 'a year', |
|
|
|
|
|
other: '{{count}} years' |
|
|
|
|
|
} |
|
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
function localize(token: string, count: number, options: any) { |
|
|
|
|
|
options = options || {}; |
|
|
|
|
|
|
|
|
|
|
|
const result = distanceInWordsLocale[token]; |
|
|
|
|
|
|
|
|
|
|
|
let value: string; |
|
|
|
|
|
if (typeof result === 'string') { |
|
|
|
|
|
value = result; |
|
|
|
|
|
} |
|
|
|
|
|
else { |
|
|
|
|
|
if (count === 12 && token === 'xMonths') { |
|
|
|
|
|
token = 'aboutXYears'; |
|
|
|
|
|
count = 1; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
if (count === 1) { |
|
|
|
|
|
value = result.one; |
|
|
|
|
|
} |
|
|
|
|
|
else { |
|
|
|
|
|
value = result.other.replace('{{count}}', count.toString()); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
if (!options.addSuffix) return value; |
|
|
|
|
|
|
|
|
|
|
|
if (options.comparison > 0) return 'in ' + value; |
|
|
|
|
|
|
|
|
|
|
|
return value + ' ago'; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
return { |
|
|
|
|
|
localize: localize |
|
|
|
|
|
}; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// Monkey patch the locale to customize the wording
|
|
|
|
|
|
(en as any).distanceInWords = buildDistanceInWordsLocale(); |
|
|
|
|
|
|
|
|
|
|
|
const formatterOptions = { addSuffix: true, locale: en }; |
|
|
|
|
|
|
|
|
export namespace Dates { |
|
|
export namespace Dates { |
|
|
|
|
|
|
|
|
export interface IDateFormatter { |
|
|
export interface IDateFormatter { |
|
@ -28,6 +140,9 @@ export namespace Dates { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
export function toFormatter(date: Date): IDateFormatter { |
|
|
export function toFormatter(date: Date): IDateFormatter { |
|
|
return moment(date); |
|
|
|
|
|
|
|
|
return { |
|
|
|
|
|
fromNow: () => _fromNow(date, formatterOptions), |
|
|
|
|
|
format: (format: string) => _format(date, format) |
|
|
|
|
|
}; |
|
|
} |
|
|
} |
|
|
} |
|
|
} |