mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-07 08:02:23 +00:00
Closes #29750. Reporter flagged that 繁體中文 displayed the TW flag instead of the PRC flag. Rather than picking a side, drop the language-flag pairings entirely — languages aren't countries (English ≠ GB, Portuguese ≠ PT, Mandarin variants ≠ any single jurisdiction), and endonyms are unambiguous. - LOCALE_META: strip flagCountryCode field - LanguageSwitcher: remove LocaleFlagIcon component + both call sites - main.tsx: drop flag-icons CSS import - package.json: uninstall flag-icons
123 lines
3 KiB
TypeScript
123 lines
3 KiB
TypeScript
import { createContext, useContext, useState, useCallback, type ReactNode } from "react";
|
|
import type { Locale, Translations } from "./types";
|
|
import { en } from "./en";
|
|
import { zh } from "./zh";
|
|
import { zhHant } from "./zh-hant";
|
|
import { ja } from "./ja";
|
|
import { de } from "./de";
|
|
import { es } from "./es";
|
|
import { fr } from "./fr";
|
|
import { tr } from "./tr";
|
|
import { uk } from "./uk";
|
|
import { af } from "./af";
|
|
import { ko } from "./ko";
|
|
import { it } from "./it";
|
|
import { ga } from "./ga";
|
|
import { pt } from "./pt";
|
|
import { ru } from "./ru";
|
|
import { hu } from "./hu";
|
|
|
|
const TRANSLATIONS: Record<Locale, Translations> = {
|
|
en,
|
|
zh,
|
|
"zh-hant": zhHant,
|
|
ja,
|
|
de,
|
|
es,
|
|
fr,
|
|
tr,
|
|
uk,
|
|
af,
|
|
ko,
|
|
it,
|
|
ga,
|
|
pt,
|
|
ru,
|
|
hu,
|
|
};
|
|
|
|
// Display metadata for the language picker — endonym (native name) so users
|
|
// recognize their language even if they don't speak the current UI language.
|
|
// Exposed as a constant so the LanguageSwitcher and any future settings page
|
|
// can share the same list.
|
|
//
|
|
// We intentionally do NOT pair locales with country flags. Languages are not
|
|
// countries (English ≠ GB, Portuguese ≠ PT, Spanish ≠ ES, Chinese variants ≠
|
|
// any single jurisdiction). Endonyms are unambiguous and avoid the political
|
|
// mismapping that flag pairings inevitably create.
|
|
export const LOCALE_META: Record<Locale, { name: string }> = {
|
|
en: { name: "English" },
|
|
zh: { name: "简体中文" },
|
|
"zh-hant": { name: "繁體中文" },
|
|
ja: { name: "日本語" },
|
|
de: { name: "Deutsch" },
|
|
es: { name: "Español" },
|
|
fr: { name: "Français" },
|
|
tr: { name: "Türkçe" },
|
|
uk: { name: "Українська" },
|
|
af: { name: "Afrikaans" },
|
|
ko: { name: "한국어" },
|
|
it: { name: "Italiano" },
|
|
ga: { name: "Gaeilge" },
|
|
pt: { name: "Português" },
|
|
ru: { name: "Русский" },
|
|
hu: { name: "Magyar" },
|
|
};
|
|
|
|
const SUPPORTED_LOCALES = Object.keys(TRANSLATIONS) as Locale[];
|
|
const STORAGE_KEY = "hermes-locale";
|
|
|
|
function isLocale(value: string): value is Locale {
|
|
return (SUPPORTED_LOCALES as string[]).includes(value);
|
|
}
|
|
|
|
function getInitialLocale(): Locale {
|
|
try {
|
|
const stored = localStorage.getItem(STORAGE_KEY);
|
|
if (stored && isLocale(stored)) return stored;
|
|
} catch {
|
|
// SSR or privacy mode
|
|
}
|
|
return "en";
|
|
}
|
|
|
|
interface I18nContextValue {
|
|
locale: Locale;
|
|
setLocale: (l: Locale) => void;
|
|
t: Translations;
|
|
}
|
|
|
|
const I18nContext = createContext<I18nContextValue>({
|
|
locale: "en",
|
|
setLocale: () => {},
|
|
t: en,
|
|
});
|
|
|
|
export function I18nProvider({ children }: { children: ReactNode }) {
|
|
const [locale, setLocaleState] = useState<Locale>(getInitialLocale);
|
|
|
|
const setLocale = useCallback((l: Locale) => {
|
|
setLocaleState(l);
|
|
try {
|
|
localStorage.setItem(STORAGE_KEY, l);
|
|
} catch {
|
|
// ignore
|
|
}
|
|
}, []);
|
|
|
|
const value: I18nContextValue = {
|
|
locale,
|
|
setLocale,
|
|
t: TRANSLATIONS[locale],
|
|
};
|
|
|
|
return (
|
|
<I18nContext.Provider value={value}>
|
|
{children}
|
|
</I18nContext.Provider>
|
|
);
|
|
}
|
|
|
|
export function useI18n() {
|
|
return useContext(I18nContext);
|
|
}
|