This repository has been archived on 2023-08-20. You can view files and clone it, but cannot push or open issues or pull requests.
symfony/src/Symfony/Bundle/FrameworkBundle/Templating/Engine.php

188 lines
5.8 KiB
PHP
Raw Normal View History

<?php
namespace Symfony\Bundle\FrameworkBundle\Templating;
use Symfony\Component\Templating\Engine as BaseEngine;
use Symfony\Component\Templating\Loader\LoaderInterface;
use Symfony\Component\OutputEscaper\Escaper;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\HttpFoundation\Response;
/*
* This file is part of the Symfony package.
*
* (c) Fabien Potencier <fabien.potencier@symfony-project.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
/**
* This engine knows how to render Symfony templates and automatically
* escapes template parameters.
*
* @author Fabien Potencier <fabien.potencier@symfony-project.com>
*/
class Engine extends BaseEngine
{
protected $container;
protected $escaper;
protected $level;
/**
* Constructor.
*
* @param ContainerInterface $container A ContainerInterface instance
* @param LoaderInterface $loader A loader instance
* @param array $renderers An array of renderer instances
* @param mixed $escaper The escaper to use (or false to disable escaping)
*/
public function __construct(ContainerInterface $container, LoaderInterface $loader, array $renderers = array(), $escaper = false)
{
$this->level = 0;
$this->container = $container;
$this->escaper = $escaper;
parent::__construct($loader);
2010-08-05 06:34:53 +01:00
foreach ($this->container->findTaggedServiceIds('templating.renderer') as $id => $attributes) {
if (isset($attributes[0]['alias'])) {
$this->renderers[$attributes[0]['alias']] = $id;
}
}
$this->helpers = array();
2010-08-05 06:34:53 +01:00
foreach ($this->container->findTaggedServiceIds('templating.helper') as $id => $attributes) {
if (isset($attributes[0]['alias'])) {
$this->helpers[$attributes[0]['alias']] = $id;
}
}
}
public function render($name, array $parameters = array())
{
++$this->level;
list(, $options) = $this->splitTemplateName($name);
$renderer = $options['renderer'];
if (isset($this->renderers[$renderer]) && is_string($this->renderers[$renderer])) {
$this->renderers[$renderer] = $this->container->get($this->renderers[$renderer]);
$this->renderers[$renderer]->setEngine($this);
}
if ('php' === $renderer) {
// escape only once
if (1 === $this->level && !isset($parameters['_data'])) {
$parameters = $this->escapeParameters($parameters);
}
}
$content = parent::render($name, $parameters);
--$this->level;
return $content;
}
2010-08-14 18:51:28 +01:00
/**
* Renders a view and returns a Response.
*
* @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 renderResponse($view, array $parameters = array(), Response $response = null)
{
if (null === $response) {
$response = $this->container->get('response');
}
$response->setContent($this->render($view, $parameters));
return $response;
}
public function has($name)
{
return isset($this->helpers[$name]);
}
/**
* @throws \InvalidArgumentException When the helper is not defined
*/
public function get($name)
{
if (!isset($this->helpers[$name])) {
throw new \InvalidArgumentException(sprintf('The helper "%s" is not defined.', $name));
}
if (is_string($this->helpers[$name])) {
$this->helpers[$name] = $this->container->get($this->helpers[$name]);
$this->helpers[$name]->setCharset($this->charset);
}
return $this->helpers[$name];
}
protected function escapeParameters(array $parameters)
{
if (false !== $this->escaper) {
Escaper::setCharset($this->getCharset());
$parameters['_data'] = Escaper::escape($this->escaper, $parameters);
foreach ($parameters['_data'] as $key => $value) {
$parameters[$key] = $value;
}
} else {
$parameters['_data'] = Escaper::escape('raw', $parameters);
}
return $parameters;
}
// parses template names following the following pattern:
2010-10-04 13:01:33 +01:00
// bundle:section:template(.format).renderer
public function splitTemplateName($name, array $defaults = array())
{
$parts = explode(':', $name);
if (3 !== count($parts)) {
throw new \InvalidArgumentException(sprintf('Template name "%s" is not valid.', $name));
}
$options = array_replace(
array(
'format' => '',
),
$defaults,
array(
'bundle' => str_replace('\\', '/', $parts[0]),
'controller' => $parts[1],
)
);
$elements = explode('.', $parts[2]);
if (3 === count($elements)) {
$parts[2] = $elements[0];
if ('html' !== $elements[1]) {
$options['format'] = '.'.$elements[1];
}
$options['renderer'] = $elements[2];
} elseif (2 === count($elements)) {
$parts[2] = $elements[0];
$options['renderer'] = $elements[1];
$format = $this->container->get('request')->getRequestFormat();
if (null !== $format && 'html' !== $format) {
$options['format'] = '.'.$format;
}
} else {
throw new \InvalidArgumentException(sprintf('Template name "%s" is not valid.', $name));
}
return array($parts[2], $options);
}
}