首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >Symfony 3多级嵌套形式

Symfony 3多级嵌套形式
EN

Stack Overflow用户
提问于 2016-10-19 18:08:09
回答 1查看 4K关注 0票数 5

我一直在研究关于嵌入 (一组表单)的Symfony 3教程,我想将这个想法扩展到额外的嵌套级别。我环顾四周,找到了Symfony 2的部分答案,但没有全面的答案(3没有)。

如果我们学习教程Task有许多Tag示例,我将如何编写它的代码以便扩展到:Task有很多Tag有很多SubTag

到目前为止,我认为我理解表单类:

任务:

代码语言:javascript
运行
复制
class TaskType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder->add('description');

        $builder->add('tags', CollectionType::class, array(
            'entry_type' => TagType::class,
            'allow_add' => true,
            'by_reference' => false,
            'allow_delete' => true
        ));
    }

    public function configureOptions(OptionsResolver $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'AppBundle\Entity\Task',
        ));
    }
}

标签:

代码语言:javascript
运行
复制
class TagType extends AbstractType
    {
        public function buildForm(FormBuilderInterface $builder, array $options)
        {
            $builder->add('name');

            $builder->add('sub_tags', CollectionType::class, array(
                'entry_type' => SubTagType::class,
                'allow_add' => true,
                'by_reference' => false,
                'allow_delete' => true
            ));
        }

        public function configureOptions(OptionsResolver $resolver)
        {
            $resolver->setDefaults(array(
                'data_class' => 'AppBundle\Entity\Tag',
            ));
        }
    }

SubTag:

代码语言:javascript
运行
复制
class SubTagType extends AbstractType
    {
        public function buildForm(FormBuilderInterface $builder, array $options)
        {
            $builder->add('name');
        }

        public function configureOptions(OptionsResolver $resolver)
        {
            $resolver->setDefaults(array(
                'data_class' => 'AppBundle\Entity\SubTag',
            ));
        }
    }

和基本的Twig类:

代码语言:javascript
运行
复制
{{ form_start(form) }}
    {# render the task's only field: description #}
    {{ form_row(form.description) }}

    <h3>Tags</h3>
    <ul class="tags">
        {# iterate over each existing tag and render its only field: name #}
        {% for tag in form.tags %}
            <li>{{ form_row(tag.name) }}</li>
            {% for sub_tag in tag.sub_tags %}
                <li>{{ form_row(sub_tag.name) }}</li>
            {% endfor %}
        {% endfor %}
    </ul>
{{ form_end(form) }}

但在这一点上,我不确定原型和javascript将如何工作。有人能解释一下我下一步该怎么做吗?这就是正确的方法吗?

我的第一个想法是,如果我们要进行额外的级别,那么将JS推广到任意数量的级别可能是明智的,因为本教程使用的JS非常多,只能在单个级别上工作。

我能找到的最接近的工作代码是这个堆栈溢出响应这里。然而,它似乎不像描述的那样起作用,而且我很难准确地知道出了什么问题。

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2016-10-19 19:32:06

它与普通的嵌入式表单集合没有什么不同。

但是,如果您想避免与父窗体的原型字符串发生冲突的默认__NAME__原型的麻烦,那么您应该考虑为TagTypeSubTag类型选择不同的值。

来自CollectionType

prototype_name

  • 类型: string默认值:name
  • 如果窗体中有多个集合,或者更糟的是,嵌套集合,则可能需要更改占位符,以便不用相同的值替换不相关的占位符。

如果您想用javascript抽象您的克隆操作,比如这篇文章 (粘贴在下面)中的操作,顺便说一句,这些操作似乎是针对symfony3的,这可能非常有帮助!

例如,您可能希望将传递给prototype_name的相同值包含在集合持有者的html上,以便在对data-prototype html执行replace时能够动态地访问它。

代码语言:javascript
运行
复制
var $collectionHolder;

// setup an "add a tag" link
var $addTagLink = $('<a href="#" class="add_tag_link">Add a tag</a>');
var $newLinkLi = $('<li></li>').append($addTagLink);

jQuery(document).ready(function() {
// Get the ul that holds the collection of tags
$collectionHolder = $('ul.tags');

// add the "add a tag" anchor and li to the tags ul
$collectionHolder.append($newLinkLi);

// count the current form inputs we have (e.g. 2), use that as the new
// index when inserting a new item (e.g. 2)
$collectionHolder.data('index', $collectionHolder.find(':input').length);

$addTagLink.on('click', function(e) {
    // prevent the link from creating a "#" on the URL
    e.preventDefault();

    // add a new tag form (see next code block)
    addTagForm($collectionHolder, $newLinkLi);
});

function addTagForm($collectionHolder, $newLinkLi) {
    // Get the data-prototype explained earlier
    var prototype = $collectionHolder.data('prototype');

    // get the new index
    var index = $collectionHolder.data('index');

    // Replace '__name__' in the prototype's HTML to
    // instead be a number based on how many items we have
    var newForm = prototype.replace(/__name__/g, index);

    // increase the index with one for the next item
    $collectionHolder.data('index', index + 1);

    // Display the form in the page in an li, before the "Add a tag" link li
    var $newFormLi = $('<li></li>').append(newForm);
    $newLinkLi.before($newFormLi);
}
票数 3
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/40138645

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档