Symfony2 - 使用表单类时如何设置和获取选项?

Mr *_*blo 6 php symfony

我正在使用Form Classes在我的项目中构建各种表单.

在实体类型文件中,对于buildForm函数,有一个辅助参数"array $ options"(这在Symfony 2官方文档中显示,但从未提及过!)

我已经将数组输入到我的控制器中的createForm函数中,但现在我完全不知道如何检索存储的值?

$form = $this->createForm(new ProductType(array(), array('id' => '2')), $product);
Run Code Online (Sandbox Code Playgroud)

我找到的关于获取选项的唯一方法是使用getOption()函数,但这在Symfony 2中不存在(我发现的帖子是2010年).

我试过用:

$id = $options['id'];
Run Code Online (Sandbox Code Playgroud)

但是当我尝试在任何地方使用$ id时,我得到错误:

注意:未定义的索引:id

是什么赋予了?

如何从$ options数组中检索我的值?我甚至在第一时间正确设置它们吗?

编辑 - 更多代码:

表格类

<?php

namespace DEMO\DemoBundle\Form\Product;

use Doctrine\ORM\EntityRepository;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilder;

class ProductType extends AbstractType
{
    public function buildForm(FormBuilder $builder, array $options)
    {
        $builder
            ->add('name')
            ->add('slug')
            ->add('reference')
            ->add('description')
            ->add('active_from')
            ->add('active_till')
            ->add('is_active')
            ->add('category', 'entity', array(
                'class' => 'DEMO\DemoBundle\Entity\Product\ProductCategory',
                'query_builder' => function(EntityRepository $er) {
                    return $er->createQueryBuilder('u')
                        ->where('u.section = :id')
                        ->setParameter('id', ***ID VARIABLE NEEDS TO GO HERE***)
                        ->orderBy('u.root', 'ASC')
                        ->addOrderBy('u.lft', 'ASC');
                },
                'empty_value' => 'Choose an option',
                'property' => 'indentedName',
            ));
    }

    public function getDefaultOptions()
    {
        return array(
            'data_class' => 'DEMO\DemoBundle\Entity\Product\Product'
        );
    }

    public function getName()
    {
        return 'demo_demobundle_product_type';
    }
}
Run Code Online (Sandbox Code Playgroud)

gre*_*ire 17

我认为你首先没有正确设置它们.你应该把它们作为第三个参数createForm()

编辑:以下是您的表单类的外观:

<?php
namespace DEMO\DemoBundle\Form\Product;

use Doctrine\ORM\EntityRepository;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilder;

class ProductType extends AbstractType
{
    public function buildForm(FormBuilder $builder, array $options)
    {
        $builder
            ->add('name')
            ->add('slug')
            ->add('reference')
            ->add('description')
            ->add('active_from')
            ->add('active_till')
            ->add('is_active')
            ->add('category', 'entity', array(
                'class' => 'DEMO\DemoBundle\Entity\Product\ProductCategory',
                'query_builder' => function(EntityRepository $er) use($options) {
                    return $er->createQueryBuilder('u')
                        ->where('u.section = :id')
                        ->setParameter('id', $options['id'])
                        ->orderBy('u.root', 'ASC')
                        ->addOrderBy('u.lft', 'ASC');
                },
                'empty_value' => 'Choose an option',
                'property' => 'indentedName',
            ));
    }

    public function getDefaultOptions()
    {
        return array(
            'data_class' => 'DEMO\DemoBundle\Entity\Product\Product',
            'id'         => null
        );
    }

    public function getName()
    {
        return 'demo_demobundle_product_type';
    }
}
Run Code Online (Sandbox Code Playgroud)


小智 8

让我告诉你什么对我有用

在控制器中:

$form = $this->createForm(new UsersType(), $entity, array(
    'attr' => array('locationId' => $currentLocationId)));
Run Code Online (Sandbox Code Playgroud)

在FormType中:

->add('location', 'entity', array(
        'class' => 'Ro\RoinventBundle\Entity\Locations',
         'query_builder' => function (\Doctrine\ORM\EntityRepository $er) use ($options)
        {
            if (isset($options['attr']['locationId']) && ($options['attr']['locationId'] != NULL))
            {
                return $er->createQueryBuilder('Locations')
                    ->where('Locations.id = :param')
                    ->setParameter('param', $options['attr']['locationId']);
            }
            //else do what you want
},
));
Run Code Online (Sandbox Code Playgroud)


Bon*_*uar 7

显然它已经不再存在getDefaultOptions()了,但随之而来setDefaultOptions().

不然它说

选项"my_custom_option"不存在.已知选项包括:"action","attr","auto_initialize",...

所以,对我来说,我必须像这样更新setDefaultOptions:

public function setDefaultOptions(OptionsResolverInterface $resolver)
{
    $resolver->setDefaults(array('my_custom_option' => false));
    // Others if needed, like in the documentation for : 'data_class' => 'VENDOR\Bundle\Entity\MyEntity', 'csrf_protection' => true
}
Run Code Online (Sandbox Code Playgroud)

然后你可以在buildForm方法中检索它

public function buildForm(FormBuilderInterface $builder, array $options)
{
    $myCustomOption = $options['my_custom_option'];
}
Run Code Online (Sandbox Code Playgroud)


Ric*_*rez 5

根据这个Google网上论坛的说法

"greg0ire"是对的,事实上我已经尝试过并且完美无缺!!! 你说"我真的不想去"黑客攻击"任何核心结构"但你最终没有使用最好的方法......事实上,从我的观点来看,你最终做的不是你没有想做.

所以最后你应该这样做:

在formType

public function buildForm(FormBuilder $builder, array $options)
{
    $builder
        ->add('name')
        ->add('slug')
        ->add('reference')
        ->add('description')
        ->add('active_from')
        ->add('active_till')
        ->add('is_active')
        ->add('category', 'entity', array(
            'class' => 'DEMO\DemoBundle\Entity\Product\ProductCategory',
            'query_builder' => function(EntityRepository $er) {
                return $er->createQueryBuilder('u')
                    ->where('u.section = :id')
                    ->setParameter('id', $options['id'])
                    ->orderBy('u.root', 'ASC')
                    ->addOrderBy('u.lft', 'ASC');
            },
            'empty_value' => 'Choose an option',
            'property' => 'indentedName',
        ));
}

public function getDefaultOptions()
{
    return array(
        'data_class' => 'DEMO\DemoBundle\Entity\Product\Product'
        'id' => null
    );
}
Run Code Online (Sandbox Code Playgroud)

并在控制器中

$form = $this->createForm(new ProductType(), $product, array('id' => $id ));
Run Code Online (Sandbox Code Playgroud)


Mr *_*blo 1

好吧,事实证明,Gregoires 的答案非常接近,但在尝试实际将变量放入 createQueryBuilder 函数时却给了我“未定义的变量”错误。

我花了一段时间试图找出原因并发现问题所在。您必须在“query_builder”选项中向函数添加一个额外的参数,如下所示:

'query_builder' => function(EntityRepository $er) use ($options) {
                    return $er->createQueryBuilder('u')
                        ->where('u.section = :id')
                        ->setParameter('id', $options['id'])
                        ->orderBy('u.root', 'ASC')
                        ->addOrderBy('u.lft', 'ASC');
                },
Run Code Online (Sandbox Code Playgroud)

神奇的设置是“use ($options)”。这使您可以在查询生成器中成功使用 $options['id']。