Twig не позволяет сделать это напрямую. Вы можете либо добавить в Twig простую функцию для выполнения замыканий, либо обернуть замыкание в класс, чтобы иметь возможность использовать атрибутивную функцию Twig (поскольку прямой вызов attribute(_context, 'myclosure', args)
вызовет фатальную ошибку, поскольку Twig вернет замыкание напрямую и игнорировать заданные аргументы, поскольку _context
— это массив).
Простое расширение Twig, которое достигает этой цели, будет выглядеть так для Symfony 2.8+. (Для Symfony 4 см. новую документацию)
// src/AppBundle/Twig/Extension/CoreExtensions.php
namespace AppBundle\Twig\Extension;
class CoreExtensions extends \Twig_Extension
{
public function getFunctions()
{
return [
new \Twig_SimpleFunction('execute', [$this, 'executeClosure'])
];
}
public function executeClosure(\Closure $closure, $arguments)
{
return $closure(...$arguments);
}
public function getName()
{
return 'core_extensions_twig_extension';
}
}
Затем в ваших шаблонах вам просто нужно вызвать execute:
{{ execute(closure, [argument1, argument2]) }}
Без расширения Twig один из способов обойти эту проблему — использовать класс, который действует как оболочка для вашего замыкания, и использовать attribute
функции Twig, поскольку ее можно использовать для вызова метода объекта.
// src/AppBundle/Twig/ClosureWrapper.php
namespace AppBundle\Twig;
/**
* Wrapper to get around the issue of not being able to use closures in Twig
* Since it is possible to call a method of a given object in Twig via "attribute",
* the only purpose of this class is to store the closure and give a method to execute it
*/
class ClosureWrapper
{
private $closure;
public function __construct($closure)
{
$this->closure = $closure;
}
public function execute()
{
return ($this->closure)(...func_get_args());
}
}
Затем вам просто нужно передать экземпляр ClosureWrapper вашему шаблону при рендеринге вместо самого замыкания:
use AppBundle\Twig\ClosureWrapper;
class MyController extends Controller
{
public function myAction()
{
$localValue = 2;
$closure = new ClosureWrapper(function($param1, $param2) use ($localValue) {
return $localValue + $param1 + $param2;
});
return $this->render('mytemplate.html.twig', ['closure' => $closure]);
}
...
В конце концов, в вашем шаблоне вам нужно использовать attribute
для выполните закрытие, которое вы определили в своем контроллере:
// Displays 12
{{ attribute(closure, 'execute', [4, 6]) }}
Однако это немного избыточно, поскольку внутренне функция Twig attribute
распаковывает также приведены аргументы. Используя приведенный выше код, для каждого вызова аргументы последовательно распаковываются, упаковываются и снова распаковываются.
person
Alan T.
schedule
24.12.2017