目前我将我的函数放在一个类中,并将此类的实例传递给模板,并将我所需的函数作为类方法调用。
{{ unneededclass.blah() }}
我需要像下面这样做
{{ blah() }}
有可能吗?
答案 0 :(得分:32)
评论者指出,我大多错了。如果您确实需要一个函数,而不是过滤器或宏,则可以将其作为suggested in the Twig docs:
$twig = new Twig_Environment($loader);
$function = new Twig_SimpleFunction('blah', function () {
// ...
});
$twig->addFunction($function);
并使用
{{ blah() }}
简而言之,不,这是不可能的。
然而,希望不会丢失!
如果您的blah()
函数用于修改现有变量,则它是过滤器。
一个例子:
//in your PHP
function format_date($date_string,$format_string) {
return date($format_string,strtotime($date_string));
}
$twig_env->addFilter('format_date',new Twig_Filter_Function('format_date'));
{# in your template #}
{{ some_date|format_date('n/j/Y') }}
(第一个参数是您要过滤的变量,第二个参数是通过常规方式提供的)
如上所述,如果您的函数只输出HTML,那么它是宏的理想选择。
一个例子:
{# in your template #}
{% macro say_hello() %}
<p>Oh! Hello, world!</p>
{% endmacro %}
{# ... later on ... #}
{{ _self.say_hello() }}
或带参数:
{% macro input(name,value,type) %}
<input type="{{ type|default('text') }}" name="{{ name }}" value="{{ value }}">
{% endmacro %}
{{ _self.input('phone_number','867-5309') }}
{{ _self.input('subscribe','yes','checkbox') }}
要记住的是,根据MVC,Twig模板代表视图。这意味着它们在环境方面是孤立的,并且只能代表您通过$template->render()
方法传递的数据数组传递的上下文。
这是一件好事,因为它将您的演示文稿与逻辑和数据分离。如果你可以任意调用函数,那么你会突然增加耦合,这是坏的事情。
另一个原因是PHP处理回调的方式。考虑一下如何将该功能传递到模板中......可能是这样的:
function blah() {
return "<p>Oh! Hello, world!</p>";
}
$template = $twig_env->loadTemplate('template.html');
echo $template->render(array('blah'=>'blah'));
在模板中,上下文变量 blah
现在只是一个包含'blah'
的字符串。
在vanilla PHP中,当你使用像这样的变量函数时(尝试使用像函数一样的字符串变量),它(或多或少)执行查找该函数,然后调用它。你不传递函数,只是它的名字。
问题是,您不可能将函数传递给模板,因为PHP执行此操作的唯一机制是通过名称字符串,并且一旦在模板内,该名称不再是函数名称而只是字符串。击>
有点长的啰嗦,但我希望有所帮助!
如果您需要更多文档,官方文档为here。
答案 1 :(得分:6)
我和你的朋友一样迷茫,但在网上搜索答案后却没有找到答案,我决定看看自己能不能做到。因此,我想在这里发布我的解决方案(尽管我知道这篇文章已经过时了),因为如果你搜索这个问题,这是谷歌的第一次点击。
实际上,实际上非常简单:
我创建了一个包含我的函数和变量的类,例如:
class functionContainer{
function getRandomNumber()
{
return rand();
}
}
$values = array(
'functions'=> new functionContainer()
);
所以现在我们将$ value作为一个数组,其中包含一个带有函数“getRandomNumber()”的对象。
渲染模板文件时,请将此类包含为值:
$twig->render('random.html', $values);
这样,在模板文件中,您可以调用此方法来调用该函数并获得结果:
{{ functions.getRandomNumber }}
答案 2 :(得分:6)
虽然你不能直接PHP调用,但twig是可扩展的。您可以添加可调用的过滤器,以便可以应用于传递给模板的PHP函数。
namespace My\Twig\Extension;
class LambdaFilter extends \Twig_Extension {
public function getName() {
return 'lambda_filter';
}
public function getFilters() {
return array(
new \Twig_SimpleFilter('call', array($this, 'doCall'))
);
}
public function doCall() {
$arguments = func_get_args();
$callable = array_shift($arguments);
if(!is_callable($callable)) {
throw new InvalidArgumentException();
}
return call_user_func_array($callable, $arguments);
}
}
现在,如果您将变量my_func
传递给模板,则可以执行my_func|call(arg1, arg2)
。您甚至可以执行更高阶函数"array_filter"|call(my_array, my_func)
,并且您可以在过滤器中执行更多操作,例如将数组作为参数接受等等。
答案 3 :(得分:2)
完整答案: http://twig.sensiolabs.org/doc/advanced.html#id2
我更喜欢像这样使用Twig Extension:
namespace Some\Twig\Extensions;
class MenuExtensions extends \Twig_Extension
{
public function getFunctions()
{
return array(
new \Twig_SimpleFunction('sidebar_menu', [$this, 'generate_sidebar_menu']),
);
}
public function generate_sidebar_menu($menu){
return $menu;
}
public function getName()
{
return 'menu';
}
}
在模板中:
{{ sidebar_menu('some text') }}
答案 4 :(得分:1)
在manager / controller / service中创建匿名类..
$functions = new class($router)
{
public function __construct($router)
{
$this->router = $router;
}
public function getRowUrl(FileManager $fileManager)
{
if ($fileManager->isNode()) {
return $this->router->generate('...', ['parent' => ...]);
}
return $this->router->generate('...', ['entity' => ...]);
}
};
将参数粘贴到视图中
$params=[
'functions' => $functions
];
return new Response($this->twig->render('...:index.html.twig', $params));
在视图中使用功能
{% set rowUrl = functions.rowUrl(entity) %}