vite-plugin-i18next-loader
yarn add -D vite-plugin-i18next-loader
Vite plugin to client bundle i18next locales composited from one to many json/yaml files from one to many libraries. Zero config HMR support included.
This vite-plugin i18next loader generates the resources
structure necessary for i18next. The structure is made available as a virtual module to the client bundle at build time, thus avoiding loading any language resources via extra HTTP requests.
Features
Given a locales directory, by default, the loader will find and parse any json|yaml|yml
file and attribute the
contents to the containing lang folder e.g. en
. There is no need to add lang such as en
or de
inside your
json
or yaml
files.
See the test/data
directory for structure and example data.
Usage
Sample app structure
└── app
└── src
│ └── index.js
└── locales
├── de
│ ├── foo.json
│ └── bar.yaml
└── en
├── foo.json
└── bar.yaml
vite.config.ts
import { defineConfig } from 'vite'
import i18nextLoader from 'vite-plugin-i18next-loader'
export default defineConfig({
plugins: [i18nextLoader({ paths: ['./node_modules/foo/locales', './locales'] })],
})
app.ts
import i18n from 'i18next'
import resources from 'virtual:i18next-loader'
i18n.init({
resources,
})
i18n.t('key')
Options
export interface Options {
logLevel?: LogLevel
include?: string[]
ignore?: string | string[] | IgnoreLike
paths: string[]
namespaceResolution?: 'basename' | 'relativePath'
}
include
to filtering files read
You can filter files in your file structure by specifying any glob supported by glob
. By default, any json|yaml|yml
in the paths
directories will be loaded.
Only json
const options = {
include: ['**/*.json'],
}
All json except one file
const options = {
include: ['**/*.json'],
ignore: ['**/excludeThis.json'],
}
paths
for overriding/white labeling
Applications that reuse libraries e.g. white labeling, can utilize one to many sets of locale directories that
the app will override.
const options = {
include: ['../node_modules/lib1/locales', './locales'],
}
This configures the loader to work on a file structure like the following:
└── app
├── src
│ └── app.js
├── locales
│ └── en
│ ├── foo.json
│ └── bar.yaml
└── node_modules
└── lib1
└── locales
└── en
├── foo.json
└── bar.yaml
Everything from ./locales
will override anything specified in one to many libraries.
namespaceResolution
Namespace resolution will impact the structure of the bundle. If you want the files' basename
or relative path to be injected, look at the following options.
namespaceResolution: 'basename'
const options = {
namespaceResolution: 'basename',
}
The following file structure would result in resources loaded as below:
└── app
├── src
│ └── index.js
└── locales
└── en
├── foo.json
└── bar.yaml
foo.json
{
"header": {
"title": "TITLE"
}
}
bar.yaml
footer:
aboutUs: About us
Results in this object loaded:
{
"en": {
"foo": {
"header": {
"title": "TITLE"
}
},
"bar": {
"footer": {
"aboutUs": "About us"
}
}
}
}
namespaceResolution: 'relativePath'
const options = {
namespaceResolution: 'relativePath',
}
The following file structure would result in resources loaded as below:
└── app
└── locales
├── index.js
└── en
├── green.yaml
└── blue
└── foo.yaml
green.yaml
tree:
species: Oak
blue/foo.yaml
water:
ocean: Quite large
Results in this object loaded:
{
"en": {
"green": {
"tree": {
"species": "Oak"
}
},
"blue": {
"foo": {
"water": {
"ocean": "Quite large"
}
}
}
}
}
NOTE: If you have a file and a folder with the same name, you MIGHT overwrite one with the other. For example:
└── app
└── locales
├── index.js
└── en
├── blue.yaml
└── blue
└── foo.yaml
blue.yaml
foo: Welcome
blue/foo.yaml
eggs: delicious
Results in this object loaded:
{
"en": {
"blue": {
"foo": {
"eggs": "delicious"
}
}
}
}
But it's just overwriting based on the return value of glob-all
, so you shouldn't depend on it.
Output
Note that the virtual module generated has contents that conform to the i18next resource format.
While using the output with import resources from 'virtual:i18next-loader'
will not be tree-shaken, it is possible to use the named outputs with a dynamic import
for tree shaking/chunking optimizations. If you take advantage of this, please see #4 and take a moment to update this doc with more information.
NOTE as shown by the test output below, due to ES syntactical rules, we cannot use hyphenated lang codes. I'm open to ideas, but in the interim, affected lang codes are exported with the hyphen converted to underscore e.g. zh-cn
has a named export of zh_cn
. I noted that vite allows for tree-shaking of JSON files, perhaps that is worth looking at to consider how it might help us and inform our output?
export const en = {
foo: { test: 'app foo.test en' },
main: {
test: 'app test en',
sub: {
slug: 'app sub.slug en',
test: 'lib sub.test en',
subsub: { slugslug: 'app sub.subsub.slugsub en', test: 'lib sub.subsub.test en' },
},
},
}
export const zh_cn = {
foo: { test: 'app foo.test zh-cn' },
main: {
test: 'app test zh-cn',
sub: {
slug: 'app sub.slug zh-cn',
test: 'lib sub.test zh-cn',
subsub: { slugslug: 'app sub.subsub.slugsub zh-cn', test: 'lib sub.subsub.test zh-cn' },
},
},
}
const resources = {
en,
'zh-cn': zh_cn,
}
export default resources
Vite typescript definitions
In order for the vite virtual module to be typechecked, you will need to a declaration. Below is an example of a common type file included in a project for vite:
interface ViteHotContext {
readonly data: any
accept(cb?: (mod: ModuleNamespace | undefined) => void): void
accept(dep: string, cb: (mod: ModuleNamespace | undefined) => void): void
accept(deps: readonly string[], cb: (mods: Array<ModuleNamespace | undefined>) => void): void
dispose(cb: (data: any) => void): void
decline(): void
invalidate(): void
on<T extends string>(event: T, cb: (payload: InferCustomEventPayload<T>) => void): void
send<T extends string>(event: T, data?: InferCustomEventPayload<T>): void
}
declare module 'virtual:*'
Credit
This was forked from @alienfast/i18next-loader, converted to be a vite plugin and improved. Thanks to the original authors and contributors.