This behavior allow you to create multilingual models and to use them (almost) like normal models. For each model, translations have to be stored in a separate table of the database (ex: PostLocalized or ProductLocalized), which allow you to easily add or remove a language without modifying your database.
First example: by default translations of current language are inserted in the model as normal attributes.
// Assuming current language is english (in protected/config/main.php: 'sourceLanguage' => 'en')
$model = Post::model()->findByPk((int) $id);
echo $model->title; // echo "English title"
// Now let's imagine current language is french (in
protected/config/main.php: 'sourceLanguage' => 'fr')
$model = Post::model()->findByPk((int) $id);
echo $model->title; // echo "Titre en Français"
$model = Post::model()->localized('en')->findByPk((int) $id);
echo $model->title; // echo "English title"
// Here current language is still french
Second example: if you use multilang() in a "find" query, every translation of the model are loaded as virtual attributes (title_en, title_fr, title_de, ...).
$model = Post::model()->multilang()->findByPk((int) $id);
echo $model->title_en; // echo "English title"
echo $model->title_fr; // echo "Titre en Français"
CREATE TABLE IF NOT EXISTS `post_localized` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`post_id` int(11) NOT NULL,
`language` varchar(6) NOT NULL,
`localized_title` varchar(255) NOT NULL,
`localized_content` TEXT NOT NULL,
PRIMARY KEY (`id`),
KEY `post_id` (`post_id`),
KEY `language` (`language`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
ALTER TABLE `post_localized`
ADD CONSTRAINT `post_localized_ibfk_1` FOREIGN KEY (`post_id`) REFERENCES
`post` (`id`) ON
DELETE CASCADE ON UPDATE CASCADE;
Attach this behavior to the model (Post in the example). Everything that is commented is with default values.
public function behaviors() {
return [
'ml' => [
'class' => 'application.models.behaviors.MultilingualBehavior',
// 'localizedModelName' => 'PostLocalized',
// 'localizedTableName' => 'postLocalized',
// 'localizedForeignKey' => 'post_id',
// 'languageField' => 'language',
// attributes of the model to be translated
'localizedAttributes' => ['title', 'content'],
// 'localizedPrefix' => 'localized_',
// array of your translated languages.
// Example: ['fr' => 'Français', 'en' => 'English']
'languages' => Yii::app()->params['translatedLanguages'],
// your main language. Example: 'fr'
'defaultLanguage' => Yii::app()->params['defaultLanguage'],
// 'createScenarios' => ['insert'],
// 'localizedRelationName' => 'localizedPost',
// 'internationalizedRelationName' => 'internationalizedPost',
// 'forceOverwrite' => false,
// 'forceDelete' => true,
// Set to true if you don't want to create a 'PostLocalized.php' in your models folder
// 'dynamicLocalizedModel' => true,
],
];
}
Look into the behavior source code and read the comments of each attribute to understand how to configure them and how to use the behavior.
In order to retrieve translated models by default, add this function in the model class:
public function loadModel($id, $ml = false) {
if ($ml) {
$model = Post::model()->multilang()->findByPk((int) $id);
} else {
$model = Post::model()->findByPk((int) $id);
}
if ($model === null) {
throw new CHttpException(404, 'The requested post does not exist.');
}
return $model;
}
and use it like this in the update action:
public function actionUpdate($id) {
$model = $this->loadModel($id, true);
...
}
Here is a very simple example for the form view:
<?php foreach (Yii::app()->params['translatedLanguages'] as $l => $lang):
if ($l === Yii::app()->params['defaultLanguage']) $suffix = '';
else $suffix = '_'.$l;
?>
<fieldset>
<legend><?= $lang ?></legend>
<div class="row">
<?= $form->labelEx($model,'title') ?>
<?= $form->textField($model, 'title'.$suffix, ['size' => 60, 'maxlength' => 255]) ?>
<?= $form->error($model, 'title'.$suffix) ?>
</div>
<div class="row">
<?= $form->labelEx($model, 'content') ?>
<?= $form->textArea($model, 'content'.$suffix) ?>
<?= $form->error($model, 'content'.$suffix) ?>
</div>
</fieldset>
<?php endforeach ?>
To enable search on translated fields, you can modify the search() function in the model like this:
public function search()
{
$criteria = new CDbCriteria;
// ...
// here your criteria definition
// ...
return new CActiveDataProvider($this, [
'criteria' => $this->ml->modifySearchCriteria($criteria),
// instead of
// 'criteria' => $criteria,
));
}
Warning: the modification of the search criteria is based on a simple str_replace so it may not work properly under certain circumstances.
It's also possible to retrieve languages translation of two or more related models in one query. Example for a Page model with a "articles" HAS_MANY relation:
$model = Page::model()->multilang()->with('articles', 'articles.internationalizedArticle')->findByPk((int) $id);
echo $model->articles[0]->content_en;
With this method it's possible to make multi model forms like it's explained here
24/03/2012: First release
28/03/2012: It's now possible to modify language when retrieving data with the localized relation.
Example:
$model = Post::model()->localized('en')->findByPk((int) $id);
30/03/2012: Modification of the rules definition for translated attributes:
- if you set forceOverwrite to true, every rules defined in the model for the attributes to translate will be applied to the translations.
- if you set forceOverwrite to false (default), every rules defined in the model for the attributes to translate will be applied to the translations except "required" rules that will only be applied to the default translation.
28/06/2012: (Bug Fix) two2wyes found and fixed a bug that prevented translations to be correctly saved on attributes that only have a "required" rule and with the "forceOverwrite" option set to false. Thanks again to him. See the thread
Many thanks to guillemc who made the biggest part of the work on this behavior (see original thread).