RelativeSlugHandler.php 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. <?php
  2. namespace Gedmo\Sluggable\Handler;
  3. use Doctrine\Common\Persistence\ObjectManager;
  4. use Gedmo\Sluggable\SluggableListener;
  5. use Gedmo\Sluggable\Mapping\Event\SluggableAdapter;
  6. use Gedmo\Tool\Wrapper\AbstractWrapper;
  7. use Gedmo\Exception\InvalidMappingException;
  8. use Gedmo\Sluggable\Util\Urlizer;
  9. /**
  10. * Sluggable handler which should be used in order to prefix
  11. * a slug of related object. For instance user may belong to a company
  12. * in this case user slug could look like 'company-name/user-firstname'
  13. * where path separator separates the relative slug
  14. *
  15. * @author Gediminas Morkevicius <gediminas.morkevicius@gmail.com>
  16. * @package Gedmo.Sluggable.Handler
  17. * @subpackage RelativeSlugHandler
  18. * @link http://www.gediminasm.org
  19. * @license MIT License (http://www.opensource.org/licenses/mit-license.php)
  20. */
  21. class RelativeSlugHandler implements SlugHandlerInterface
  22. {
  23. const SEPARATOR = '/';
  24. /**
  25. * @var Doctrine\Common\Persistence\ObjectManager
  26. */
  27. protected $om;
  28. /**
  29. * @var Gedmo\Sluggable\SluggableListener
  30. */
  31. protected $sluggable;
  32. /**
  33. * Used options
  34. *
  35. * @var array
  36. */
  37. private $usedOptions;
  38. /**
  39. * Callable of original transliterator
  40. * which is used by sluggable
  41. *
  42. * @var callable
  43. */
  44. private $originalTransliterator;
  45. /**
  46. * $options = array(
  47. * 'separator' => '/',
  48. * 'relationField' => 'something',
  49. * 'relationSlugField' => 'slug'
  50. * )
  51. * {@inheritDoc}
  52. */
  53. public function __construct(SluggableListener $sluggable)
  54. {
  55. $this->sluggable = $sluggable;
  56. }
  57. /**
  58. * {@inheritDoc}
  59. */
  60. public function getOptions($object)
  61. {
  62. $meta = $this->om->getClassMetadata(get_class($object));
  63. if (!isset($this->options[$meta->name])) {
  64. $config = $this->sluggable->getConfiguration($this->om, $meta->name);
  65. $options = $config['handlers'][get_called_class()];
  66. $default = array(
  67. 'separator' => '/'
  68. );
  69. $this->options[$meta->name] = array_merge($default, $options);
  70. }
  71. return $this->options[$meta->name];
  72. }
  73. /**
  74. * {@inheritDoc}
  75. */
  76. public function handlesUrlization()
  77. {
  78. return true;
  79. }
  80. /**
  81. * {@inheritDoc}
  82. */
  83. public function onChangeDecision(SluggableAdapter $ea, $config, $object, &$slug, &$needToChangeSlug)
  84. {
  85. $this->om = $ea->getObjectManager();
  86. $isInsert = $this->om->getUnitOfWork()->isScheduledForInsert($object);
  87. $this->usedOptions = $config['handlers'][get_called_class()];
  88. if (!isset($this->usedOptions['separator'])) {
  89. $this->usedOptions['separator'] = self::SEPARATOR;
  90. }
  91. if (!$isInsert && !$needToChangeSlug) {
  92. $changeSet = $ea->getObjectChangeSet($this->om->getUnitOfWork(), $object);
  93. if (isset($changeSet[$this->usedOptions['relationField']])) {
  94. $needToChangeSlug = true;
  95. }
  96. }
  97. }
  98. /**
  99. * {@inheritDoc}
  100. */
  101. public function postSlugBuild(SluggableAdapter $ea, array &$config, $object, &$slug)
  102. {
  103. $this->originalTransliterator = $this->sluggable->getTransliterator();
  104. $this->sluggable->setTransliterator(array($this, 'transliterate'));
  105. }
  106. /**
  107. * {@inheritDoc}
  108. */
  109. public static function validate(array $options, $meta)
  110. {
  111. if (!$meta->isSingleValuedAssociation($options['relationField'])) {
  112. throw new InvalidMappingException("Unable to find slug relation through field - [{$options['relationField']}] in class - {$meta->name}");
  113. }
  114. }
  115. /**
  116. * {@inheritDoc}
  117. */
  118. public function onSlugCompletion(SluggableAdapter $ea, array &$config, $object, &$slug)
  119. {}
  120. /**
  121. * Transliterates the slug and prefixes the slug
  122. * by relative one
  123. *
  124. * @param string $text
  125. * @param string $separator
  126. * @param object $object
  127. * @return string
  128. */
  129. public function transliterate($text, $separator, $object)
  130. {
  131. $result = call_user_func_array(
  132. $this->originalTransliterator,
  133. array($text, $separator, $object)
  134. );
  135. $result = Urlizer::urlize($result, $separator);
  136. $wrapped = AbstractWrapper::wrap($object, $this->om);
  137. $relation = $wrapped->getPropertyValue($this->usedOptions['relationField']);
  138. if ($relation) {
  139. $wrappedRelation = AbstractWrapper::wrap($relation, $this->om);
  140. $slug = $wrappedRelation->getPropertyValue($this->usedOptions['relationSlugField']);
  141. $result = $slug . $this->usedOptions['separator'] . $result;
  142. }
  143. $this->sluggable->setTransliterator($this->originalTransliterator);
  144. return $result;
  145. }
  146. }