twig函数和pass-by-reference变量

时间:2012-08-29 11:57:45

标签: php function templates pass-by-reference twig

情况如下:

{% for entry in entries %}
{{ action('entry_modify', entry) }}
{{ entry.title }}
{% endfor %}

action($ action,$ params)方法是为自己注册某个动作的插件的某种触发器。在这种情况下,我调用'entry_modify'操作,插件使用'entry'参数响应此操作,该参数是一个数组。

function plugin(&$params)
{
    $params['title'] = 'changed to ...'; 
}

但据我所知,Twig不是通过引用传递变量,有没有办法做到这一点?

我想要的只是修改传递给动作函数的变量。

...谢谢

2 个答案:

答案 0 :(得分:1)

Twig函数总是通过引用传递对象。您可以考虑使用对象而不是数组来表示每个条目。

答案 1 :(得分:1)

我可能会为您的问题找到解决方案。我正在用Twig开发一个Wordpress主题,我想通过引用传递变量并向它们添加一些数据。 我的目标是制作这样的东西

{{ load_data('all_posts',foo) }}

我想执行一个名为load_all_posts的函数,并将此函数的结果赋给foo(我作为第二个参数传递的twig变量)。 此外,我希望以下内容也有效:

{% set foo = {'bar':'bar'} %}

{#foo is#}
array(1) {
  ["bar"]=>
  string(3) "bar"
}

{{load_data('all_posts',foo.posts)}}

{#foo is#}
array(2) {
  ["bar"]=>
  string(3) "bar"
  ["posts"]=>
  array(3) {
    [0]=>
    string(5) "post1"
    [1]=>
    string(5) "post2"
    [2]=>
    string(5) "post3"
  }
}

由于并非所有变量都通过引用传递给函数。我们不能为此目的使用函数。相反,我创建了自己的标签并像这样使用它:

{% set foo = {'bar':'bar'} %}
{% load all_pots in foo.posts %}

要做到以下几点,我们必须做两件事:

  • 创建“加载”令牌
  • 调用正确的函数并将数据加载到正确的位置

创建令牌解析器:

令牌的创建很简单。我们只需告诉twig如何使用以下类解析此标记:

class Load_Data_TokenParser  extends Twig_TokenParser
{
    public function parse(Twig_Token $token)
    {
        $stream = $this->parser->getStream();
        $variable_array = array(); //an array where we'll store the destination variable, as it can be just foo or foo.post, or foo.bar.foo.bar.posts

        $function_name = $stream->expect(Twig_Token::NAME_TYPE)->getValue();  //here we say that we expect a name. In our case it'll be 'all_posts' (the name of the function we will execute)
        $in = $stream->expect(Twig_Token::OPERATOR_TYPE)->getValue();   //here we say that the next thing should be the operator 'in'
        while(Twig_Token::typeToEnglish($stream->getCurrent()->getType()) === 'name'){ //and with this while, we get all the variable names that are separated with '.'
            array_push($variable_array,$stream->getCurrent()->getValue());

            $stream->next();
            $new  = $stream->getCurrent();

            if(Twig_Token::typeToEnglish($new->getType()) === 'punctuation'){
                $stream->next();
            } else {
                break;
            }
        }
        //in our case $variable_array here will be ['foo','posts']

        $stream->expect(Twig_Token::BLOCK_END_TYPE);

        return new Load_Node($variable_array, $function_name, $token->getLine(), $this->getTag());
    }

    public function getTag()
    {
        return 'load';
    }
}

最后我们返回一个新的“Load_Node”并将$variable_array, $function_name行和当前标记传递给它。

加载数据

class Load_Node extends Twig_Node
{
    public function __construct($variable_array, $function_name, $line, $tag = null)
    {
        $value = new Twig_Node();
        parent::__construct(array('value'=>$value),array(
            'variable_array'=>$variable_array,
            'function_name'=>$function_name
        ),$line,$tag);
    }

    public function compile(Twig_Compiler $compiler)
    {
        $variable_access_string = '';
        $variable_array = $this->getAttribute('variable_array');
        foreach ($variable_array as $single_variable) {
            $variable_access_string.= '[\''.$single_variable.'\']';
        } //here we prepare the string that will be used to access the correct variables. In our case here $variable_access_string will be "['foo']['posts']"

        $compiler
            ->addDebugInfo($this)
            ->write('$context'.$variable_access_string.' = load_'.$this->getAttribute('function_name').'()')
            ->raw(";\n"); //and the here we call load_{function_name} and assign the result to the correct variable
    }
}

目前它仅适用于数组。如果您尝试在对象内部加载数据,则会引发异常。我稍后会编辑它并添加对象的支持。 我希望这会对你和其他人有所帮助。 祝你有愉快的一天:)