Skip to content

Instantly share code, notes, and snippets.

@selbekk
Created April 12, 2023 20:06
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save selbekk/3c409a7d1a14c70a37e20fdb7c6dbce1 to your computer and use it in GitHub Desktop.
Save selbekk/3c409a7d1a14c70a37e20fdb7c6dbce1 to your computer and use it in GitHub Desktop.
A simple i18n feature for next.js
import { useRouter } from "next/router";
/** The supported languages as enums */
export enum Language {
English = "en",
Norwegian = "no",
}
export type LanguageObject = {
[key in Language]: string | React.ReactNode;
};
type LanguageFunction = (...args: (string | number)[]) => LanguageObject;
type Translations = {
[key: string]:
| LanguageObject
| LanguageFunction
| Translations
| Array<LanguageFunction | LanguageObject | Translations>;
};
/** Converts a Next.js locale string to a language enum */
const toLanguage = (locale?: string): Language => {
switch (locale) {
case "en":
return Language.English;
case "no":
return Language.Norwegian;
default:
console.warn(`Unknown locale "${locale}", defaulting to Norwegian`);
return Language.Norwegian;
}
};
/** Utility function that creates type safe text objects with useTranslation
*
* ```tsx
* const texts = createTexts({
* example: {
* no: "Eksempel",
* en: "Example",
* }
* })
* ```
*/
export function createTexts<T extends Translations>(texts: T) {
return texts;
}
/** Utility for localizing texts
*
* ```tsx
* const texts = createTexts({
* example: { no: "Eksempel", en: "Example" }
* });
* const Example = () => {
* const { t } = useTranslation();
* return t(texts.example);
* }
*/
export const useTranslation = () => {
const { locale } = useRouter();
const language = toLanguage(locale);
return {
t: (translations: LanguageObject) => translations[language] as string,
};
};
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment