TranslationRepository.php 8.6 KB

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