Twig_SimpleFunction可以修改上下文吗?

Fyr*_*ase 5 php templates symfony twig

是否可以通过调用Twig_SimpleFunction来修改当前的twig上下文?

我注册了以下功能:

<?php
namespace Craft;

class TwiggedTwigExtension extends \Twig_Extension
{
    public function getName()
    {
      return 'Twigged';
    }

    public function getFunctions()
    {
        return array(
            'setContextVar' => new \Twig_SimpleFunction('setContextVar', array($this, 'setContextVar'), array('needs_context' => true)),
        );
    }

    public function setContextVar($context, $str, $val)
    {
        $context['context'][$str] = $val;

        var_dump(array_keys($context['context']));
    }
}
Run Code Online (Sandbox Code Playgroud)

从类似模板调用时{{ setContextVar('hellow', 'world') }},var_dump显示修改后的上下文.但是像这样快速检查模板{{ dump(_context|keys) }}并不会显示修改后的上下文.

我是以错误的方式来做这件事的吗?

Chr*_*oet 5

这对于函数来说是不可能的,因为上下文不是通过引用传递的。

在您的扩展中,您甚至正在访问$context['context'],这意味着一个名为 的变量context,而不是上下文本身(_context是用于访问模板中的上下文的特殊变量名称,但它没有出现在发送给函数的上下文数组中,因为它是数组本身)。

可能有一种方法可以通过更改节点类来自定义函数编译来执行此操作。但我还没有尝试过,维护表达式语义可能会有点困难。
无论如何,我建议不要编写这样的函数。在 Twig 中分配变量没有函数语义,不能作为表达式的一部分完成(函数当然可以在表达式中使用)。改变这个语义可能会导致奇怪的行为。


mar*_*ias 5

就像他评论中提到的@DarkBee 一样,您可以通过引用传递上下文来修改上下文:

function setContextVar(&$context, ...)
Run Code Online (Sandbox Code Playgroud)

但是,仅向 Twig 添加一个函数似乎不起作用(我使用的是 Twig 2.4.4):

$twig->addFunction(new Twig_Function('setContextVar', function(&$context, $name, $value) {
    $context[$name] = $value;
}, ['needs_context' => true]));
Run Code Online (Sandbox Code Playgroud)

当您在 Twig 文件中使用该函数时,上下文不会被修改,您会收到以下警告:

警告:参数 1 到 {closure}() 预期为参考,值在 C:\...\vendor\twig\twig\lib\Twig\Environment.php(378) 中给出:eval()'d code on线 ...

相反,您需要创建一个 Twig 扩展:

$twig->addExtension(new class extends Twig_Extension {
    public function getFunctions() {
        return [
            new Twig_Function('setContextVar', [$this, 'setContextVar'], ['needs_context' => true]),
        ];
    }

    public function setContextVar(&$context, $name, $value) {
        $context[$name] = $value;
    }
});
Run Code Online (Sandbox Code Playgroud)

然后你可以在 Twig 中使用它而不会收到警告:

{{ dump() }}
{% do setContextVar('foo', 'bar') %}
{{ dump() }}
Run Code Online (Sandbox Code Playgroud)

上面的打印例如:

array(0) {
}

array(1) {
  ["foo"]=>
  string(3) "bar"
}
Run Code Online (Sandbox Code Playgroud)