vendor/symfony/event-dispatcher/EventDispatcher.php line 270

  1. use Psr\EventDispatcher\StoppableEventInterface;
  2. use Symfony\Component\EventDispatcher\Debug\WrappedListener;
  3. /**
  4.  * The EventDispatcherInterface is the central point of Symfony's event listener system.
  5.  *
  6.  * Listeners are registered on the manager and events are dispatched through the
  7.  * manager.
  8.  *
  9.  * @author Guilherme Blanco <guilhermeblanco@hotmail.com>
  10.  * @author Jonathan Wage <jonwage@gmail.com>
  11.  * @author Roman Borschel <roman@code-factory.org>
  12.  * @author Bernhard Schussek <bschussek@gmail.com>
  13.  * @author Fabien Potencier <fabien@symfony.com>
  14.  * @author Jordi Boggiano <j.boggiano@seld.be>
  15.  * @author Jordan Alliot <jordan.alliot@gmail.com>
  16.  * @author Nicolas Grekas <p@tchwork.com>
  17.  */
  18. class EventDispatcher implements EventDispatcherInterface
  19. {
  20.     private array $listeners = [];
  21.     private array $sorted = [];
  22.     private array $optimized;
  23.     public function __construct()
  24.     {
  25.         if (__CLASS__ === static::class) {
  26.             $this->optimized = [];
  27.         }
  28.     }
  29.     public function dispatch(object $event, ?string $eventName null): object
  30.     {
  31.         $eventName ??= $event::class;
  32.         if (isset($this->optimized)) {
  33.             $listeners $this->optimized[$eventName] ?? (empty($this->listeners[$eventName]) ? [] : $this->optimizeListeners($eventName));
  34.         } else {
  35.             $listeners $this->getListeners($eventName);
  36.         }
  37.         if ($listeners) {
  38.             $this->callListeners($listeners$eventName$event);
  39.         }
  40.         return $event;
  41.     }
  42.     public function getListeners(?string $eventName null): array
  43.     {
  44.         if (null !== $eventName) {
  45.             if (empty($this->listeners[$eventName])) {
  46.                 return [];
  47.             }
  48.             if (!isset($this->sorted[$eventName])) {
  49.                 $this->sortListeners($eventName);
  50.             }
  51.             return $this->sorted[$eventName];
  52.         }
  53.         foreach ($this->listeners as $eventName => $eventListeners) {
  54.             if (!isset($this->sorted[$eventName])) {
  55.                 $this->sortListeners($eventName);
  56.             }
  57.         }
  58.         return array_filter($this->sorted);
  59.     }
  60.     public function getListenerPriority(string $eventName, callable|array $listener): ?int
  61.     {
  62.         if (empty($this->listeners[$eventName])) {
  63.             return null;
  64.         }
  65.         if (\is_array($listener) && isset($listener[0]) && $listener[0] instanceof \Closure && >= \count($listener)) {
  66.             $listener[0] = $listener[0]();
  67.             $listener[1] ??= '__invoke';
  68.         }
  69.         foreach ($this->listeners[$eventName] as $priority => &$listeners) {
  70.             foreach ($listeners as &$v) {
  71.                 if ($v !== $listener && \is_array($v) && isset($v[0]) && $v[0] instanceof \Closure && >= \count($v)) {
  72.                     $v[0] = $v[0]();
  73.                     $v[1] ??= '__invoke';
  74.                 }
  75.                 if ($v === $listener || ($listener instanceof \Closure && $v == $listener)) {
  76.                     return $priority;
  77.                 }
  78.             }
  79.         }
  80.         return null;
  81.     }
  82.     public function hasListeners(?string $eventName null): bool
  83.     {
  84.         if (null !== $eventName) {
  85.             return !empty($this->listeners[$eventName]);
  86.         }
  87.         foreach ($this->listeners as $eventListeners) {
  88.             if ($eventListeners) {
  89.                 return true;
  90.             }
  91.         }
  92.         return false;
  93.     }
  94.     public function addListener(string $eventName, callable|array $listenerint $priority 0): void
  95.     {
  96.         $this->listeners[$eventName][$priority][] = $listener;
  97.         unset($this->sorted[$eventName], $this->optimized[$eventName]);
  98.     }
  99.     public function removeListener(string $eventName, callable|array $listener): void
  100.     {
  101.         if (empty($this->listeners[$eventName])) {
  102.             return;
  103.         }
  104.         if (\is_array($listener) && isset($listener[0]) && $listener[0] instanceof \Closure && >= \count($listener)) {
  105.             $listener[0] = $listener[0]();
  106.             $listener[1] ??= '__invoke';
  107.         }
  108.         foreach ($this->listeners[$eventName] as $priority => &$listeners) {
  109.             foreach ($listeners as $k => &$v) {
  110.                 if ($v !== $listener && \is_array($v) && isset($v[0]) && $v[0] instanceof \Closure && >= \count($v)) {
  111.                     $v[0] = $v[0]();
  112.                     $v[1] ??= '__invoke';
  113.                 }
  114.                 if ($v === $listener || ($listener instanceof \Closure && $v == $listener)) {
  115.                     unset($listeners[$k], $this->sorted[$eventName], $this->optimized[$eventName]);
  116.                 }
  117.             }
  118.             if (!$listeners) {
  119.                 unset($this->listeners[$eventName][$priority]);
  120.             }
  121.         }
  122.     }
  123.     public function addSubscriber(EventSubscriberInterface $subscriber): void
  124.     {
  125.         foreach ($subscriber->getSubscribedEvents() as $eventName => $params) {
  126.             if (\is_string($params)) {
  127.                 $this->addListener($eventName, [$subscriber$params]);
  128.             } elseif (\is_string($params[0])) {
  129.                 $this->addListener($eventName, [$subscriber$params[0]], $params[1] ?? 0);
  130.             } else {
  131.                 foreach ($params as $listener) {
  132.                     $this->addListener($eventName, [$subscriber$listener[0]], $listener[1] ?? 0);
  133.                 }
  134.             }
  135.         }
  136.     }
  137.     public function removeSubscriber(EventSubscriberInterface $subscriber): void
  138.     {
  139.         foreach ($subscriber->getSubscribedEvents() as $eventName => $params) {
  140.             if (\is_array($params) && \is_array($params[0])) {
  141.                 foreach ($params as $listener) {
  142.                     $this->removeListener($eventName, [$subscriber$listener[0]]);
  143.                 }
  144.             } else {
  145.                 $this->removeListener($eventName, [$subscriber\is_string($params) ? $params $params[0]]);
  146.             }
  147.         }
  148.     }
  149.     /**
  150.      * Triggers the listeners of an event.
  151.      *
  152.      * This method can be overridden to add functionality that is executed
  153.      * for each listener.
  154.      *
  155.      * @param callable[] $listeners The event listeners
  156.      * @param string     $eventName The name of the event to dispatch
  157.      * @param object     $event     The event object to pass to the event handlers/listeners
  158.      */
  159.     protected function callListeners(iterable $listenersstring $eventNameobject $event): void
  160.     {
  161.         $stoppable $event instanceof StoppableEventInterface;
  162.         foreach ($listeners as $listener) {
  163.             if ($stoppable && $event->isPropagationStopped()) {
  164.                 break;
  165.             }
  166.             $listener($event$eventName$this);
  167.         }
  168.     }
  169.     /**
  170.      * Sorts the internal list of listeners for the given event by priority.
  171.      */
  172.     private function sortListeners(string $eventName): void
  173.     {
  174.         krsort($this->listeners[$eventName]);
  175.         $this->sorted[$eventName] = [];
  176.         foreach ($this->listeners[$eventName] as &$listeners) {
  177.             foreach ($listeners as &$listener) {
  178.                 if (\is_array($listener) && isset($listener[0]) && $listener[0] instanceof \Closure && >= \count($listener)) {
  179.                     $listener[0] = $listener[0]();
  180.                     $listener[1] ??= '__invoke';
  181.                 }
  182.                 $this->sorted[$eventName][] = $listener;
  183.             }
  184.         }
  185.     }
  186.     /**
  187.      * Optimizes the internal list of listeners for the given event by priority.
  188.      */
  189.     private function optimizeListeners(string $eventName): array
  190.     {
  191.         krsort($this->listeners[$eventName]);
  192.         $this->optimized[$eventName] = [];
  193.         foreach ($this->listeners[$eventName] as &$listeners) {
  194.             foreach ($listeners as &$listener) {
  195.                 $closure = &$this->optimized[$eventName][];
  196.                 if (\is_array($listener) && isset($listener[0]) && $listener[0] instanceof \Closure && >= \count($listener)) {
  197.                     $closure = static function (...$args) use (&$listener, &$closure) {
  198.                         if ($listener[0] instanceof \Closure) {
  199.                             $listener[0] = $listener[0]();
  200.                             $listener[1] ??= '__invoke';
  201.                         }
  202.                         ($closure $listener(...))(...$args);
  203.                     };
  204.                 } else {
  205.                     $closure $listener instanceof WrappedListener $listener $listener(...);
  206.                 }
  207.             }
  208.         }
  209.         return $this->optimized[$eventName];
  210.     }
  211. }