Мне интересно, как лучше всего создать полностью совместимое приложение с плагинами.
Я привык к концепции подключаемых модулей Wordpress, что вы можете определять действия и фильтры, а затем использовать их в своих подключаемых модулях. Таким образом, другие могут определять методы своих плагинов, которые выполняются при вызове действия (или фильтра).
Моя идея состоит в том, чтобы создать мое приложение с некоторыми действиями и фильтрами, а затем другие разработчики могут создать пакет, который будет мешать «нормальному» потоку приложения...
Я читал об внедрении зависимостей Symfony2, но не нашел подробного примера, чтобы сделать что-то похожее на то, что я хочу.
- У кого-то есть реальный пример чего-то подобного, что я ищу?
- Является ли Dependency Injection лучшим решением или мне следует создать собственный обработчик плагинов?
РЕДАКТИРОВАТЬ:
Что я сделал, чтобы другие пакеты могли добавлять элементы в мое меню knp-menu.
В моем базовом комплекте:
Определение фильтра, позволяющего подписчику получать и устанавливать данные меню:
# BaseBundle/Event/FilterMenuEvent.php
class FilterMenuEvent extends Event
{
protected $menu;
public function __construct($menu)
{
$this->menu = $menu;
}
public function getMenu()
{
return $this->menu;
}
}
Определение событий меню:
# Event/MenuEvents.php
final class MenuEvents
{
const BEFORE_ITEMS = 'menu.before.items';
const AFTER_ITEMS = 'menu.after.items';
}
Настройка абонента:
# Event/MenuSubscriber.php
class MenuSubscriber implements EventSubscriberInterface
{
public static function getSubscribedEvents()
{
return array(
'menu.after.items' => array(
array('homeItems', 9000),
array('quickactionsItems', 80),
array('adminItems', 70),
...
array('logoutItems', -9000),
)
);
}
public function homeItems(FilterMenuEvent $menu_filter)
{
$menu = $menu_filter->getMenu();
$menu->addChild('Home', array('route' => 'zashost_zaspanel_homepage'));
}
public function quickactionsItems(FilterMenuEvent $menu_filter)
{
$menu = $menu_filter->getMenu();
$menu->addChild('Quick actions', array( 'route' => null));
$menu['Quick actions']->addChild('Add hosting', array( 'route' => 'zashost_zaspanel_register_host'));
}
}
Диспетчеризация событий при генерации меню:
# Menu\Builder.php
class Builder extends ContainerAware
{
public function userMenu(FactoryInterface $factory, array $options)
{
$menu = $factory->createItem('root');
$this->container->get('event_dispatcher')->dispatch(MenuEvents::AFTER_ITEMS , new FilterMenuEvent($menu));
return $menu;
}
}
Прикрепите подписчика к подписчику событий ядра:
# services.yml
services:
# Menu items added with event listener
base_menu_subscriber:
class: Acme\BaseBundle\Event\MenuSubscriber
arguments: ['@event_dispatcher']
tags:
- {name: kernel.event_subscriber}
Затем в стороннем комплекте:
Настройка моего стороннего подписчика событий:
class MenuSubscriber implements EventSubscriberInterface
{
public static function getSubscribedEvents()
{
return array(
'menu.after.items' => array('afterItems', 55)
);
}
public function afterItems(FilterMenuEvent $menu_filter)
{
$menu = $menu_filter->getMenu();
$menu->addChild('Backups', array( 'route' => null));
$menu['Backups']->addChild('Create new backup', array( 'route' => null));
return $menu;
}
}
И присоединение к подписчику событий ядра:
# srevices.yml
services:
menu_subscriber:
class: Acme\ThirdPartyBundle\Event\MenuSubscriber
arguments: ['@event_dispatcher']
tags:
- {name: kernel.event_subscriber}
Таким образом, я могу использовать приоритет диспетчера событий для установки позиции каждой группы пунктов меню.