2

我正在使用带有 Doctrine 的 Symfony 2.2 构建一个表单生成器。基本原则是用户通过填写​​其名称并在选择菜单中选择他想拥有的小部件来创建一个新表单。

我们可以想到 WidgetInputText、WidgetSelect、WidgetFile 等。

这是我的模型的示例:

<?php

namespace Ineat\FormGeneratorBundle\Entity\Widget;
use Symfony\Component\Validator\Constraints as Assert;


use Doctrine\ORM\Mapping as ORM;

/**
 * Widget
 *
 * @ORM\Table(name="widget")
 * @ORM\Entity
 * @ORM\InheritanceType("SINGLE_TABLE")
 * @ORM\DiscriminatorColumn(name="discr", type="string")
 * @ORM\DiscriminatorMap({"widget_text" = "WidgetText", "widget_input_text" = "WidgetInputText", "widget_select" = "WidgetSelect"})
 */
abstract class Widget
{
    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="integer")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

    /**
     * @ORM\ManyToOne(targetEntity="Form", inversedBy="widgets")
     */
    private $form;

    /**
     * @var integer
     *
     * @ORM\OneToOne(targetEntity="Question")
     */
    private $question;

    /**
     * Get id
     *
     * @return integer 
     */
    public function getId()
    {
        return $this->id;
    }

    /**
     * Set form
     *
     * @param \Ineat\FormGeneratorBundle\Entity\Form $form
     * @return Widget
     */
    public function setForm(\Ineat\FormGeneratorBundle\Entity\Form $form = null)
    {
        $this->form = $form;

        return $this;
    }

    /**
     * Get form
     *
     * @return \Ineat\FormGeneratorBundle\Entity\Form 
     */
    public function getForm()
    {
        return $this->form;
    }

    /**
     * Set question
     *
     * @param \Ineat\FormGeneratorBundle\Entity\Question $question
     * @return Widget
     */
    public function setQuestion(\Ineat\FormGeneratorBundle\Entity\Question $question = null)
    {
        $this->question = $question;

        return $this;
    }

    /**
     * Get question
     *
     * @return \Ineat\FormGeneratorBundle\Entity\Question 
     */
    public function getQuestion()
    {
        return $this->question;
    }
}

<?php

namespace Ineat\FormGeneratorBundle\Entity\Widget;

use Symfony\Component\Validator\Constraints as Assert;
use Doctrine\ORM\Mapping as ORM;

/**
 * Widget
 *
 * @ORM\Entity
 * @ORM\Table(name="widget_text")
 */
class WidgetText extends Widget
{
    /**
     * @var string
     *
     * @ORM\Column(type="text")
     */
    private $text;

    /**
     * Set text
     *
     * @param string $text
     * @return WidgetText
     */
    public function setText($text)
    {
        $this->text = $text;

        return $this;
    }

    /**
     * Get text
     *
     * @return string 
     */
    public function getText()
    {
        return $this->text;
    }
}

<?php

namespace Ineat\FormGeneratorBundle\Entity;

use Doctrine\ORM\Mapping as ORM;
use Symfony\Component\Validator\Constraints as Assert;
use Symfony\Bridge\Doctrine\Validator\Constraints\UniqueEntity;
use Doctrine\Common\Collections\ArrayCollection;

/**
 * Form
 *
 * @ORM\Table(name="form")
 * @ORM\Entity(repositoryClass="Ineat\FormGeneratorBundle\Entity\FormRepository")
 * @UniqueEntity("name")
 * @UniqueEntity("slug")
 */
