i18n
Lightweight simple translation module with dynamic json storage. Supports plain vanilla node.js apps and should work with any framework (like express, restify and probably more) that exposes an app.use()
method passing in res
and req
objects.
Uses common __('...') syntax in app and templates.
Stores language files in json files compatible to webtranslateit json format.
Adds new strings on-the-fly when first used in your app.
No extra parsing needed.
Can create new i18n instances with isolated environment.
Install
npm install i18n-factory
Test
npm test
Load
// load modulesvar express = i18n = ;
now you are ready to use a global i18n.__('Hello')
. Global assumes you share a common state of localization in any time and any part of your app. This is usually fine in cli-style scripts. When serving responses to http requests you'll need to make sure that scope is NOT shared globally but attached to your request object.
Configure
Minimal example, just setup two locales and a project specific directory
i18n;
list of configuration options
i18n;
Example usage in global scope
In your app, when not registered to a specific object:
var greeting = i18n;
Example usage in express.js
In an express app, you might use i18n.init to gather language settings of your visitors and also bind your helpers to response object honoring request objects locale, ie:
// Configurationapp;
in your apps methods:
app;
in your templates (depending on your template engine)
<%= __('Hello') %> ${__('Hello')}
For serving the same static files with different language url, you could:
app;app;app;
Examples for common setups
See tested examples inside /examples
or browse these gists:
- plain node.js + http
- plain node.js + restify
- express 3 + cookie
- express 3 + hbs 2 (+ cookie)
- express 3 + mustache (+ cookie)
- express 3 + jade 0.3 (+ cookie)
- express 4 + cookie
API
create()
Returns a new instance with isolated scope.
getLang()
Returns language code from locale.
parseLocale()
Returns a locale found in the directory where json files are stored.
__()
Translates a single phrase and adds it to locales if unknown. Returns translated parsed and substituted string.
// template and global (this.locale == 'de'); // Hallo; // Hallo Marcus; // Hallo Marcus // scoped via req object (req.locale == 'de')req; // Halloreq; // Hallo Marcusreq; // Hallo Marcus // passing specific locale; // Salut; // Salut Marcus; // Salut Marcus
__n()
Plurals translation of a single phrase. Singular and plural forms will get added to locales if unknown. Returns translated parsed and substituted string based on count
parameter.
// template and global (this.locale == 'de'); // 1 Katze; // 3 Katzen // scoped via req object (req.locale == 'de')req; // 1 Katzereq; // 3 Katzen // passing specific locale; // 1 chat; // 3 chat ; // 1 chat; // 3 chat
setLocale()
Setting the current locale (ie.: en
) globally or in current scope.
;;req;
To change the initial locale (when you set it on i18n.init()
) for all the user session (eg.: you have a language
selector on your web page to let the user select the preferred language), you have some options.
You could set it via res.setLocale('de')
on each loop before load the each page.
Or you could manage it via any session middleware or by setting a cookie in the client and let i18n read it's value.
In the last case you will need to enable cookies (eg. for express will be app.use(express.cookieParser())
) and then
you can use the i18n.configure.cookie
to let i18n which language must use. Simply use the same cookie name when setting it in the user preferred language, like here:
rescookie'yourcookiename' 'de' maxAge: 900000 httpOnly: true ;
After this and until the cookie expires, i18n will get the value of the cookie and will set that language instead of default for every page.
getLocale()
Getting the current locale (ie.: en
) from current scope or globally.
; // --> de; // --> dereq; // --> de
getCatalog()
Returns a whole catalog optionally based on current scope and locale.
; // returns all locales; // returns just 'de' ; // returns all locales; // returns just 'de' req; // returns all localesreq; // returns just 'de'
Optionally manual attaching helpers for different template engines
In general i18n has to be attached to the response object to let it's public api get accessible in your templates and methods. As of 0.4.0 i18n tries to do so internally via i18n.init
, as if you were doing it in app.configure
on your own:
app;
Different engines need different implementations, so yours might miss or not work with the current default helpers. This one showing an example for mustache in express:
// register helper as a locals function wrapped as mustache expectsapp;
You could still setup your own implementation. Please refer to Examples below, post an issue or contribute your setup.
Output parsing of expressions
As inspired by gettext there is currently support for sprintf-style expressions. You can also use mustache syntax for named parameters.
sprintf support
var greeting = ;
this puts Hello Marcus, how are you today?. You might add endless arguments and even nest it.
var greeting = ;
which puts Hello Marcus, how are you today? How was your weekend.
You might need to have repeated references to the same argument, which can be done with sprintf.
var example = ;
which puts
repeat, repeat, repeat
In some cases the argument order will need to be switched for different locales. The arguments can be strings, floats, numbers, etc.
var example = ;
which puts
2 then First then 333.33
mustache support
You may also use mustache syntax for your message strings. To pass named parameters to your message, just provide an object as the second parameter. You can still pass unnamed parameters by adding additional arguments.
var greeting = ;
this puts Hello Marcus, how are you today?. You might also combine it with sprintf arguments and also nest it.
var greeting = ;
which puts Hello Marcus, how was your weekend.
variable support
you might even use dynamic variables as they get interpreted on the fly. Better make sure no user input finds it's way to that point as they all get added to the en.js
file if not yet existing.
var greetings = 'Hi' 'Hello' 'Howdy';for var i=0; i < greetingslength; i++ console;;
which puts
Hi
Hello
Howdy
basic plural support
two different plural forms are supported as response to count
:
var singular = ;var plural = ;
this puts 1 cat or 3 cats and again these could get nested:
var singular = ;var plural = ;
putting There is one monkey in the tree or There are 3 monkeys in the tree
Storage
Will get modular support for different storage engines, currently just json files are stored in filesystem.
json file
the above will automatically generate a en.json
by default inside ./locales/
which looks like
that file can be edited or just uploaded to webtranslateit for any kind of collaborative translation workflow:
Logging & Debugging
Logging any kind of output is moved to debug module. To let i18n output anything run your app with DEBUG
env set like so:
$ DEBUG=i18n:* node app.js
i18n exposes three log-levels:
- i18n:debug
- i18n:warn
- i18n:error
if you only want to get errors and warnings reported start your node server like so:
$ DEBUG=i18n:warn,i18n:error node app.js
Combine those settings with you existing application if any of you other modules or libs also uses debug
Using custom logger
You can configure i18n to use a custom logger. For example attach some simple console
-logging:
i18n;
Object notation
In addition to the traditional, linear translation lists, i18n also supports hierarchical translation catalogs.
To enable this feature, be sure to set objectNotation
to true
in your configure()
call. Note: If you can't or don't want to use .
as a delimiter, set objectNotation
to any other delimiter you like.
Instead of calling __("Hello")
you might call __("greeting.formal")
to retrieve a formal greeting from a translation document like this one:
"greeting": "formal": "Hello" "informal": "Hi" "placeholder": "formal": "Hello %s" "informal": "Hi %s"
In the document, the translation terms, which include placeholders, are nested inside the "greeting" translation. They can be accessed and used in the same way, like so __('greeting.placeholder.informal', 'Marcus')
.
Pluralization
Object notation also supports pluralization. When making use of it, the "one" and "other" entries are used implicitly for an object in the translation document. For example, consider the following document:
"cat": "one": "Katze" "other": "Katzen"
When accessing these, you would use __n("cat", "cat", 3)
to tell i18n to use both the singular and plural form of the "cat" entry. Naturally, you could also access these members explicitly with __("cat.one")
and __("cat.other")
.
Defaults
When starting a project from scratch, your translation documents will probably be empty. i18n takes care of filling your translation documents for you. Whenever you use an unknown object, it is added to the translation documents.
By default, when using object notation, the provided string literal will be inserted and returned as the default string. As an example, this is what the "greeting" object shown earlier would look like by default:
"greeting": "formal": "greeting.formal" "informal": "greeting.informal"
In case you would prefer to have a default string automatically inserted and returned, you can provide that default string by appending it to your object literal, delimited by a :
. For example:
Changelog
- 0.5.0: feature release; added {{mustache}} parsing by #85, added "object.notation" by #110, fixed buggy req.__() implementation by #111 and closed 13 issues
- 0.4.1: stable release; merged/closed: #57, #60, #67 typo fixes; added more examples and new features: #53, #65, #66 - and some more api reference
- 0.4.0: stable release; closed: #22, #24, #4, #10, #54; added examples, clarified concurrency usage in different template engines, added
i18n.getCatalog
- 0.3.9: express.js usage, named api, jscoverage + more test, refactored configure, closed: #51, #20, #16, #49
- 0.3.8: fixed: #44, #49; merged: #47, #45, #50; added: #33; updated: README
- 0.3.7: tests by mocha.js, added
this.locale
to__
and__n
- 0.3.6: travisCI, writeFileSync, devDependencies, jslint, MIT, fixed: #29, #9, merged: #25, #30, #43
- 0.3.5: fixed some issues, prepared refactoring, prepared publishing to npm finally
- 0.3.4: merged pull request #13 from Fuitad/master and updated README
- 0.3.3: merged pull request from codders/master and modified for backward compatibility. Usage and tests pending
- 0.3.2: merged pull request #7 from carlptr/master and added tests, modified fswrite to do sync writes
- 0.3.0: added configure and init with express support (calling guessLanguage() via 'accept-language')
- 0.2.0: added plurals
- 0.1.0: added tests
- 0.0.1: start
Licensed under MIT
Copyright (c) 2011-2014 Marcus Spiegel marcus.spiegel@gmail.com
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.