ZF3 - событие, прикрепленное к EventManager, не запускается - PullRequest
1 голос
/ 17 мая 2019

У меня есть класс EntityRepository, и я хочу прикрепить событие к его методу "save". Вот класс AbstractRepository, в котором я устанавливаю идентификатор менеджера событий:

abstract class AbstractRepository extends EntityRepository implements RepositoryInterface
{
    /**
     * @var EventManagerInterface
     */
    protected $events;

    /**
     * Set the event manager instance used by this context.
     *
     * @param EventManagerInterface $events
     *
     * @return mixed
     */
    public function setEventManager(EventManagerInterface $events)
    {
        $events->setIdentifiers([
            __CLASS__,
            get_class($this),
        ]);

        $this->events = $events;

        return $this;
    }

    /**
     * Retrieve the event manager.
     *
     * Lazy-loads an EventManager instance if none registered.
     *
     * @return EventManagerInterface
     */
    public function getEventManager()
    {
        if (!$this->events) {
            $this->setEventManager(new EventManager());
        }

        return $this->events;
    }
}

и класс хранилища под названием DocumentRepository:

class DocumentsRepository extends AbstractRepository
{
    public function save(Documents $entity)
    {
        $this->getEventManager()->trigger(RepositoryInterface::EVENT_BEFORE_SAVE);

        $this->getEntityManager()->persist($entity);
        $this->getEntityManager()->flush($entity);

        $this->getEventManager()->trigger(RepositoryInterface::EVENT_AFTER_SAVE);

        return $entity;
    }
}

Также интерфейс для определения имен событий:

interface RepositoryInterface extends EventManagerAwareInterface
{
    const EVENT_BEFORE_SAVE   = 'before.save';
    const EVENT_AFTER_SAVE    = 'after.save';
}

Чтобы зарегистрировать слушателя, я добавил следующий код в module.php.

public function onBootstrap(MvcEvent $event)
{
    $application    = $event->getApplication();
    $eventManager   = $application->getEventManager();

    /* Register event listener(s) */
    (new DocumentsRepositoryListener($serviceManager))->attach($eventManager);
}

И, наконец, добавил класс слушателя:

class DocumentsRepositoryListener extends AbstractListener implements ListenerAggregateInterface
{
    public function attach(EventManagerInterface $events, $priority = 1)
    {
        $sharedEvents = $events->getSharedManager();

        $this->listeners[] = $sharedEvents
            ->attach(DocumentsRepository::class, RepositoryInterface::EVENT_BEFORE_SAVE, [$this, 'beforeSave']);
        $this->listeners[] = $sharedEvents
            ->attach(DocumentsRepository::class, RepositoryInterface::EVENT_AFTER_SAVE, [$this, 'afterSave']);
    }

    public function detach(EventManagerInterface $events)
    {
        foreach ($this->listeners as $index => $listener) {
            if ($events->detach($listener)) {
                unset($this->listeners[$index]);
            }
        }
    }

    public function beforeSave(EventInterface $event)
    {
        /* Something to do before saving */
    }

    public function afterSave(EventInterface $event)
    {
        /* Something to do after saving */
    }
}

Класс AbstractListener также содержит конструктор для внедрения методов serviceLocator и setter / getter.

Теперь я не знаю, почему триггер не работает. Есть что-нибудь еще, что может быть пропущено !?

1 Ответ

3 голосов
/ 18 мая 2019

Код, который вы записали в , вызывает , события в DocumentsRepository не имеют проблем. Однако у вас есть недоразумение относительно использования менеджера событий; в частности, как вы присоединяете слушателей событий.

Первое, что нужно отметить, это то, что вы дали классу репозитория собственный экземпляр EventManager; this - это экземпляр, к которому вы должны присоединить своих слушателей событий. Код в Module::onBootstrap() регистрируется в диспетчере событий application , что является неправильным экземпляром. Менеджер событий приложения работает с событиями приложения, поэтому он никогда не будет запускать ваши пользовательские события.

/* Register event listener(s) */
(new DocumentsRepositoryListener($serviceManager))->attach($eventManager);

Будет заменен на:

$documentRepositoryListener->attach($repository->getEventManager());

Логика для регистрации слушателей может лучше подходить в RepositoryFactory, например:

class RepositoryFactory implements FactoryInterface
{
    public function __invoke(ContainerInterface $container, $requestedName, array $options = null)
    {       
        $repository = new Repository();

        if ($repository instanceof EventManagerAwareInterface) {
            $eventManager = $repository->getEventManager();
            $listener = $container->get('Foo/Event/Listener');

            $listener->attach($eventManager);
        }

        return $repository;
    }
}

Наконец, вам также нужно будет удалить использование $sharedEvents в DocumentsRepositoryListener::attach(); поскольку вы хотите присоединить прослушиватели событий к менеджеру событий репозитория, который будет аргументом $events.

...