【问题标题】:Reduce form groups layout in Twig with Symfony2(.7)使用 Symfony2(.7) 减少 Twig 中的表单组布局
【发布时间】:2015-10-02 08:53:20
【问题描述】:

我正在尝试摆脱 symfony2 表单中 Twig 中的重复布局 目前我的布局看起来像:

<form name="step2" method="post" action="" class="productForm">
    <h2>Step2:</h2>

    <div id="step2">
        <div class="form-group">
            <div class="form-widget">
                <div id="step2_client1">
                    <div class="form-group"><label class="control-label required" for="step2_client1_clientTitle">Client 1 title:</label>

                        <div class="form-widget"><select id="step2_client1_clientTitle" name="step2[client1][clientTitle]" class="form-control">
                            <option value="Mr">Mr</option>
                            <option value="Mrs">Mrs</option>
                            <option value="Miss">Miss</option>
                            <option value="Ms">Ms</option>
                            <option value="Dr">Dr</option>
                            <option value="Sir">Sir</option>
                        </select></div>
                    </div>
                    <div class="form-group"><label class="control-label required" for="step2_client1_firstName">Client 1 first name:</label>

                        <div class="form-widget"><input type="text" id="step2_client1_firstName" name="step2[client1][firstName]" required="required" class="form-control">
                        </div>
                    </div>
                    <div class="form-group"><label class="control-label required" for="step2_client1_surname">Client 1 last name:</label>

                        <div class="form-widget"><input type="text" id="step2_client1_surname" name="step2[client1][surname]" required="required" class="form-control"></div>
                    </div>
                </div>
            </div>
        </div>
        
        ...
        
        <div class="form-group form-group-submit">
            <div class=""><a id="step2_save" class="has-spinner btn btn-default btn-submit"><span class="spinner"><i class="fa fa-spinner fa-spin"></i></span>Complete the quote
            </a></div>
        </div>
        <input type="hidden" id="step2__token" name="step2[_token]" class="form-control" value="xxx"></div>
</form>

而且我有一个肮脏的 js 脚本,可以按照我希望它显示的方式修复我的布局,但正如我所说,这很脏。所以我想知道是否有人可以帮助我更改我的 twig 文件以仅显示子元素而不是父元素的表单组和表单小部件 css 类。

肮脏的js脚本:

$('.form-group').each(function(e){
    if($(this).parents('.form-group').length) {
        var $realGroup = $(this).parents('.form-group');
        $realGroup.toggleClass('form-group');
        $realGroup.children('.form-widget').toggleClass('form-widget');
    }
});

还有小树枝:

{% block form_row -%}
    {% spaceless %}
    <div class="form-group{% if (not compound or force_error|default(false)) and not valid %} has-error{% endif %}">
        {{ form_label(form) }}
        <div class="form-widget">
            {{ form_widget(form) }}
            {{ form_errors(form) }}
        </div>
    </div>
    {% endspaceless %}
{%- endblock form_row %}

现在树枝位是标准的 SF2 引导表单布局,我很确定我没有改变它。

所以任何帮助摆脱 js 脚本并实际在 twig 中执行它都会很棒,我尝试了一些东西,但每一个都比我用 js 做的更糟糕,所以我有点卡住了。

编辑:

js 脚本所做的只是删除 css 类,因此布局有效。它看起来很平坦,因为它只是空的 div,但我所追求的并不是首先显示 div。我正在尝试更改 twig 脚本,而不是 css,因为这正是我们想要的方式。

【问题讨论】:

  • 您是否只是想使用 CSS 定位表单的特定部分?因为如果是这样,那么使用child selectors 肯定会更容易?
  • 我重读了 3 次,我不知道你在说什么(与他的例子有关)我不想用 css 来定位任何东西,我只是想要任何嵌套的表单,只是简单的形式,所以如果它们是嵌套的,我不希望父母出现,如if form_row.hasChildren() 没有为该父行输出html。
  • 对,明白了,你的问题根本不清楚,因为你提到了 CSS 类,而你发布的 JavaScript 代码是切换 CSS 类,所以我试图找到你问题的根源即你真正想要做的事情。
  • 很公平,我会更新问题

标签: php forms symfony twitter-bootstrap-3 twig


【解决方案1】:

你有三个选择。

1.手动创建每个字段:

{{ form_start(form, {'action': path(''), 'attr': {'class':'productForm'}}) }}
    {{ form_errors(form) }}

<div class="form-group">
    {{ form_label(form.clientTitle, 'Client 1 title:', {'label_attr': {'class': 'control-label required'}}) }}
    <div class="form-widget">
        {{ form_widget(form.clientTitle, { 'attr': {'class': 'form-control' }}) }}
    </div>
    {{ form_errors(form.clientTitle) }}
</div>

<div class="form-group">
    {{ form_label(form.firstName, 'Client 1 first name:', {'label_attr': {'class': 'control-label required'}}) }}
    <div class="form-widget">
        {{ form_widget(form.firstName, { 'attr': {'class': 'form-control' }}) }}
    </div>
    {{ form_errors(form.firstName) }}
</div>

<div class="form-group">
    {{ form_label(form.firstName, 'Client 1 last name:', {'label_attr': {'class': 'control-label required'}}) }}
    <div class="form-widget">
        {{ form_widget(form.firstName, { 'attr': {'class': 'form-control' }}) }}
    </div>
    {{ form_errors(form.firstName) }}
</div>


<div class="form-group form-group-submit">
    <div class="">
        <a id="step2_save" class="has-spinner btn btn-default btn-submit">
            <span class="spinner">
                <i class="fa fa-spinner fa-spin"></i>
            </span>
            Complete the quote
        </a>
    </div>
