我在kernel.view
事件监听器的帮助下分离了移动和网络请求。
逻辑的工作原理如下:
xxx.mobile.twig
xxx.html.twig
这对我的CustomBundle没有任何问题。除此之外,我正在使用FOSUserBundle和HWIOAuthBundle以及它们的一些路线。我检查了var/logs/dev.log
,我看不到有关这些捆绑路由的kernel.view
个事件,最终我的监听器无法使用这些捆绑包。
您能否告诉我如何为这些捆绑包绑定kernel.view
事件?
/**
* @param GetResponseForControllerResultEvent $event
* @return bool
*/
public function onKernelView(GetResponseForControllerResultEvent $event)
{
if (!$this->isMobileRequest($event->getRequest()->headers->get('user-agent'))) {
return false;
}
$template = $event->getRequest()->attributes->get('_template');
if (!$template) {
return false;
}
$templateReference = $this->templateNameParser->parse($template);
if ($templateReference->get('format') == 'html' && $templateReference->get('bundle') == 'CustomBundle') {
$mobileTemplate = sprintf(
'%s:%s:%s.mobile.twig',
$templateReference->get('bundle'),
$templateReference->get('controller'),
$templateReference->get('name')
);
if ($this->templating->exists($mobileTemplate)) {
$templateReference->set('format', 'mobile');
$event->getRequest()->attributes->set('_template', $templateReference);
}
}
}
答案 0 :(得分:2)
在Symfony2上调试与事件相关的问题时,您应该考虑以下几点。
可以停止事件传播
$event->stopPropagation()
来停止事件传播。在这种情况下,请确保首先执行您的侦听器(请参阅下面的第2点)。事件监听器具有优先级
view_response_listener:
class: AppBundle\EventListener\ViewResponseListener
tags:
# The highest the priority, the earlier a listener is executed
# @see http://symfony.com/doc/2.7/cookbook/event_dispatcher/event_listener.html#creating-an-event-listener
- { name: kernel.event_listener, event: kernel.view, method: onKernelView, priority: 101 }
另一个可选标记属性称为优先级,默认为0,它控制执行侦听器的顺序(优先级最高,执行侦听器的时间越早)。当您需要保证一个侦听器在另一个侦听器之前执行时,这非常有用。内部Symfony侦听器的优先级通常在-255到255之间,但您自己的侦听器可以使用任何正整数或负整数。
来源:http://symfony.com/doc/2.7/cookbook/event_dispatcher/event_listener.html#creating-an-event-listener
通常,这些事件的分派是在引导程序文件
中完成的
composer run-script post-update-cmd
php app/console cache:clear --env=dev
作曲家post-update-cmd
将重新生成你的引导程序文件,但它也会做其他事情,比如重新安装资产,这可能是你不需要的东西。要重新生成引导程序文件,请检查我的答案here。
答案 1 :(得分:0)
我找到了解决方案,但它有点解决方法,现在正常工作。
我将以下功能放到我的MobileTemplateListener.php
文件中。
更多细节在这里 - > http://www.99bugs.com/handling-mobile-template-switching-in-symfony2/
/**
* @param GetResponseEvent $event
*/
public function onKernelRequest(GetResponseEvent $event)
{
$request = $event->getRequest();
if ($this->isMobileRequest($request->headers->get('user-agent')))
{
//ONLY AFFECT HTML REQUESTS
//THIS ENSURES THAT YOUR JSON REQUESTS TO E.G. REST API, DO NOT GET SERVED TEXT/HTML CONTENT-TYPE
if ($request->getRequestFormat() == "html")
{
$request->setRequestFormat('mobile');
}
}
}
/**
* Returns true if request is from mobile device, otherwise false
* @return boolean mobileUA
*/
private function isMobileRequest($userAgent)
{
if (preg_match('/(android|blackberry|iphone|ipad|phone|playbook|mobile)/i', $userAgent)) {
return true;
}
return false;
}
当kernel.request
事件侦听器开始处理时,它正在设置值为mobile
的格式
我通过子捆绑使用FOSUserBundle来操作我的需求。您可以在此处找到更多详细信息 - > https://github.com/FriendsOfSymfony/FOSUserBundle/blob/master/Resources/doc/overriding_controllers.md
例如:我们假设SecurityController.php
我在UserBundle下创建了一个名为SecurityController.php的文件。它看起来像是在跟随。
<?php
namespace Acme\UserBundle\Controller;
use Symfony\Component\HttpFoundation\RedirectResponse;
use FOS\UserBundle\Controller\SecurityController as BaseController;
use Symfony\Component\HttpFoundation\Request;
use Acme\UserBundle\Overrides\ControllerOverrideRenderTrait;
class SecurityController extends BaseController
{
use ControllerOverrideRenderTrait;
public function loginAction(Request $request)
{
$securityContext = $this->container->get('security.context');
$router = $this->container->get('router');
if ($securityContext->isGranted('IS_AUTHENTICATED_FULLY')) {
return new RedirectResponse($router->generate('my_profile_dashboard'), 307);
}
return parent::loginAction($request);
}
}
对于所有其他FOS控制器,我必须覆盖渲染功能。这是由smyfony Symfony\Bundle\FrameworkBundle\Controller
但是我的子包已经扩展了FOSUserBundle的控制器,在没有重复代码的情况下覆盖它的唯一方法就是使用特征。
我创建了一个特征如下。
<?php
namespace Acme\UserBundle\Overrides;
use Symfony\Component\HttpFoundation\Response;
trait ControllerOverrideRenderTrait {
/**
* This overrides vendor/symfony/symfony/src/Symfony/Bundle/FrameworkBundle/Controller/Controller.php
* Renders a view.
*
* @param string $view The view name
* @param array $parameters An array of parameters to pass to the view
* @param Response $response A response instance
*
* @return Response A Response instance
*/
public function render($view, array $parameters = array(), Response $response = null)
{
$format = $this->getRequest()->getRequestFormat();
$view = str_replace('.html.', '.' . $format . '.', $view);
return $this->container->get('templating')->renderResponse($view, $parameters, $response);
}
}
symfony提供的原始功能如下。
/**
* Renders a view.
*
* @param string $view The view name
* @param array $parameters An array of parameters to pass to the view
* @param Response $response A response instance
*
* @return Response A Response instance
*/
public function render($view, array $parameters = array(), Response $response = null)
{
return $this->container->get('templating')->renderResponse($view, $parameters, $response);
}
基本上,我的更改通过提供'.html.'
通过$format
EventListener设置来替换模板名称中的onKernelRequest
部分。
不要忘记在services.yml
中添加服务定义services:
acme.frontend.listener.mobile_template_listener:
class: Acme\FrontendBundle\EventListener\MobileTemplateListener
arguments: ['@templating', '@templating.name_parser']
tags:
- { name: kernel.event_listener, event: kernel.view, method: onKernelView }
- { name: kernel.event_listener, event: kernel.request, method: onKernelRequest }