RelativeSlugHandler.php 4.7 KB

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