Difference between revisions of "Internationalization/User guide"

m (Interface)
 
(11 intermediate revisions by 3 users not shown)
Line 1: Line 1:
=User documentation for the i18n library=
+
[[Category:Internationalization]]
 
+
 
==Overview==
 
==Overview==
  
Line 12: Line 11:
 
==Interface==
 
==Interface==
  
The library provides most of it's services through one class: LOCALE. This presents all formatting and translation facilities for a given locale.
+
The library provides most of its services through one class: LOCALE. This presents all formatting and translation facilities for a given locale.
 
LOCALE objects can't be created directly: one must go though the LOCALE_MANAGER class. A LOCALE_MANAGER finds out what information for which locales is available, and offers a list to chose from. It will then load the information for the chosen locale into a LOCALE object and give it to you. With this LOCALE object you can do the really interesting things, like formatting dates and translating strings.
 
LOCALE objects can't be created directly: one must go though the LOCALE_MANAGER class. A LOCALE_MANAGER finds out what information for which locales is available, and offers a list to chose from. It will then load the information for the chosen locale into a LOCALE object and give it to you. With this LOCALE object you can do the really interesting things, like formatting dates and translating strings.
  
Line 18: Line 17:
  
 
First you must have a LOCALE_MANAGER. For details on creating them, please see the '''Datasources''' section.
 
First you must have a LOCALE_MANAGER. For details on creating them, please see the '''Datasources''' section.
If you just want to use whatever locale is the default on the user's machine, as in most cases, then it's easy to get a LOCALE object: just call get_system_locale.
+
If you just want to use whatever locale is the default on the user's machine, as in most cases, then it's easy to get a LOCALE object: just call default_locale.
 
<code>[eiffel, N]
 
<code>[eiffel, N]
my_locale := locale_mananger.get_system_locale
+
my_locale := locale_mananger.default_locale
 
</code>
 
</code>
  
Line 28: Line 27:
 
A locale is identified by a LOCALE_ID object, and normally this has two components: language code and country code.
 
A locale is identified by a LOCALE_ID object, and normally this has two components: language code and country code.
 
For example, the US english locale has language code "en" and country code "US".  
 
For example, the US english locale has language code "en" and country code "US".  
If you've found a locale id  that you like in the list returned by available_locales, you can check exactly what is vailable for it by using  
+
If you've found a locale id  that you like in the list returned by available_locales, you can check exactly what is available for it by using  
 
<code>[eiffel, N]
 
<code>[eiffel, N]
 
has_translations (a_locale_id: I18N_LOCALE_ID): BOOLEAN
 
has_translations (a_locale_id: I18N_LOCALE_ID): BOOLEAN
 +
has_localised_translations (a_locale_id: I18N_LOCALE_ID): BOOLEAN
 
has_formatting_info (a_locale_id: I18N_LOCALE_ID): BOOLEAN
 
has_formatting_info (a_locale_id: I18N_LOCALE_ID): BOOLEAN
 
</code>
 
</code>
  
You can then retrieve the corresponding LOCALE object by calling get_locale.
+
(The difference between a translation and a localised translation is that a localised translation is specific to a region: fr_CA or de_CH, for example. A translation is specific only to a language: fr or de, for example)
 +
 
 +
You can then retrieve the corresponding LOCALE object by calling locale_with_id, providing the desired LOCALE_ID.
  
  
Line 44: Line 46:
 
=====Interface=====
 
=====Interface=====
 
<code>[eiffel, N]
 
<code>[eiffel, N]
translate (original: STRING_GENERAL): STRING_32
+
translated (original: STRING_GENERAL): STRING_32
translate_plural (original_singular, original_plural: STRING_GENERAL; plural_form : INTEGER): STRING_32
+
translated_plural (original_singular, original_plural: STRING_GENERAL; plural_form : INTEGER): STRING_32
format_string (original: STRING_GENERAL; token_values: TUPLE[STRING_GENERAL]): STRING_32
+
formatted_string (original: STRING_GENERAL; token_values: TUPLE[STRING_GENERAL]): STRING_32
 
</code>
 
</code>
 
=====Usage=====
 
