2023-04-07 15:56:02 +03:00
|
|
|
import {parseUTCDate, formatMonthYYYY, formatDayShort} from '../../util/date'
|
2022-11-22 15:50:58 +03:00
|
|
|
|
|
|
|
const browserDateFormat = Intl.DateTimeFormat(navigator.language, { hour: 'numeric' })
|
|
|
|
|
2023-02-14 14:54:22 +03:00
|
|
|
const is12HourClock = function () {
|
2022-11-22 15:50:58 +03:00
|
|
|
return browserDateFormat.resolvedOptions().hour12
|
|
|
|
}
|
|
|
|
|
|
|
|
const monthIntervalFormatter = {
|
|
|
|
long(isoDate, options) {
|
|
|
|
const formatted = this.short(isoDate, options)
|
|
|
|
return options.isBucketPartial ? `Partial of ${formatted}` : formatted
|
|
|
|
},
|
|
|
|
short(isoDate, _options) {
|
2023-04-07 15:56:02 +03:00
|
|
|
return formatMonthYYYY(parseUTCDate(isoDate))
|
2022-11-22 15:50:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const weekIntervalFormatter = {
|
|
|
|
long(isoDate, options) {
|
|
|
|
const formatted = this.short(isoDate, options)
|
|
|
|
return options.isBucketPartial ? `Partial week of ${formatted}` : `Week of ${formatted}`
|
|
|
|
},
|
2023-02-14 14:54:22 +03:00
|
|
|
short(isoDate, options) {
|
2023-04-07 15:56:02 +03:00
|
|
|
return formatDayShort(parseUTCDate(isoDate), options.shouldShowYear)
|
2022-11-22 15:50:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const dateIntervalFormatter = {
|
|
|
|
long(isoDate, _options) {
|
2023-04-07 15:56:02 +03:00
|
|
|
return parseUTCDate(isoDate).format('ddd, D MMM')
|
2022-11-22 15:50:58 +03:00
|
|
|
},
|
2023-02-14 14:54:22 +03:00
|
|
|
short(isoDate, options) {
|
2023-04-07 15:56:02 +03:00
|
|
|
return formatDayShort(parseUTCDate(isoDate), options.shouldShowYear)
|
2022-11-22 15:50:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const hourIntervalFormatter = {
|
|
|
|
long(isoDate, options) {
|
|
|
|
return this.short(isoDate, options)
|
|
|
|
},
|
|
|
|
short(isoDate, _options) {
|
|
|
|
if (is12HourClock()) {
|
2023-04-07 15:56:02 +03:00
|
|
|
return parseUTCDate(isoDate).format('ha')
|
2022-11-22 15:50:58 +03:00
|
|
|
} else {
|
2023-04-07 15:56:02 +03:00
|
|
|
return parseUTCDate(isoDate).format('HH:mm')
|
2022-11-22 15:50:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const minuteIntervalFormatter = {
|
|
|
|
long(isoDate, options) {
|
|
|
|
if (options.period == 'realtime') {
|
|
|
|
const minutesAgo = Math.abs(isoDate)
|
|
|
|
return minutesAgo === 1 ? '1 minute ago' : minutesAgo + ' minutes ago'
|
|
|
|
} else {
|
|
|
|
return this.short(isoDate, options)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
short(isoDate, options) {
|
|
|
|
if (options.period === 'realtime') return isoDate + 'm'
|
|
|
|
|
|
|
|
if (is12HourClock()) {
|
2023-04-07 15:56:02 +03:00
|
|
|
return parseUTCDate(isoDate).format('h:mma')
|
2022-11-22 15:50:58 +03:00
|
|
|
} else {
|
2023-04-07 15:56:02 +03:00
|
|
|
return parseUTCDate(isoDate).format('HH:mm')
|
2022-11-22 15:50:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Each interval has a different date and time format. This object maps each
|
|
|
|
// interval with two functions: `long` and `short`, that formats date and time
|
|
|
|
// accordingly.
|
|
|
|
const factory = {
|
|
|
|
month: monthIntervalFormatter,
|
|
|
|
week: weekIntervalFormatter,
|
|
|
|
date: dateIntervalFormatter,
|
|
|
|
hour: hourIntervalFormatter,
|
|
|
|
minute: minuteIntervalFormatter
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a function that formats a ISO 8601 timestamp based on the given
|
|
|
|
* arguments.
|
|
|
|
*
|
|
|
|
* The preferred date and time format in the dashboard depends on the selected
|
|
|
|
* interval and period. For example, in real-time view only the time is necessary,
|
|
|
|
* while other intervals require dates to be displayed.
|
2023-02-14 14:54:22 +03:00
|
|
|
* @param {Object} config - Configuration object for determining formatter.
|
2022-11-22 15:50:58 +03:00
|
|
|
*
|
2023-02-14 14:54:22 +03:00
|
|
|
* @param {string} config.interval - The interval of the query, e.g. `minute`, `hour`
|
|
|
|
* @param {boolean} config.longForm - Whether the formatted result should be in long or
|
2022-11-22 15:50:58 +03:00
|
|
|
* short form.
|
2023-02-14 14:54:22 +03:00
|
|
|
* @param {string} config.period - The period of the query, e.g. `12mo`, `day`
|
|
|
|
* @param {boolean} config.isPeriodFull - Indicates whether the interval has been cut
|
2022-11-22 15:50:58 +03:00
|
|
|
* off by the requested date range or not. If false, the returned formatted date
|
|
|
|
* indicates this cut off, e.g. `Partial week of November 8`.
|
2023-02-14 14:54:22 +03:00
|
|
|
* @param {boolean} config.shouldShowYear - Should the year be appended to the date?
|
|
|
|
* Defaults to false. Rendering year string is a newer opt-in feature to be enabled where needed.
|
2022-11-22 15:50:58 +03:00
|
|
|
*/
|
2023-02-14 14:54:22 +03:00
|
|
|
export default function dateFormatter({ interval, longForm, period, isPeriodFull, shouldShowYear = false }) {
|
2022-11-22 15:50:58 +03:00
|
|
|
const displayMode = longForm ? 'long' : 'short'
|
2023-02-14 14:54:22 +03:00
|
|
|
const options = { period: period, interval: interval, isBucketPartial: !isPeriodFull, shouldShowYear }
|
|
|
|
return function (isoDate, _index, _ticks) {
|
2022-11-22 15:50:58 +03:00
|
|
|
return factory[interval][displayMode](isoDate, options)
|
|
|
|
}
|
|
|
|
}
|