Skip to content
This repository has been archived by the owner on Jun 26, 2020. It is now read-only.

Provide support for translating plural forms #334

Merged
merged 37 commits into from
Apr 23, 2020
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
37 commits
Select commit Hold shift + click to select a range
962f375
WIP - Adding support for passing object as a first argument for `t()`…
ma2ciek Apr 8, 2020
c14c653
Docs: Improved API docs.
ma2ciek Apr 17, 2020
a577cf6
Docs: Improved API docs.
ma2ciek Apr 17, 2020
05bb63a
Renamed `getFormIndex` to `getPluralForm`.
ma2ciek Apr 17, 2020
bf676ba
Changed: The message id used to determine the translation will be gen…
ma2ciek Apr 17, 2020
020072b
Tests: Added tests for the translation service.
ma2ciek Apr 17, 2020
2740664
Fix: Plural form will be set based on English rules if a function tha…
ma2ciek Apr 17, 2020
53f4639
Tests: Added tests for the locale.js.
ma2ciek Apr 17, 2020
7361eab
Merge branch 'master' into i/6526
ma2ciek Apr 17, 2020
0b01d9b
Internal: Fixed code style issues.
ma2ciek Apr 17, 2020
a36cad2
Docs: Improved `Locale` class API docs.
ma2ciek Apr 19, 2020
f116217
Docs: Improved `Locale` class API docs.
ma2ciek Apr 19, 2020
fc1c318
Docs: Improved `Locale` class and translation service API docs.
ma2ciek Apr 19, 2020
7515252
Fix: Simplified 'magic' code.
ma2ciek Apr 19, 2020
e36132d
Tests: Removed usage of deprecated API.
ma2ciek Apr 19, 2020
ea3ae82
Docs: Improved translation service API docs.
ma2ciek Apr 20, 2020
bf15d06
Other: Simplified the `Locale._t()` function.
ma2ciek Apr 20, 2020
e1c569b
Tests: Added missing test for the message context.
ma2ciek Apr 20, 2020
48d69e1
Docs: Improved translation service API docs.
ma2ciek Apr 20, 2020
30c31af
Docs: Improved `Local#t()` API docs.
ma2ciek Apr 20, 2020
aade538
Docs: Improved `Local#t()` API docs.
ma2ciek Apr 20, 2020
d0ddf47
Docs: Improved `Local#t()` API docs.
ma2ciek Apr 20, 2020
2acf51b
Added support for the shorthand `t` call signature: `t( message, valu…
ma2ciek Apr 21, 2020
7f0096d
Docs: Improved API docs for the `Locale#t()` call with a message and …
ma2ciek Apr 21, 2020
38db678
Docs: Fixed API docs issues.
ma2ciek Apr 21, 2020
db4f3c5
Docs: Fixed API docs issues in `translation-service`.
ma2ciek Apr 21, 2020
7797511
Apply suggestions from code review
ma2ciek Apr 21, 2020
33acb99
Other: Reworded amount to quantity in a few places.
ma2ciek Apr 21, 2020
358d8da
Applied suggestions from code review
ma2ciek Apr 21, 2020
11d5b8c
Other: Removed `message.context`, introduced the `message.id`.
ma2ciek Apr 22, 2020
9083b73
Other: Added support for plural rules that returns a boolean value.
ma2ciek Apr 22, 2020
da86205
Docs: Added a simpler scenario.
ma2ciek Apr 22, 2020
bf71980
Docs: Added a simpler scenario.
ma2ciek Apr 22, 2020
9fe6f52
Docs: Fixed API docs issue in `Locale`.
ma2ciek Apr 22, 2020
14d29a6
Docs: Added missing `module:` part.
ma2ciek Apr 22, 2020
cb5cd06
Applied suggestions from code review
ma2ciek Apr 22, 2020
3d8c423
Docs: Added docs for `translation-service#add() function`.
ma2ciek Apr 22, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
53 changes: 34 additions & 19 deletions src/locale.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@

/* globals console */

import { translate } from './translation-service';
import { _translate } from './translation-service';

const RTL_LANGUAGE_CODES = [ 'ar', 'fa', 'he', 'ku', 'ug' ];