=====Usage=====
Line 58: Line 60:
 
The resulting code would look like this
 
The resulting code would look like this
 
<code>[eiffel, N]
 
<code>[eiffel, N]
io.put_string(my_locale.translate("My hovercraft is full of eels"))
+
io.put_string(my_locale.translated("My hovercraft is full of eels"))
 
</code>
 
</code>
 
If the translate function can't find a translation for this string it will simply return the original string - better then nothing!
 
If the translate function can't find a translation for this string it will simply return the original string - better then nothing!
Line 64: Line 66:
 
But life is, of course, not always that simple. What if we have to deal with plurals? The "traditional" way of doing this is something like:
 
But life is, of course, not always that simple. What if we have to deal with plurals? The "traditional" way of doing this is something like:
 
<code>[eiffel, N]
 
<code>[eiffel, N]
n := get_number_of_hovercraft
+
n := number_of_hovercraft
 
if n = 1 then
 
if n = 1 then
 
io.put_string("My hovercraft is full of eels")
 
io.put_string("My hovercraft is full of eels")
Line 74: Line 76:
 
This is not so easy to translate as the above. Why can't we just translate both strings?  
 
This is not so easy to translate as the above. Why can't we just translate both strings?  
  
Depending on the language, there may be up to 4 different types of plural forms, used in strange and exotic ways. Clearly, it is important to know exactly _how_ many hovercraft there are so that we can choose the right plural form. This can be done by the translate_plural function, whoch we can use in this way:
+
Depending on the language, there may be up to 4 different types of plural forms, used in strange and exotic ways. Clearly, it is important to know exactly _how_ many hovercraft there are so that we can choose the right plural form. This can be done by the translate_plural function, which we can use in this way:
 
<code>[eiffel,n]
 
<code>[eiffel,n]
n := get_number_of_hovercraft
+
n := number_of_hovercraft
io.put_string(my_locale.translate_plural("My hovercraft is full of eels","My hovercraft are full of eels",n))
+
io.put_string(my_locale.translated_plural("My hovercraft is full of eels","My hovercraft are full of eels",n))
 
</code>
 
</code>
 
This function will choose and return a translation in the correct plural form. If it can't find one, it will behave like translate and return either the original singular string or the original plural string, following English grammatical rules.
 
This function will choose and return a translation in the correct plural form. If it can't find one, it will behave like translate and return either the original singular string or the original plural string, following English grammatical rules.
Line 83: Line 85:
 
Often even the above is not enough. What if you want to tell the world exactly how many hovercraft you have? You might write something like this:
 
Often even the above is not enough. What if you want to tell the world exactly how many hovercraft you have? You might write something like this:
 
<code>[eiffel, N]
 
<code>[eiffel, N]
n := get_number_of_hovercraft
+
n := number_of_hovercraft
 
if n = 1 then
 
if n = 1 then
 
io.put_string("My hovercraft is full of eels")
 
io.put_string("My hovercraft is full of eels")
Line 94: Line 96:
 
Let's see how this works:
 
Let's see how this works:
 
<code>[eiffel,n]
 
<code>[eiffel,n]
n := get_number_of_hovercraft
+
n := number_of_hovercraft
plural_string := my_locale.translate_plural("My hovercraft is full of eels","My $1 hovercraft are full of eels",n)
+
plural_string := my_locale.translated_plural("My hovercraft is full of eels","My $1 hovercraft are full of eels",n)
io.put_string(my_locale.format_string(plural_string, [n]))
+
io.put_string(my_locale.formatted_string(plural_string, [n]))
 
</code>
 
</code>
To replace the ''escape codes'', such as $1, $2, we use the function format_string. This replaces all the escape codes it finds by the values in a tuple that you give to it a an argument.  
+
To replace the ''escape codes'', such as $1, $2, we use the function format_string. This replaces all the escape codes it finds by the values in a tuple that you give to it a an argument.
 +
 
 +
=====Misuse=====
 +
 
 +
It is not a good idea, generally speaking, to do something like the following:
 +
 
 +
<code>[eiffel, N]
 +
n := number_of_eels
 +
io.put_string(my_locale.translated("My hovercraft has "+n.out+"eels"))
 +
