Commit 14cd7eed2849cf5ca2b52e8caaaf957f9ea0d0e3
1 parent
96410438
LanguageBehavior optimized
Showing
4 changed files
with
112 additions
and
88 deletions
Show diff stats
backend/controllers/ArticlesController.php
| ... | ... | @@ -82,10 +82,9 @@ |
| 82 | 82 | public function actionCreate() |
| 83 | 83 | { |
| 84 | 84 | $model = new Articles(); |
| 85 | - $model_langs = $model->generateLangs(); | |
| 85 | + $model->generateLangs(); | |
| 86 | 86 | if($model->load(Yii::$app->request->post())) { |
| 87 | - $model->loadLangs(\Yii::$app->request, $model_langs); | |
| 88 | - $model->model_langs = $model_langs; | |
| 87 | + $model->loadLangs(\Yii::$app->request); | |
| 89 | 88 | if($model->save() && $model->transactionStatus) { |
| 90 | 89 | return $this->redirect([ |
| 91 | 90 | 'view', |
| ... | ... | @@ -95,7 +94,7 @@ |
| 95 | 94 | } |
| 96 | 95 | return $this->render('create', [ |
| 97 | 96 | 'model' => $model, |
| 98 | - 'model_langs' => $model_langs, | |
| 97 | + 'model_langs' => $model->model_langs, | |
| 99 | 98 | ]); |
| 100 | 99 | } |
| 101 | 100 | |
| ... | ... | @@ -110,10 +109,9 @@ |
| 110 | 109 | public function actionUpdate($id) |
| 111 | 110 | { |
| 112 | 111 | $model = $this->findModel($id); |
| 113 | - $model_langs = $model->generateLangs(); | |
| 112 | + $model->generateLangs(); | |
| 114 | 113 | if($model->load(Yii::$app->request->post())) { |
| 115 | - $model->loadLangs(\Yii::$app->request, $model_langs); | |
| 116 | - $model->model_langs = $model_langs; | |
| 114 | + $model->loadLangs(\Yii::$app->request); | |
| 117 | 115 | if($model->save() && $model->transactionStatus) { |
| 118 | 116 | return $this->redirect([ |
| 119 | 117 | 'view', |
| ... | ... | @@ -123,7 +121,7 @@ |
| 123 | 121 | } |
| 124 | 122 | return $this->render('update', [ |
| 125 | 123 | 'model' => $model, |
| 126 | - 'model_langs' => $model_langs, | |
| 124 | + 'model_langs' => $model->model_langs, | |
| 127 | 125 | ]); |
| 128 | 126 | } |
| 129 | 127 | ... | ... |
common/models/Articles.php
| 1 | 1 | <?php |
| 2 | 2 | |
| 3 | 3 | namespace common\models; |
| 4 | - | |
| 4 | + | |
| 5 | 5 | use common\modules\language\behaviors\TransactionBehavior; |
| 6 | 6 | use common\modules\language\behaviors\LanguageBehavior; |
| 7 | 7 | use common\behaviors\SaveImgBehavior; |
| ... | ... | @@ -22,26 +22,23 @@ |
| 22 | 22 | * @property ArticlesLang $object_lang |
| 23 | 23 | * @property string $ownerKey |
| 24 | 24 | * @property string $langKey |
| 25 | + * @property ArticlesLang[] $model_langs | |
| 26 | + * @property bool $transactionStatus | |
| 25 | 27 | * @method string getOwnerKey() |
| 26 | - * @method void setOwnerKey(string $value) | |
| 28 | + * @method void setOwnerKey( string $value ) | |
| 27 | 29 | * @method string getLangKey() |
| 28 | - * @method void setLangKey(string $value) | |
| 30 | + * @method void setLangKey( string $value ) | |
| 29 | 31 | * @method ActiveQuery getLangs() |
| 30 | 32 | * @method ActiveQuery getLang( integer $language_id ) |
| 31 | 33 | * @method ArticlesLang[] generateLangs() |
| 32 | - * @method void loadLangs(Request $request, ActiveRecord[] $model_langs) | |
| 33 | - * @method bool linkLangs(ActiveRecord[] $model_langs) | |
| 34 | - * @method bool saveLangs(ActiveRecord[] $model_langs) | |
| 35 | - * * End language behavior * | |
| 36 | - * * From transaction behavior * | |
| 37 | - * @property ArticlesLang[] $model_langs | |
| 38 | - * @property bool $transactionStatus | |
| 34 | + * @method void loadLangs( Request $request ) | |
| 35 | + * @method bool linkLangs() | |
| 36 | + * @method bool saveLangs() | |
| 39 | 37 | * @method bool getTransactionStatus() |
| 40 | - * * End transaction behavior * | |
| 38 | + * * End language behavior * | |
| 41 | 39 | */ |
| 42 | 40 | class Articles extends \yii\db\ActiveRecord |
| 43 | 41 | { |
| 44 | - public $imageUpload; | |
| 45 | 42 | |
| 46 | 43 | /** |
| 47 | 44 | * @inheritdoc |
| ... | ... | @@ -58,20 +55,17 @@ |
| 58 | 55 | { |
| 59 | 56 | return [ |
| 60 | 57 | [ |
| 61 | - 'class' => SaveImgBehavior::className(), | |
| 58 | + 'class' => SaveImgBehavior::className(), | |
| 62 | 59 | 'fields' => [ |
| 63 | 60 | [ |
| 64 | - 'name' => 'image', | |
| 61 | + 'name' => 'image', | |
| 65 | 62 | 'directory' => 'articles', |
| 66 | - ] | |
| 63 | + ], | |
| 67 | 64 | ], |
| 68 | 65 | ], |
| 69 | 66 | 'language' => [ |
| 70 | 67 | 'class' => LanguageBehavior::className(), |
| 71 | 68 | ], |
| 72 | - 'transaction' => [ | |
| 73 | - 'class' => TransactionBehavior::className(), | |
| 74 | - ], | |
| 75 | 69 | ]; |
| 76 | 70 | } |
| 77 | 71 | |
| ... | ... | @@ -93,22 +87,6 @@ |
| 93 | 87 | 'safe', |
| 94 | 88 | ], |
| 95 | 89 | [ |
| 96 | - [ | |
| 97 | - 'image', | |
| 98 | - ], | |
| 99 | - 'string', | |
| 100 | - 'max' => 255, | |
| 101 | - ], | |
| 102 | - [ | |
| 103 | - [ 'imageUpload' ], | |
| 104 | - 'safe', | |
| 105 | - ], | |
| 106 | - [ | |
| 107 | - [ 'imageUpload' ], | |
| 108 | - 'file', | |
| 109 | - 'extensions' => 'jpg, gif, png', | |
| 110 | - ], | |
| 111 | - [ | |
| 112 | 90 | [ 'date' ], |
| 113 | 91 | 'filter', |
| 114 | 92 | 'filter' => function($value) { |
| ... | ... | @@ -124,23 +102,13 @@ |
| 124 | 102 | public function attributeLabels() |
| 125 | 103 | { |
| 126 | 104 | return [ |
| 127 | - 'id' => Yii::t('app', 'ID'), | |
| 128 | - 'date' => Yii::t('app', 'Date'), | |
| 129 | - 'image' => Yii::t('app', 'Image'), | |
| 130 | - 'imageUrl' => Yii::t('app', 'Image'), | |
| 105 | + 'id' => Yii::t('app', 'ID'), | |
| 106 | + 'date' => Yii::t('app', 'Date'), | |
| 107 | + 'image' => Yii::t('app', 'Image'), | |
| 108 | + 'imageUrl' => Yii::t('app', 'Image'), | |
| 131 | 109 | ]; |
| 132 | 110 | } |
| 133 | 111 | |
| 134 | - public function getImageFile() | |
| 135 | - { | |
| 136 | - return empty( $this->image ) ? NULL : Yii::getAlias('@imagesDir/articles/' . $this->image); | |
| 137 | - } | |
| 138 | - | |
| 139 | - public function getImageUrl() | |
| 140 | - { | |
| 141 | - return empty( $this->image ) ? NULL : Yii::getAlias('@imagesUrl/articles/' . $this->image); | |
| 142 | - } | |
| 143 | - | |
| 144 | 112 | public function recalculateRating() |
| 145 | 113 | { |
| 146 | 114 | /** | ... | ... |
common/modules/language/behaviors/LanguageBehavior.php
| ... | ... | @@ -6,15 +6,16 @@ |
| 6 | 6 | use yii\base\InvalidConfigException; |
| 7 | 7 | use yii\db\ActiveQuery; |
| 8 | 8 | use yii\db\ActiveRecord; |
| 9 | + use yii\db\Transaction; | |
| 9 | 10 | use yii\web\Request; |
| 10 | 11 | |
| 11 | 12 | /** |
| 12 | 13 | * Class LanguageBehavior |
| 13 | - * @property ActiveRecord $owner | |
| 14 | - * @property string $ownerKey | |
| 15 | - * @property string $langKey | |
| 14 | + * @property ActiveRecord $owner | |
| 15 | + * @property string $ownerKey | |
| 16 | + * @property string $langKey | |
| 16 | 17 | * @property ActiveRecord[] $langs |
| 17 | - * @property ActiveRecord $lang | |
| 18 | + * @property ActiveRecord $lang | |
| 18 | 19 | */ |
| 19 | 20 | class LanguageBehavior extends Behavior |
| 20 | 21 | { |
| ... | ... | @@ -24,11 +25,36 @@ |
| 24 | 25 | */ |
| 25 | 26 | public $object_lang; |
| 26 | 27 | |
| 28 | + /** | |
| 29 | + * @var ActiveRecord[] $model_langs | |
| 30 | + */ | |
| 31 | + public $model_langs = []; | |
| 32 | + | |
| 27 | 33 | private $_owner_key; |
| 28 | 34 | |
| 29 | 35 | private $_lang_key; |
| 30 | 36 | |
| 31 | 37 | /** |
| 38 | + * @var Transaction $_transaction | |
| 39 | + */ | |
| 40 | + private $_transaction; | |
| 41 | + | |
| 42 | + /** | |
| 43 | + * @var bool $_transaction_status | |
| 44 | + */ | |
| 45 | + private $_transaction_status = false; | |
| 46 | + | |
| 47 | + public function events() | |
| 48 | + { | |
| 49 | + return [ | |
| 50 | + ActiveRecord::EVENT_BEFORE_INSERT => 'beforeSave', | |
| 51 | + ActiveRecord::EVENT_BEFORE_UPDATE => 'beforeSave', | |
| 52 | + ActiveRecord::EVENT_AFTER_INSERT => 'afterSave', | |
| 53 | + ActiveRecord::EVENT_AFTER_UPDATE => 'afterSave', | |
| 54 | + ]; | |
| 55 | + } | |
| 56 | + | |
| 57 | + /** | |
| 32 | 58 | * Get $owner primary key to link language model |
| 33 | 59 | * @return string |
| 34 | 60 | */ |
| ... | ... | @@ -129,19 +155,20 @@ |
| 129 | 155 | $table_name = $object_lang::getTableSchema()->name; |
| 130 | 156 | $owner = $this->owner; |
| 131 | 157 | return $owner->hasOne($object_lang::className(), [ $this->getLangKey() => $this->getOwnerKey() ]) |
| 132 | - ->where([ $table_name.'.language_id' => $language_id ]); | |
| 158 | + ->where([ $table_name . '.language_id' => $language_id ]); | |
| 133 | 159 | } |
| 134 | 160 | |
| 135 | 161 | /** |
| 136 | 162 | * Generate language models for $owner for active languages. If $owner not new and language |
| 137 | 163 | * models already inserted, models will be filled with them. |
| 138 | - * @return ActiveRecord[] | |
| 164 | + * @return void | |
| 139 | 165 | */ |
| 140 | 166 | public function generateLangs() |
| 141 | 167 | { |
| 142 | 168 | $owner = $this->owner; |
| 143 | 169 | $languages = Language::find() |
| 144 | 170 | ->where([ 'status' => true ]) |
| 171 | + ->orderBy([ 'language_id' => SORT_ASC ]) | |
| 145 | 172 | ->asArray() |
| 146 | 173 | ->column(); |
| 147 | 174 | $object_lang = $this->object_lang; |
| ... | ... | @@ -150,6 +177,7 @@ |
| 150 | 177 | if(!$owner->isNewRecord) { |
| 151 | 178 | $langs = $this->getLangs() |
| 152 | 179 | ->andFilterWhere([ 'language_id' => $languages ]) |
| 180 | + ->orderBy([ 'language_id' => SORT_ASC ]) | |
| 153 | 181 | ->all(); |
| 154 | 182 | } |
| 155 | 183 | foreach($languages as $language) { |
| ... | ... | @@ -161,21 +189,20 @@ |
| 161 | 189 | ]); |
| 162 | 190 | } |
| 163 | 191 | } |
| 164 | - return $langs; | |
| 192 | + $this->model_langs = $langs; | |
| 165 | 193 | } |
| 166 | 194 | |
| 167 | 195 | /** |
| 168 | 196 | * Load language models with post data. |
| 169 | 197 | * |
| 170 | - * @param Request $request | |
| 171 | - * @param ActiveRecord[] $model_langs | |
| 198 | + * @param Request $request | |
| 172 | 199 | */ |
| 173 | - public function loadLangs(Request $request, array $model_langs) | |
| 200 | + public function loadLangs(Request $request) | |
| 174 | 201 | { |
| 175 | 202 | foreach($request->post($this->object_lang->formName(), []) as $lang => $value) { |
| 176 | - if(!empty( $model_langs[ $lang ] )) { | |
| 177 | - $model_langs[ $lang ]->attributes = $value; | |
| 178 | - $model_langs[ $lang ]->language_id = $lang; | |
| 203 | + if(!empty( $this->model_langs[ $lang ] )) { | |
| 204 | + $this->model_langs[ $lang ]->attributes = $value; | |
| 205 | + $this->model_langs[ $lang ]->language_id = $lang; | |
| 179 | 206 | } |
| 180 | 207 | } |
| 181 | 208 | } |
| ... | ... | @@ -183,12 +210,9 @@ |
| 183 | 210 | /** |
| 184 | 211 | * Link language models with $owner by setting language model language key to owner key of |
| 185 | 212 | * owner |
| 186 | - * | |
| 187 | - * @param ActiveRecord[] $model_langs | |
| 188 | - * | |
| 189 | 213 | * @return bool If $owner is new record then return false else true |
| 190 | 214 | */ |
| 191 | - public function linkLangs(array $model_langs) | |
| 215 | + public function linkLangs() | |
| 192 | 216 | { |
| 193 | 217 | $owner = $this->owner; |
| 194 | 218 | if($owner->isNewRecord) { |
| ... | ... | @@ -196,6 +220,7 @@ |
| 196 | 220 | } |
| 197 | 221 | $lang_key = $this->getLangKey(); |
| 198 | 222 | $owner_key = $this->getOwnerKey(); |
| 223 | + $model_langs = $this->model_langs; | |
| 199 | 224 | foreach($model_langs as $model_lang) { |
| 200 | 225 | $model_lang->$lang_key = $owner->$owner_key; |
| 201 | 226 | } |
| ... | ... | @@ -204,14 +229,12 @@ |
| 204 | 229 | |
| 205 | 230 | /** |
| 206 | 231 | * Try to save all language models to the db. Validation function is run for all models. |
| 207 | - * | |
| 208 | - * @param ActiveRecord[] $model_langs | |
| 209 | - * | |
| 210 | 232 | * @return bool Whether all models are valid |
| 211 | 233 | */ |
| 212 | - public function saveLangs(array $model_langs) | |
| 234 | + public function saveLangs() | |
| 213 | 235 | { |
| 214 | 236 | $success = true; |
| 237 | + $model_langs = $this->model_langs; | |
| 215 | 238 | foreach($model_langs as $model_lang) { |
| 216 | 239 | if(!$model_lang->save()) { |
| 217 | 240 | $success = false; |
| ... | ... | @@ -219,4 +242,42 @@ |
| 219 | 242 | } |
| 220 | 243 | return $success; |
| 221 | 244 | } |
| 245 | + | |
| 246 | + public function beforeSave($event) | |
| 247 | + { | |
| 248 | + /** | |
| 249 | + * @var ActiveRecord $owner | |
| 250 | + */ | |
| 251 | + $owner = $this->owner; | |
| 252 | + $db = $owner::getDb(); | |
| 253 | + $this->_transaction = $db->beginTransaction(); | |
| 254 | + } | |
| 255 | + | |
| 256 | + public function afterSave($event) | |
| 257 | + { | |
| 258 | + /** | |
| 259 | + * @var ActiveRecord $owner | |
| 260 | + */ | |
| 261 | + $owner = $this->owner; | |
| 262 | + if(!empty( $this->model_langs )) { | |
| 263 | + if($this->linkLangs() && $this->saveLangs()) { | |
| 264 | + $this->_transaction->commit(); | |
| 265 | + $this->_transaction_status = true; | |
| 266 | + } else { | |
| 267 | + $this->_transaction->rollBack(); | |
| 268 | + $this->_transaction_status = false; | |
| 269 | + } | |
| 270 | + } else { | |
| 271 | + $this->_transaction->commit(); | |
| 272 | + $this->_transaction_status = true; | |
| 273 | + } | |
| 274 | + } | |
| 275 | + | |
| 276 | + /** | |
| 277 | + * @return bool | |
| 278 | + */ | |
| 279 | + public function getTransactionStatus():bool | |
| 280 | + { | |
| 281 | + return $this->_transaction_status; | |
| 282 | + } | |
| 222 | 283 | } |
| 223 | 284 | \ No newline at end of file | ... | ... |
common/modules/language/readme.txt
| ... | ... | @@ -57,6 +57,8 @@ public function behaviors() { |
| 57 | 57 | * @property {TableLang} $object_lang |
| 58 | 58 | * @property string $ownerKey |
| 59 | 59 | * @property string $langKey |
| 60 | + * @property {TableLang}[] $model_langs | |
| 61 | + * @property bool $transactionStatus | |
| 60 | 62 | * @method string getOwnerKey() |
| 61 | 63 | * @method void setOwnerKey(string $value) |
| 62 | 64 | * @method string getLangKey() |
| ... | ... | @@ -67,20 +69,16 @@ public function behaviors() { |
| 67 | 69 | * @method void loadLangs(Request $request, ActiveRecord[] $model_langs) |
| 68 | 70 | * @method bool linkLangs(ActiveRecord[] $model_langs) |
| 69 | 71 | * @method bool saveLangs(ActiveRecord[] $model_langs) |
| 70 | - * * End language behavior * | |
| 71 | - * * From transaction behavior * | |
| 72 | - * @property {TableLang}[] $model_langs | |
| 73 | - * @property bool $transactionStatus | |
| 74 | 72 | * @method bool getTransactionStatus() |
| 75 | - * * End transaction behavior * | |
| 73 | + * * End language behavior * | |
| 76 | 74 | 3.2. Убрать language behavior с наследуемых таблиц от {Table} ({TableSearch}...) |
| 77 | 75 | 4. Доступные полезные методы: |
| 78 | 76 | {Table}->getLangs() - получить все текущие {TableLang} для {Table} проиндексированные по language_id |
| 79 | 77 | {Table}->getLang($language_id = NULL) - получить {TableLang} для определенного языка (default: текущий язык) для {Table} |
| 80 | 78 | {Table}->generateLangs() - получить массив {TableLang} под каждый язык, включая существующие записи, для {Table} |
| 81 | - {Table}->loadLangs($request, $model_langs) - заполнить массив {TableLang} данными с POST | |
| 82 | - {Table}->linkLangs($model_langs) - связать каждый элемент массива {TableLang} с текущей {Table} | |
| 83 | - {Table}->saveLangs($model_langs) - провалидировать и сохранить каждый элемент массива {TableLang} | |
| 79 | + {Table}->loadLangs($request) - заполнить массив {TableLang} данными с POST | |
| 80 | + {Table}->linkLangs() - связать каждый элемент массива {TableLang} с текущей {Table} | |
| 81 | + {Table}->saveLangs() - провалидировать и сохранить каждый элемент массива {TableLang} | |
| 84 | 82 | 5. Добавить поля в форму (к примеру через Bootstrap Tabs). |
| 85 | 83 | В наличии: |
| 86 | 84 | LanguageForm::widget([ |
| ... | ... | @@ -90,7 +88,6 @@ public function behaviors() { |
| 90 | 88 | ]); |
| 91 | 89 | 6. Обрабатывать данные в контроллере. |
| 92 | 90 | 1. После создания/поиска {Table} создаем/находим языковые модели {Table}->generateLangs() |
| 93 | - 2. При POST запросе загружаем данные в языковые модели {Table}->loadLangs(Request $request, {TableLangs[]} $model_langs) | |
| 94 | - 3. Передаем в свойство model_langs модели {Table} массив {TableLang[]}: $model->model_langs = $model_langs | |
| 95 | - 4. После сохранения, если транзанкция успешна, то свойство {Table}->transactionStatus будет true, иначе возникла ошибка в какой то модели. | |
| 91 | + 2. При POST запросе загружаем данные в языковые модели {Table}->loadLangs(Request $request) | |
| 92 | + 3. После сохранения, если транзанкция успешна, то свойство {Table}->transactionStatus будет true, иначе возникла ошибка в какой то модели. | |
| 96 | 93 | 7. Получать данные на публичной части сайта через {Table}->lang. | ... | ... |