2024-01-03 20:29:33 +01:00
|
|
|
import countryLanguages, { LanguageObj } from "@ladjs/country-language";
|
2024-01-03 20:06:08 +01:00
|
|
|
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.
|
2024-01-06 10:46:16 +00:00
|
|
|
// This mapping purely exists to prioritize a country over another in languages where the base language code does
|
|
|
|
// not contain a region (i.e. if the language code is zh-Hant where Hant is a script) or if the region in the language code is incorrect
|
2024-01-03 20:06:08 +01:00
|
|
|
// iso639_1 -> iso3166 Alpha-2
|
|
|
|
const countryPriority: Record<string, string> = {
|
|
|
|
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",
|
2024-01-03 21:20:57 +01:00
|
|
|
"bg-BG",
|
|
|
|
"bn-BD",
|
|
|
|
"el-GR",
|
|
|
|
"fa-IR",
|
|
|
|
"gu-IN",
|
|
|
|
"id-ID",
|
|
|
|
"ja-JP",
|
|
|
|
"ko-KR",
|
|
|
|
"sl-SI",
|
|
|
|
"ta-LK",
|
|
|
|
"ru-RU",
|
2024-01-03 23:14:26 +00:00
|
|
|
"gl-ES",
|
2024-01-03 20:06:08 +01:00
|
|
|
];
|
|
|
|
|
|
|
|
export interface LocaleInfo {
|
|
|
|
name: string;
|
|
|
|
nativeName?: string;
|
|
|
|
code: string;
|
|
|
|
isRtl?: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-01-06 10:42:48 +00:00
|
|
|
* @param locale idk what kinda code this takes, anything in ietf format I guess
|
2024-01-03 20:06:08 +01:00
|
|
|
* @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}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-01-06 10:42:48 +00:00
|
|
|
* Sort locale codes by occurrence, rest on alphabetical order
|
2024-01-03 20:06:08 +01:00
|
|
|
* @param langCodes list language codes to sort
|
|
|
|
* @returns sorted version of inputted list
|
|
|
|
*/
|
|
|
|
export function sortLangCodes(langCodes: string[]) {
|
2024-01-06 10:42:48 +00:00
|
|
|
const languagesOrder = [...languageOrder].reverse(); // Reverse is necessary, not sure why
|
2024-01-03 20:06:08 +01:00
|
|
|
|
|
|
|
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;
|
2024-01-06 12:31:22 +00:00
|
|
|
const tag = getTag(populateLanguageCode(locale), true);
|
2024-01-03 23:14:26 +00:00
|
|
|
|
2024-01-03 20:06:08 +01:00
|
|
|
if (!tag?.language?.Subtag) return null;
|
2024-01-06 10:42:48 +00:00
|
|
|
// this function isn't async, so its guaranteed to work like this
|
2024-01-03 20:29:33 +01:00
|
|
|
countryLanguages.getLanguage(tag.language.Subtag, (_err, lang) => {
|
|
|
|
if (lang) output = lang;
|
|
|
|
});
|
2024-01-06 10:42:48 +00:00
|
|
|
|
2024-01-03 20:06:08 +01:00
|
|
|
if (!output) return null;
|
|
|
|
const priority = countryPriority[output.iso639_1.toLowerCase()];
|
2024-01-03 23:14:26 +00:00
|
|
|
if (output.countries.length === 0) {
|
|
|
|
return priority ?? null;
|
|
|
|
}
|
2024-01-06 10:42:48 +00:00
|
|
|
|
2024-01-06 10:46:16 +00:00
|
|
|
if (priority) {
|
|
|
|
const prioritizedCountry = output.countries.find(
|
|
|
|
(v) => v.code_2.toLowerCase() === priority,
|
|
|
|
);
|
|
|
|
if (prioritizedCountry) return prioritizedCountry.code_2.toLowerCase();
|
|
|
|
}
|
|
|
|
|
2024-01-06 10:42:48 +00:00
|
|
|
// If the language contains a region, check that against the countries and
|
|
|
|
// return the region if it matches
|
|
|
|
const regionSubtag = tag?.region?.Subtag.toLowerCase();
|
|
|
|
if (regionSubtag) {
|
|
|
|
const regionCode = output.countries.find(
|
|
|
|
(c) =>
|
|
|
|
c.code_2.toLowerCase() === regionSubtag ||
|
|
|
|
c.code_3.toLowerCase() === regionSubtag,
|
|
|
|
);
|
|
|
|
if (regionCode) return regionCode.code_2.toLowerCase();
|
|
|
|
}
|
2024-01-03 20:06:08 +01:00
|
|
|
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
|
2024-01-03 20:29:33 +01:00
|
|
|
countryLanguages.getLanguage(tag.language.Subtag, (_err, lang) => {
|
|
|
|
if (lang) output = lang;
|
|
|
|
});
|
2024-01-03 20:06:08 +01:00
|
|
|
if (!output) return null;
|
|
|
|
|
2024-01-03 21:20:57 +01:00
|
|
|
const extras = [];
|
|
|
|
if (tag.region?.Description) extras.push(tag.region.Description[0]);
|
|
|
|
if (tag.script?.Description) extras.push(tag.script.Description[0]);
|
|
|
|
const extraStringified = extras.map((v) => `(${v})`).join(" ");
|
|
|
|
|
2024-01-03 20:06:08 +01:00
|
|
|
return {
|
|
|
|
code: tag.parts.langtag ?? realLocale,
|
|
|
|
isRtl: output.direction === "RTL",
|
2024-01-03 21:20:57 +01:00
|
|
|
name: output.name[0] + (extraStringified ? ` ${extraStringified}` : ""),
|
2024-01-03 20:06:08 +01:00
|
|
|
nativeName: output.nativeName[0] ?? undefined,
|
|
|
|
};
|
|
|
|
}
|