</code>
 +
 
 +
There are two reasons for this:
 +
 
 +
* What the translation function will see as an argument is runtime-dependant. It may correspond to an actual entry in the datasource, if you are careful, but in the example above it probably won't unless you have entries for every possible value of n. This is the reason for which we have the format_string feature in LOCALE - please use it!
 +
 
 +
*When you want to generate a .po file to give to translators, this sort of thing is not going to make the command that extracts it very happy. What will happen is that it will extract the first constant STRING argument and ignore the rest, so your .po file will have an "My hovercraft has " entry and unless you correct this by hand very probably at runtime the i18n library will not be able to find a translation.  
  
 
====Formatting====
 
====Formatting====
 
=====Interface=====
 
=====Interface=====
TODO!
+
DATE_FORMATTER provides:
 +
<code>[eiffel, N]
 +
formatted_date(date:DATE):STRING_32
 +
formatted_time(time: TIME): STRING_32
 +
formatted_date_time(date_time:DATE_TIME):STRING_32
 +
</code>
 +
 
 +
CURRENCY_FORMATTER provides:
 +
<code>[eiffel, N]
 +
formatted_currency (a_value: REAL_64): STRING_32
 +
</code>
 +
 
 +
VALUE_FORMATTER provides:
 +
<code>[eiffel, N]
 +
formatted_integer_8 (a_integer_8: INTEGER_8): STRING_32
 +
formatted_integer_16 (a_integer_16: INTEGER_16): STRING_32
 +
formatted_integer_32 (a_integer_32: INTEGER_32): STRING_32
 +
formatted_integer_64 (a_integer_64: INTEGER_64): STRING_32
 +
formatted_real_32 (a_real_32: REAL_32): STRING_32
 +
formatted_real_64 (a_real_64: REAL_64): STRING_32
 +
</code>
 
=====Usage=====
 
=====Usage=====
TODO!
+
 
 +
The LOCALE class makes 3 formatters accessible to clients: a VALUE_FORMATTER, a DATE_FORMATTER and a CURRENCY_FORMATTER, exposed as features under the names value_formatter, date_formatter and currency_formatter respectively.
 +
Using these formatters is fairly straightforward: you simply call the appropriate function for  the type of object that you want to format.
 +
 
 +
======Date formatting======
 +
 
 +
The DATE_FORMATTER class can format EiffelTime DATE, TIME and DATE_TIME classes in a way appropriate to the locale. For example, to get a string representation of today's date in a given locale, you might write:
 +
 
 +
<code>[eiffel, n]
 +
io.put_string(my_locale.formatted_date(create {DATE}.make_now))
 +
</code>
 +
 
 +
Currently eras of non-gregorian calendars are not well supported.
 +
 
 +
======Value and currency formatting======
 +
 
 +
The VALUE_FORMATTER and CURRENCY_FORMATTER classes can format integers and reals according to the conventions of a given locale - number of digits after the decimal separator, the decimal separator itself, grouping of digits and so on.
 +
 
 +
Using them is just as easy as DATE_FORMATTER: just call the function appropriate to the type of object who's value you want to format.
  
 
==String extraction==
 
==String extraction==
Line 112: Line 167:
 
