3 * Try to be consistent about string vs. message (it's probably not yet).
7 You need to put all of its user-visible strings into a file
8 named
<a href=
"i18n-messages.html"><code>messages.json
</code></a>.
9 Each time you add a new locale,
10 you add a messages file
12 named
<code>_locales/
<em>localeCode
</em></code>,
13 where
<em>localeCode
</em> is a code such as
14 <code>en
</code> for English.
18 Here's the file hierarchy
19 for an internationalized extension that supports
20 English (
<code>en
</code>),
21 Spanish (
<code>es
</code>), and
22 Korean (
<code>ko
</code>):
25 <img src=
"{{static}}/images/i18n-hierarchy.gif"
26 alt='In the extension directory: manifest.json, *.html, *.js, _locales directory. In the _locales directory: en, es, and ko directories, each with a messages.json file.'
27 width=
"385" height=
"77" />
30 <h2 id=
"l10">How to support multiple languages
</h2>
33 Say you have an extension
34 with the files shown in the following figure:
37 <img src=
"{{static}}/images/i18n-before.gif"
38 alt='A manifest.json file and a file with JavaScript. The .json file has
"name":
"Hello World". The JavaScript file has title =
"Hello World";'
39 width=
"323" height=
"148">
42 To internationalize this extension,
43 you name each user-visible string
44 and put it into a messages file.
45 The extension's manifest,
48 use each string's name to get its localized version.
52 Here's what the extension looks like when it's internationalized
53 (note that it still has only English strings):
56 <img src=
"{{static}}/images/i18n-after-1.gif"
57 alt='In the manifest.json file,
"Hello World" has been changed to
"__MSG_extName__", and a new
"default_locale" item has the value
"en". In the JavaScript file,
"Hello World" has been changed to chrome.i18n.getMessage(
"extName"). A new file named _locales/en/messages.json defines
"extName".'
58 width=
"782" height=
"228">
62 If an extension has a
<code>_locales
</code> directory,
63 the
<a href=
"manifest.html">manifest
</a>
64 <b>must
</b> define
"default_locale".
68 Some notes about internationalizing:
73 You can use any of the
<a href=
"#overview-locales">supported locales
</a>.
74 If you use an unsupported locale,
75 Google Chrome ignores it.
79 In
<code>manifest.json
</code>
81 refer to a string named
<em>messagename
</em> like this:
82 <pre>__MSG_
<em>messagename
</em>__
</pre>
86 In your extension or app's JavaScript code,
87 refer to a string named
<em>messagename
</em>
89 <pre>chrome.i18n.getMessage(
"<em>messagename</em>")
</pre>
92 In each call to
<code>getMessage()
</code>,
93 you can supply up to
9 strings
94 to be included in the message.
95 See
<a href=
"#examples-getMessage">Examples: getMessage
</a>
101 Some messages, such as
<code>@@bidi_dir
</code> and
<code>@@ui_locale
</code>,
102 are provided by the internationalization system.
103 See the
<a href=
"#overview-predefined">Predefined messages
</a> section
104 for a full list of predefined message names.
109 In
<code>messages.json
</code>,
110 each user-visible string has a name, a
"message" item,
111 and an optional
"description" item.
113 such as
"extName" or
"search_string"
114 that identifies the string.
115 The
"message" specifies
116 the value of the string in this locale.
117 The optional
"description"
118 provides help to translators,
119 who might not be able to see how the string is used in your extension.
121 <pre data-filename=
"messages.json">
124 "message":
"hello%20world",
125 "description":
"The string we search for. Put %20 between words that go together."
131 For more information, see
132 <a href=
"i18n-messages.html">Formats: Locale-Specific Messages
</a>.
138 Once an extension or app is internationalized,
139 translating it is simple.
140 You copy
<code>messages.json
</code>,
142 and put the copy into a new directory under
<code>_locales
</code>.
143 For example, to support Spanish,
144 just put a translated copy of
<code>messages.json
</code>
145 under
<code>_locales/es
</code>.
146 The following figure shows the previous extension
147 with a new Spanish translation.
150 <img src=
"{{static}}/images/i18n-after-2.gif"
151 alt='This looks the same as the previous figure, but with a new file at _locales/es/messages.json that contains a Spanish translation of the messages.'
152 width=
"782" height=
"358">
155 <h2 id=
"overview-predefined">Predefined messages
</h2>
158 The internationalization system provides a few predefined
159 messages to help you localize.
160 These include
<code>@@ui_locale
</code>,
161 so you can detect the current UI locale,
162 and a few
<code>@@bidi_...
</code> messages
163 that let you detect the text direction.
164 The latter messages have similar names to constants in the
165 <a href=
"http://code.google.com/apis/gadgets/docs/i18n.html#BIDI">
166 gadgets BIDI (bi-directional) API
</a>.
170 The special message
<code>@@extension_id
</code>
171 can be used in the CSS and JavaScript files,
172 whether or not the extension or app is localized.
173 This message doesn't work in manifest files.
177 The following table describes each predefined message.
182 <th>Message name
</th> <th>Description
</th>
185 <td> <code>@@extension_id
</code> </td>
186 <td>The extension or app ID;
187 you might use this string to construct URLs
188 for resources inside the extension.
189 Even unlocalized extensions can use this message.
191 <b>Note:
</b> You can't use this message in a manifest file.
195 <td> <code>@@ui_locale
</code> </td>
196 <td>The current locale;
197 you might use this string to construct locale-specific URLs.
</td>
200 <td> <code>@@bidi_dir
</code> </td>
201 <td> The text direction for the current locale,
202 either
"ltr" for left-to-right languages such as English
203 or
"rtl" for right-to-left languages such as Japanese.
</td>
206 <td> <code>@@bidi_reversed_dir
</code> </td>
207 <td> If the
<code>@@bidi_dir
</code> is
"ltr", then this is
"rtl";
208 otherwise, it's
"ltr".
</td>
211 <td> <code>@@bidi_start_edge
</code> </td>
212 <td> If the
<code>@@bidi_dir
</code> is
"ltr", then this is
"left";
213 otherwise, it's
"right".
</td>
216 <td> <code>@@bidi_end_edge
</code> </td>
217 <td> If the
<code>@@bidi_dir
</code> is
"ltr", then this is
"right";
218 otherwise, it's
"left".
</td>
223 Here's an example of using
<code>@@extension_id
</code> in a CSS file
229 <b>background-image:url('chrome-extension://__MSG_@@extension_id__/background.png');
</b>
234 If the extension ID is abcdefghijklmnopqrstuvwxyzabcdef,
235 then the bold line in the previous code snippet becomes:
239 background-image:url('chrome-extension://abcdefghijklmnopqrstuvwxyzabcdef/background.png');
243 Here's an example of using
<code>@@bidi_*
</code> messages in a CSS file:
248 <b>direction: __MSG_@@bidi_dir__;
</b>
252 margin-bottom:
1.05em;
254 padding-bottom:
1.5em;
255 <b>padding-__MSG_@@bidi_start_edge__:
0;
</b>
256 <b>padding-__MSG_@@bidi_end_edge__:
1.5em;
</b>
262 For left-to-right languages such as English,
263 the bold lines become:
269 padding-right:
1.5em;
273 <h2 id=
"overview-locales">Locales
</h2>
276 You can choose from many locales,
277 including some (such as
<code>en
</code>)
278 that let a single translation support multiple variations of a language
279 (such as
<code>en_GB
</code> and
<code>en_US
</code>).
283 <h3 id=
"locales-supported">Supported locales
</h3>
286 You can use any of the
287 <a href=
"http://code.google.com/chrome/webstore/docs/i18n.html#localeTable">locales that the Chrome Web Store supports
</a>.
291 <h3 id=
"locales-usage">Searching for messages
</h3>
294 You don't have to define every string for every supported locale.
295 As long as the default locale's
<code>messages.json
</code> file
296 has a value for every string,
297 your extension or app will run no matter how sparse a translation is.
298 Here's how the extension system searches for a message:
303 Search the messages file (if any)
304 for the user's preferred locale.
305 For example, when Google Chrome's locale is set to
306 British English (
<code>en_GB
</code>),
307 the system first looks for the message in
308 <code>_locales/en_GB/messages.json
</code>.
309 If that file exists and the message is there,
310 the system looks no further.
313 If the user's preferred locale has a region
314 (that is, the locale has an underscore: _),
315 search the locale without that region.
316 For example, if the
<code>en_GB
</code> messages file
317 doesn't exist or doesn't contain the message,
318 the system looks in the
<code>en
</code> messages file.
319 If that file exists and the message is there,
320 the system looks no further.
323 Search the messages file for the default locale.
324 For example, if the extension's
"default_locale" is set to
"es",
325 and neither
<code>_locales/en_GB/messages.json
</code>
326 nor
<code>_locales/en/messages.json
</code> contains the message,
327 the extension uses the message from
328 <code>_locales/es/messages.json
</code>.
333 In the following figure,
334 the message named
"colores" is in all three locales
335 that the extension supports,
336 but
"extName" is in only two of the locales.
337 Wherever a user running Google Chrome in US English sees the label
"Colors",
338 a user of British English sees
"Colours".
339 Both US English and British English users
340 see the extension name
"Hello World".
341 Because the default language is Spanish,
342 users running Google Chrome in any non-English language
343 see the label
"Colores" and the extension name
"Hola mundo".
346 <img src=
"{{static}}/images/i18n-strings.gif"
347 alt='Four files: manifest.json and three messages.json files (for es, en, and en_GB). The es and en files show entries for messages named
"extName" and
"colores"; the en_GB file has just one entry (for
"colores").'
348 width=
"493" height=
"488" />
350 <h3 id=
"locales-testing">How to set your browser's locale
</h3>
353 To test translations, you might want to set your browser's locale.
354 This section tells you how to set the locale in
355 <a href=
"#testing-win">Windows
</a>,
356 <a href=
"#testing-mac">Mac OS X
</a>, and
357 <a href=
"#testing-linux">Linux
</a>.
360 <h4 id=
"testing-win">Windows
</h4>
363 You can change the locale using either
364 a locale-specific shortcut
365 or the Google Chrome UI.
366 The shortcut approach is quicker, once you've set it up,
367 and it lets you use several languages at once.
370 <h5 id=
"win-shortcut">Using a locale-specific shortcut
</h5>
373 To create and use a shortcut that launches Google Chrome
374 with a particular locale:
379 Make a copy of the Google Chrome shortcut
380 that's already on your desktop.
383 Rename the new shortcut to match the new locale.
386 Change the shortcut's properties
387 so that the Target field specifies the
388 <code>--lang
</code> and
389 <code>--user-data-dir
</code> flags.
390 The target should look something like this:
392 <pre><em>path_to_chrome.exe
</em> --lang=
<em>locale
</em> --user-data-dir=c:\
<em>locale_profile_dir
</em></pre>
396 Launch Google Chrome by double-clicking the shortcut.
401 For example, to create a shortcut
402 that launches Google Chrome in Spanish (
<code>es
</code>),
403 you might create a shortcut named
<code>chrome-es
</code>
404 that has the following target:
407 <pre><em>path_to_chrome.exe
</em> --lang=es --user-data-dir=c:\chrome-profile-es
</pre>
410 You can create as many shortcuts as you like,
411 making it easy to test in multiple languages.
415 <pre><em>path_to_chrome.exe
</em> --lang=en --user-data-dir=c:\chrome-profile-en
416 <em>path_to_chrome.exe
</em> --lang=en_GB --user-data-dir=c:\chrome-profile-en_GB
417 <em>path_to_chrome.exe
</em> --lang=ko --user-data-dir=c:\chrome-profile-ko
</pre>
421 Specifying
<code>--user-data-dir
</code> is optional but handy.
422 Having one data directory per locale
423 lets you run the browser
424 in several languages at the same time.
425 A disadvantage is that because the locales' data isn't shared,
426 you have to install your extension multiple times
— once per locale,
427 which can be challenging when you don't speak the language.
428 For more information, see
429 <a href=
"http://www.chromium.org/developers/creating-and-using-profiles">Creating and Using Profiles
</a>.
433 <h5 id=
"win-ui">Using the UI
</h5>
436 Here's how to change the locale using the UI on Google Chrome for Windows:
440 <li> Wrench icon
> <b>Options
</b> </li>
441 <li> Choose the
<b>Under the Hood
</b> tab
</li>
442 <li> Scroll down to
<b>Web Content
</b> </li>
443 <li> Click
<b>Change font and language settings
</b> </li>
444 <li> Choose the
<b>Languages
</b> tab
</li>
445 <li> Use the drop down to set the
<b>Google Chrome language
</b> </li>
446 <li> Restart Chrome
</li>
450 <h4 id=
"testing-mac">Mac OS X
</h4>
453 To change the locale on Mac,
454 you use the system preferences.
458 <li> From the Apple menu, choose
<b>System Preferences
</b> </li>
459 <li> Under the
<b>Personal
</b> section, choose
<b>International
</b> </li>
460 <li> Choose your language and location
</li>
461 <li> Restart Chrome
</li>
465 <h4 id=
"testing-linux">Linux
</h4>
468 To change the locale on Linux,
469 first quit Google Chrome.
470 Then, all in one line,
471 set the LANGUAGE environment variable
472 and launch Google Chrome.
481 <h2 id=
"overview-examples">Examples
</h2>
484 You can find simple examples of internationalization in the
485 <a href=
"http://src.chromium.org/viewvc/chrome/trunk/src/chrome/common/extensions/docs/examples/api/i18n/">examples/api/i18n
</a>
487 For a complete example, see
488 <a href=
"http://src.chromium.org/viewvc/chrome/trunk/src/chrome/common/extensions/docs/examples/extensions/news/">examples/extensions/news
</a>.
489 For other examples and for help in viewing the source code, see
490 <a href=
"samples.html">Samples
</a>.
494 <h3 id=
"examples-getMessage">Examples: getMessage
</h3>
497 [PENDING: improve this section. it should probably start with a
498 one-variable example that includes the messages.json code.]
502 The following code gets a localized message from the browser
503 and displays it as a string.
504 It replaces two placeholders within the message with the strings
505 "string1" and
"string2".
509 function getMessage() {
510 var message = chrome.i18n.getMessage(
"click_here", [
"string1",
"string2"]);
511 document.getElementById(
"languageSpan").innerHTML = message;
516 Here's how you'd supply and use a single string:
520 <em>// In JavaScript code
</em>
521 status.innerText = chrome.i18n.getMessage(
"error", errorDetails);
523 <pre data-filename=
"messages.json">
525 "message":
"Error: $details$",
526 "description":
"Generic error template. Expects error parameter to be passed in.",
530 "example":
"Failed to fetch RSS feed."
537 For more information about placeholders, see the
538 <a href=
"i18n-messages.html">Locale-Specific Messages
</a> page.
539 For details on calling
<code>getMessage()
</code>, see the
540 $ref:[i18n.getMessage API reference].
543 <h3 id=
"example-accept-languages">Example: getAcceptLanguages
</h3>
545 The following code gets accept-languages from the browser and displays them as a
546 string by separating each accept-language with ','.
550 function getAcceptLanguages() {
551 chrome.i18n.getAcceptLanguages(function(languageList) {
552 var languages = languageList.join(
",");
553 document.getElementById(
"languageSpan").innerHTML = languages;
559 For details on calling
<code>getAcceptLanguages()
</code>, see the
560 $ref:[i18n.getAcceptLanguages API reference].