TranslationRepository.php 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. <?php
  2. namespace Gedmo\Translatable\Entity\Repository;
  3. use Gedmo\Translatable\TranslatableListener;
  4. use Doctrine\ORM\EntityRepository;
  5. use Doctrine\ORM\Query;
  6. use Doctrine\ORM\EntityManager;
  7. use Doctrine\ORM\Mapping\ClassMetadata;
  8. use Gedmo\Tool\Wrapper\EntityWrapper;
  9. use Gedmo\Translatable\Mapping\Event\Adapter\ORM as TranslatableAdapterORM;
  10. use Doctrine\DBAL\Types\Type;
  11. /**
  12. * The TranslationRepository has some useful functions
  13. * to interact with translations.
  14. *
  15. * @author Gediminas Morkevicius <gediminas.morkevicius@gmail.com>
  16. * @package Gedmo.Translatable.Entity.Repository
  17. * @subpackage TranslationRepository
  18. * @link http://www.gediminasm.org
  19. * @license MIT License (http://www.opensource.org/licenses/mit-license.php)
  20. */
  21. class TranslationRepository extends EntityRepository
  22. {
  23. /**
  24. * Current TranslatableListener instance used
  25. * in EntityManager
  26. *
  27. * @var TranslatableListener
  28. */
  29. private $listener;
  30. /**
  31. * {@inheritdoc}
  32. */
  33. public function __construct(EntityManager $em, ClassMetadata $class)
  34. {
  35. if ($class->getReflectionClass()->isSubclassOf('Gedmo\Translatable\Entity\MappedSuperclass\AbstractPersonalTranslation')) {
  36. throw new \Gedmo\Exception\UnexpectedValueException('This repository is useless for personal translations');
  37. }
  38. parent::__construct($em, $class);
  39. }
  40. /**
  41. * Makes additional translation of $entity $field into $locale
  42. * using $value
  43. *
  44. * @param object $entity
  45. * @param string $field
  46. * @param string $locale
  47. * @param mixed $value
  48. * @return TranslationRepository
  49. */
  50. public function translate($entity, $field, $locale, $value)
  51. {
  52. $meta = $this->_em->getClassMetadata(get_class($entity));
  53. $listener = $this->getTranslatableListener();
  54. $config = $listener->getConfiguration($this->_em, $meta->name);
  55. if (!isset($config['fields']) || !in_array($field, $config['fields'])) {
  56. throw new \Gedmo\Exception\InvalidArgumentException("Entity: {$meta->name} does not translate field - {$field}");
  57. }
  58. $needsPersist = TRUE;
  59. if ($locale === $listener->getTranslatableLocale($entity, $meta)) {
  60. $meta->getReflectionProperty($field)->setValue($entity, $value);
  61. $this->_em->persist($entity);
  62. } else {
  63. if (isset($config['translationClass'])) {
  64. $class = $config['translationClass'];
  65. } else {
  66. $ea = new TranslatableAdapterORM();
  67. $class = $listener->getTranslationClass($ea, $config['useObjectClass']);
  68. }
  69. $foreignKey = $meta->getReflectionProperty($meta->getSingleIdentifierFieldName())->getValue($entity);
  70. $objectClass = $config['useObjectClass'];
  71. $transMeta = $this->_em->getClassMetadata($class);
  72. $trans = $this->findOneBy(compact('locale', 'objectClass', 'field', 'foreignKey'));
  73. if (!$trans) {
  74. $trans = $transMeta->newInstance();
  75. $transMeta->getReflectionProperty('foreignKey')->setValue($trans, $foreignKey);
  76. $transMeta->getReflectionProperty('objectClass')->setValue($trans, $objectClass);
  77. $transMeta->getReflectionProperty('field')->setValue($trans, $field);
  78. $transMeta->getReflectionProperty('locale')->setValue($trans, $locale);
  79. if ($listener->getDefaultLocale() != $listener->getTranslatableLocale($entity, $meta) &&
  80. $locale === $listener->getDefaultLocale()) {
  81. $listener->setTranslationInDefaultLocale(spl_object_hash($entity), $trans);
  82. $needsPersist = $listener->getPersistDefaultLocaleTranslation();
  83. }
  84. }
  85. $type = Type::getType($meta->getTypeOfField($field));
  86. $transformed = $type->convertToDatabaseValue($value, $this->_em->getConnection()->getDatabasePlatform());
  87. $transMeta->getReflectionProperty('content')->setValue($trans, $transformed);
  88. if ($needsPersist) {
  89. if ($this->_em->getUnitOfWork()->isInIdentityMap($entity)) {
  90. $this->_em->persist($trans);
  91. } else {
  92. $oid = spl_object_hash($entity);
  93. $listener->addPendingTranslationInsert($oid, $trans);
  94. }
  95. }
  96. }
  97. return $this;
  98. }
  99. /**
  100. * Loads all translations with all translatable
  101. * fields from the given entity
  102. *
  103. * @param object $entity Must implement Translatable
  104. * @return array list of translations in locale groups
  105. */
  106. public function findTranslations($entity)
  107. {
  108. $result = array();
  109. $wrapped = new EntityWrapper($entity, $this->_em);
  110. if ($wrapped->hasValidIdentifier()) {
  111. $entityId = $wrapped->getIdentifier();
  112. $entityClass = $wrapped->getMetadata()->rootEntityName;
  113. $translationMeta = $this->getClassMetadata(); // table inheritance support
  114. $qb = $this->_em->createQueryBuilder();
  115. $qb->select('trans.content, trans.field, trans.locale')
  116. ->from($translationMeta->rootEntityName, 'trans')
  117. ->where('trans.foreignKey = :entityId', 'trans.objectClass = :entityClass')
  118. ->orderBy('trans.locale');
  119. $q = $qb->getQuery();
  120. $data = $q->execute(
  121. compact('entityId', 'entityClass'),
  122. Query::HYDRATE_ARRAY
  123. );
  124. if ($data && is_array($data) && count($data)) {
  125. foreach ($data as $row) {
  126. $result[$row['locale']][$row['field']] = $row['content'];
  127. }
  128. }
  129. }
  130. return $result;
  131. }
  132. /**
  133. * Find the entity $class by the translated field.
  134. * Result is the first occurence of translated field.
  135. * Query can be slow, since there are no indexes on such
  136. * columns
  137. *
  138. * @param string $field
  139. * @param string $value
  140. * @param string $class
  141. * @return object - instance of $class or null if not found
  142. */
  143. public function findObjectByTranslatedField($field, $value, $class)
  144. {
  145. $entity = null;
  146. $meta = $this->_em->getClassMetadata($class);
  147. $translationMeta = $this->getClassMetadata(); // table inheritance support
  148. if ($meta->hasField($field)) {
  149. $dql = "SELECT trans.foreignKey FROM {$translationMeta->rootEntityName} trans";
  150. $dql .= ' WHERE trans.objectClass = :class';
  151. $dql .= ' AND trans.field = :field';
  152. $dql .= ' AND trans.content = :value';
  153. $q = $this->_em->createQuery($dql);
  154. $q->setParameters(compact('class', 'field', 'value'));
  155. $q->setMaxResults(1);
  156. $result = $q->getArrayResult();
  157. $id = count($result) ? $result[0]['foreignKey'] : null;
  158. if ($id) {
  159. $entity = $this->_em->find($class, $id);
  160. }
  161. }
  162. return $entity;
  163. }
  164. /**
  165. * Loads all translations with all translatable
  166. * fields by a given entity primary key
  167. *
  168. * @param mixed $id - primary key value of an entity
  169. * @return array
  170. */
  171. public function findTranslationsByObjectId($id)
  172. {
  173. $result = array();
  174. if ($id) {
  175. $translationMeta = $this->getClassMetadata(); // table inheritance support
  176. $qb = $this->_em->createQueryBuilder();
  177. $qb->select('trans.content, trans.field, trans.locale')
  178. ->from($translationMeta->rootEntityName, 'trans')
  179. ->where('trans.foreignKey = :entityId')
  180. ->orderBy('trans.locale');
  181. $q = $qb->getQuery();
  182. $data = $q->execute(
  183. array('entityId' => $id),
  184. Query::HYDRATE_ARRAY
  185. );
  186. if ($data && is_array($data) && count($data)) {
  187. foreach ($data as $row) {
  188. $result[$row['locale']][$row['field']] = $row['content'];
  189. }
  190. }
  191. }
  192. return $result;
  193. }
  194. /**
  195. * Get the currently used TranslatableListener
  196. *
  197. * @throws \Gedmo\Exception\RuntimeException - if listener is not found
  198. * @return TranslatableListener
  199. */
  200. private function getTranslatableListener()
  201. {
  202. if (!$this->listener) {
  203. foreach ($this->_em->getEventManager()->getListeners() as $event => $listeners) {
  204. foreach ($listeners as $hash => $listener) {
  205. if ($listener instanceof TranslatableListener) {
  206. $this->listener = $listener;
  207. break;
  208. }
  209. }
  210. if ($this->listener) {
  211. break;
  212. }
  213. }
  214. if (is_null($this->listener)) {
  215. throw new \Gedmo\Exception\RuntimeException('The translation listener could not be found');
  216. }
  217. }
  218. return $this->listener;
  219. }
  220. }