|
|
|
@ -1,19 +1,12 @@
|
|
|
|
|
# Internationalization (i18n)
|
|
|
|
|
|
|
|
|
|
{@a top}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Angular's _internationalization_ (_i18n_) tools help make your app available in multiple languages.
|
|
|
|
|
|
|
|
|
|
Try this <live-example name="i18n" title="i18n Example in Spanish">live example</live-example>
|
|
|
|
|
of a JIT-compiled app, translated into Spanish.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a angular-i18n}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
## Angular and _i18n_ template translation
|
|
|
|
|
|
|
|
|
|
Application internationalization is a challenging, many-faceted effort that
|
|
|
|
@ -23,20 +16,13 @@ Angular's _i18n_ internationalization facilities can help.
|
|
|
|
|
This page describes the _i18n_ tools available to assist translation of component template text
|
|
|
|
|
into multiple languages.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<div class="l-sub-section">
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Practitioners of _internationalization_ refer to a translatable text as a "_message_".
|
|
|
|
|
This page uses the words "_text_" and "_message_" interchangably and in the combination, "_text message_".
|
|
|
|
|
|
|
|
|
|
This page uses the words "_text_" and "_message_" interchangeably and in the combination, "_text message_".
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
The _i18n_ template translation process has four phases:
|
|
|
|
|
|
|
|
|
|
1. Mark static text messages in your component templates for translation.
|
|
|
|
@ -52,11 +38,8 @@ in the target language.
|
|
|
|
|
|
|
|
|
|
You need to build and deploy a separate version of the application for each supported language.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a i18n-attribute}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
## Mark text with the _i18n_ attribute
|
|
|
|
|
|
|
|
|
|
The Angular `i18n` attribute is a marker for translatable content.
|
|
|
|
@ -65,73 +48,112 @@ Place it on every element tag whose fixed text should be translated.
|
|
|
|
|
|
|
|
|
|
<div class="alert is-helpful">
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
`i18n` is not an Angular _directive_.
|
|
|
|
|
It's a custom _attribute_, recognized by Angular tools and compilers.
|
|
|
|
|
After translation, the compiler removes it.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
In the accompanying sample, an `<h1>` tag displays a simple English language greeting
|
|
|
|
|
that you translate into Spanish:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.1.html" region="greeting" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Add the `i18n` attribute to the tag to mark it for translation.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.1.html" region="i18n-attribute" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a help-translator}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Help the translator with a _description_ and _intent_
|
|
|
|
|
### Help the translator with a _description_ and _meaning_
|
|
|
|
|
|
|
|
|
|
In order to translate it accurately, the translator may
|
|
|
|
|
need a description of the message.
|
|
|
|
|
Assign a description to the i18n attribute:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.1.html" region="i18n-attribute-desc" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
In order to deliver a correct translation, the translator may need to
|
|
|
|
|
know your _intent_—the true _meaning_ of the text
|
|
|
|
|
within _this particular_ application context.
|
|
|
|
|
In front of the description, add some contextual meaning to the assigned string,
|
|
|
|
|
separating it from the description with the `|` character (`<meaning>|<description>`):
|
|
|
|
|
know the _meaning_ or _intent_ of the text within _this particular_ application context.
|
|
|
|
|
|
|
|
|
|
You add context by beginning the string with the _meaning_ and
|
|
|
|
|
separating it from the _description_ with the `|` character (`<meaning>|<description>`):
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.html" region="i18n-attribute-meaning" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.1.html" region="i18n-attribute-meaning" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
While all appearances of a message with the _same_ meaning have the _same_ translation,
|
|
|
|
|
a message with *a variety of possible meanings* could have different translations.
|
|
|
|
|
The Angular extraction tool preserves both the _meaning_ and the _description_ in the translation source file
|
|
|
|
|
to facilitiate contextually-specific translations.
|
|
|
|
|
to facilitate contextually-specific translations.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a custom-id}
|
|
|
|
|
|
|
|
|
|
### Set a custom _id_ to improve search and maintenance
|
|
|
|
|
|
|
|
|
|
The angular _i18n_ extractor tool generates a file with a _translation unit_ entry for each `i18n` attribute in a template. By default, it assigns each translation unit a unique _id_ such as this one:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="generated-id" linenums="false">
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
This _id_ is obscure and difficult for humans to read or remember.
|
|
|
|
|
|
|
|
|
|
Worse, when you change the translatable text, perhaps to fix a typo,
|
|
|
|
|
the extractor tool generates a new _id_ for that translation.
|
|
|
|
|
You will lose the translation unless you update it with the new _id_.
|
|
|
|
|
That [complicates maintenance](#maintenance).
|
|
|
|
|
|
|
|
|
|
Consider specifying your own, meaningful _id_ in the `i18n` attribute, **prefixed with `@@`**.
|
|
|
|
|
|
|
|
|
|
<code-example path='i18n/src/app/app.component.1.html' region='i18n-attribute-solo-id' title='app/app.component.html' linenums="false">
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
Now the extractor tool and compiler will generate a translation unit with _your custom id_ and never change it.
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="custom-id" linenums="false">
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
Here is the `i18n` attribute with a _definition_, followed by the custom `id`:
|
|
|
|
|
|
|
|
|
|
<code-example path='i18n/src/app/app.component.1.html' region='i18n-attribute-id' title='app/app.component.html' linenums="false">
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
Here is a _meaning_ and a _description_ and the _id_ at the end:
|
|
|
|
|
|
|
|
|
|
<code-example path='i18n/src/app/app.component.1.html' region='i18n-attribute-meaning-and-id' title='app/app.component.html' linenums="false">
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
<div class="l-sub-section">
|
|
|
|
|
|
|
|
|
|
Be sure to define _unique_ custom ids. If you use the same id for 2 _different_ blocks of text, only the first one will be extracted,
|
|
|
|
|
and its translation used in both blocks of text.
|
|
|
|
|
|
|
|
|
|
For example:
|
|
|
|
|
|
|
|
|
|
```html
|
|
|
|
|
<p i18n="@@myId">Hello</p>
|
|
|
|
|
<p i18n="@@myId">Good bye</p>
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
with the translation:
|
|
|
|
|
|
|
|
|
|
```xml
|
|
|
|
|
<trans-unit id="myId" datatype="html">
|
|
|
|
|
<source>Hello</source>
|
|
|
|
|
<target state="new">Hola</target>
|
|
|
|
|
</trans-unit>
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Both `<p>` elements will contain the text `Hola`.
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
{@a no-element}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Translate text without creating an element
|
|
|
|
|
|
|
|
|
|
Suppose there is a stretch of text that you'd like to translate.
|
|
|
|
@ -140,58 +162,37 @@ you don't want to create a new DOM element merely to facilitate translation.
|
|
|
|
|
|
|
|
|
|
Here are two techniques to try.
|
|
|
|
|
|
|
|
|
|
(1) Wrap the text in an `<ng-container>` element. The `<ng-container>` is never renderered:
|
|
|
|
|
|
|
|
|
|
(1) Wrap the text in an `<ng-container>` element. The `<ng-container>` is never rendered:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.html" region="i18n-ng-container" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
(2) Wrap the text in a pair of HTML comments:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.html" region="i18n-with-comment" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a translate-attributes}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
## Add _i18n-..._ translation attributes
|
|
|
|
|
## Add _i18n_ translation attributes
|
|
|
|
|
You've added an image to your template. You care about accessibility too so you add a `title` attribute:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.1.html" region="i18n-title" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
The `title` attribute needs to be translated.
|
|
|
|
|
Angular i18n support has more translation attributes in the form,`i18n-x`, where `x` is the
|
|
|
|
|
name of the attribute to translate.
|
|
|
|
|
|
|
|
|
|
To translate the `title` on the `img` tag from the previous example, write:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.html" region="i18n-title-translate" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
You can also assign a meaning and a description with the `i18n-x="<meaning>|<description>"` syntax.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a cardinality}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
## Handle singular and plural
|
|
|
|
|
|
|
|
|
|
Different languages have different pluralization rules.
|
|
|
|
@ -202,13 +203,9 @@ Other languages might express the _cardinality_ differently.
|
|
|
|
|
|
|
|
|
|
Here's how you could mark up the component template to display the phrase appropriate to the number of wolves:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.html" region="i18n-plural" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
* The first parameter is the key. It is bound to the component property (`wolves`)
|
|
|
|
|
that determines the number of wolves.
|
|
|
|
|
* The second parameter identifies this as a `plural` translation type.
|
|
|
|
@ -217,43 +214,41 @@ categories and their matching values.
|
|
|
|
|
|
|
|
|
|
Pluralization categories include:
|
|
|
|
|
|
|
|
|
|
* =0
|
|
|
|
|
* =1
|
|
|
|
|
* =5
|
|
|
|
|
* =0 (or any other number)
|
|
|
|
|
* zero
|
|
|
|
|
* one
|
|
|
|
|
* two
|
|
|
|
|
* few
|
|
|
|
|
* many
|
|
|
|
|
* other
|
|
|
|
|
|
|
|
|
|
Put the default _English_ translation in braces (`{}`) next to the pluralization category.
|
|
|
|
|
|
|
|
|
|
* When you're talking about one wolf, you could write `=1 {one wolf}`.
|
|
|
|
|
|
|
|
|
|
* For zero wolves, you could write `=0 {no wolves}`.
|
|
|
|
|
|
|
|
|
|
* For two wolves, you could write `=2 {two wolves}`.
|
|
|
|
|
|
|
|
|
|
You could keep this up for three, four, and every other number of wolves.
|
|
|
|
|
Or you could specify the **`other`** category as a catch-all for any unmatched cardinality
|
|
|
|
|
and write something like: `other {a wolf pack}`.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<div class="l-sub-section">
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
This syntax conforms to the
|
|
|
|
|
<a href="http://userguide.icu-project.org/formatparse/messages" title="ICU Message Format">ICU Message Format</a>
|
|
|
|
|
that derives from the
|
|
|
|
|
<a href="http://cldr.unicode.org/" title="CLDR">Common Locale Data Repository (CLDR)</a>,
|
|
|
|
|
which specifies the
|
|
|
|
|
<a href="http://unicode.org/reports/tr35/tr35-numbers.html#Language_Plural_Rules" title="Pluralization Rules">pluralization rules</a>.
|
|
|
|
|
|
|
|
|
|
<a href="http://cldr.unicode.org/index/cldr-spec/plural-rules" title="Pluralization Rules">pluralization rules</a>.
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a select}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
## Select among alternative texts
|
|
|
|
|
|
|
|
|
|
The application displays different text depending upon whether the hero is male or female.
|
|
|
|
|
These text alternatives require translation too.
|
|
|
|
|
|
|
|
|
@ -266,17 +261,18 @@ The following format message in the component template binds to the component's
|
|
|
|
|
property, which outputs either an "m" or an "f".
|
|
|
|
|
The message maps those values to the appropriate translation:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.html" region="i18n-select" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
## Nesting pluralization and selection expressions
|
|
|
|
|
|
|
|
|
|
You can also nest different ICU expressions together. For example:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.html" region="i18n-nested" title="src/app/app.component.html">
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
{@a ng-xi18n}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
## Create a translation source file with the _ng-xi18n_ tool
|
|
|
|
|
|
|
|
|
|
Use the **_ng-xi18n_ extraction tool** to extract the `i18n`-marked texts
|
|
|
|
@ -285,63 +281,46 @@ into a translation source file in an industry standard format.
|
|
|
|
|
This is an Angular CLI tool in the `@angular/compiler-cli` npm package.
|
|
|
|
|
If you haven't already installed the CLI and its `platform-server` peer dependency, do so now:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example language="sh" class="code-shell">
|
|
|
|
|
npm install @angular/compiler-cli @angular/platform-server --save
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Open a terminal window at the root of the application project and enter the `ng-xi18n` command:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example language="sh" class="code-shell">
|
|
|
|
|
./node_modules/.bin/ng-xi18n
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<div class="l-sub-section">
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Windows users may have to quote the command like this: `"./node_modules/.bin/ng-xi18n"`
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
By default, the tool generates a translation file named **`messages.xlf`** in the
|
|
|
|
|
<a href="https://en.wikipedia.org/wiki/XLIFF">XML Localisation Interchange File Format (XLIFF, version 1.2)</a>.
|
|
|
|
|
|
|
|
|
|
<a href="https://en.wikipedia.org/wiki/XLIFF">XML Localization Interchange File Format (XLIFF, version 1.2)</a>.
|
|
|
|
|
|
|
|
|
|
{@a other-formats}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Other translation formats
|
|
|
|
|
|
|
|
|
|
You can generate a file named **`messages.xmb`** in the
|
|
|
|
|
<a href="http://cldr.unicode.org/development/development-process/design-proposals/xmb" >XML Message Bundle (XMB)</a> format
|
|
|
|
|
by adding the `--i18nFormat=xmb` flag.
|
|
|
|
|
Angular i18n tooling supports XLIFF 1.2 and XLIFF 2 as well as the
|
|
|
|
|
<a href="http://cldr.unicode.org/development/development-process/design-proposals/xmb" >XML Message Bundle (XMB)</a>.
|
|
|
|
|
|
|
|
|
|
You can specify your choice of format _explicitly_ with the `--i18nFormat` flag as illustrated in these example commands
|
|
|
|
|
|
|
|
|
|
<code-example language="sh" class="code-shell">
|
|
|
|
|
./node_modules/.bin/ng-xi18n --i18nFormat=xmb
|
|
|
|
|
|
|
|
|
|
./node_modules/.bin/ng-xi18n --i18nFormat=xlf --outFile=messages.xlf
|
|
|
|
|
./node_modules/.bin/ng-xi18n --i18nFormat=xlf2 --outFile=messages.xliff2.xlf
|
|
|
|
|
./node_modules/.bin/ng-xi18n --i18nFormat=xmb --outFile=messages.xmb
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
This sample sticks with the _XLIFF_ format.
|
|
|
|
|
|
|
|
|
|
The sample in _this_ guide sticks with the default _XLIFF 1.2_ format.
|
|
|
|
|
|
|
|
|
|
{@a ng-xi18n-options}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Other options
|
|
|
|
|
|
|
|
|
|
You may have to specify additional options.
|
|
|
|
|
For example, if the `tsconfig.json` TypeScript configuration
|
|
|
|
|
file is located somewhere other than in the root folder,
|
|
|
|
@ -350,15 +329,10 @@ you must identify the path to it with the `-p` option:
|
|
|
|
|
<code-example language="sh" class="code-shell">
|
|
|
|
|
./node_modules/.bin/ng-xi18n -p path/to/tsconfig.json
|
|
|
|
|
./node_modules/.bin/ng-xi18n --i18nFormat=xmb -p path/to/tsconfig.json
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a npm-i18n-script}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Add an _npm_ script for convenience
|
|
|
|
|
|
|
|
|
|
Consider adding a convenience shortcut to the `scripts` section of the `package.json`
|
|
|
|
@ -371,8 +345,6 @@ to make the command easier to remember and run:
|
|
|
|
|
}
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Now you can issue command variations such as these:
|
|
|
|
|
|
|
|
|
|
<code-example language="sh" class="code-shell">
|
|
|
|
@ -381,28 +353,21 @@ Now you can issue command variations such as these:
|
|
|
|
|
npm run i18n -- --i18nFormat=xmb -p path/to/tsconfig.json
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Note the `--` flag before the options.
|
|
|
|
|
It tells _npm_ to pass every flag thereafter to `ng-xi18n`.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a translate}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
## Translate text messages
|
|
|
|
|
|
|
|
|
|
The `ng-xi18n` command generates a translation source file
|
|
|
|
|
in the project root folder named `messages.xlf`.
|
|
|
|
|
The next step is to translate the English language template
|
|
|
|
|
text into the specific language translation
|
|
|
|
|
files. The cookbook sample creates a Spanish translation file.
|
|
|
|
|
|
|
|
|
|
files. The guide sample creates a Spanish translation file.
|
|
|
|
|
|
|
|
|
|
{@a localization-folder}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Create a localization folder
|
|
|
|
|
|
|
|
|
|
You will probably translate into more than one other language so it's a good idea
|
|
|
|
@ -413,16 +378,12 @@ such as internationalization files, there.
|
|
|
|
|
|
|
|
|
|
<div class="l-sub-section">
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Localization and internationalization are
|
|
|
|
|
<a href="https://en.wikipedia.org/wiki/Internationalization_and_localization">different but closely related terms</a>.
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
This cookbook follows that suggestion. It has a `locale` folder under the `src/`.
|
|
|
|
|
This guide follows that suggestion. It has a `locale` folder under `src/`.
|
|
|
|
|
Assets within the folder carry a filename extension that matches a language-culture code from a
|
|
|
|
|
<a href="https://msdn.microsoft.com/en-us/library/ee825488(v=cs.20).aspx">well-known codeset</a>.
|
|
|
|
|
|
|
|
|
@ -432,7 +393,6 @@ Do the same for each target language.
|
|
|
|
|
|
|
|
|
|
{@a translate-text-nodes}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Translate text nodes
|
|
|
|
|
In the real world, you send the `messages.es.xlf` file to a Spanish translator who fills in the translations
|
|
|
|
|
using one of the
|
|
|
|
@ -441,50 +401,41 @@ using one of the
|
|
|
|
|
This sample file is easy to translate without a special editor or knowledge of Spanish.
|
|
|
|
|
Open `messages.es.xlf` and find the first `<trans-unit>` section:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="translated-hello" title="src/locale/messages.es.xlf (<trans-unit>)" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
This XML element represents the translation of the `<h1>` greeting tag you marked with the `i18n` attribute.
|
|
|
|
|
|
|
|
|
|
<div class="l-sub-section">
|
|
|
|
|
|
|
|
|
|
Note that the translation unit `id=introductionHeader` is derived from the _custom_ `id`](#custom-id "Set a custom id") that you set earlier, but **without the `@@` prefix** required in the source HTML.
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
Using the _source_, _description_, and _meaning_ elements to guide your translation,
|
|
|
|
|
replace the `<target/>` tag with the Spanish greeting:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="translated-hello" title="src/locale/messages.es.xlf (<trans-unit>, after translation)" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<div class="alert is-important">
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Note that the tool generates the `id`. **Don't touch it.**
|
|
|
|
|
Its value depends on the content of the message and its assigned meaning.
|
|
|
|
|
Change either factor and the `id` changes as well.
|
|
|
|
|
See the **[translation file maintenance discussion](guide/i18n#maintenance)**.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Translate the other text nodes the same way:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="translated-other-nodes" title="src/locale/messages.es.xlf (<trans-unit>)" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
<div class="alert is-important">
|
|
|
|
|
|
|
|
|
|
**The tool generated the `id`s for _these_ translation units. Don't touch them.**
|
|
|
|
|
Each `id` depends upon the content of the message and its assigned meaning.
|
|
|
|
|
Change either factor and the `id` changes as well.
|
|
|
|
|
See the **[translation file maintenance discussion](#maintenance)**.
|
|
|
|
|
|
|
|
|
|
This is why you should **[specify custom ids](#custom-id "Set a custom id")** and avoid tool generated ids.
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
{@a translate-plural-select}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
## Translate _plural_ and _select_
|
|
|
|
|
Translating _plural_ and _select_ messages is a little tricky.
|
|
|
|
|
|
|
|
|
@ -496,121 +447,92 @@ However, the `XMB` format does support the ICU rules.
|
|
|
|
|
You'll just have to look for them in relation to other translation units that you recognize from elsewhere in the source template.
|
|
|
|
|
In this example, you know the translation unit for the `select` must be just below the translation unit for the logo.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a translate-plural}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Translate _plural_
|
|
|
|
|
To translate a `plural`, translate its ICU format match values:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="translated-plural" title="src/locale/messages.es.xlf (<trans-unit>)" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a translate-select}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Translate _select_
|
|
|
|
|
The `select` behaves a little differently. Here again is the ICU format message in the component template:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/app.component.html" region="i18n-select" title="src/app/app.component.html" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
The extraction tool broke that into _two_ translation units.
|
|
|
|
|
|
|
|
|
|
The first unit contains the text that was _outside_ the `select`.
|
|
|
|
|
In place of the `select` is a placeholder, `<x id="ICU">`, that represents the `select` message.
|
|
|
|
|
Translate the text and leave the placeholder where it is.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="translate-select-1" title="src/locale/messages.es.xlf (<trans-unit>)" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
The second translation unit, immediately below the first one, contains the `select` message. Translate that.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="translate-select-2" title="src/locale/messages.es.xlf (<trans-unit>)" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Here they are together, after translation:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="translated-select" title="src/locale/messages.es.xlf (<trans-unit>)" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
### Translate a nested expression
|
|
|
|
|
|
|
|
|
|
A nested expression is not different from the previous ones. As in the previous example, we have _two_ translation units.
|
|
|
|
|
|
|
|
|
|
<div class='l-main-content'>
|
|
|
|
|
The first one contains the text outside the nested expression:
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="translate-nested-1" title="src/locale/messages.es.xlf (<trans-unit>)" linenums="false">
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
The second unit contains the complete nested expression:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="translate-nested-2" title="src/locale/messages.es.xlf (<trans-unit>)" linenums="false">
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
And both together:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/locale/messages.es.xlf.html" region="translate-nested" title="src/locale/messages.es.xlf (<trans-unit>)" linenums="false">
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
The entire template translation is complete. It's
|
|
|
|
|
time to incorporate that translation into the application.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<div id='app-pre-translation'>
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<a id='app-pre-translation'></a>
|
|
|
|
|
|
|
|
|
|
### The app before translation
|
|
|
|
|
|
|
|
|
|
When the previous steps finish, the sample app _and_ its translation file are as follows:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<code-tabs>
|
|
|
|
|
|
|
|
|
|
<code-pane title="src/app/app.component.html" path="i18n/src/app/app.component.html">
|
|
|
|
|
|
|
|
|
|
</code-pane>
|
|
|
|
|
|
|
|
|
|
<code-pane title="src/app/app.component.ts" path="i18n/src/app/app.component.ts">
|
|
|
|
|
|
|
|
|
|
</code-pane>
|
|
|
|
|
|
|
|
|
|
<code-pane title="src/app/app.module.ts" path="i18n/src/app/app.module.ts">
|
|
|
|
|
|
|
|
|
|
</code-pane>
|
|
|
|
|
|
|
|
|
|
<code-pane title="src/main.ts" path="i18n/src/main.1.ts">
|
|
|
|
|
|
|
|
|
|
</code-pane>
|
|
|
|
|
|
|
|
|
|
<code-pane title="src/locale/messages.es.xlf" path="i18n/src/locale/messages.es.xlf.html">
|
|
|
|
|
|
|
|
|
|
</code-pane>
|
|
|
|
|
|
|
|
|
|
</code-tabs>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a merge}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
## Merge the completed translation file into the app
|
|
|
|
|
|
|
|
|
|
To merge the translated text into component templates,
|
|
|
|
|
compile the application with the completed translation file.
|
|
|
|
|
The process is the same whether the file is in `.xlf` format or
|
|
|
|
|
in another format that Angular understands, such as `.xlif` or `.xtb`.
|
|
|
|
|
in another format that Angular understands, such as `.xtb`.
|
|
|
|
|
|
|
|
|
|
You provide the Angular compiler with three new pieces of information:
|
|
|
|
|
|
|
|
|
@ -625,11 +547,8 @@ the JIT (_Just-in-Time_) compiler or the AOT (_Ahead-of-Time_) compiler.
|
|
|
|
|
* With [JIT](guide/i18n#jit), you provide the information at bootstrap time.
|
|
|
|
|
* With [AOT](guide/i18n#aot), you pass the information as `ngc` options.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a jit}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Merge with the JIT compiler
|
|
|
|
|
|
|
|
|
|
The JIT compiler compiles the application in the browser as the application loads.
|
|
|
|
@ -643,20 +562,22 @@ Translation with the JIT compiler is a dynamic process of:
|
|
|
|
|
Open `index.html` and revise the launch script as follows:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/index.html" region="i18n" title="index.html (launch script)" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
In this sample, the user's language is hardcoded as a global `document.locale` variable
|
|
|
|
|
In this sample, the user's language is hard-coded as a global `document.locale` variable
|
|
|
|
|
in the `index.html`.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a text-plugin}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### SystemJS text plugin
|
|
|
|
|
|
|
|
|
|
<div class="alert is-important">
|
|
|
|
|
|
|
|
|
|
This plugin only applies to an application using SystemJS. If you are using the Angular CLI, please refer to their
|
|
|
|
|
[docs](https://github.com/angular/angular-cli/wiki/xi18n).
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
Notice the SystemJS mapping of `text` to a `systemjs-text-plugin.js`.
|
|
|
|
|
With the help of a text plugin, SystemJS can read any file as raw text and
|
|
|
|
|
return the contents as a string.
|
|
|
|
@ -666,33 +587,26 @@ SystemJS doesn't ship with a raw text plugin but it's easy to add.
|
|
|
|
|
Create the following `systemjs-text-plugin.js` in the `src/` folder:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/systemjs-text-plugin.js" title="src/systemjs-text-plugin.js" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a create-translation-providers}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Create translation providers
|
|
|
|
|
|
|
|
|
|
Three providers tell the JIT compiler how to translate the template texts for a particular language
|
|
|
|
|
while compiling the application:
|
|
|
|
|
|
|
|
|
|
* `TRANSLATIONS` is a string containing the content of the translation file.
|
|
|
|
|
* `TRANSLATIONS_FORMAT` is the format of the file: `xlf`, `xlif`, or `xtb`.
|
|
|
|
|
* `TRANSLATIONS_FORMAT` is the format of the file: `xlf`, `xlf2`, or `xtb`.
|
|
|
|
|
* `LOCALE_ID` is the locale of the target language.
|
|
|
|
|
|
|
|
|
|
The `getTranslationProviders()` function in the following `src/app/i18n-providers.ts`
|
|
|
|
|
creates those providers based on the user's _locale_
|
|
|
|
|
and the corresponding translation file:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/i18n-providers.ts" title="src/app/i18n-providers.ts">
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/i18n-providers.ts" region="without-missing-translation" title="src/app/i18n-providers.ts">
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1. It gets the locale from the global `document.locale` variable that was set in `index.html`.
|
|
|
|
|
|
|
|
|
|
1. If there is no locale or the language is U.S. English (`en-US`), there is no need to translate.
|
|
|
|
@ -709,10 +623,15 @@ Notice that it appends `!text` to the filename, telling SystemJS to use the [tex
|
|
|
|
|
|
|
|
|
|
1. Finally, `getTranslationProviders()` returns the entire effort as a promise.
|
|
|
|
|
|
|
|
|
|
<div class="alert is-important">
|
|
|
|
|
|
|
|
|
|
The `LOCALE_ID` has to be a valid locale id as explained in [here](http://userguide.icu-project.org/locale).
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
{@a bootstrap-the-app}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Bootstrap the app with translation providers
|
|
|
|
|
### Bootstrap with translation providers
|
|
|
|
|
|
|
|
|
|
The Angular `bootstrapModule()` method has a second _options_ parameter
|
|
|
|
|
that can influence the behavior of the compiler.
|
|
|
|
@ -722,22 +641,16 @@ and pass it to `bootstrapModule`.
|
|
|
|
|
Open the `src/main.ts` and modify the bootstrap code as follows:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/main.ts" title="src/main.ts" linenums="false">
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Notice that it waits for the `getTranslationProviders()` promise to resolve before
|
|
|
|
|
bootstrapping the app.
|
|
|
|
|
|
|
|
|
|
The app is now _internationalized_ for English and Spanish and there is a clear path for adding
|
|
|
|
|
more languages.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{@a aot}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### _Internationalization_ with the AOT compiler
|
|
|
|
|
|
|
|
|
|
The JIT compiler translates the application into the target language
|
|
|
|
@ -752,7 +665,7 @@ language. Then in the host web page, in this case `index.html`,
|
|
|
|
|
you determine which language the user needs
|
|
|
|
|
and serve the appropriate application package.
|
|
|
|
|
|
|
|
|
|
This cookbook doesn't cover how to build multiple application packages and
|
|
|
|
|
This guide doesn't cover how to build multiple application packages and
|
|
|
|
|
serve them according to the user's language preference.
|
|
|
|
|
It does explain the few steps necessary to tell the AOT compiler to apply a translations file.
|
|
|
|
|
|
|
|
|
@ -760,7 +673,7 @@ Internationalization with the AOT compiler requires
|
|
|
|
|
some setup specifically for AOT compilation.
|
|
|
|
|
Start with the application project as shown
|
|
|
|
|
[just before merging the translation file](guide/i18n#app-pre-translation)
|
|
|
|
|
and refer to the [AOT cookbook](guide/aot-compiler) to make it _AOT-ready_.
|
|
|
|
|
and refer to the [AOT guide](guide/aot-compiler) to make it _AOT-ready_.
|
|
|
|
|
|
|
|
|
|
Next, issue an `ngc` compile command for each supported language, including English.
|
|
|
|
|
The result is a separate version of the application for each language.
|
|
|
|
@ -775,45 +688,65 @@ For this sample, the Spanish language command would be:
|
|
|
|
|
|
|
|
|
|
<code-example language="sh" class="code-shell">
|
|
|
|
|
./node_modules/.bin/ngc --i18nFile=./locale/messages.es.xlf --locale=es --i18nFormat=xlf
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<div class="l-sub-section">
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Windows users may have to quote the command:
|
|
|
|
|
|
|
|
|
|
<code-example language="sh" class="code-shell">
|
|
|
|
|
"./node_modules/.bin/ngc" --i18nFile=./locale/messages.es.xlf --locale=es --i18nFormat=xlf
|
|
|
|
|
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
### Report missing translations
|
|
|
|
|
|
|
|
|
|
If you forgot to provide a translation, the build will succeed with a warning that you might easily overlook.
|
|
|
|
|
You can configure the Angular compiler for different "missing translation" behaviors:
|
|
|
|
|
|
|
|
|
|
* Error
|
|
|
|
|
* Warning (default)
|
|
|
|
|
* Ignore
|
|
|
|
|
|
|
|
|
|
To change the behavior in JIT, you can use the following configuration:
|
|
|
|
|
|
|
|
|
|
<code-example language="typescript">
|
|
|
|
|
{ provide: CompilerConfig, useValue: new CompilerConfig({ missingTranslation: MissingTranslationStrategy.Error }) }
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
A good place to use it is the translation providers:
|
|
|
|
|
|
|
|
|
|
<code-example path="i18n/src/app/i18n-providers.ts" region="missing-translation" title="src/app/i18n-providers.ts"></code-example>
|
|
|
|
|
|
|
|
|
|
To change the behavior in AOT, add the `--missingTranslation` flag to the compilation command:
|
|
|
|
|
|
|
|
|
|
<code-example language="sh" class="code-shell">
|
|
|
|
|
./node_modules/.bin/ngc --i18nFile=./locale/messages.es.xlf --locale=es --i18nFormat=xlf --missingTranslation=error
|
|
|
|
|
</code-example>
|
|
|
|
|
|
|
|
|
|
{@a maintenance}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
## Translation file maintenance and _id_ changes
|
|
|
|
|
## File maintenance and _id_ changes
|
|
|
|
|
|
|
|
|
|
As the application evolves, you will change the _i18n_ markup
|
|
|
|
|
and re-run the `ng-xi18n` extraction tool many times.
|
|
|
|
|
The _new_ markup that you add is not a problem;
|
|
|
|
|
but _most_ changes to existing markup trigger
|
|
|
|
|
generation of new `id`s for the affected translation units.
|
|
|
|
|
The _new_ markup that you add is not a problem.
|
|
|
|
|
But the `id` _can be a serious problem!_
|
|
|
|
|
|
|
|
|
|
If the `id` is generated by the tool, _most_ changes to _existing_ markup
|
|
|
|
|
cause the tool to generate a _new_ `id` for the affected translation unit.
|
|
|
|
|
|
|
|
|
|
After an `id` changes, the translation files are no longer in sync.
|
|
|
|
|
**All translated versions of the application will fail** during re-compilation.
|
|
|
|
|
The error messages identify the old `id`s that are no longer valid but
|
|
|
|
|
they don't tell you what the new `id`s should be.
|
|
|
|
|
Because of that, you get some warning messages during re-compilation.
|
|
|
|
|
The warning messages identify that some translations are missing, but they don't tell you which
|
|
|
|
|
old `ids` are no longer valid.
|
|
|
|
|
|
|
|
|
|
**Commit all translation message files to source control**,
|
|
|
|
|
If you use a [custom id](#custom-id "Set a custom id"),
|
|
|
|
|
the tooling preserves the custom `id` as you make changes to the corresponding translation unit. **Use custom _ids_ unless you have a very good reason to do otherwise.**
|
|
|
|
|
|
|
|
|
|
Whether you use generated or custom `ids`, **always commit all translation message files to source control**,
|
|
|
|
|
especially the English source `messages.xlf`.
|
|
|
|
|
The difference between the old and the new `messages.xlf` file
|
|
|
|
|
help you find and update `id` changes across your translation files.
|
|
|
|
|
will help you find and update `ids` and other changes across your translation files.
|
|
|
|
|
|
|
|
|
|