mirror of
https://github.com/movie-web/movie-web.git
synced 2025-01-28 18:58:24 +00:00
Implement new country code system and new language code system
This commit is contained in:
parent
bc5e2d6f30
commit
aca7827a15
|
@ -28,6 +28,7 @@
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@formkit/auto-animate": "^0.8.1",
|
"@formkit/auto-animate": "^0.8.1",
|
||||||
"@headlessui/react": "^1.7.17",
|
"@headlessui/react": "^1.7.17",
|
||||||
|
"@ladjs/country-language": "^1.0.3",
|
||||||
"@movie-web/providers": "^2.0.2",
|
"@movie-web/providers": "^2.0.2",
|
||||||
"@noble/hashes": "^1.3.3",
|
"@noble/hashes": "^1.3.3",
|
||||||
"@react-spring/web": "^9.7.3",
|
"@react-spring/web": "^9.7.3",
|
||||||
|
@ -44,7 +45,6 @@
|
||||||
"hls.js": "^1.4.14",
|
"hls.js": "^1.4.14",
|
||||||
"i18next": "^23.7.11",
|
"i18next": "^23.7.11",
|
||||||
"immer": "^10.0.3",
|
"immer": "^10.0.3",
|
||||||
"iso-639-1": "^3.1.0",
|
|
||||||
"jwt-decode": "^4.0.0",
|
"jwt-decode": "^4.0.0",
|
||||||
"lodash.isequal": "^4.5.0",
|
"lodash.isequal": "^4.5.0",
|
||||||
"million": "^2.6.4",
|
"million": "^2.6.4",
|
||||||
|
|
|
@ -17,6 +17,9 @@ dependencies:
|
||||||
'@headlessui/react':
|
'@headlessui/react':
|
||||||
specifier: ^1.7.17
|
specifier: ^1.7.17
|
||||||
version: 1.7.17(react-dom@18.2.0)(react@18.2.0)
|
version: 1.7.17(react-dom@18.2.0)(react@18.2.0)
|
||||||
|
'@ladjs/country-language':
|
||||||
|
specifier: ^1.0.3
|
||||||
|
version: 1.0.3
|
||||||
'@movie-web/providers':
|
'@movie-web/providers':
|
||||||
specifier: ^2.0.2
|
specifier: ^2.0.2
|
||||||
version: 2.0.3
|
version: 2.0.3
|
||||||
|
@ -65,9 +68,6 @@ dependencies:
|
||||||
immer:
|
immer:
|
||||||
specifier: ^10.0.3
|
specifier: ^10.0.3
|
||||||
version: 10.0.3
|
version: 10.0.3
|
||||||
iso-639-1:
|
|
||||||
specifier: ^3.1.0
|
|
||||||
version: 3.1.0
|
|
||||||
jwt-decode:
|
jwt-decode:
|
||||||
specifier: ^4.0.0
|
specifier: ^4.0.0
|
||||||
version: 4.0.0
|
version: 4.0.0
|
||||||
|
@ -1912,6 +1912,11 @@ packages:
|
||||||
'@jridgewell/resolve-uri': 3.1.1
|
'@jridgewell/resolve-uri': 3.1.1
|
||||||
'@jridgewell/sourcemap-codec': 1.4.15
|
'@jridgewell/sourcemap-codec': 1.4.15
|
||||||
|
|
||||||
|
/@ladjs/country-language@1.0.3:
|
||||||
|
resolution: {integrity: sha512-FJROu9/hh4eqVAGDyfL8vpv6Vb0qKHX1ozYLRZ+beUzD5xFf+3r0J+SVIWKviEa7W524Qvqou+ta1WrsRgzxGw==}
|
||||||
|
engines: {node: '>= 14'}
|
||||||
|
dev: false
|
||||||
|
|
||||||
/@movie-web/providers@2.0.3:
|
/@movie-web/providers@2.0.3:
|
||||||
resolution: {integrity: sha512-6UNk5EebiNjGoFTuyHuu0eZZTreRYv0cdsn52CVYjm6CXG63w4dMbx8ybxcvMUrDF3o8bWlqnlovG142sdOmNw==}
|
resolution: {integrity: sha512-6UNk5EebiNjGoFTuyHuu0eZZTreRYv0cdsn52CVYjm6CXG63w4dMbx8ybxcvMUrDF3o8bWlqnlovG142sdOmNw==}
|
||||||
dependencies:
|
dependencies:
|
||||||
|
|
|
@ -50,5 +50,3 @@ export const locales = {
|
||||||
uk,
|
uk,
|
||||||
};
|
};
|
||||||
export type Locales = keyof typeof locales;
|
export type Locales = keyof typeof locales;
|
||||||
|
|
||||||
export const rtlLocales: Locales[] = ["he", "ar"];
|
|
||||||
|
|
|
@ -1,53 +1,32 @@
|
||||||
import classNames from "classnames";
|
import classNames from "classnames";
|
||||||
|
|
||||||
|
import { getCountryCodeForLocale } from "@/utils/language";
|
||||||
import "flag-icons/css/flag-icons.min.css";
|
import "flag-icons/css/flag-icons.min.css";
|
||||||
|
|
||||||
export interface FlagIconProps {
|
export interface FlagIconProps {
|
||||||
countryCode?: string;
|
country?: string;
|
||||||
|
langCode?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Country code overrides
|
|
||||||
const countryOverrides: Record<string, string> = {
|
|
||||||
en: "gb",
|
|
||||||
cs: "cz",
|
|
||||||
el: "gr",
|
|
||||||
fa: "ir",
|
|
||||||
ko: "kr",
|
|
||||||
he: "il",
|
|
||||||
ze: "cn",
|
|
||||||
ar: "sa",
|
|
||||||
ja: "jp",
|
|
||||||
bs: "ba",
|
|
||||||
vi: "vn",
|
|
||||||
zh: "cn",
|
|
||||||
sl: "si",
|
|
||||||
sv: "se",
|
|
||||||
et: "ee",
|
|
||||||
ne: "np",
|
|
||||||
uk: "ua",
|
|
||||||
hi: "in",
|
|
||||||
};
|
|
||||||
|
|
||||||
export function FlagIcon(props: FlagIconProps) {
|
export function FlagIcon(props: FlagIconProps) {
|
||||||
let countryCode =
|
let countryCode: string | null = props.country ?? null;
|
||||||
(props.countryCode || "")?.split("-").pop()?.toLowerCase() || "";
|
if (props.langCode) countryCode = getCountryCodeForLocale(props.langCode);
|
||||||
if (countryOverrides[countryCode])
|
|
||||||
countryCode = countryOverrides[countryCode];
|
|
||||||
|
|
||||||
if (countryCode === "tok")
|
if (props.langCode === "tok")
|
||||||
return (
|
return (
|
||||||
<div className="w-8 h-6 rounded bg-[#c8e1ed] flex justify-center items-center">
|
<div className="w-8 h-6 rounded bg-[#c8e1ed] flex justify-center items-center">
|
||||||
<img src="/tokiPona.svg" className="w-7 h-5" />
|
<img src="/tokiPona.svg" className="w-7 h-5" />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
if (countryCode === "pirate")
|
if (props.langCode === "pirate")
|
||||||
return (
|
return (
|
||||||
<div className="w-8 h-6 rounded bg-[#2E3439] flex justify-center items-center">
|
<div className="w-8 h-6 rounded bg-[#2E3439] flex justify-center items-center">
|
||||||
<img src="/skull.svg" className="w-4 h-4" />
|
<img src="/skull.svg" className="w-4 h-4" />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
if (countryCode === "minion")
|
if (props.langCode === "minion")
|
||||||
return (
|
return (
|
||||||
<div className="w-8 h-6 rounded bg-[#ffff1a] flex justify-center items-center">
|
<div className="w-8 h-6 rounded bg-[#ffff1a] flex justify-center items-center">
|
||||||
<div className="w-4 h-4 border-2 border-gray-500 rounded-full bg-white flex justify-center items-center">
|
<div className="w-4 h-4 border-2 border-gray-500 rounded-full bg-white flex justify-center items-center">
|
||||||
|
@ -66,7 +45,7 @@ export function FlagIcon(props: FlagIconProps) {
|
||||||
className={classNames(
|
className={classNames(
|
||||||
"!w-8 h-6 rounded overflow-hidden bg-cover bg-center block fi",
|
"!w-8 h-6 rounded overflow-hidden bg-cover bg-center block fi",
|
||||||
backgroundClass,
|
backgroundClass,
|
||||||
props.countryCode ? `fi-${countryCode}` : undefined,
|
countryCode ? `fi-${countryCode}` : undefined,
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
|
|
@ -17,7 +17,7 @@ interface DropdownProps {
|
||||||
|
|
||||||
export function Dropdown(props: DropdownProps) {
|
export function Dropdown(props: DropdownProps) {
|
||||||
return (
|
return (
|
||||||
<div className="relative my-4 max-w-[18rem]">
|
<div className="relative my-4 max-w-[25rem]">
|
||||||
<Listbox value={props.selectedItem} onChange={props.setSelectedItem}>
|
<Listbox value={props.selectedItem} onChange={props.setSelectedItem}>
|
||||||
{() => (
|
{() => (
|
||||||
<>
|
<>
|
||||||
|
|
|
@ -10,12 +10,14 @@ import { useCaptions } from "@/components/player/hooks/useCaptions";
|
||||||
import { Menu } from "@/components/player/internals/ContextMenu";
|
import { Menu } from "@/components/player/internals/ContextMenu";
|
||||||
import { Input } from "@/components/player/internals/ContextMenu/Input";
|
import { Input } from "@/components/player/internals/ContextMenu/Input";
|
||||||
import { SelectableLink } from "@/components/player/internals/ContextMenu/Links";
|
import { SelectableLink } from "@/components/player/internals/ContextMenu/Links";
|
||||||
import { getLanguageFromIETF } from "@/components/player/utils/language";
|
|
||||||
import { useOverlayRouter } from "@/hooks/useOverlayRouter";
|
import { useOverlayRouter } from "@/hooks/useOverlayRouter";
|
||||||
import { CaptionListItem } from "@/stores/player/slices/source";
|
import { CaptionListItem } from "@/stores/player/slices/source";
|
||||||
import { usePlayerStore } from "@/stores/player/store";
|
import { usePlayerStore } from "@/stores/player/store";
|
||||||
import { useSubtitleStore } from "@/stores/subtitles";
|
import { useSubtitleStore } from "@/stores/subtitles";
|
||||||
import { sortLangCodes } from "@/utils/sortLangCodes";
|
import {
|
||||||
|
getPrettyLanguageNameFromLocale,
|
||||||
|
sortLangCodes,
|
||||||
|
} from "@/utils/language";
|
||||||
|
|
||||||
export function CaptionOption(props: {
|
export function CaptionOption(props: {
|
||||||
countryCode?: string;
|
countryCode?: string;
|
||||||
|
@ -37,7 +39,7 @@ export function CaptionOption(props: {
|
||||||
className="flex items-center"
|
className="flex items-center"
|
||||||
>
|
>
|
||||||
<span data-code={props.countryCode} className="mr-3 inline-flex">
|
<span data-code={props.countryCode} className="mr-3 inline-flex">
|
||||||
<FlagIcon countryCode={props.countryCode} />
|
<FlagIcon langCode={props.countryCode} />
|
||||||
</span>
|
</span>
|
||||||
<span>{props.children}</span>
|
<span>{props.children}</span>
|
||||||
</span>
|
</span>
|
||||||
|
@ -89,7 +91,8 @@ function useSubtitleList(subs: CaptionListItem[], searchQuery: string) {
|
||||||
return useMemo(() => {
|
return useMemo(() => {
|
||||||
const input = subs.map((t) => ({
|
const input = subs.map((t) => ({
|
||||||
...t,
|
...t,
|
||||||
languageName: getLanguageFromIETF(t.language) ?? unknownChoice,
|
languageName:
|
||||||
|
getPrettyLanguageNameFromLocale(t.language) ?? unknownChoice,
|
||||||
}));
|
}));
|
||||||
const sorted = sortLangCodes(input.map((t) => t.language));
|
const sorted = sortLangCodes(input.map((t) => t.language));
|
||||||
let results = input.sort((a, b) => {
|
let results = input.sort((a, b) => {
|
||||||
|
|
|
@ -6,11 +6,11 @@ import { Toggle } from "@/components/buttons/Toggle";
|
||||||
import { Icon, Icons } from "@/components/Icon";
|
import { Icon, Icons } from "@/components/Icon";
|
||||||
import { useCaptions } from "@/components/player/hooks/useCaptions";
|
import { useCaptions } from "@/components/player/hooks/useCaptions";
|
||||||
import { Menu } from "@/components/player/internals/ContextMenu";
|
import { Menu } from "@/components/player/internals/ContextMenu";
|
||||||
import { getLanguageFromIETF } from "@/components/player/utils/language";
|
|
||||||
import { useOverlayRouter } from "@/hooks/useOverlayRouter";
|
import { useOverlayRouter } from "@/hooks/useOverlayRouter";
|
||||||
import { usePlayerStore } from "@/stores/player/store";
|
import { usePlayerStore } from "@/stores/player/store";
|
||||||
import { qualityToString } from "@/stores/player/utils/qualities";
|
import { qualityToString } from "@/stores/player/utils/qualities";
|
||||||
import { useSubtitleStore } from "@/stores/subtitles";
|
import { useSubtitleStore } from "@/stores/subtitles";
|
||||||
|
import { getPrettyLanguageNameFromLocale } from "@/utils/language";
|
||||||
|
|
||||||
export function SettingsMenu({ id }: { id: string }) {
|
export function SettingsMenu({ id }: { id: string }) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
@ -31,7 +31,7 @@ export function SettingsMenu({ id }: { id: string }) {
|
||||||
const { toggleLastUsed } = useCaptions();
|
const { toggleLastUsed } = useCaptions();
|
||||||
|
|
||||||
const selectedLanguagePretty = selectedCaptionLanguage
|
const selectedLanguagePretty = selectedCaptionLanguage
|
||||||
? getLanguageFromIETF(selectedCaptionLanguage) ??
|
? getPrettyLanguageNameFromLocale(selectedCaptionLanguage) ??
|
||||||
t("player.menus.subtitles.unknownLanguage")
|
t("player.menus.subtitles.unknownLanguage")
|
||||||
: undefined;
|
: undefined;
|
||||||
|
|
||||||
|
|
|
@ -1,14 +0,0 @@
|
||||||
import { getTag } from "@sozialhelden/ietf-language-tags";
|
|
||||||
|
|
||||||
export function getLanguageFromIETF(ietf: string): string | null {
|
|
||||||
const tag = getTag(ietf, true);
|
|
||||||
|
|
||||||
const lang = tag?.language?.Description?.[0] ?? null;
|
|
||||||
if (!lang) return null;
|
|
||||||
|
|
||||||
const region = tag?.region?.Description?.[0] ?? null;
|
|
||||||
let regionText = "";
|
|
||||||
if (region) regionText = ` (${region})`;
|
|
||||||
|
|
||||||
return `${lang}${regionText}`;
|
|
||||||
}
|
|
|
@ -23,10 +23,9 @@ import { MigrationPart } from "@/pages/parts/migrations/MigrationPart";
|
||||||
import { LargeTextPart } from "@/pages/parts/util/LargeTextPart";
|
import { LargeTextPart } from "@/pages/parts/util/LargeTextPart";
|
||||||
import App from "@/setup/App";
|
import App from "@/setup/App";
|
||||||
import { conf } from "@/setup/config";
|
import { conf } from "@/setup/config";
|
||||||
import i18n from "@/setup/i18n";
|
|
||||||
import { useAuthStore } from "@/stores/auth";
|
import { useAuthStore } from "@/stores/auth";
|
||||||
import { BookmarkSyncer } from "@/stores/bookmarks/BookmarkSyncer";
|
import { BookmarkSyncer } from "@/stores/bookmarks/BookmarkSyncer";
|
||||||
import { useLanguageStore } from "@/stores/language";
|
import { changeAppLanguage, useLanguageStore } from "@/stores/language";
|
||||||
import { ProgressSyncer } from "@/stores/progress/ProgressSyncer";
|
import { ProgressSyncer } from "@/stores/progress/ProgressSyncer";
|
||||||
import { SettingsSyncer } from "@/stores/subtitles/SettingsSyncer";
|
import { SettingsSyncer } from "@/stores/subtitles/SettingsSyncer";
|
||||||
import { ThemeProvider } from "@/stores/theme";
|
import { ThemeProvider } from "@/stores/theme";
|
||||||
|
@ -123,7 +122,7 @@ function AuthWrapper() {
|
||||||
|
|
||||||
function MigrationRunner() {
|
function MigrationRunner() {
|
||||||
const status = useAsync(async () => {
|
const status = useAsync(async () => {
|
||||||
i18n.changeLanguage(useLanguageStore.getState().language);
|
changeAppLanguage(useLanguageStore.getState().language);
|
||||||
await initializeOldStores();
|
await initializeOldStores();
|
||||||
}, []);
|
}, []);
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
|
@ -4,7 +4,7 @@ import { FlagIcon } from "@/components/FlagIcon";
|
||||||
import { Dropdown } from "@/components/form/Dropdown";
|
import { Dropdown } from "@/components/form/Dropdown";
|
||||||
import { Heading1 } from "@/components/utils/Text";
|
import { Heading1 } from "@/components/utils/Text";
|
||||||
import { appLanguageOptions } from "@/setup/i18n";
|
import { appLanguageOptions } from "@/setup/i18n";
|
||||||
import { sortLangCodes } from "@/utils/sortLangCodes";
|
import { getLocaleInfo, sortLangCodes } from "@/utils/language";
|
||||||
|
|
||||||
export function LocalePart(props: {
|
export function LocalePart(props: {
|
||||||
language: string;
|
language: string;
|
||||||
|
@ -17,11 +17,13 @@ export function LocalePart(props: {
|
||||||
.sort((a, b) => sorted.indexOf(a.code) - sorted.indexOf(b.code))
|
.sort((a, b) => sorted.indexOf(a.code) - sorted.indexOf(b.code))
|
||||||
.map((opt) => ({
|
.map((opt) => ({
|
||||||
id: opt.code,
|
id: opt.code,
|
||||||
name: `${opt.name} — ${opt.nativeName}`,
|
name: `${opt.name}${opt.nativeName ? ` — ${opt.nativeName}` : ""}`,
|
||||||
leftIcon: <FlagIcon countryCode={opt.code} />,
|
leftIcon: <FlagIcon langCode={opt.code} />,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
const selected = options.find((item) => item.id === props.language);
|
const selected = options.find(
|
||||||
|
(item) => item.id === getLocaleInfo(props.language)?.code,
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
import i18n from "i18next";
|
import i18n from "i18next";
|
||||||
import ISO6391 from "iso-639-1";
|
|
||||||
import { initReactI18next } from "react-i18next";
|
import { initReactI18next } from "react-i18next";
|
||||||
|
|
||||||
import { locales } from "@/assets/languages";
|
import { locales } from "@/assets/languages";
|
||||||
|
import { getLocaleInfo } from "@/utils/language";
|
||||||
|
|
||||||
// Languages
|
// Languages
|
||||||
const langCodes = Object.keys(locales);
|
const langCodes = Object.keys(locales);
|
||||||
|
@ -10,43 +10,15 @@ const resources = Object.fromEntries(
|
||||||
Object.entries(locales).map((entry) => [entry[0], { translation: entry[1] }]),
|
Object.entries(locales).map((entry) => [entry[0], { translation: entry[1] }]),
|
||||||
);
|
);
|
||||||
i18n.use(initReactI18next).init({
|
i18n.use(initReactI18next).init({
|
||||||
fallbackLng: "en",
|
fallbackLng: "en-US",
|
||||||
resources,
|
resources,
|
||||||
interpolation: {
|
interpolation: {
|
||||||
escapeValue: false, // not needed for react as it escapes by default
|
escapeValue: false, // not needed for react as it escapes by default
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const extraLanguages: Record<
|
|
||||||
string,
|
|
||||||
{
|
|
||||||
code: string;
|
|
||||||
name: string;
|
|
||||||
nativeName: string;
|
|
||||||
}
|
|
||||||
> = {
|
|
||||||
pirate: {
|
|
||||||
code: "pirate",
|
|
||||||
name: "Pirate",
|
|
||||||
nativeName: "Pirate Tongue",
|
|
||||||
},
|
|
||||||
minion: {
|
|
||||||
code: "minion",
|
|
||||||
name: "Minion",
|
|
||||||
nativeName: "Minionese",
|
|
||||||
},
|
|
||||||
tok: {
|
|
||||||
code: "tok",
|
|
||||||
name: "Toki pona",
|
|
||||||
nativeName: "Toki pona",
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
export const appLanguageOptions = langCodes.map((lang) => {
|
export const appLanguageOptions = langCodes.map((lang) => {
|
||||||
const extraLang = extraLanguages[lang];
|
const langObj = getLocaleInfo(lang);
|
||||||
if (extraLang) return extraLang;
|
|
||||||
|
|
||||||
const [langObj] = ISO6391.getLanguages([lang]);
|
|
||||||
if (!langObj)
|
if (!langObj)
|
||||||
throw new Error(`Language with code ${lang} cannot be found in database`);
|
throw new Error(`Language with code ${lang} cannot be found in database`);
|
||||||
return langObj;
|
return langObj;
|
||||||
|
|
|
@ -4,8 +4,8 @@ import { create } from "zustand";
|
||||||
import { persist } from "zustand/middleware";
|
import { persist } from "zustand/middleware";
|
||||||
import { immer } from "zustand/middleware/immer";
|
import { immer } from "zustand/middleware/immer";
|
||||||
|
|
||||||
import { rtlLocales } from "@/assets/languages";
|
|
||||||
import i18n from "@/setup/i18n";
|
import i18n from "@/setup/i18n";
|
||||||
|
import { getLocaleInfo } from "@/utils/language";
|
||||||
|
|
||||||
export interface LanguageStore {
|
export interface LanguageStore {
|
||||||
language: string;
|
language: string;
|
||||||
|
@ -26,14 +26,25 @@ export const useLanguageStore = create(
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
export function changeAppLanguage(language: string) {
|
||||||
|
const lang = getLocaleInfo(language);
|
||||||
|
if (lang) i18n.changeLanguage(lang.code);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isRightToLeft(language: string) {
|
||||||
|
const lang = getLocaleInfo(language);
|
||||||
|
if (!lang) return false;
|
||||||
|
return lang.isRtl;
|
||||||
|
}
|
||||||
|
|
||||||
export function LanguageProvider() {
|
export function LanguageProvider() {
|
||||||
const language = useLanguageStore((s) => s.language);
|
const language = useLanguageStore((s) => s.language);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
i18n.changeLanguage(language);
|
changeAppLanguage(language);
|
||||||
}, [language]);
|
}, [language]);
|
||||||
|
|
||||||
const isRtl = rtlLocales.includes(language as any);
|
const isRtl = isRightToLeft(language);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Helmet>
|
<Helmet>
|
||||||
|
|
188
src/utils/language.ts
Normal file
188
src/utils/language.ts
Normal file
|
@ -0,0 +1,188 @@
|
||||||
|
import countryLanguages from "@ladjs/country-language";
|
||||||
|
import { getTag } from "@sozialhelden/ietf-language-tags";
|
||||||
|
|
||||||
|
const languageOrder = ["en", "hi", "fr", "de", "nl", "pt"];
|
||||||
|
|
||||||
|
// mapping of language code to country code.
|
||||||
|
// multiple mappings can exist, since languages are spoken in multiple countries.
|
||||||
|
// This mapping purely exists to prioritize a country over another in languages.
|
||||||
|
// iso639_1 -> iso3166 Alpha-2
|
||||||
|
const countryPriority: Record<string, string> = {
|
||||||
|
en: "gb",
|
||||||
|
nl: "nl",
|
||||||
|
fr: "fr",
|
||||||
|
de: "de",
|
||||||
|
pt: "pt",
|
||||||
|
ar: "sa",
|
||||||
|
es: "es",
|
||||||
|
zh: "cn",
|
||||||
|
};
|
||||||
|
|
||||||
|
// list of iso639_1 Alpha-2 codes used as default languages
|
||||||
|
const defaultLanguageCodes: string[] = [
|
||||||
|
"en-US",
|
||||||
|
"cs-CZ",
|
||||||
|
"de-DE",
|
||||||
|
"fr-FR",
|
||||||
|
"pt-BR",
|
||||||
|
"it-IT",
|
||||||
|
"nl-NL",
|
||||||
|
"pl-PL",
|
||||||
|
"tr-TR",
|
||||||
|
"vi-VN",
|
||||||
|
"zh-CN",
|
||||||
|
"he-IL",
|
||||||
|
"sv-SE",
|
||||||
|
"lv-LV",
|
||||||
|
"th-TH",
|
||||||
|
"ne-NP",
|
||||||
|
"ar-SA",
|
||||||
|
"es-ES",
|
||||||
|
"et-EE",
|
||||||
|
];
|
||||||
|
|
||||||
|
export interface LocaleInfo {
|
||||||
|
name: string;
|
||||||
|
nativeName?: string;
|
||||||
|
code: string;
|
||||||
|
isRtl?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface LanguageObj {
|
||||||
|
countries: Array<{
|
||||||
|
code_2: string;
|
||||||
|
code_3: string;
|
||||||
|
numCode: string;
|
||||||
|
}>;
|
||||||
|
direction: "RTL" | "LTR";
|
||||||
|
name: string[];
|
||||||
|
nativeName: string[];
|
||||||
|
iso639_1: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const extraLanguages: Record<string, LocaleInfo> = {
|
||||||
|
pirate: {
|
||||||
|
code: "pirate",
|
||||||
|
name: "Pirate",
|
||||||
|
nativeName: "Pirate Tongue",
|
||||||
|
},
|
||||||
|
minion: {
|
||||||
|
code: "minion",
|
||||||
|
name: "Minion",
|
||||||
|
nativeName: "Minionese",
|
||||||
|
},
|
||||||
|
tok: {
|
||||||
|
code: "tok",
|
||||||
|
name: "Toki pona",
|
||||||
|
nativeName: "Toki pona",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
function populateLanguageCode(language: string): string {
|
||||||
|
if (language.includes("-")) return language;
|
||||||
|
if (language.length !== 2) return language;
|
||||||
|
return (
|
||||||
|
defaultLanguageCodes.find((v) => v.startsWith(`${language}-`)) ?? language
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param locale idk what kinda code this takes, anytihhng in ietf format I guess
|
||||||
|
* @returns pretty format for language, null if it no info can be found for language
|
||||||
|
*/
|
||||||
|
export function getPrettyLanguageNameFromLocale(locale: string): string | null {
|
||||||
|
const tag = getTag(populateLanguageCode(locale), true);
|
||||||
|
|
||||||
|
const lang = tag?.language?.Description?.[0] ?? null;
|
||||||
|
if (!lang) return null;
|
||||||
|
|
||||||
|
const region = tag?.region?.Description?.[0] ?? null;
|
||||||
|
let regionText = "";
|
||||||
|
if (region) regionText = ` (${region})`;
|
||||||
|
|
||||||
|
return `${lang}${regionText}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sort locale codes by occurance, rest on alphabetical order
|
||||||
|
* @param langCodes list language codes to sort
|
||||||
|
* @returns sorted version of inputted list
|
||||||
|
*/
|
||||||
|
export function sortLangCodes(langCodes: string[]) {
|
||||||
|
const languagesOrder = [...languageOrder].reverse(); // Reverse is neccesary, not sure why
|
||||||
|
|
||||||
|
const results = langCodes.sort((a, b) => {
|
||||||
|
const langOrderA = languagesOrder.findIndex(
|
||||||
|
(v) => a.startsWith(`${v}-`) || a === v,
|
||||||
|
);
|
||||||
|
const langOrderB = languagesOrder.findIndex(
|
||||||
|
(v) => b.startsWith(`${v}-`) || b === v,
|
||||||
|
);
|
||||||
|
if (langOrderA !== -1 || langOrderB !== -1) return langOrderB - langOrderA;
|
||||||
|
|
||||||
|
return a.localeCompare(b);
|
||||||
|
});
|
||||||
|
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get country code for locale
|
||||||
|
* @param locale input locale
|
||||||
|
* @returns country code or null
|
||||||
|
*/
|
||||||
|
export function getCountryCodeForLocale(locale: string): string | null {
|
||||||
|
let output: LanguageObj | null = null as any as LanguageObj;
|
||||||
|
const tag = getTag(locale, true);
|
||||||
|
if (!tag?.language?.Subtag) return null;
|
||||||
|
// this function isnt async, so its garuanteed to work like this
|
||||||
|
countryLanguages.getLanguage(
|
||||||
|
tag.language.Subtag,
|
||||||
|
(_err: string, lang: LanguageObj) => {
|
||||||
|
if (lang) output = lang;
|
||||||
|
},
|
||||||
|
);
|
||||||
|
if (!output) return null;
|
||||||
|
if (output.countries.length === 0) return null;
|
||||||
|
const priority = countryPriority[output.iso639_1.toLowerCase()];
|
||||||
|
if (priority) {
|
||||||
|
const priotizedCountry = output.countries.find(
|
||||||
|
(v) => v.code_2.toLowerCase() === priority,
|
||||||
|
);
|
||||||
|
if (priotizedCountry) return priotizedCountry.code_2.toLowerCase();
|
||||||
|
}
|
||||||
|
return output.countries[0].code_2.toLowerCase();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get information for a specific local
|
||||||
|
* @param locale local code
|
||||||
|
* @returns locale object
|
||||||
|
*/
|
||||||
|
export function getLocaleInfo(locale: string): LocaleInfo | null {
|
||||||
|
const realLocale = populateLanguageCode(locale);
|
||||||
|
const extraLang = extraLanguages[realLocale];
|
||||||
|
if (extraLang) return extraLang;
|
||||||
|
|
||||||
|
const tag = getTag(realLocale, true);
|
||||||
|
if (!tag?.language?.Subtag) return null;
|
||||||
|
|
||||||
|
let output: LanguageObj | null = null as any as LanguageObj;
|
||||||
|
// this function isnt async, so its garuanteed to work like this
|
||||||
|
countryLanguages.getLanguage(
|
||||||
|
tag.language.Subtag,
|
||||||
|
(_err: string, lang: LanguageObj) => {
|
||||||
|
if (lang) output = lang;
|
||||||
|
},
|
||||||
|
);
|
||||||
|
if (!output) return null;
|
||||||
|
|
||||||
|
return {
|
||||||
|
code: tag.parts.langtag ?? realLocale,
|
||||||
|
isRtl: output.direction === "RTL",
|
||||||
|
name:
|
||||||
|
output.name[0] +
|
||||||
|
(tag.region?.Description ? ` (${tag.region.Description[0]})` : ""),
|
||||||
|
nativeName: output.nativeName[0] ?? undefined,
|
||||||
|
};
|
||||||
|
}
|
|
@ -1,12 +0,0 @@
|
||||||
export function sortLangCodes(langCodes: string[]) {
|
|
||||||
const languagesOrder = ["en", "hi", "fr", "de", "nl", "pt"].reverse(); // Reverse is neccesary, not sure why
|
|
||||||
|
|
||||||
const results = langCodes.sort((a, b) => {
|
|
||||||
if (languagesOrder.indexOf(b) !== -1 || languagesOrder.indexOf(a) !== -1)
|
|
||||||
return languagesOrder.indexOf(b) - languagesOrder.indexOf(a);
|
|
||||||
|
|
||||||
return a.localeCompare(b);
|
|
||||||
});
|
|
||||||
|
|
||||||
return results;
|
|
||||||
}
|
|
|
@ -25,7 +25,7 @@ export default defineConfig(({ mode }) => {
|
||||||
const env = loadEnv(mode, process.cwd());
|
const env = loadEnv(mode, process.cwd());
|
||||||
return {
|
return {
|
||||||
plugins: [
|
plugins: [
|
||||||
million.vite({ auto: true }),
|
million.vite({ auto: true, mute: true }),
|
||||||
handlebars({
|
handlebars({
|
||||||
vars: {
|
vars: {
|
||||||
opensearchEnabled: env.VITE_OPENSEARCH_ENABLED === "true",
|
opensearchEnabled: env.VITE_OPENSEARCH_ENABLED === "true",
|
||||||
|
|
Loading…
Reference in a new issue