Internationalization in Drupal 9
Same name and namespace in other branches
Internationalization and translation
The principle of internationalization is that it should be possible to make a Drupal site in any language (or a multi-lingual site), where only content in the desired language is displayed for any particular page request. In order to make this happen, developers of modules, themes, and installation profiles need to make sure that all of the displayable content and user interface (UI) text that their project deals with is internationalized properly, so that it can be translated using the standard Drupal translation mechanisms.
Internationalization
Different types of information in Drupal have different methods for internationalization, and different portions of the UI also have different methods for internationalization. Here is a list of the different mechanisms for internationalization, and some notes:
- UI text is always put into code and related files in English.
- Any time UI text is displayed using PHP code, it should be passed through either the global t() function or a t() method on the class. If it involves plurals, it should be passed through either the global \Drupal\Core\StringTranslation\PluralTranslatableMarkup::createFromTranslatedString() or a formatPlural() method on the class. Use \Drupal\Core\StringTranslation\StringTranslationTrait to get these methods into a class.
- Dates displayed in the UI should be passed through the 'date' service class's format() method. Again see the Services topic; the method to call is \Drupal\Core\Datetime\Date::format().
- Some YML files contain UI text that is automatically translatable:
- *.routing.yml files: route titles. This also applies to *.links.task.yml, *.links.action.yml, and *.links.contextual.yml files.
- *.info.yml files: module names and descriptions.
 
- For configuration, make sure any configuration that is displayable to users is marked as translatable in the configuration schema. Configuration types label, text, and date_format are translatable; string is non-translatable text. See the Config API topic for more information.
- For annotation, make sure that any text that is displayable in the UI is wrapped in @Translation(). See the Plugin translatables topic for more information.
- Content entities are translatable if they have
  translatable = TRUE,
  in their annotation. The use of entities to store user-editable content to be displayed in the site is highly recommended over creating your own method for storing, retrieving, displaying, and internationalizing content.
- For Twig templates, use 't' or 'trans' filters to indicate translatable text. See https://www.drupal.org/node/2133321 for more information.
- In JavaScript code, use the Drupal.t() and Drupal.formatPlural() functions (defined in core/misc/drupal.js) to translate UI text.
- If you are using a custom module, theme, etc. that is not hosted on Drupal.org, see Interface translation properties topic for information on how to make sure your UI text is translatable.
Translation
Once your data and user interface are internationalized, the following Core modules are used to translate it into different languages (machine names of modules in parentheses):
- Language (language): Define which languages are active on the site.
- Interface Translation (locale): Translate UI text.
- Content Translation (content_translation): Translate content entities.
- Configuration Translation (config_translation): Translate configuration.
The Interface Translation module deserves special mention, because besides providing a UI for translating UI text, it also imports community translations from the Drupal translation server. If UI text and provided configuration in Drupal Core and contributed modules, themes, and installation profiles is properly internationalized (as described above), the text is automatically added to the translation server for community members to translate, via *.po files that are generated by scanning the project files.
Translation string sharing and context
By default, translated strings are only translated once, no matter where they are being used. For instance, there are many forms with Save buttons on them, and they all would have t('Save') in their code. The translation system will only store this string once in the translation database, so that if the translation is updated, all forms using that text will get the updated translation.
Because the source of translation strings is English, and some words in English have multiple meanings or uses, this centralized, shared translation string storage can sometimes lead to ambiguous translations that are not correct for every place the string is used. As an example, the English word "May", in a string by itself, could be part of a list of full month names or part of a list of 3-letter abbreviated month names. So, in languages where the month name for May is longer than 3 letters, you'd need to translate May differently depending on how it's being used. To address this problem, the translation system includes the concept of the "context" of a translated string, which can be used to disambiguate text for translators, and obtain the correct translation for each usage of the string.
Here are some examples of how to provide translation context with strings, so that this information can be included in *.po files, displayed on the localization server for translators, and used to obtain the correct translation in the user interface:
// PHP code
t('May', [], ['context' => 'Long month name']);
\Drupal::translation()->formatPlural($count, '1 something',
  '@count somethings', [], ['context' => 'My context']);
// JavaScript code
Drupal.t('May', {}, {'context': 'Long month name'});
Drupal.formatPlural(count, '1 something', '@count somethings', {},
  {'context': 'My context'});
// *.links.yml file
title: 'May'
title_context: 'Long month name'
// *.routing.yml file
my.route.name:
  pattern: '/something'
  defaults:
    _title: 'May'
    _title_context: 'Long month name'
// Config schema to say that a certain piece of configuration should be
// translatable using the Config Translation API. Note that the schema label
// is also translatable, but it cannot have context.
date_format:
 type: string
 label: 'PHP date format'
 translatable: true
 translation context: 'PHP date format'
// Twig template
{% trans with {'context': 'Long month name'} %}
 May
{% endtrans %}
See also
t()
File
- core/lib/ Drupal/ Core/ Language/ language.api.php, line 10 
- Hooks provided by the base system for language support.
Classes
| Name   | Location | Description | 
|---|---|---|
| DateFormatter | core/ | Provides a service to handle various date related functionality. | 
Interfaces
| Name   | Location | Description | 
|---|---|---|
| LanguageInterface | core/ | Defines an interface for languages. | 
| TranslationInterface | core/ | Interface for the translation.manager translation service. | 
Traits
| Name   | Location | Description | 
|---|---|---|
| StringTranslationTrait | core/ | Wrapper methods for \Drupal\Core\StringTranslation\TranslationInterface. | 