Expand Down Expand Up @@ -76,23 +76,40 @@ export default class Locale {
this.contentLanguageDirection = getLanguageDirection( this.contentLanguage );

/**
* Translates the given string to the {@link #uiLanguage}. This method is also available in
* Translates the given message to the {@link #uiLanguage}. This method is also available in
* {@link module:core/editor/editor~Editor#t} and {@link module:ui/view~View#t}.
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
*
* The strings may contain placeholders (`%<index>`) for values which are passed as the second argument.
* This method's context is statically bound to the `Locale` instance and **always should be called as a function**:
*
* const t = locale.t;
* t( 'Label' );
*
* The message can be either a string or an object implementing the {@link module:translation-service~Message} interface.
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
*
* A Message may contain placeholders (`%<index>`) for values that are passed as the second argument.
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
* `<index>` is the index in the `values` array.
*
* editor.t( 'Created file "%0" in %1ms.', [ fileName, timeTaken ] );
* t( 'Created file "%0" in %1ms.', [ fileName, timeTaken ] );
*
* This method's context is statically bound to Locale instance,
* so it can be called as a function:
* A Message can provide a plural form using the `plural` property and a value - that should be always the first element
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
* of the `values` array based on which the plural form of the target language should be picked. That property value will
* be used as a default plural translation when the translation for the target language will be missing.
*
* const t = this.t;
* t( 'Label' );
* t( { string: 'Add a space', plural: 'Add %0 spaces' }, [ spaces ] );
* t( { string: '%1 a space', plural: '%1 %0 spaces' }, [ spaces, 'Add' ] );
*
* A Message can provide a context using the `context` property when the message string may be not enough unique
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
* across all messages. When the `context` property is set the message id will be constructed in
* the following way: `${ message.string }_${ message.context }`. This context will be also used
* by translators later as a context for the message that should be translated.
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
*
* t( { string: 'image', context: 'Add/Remove image' } );
*
* @method #t
* @param {String} message The string to translate.
* @param {String[]} [values] Values that should be used to interpolate the string.
* @param {String|module:utils/translation-service~Message} message A message that will be localized (translated).
* @param {Array.<String>} [values] An array of values that will fill message placeholders.
* For messages supporting plural forms the first value will determine the plural form.
* @returns {String}
*/
this.t = ( message, values ) => this._t( message, values );
}
Expand Down Expand Up @@ -122,11 +139,12 @@ export default class Locale {
}