class Form
{
    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="integer")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

    /**
     * @var string
     *
     * @ORM\Column(name="name", type="string", length=255)
     */
    private $name;

    /**
     * @var string
     *
     * @ORM\Column(name="slug", type="string", length=255)
     */
    private $slug;

    /**
     * @var ArrayCollection
     *
     * @ORM\OneToMany(targetEntity="Widget", mappedBy="form", cascade={"persist"})
     */
    private $widgets;


    public function __construct()
    {
        $this->widgets = new ArrayCollection();
    }

    /**
     * Get id
     *
     * @return integer 
     */
    public function getId()
    {
        return $this->id;
    }

    /**
     * Set name
     *
     * @param string $name
     * @return Form
     */
    public function setName($name)
    {
        $this->name = $name;

        return $this;
    }

    /**
     * Get name
     *
     * @return string 
     */
    public function getName()
    {
        return $this->name;
    }

    /**
     * Set slug
     *
     * @param string $slug
     * @return Form
     */
    public function setSlug($slug)
    {
        $this->slug = $slug;

        return $this;
    }

    /**
     * Get slug
     *
     * @return string 
     */
    public function getSlug()
    {
        return $this->slug;
    }

    /**
     * Add widgets
     *
     * @param \Ineat\FormGeneratorBundle\Entity\Widget\Widget $widget
     * @return Form
     */
    public function addWidget(\Ineat\FormGeneratorBundle\Entity\Widget\Widget $widget)
    {
        $this->widgets[] = $widget;

        return $this;
    }

    /**
     * Remove widgets
     *
     * @param \Ineat\FormGeneratorBundle\Entity\Widget\Widget $widget
     */
    public function removeWidget(\Ineat\FormGeneratorBundle\Entity\Widget\Widget $widget)
    {
        $this->widgets->removeElement($widget);
    }

    /**
     * Get widgets
     *
     * @return \Doctrine\Common\Collections\Collection 
     */
    public function getWidgets()
    {
        return $this->widgets;
    }

    /**
     * Set widgets
     *
     * @param \Doctrine\Common\Collections\Collection $widgets
     */
    public function setWidget(\Doctrine\Common\Collections\Collection $widgets)
    {
        $this->widgets = $widgets;
    }

    public function __set($name, $obj)
    {
        if (is_a($obj, '\Ineat\FormGeneratorBundle\Entity\Widget\Widget')) {
            $this->addWidget($obj);
        }
    }
}

如您所见,一个表单可以附加多个小部件。

我创建了一个抽象类 Widget,因为所有小部件都有公共字段并且属于 Widget 类型,并且因为在 Form 实体中,每个 Widget 类型都有一个集合似乎真的很糟糕(糟糕且无聊)。

这个模型有效,我已经对其进行了单元测试,并且能够将 WidgetText 附加到表单然后检索它。

当我尝试使用表单时,问题就来了。

<?php

namespace Ineat\FormGeneratorBundle\Form;

use Ineat\FormGeneratorBundle\Entity\Widget\WidgetText;
use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolverInterface;

class FormType extends AbstractType
{
    protected $widgets;

    public function __construct(array $widgets = array())
    {
        $this->widgets = $widgets;
    }

    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder
            ->add('name', 'text')
            ->add('slug', 'text')
            ->add('WidgetText', 'collection', array(
                'type'         => new WidgetTextType(),
                'allow_add'    => true,
                'attr'         => array('class' => 'widget-text'),
                'by_reference' => false
            ))
        ;
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'Ineat\FormGeneratorBundle\Entity\Form',
        ));
    }

    public function getName()
    {
        return 'ineat_formgeneratorbundle_formtype';
    }
}

<?php

namespace Ineat\FormGeneratorBundle\Form;

use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\OptionsResolver\OptionsResolverInterface;

class WidgetTextType extends AbstractType
{
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder
            ->add('text', 'text')
        ;
    }

    public function getName()
    {
        return 'ineat_formgeneratorbundle_widgettexttype';
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'Ineat\FormGeneratorBundle\Entity\Widget\WidgetText',
        ));
    }
}

当我尝试显示表单时,出现以下错误:

类“Ineat\FormGeneratorBundle\Entity\Form”中不存在属性“WidgetText”或方法“getWidgetText()”或方法“isWidgetText()”

就像 Symfony 不知道我的 WidgetText 也是 Widget 类型。

如果在控制器中(由 Symfony 生成)我改变这一行:

$this->createForm(new FormType(), new Form())

至:

$this->createForm(new FormType())

表单显示良好,但提交时我没有绑定数据。

我完全被困在那里,从 OOP 的角度来看,我认为这应该可行,但我不确定 Symfony 是否允许我做我想做的事。

4

1 回答 1

0

如您的问题评论中所述,您应该将“WidgetText”字段的名称更改为“widgets”。其背后的原因是字段的名称应该与模型中的访问器匹配(即“名称”用于(set|get)Name(),“小部件”用于(set|get)Widgets()等)

如果您确实希望字段名称与模型中的访问器不同,您还可以使用“property_path”选项(默认设置为字段名称):

$builder->add('WidgetText', ..., array(
    ...
    'property_path' => 'widgets',
));
于 2013-08-20T08:38:18.250 回答