I am trying to send var to view from event listener in symfony2 but I am stacked.
1) Is this possible?
2) What kernel event to use (kernel.view or kernel.request)?
3) Finally how to send some var to view?
My best guess is that I have to extend return from controller and then to let controller do his job.
I need to send some array of objects (entities).
I see several ways to handle this.
kernel.request listenerThe idea is to add a global variable straight after the kernel.request event.  
services.yml
services:
    class: Acme\FooBundle\Listener\MyListener
    arguments:
        - @twig
    tags:
        -
            name: kernel.event_listener
            event: kernel.request
            method: onKernelRequest
MyListener
class MyListener
{
    protected $twig;
    public function __construct(\Twig_Environment $twig)
    {
        $this->twig = $twig;
    }
    public function onKernelRequest(GetResponseEvent $event)
    {
        $myVar = 'foo'; // Process data
        $this->twig->addGlobal('myvar', $myVar);
    }
}
You can now use it at any time by doing
{{ myvar }}
kernel.view listenerFirst, you need to understand when kernel.view is called. It's only called when the return of the controller is not an instance of Response object.
That said, doing 
// Acme/FooBundle/FooController#fooAction
return $this->render();
returns a Response object, so kernel.view is not called.
The idea is to make all controller returns an array of data, just like @Template requirements.
// Acme/FooBundle/FooController#fooAction
return array(
    'template' => 'AcmeFooBundle:Foo:foo.html.twig',
    'data' => array(
        'entity' => $entity
    )
);
Since you already have your service definition, you just need to add some requirements in your service declaration.
You need the @templating service to render the data.
You need to set itself as a kernel.view listener
// Acme/FooBundle/Resources/config/services.yml
services:
    acme_foo.my_listener:
        class: Acme\FooBundle\Listener\MyListener
        arguments:
            - @templating
        tags:
            -
                name: kernel.event_listener
                event: kernel.request
                method: onKernelRequest
            -
                name: kernel.event_listener
                event: kernel.view
                method: onKernelView
// Acme/FooBundle/Listener/MyListener.php
use Symfony\Component\Templating\EngineInterface;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\HttpKernel\Event\GetResponseEvent;
use Symfony\Component\HttpKernel\Event\GetResponseForControllerResultEvent;
class MyListener
{
    protected $templating;
    protected $myVar;
    public function __construct(EngineInterface $templating)
    {
        $this->templating = $templating;
    }
    public function getMyVar()
    {
        return $this->myVar;
    }
    public function onKernelRequest(GetResponseEvent $event)
    {
        $this->myVar = ""; // Process MyVar data
    }
    public function onKernelView(GetResponseForControllerResultEvent $event)
    {
        $result = $event->getControllerResult();
        if (null === $this->myVar || !isset($result['template']) || !isset($result['data'])) {
            return;
        }
        $data = array_merge($result['data'], array('myvar' => $this->myVar));
        $rendered = $this->templating->render($result['template'], $data);
        $event->setResponse(new Response($rendered));
    }
}
And there you are. The listener is creating a new response, adding your custom definition of myvar to any template rendered by him.
An alternative is to create a TWIG extension. In the following example, I'm assuming the MyListener definition is the same as above.
As per the documentation given above, you just have to create a simple extension class.
// Acme/FooBundle/Resources/config/services.yml
services:
    acme_foo.my_listener:
        class: Acme\FooBundle\Listener\MyListener
        tags:
            - { name: kernel.event_listener, event: kernel.request, method: onKernelRequest }
    acme_foo.my_extension:
        class: Acme\FooBundle\Extension\MyExtension
        arguments:
            - @acme_foo.my_listener
        tags:
            - { name: twig.extension }
Just like in documentation, we'll create a simple function.
// Acme/FooBundle/Extension/MyExtension.php
use Acme\FooBundle\Listener\MyListener;
class MyExtension extends \Twig_Extension
{
    protected $listener;
    public function __construct(MyListener $listener)
    {
        $this->listener = $listener;
    }
    public function getName()
    {
        return 'my_extension';
    }
    public function getFunctions()
    {
        return array(
            'myvar' => new \Twig_Function_Method($this, 'getMyVar')
        );
    }
    public function getMyVar()
    {
        return $this->listener->getMyVar();
    }
}
Then you can use it in any view by doing
{{ myvar() }}
I don't like this idea, but this is an alternative. You just have to create a BaseController which will override the default render method.
// Acme/FooBundle/Controller/BaseController.php
abstract class BaseController extends Controller
{
    public function render($view, array $parameters = array(), Response $response = null)
    {
        $parameters = array_merge(
            $parameters,
            array(
                'myvar' => $this->get('my_listener')->getMyVar()
            )
        );
        return parent::render($view, $parameters, $response);
    }
}
                        There's an alternative method here that I've had to do. I wanted to get some data, run it through json_encode(), then add that as a JavaScript variable to the response. Here's what I ended up doing.
I'm subscribing to kernel.response:
public static function getSubscribedEvents()
{
    return [
        KernelEvents::RESPONSE => 'onKernelResponse'
    ];
}
public function onKernelResponse(FilterResponseEvent $event)
{
    /** -- SNIP -- Cutting out how I get my serialised data **/
    $serialized = json_encode($data);
    /** Shove it into the response as some JS at the bottom **/
    $dom = new \DOMDocument;
    libxml_use_internal_errors(true);
    $dom->loadHTML($event->getResponse()->getContent());
    libxml_use_internal_errors(false);
    $node = $dom->createElement('script', "var data = $serialized;");
    $dom->getElementsByTagName('body')->item(0)->appendChild($node);
    $event->getResponse()->setContent($dom->saveHTML());
}
This is one way of doing it. Honestly, I don't like any of the methods described on this page. There should be a better way, but there isn't. This is what I'm using, though, and it works well. Just make sure you don't call your variable "data"; use something that won't be taken up elsewhere and preferably shove it in it's own (function() { } JS namespace.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With