set up i18n
This commit is contained in:
parent
490cf0c677
commit
27aac495b9
10 changed files with 183 additions and 3 deletions
BIN
bun.lockb
BIN
bun.lockb
Binary file not shown.
|
@ -1,10 +1,13 @@
|
|||
{
|
||||
"name": "calque",
|
||||
"dependencies": {
|
||||
"@solid-primitives/i18n": "^2.1.1",
|
||||
"@solid-primitives/storage": "^4.2.1",
|
||||
"@solidjs/meta": "^0.29.4",
|
||||
"@solidjs/router": "^0.15.2",
|
||||
"@solidjs/start": "^1.0.10",
|
||||
"dexie": "^4.0.10",
|
||||
"flag-icons": "^7.2.3",
|
||||
"iterator-helpers-polyfill": "^3.0.1",
|
||||
"sitemap": "^8.0.0",
|
||||
"solid-icons": "^1.1.0",
|
||||
|
|
|
@ -2,8 +2,9 @@ import { MetaProvider } from "@solidjs/meta";
|
|||
import { Router } from "@solidjs/router";
|
||||
import { FileRoutes } from "@solidjs/start/router";
|
||||
import { Suspense } from "solid-js";
|
||||
import "./app.css";
|
||||
import { ThemeProvider } from "./components/colorschemepicker";
|
||||
import { I18nProvider } from "./features/i18n";
|
||||
import "./app.css";
|
||||
|
||||
export default function App() {
|
||||
return (
|
||||
|
@ -11,9 +12,11 @@ export default function App() {
|
|||
root={props => (
|
||||
<MetaProvider>
|
||||
<ThemeProvider>
|
||||
<Suspense>{props.children}</Suspense>
|
||||
<I18nProvider>
|
||||
<Suspense>{props.children}</Suspense>
|
||||
</I18nProvider>
|
||||
</ThemeProvider>
|
||||
</MetaProvider>
|
||||
</ MetaProvider>
|
||||
)}
|
||||
>
|
||||
<FileRoutes />
|
||||
|
|
9
src/features/i18n/constants.ts
Normal file
9
src/features/i18n/constants.ts
Normal file
|
@ -0,0 +1,9 @@
|
|||
import { Locale } from "./context";
|
||||
|
||||
import Flag_en_GB from 'flag-icons/flags/4x3/gb.svg';
|
||||
import Flag_nl_NL from 'flag-icons/flags/4x3/nl.svg';
|
||||
|
||||
export const locales: Record<Locale, { label: string, flag: any }> = {
|
||||
'en-GB': { label: 'English', flag: Flag_en_GB },
|
||||
'nl-NL': { label: 'Nederlands', flag: Flag_nl_NL },
|
||||
} as const;
|
60
src/features/i18n/context.tsx
Normal file
60
src/features/i18n/context.tsx
Normal file
|
@ -0,0 +1,60 @@
|
|||
import { Accessor, createContext, createMemo, createSignal, ParentComponent, Setter, useContext } from 'solid-js';
|
||||
import { translator, flatten, Translator, Flatten } from "@solid-primitives/i18n";
|
||||
import en from '~/i18n/en-GB.json';
|
||||
import nl from '~/i18n/nl-NL.json';
|
||||
import { makePersisted } from '@solid-primitives/storage';
|
||||
|
||||
type RawDictionary = typeof en;
|
||||
type Dictionary = Flatten<RawDictionary>;
|
||||
export type Locale = 'en-GB' | 'nl-NL';
|
||||
|
||||
const dictionaries = {
|
||||
'en-GB': en,
|
||||
'nl-NL': nl,
|
||||
} as const;
|
||||
|
||||
interface I18nContextType {
|
||||
readonly t: Translator<Dictionary>;
|
||||
readonly locale: Accessor<Locale>;
|
||||
readonly setLocale: Setter<Locale>;
|
||||
readonly dictionaries: Accessor<Record<Locale, RawDictionary>>;
|
||||
readonly availableLocales: Accessor<Locale[]>;
|
||||
}
|
||||
|
||||
const I18nContext = createContext<I18nContextType>();
|
||||
|
||||
export const I18nProvider: ParentComponent = (props) => {
|
||||
const [locale, setLocale, initLocale] = makePersisted(createSignal<Locale>('en-GB'), { name: 'locale' });
|
||||
const dictionary = createMemo(() => flatten(dictionaries[locale()]));
|
||||
const t = translator(dictionary);
|
||||
|
||||
const ctx: I18nContextType = {
|
||||
t,
|
||||
locale,
|
||||
setLocale,
|
||||
dictionaries: createMemo(() => dictionaries),
|
||||
availableLocales: createMemo(() => Object.keys(dictionaries) as Locale[]),
|
||||
};
|
||||
|
||||
return <I18nContext.Provider value={ctx}>{props.children}</I18nContext.Provider>
|
||||
};
|
||||
|
||||
export const useI18n = () => {
|
||||
const context = useContext(I18nContext);
|
||||
|
||||
if (!context) {
|
||||
throw new Error(`'useI18n' is called outside the scope of an <I18nProvider />`);
|
||||
}
|
||||
|
||||
return { t: context.t, locale: context.locale };
|
||||
};
|
||||
|
||||
export const internal_useI18n = () => {
|
||||
const context = useContext(I18nContext);
|
||||
|
||||
if (!context) {
|
||||
throw new Error(`'useI18n' is called outside the scope of an <I18nProvider />`);
|
||||
}
|
||||
|
||||
return context;
|
||||
};
|
2
src/features/i18n/index.tsx
Normal file
2
src/features/i18n/index.tsx
Normal file
|
@ -0,0 +1,2 @@
|
|||
export { I18nProvider, useI18n } from './context';
|
||||
export { LocalePicker } from './picker';
|
7
src/features/i18n/picker.module.css
Normal file
7
src/features/i18n/picker.module.css
Normal file
|
@ -0,0 +1,7 @@
|
|||
.box {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.flag {
|
||||
inline-size: 1em;
|
||||
}
|
22
src/features/i18n/picker.tsx
Normal file
22
src/features/i18n/picker.tsx
Normal file
|
@ -0,0 +1,22 @@
|
|||
import { Component } from "solid-js";
|
||||
import { internal_useI18n } from "./context";
|
||||
import { locales } from "./constants";
|
||||
import { ComboBox } from "~/components/combobox";
|
||||
import { Dynamic } from "solid-js/web";
|
||||
import css from './picker.module.css';
|
||||
|
||||
interface LocalePickerProps { }
|
||||
|
||||
export const LocalePicker: Component<LocalePickerProps> = (props) => {
|
||||
const { locale, setLocale } = internal_useI18n();
|
||||
|
||||
return <ComboBox
|
||||
id="locale-picker"
|
||||
class={css.box}
|
||||
value={locale()}
|
||||
setValue={setLocale}
|
||||
values={locales}
|
||||
>
|
||||
{(locale, { flag, label }) => <Dynamic component={flag} lang={locale} aria-label={label} class={css.flag} />}
|
||||
</ComboBox>
|
||||
};
|
37
src/i18n/en-GB.json
Normal file
37
src/i18n/en-GB.json
Normal file
|
@ -0,0 +1,37 @@
|
|||
{
|
||||
"page": {
|
||||
"welcome": {
|
||||
"title": "Hi, welcome!",
|
||||
"subtitle": "Lets get started",
|
||||
"edit": "Start editing",
|
||||
"instructions": "Read the instructions",
|
||||
"about": "Abut this app"
|
||||
},
|
||||
"edit": {
|
||||
"menu": {
|
||||
"file": "File",
|
||||
"edit": "Edit",
|
||||
"selection": "Selection"
|
||||
},
|
||||
"command": {
|
||||
"open": "Open folder",
|
||||
"close": "Close folder",
|
||||
"closeTab": "Close tab",
|
||||
"save": "Save",
|
||||
"saveAs": "Save as ...",
|
||||
"selectAll": "Select all",
|
||||
"clearSelection": "Clear selection",
|
||||
"insertKey": "Insert new key",
|
||||
"insertLanguage": "Insert new language",
|
||||
"delete": "Delete selected items"
|
||||
}
|
||||
}
|
||||
},
|
||||
"feature": {
|
||||
"file": {
|
||||
"grid": {
|
||||
"key": "Key"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
37
src/i18n/nl-NL.json
Normal file
37
src/i18n/nl-NL.json
Normal file
|
@ -0,0 +1,37 @@
|
|||
{
|
||||
"page": {
|
||||
"welcome": {
|
||||
"title": "Hoi, welkom!",
|
||||
"subtitle": "Laten we beginnen",
|
||||
"edit": "Begin met bewerken",
|
||||
"instructions": "Lees de instructies",
|
||||
"about": "Over deze app"
|
||||
},
|
||||
"edit": {
|
||||
"menu": {
|
||||
"file": "Bestand",
|
||||
"edit": "Bewerken",
|
||||
"selection": "Selectie"
|
||||
},
|
||||
"command": {
|
||||
"open": "Map openen",
|
||||
"close": "Map sluiten",
|
||||
"closeTab": "Tabblad sluiten",
|
||||
"save": "Opslaan",
|
||||
"saveAs": "Opslaan als ...",
|
||||
"selectAll": "Selecteer alles",
|
||||
"clearSelection": "Selectie leeg maken",
|
||||
"insertKey": "Voeg nieuwe sleutel toe",
|
||||
"insertLanguage": "Voeg nieuwe taal toe",
|
||||
"delete": "Verwijder geselecteerde items"
|
||||
}
|
||||
}
|
||||
},
|
||||
"feature": {
|
||||
"file": {
|
||||
"grid": {
|
||||
"key": "Sleutel"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue