Symfony2形成事件和模型变换器

cap*_*ica 23 php forms symfony doctrine-orm

我正试图与Symfony2的表格制作者,活动和变形金刚搏斗...希望有人在这里更有经验,可以提供帮助!

我有一个表单字段(选择下拉列表),其中包含一些映射到实体的值(一个候选名单).其中一个选项是"其他".假设现在没有AJAX,当用户提交表格我想要检测他们是否选择了"其他"(或者不在候选名单中的任何其他选项).如果他们选择了其中一个选项,则应显示完整的选项列表,否则只显示候选名单.应该很容易吧?;)

所以,我有我的表格类型,它显示基本的候选名单就好了.代码看起来像这样:

namespace Company\ProjectBundle\Form\Type;

use ...

class FancyFormType extends AbstractType {
    private $fooRepo;

    public function __construct(EntityManager $em, FooRepository $fooRepo)
    {
        $this->fooRepo = $fooRepo;
    }

    public function buildForm(FormBuilderInterface $builder, array $options) {
        /** @var Bar $bar */
        $bar = $builder->getData();
        $fooTransformer = new FooToStringTransformer($options['em']);

        $builder
            ->add($builder
                ->create('linkedFoo', 'choice', array(
                    'choices' => $this->fooRepo->getListAsArray(
                        $bar->getLinkedfoo()->getId()
                    ),
                ))
                ->addModelTransformer($fooTransformer)
            )
        ;

        // ...

    }

    // ...
}
Run Code Online (Sandbox Code Playgroud)

现在,我想检查提交的值,因此我使用Form Event Listener,如下所示.

public function buildForm(FormBuilderInterface $builder, array $options) {
    // ... This code comes just after the snippet shown above

    $builder->addEventListener(FormEvents::PRE_SUBMIT, function(FormEvent $event) {
        /** @var EntityManager $em */
        $em = $event->getForm()->getConfig()->getOption('em');

        $data = $event->getData();
        if (empty($data['linkedFoo'])) return;
        $selectedFoo = $data['linkedfoo'];

        $event->getForm()->add('linkedFoo', 'choice', array(
            'choices' => $em
                ->getRepository('CompanyProjectBundle:FooShortlist')
                ->getListAsArray($selectedFoo)
            ,
        ));
        //@todo - needs transformer?
    });
}
Run Code Online (Sandbox Code Playgroud)

但是,它失败并显示如下错误消息:

Notice: Object of class Proxies\__CG__\Company\ProjectBundle\Entity\Foo could not be converted to int in \path\to\project\symfony\symfony\src\Symfony\Component\Form\Extension\Core\ChoiceList\ChoiceList.php line 458 
Run Code Online (Sandbox Code Playgroud)

我认为这个错误是因为当linkedFoo被覆盖时它删除了modelTransformer?我尝试了在事件关闭时访问构建器的各种方法,但这似乎不起作用(返回值是意外的).除了以外的事件,我应该使用其他方法$event->getForm()->add()吗?或者我的方法是否存在更基本的问题?

基本上我不想搞乱linkedFoo字段的配置/变换器/标签,除了改变可用的选择......还有其他方法吗?像这样的东西$form->getField()->updateChoices()

提前感谢您提供的任何帮助!

C

PS是否有比Symfony网站更好的文档或讨论形式,事件等?例如,PRE_SET_DATA,PRE_SUBMIT,SUBMIT等之间的区别是什么?他们什么时候开除?它们应该用于什么?继承如何使用自定义表单字段?什么是表单和构建器,它们如何交互以及何时处理每个表单?如何,何时以及为什么要使用您可以访问的FormFactory $form->getConfig()->getFormFactory()?等等..


编辑:回应弗洛里安的建议,这里有一些关于已尝试但不起作用的事情的更多信息:

如果您尝试在此事件中获取FormBuilder,请执行以下操作:

/** @var FormBuilder $builder */
$builder = $event->getForm()->get('linkedFoo')->getConfig();