</div>

{{ form_end(change_password) }}

2.或者您可以创建自己的form theme

3.或者为每个表单使用一个循环:

{% for child in form.children %}

    <div class="form-group">
        {{ form_label(child, null, {'label_attr': {'class': 'control-label required'}}) }}
        <div class="form-widget">
        {{ form_widget(child, { 'attr': {'class': 'form-control' }}) }}
        </div>
        {{ form_errors(child) }}
    </div>

{% endfor %}

这完全取决于您未来的表单将拥有多少个字段以及您拥有多少类似的表单。

但是,使用 JS 来做这样的事情是不行的。

【讨论】:

  • 是的,我知道,但是您所看到的只是表单的一小部分,大约有 30 多个表单,每个表单有 40 多个字段,它们都可以正常工作,但手动解决方案不是会发生,创建我的表单主题一切都很好,但这仍然不能解决知道该行何时是顶级的问题
  • 什么是顶级行?在想要的效果之前和之后显示一个想法。另外,你看过form collections 吗?也许你能想到一个主意。
  • 之前和之后...... js 确实说明了这一点。顶级行将是不嵌套在其他行中的 form_row。
  • nounty 是自动授予的,但我的问题从未得到解答,我认为我需要什么已经很清楚了,但 no1 从未给我一个确切的解决方案
【解决方案2】:

最简单的方法是创建自定义表单主题。

如果你看at the current Symfony2's bootstrap one,你可以看到:

{% block form_row -%}
    <div class="form-group{% if (not compound or force_error|default(false)) and not valid %} has-error{% endif %}">
        {{- form_label(form) -}}
        {{- form_widget(form) -}}
        {{- form_errors(form) -}}
    </div>
{%- endblock form_row %}

“复合”内容实际上是在告诉您当前元素是否包含子元素(表单、集合……基本上是所有父元素),以及您是否不想为父母提供 form-group 类,您只需要在自定义主题中重载此块即可。

{# AppBundle::custom_theme.html.twig #}

{% use "bootstrap_3_layout.html.twig" %}

{% block form_row -%}
    <div class="{% if not compound %}form-group{% endif %}{% if (not compound or force_error|default(false)) and not valid %} has-error{% endif %}">
        {{- form_label(form) -}}
        {{- form_widget(form) -}}
        {{- form_errors(form) -}}
    </div>
{%- endblock form_row %}

如果您根本不需要 div,只需将其删除!现在就可以了。


要在整个项目中应用更改,您可以将其添加到 config.yml

twig:
    # ...
    form_themes:
        - 'AppBundle::custom_theme.html.twig'

或仅在特定视图上使用它:

{% form_theme yourForm 'AppBundle::custom_theme.html.twig' %}

注意:yourForm 是您当前的表单对象。

【讨论】:

  • 感谢您的回答,但您是否注意到我的表单行(树枝位)和您的块 form_row 之间有什么区别?因为那个树枝位位于Acme/MyBundle/Resources/views/Form/form_horizontal_layout.html.twig 并且已经在扩展引导表单主题,所以我正在寻找的是一种判断行中是否有子表单的方法,因为如果有,那么我没有想要显示该表单行布局的 div,只有那些嵌套表单中最深的表单行应该作为 div 可见,如展平嵌套表单
【解决方案3】:

George Irimiciuc 已经帮助了你,我的意思是你可以使用 George 回答中提供的第二个选项,你需要专注于表单主题的这一部分。

Symfony2 twig 根据选定的主题创建表单,如果您尝试添加表单主题,那会很好,其他选项意味着您只是将问题推到其他地方,这只是一个技巧。无需更新完整的表单,只需添加您的 form_row 即可,可以试试这个

http://symfony.com/doc/current/cookbook/form/form_customization.html#customizing-the-form-row

我使用这是我的项目,最好的一点是您可以从表单构建器发送属性,然后检查 form_row 下的属性,这会呈现您的表单行并可以在检查条件后创建您想要的表单 -

attr => array( 'parent_class' => false)

并且可以像这样在 form_row 下查看这个

if( form.vars.attr.parent_class is defined )

并且可以在树枝中添加条件。

如果您遇到问题,例如您的表单有子表单,而 twig form_row 以奇怪的方式显示 -

我建立了这个项目,直到现在只有这个,因为有学生实体与各种实体相结合,所以在像 Profile 这样的表单条目时。我同时使用了实体 Student 和 Address 我所做的是,我刚刚检查了这个

 {% if ( not form.children ) or ( form.vars.expanded is defined  and form.vars.expanded ) %}
  <div class="{{ form.vars.attr['parent-div-class'] is defined ? form.vars.attr['parent-div-class'] : 'col-sm-12'}}">
    <div class="form-group">
      {{ form_label(form) }}
      {{ form_widget(form) }}
      {% if form_errors(form) %}
         {{ form_errors(form) }}
      {% endif %}
    </div>
   </div>    
{% else %}
    <div class="clear-both"></div>
    {{ form_label(form) }}           
    {{ form_widget(form) }}
{% endif %}

代码并不完全像这样,但主要概念是这样的。

希望你明白我的意思。

【讨论】:

  • 我的问题中的树枝位位于Acme/MyBundle/Resources/views/Form/form_horizontal_layout.html.twig。这不是我自己的布局吗?我知道我必须将表单扩展为我自己的主题,但我真正想要的是如何在 twig 中判断该行是否有子表单
  • 可以试试这个 - {% if ( not form.children ) or ( form.vars.expanded 被定义并且 form.vars.expanded ) %}
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2014-03-24
  • 2013-03-12
  • 2013-07-08
  • 2018-05-19
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多