vendor/gedmo/doctrine-extensions/src/Translatable/TranslatableListener.php line 527

  1. <?php
  2. /*
  3.  * This file is part of the Doctrine Behavioral Extensions package.
  4.  * (c) Gediminas Morkevicius <gediminas.morkevicius@gmail.com> http://www.gediminasm.org
  5.  * For the full copyright and license information, please view the LICENSE
  6.  * file that was distributed with this source code.
  7.  */
  8. namespace Gedmo\Translatable;
  9. use Doctrine\Common\EventArgs;
  10. use Doctrine\ODM\MongoDB\DocumentManager;
  11. use Doctrine\ORM\ORMInvalidArgumentException;
  12. use Doctrine\Persistence\Event\LifecycleEventArgs;
  13. use Doctrine\Persistence\Event\LoadClassMetadataEventArgs;
  14. use Doctrine\Persistence\Event\ManagerEventArgs;
  15. use Doctrine\Persistence\Mapping\ClassMetadata;
  16. use Doctrine\Persistence\ObjectManager;
  17. use Gedmo\Exception\InvalidArgumentException;
  18. use Gedmo\Exception\RuntimeException;
  19. use Gedmo\Mapping\MappedEventSubscriber;
  20. use Gedmo\Tool\Wrapper\AbstractWrapper;
  21. use Gedmo\Translatable\Mapping\Event\TranslatableAdapter;
  22. /**
  23.  * The translation listener handles the generation and
  24.  * loading of translations for entities which implements
  25.  * the Translatable interface.
  26.  *
  27.  * This behavior can impact the performance of your application
  28.  * since it does an additional query for each field to translate.
  29.  *
  30.  * Nevertheless the annotation metadata is properly cached and
  31.  * it is not a big overhead to lookup all entity annotations since
  32.  * the caching is activated for metadata
  33.  *
  34.  * @author Gediminas Morkevicius <gediminas.morkevicius@gmail.com>
  35.  *
  36.  * @phpstan-type TranslatableConfiguration = array{
  37.  *   fields?: string[],
  38.  *   fallback?: array<string, bool>,
  39.  *   locale?: string,
  40.  *   translationClass?: class-string,
  41.  *   useObjectClass?: class-string,
  42.  * }
  43.  *
  44.  * @phpstan-method TranslatableConfiguration getConfiguration(ObjectManager $objectManager, $class)
  45.  *
  46.  * @method TranslatableAdapter getEventAdapter(EventArgs $args)
  47.  *
  48.  * @final since gedmo/doctrine-extensions 3.11
  49.  */
  50. class TranslatableListener extends MappedEventSubscriber
  51. {
  52.     /**
  53.      * Query hint to override the fallback of translations
  54.      * integer 1 for true, 0 false
  55.      */
  56.     public const HINT_FALLBACK 'gedmo.translatable.fallback';
  57.     /**
  58.      * Query hint to override the fallback locale
  59.      */
  60.     public const HINT_TRANSLATABLE_LOCALE 'gedmo.translatable.locale';
  61.     /**
  62.      * Query hint to use inner join strategy for translations
  63.      */
  64.     public const HINT_INNER_JOIN 'gedmo.translatable.inner_join.translations';
  65.     /**
  66.      * Locale which is set on this listener.
  67.      * If Entity being translated has locale defined it
  68.      * will override this one
  69.      *
  70.      * @var string
  71.      */
  72.     protected $locale 'en_US';
  73.     /**
  74.      * Default locale, this changes behavior
  75.      * to not update the original record field if locale
  76.      * which is used for updating is not default. This
  77.      * will load the default translation in other locales
  78.      * if record is not translated yet
  79.      */
  80.     private string $defaultLocale 'en_US';
  81.     /**
  82.      * If this is set to false, when if entity does
  83.      * not have a translation for requested locale
  84.      * it will show a blank value
  85.      */
  86.     private bool $translationFallback false;
  87.     /**
  88.      * List of translations which do not have the foreign
  89.      * key generated yet - MySQL case. These translations
  90.      * will be updated with new keys on postPersist event
  91.      *
  92.      * @var array<int, array<int, object|Translatable>>
  93.      */
  94.     private array $pendingTranslationInserts = [];
  95.     /**
  96.      * Currently in case if there is TranslationQueryWalker
  97.      * in charge. We need to skip issuing additional queries
  98.      * on load
  99.      */
  100.     private bool $skipOnLoad false;
  101.     /**
  102.      * Tracks locale the objects currently translated in
  103.      *
  104.      * @var array<int, string>
  105.      */
  106.     private array $translatedInLocale = [];
  107.     /**
  108.      * Whether or not, to persist default locale
  109.      * translation or keep it in original record
  110.      */
  111.     private bool $persistDefaultLocaleTranslation false;
  112.     /**
  113.      * Tracks translation object for default locale
  114.      *
  115.      * @var array<int, array<string, object|Translatable>>
  116.      */
  117.     private array $translationInDefaultLocale = [];
  118.     /**
  119.      * Default translation value upon missing translation
  120.      */
  121.     private ?string $defaultTranslationValue null;
  122.     /**
  123.      * Specifies the list of events to listen
  124.      *
  125.      * @return string[]
  126.      */
  127.     public function getSubscribedEvents()
  128.     {
  129.         return [
  130.             'postLoad',
  131.             'postPersist',
  132.             'preFlush',
  133.             'onFlush',
  134.             'loadClassMetadata',
  135.         ];
  136.     }
  137.     /**
  138.      * Set to skip or not onLoad event
  139.      *
  140.      * @param bool $bool
  141.      *
  142.      * @return static
  143.      */
  144.     public function setSkipOnLoad($bool)
  145.     {
  146.         $this->skipOnLoad = (bool) $bool;
  147.         return $this;
  148.     }
  149.     /**
  150.      * Whether or not, to persist default locale
  151.      * translation or keep it in original record
  152.      *
  153.      * @param bool $bool
  154.      *
  155.      * @return static
  156.      */
  157.     public function setPersistDefaultLocaleTranslation($bool)
  158.     {
  159.         $this->persistDefaultLocaleTranslation = (bool) $bool;
  160.         return $this;
  161.     }
  162.     /**
  163.      * Check if should persist default locale
  164.      * translation or keep it in original record
  165.      *
  166.      * @return bool
  167.      */
  168.     public function getPersistDefaultLocaleTranslation()
  169.     {
  170.         return (bool) $this->persistDefaultLocaleTranslation;
  171.     }
  172.     /**
  173.      * Add additional $translation for pending $oid object
  174.      * which is being inserted
  175.      *
  176.      * @param int    $oid
  177.      * @param object $translation
  178.      *
  179.      * @return void
  180.      */
  181.     public function addPendingTranslationInsert($oid$translation)
  182.     {
  183.         $this->pendingTranslationInserts[$oid][] = $translation;
  184.     }
  185.     /**
  186.      * Maps additional metadata
  187.      *
  188.      * @param LoadClassMetadataEventArgs $eventArgs
  189.      *
  190.      * @phpstan-param LoadClassMetadataEventArgs<ClassMetadata<object>, ObjectManager> $eventArgs
  191.      *
  192.      * @return void
  193.      */
  194.     public function loadClassMetadata(EventArgs $eventArgs)
  195.     {
  196.         $this->loadMetadataForObjectClass($eventArgs->getObjectManager(), $eventArgs->getClassMetadata());
  197.     }
  198.     /**
  199.      * Get the translation class to be used
  200.      * for the object $class
  201.      *
  202.      * @param string $class
  203.      *
  204.      * @phpstan-param class-string $class
  205.      *
  206.      * @return string
  207.      *
  208.      * @phpstan-return class-string
  209.      */
  210.     public function getTranslationClass(TranslatableAdapter $ea$class)
  211.     {
  212.         return self::$configurations[$this->name][$class]['translationClass'] ?? $ea->getDefaultTranslationClass()
  213.         ;
  214.     }
  215.     /**
  216.      * Enable or disable translation fallback
  217.      * to original record value
  218.      *
  219.      * @param bool $bool
  220.      *
  221.      * @return static
  222.      */
  223.     public function setTranslationFallback($bool)
  224.     {
  225.         $this->translationFallback = (bool) $bool;
  226.         return $this;
  227.     }
  228.     /**
  229.      * Weather or not is using the translation
  230.      * fallback to original record
  231.      *
  232.      * @return bool
  233.      */
  234.     public function getTranslationFallback()
  235.     {
  236.         return $this->translationFallback;
  237.     }
  238.     /**
  239.      * Set the locale to use for translation listener
  240.      *
  241.      * @param string $locale
  242.      *
  243.      * @return static
  244.      */
  245.     public function setTranslatableLocale($locale)
  246.     {
  247.         $this->validateLocale($locale);
  248.         $this->locale $locale;
  249.         return $this;
  250.     }
  251.     /**
  252.      * Set the default translation value on missing translation
  253.      *
  254.      * @deprecated usage of a non nullable value for defaultTranslationValue is deprecated
  255.      * and will be removed on the next major release which will rely on the expected types
  256.      */
  257.     public function setDefaultTranslationValue(?string $defaultTranslationValue): void
  258.     {
  259.         $this->defaultTranslationValue $defaultTranslationValue;
  260.     }
  261.     /**
  262.      * Sets the default locale, this changes behavior
  263.      * to not update the original record field if locale
  264.      * which is used for updating is not default
  265.      *
  266.      * @param string $locale
  267.      *
  268.      * @return static
  269.      */
  270.     public function setDefaultLocale($locale)
  271.     {
  272.         $this->validateLocale($locale);
  273.         $this->defaultLocale $locale;
  274.         return $this;
  275.     }
  276.     /**
  277.      * Gets the default locale
  278.      *
  279.      * @return string
  280.      */
  281.     public function getDefaultLocale()
  282.     {
  283.         return $this->defaultLocale;
  284.     }
  285.     /**
  286.      * Get currently set global locale, used
  287.      * extensively during query execution
  288.      *
  289.      * @return string
  290.      */
  291.     public function getListenerLocale()
  292.     {
  293.         return $this->locale;
  294.     }
  295.     /**
  296.      * Gets the locale to use for translation. Loads object
  297.      * defined locale first.
  298.      *
  299.      * @param object        $object
  300.      * @param ClassMetadata $meta
  301.      * @param object        $om
  302.      *
  303.      * @throws RuntimeException if language or locale property is not found in entity
  304.      *
  305.      * @return string
  306.      */
  307.     public function getTranslatableLocale($object$meta$om null)
  308.     {
  309.         $locale $this->locale;
  310.         $configurationLocale self::$configurations[$this->name][$meta->getName()]['locale'] ?? null;
  311.         if (null !== $configurationLocale) {
  312.             $class $meta->getReflectionClass();
  313.             if (!$class->hasProperty($configurationLocale)) {
  314.                 throw new RuntimeException("There is no locale or language property ({$configurationLocale}) found on object: {$meta->getName()}");
  315.             }
  316.             $reflectionProperty $class->getProperty($configurationLocale);
  317.             $reflectionProperty->setAccessible(true);
  318.             $value $reflectionProperty->getValue($object);
  319.             if (is_object($value) && method_exists($value'__toString')) {
  320.                 $value $value->__toString();
  321.             }
  322.             if ($this->isValidLocale($value)) {
  323.                 $locale $value;
  324.             }
  325.         } elseif ($om instanceof DocumentManager) {
  326.             [, $parentObject] = $om->getUnitOfWork()->getParentAssociation($object);
  327.             if (null !== $parentObject) {
  328.                 $parentMeta $om->getClassMetadata(get_class($parentObject));
  329.                 $locale $this->getTranslatableLocale($parentObject$parentMeta$om);
  330.             }
  331.         }
  332.         return $locale;
  333.     }
  334.     /**
  335.      * Handle translation changes in default locale
  336.      *
  337.      * This has to be done in the preFlush because, when an entity has been loaded
  338.      * in a different locale, no changes will be detected.
  339.      *
  340.      * @param ManagerEventArgs $args
  341.      *
  342.      * @phpstan-param ManagerEventArgs<ObjectManager> $args
  343.      *
  344.      * @return void
  345.      */
  346.     public function preFlush(EventArgs $args)
  347.     {
  348.         $ea $this->getEventAdapter($args);
  349.         $om $ea->getObjectManager();
  350.         $uow $om->getUnitOfWork();
  351.         foreach ($this->translationInDefaultLocale as $oid => $fields) {
  352.             $trans reset($fields);
  353.             assert(false !== $trans);
  354.             if ($ea->usesPersonalTranslation(get_class($trans))) {
  355.                 $entity $trans->getObject();
  356.             } else {
  357.                 $entity $uow->tryGetById($trans->getForeignKey(), $trans->getObjectClass());
  358.             }
  359.             if (!$entity) {
  360.                 continue;
  361.             }
  362.             try {
  363.                 $uow->scheduleForUpdate($entity);
  364.             } catch (ORMInvalidArgumentException $e) {
  365.                 foreach ($fields as $field => $trans) {
  366.                     $this->removeTranslationInDefaultLocale($oid$field);
  367.                 }
  368.             }
  369.         }
  370.     }
  371.     /**
  372.      * Looks for translatable objects being inserted or updated
  373.      * for further processing
  374.      *
  375.      * @param ManagerEventArgs $args
  376.      *
  377.      * @phpstan-param ManagerEventArgs<ObjectManager> $args
  378.      *
  379.      * @return void
  380.      */
  381.     public function onFlush(EventArgs $args)
  382.     {
  383.         $ea $this->getEventAdapter($args);
  384.         $om $ea->getObjectManager();
  385.         $uow $om->getUnitOfWork();
  386.         // check all scheduled inserts for Translatable objects
  387.         foreach ($ea->getScheduledObjectInsertions($uow) as $object) {
  388.             $meta $om->getClassMetadata(get_class($object));
  389.             $config $this->getConfiguration($om$meta->getName());
  390.             if (isset($config['fields'])) {
  391.                 $this->handleTranslatableObjectUpdate($ea$objecttrue);
  392.             }
  393.         }
  394.         // check all scheduled updates for Translatable entities
  395.         foreach ($ea->getScheduledObjectUpdates($uow) as $object) {
  396.             $meta $om->getClassMetadata(get_class($object));
  397.             $config $this->getConfiguration($om$meta->getName());
  398.             if (isset($config['fields'])) {
  399.                 $this->handleTranslatableObjectUpdate($ea$objectfalse);
  400.             }
  401.         }
  402.         // check scheduled deletions for Translatable entities
  403.         foreach ($ea->getScheduledObjectDeletions($uow) as $object) {
  404.             $meta $om->getClassMetadata(get_class($object));
  405.             $config $this->getConfiguration($om$meta->getName());
  406.             if (isset($config['fields'])) {
  407.                 $wrapped AbstractWrapper::wrap($object$om);
  408.                 $transClass $this->getTranslationClass($ea$meta->getName());
  409.                 \assert($wrapped instanceof AbstractWrapper);
  410.                 $ea->removeAssociatedTranslations($wrapped$transClass$config['useObjectClass']);
  411.             }
  412.         }
  413.     }
  414.     /**
  415.      * Checks for inserted object to update their translation
  416.      * foreign keys
  417.      *
  418.      * @param LifecycleEventArgs $args
  419.      *
  420.      * @phpstan-param LifecycleEventArgs<ObjectManager> $args
  421.      *
  422.      * @return void
  423.      */
  424.     public function postPersist(EventArgs $args)
  425.     {
  426.         $ea $this->getEventAdapter($args);
  427.         $om $ea->getObjectManager();
  428.         $object $ea->getObject();
  429.         $meta $om->getClassMetadata(get_class($object));
  430.         // check if entity is tracked by translatable and without foreign key
  431.         if ($this->getConfiguration($om$meta->getName()) && [] !== $this->pendingTranslationInserts) {
  432.             $oid spl_object_id($object);
  433.             if (array_key_exists($oid$this->pendingTranslationInserts)) {
  434.                 // load the pending translations without key
  435.                 $wrapped AbstractWrapper::wrap($object$om);
  436.                 $objectId $wrapped->getIdentifier();
  437.                 $translationClass $this->getTranslationClass($eaget_class($object));
  438.                 foreach ($this->pendingTranslationInserts[$oid] as $translation) {
  439.                     if ($ea->usesPersonalTranslation($translationClass)) {
  440.                         $translation->setObject($objectId);
  441.                     } else {
  442.                         $translation->setForeignKey($objectId);
  443.                     }
  444.                     $ea->insertTranslationRecord($translation);
  445.                 }
  446.                 unset($this->pendingTranslationInserts[$oid]);
  447.             }
  448.         }
  449.     }
  450.     /**
  451.      * After object is loaded, listener updates the translations
  452.      * by currently used locale
  453.      *
  454.      * @param ManagerEventArgs $args
  455.      *
  456.      * @phpstan-param ManagerEventArgs<ObjectManager> $args
  457.      *
  458.      * @return void
  459.      */
  460.     public function postLoad(EventArgs $args)
  461.     {
  462.         $ea $this->getEventAdapter($args);
  463.         $om $ea->getObjectManager();
  464.         $object $ea->getObject();
  465.         $meta $om->getClassMetadata(get_class($object));
  466.         $config $this->getConfiguration($om$meta->getName());
  467.         $locale $this->defaultLocale;
  468.         $oid null;
  469.         if (isset($config['fields'])) {
  470.             $locale $this->getTranslatableLocale($object$meta$om);
  471.             $oid spl_object_id($object);
  472.             $this->translatedInLocale[$oid] = $locale;
  473.         }
  474.         if ($this->skipOnLoad) {
  475.             return;
  476.         }
  477.         if (isset($config['fields']) && ($locale !== $this->defaultLocale || $this->persistDefaultLocaleTranslation)) {
  478.             // fetch translations
  479.             $translationClass $this->getTranslationClass($ea$config['useObjectClass']);
  480.             $result $ea->loadTranslations(
  481.                 $object,
  482.                 $translationClass,
  483.                 $locale,
  484.                 $config['useObjectClass']
  485.             );
  486.             // translate object's translatable properties
  487.             foreach ($config['fields'] as $field) {
  488.                 $translated $this->defaultTranslationValue;
  489.                 foreach ($result as $entry) {
  490.                     if ($entry['field'] == $field) {
  491.                         $translated $entry['content'] ?? null;
  492.                         break;
  493.                     }
  494.                 }
  495.                 // update translation
  496.                 if ($this->defaultTranslationValue !== $translated
  497.                     || (!$this->translationFallback && (!isset($config['fallback'][$field]) || !$config['fallback'][$field]))
  498.                     || ($this->translationFallback && isset($config['fallback'][$field]) && !$config['fallback'][$field])
  499.                 ) {
  500.                     $ea->setTranslationValue($object$field$translated);
  501.                     // ensure clean changeset
  502.                     $ea->setOriginalObjectProperty(
  503.                         $om->getUnitOfWork(),
  504.                         $object,
  505.                         $field,
  506.                         $meta->getReflectionProperty($field)->getValue($object)
  507.                     );
  508.                 }
  509.             }
  510.         }
  511.     }
  512.     /**
  513.      * Sets translation object which represents translation in default language.
  514.      *
  515.      * @param int                 $oid   hash of basic entity
  516.      * @param string              $field field of basic entity
  517.      * @param object|Translatable $trans Translation object
  518.      *
  519.      * @return void
  520.      */
  521.     public function setTranslationInDefaultLocale($oid$field$trans)
  522.     {
  523.         if (!isset($this->translationInDefaultLocale[$oid])) {
  524.             $this->translationInDefaultLocale[$oid] = [];
  525.         }
  526.         $this->translationInDefaultLocale[$oid][$field] = $trans;
  527.     }
  528.     /**
  529.      * @return bool
  530.      */
  531.     public function isSkipOnLoad()
  532.     {
  533.         return $this->skipOnLoad;
  534.     }
  535.     /**
  536.      * Check if object has any translation object which represents translation in default language.
  537.      * This is for internal use only.
  538.      *
  539.      * @param int $oid hash of the basic entity
  540.      *
  541.      * @return bool
  542.      */
  543.     public function hasTranslationsInDefaultLocale($oid)
  544.     {
  545.         return array_key_exists($oid$this->translationInDefaultLocale);
  546.     }
  547.     protected function getNamespace()
  548.     {
  549.         return __NAMESPACE__;
  550.     }
  551.     /**
  552.      * Validates the given locale
  553.      *
  554.      * @param string $locale locale to validate
  555.      *
  556.      * @throws InvalidArgumentException if locale is not valid
  557.      *
  558.      * @return void
  559.      */
  560.     protected function validateLocale($locale)
  561.     {
  562.         if (!$this->isValidLocale($locale)) {
  563.             throw new InvalidArgumentException('Locale or language cannot be empty and must be set through Listener or Entity');
  564.         }
  565.     }
  566.     /**
  567.      * Check if the given locale is valid
  568.      */
  569.     private function isValidLocale(?string $locale): bool
  570.     {
  571.         return is_string($locale) && strlen($locale);
  572.     }
  573.     /**
  574.      * Creates the translation for object being flushed
  575.      *
  576.      * @throws \UnexpectedValueException if locale is not valid, or
  577.      *                                   primary key is composite, missing or invalid
  578.      */
  579.     private function handleTranslatableObjectUpdate(TranslatableAdapter $eaobject $objectbool $isInsert): void
  580.     {
  581.         $om $ea->getObjectManager();
  582.         $wrapped AbstractWrapper::wrap($object$om);
  583.         $meta $wrapped->getMetadata();
  584.         $config $this->getConfiguration($om$meta->getName());
  585.         // no need cache, metadata is loaded only once in MetadataFactoryClass
  586.         $translationClass $this->getTranslationClass($ea$config['useObjectClass']);
  587.         $translationMetadata $om->getClassMetadata($translationClass);
  588.         // check for the availability of the primary key
  589.         $objectId $wrapped->getIdentifier();
  590.         // load the currently used locale
  591.         $locale $this->getTranslatableLocale($object$meta$om);
  592.         $uow $om->getUnitOfWork();
  593.         $oid spl_object_id($object);
  594.         $changeSet $ea->getObjectChangeSet($uow$object);
  595.         $translatableFields $config['fields'];
  596.         foreach ($translatableFields as $field) {
  597.             $wasPersistedSeparetely false;
  598.             $skip = isset($this->translatedInLocale[$oid]) && $locale === $this->translatedInLocale[$oid];
  599.             $skip $skip && !isset($changeSet[$field]) && !$this->getTranslationInDefaultLocale($oid$field);
  600.             if ($skip) {
  601.                 continue; // locale is same and nothing changed
  602.             }
  603.             $translation null;
  604.             foreach ($ea->getScheduledObjectInsertions($uow) as $trans) {
  605.                 if ($locale !== $this->defaultLocale
  606.                     && get_class($trans) === $translationClass
  607.                     && $trans->getLocale() === $this->defaultLocale
  608.                     && $trans->getField() === $field
  609.                     && $this->belongsToObject($ea$trans$object)) {
  610.                     $this->setTranslationInDefaultLocale($oid$field$trans);
  611.                     break;
  612.                 }
  613.             }
  614.             // lookup persisted translations
  615.             foreach ($ea->getScheduledObjectInsertions($uow) as $trans) {
  616.                 if (get_class($trans) !== $translationClass
  617.                     || $trans->getLocale() !== $locale
  618.                     || $trans->getField() !== $field) {
  619.                     continue;
  620.                 }
  621.                 if ($ea->usesPersonalTranslation($translationClass)) {
  622.                     $wasPersistedSeparetely $trans->getObject() === $object;
  623.                 } else {
  624.                     $wasPersistedSeparetely $trans->getObjectClass() === $config['useObjectClass']
  625.                         && $trans->getForeignKey() === $objectId;
  626.                 }
  627.                 if ($wasPersistedSeparetely) {
  628.                     $translation $trans;
  629.                     break;
  630.                 }
  631.             }
  632.             // check if translation already is created
  633.             if (!$isInsert && !$translation) {
  634.                 \assert($wrapped instanceof AbstractWrapper);
  635.                 $translation $ea->findTranslation(
  636.                     $wrapped,
  637.                     $locale,
  638.                     $field,
  639.                     $translationClass,
  640.                     $config['useObjectClass']
  641.                 );
  642.             }
  643.             // create new translation if translation not already created and locale is different from default locale, otherwise, we have the date in the original record
  644.             $persistNewTranslation = !$translation
  645.                 && ($locale !== $this->defaultLocale || $this->persistDefaultLocaleTranslation)
  646.             ;
  647.             if ($persistNewTranslation) {
  648.                 $translation $translationMetadata->newInstance();
  649.                 $translation->setLocale($locale);
  650.                 $translation->setField($field);
  651.                 if ($ea->usesPersonalTranslation($translationClass)) {
  652.                     $translation->setObject($object);
  653.                 } else {
  654.                     $translation->setObjectClass($config['useObjectClass']);
  655.                     $translation->setForeignKey($objectId);
  656.                 }
  657.             }
  658.             if ($translation) {
  659.                 // set the translated field, take value using reflection
  660.                 $content $ea->getTranslationValue($object$field);
  661.                 $translation->setContent($content);
  662.                 // check if need to update in database
  663.                 $transWrapper AbstractWrapper::wrap($translation$om);
  664.                 if (((null === $content && !$isInsert) || is_bool($content) || is_int($content) || is_string($content) || !empty($content)) && ($isInsert || !$transWrapper->getIdentifier() || isset($changeSet[$field]))) {
  665.                     if ($isInsert && !$objectId && !$ea->usesPersonalTranslation($translationClass)) {
  666.                         // if we do not have the primary key yet available
  667.                         // keep this translation in memory to insert it later with foreign key
  668.                         $this->pendingTranslationInserts[spl_object_id($object)][] = $translation;
  669.                     } else {
  670.                         // persist and compute change set for translation
  671.                         if ($wasPersistedSeparetely) {
  672.                             $ea->recomputeSingleObjectChangeset($uow$translationMetadata$translation);
  673.                         } else {
  674.                             $om->persist($translation);
  675.                             $uow->computeChangeSet($translationMetadata$translation);
  676.                         }
  677.                     }
  678.                 }
  679.             }
  680.             if ($isInsert && null !== $this->getTranslationInDefaultLocale($oid$field)) {
  681.                 // We can't rely on object field value which is created in non-default locale.
  682.                 // If we provide translation for default locale as well, the latter is considered to be trusted
  683.                 // and object content should be overridden.
  684.                 $wrapped->setPropertyValue($field$this->getTranslationInDefaultLocale($oid$field)->getContent());
  685.                 $ea->recomputeSingleObjectChangeset($uow$meta$object);
  686.                 $this->removeTranslationInDefaultLocale($oid$field);
  687.             }
  688.         }
  689.         $this->translatedInLocale[$oid] = $locale;
  690.         // check if we have default translation and need to reset the translation
  691.         if (!$isInsert && strlen($this->defaultLocale)) {
  692.             $this->validateLocale($this->defaultLocale);
  693.             $modifiedChangeSet $changeSet;
  694.             foreach ($changeSet as $field => $changes) {
  695.                 if (in_array($field$translatableFieldstrue)) {
  696.                     if ($locale !== $this->defaultLocale) {
  697.                         $ea->setOriginalObjectProperty($uow$object$field$changes[0]);
  698.                         unset($modifiedChangeSet[$field]);
  699.                     }
  700.                 }
  701.             }
  702.             $ea->recomputeSingleObjectChangeset($uow$meta$object);
  703.             // cleanup current changeset only if working in a another locale different than de default one, otherwise the changeset will always be reverted
  704.             if ($locale !== $this->defaultLocale) {
  705.                 $ea->clearObjectChangeSet($uow$object);
  706.                 // recompute changeset only if there are changes other than reverted translations
  707.                 if ($modifiedChangeSet || $this->hasTranslationsInDefaultLocale($oid)) {
  708.                     foreach ($modifiedChangeSet as $field => $changes) {
  709.                         $ea->setOriginalObjectProperty($uow$object$field$changes[0]);
  710.                     }
  711.                     foreach ($translatableFields as $field) {
  712.                         if (null !== $this->getTranslationInDefaultLocale($oid$field)) {
  713.                             $wrapped->setPropertyValue($field$this->getTranslationInDefaultLocale($oid$field)->getContent());
  714.                             $this->removeTranslationInDefaultLocale($oid$field);
  715.                         }
  716.                     }
  717.                     $ea->recomputeSingleObjectChangeset($uow$meta$object);
  718.                 }
  719.             }
  720.         }
  721.     }
  722.     /**
  723.      * Removes translation object which represents translation in default language.
  724.      * This is for internal use only.
  725.      *
  726.      * @param int    $oid   hash of the basic entity
  727.      * @param string $field field of basic entity
  728.      */
  729.     private function removeTranslationInDefaultLocale(int $oidstring $field): void
  730.     {
  731.         if (isset($this->translationInDefaultLocale[$oid])) {
  732.             if (isset($this->translationInDefaultLocale[$oid][$field])) {
  733.                 unset($this->translationInDefaultLocale[$oid][$field]);
  734.             }
  735.             if (!$this->translationInDefaultLocale[$oid]) {
  736.                 // We removed the final remaining elements from the
  737.                 // translationInDefaultLocale[$oid] array, so we might as well
  738.                 // completely remove the entry at $oid.
  739.                 unset($this->translationInDefaultLocale[$oid]);
  740.             }
  741.         }
  742.     }
  743.     /**
  744.      * Gets translation object which represents translation in default language.
  745.      * This is for internal use only.
  746.      *
  747.      * @param int    $oid   hash of the basic entity
  748.      * @param string $field field of basic entity
  749.      *
  750.      * @return object|Translatable|null Returns translation object if it exists or NULL otherwise
  751.      */
  752.     private function getTranslationInDefaultLocale(int $oidstring $field)
  753.     {
  754.         return $this->translationInDefaultLocale[$oid][$field] ?? null;
  755.     }
  756.     /**
  757.      * Checks if the translation entity belongs to the object in question
  758.      */
  759.     private function belongsToObject(TranslatableAdapter $eaobject $transobject $object): bool
  760.     {
  761.         if ($ea->usesPersonalTranslation(get_class($trans))) {
  762.             return $trans->getObject() === $object;
  763.         }
  764.         return $trans->getForeignKey() === $object->getId()
  765.             && ($trans->getObjectClass() === get_class($object));
  766.     }
  767. }