@@ -243,6 +243,7 @@ protected function _traverseClause(Query $query, $name = '', $config = [])
243
243
if (in_array ($ field , $ fields )) {
244
244
$ joinRequired = true ;
245
245
$ expression ->setField ("$ alias. $ field " );
246
+
246
247
return ;
247
248
}
248
249
@@ -269,20 +270,52 @@ public function beforeSave(Event $event, EntityInterface $entity, ArrayObject $o
269
270
$ newOptions = [$ this ->_translationTable ->alias () => ['validate ' => false ]];
270
271
$ options ['associated ' ] = $ newOptions + $ options ['associated ' ];
271
272
273
+ // Check early if empty translations are present in the entity.
274
+ // If this is the case, unset them to prevent persistence.
275
+ // This only applies if $this->_config['allowEmptyTranslations'] is false
276
+ if ($ this ->_config ['allowEmptyTranslations ' ] === false ) {
277
+ $ this ->_unsetEmptyFields ($ entity );
278
+ }
279
+
272
280
$ this ->_bundleTranslatedFields ($ entity );
273
281
$ bundled = $ entity ->get ('_i18n ' ) ?: [];
282
+ $ noBundled = count ($ bundled ) === 0 ;
274
283
275
- if ($ locale === $ this ->config ('defaultLocale ' )) {
284
+ // No additional translation records need to be saved,
285
+ // as the entity is in the default locale.
286
+ if ($ noBundled && $ locale === $ this ->config ('defaultLocale ' )) {
276
287
return ;
277
288
}
289
+
278
290
$ values = $ entity ->extract ($ this ->_translationFields (), true );
279
291
$ fields = array_keys ($ values );
292
+ $ noFields = empty ($ fields );
280
293
281
- if (empty ($ fields )) {
294
+ // If there are no fields and no bundled translations, or both fields
295
+ // in the default locale and bundled translations we can
296
+ // skip the remaining logic as its not necessary.
297
+ if ($ noFields && $ noBundled || ($ fields && $ bundled )) {
282
298
return ;
283
299
}
300
+
284
301
$ primaryKey = (array )$ this ->_table ->primaryKey ();
285
302
$ id = $ entity ->get (current ($ primaryKey ));
303
+
304
+ // When we have no key and bundled translations, we
305
+ // need to mark the entity dirty so the root
306
+ // entity persists.
307
+ if ($ noFields && $ bundled && !$ id ) {
308
+ foreach ($ this ->_translationFields () as $ field ) {
309
+ $ entity ->dirty ($ field , true );
310
+ }
311
+
312
+ return ;
313
+ }
314
+
315
+ if ($ noFields ) {
316
+ return ;
317
+ }
318
+
286
319
$ where = compact ('id ' , 'locale ' );
287
320
288
321
$ translation = $ this ->_translationTable ()->find ()
@@ -314,6 +347,30 @@ public function beforeSave(Event $event, EntityInterface $entity, ArrayObject $o
314
347
}
315
348
}
316
349
350
+ /**
351
+ * Returns a fully aliased field name for translated fields.
352
+ *
353
+ * If the requested field is configured as a translation field, field with
354
+ * an alias of a corresponding association is returned. Table-aliased
355
+ * field name is returned for all other fields.
356
+ *
357
+ * @param string $field Field name to be aliased.
358
+ * @return string
359
+ */
360
+ public function translationField ($ field )
361
+ {
362
+ if ($ this ->locale () === $ this ->getConfig ('defaultLocale ' )) {
363
+ return $ this ->_table ->aliasField ($ field );
364
+ }
365
+
366
+ $ translatedFields = $ this ->_translationFields ();
367
+ if (in_array ($ field , $ translatedFields )) {
368
+ return $ this ->config ('hasOneAlias ' ) . '. ' . $ field ;
369
+ }
370
+
371
+ return $ this ->_table ->aliasField ($ field );
372
+ }
373
+
317
374
/**
318
375
* Modifies the results from a table find in order to merge the translated fields
319
376
* into each entity for a given locale.
0 commit comments