如何检测表单是否还有要呈现的元素?

时间:2016-07-28 14:43:20

标签: php symfony twig symfony-forms

我们假设我使用Symfony Form Component呈现一个只有两个字段(用户名和电子邮件)的简单表单

我使用form_row()呈现两个字段以添加自定义css,然后调用form_widget()呈现其余元素(另一个选项为form_rest()) 我正在寻找的是一种预先检查form_widget()是否会打印任何对象的方法,以便在有额外字段的情况下添加自定义html。 我这样做的方式是这样的:

//app/form/index.html.twig
{# Other template code #}
...
{# Rendering form: #}
{{ form_start(form) }}

{{ form_row(form.username) }}
{{ form_row(form.email) }}

{% set form_rendered = form_widget(form) %}
{% if form_rendered %}
    <h3>Other fields</h3>
    {{ form_rendered | raw }}
{% endif %}

{{ form_end(form) }}
{# End form #}
...
{# Other template code #}

然而,我对此并不满意。还有更好的办法吗?

编辑:当使用CSRF保护(默认情况下已激活)时,之前的代码将始终打印<h3>Other fields</h3>,因为表单有一个额外的隐藏字段用于我们没有的令牌#39 ;打印。我们需要使用{{ form_row(form._token) }}将其呈现在某处。

1 个答案:

答案 0 :(得分:3)

通过检查src/Symfony/Component/Form/FormView.php中的代码,我找到了一个完全符合我想要的功能。

//src/Symfony/Component/Form/FormView.php

    /**                                                                  
     * Returns whether the view was already rendered. 
     * 
     * @return bool Whether this view's widget is rendered 
     */
     public function isRendered()
     {
        $hasChildren = 0 < count($this->children);

        if (true === $this->rendered || !$hasChildren) {
            return $this->rendered;
        }

        if ($hasChildren) {
            foreach ($this->children as $child) {
                if (!$child->isRendered()) {
                    return false;
                }
            }

            return $this->rendered = true;
        }

        return false;
    }

现在该功能可以在模板中使用:

{% if not form.isRendered() %}
    <h3>Other fields</h3>
    {{ form_widget(form) }}
{% endif %}