sendou.ink/app/hooks/intl/useUserIntlPreference.ts

46 lines
1.4 KiB
TypeScript

import { useTranslation } from "react-i18next";
import type { UserPreferences } from "~/db/tables";
import { useUser } from "~/features/auth/core/user";
import { useHydrated } from "../useHydrated";
/**
* Resolves the language and hour cycle to use when formatting dates and times
* for the current user. Prefers a browser language sharing a base tag with the
* active i18n language (e.g. `en-GB` over site `en`) for regional formatting.
* `isLoaded` is `false` until hydration; gate locale-dependent output on it to
* avoid hydration mismatches.
*/
export function useUserIntlPreference() {
const { i18n } = useTranslation();
const user = useUser();
const hydrated = useHydrated();
const browserLanguages = hydrated ? navigator.languages : [];
// does the user want to use their browser language even if the site is in another language?
const language =
browserLanguages.find((lang) => compareLanguages(lang, i18n.language)) ??
i18n.language;
return {
language,
hourCycle: resolveHourCycle(user?.preferences?.clockFormat),
isLoaded: hydrated,
};
}
function resolveHourCycle(
clockFormat: UserPreferences["clockFormat"],
): "h12" | "h23" | undefined {
if (clockFormat === "12h") return "h12";
if (clockFormat === "24h") return "h23";
return undefined;
}
function compareLanguages(a: string, b: string) {
const baseA = a.split("-")[0];
const baseB = b.split("-")[0];
return baseA.toUpperCase() === baseB.toUpperCase();
}