revite/src/pages/settings/panes/Languages.tsx

156 lines
4.8 KiB
TypeScript
Raw Normal View History

2021-12-11 11:56:33 +00:00
import { observer } from "mobx-react-lite";
2021-06-19 22:37:12 +01:00
import styles from "./Panes.module.scss";
2021-07-05 11:23:23 +01:00
import { Text } from "preact-i18n";
2021-12-11 11:56:33 +00:00
import { useMemo } from "preact/hooks";
import { useApplicationState } from "../../../mobx/State";
2021-07-05 11:23:23 +01:00
import {
2021-07-05 11:25:20 +01:00
Language,
LanguageEntry,
Languages as Langs,
2021-07-05 11:23:23 +01:00
} from "../../../context/Locale";
2021-06-22 13:28:03 +01:00
import Emoji from "../../../components/common/Emoji";
2021-06-19 22:37:12 +01:00
import Checkbox from "../../../components/ui/Checkbox";
2021-07-05 11:23:23 +01:00
import Tip from "../../../components/ui/Tip";
2021-09-04 15:50:58 +01:00
import enchantingTableWEBP from "../assets/enchanting_table.webp";
import tamilFlagPNG from "../assets/tamil_nadu_flag.png";
2021-08-04 19:22:13 +01:00
import tokiponaSVG from "../assets/toki_pona.svg";
2021-06-19 22:37:12 +01:00
2021-12-11 11:56:33 +00:00
type Key = [Language, LanguageEntry];
2021-06-19 22:37:12 +01:00
2021-12-11 11:56:33 +00:00
interface Props {
entry: Key;
selected: boolean;
onSelect: () => void;
}
2021-12-11 11:56:33 +00:00
/**
* Component providing individual language entries.
* @param param0 Entry data
*/
function Entry({ entry: [x, lang], selected, onSelect }: Props) {
2021-07-05 11:25:20 +01:00
return (
<Checkbox
key={x}
className={styles.entry}
2021-12-11 11:56:33 +00:00
checked={selected}
onChange={onSelect}>
2021-07-05 11:25:20 +01:00
<div className={styles.flag}>
{lang.i18n === "ta" ? (
<img
src={tamilFlagPNG}
width={42}
style={{ objectFit: "contain" }}
/>
) : lang.emoji === "🙂" ? (
2021-08-04 19:22:13 +01:00
<img src={tokiponaSVG} width={42} />
2021-09-04 15:50:58 +01:00
) : lang.emoji === "🪄" ? (
<img
src={enchantingTableWEBP}
width={42}
style={{ objectFit: "contain" }}
/>
2021-08-04 19:22:13 +01:00
) : (
<Emoji size={42} emoji={lang.emoji} />
)}
2021-07-05 11:25:20 +01:00
</div>
<span className={styles.description}>{lang.display}</span>
</Checkbox>
);
}
2021-12-11 11:56:33 +00:00
/**
* Component providing the language selection menu.
*/
export const Languages = observer(() => {
const locale = useApplicationState().locale;
const language = locale.getLanguage();
// Generate languages array.
const languages = useMemo(() => {
const languages = Object.keys(Langs).map((x) => [
x,
Langs[x as keyof typeof Langs],
]) as Key[];
// Get the user's system language. Check for exact
// matches first, otherwise check for partial matches
const preferredLanguage =
navigator.languages.filter((lang) =>
languages.find((l) => l[0].replace(/_/g, "-") == lang),
)?.[0] ||
navigator.languages
?.map((x) => x.split("-")[0])
?.filter((lang) => languages.find((l) => l[0] == lang))?.[0]
?.split("-")[0];
if (preferredLanguage) {
// This moves the user's system language to the top of the language list
const prefLangKey = languages.find(
(lang) => lang[0].replace(/_/g, "-") == preferredLanguage,
2021-08-06 22:12:10 +02:00
);
2021-12-11 11:56:33 +00:00
if (prefLangKey) {
languages.splice(
0,
0,
languages.splice(languages.indexOf(prefLangKey), 1)[0],
);
}
2021-08-06 22:12:10 +02:00
}
2021-12-11 11:56:33 +00:00
return languages;
}, []);
// Creates entries with given key.
const EntryFactory = ([x, lang]: Key) => (
<Entry
key={x}
entry={[x, lang]}
selected={language === x}
onSelect={() => locale.setLanguage(x)}
/>
);
2021-07-05 11:25:20 +01:00
return (
<div className={styles.languages}>
<h3>
<Text id="app.settings.pages.language.select" />
</h3>
<div className={styles.list}>
2021-12-11 11:56:33 +00:00
{languages.filter(([, lang]) => !lang.cat).map(EntryFactory)}
2021-07-05 11:25:20 +01:00
</div>
2021-08-04 13:29:12 +02:00
<h3>
<Text id="app.settings.pages.language.const" />
</h3>
2021-08-04 12:31:51 +01:00
<div className={styles.list}>
{languages
.filter(([, lang]) => lang.cat === "const")
2021-12-11 11:56:33 +00:00
.map(EntryFactory)}
2021-08-04 12:31:51 +01:00
</div>
2021-07-05 11:25:20 +01:00
<h3>
<Text id="app.settings.pages.language.other" />
</h3>
<div className={styles.list}>
{languages
2021-08-04 12:31:51 +01:00
.filter(([, lang]) => lang.cat === "alt")
2021-12-11 11:56:33 +00:00
.map(EntryFactory)}
2021-07-05 11:25:20 +01:00
</div>
<Tip>
<span>
<Text id="app.settings.tips.languages.a" />
</span>{" "}
<a
href="https://weblate.insrt.uk/engage/revolt/?utm_source=widget"
target="_blank"
rel="noreferrer">
2021-07-05 11:25:20 +01:00
<Text id="app.settings.tips.languages.b" />
</a>
</Tip>
</div>
);
2021-07-05 11:23:23 +01:00
});