mirror of
https://github.com/Sendouc/sendou.ink.git
synced 2026-05-24 12:29:30 -05:00
46 lines
1.4 KiB
TypeScript
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();
|
|
}
|