We can extract these strings from your application fairly easily by simply looking at the arguments for each call to translate or translate_plural. By clicking on a handy button in EiffelStudio, these strings will be extracted and placed in a [http://www.gnu.org/software/gettext/manual/html_node/gettext_9.html#SEC9 .po file].
 
We can extract these strings from your application fairly easily by simply looking at the arguments for each call to translate or translate_plural. By clicking on a handy button in EiffelStudio, these strings will be extracted and placed in a [http://www.gnu.org/software/gettext/manual/html_node/gettext_9.html#SEC9 .po file].
  
TODO: add more details about clicky thing.
+
If you are using the es-i18n branch, this command is in the "Tools" menu ("Generate .po"). One click and it will prompt you for a location to write the .po file to. This is basically all there is to it.
 +
It is helpful to understand how it works (see the 'Misuse' section): it extracts the arguments of calls to the translate and translate_plural functions. Largely it will not behave satisfactorily if the string is made up of several parts glued together at run-time.
  
 
A .po file is a reasonably widespread format for storing strings to be translated. There are several tools to aid translation of these files, such as [http://www.poedit.org/ poEdit] for Windows and [http://kbabel.kde.org/ KBabel] or [http://gtranslator.sourceforge.net/ gtranslator] for KDE and Gnome.  
 
A .po file is a reasonably widespread format for storing strings to be translated. There are several tools to aid translation of these files, such as [http://www.poedit.org/ poEdit] for Windows and [http://kbabel.kde.org/ KBabel] or [http://gtranslator.sourceforge.net/ gtranslator] for KDE and Gnome.  
Line 122: Line 178:
 
Instead we load the strings from a datasource. This is appropriately generic: it could be anything, from a database to a system that queries a server via RPC or SOAP, but currently we only have one implementation: files. And in fact, we only support one type of file: the .mo file format.  
 
Instead we load the strings from a datasource. This is appropriately generic: it could be anything, from a database to a system that queries a server via RPC or SOAP, but currently we only have one implementation: files. And in fact, we only support one type of file: the .mo file format.  
 
The library can't guess the type of datasource you want to use, so you have to tell it when you create a LOCALE_MANAGER.
 
The library can't guess the type of datasource you want to use, so you have to tell it when you create a LOCALE_MANAGER.
This is done via an ''uri''. Currently all uris are interpreted as directories where string catalog files may be found, and any .mo files in this directory will be used by the i18n library. This means that creating a LOCALE_MANAGER look somewhat like this:
+
This is done via an ''uri''. Currently all uris are interpreted as directories where string catalog files may be found, and any .mo files in this directory will be used by the i18n library. This means that creating a LOCALE_MANAGER looks somewhat like this:
 
<code>[eiffel, N]
 
<code>[eiffel, N]
create my_locale_mananger.make("/path/to/my/files")
+
create my_locale_manager.make("/path/to/my/files")
 
</code>
 
</code>
  
Line 132: Line 188:
  
 
How do you get these .mo files? Once your translator has finished translating a .po file, you can convert it into a .mo file by using the ''msgfmt'' tool, which is obtainable as part of the gettext package under unix and distributed with poEdit under Windows.
 
How do you get these .mo files? Once your translator has finished translating a .po file, you can convert it into a .mo file by using the ''msgfmt'' tool, which is obtainable as part of the gettext package under unix and distributed with poEdit under Windows.
The resulting mo file should be named with the locale identifier of the locale it is intended for (zh_CN.mo or de_CH.mo, for example) and placed in the appropriate directory - this is to say, the one that you give to LOCALE_MANAGER as an uri.
+
The resulting mo file should be named with the locale identifier of the locale it is intended for (zh_CN.mo or de_CH.mo, for example) or the language identifier of the target language (zh or de, for example)  and placed in the appropriate directory - this is to say, the one that you give to LOCALE_MANAGER as an uri.
  
 
The .mo file should then be seen and used by the i18n library.
 
The .mo file should then be seen and used by the i18n library.

Latest revision as of 02:56, 5 April 2007

Overview

The i18n library is intended to enable localisation of Eiffel programs.

localisation is the process of adapting a piece of software to a specific place - the locale, often expressed as a combination of language and country codes.

This normally means not only displaying strings in the appropriate language, but also adapting number formatting, date and time formatting etc. to use local conventions. The i18n library provides formatting facilities for numbers, currency values and dates, and the ability to identify and load translated strings at run-time.

Interface

The library provides most of its services through one class: LOCALE. This presents all formatting and translation facilities for a given locale. LOCALE objects can't be created directly: one must go though the LOCALE_MANAGER class. A LOCALE_MANAGER finds out what information for which locales is available, and offers a list to chose from. It will then load the information for the chosen locale into a LOCALE object and give it to you. With this LOCALE object you can do the really interesting things, like formatting dates and translating strings.

Choosing a locale

First you must have a LOCALE_MANAGER. For details on creating them, please see the Datasources section. If you just want to use whatever locale is the default on the user's machine, as in most cases, then it's easy to get a LOCALE object: just call default_locale.

my_locale := locale_mananger.default_locale

If you want a specific locale, it's going to be a bit more complicated. A LOCALE_MANAGER knows what locales are available and exactly what information is available for a specific locale. You can get a list of all locales that are available to some degree by calling the available_locales feature. A locale is identified by a LOCALE_ID object, and normally this has two components: language code and country code. For example, the US english locale has language code "en" and country code "US". If you've found a locale id that you like in the list returned by available_locales, you can check exactly what is available for it by using

has_translations (a_locale_id: I18N_LOCALE_ID): BOOLEAN
has_localised_translations (a_locale_id: I18N_LOCALE_ID): BOOLEAN
has_formatting_info (a_locale_id: I18N_LOCALE_ID): BOOLEAN

(The difference between a translation and a localised translation is that a localised translation is specific to a region: fr_CA or de_CH, for example. A translation is specific only to a language: fr or de, for example)

You can then retrieve the corresponding LOCALE object by calling locale_with_id, providing the desired LOCALE_ID.


Using your locale

Now you've got a LOCALE. After the initial burst of euphoria has faded away, what can you do with it?

String translation

Interface
translated (original: STRING_GENERAL): STRING_32
translated_plural (original_singular, original_plural: STRING_GENERAL; plural_form : INTEGER): STRING_32
formatted_string (original: STRING_GENERAL; token_values: TUPLE[STRING_GENERAL]): STRING_32
Usage

In naïvely written software, you can often spot things like

io.put_string("My hovercraft is full of eels")

We'll use this example to illustrate the use of the string translation features of the i18n library.

If, as above, there is just one constant string to translate, the solution is very easy: simply use the translate function. The resulting code would look like this

io.put_string(my_locale.translated("My hovercraft is full of eels"))

If the translate function can't find a translation for this string it will simply return the original string - better then nothing!

But life is, of course, not always that simple. What if we have to deal with plurals? The "traditional" way of doing this is something like:

n := number_of_hovercraft
if n = 1 then
	io.put_string("My hovercraft is full of eels")
else
	io.put_string("My hovercraft are full of eels")
end

This is not so easy to translate as the above. Why can't we just translate both strings?

Depending on the language, there may be up to 4 different types of plural forms, used in strange and exotic ways. Clearly, it is important to know exactly _how_ many hovercraft there are so that we can choose the right plural form. This can be done by the translate_plural function, which we can use in this way:

n := number_of_hovercraft
io.put_string(my_locale.translated_plural("My hovercraft is full of eels","My hovercraft are full of eels",n))

This function will choose and return a translation in the correct plural form. If it can't find one, it will behave like translate and return either the original singular string or the original plural string, following English grammatical rules.

Often even the above is not enough. What if you want to tell the world exactly how many hovercraft you have? You might write something like this:

n := number_of_hovercraft
if n = 1 then
	io.put_string("My hovercraft is full of eels")
else
	io.put_string("My "+n.out+" hovercraft are full of eels")
end

How can translate_plural handle this? It needs some reinforcements: the solution is to also use string templates. This means that we can embed codes like "$1" in a string and replace them in the translation by the actual values. Let's see how this works:

n := number_of_hovercraft
plural_string := my_locale.translated_plural("My hovercraft is full of eels","My $1 hovercraft are full of eels",n)
io.put_string(my_locale.formatted_string(plural_string, [n]))

To replace the escape codes, such as $1, $2, we use the function format_string. This replaces all the escape codes it finds by the values in a tuple that you give to it a an argument.

Misuse

It is not a good idea, generally speaking, to do something like the following:

n := number_of_eels
io.put_string(my_locale.translated("My hovercraft has "+n.out+"eels"))

There are two reasons for this:

  • What the translation function will see as an argument is runtime-dependant. It may correspond to an actual entry in the datasource, if you are careful, but in the example above it probably won't unless you have entries for every possible value of n. This is the reason for which we have the format_string feature in LOCALE - please use it!
  • When you want to generate a .po file to give to translators, this sort of thing is not going to make the command that extracts it very happy. What will happen is that it will extract the first constant STRING argument and ignore the rest, so your .po file will have an "My hovercraft has " entry and unless you correct this by hand very probably at runtime the i18n library will not be able to find a translation.

Formatting

Interface

DATE_FORMATTER provides:

formatted_date(date:DATE):STRING_32 
formatted_time(time: TIME): STRING_32 
formatted_date_time(date_time:DATE_TIME):STRING_32

CURRENCY_FORMATTER provides:

formatted_currency (a_value: REAL_64): STRING_32

VALUE_FORMATTER provides:

formatted_integer_8 (a_integer_8: INTEGER_8): STRING_32 
formatted_integer_16 (a_integer_16: INTEGER_16): STRING_32 
formatted_integer_32 (a_integer_32: INTEGER_32): STRING_32
formatted_integer_64 (a_integer_64: INTEGER_64): STRING_32 
formatted_real_32 (a_real_32: REAL_32): STRING_32 
formatted_real_64 (a_real_64: REAL_64): STRING_32
Usage

The LOCALE class makes 3 formatters accessible to clients: a VALUE_FORMATTER, a DATE_FORMATTER and a CURRENCY_FORMATTER, exposed as features under the names value_formatter, date_formatter and currency_formatter respectively. Using these formatters is fairly straightforward: you simply call the appropriate function for the type of object that you want to format.

Date formatting

The DATE_FORMATTER class can format EiffelTime DATE, TIME and DATE_TIME classes in a way appropriate to the locale. For example, to get a string representation of today's date in a given locale, you might write:

io.put_string(my_locale.formatted_date(create {DATE}.make_now))

Currently eras of non-gregorian calendars are not well supported.

Value and currency formatting

The VALUE_FORMATTER and CURRENCY_FORMATTER classes can format integers and reals according to the conventions of a given locale - number of digits after the decimal separator, the decimal separator itself, grouping of digits and so on.

Using them is just as easy as DATE_FORMATTER: just call the function appropriate to the type of object who's value you want to format.

String extraction

Somebody has to translate all these strings. Mostly, this isn't a programmer, so somehow you have to be able to give this translator a list of strings that you want translated.

We can extract these strings from your application fairly easily by simply looking at the arguments for each call to translate or translate_plural. By clicking on a handy button in EiffelStudio, these strings will be extracted and placed in a .po file.

If you are using the es-i18n branch, this command is in the "Tools" menu ("Generate .po"). One click and it will prompt you for a location to write the .po file to. This is basically all there is to it. It is helpful to understand how it works (see the 'Misuse' section): it extracts the arguments of calls to the translate and translate_plural functions. Largely it will not behave satisfactorily if the string is made up of several parts glued together at run-time.

A .po file is a reasonably widespread format for storing strings to be translated. There are several tools to aid translation of these files, such as poEdit for Windows and KBabel or gtranslator for KDE and Gnome. There are also many tools to convert .po files to other formats, such as the xml xliff format.

Datasources

The library has to load the translated strings from somewhere - sadly we can't do on the fly translation but if you can, please tell us! Instead we load the strings from a datasource. This is appropriately generic: it could be anything, from a database to a system that queries a server via RPC or SOAP, but currently we only have one implementation: files. And in fact, we only support one type of file: the .mo file format. The library can't guess the type of datasource you want to use, so you have to tell it when you create a LOCALE_MANAGER. This is done via an uri. Currently all uris are interpreted as directories where string catalog files may be found, and any .mo files in this directory will be used by the i18n library. This means that creating a LOCALE_MANAGER looks somewhat like this:

create my_locale_manager.make("/path/to/my/files")

Mo files

The mo file format is defined by the GNU gettext library, a widely-used C library that allows localisation of text strings. We support UTF-8 encoded mo files.

How do you get these .mo files? Once your translator has finished translating a .po file, you can convert it into a .mo file by using the msgfmt tool, which is obtainable as part of the gettext package under unix and distributed with poEdit under Windows. The resulting mo file should be named with the locale identifier of the locale it is intended for (zh_CN.mo or de_CH.mo, for example) or the language identifier of the target language (zh or de, for example) and placed in the appropriate directory - this is to say, the one that you give to LOCALE_MANAGER as an uri.

The .mo file should then be seen and used by the i18n library.