$event->getForm()->add($builder
    ->create('linkedFoo', 'choice', array(
        'choices' => $newChoices,
        'label'   =>'label',
    ))
    ->addModelTransformer(new FooToStringTransformer($em))
);
Run Code Online (Sandbox Code Playgroud)

然后你得到错误:

FormBuilder methods cannot be accessed anymore once the builder is turned
into a FormConfigInterface instance.
Run Code Online (Sandbox Code Playgroud)

那么你尝试像Florian建议的那样,即

$event->getForm()->add('linkedFoo', 'choice', array(
    'choices' => $newChoices,
));
$event->getForm()->get('linkedFoo')->getConfig()->addModelTransformer(new FooToStringTransformer($em));
Run Code Online (Sandbox Code Playgroud)

...但是你得到了这个错误:

Notice: Object of class Proxies\__CG__\Company\ProjectBundle\Entity\Foo could not be converted to int 
in C:\path\to\vendor\symfony\symfony\src\Symfony\Component\Form\Extension\Core\ChoiceList\ChoiceList.php line 458
Run Code Online (Sandbox Code Playgroud)

这似乎表明第二行(添加了ModelTransformer)从未被调用,因为->add()在你到达那里之前调用失败了.

cap*_*ica 29

感谢sstok(在github上)的想法,我想我现在已经开始工作了.关键是创建自定义的表单类型,然后使用它来添加ModelTransformer.

创建自定义表单类型:

namespace Caponica\MagnetBundle\Form\Type;

use ...

class FooShortlistChoiceType extends AbstractType {
    protected $em;

    public function __construct(EntityManager $entityManager)
    {
        $this->em                   = $entityManager;
    }

    public function buildForm(FormBuilderInterface $builder, array $options) {
        $fooTransformer = new FooToStringTransformer($this->em);

        $builder
            ->addModelTransformer($fooTransformer)
        ;
    }

    public function getParent() {
        return 'choice';
    }

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

为新类型创建服务定义:

company_project.form.type.foo_shortlist:
    class: Company\ProjectBundle\Form\Type\FooShortlistChoiceType
    tags:
        - { name: form.type, alias: fooShortlist }
    arguments:
        - @doctrine.orm.entity_manager
Run Code Online (Sandbox Code Playgroud)

主窗体的代码现在看起来像这样:

namespace Company\ProjectBundle\Form\Type;

use ...

class FancyFormType extends AbstractType {
    private $fooRepo;

    public function __construct(FooRepository $fooRepo)
    {
        $this->fooRepo = $fooRepo;
    }

    public function buildForm(FormBuilderInterface $builder, array $options) {
        /** @var Bar $bar */
        $bar = $builder->getData();
        $fooTransformer = new FooToStringTransformer($options['em']);

        $builder
            ->add('linkedFoo', 'fooShortlist', array(
                'choices' => $this->fooRepo->getListAsArray(
                    $bar->getLinkedfoo()->getId()
                ),
            ))
        ;

        $builder->addEventListener(FormEvents::PRE_SUBMIT, function(FormEvent $event) {
            /** @var EntityManager $em */
            $em = $event->getForm()->getConfig()->getOption('em');

            $data = $event->getData();
            if (empty($data['linkedFoo'])) return;
            $selectedFoo = $data['linkedFoo'];

            $event->getForm()->add('linkedFoo', 'fooShortlist', array(
                'choices'       => $em->getRepository('CaponicaMagnetBundle:FooShortlist')->getListAsArray($selectedFoo),
                'label'         => 'label'
            ));
        });

        // ...

    }

    // ...
}
Run Code Online (Sandbox Code Playgroud)

关键是这个方法允许你在自定义字段类型中嵌入ModelTransformer,这样,无论何时添加这种类型的新实例,它都会自动为你添加ModelTransformer并阻止前一个循环"无法添加一个字段变压器AND无法在没有字段的情况下添加变压器"