button-card/src/common/format_time.ts

72 lines
2.9 KiB
TypeScript

import { HassConfig } from 'home-assistant-js-websocket';
import memoizeOne from 'memoize-one';
import { FrontendLocaleData, TimeFormat } from '../types/translation';
export const useAmPm = memoizeOne((locale: FrontendLocaleData): boolean => {
if (locale.time_format === TimeFormat.language || locale.time_format === TimeFormat.system) {
const testLanguage = locale.time_format === TimeFormat.language ? locale.language : undefined;
const test = new Date().toLocaleString(testLanguage);
return test.includes('AM') || test.includes('PM');
}
return locale.time_format === TimeFormat.am_pm;
});
export const formatTime = (dateObj: Date, locale: FrontendLocaleData, config: HassConfig) =>
formatTimeMem(locale, config.time_zone).format(dateObj);
const formatTimeMem = memoizeOne(
(locale: FrontendLocaleData, serverTimeZone: string) =>
new Intl.DateTimeFormat(locale.language === 'en' && !useAmPm(locale) ? 'en-u-hc-h23' : locale.language, {
hour: 'numeric',
minute: '2-digit',
hour12: useAmPm(locale),
timeZone: locale.time_zone === 'server' ? serverTimeZone : undefined,
}),
);
// 9:15:24 PM || 21:15:24
export const formatTimeWithSeconds = (dateObj: Date, locale: FrontendLocaleData, config: HassConfig) =>
formatTimeWithSecondsMem(locale, config.time_zone).format(dateObj);
const formatTimeWithSecondsMem = memoizeOne(
(locale: FrontendLocaleData, serverTimeZone: string) =>
new Intl.DateTimeFormat(locale.language === 'en' && !useAmPm(locale) ? 'en-u-hc-h23' : locale.language, {
hour: useAmPm(locale) ? 'numeric' : '2-digit',
minute: '2-digit',
second: '2-digit',
hour12: useAmPm(locale),
timeZone: locale.time_zone === 'server' ? serverTimeZone : undefined,
}),
);
// Tuesday 7:00 PM || Tuesday 19:00
export const formatTimeWeekday = (dateObj: Date, locale: FrontendLocaleData, config: HassConfig) =>
formatTimeWeekdayMem(locale, config.time_zone).format(dateObj);
const formatTimeWeekdayMem = memoizeOne(
(locale: FrontendLocaleData, serverTimeZone: string) =>
new Intl.DateTimeFormat(locale.language === 'en' && !useAmPm(locale) ? 'en-u-hc-h23' : locale.language, {
weekday: 'long',
hour: useAmPm(locale) ? 'numeric' : '2-digit',
minute: '2-digit',
hour12: useAmPm(locale),
timeZone: locale.time_zone === 'server' ? serverTimeZone : undefined,
}),
);
// 21:15
export const formatTime24h = (dateObj: Date, locale: FrontendLocaleData, config: HassConfig) =>
formatTime24hMem(locale, config.time_zone).format(dateObj);
const formatTime24hMem = memoizeOne(
(locale: FrontendLocaleData, serverTimeZone: string) =>
// en-GB to fix Chrome 24:59 to 0:59 https://stackoverflow.com/a/60898146
new Intl.DateTimeFormat('en-GB', {
hour: 'numeric',
minute: '2-digit',
hour12: false,
timeZone: locale.time_zone === 'server' ? serverTimeZone : undefined,
}),
);