/**
* Base for the {@link #t} method.
* An unbound version of the {@link #t} method.
*
* @param {Object|String} message
* @param {String[]} [values]
* @private
* @param {String|module:utils/translation-service~Message} message
* @param {Array.<String>} [values]
* @returns {String}
*/
_t( message, values = [] ) {
if ( typeof message === 'string' ) {
Expand All @@ -136,15 +154,12 @@ export default class Locale {
const hasPluralForm = !!message.plural;
const amount = hasPluralForm ? values[ 0 ] : 1;

let translatedString = translate( this.uiLanguage, message, amount );
const translatedString = _translate( this.uiLanguage, message, amount );

if ( values ) {
translatedString = translatedString.replace( /%(\d+)/g, ( match, index ) => {
return translatedString
.replace( /%(\d+)/g, ( match, index ) => {
return ( index < values.length ) ? values[ index ] : match;
} );
}

return translatedString;
}
}

Expand Down
107 changes: 64 additions & 43 deletions src/translation-service.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
* For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
*/

/* globals window, console */
/* globals window */

/**
* @module utils/translation-service
Expand All @@ -15,41 +15,60 @@ if ( !window.CKEDITOR_TRANSLATIONS ) {
}

/**
* Adds translations to existing ones.
* These translations will later be available for the {@link module:utils/translation-service~translate `translate()`} function.
* Adds translations to existing ones or overrides the existing translations.
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
*
* These translations will later be available for the {@link module:utils/locale~Locale#t `t()`} function.
*
* add( 'pl', {
* 'OK': 'OK',
* 'Cancel': 'Anuluj'
* } );
* 'Cancel': 'Anuluj',
* 'Heading': 'Nagłówek'
* }, n => n == 1 ? 0 : n % 10 >= 2 && n % 10 <= 4 && ( n % 100 < 10 || n % 100 >= 20 ) ? 1 : 2 );
*
* If the message supports plural forms, make sure to provide an array with all plural forms:
* If a message is supposed to support various plural forms, make sure to provide an array with the single form and all plural forms:
*
* add( 'pl', {
* 'Add editor': [ 'Dodaj edytor', 'Dodaj %0 edytory', 'Dodaj %0 edytorów' ]
* 'Add space': [ 'Dodaj spację', 'Dodaj %0 spacje', 'Dodaj %0 spacji' ]
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
* } );
*
* You should also specify the third argument (the `getPluralForm` function) that will be used to determine the plural form if the
* language is not supported by CKEditor 5 yet.
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
*
* add( 'pl', {
* 'Add space': [ 'Dodaj spację', 'Dodaj %0 spacje', 'Dodaj %0 spacji' ]
* }, n => n == 1 ? 0 : n % 10 >= 2 && n % 10 <= 4 && ( n % 100 < 10 || n % 100 >= 20 ) ? 1 : 2 );
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
*
* If you cannot import this function from this module (e.g. because you use a CKEditor 5 build), then you can
* still add translations by extending the global `window.CKEDITOR_TRANSLATIONS` object by using a function like
* the one below:
*
* function addTranslations( language, translations ) {
* function addTranslations( language, translations, getPluralForm ) {
* if ( !window.CKEDITOR_TRANSLATIONS ) {
* window.CKEDITOR_TRANSLATIONS = {};
* }

* if ( !window.CKEDITOR_TRANSLATIONS[ language ] ) {
* window.CKEDITOR_TRANSLATIONS[ language ] = {};
* }
*
* const dictionary = window.CKEDITOR_TRANSLATIONS[ language ] || ( window.CKEDITOR_TRANSLATIONS[ language ] = {} );
* const languageTranslations = window.CKEDITOR_TRANSLATIONS[ language ];
*
* languageTranslations.dictionary = languageTranslations.dictionary || {};
* languageTranslations.getPluralForm = getPluralForm || languageTranslations.getPluralForm;
*
* // Extend the dictionary for the given language.
* Object.assign( dictionary, translations );
* Object.assign( languageTranslations.dictionary, translations );
* }
*
* @param {String} language Target language.
* @param {Object.<String, String>} translations Translations which will be added to the dictionary.
* @param {Function} getPluralForm A function that returns the plural form index.
* @param {Object.<String, String|Array.<String>>} translations Translations which will be added to the dictionary.
* @param {Function} getPluralForm A function that returns the plural form index (a number).
*/
export function add( language, translations, getPluralForm ) {
const languageTranslations = window.CKEDITOR_TRANSLATIONS[ language ] || ( window.CKEDITOR_TRANSLATIONS[ language ] = {} );
if ( !window.CKEDITOR_TRANSLATIONS[ language ] ) {
window.CKEDITOR_TRANSLATIONS[ language ] = {};
}

const languageTranslations = window.CKEDITOR_TRANSLATIONS[ language ];

languageTranslations.dictionary = languageTranslations.dictionary || {};
languageTranslations.getPluralForm = getPluralForm || languageTranslations.getPluralForm;
Expand All @@ -58,37 +77,36 @@ export function add( language, translations, getPluralForm ) {
}

/**
* Translates string if the translation of the string was previously added to the dictionary.
* See {@link module:utils/translation-service Translation Service}.
* This happens in a multi-language mode were translation modules are created by the bundler.
* **Note:** this method is internal, use {@link module:utils/local~Locale#t the `t()` function} instead to translate
* editor UI parts.
*
* When no translation is defined in the dictionary or the dictionary doesn't exist this function returns
* the original string.
* This function is responsible for translating messages to the specified language. It uses perviously added translations
* by {@link module:utils/translation-service~add} (a translations dictionary and and the `getPluralForm` function
* to provide accurate translations of plural forms).
*
* In a single-language mode (when values passed to `t()` were replaced with target language strings) the dictionary
* is left empty, so this function will return the original strings always.
* When no translation is defined in the dictionary or the dictionary doesn't exist this function returns
* the original message string or message plural depending on the number of elements.
*
* translate( 'pl', { string: 'Cancel' } );
* translate( 'pl', { string: 'Cancel' } ); // 'Cancel'
*
* The third optional argument is the number of elements, based on which the single form or one of plural forms
* should be picked when the message supports plural forms.
* should be picked when the message is supposed to support various plural forms.
*
* translate( 'en', { string: 'Add a space', plural: 'Add %0 spaces' }, 1 ); // 'Add a space'
* translate( 'en', { string: 'Add a space', plural: 'Add %0 spaces' }, 3 ); // 'Add %0 spaces'
*
* translate( 'en', { string: 'Add a space', plural: 'Add %0 spaces }, 1 ); // 'Add %0 space'
* translate( 'en', { string: 'Add a space', plural: 'Add %0 spaces }, 3 ); // 'Add %0 spaces'
* A Message can provide a context using the `context` property when the message string may be not enough unique
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
* across all messages. When the `context` property is set the message id will be constructed in
* the following way: `${ message.string }_${ message.context }`. This context will be also used
* by translators later as a context for the message that should be translated.
*
* @protected
* @param {String} language Target language.
* @param {module:utils/translation-service~Message|string} message A message that will be translated.
* @param {Number} [amount] A number of elements for which a plural form should be picked from the target language dictionary.
* @returns {String} Translated sentence.
*/
export function translate( language, message, amount = 1 ) {
if ( typeof message === 'string' ) {
// TODO
console.warn( 'deprecated usage' );

message = { string: message };
}

export function _translate( language, message, amount = 1 ) {
const numberOfLanguages = getNumberOfLanguages();

if ( numberOfLanguages === 1 ) {
Expand All @@ -97,13 +115,14 @@ export function translate( language, message, amount = 1 ) {
language = Object.keys( window.CKEDITOR_TRANSLATIONS )[ 0 ];
}

// Use message context to enhance the message id when passed.
const messageId = message.context ?
message.string + '_' + message.context :
message.string;

if ( numberOfLanguages === 0 || !hasTranslation( language, messageId ) ) {
// return english forms:
if ( amount !== 1 ) {
// Return the default plural form that was passed in the `message.plural` parameter.
return message.plural;
}

Expand All @@ -113,14 +132,14 @@ export function translate( language, message, amount = 1 ) {
const dictionary = window.CKEDITOR_TRANSLATIONS[ language ].dictionary;
const getPluralForm = window.CKEDITOR_TRANSLATIONS[ language ].getPluralForm || ( n => n === 1 ? 0 : 1 );

// TODO - maybe a warning could be helpful for some mismatches.

if ( typeof dictionary[ messageId ] === 'string' ) {
return dictionary[ messageId ];
}

const pluralFormIndex = getPluralForm( amount );

// Note: The `translate` function is not responsible for replacing `%0, %1, ...` with values.
return dictionary[ messageId ][ getPluralForm( amount ) ];
return dictionary[ messageId ][ pluralFormIndex ];
}

/**
Expand All @@ -145,14 +164,16 @@ function getNumberOfLanguages() {
}

/**
* The internalization message interface. A translation for the given language can be found.
*
* TODO
* The internationalization message interface. A message that implements this interface can be passed to the `t()` function
* to be translated to the target ui language.
*
* @typedef {Object} Message
*
* @property {String} string The message string. It becomes the message id when no context is provided.
* @property {String} string The message string. It becomes the message id when no context is provided. When the message is supposed
ma2ciek marked this conversation as resolved.
Show resolved Hide resolved
* to support plural forms then the string should be the English singular form of the message.
* @property {String} [context] The message context. If passed then the message id is constructed form both,
* the message string and the message string in the following format: `<messageString>_<messageContext>`.
* @property {String} [plural] The plural form of the message.
* the message string and the message string in the following format: `<messageString>_<messageContext>`. This property is useful when
* various messages can share the same message string, when omitting a context would result in a broken translation.
* @property {String} [plural] The plural form of the message. This property should be skipped when a message is not supposed
* to support plural forms. Otherwise it should always be set to a string with the English plural form of the message.
*/
28 changes: 27 additions & 1 deletion tests/locale.js
Original file line number Diff line number Diff line change
Expand Up @@ -136,12 +136,22 @@ describe( 'Locale', () => {
} );
} );

it( 'should translate a string to the target ui language', () => {
it( 'should translate a message to the target ui language', () => {
const t = locale.t;

expect( t( 'foo' ) ).to.equal( 'foo_pl' );
} );

it( 'should translate a message including the message string and the message context', () => {
const t = locale.t;

addTranslations( 'pl', {
'foo_bar': 'foo_bar_pl'
} );

expect( t( { string: 'foo', context: 'bar' } ) ).to.equal( 'foo_bar_pl' );
} );

it( 'should translate a plural form to the target ui language based on the first value and interpolate the string', () => {
const t = locale.t;

Expand All @@ -150,6 +160,22 @@ describe( 'Locale', () => {
expect( t( { string: 'bar', plural: '%0 bars' }, [ 5 ] ), 3 ).to.equal( '5 bar_pl_2' );
} );

it( 'should translate a message supporting plural form with a context', () => {
const t = locale.t;

addTranslations( 'pl', {
'%1 a space_Add/Remove a space': [ '%1 spację', '%1 %0 spacje', '%1 %0 spacji' ],
'Add': 'Dodaj',
'Remove': 'Usuń'
} );

const addOrRemoveSpaceMessage = { string: '%1 a space', plural: '%1 %0 spaces', context: 'Add/Remove a space' };

expect( t( addOrRemoveSpaceMessage, [ 1, t( 'Add' ) ] ), 1 ).to.equal( 'Dodaj spację' );
expect( t( addOrRemoveSpaceMessage, [ 2, t( 'Remove' ) ] ), 2 ).to.equal( 'Usuń 2 spacje' );
expect( t( addOrRemoveSpaceMessage, [ 5, t( 'Add' ) ] ), 3 ).to.equal( 'Dodaj 5 spacji' );
} );

it( 'should interpolate a message with provided values', () => {
const t = locale.t;

Expand Down
Loading