const fs = require("fs"); const translationDir = "lemmy-translations/translations/"; const outDir = "src/shared/translations/"; fs.mkdirSync(outDir, { recursive: true }); fs.readdir(translationDir, (_err, files) => { files.forEach(filename => { const lang = filename.split(".")[0]; try { const json = JSON.parse( fs.readFileSync(translationDir + filename, "utf8"), ); let data = `export const ${lang} = {\n translation: {`; for (const key in json) { if (key in json) { const value = json[key].replace(/"/g, '\\"').replace("\n", "\\n"); data += `\n ${key}: "${value}",`; } } data += "\n },\n};"; const target = outDir + lang + ".ts"; fs.writeFileSync(target, data); } catch (err) { console.error(err); } }); }); // generate types for i18n keys const baseLanguage = "en"; fs.readFile(`${translationDir}${baseLanguage}.json`, "utf8", (_, fileStr) => { const noOptionKeys = []; const optionKeys = []; const optionRegex = /\{\{(.+?)\}\}/g; const optionMap = new Map(); for (const [key, val] of Object.entries(JSON.parse(fileStr))) { const options = []; for ( let match = optionRegex.exec(val); match; match = optionRegex.exec(val) ) { options.push(match[1]); } if (options.length > 0) { optionMap.set(key, options); optionKeys.push(key); } else { noOptionKeys.push(key); } } const indent = " "; const data = `import { i18n } from "i18next"; declare module "i18next" { export type NoOptionI18nKeys = ${noOptionKeys.map(key => `${indent}| "${key}"`).join("\n")}; export type OptionI18nKeys = ${optionKeys.map(key => `${indent}| "${key}"`).join("\n")}; export type I18nKeys = NoOptionI18nKeys | OptionI18nKeys; export type TTypedOptions =${Array.from( optionMap.entries(), ).reduce( (acc, [key, options]) => `${acc} TKey extends \"${key}\" ? ${ options.reduce((acc, cur) => acc + `${cur}: string | number; `, "{ ") + "}" } :\n${indent}`, "", )} (Record | string); export interface TFunctionTyped { // Translation requires options < TKey extends OptionI18nKeys | OptionI18nKeys[], TResult extends TFunctionResult = string, TInterpolationMap extends TTypedOptions = StringMap > ( key: TKey, options: TOptions | string ): TResult; // Translation does not require options < TResult extends TFunctionResult = string, TInterpolationMap extends Record = StringMap > ( key: NoOptionI18nKeys | NoOptionI18nKeys[], options?: TOptions | string ): TResult; } export interface i18nTyped extends i18n { t: TFunctionTyped; } } `; fs.writeFileSync(`${outDir}i18next.d.ts`, data); });