TranslationRepository.php 7.8 KB

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