Symfony2:从集合中删除实体

时间:2014-11-28 11:59:51

标签: php symfony doctrine-orm

1。概述

我希望能够使用symfony2表单从集合中删除实体。

1.1问题

我可以向集合添加新实体,并将其删除,只要添加或删除的实体位于集合的末尾即可。只要我从开头或中间删除一个,我就会收到以下错误:

当我尝试这样做时,我收到此错误:

  

属性“id”和其中一种方法都没有   “addId()”/“removeId()”,“setId()”,“id()”,“__ set()”或“__call()”   在课堂上存在并具有公共访问权限   “ApiBundle \实体\数据\列”。

1.2代码

以下是所有相关代码。

数据

/**
 * Data
 *
 * @ORM\Table(name="data__data")
 * @ORM\Entity(repositoryClass="ApiBundle\Repository\Data\DataRepository")
 */
class Data
{
    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="string")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="UUID")
     */
    protected $id;

    /**
     * @var ArrayCollection
     * @ORM\OneToMany(targetEntity="Column", mappedBy="parent", cascade={"all"}, orphanRemoval=true)
     */
    protected $columns;

    /**
     * Initialise the array collections
     */
    public function __construct()
    {
        $this->columns = new ArrayCollection();
    }

    /**
     * @param mixed $columns
     */
    public function setColumns($columns)
    {
        $this->columns = $columns;
    }

    /**
     * @param Column $column
     */
    public function addColumn($column)
    {
        $column->setParent($this);
        $this->columns->add($column);
    }

    /**
     * @param Column $column
     */
    public function removeColumn($column)
    {
        $this->columns->removeElement($column);
    }
}

/**
 * Data
 *
 * @ORM\Table(name="data__column")
 * @ORM\Entity
 */
class Column
{

    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="string")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="UUID")
     */
    protected $id;

    /**
     * @var Data
     * @ORM\ManyToOne(targetEntity="Data", inversedBy="columns")
     */
    protected $parent;

    /**
     * @return Data
     */
    public function getParent()
    {
        return $this->parent;
    }

    /**
     * @param Data $parent
     */
    public function setParent($parent)
    {
        $this->parent = $parent;
    }
}

DataFormType

class DataFormType extends AbstractType {

    public function buildForm(FormBuilderInterface $builder, array $options)
    {

        $builder
            ->add('id')
            ->add('columns', 'collection', array(
                'type' => new ColumnFormType(),
                'allow_add'    => true,
                'allow_delete' => true,
                'by_reference' => false
            ))
        ;
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'ApiBundle\Entity\Data\Data',
            'csrf_protection' => false
        ));
    }

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

} 

ColumnFormType

class ColumnFormType extends AbstractType
{

    public function buildForm(FormBuilderInterface $builder, array $options)
    {

        $builder
            ->add('id');
    }

    public function setDefaultOptions(OptionsResolverInterface $resolver)
    {
        $resolver->setDefaults(array(
            'data_class' => 'ApiBundle\Entity\Data\Column',
            'csrf_protection' => false
        ));
    }

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

} 

为清晰起见,我已从这些代码段中删除了一些代码

1.3结论

就像我说的那样,从集合的末尾添加或删除时我没有遇到任何问题。但是只要它在任何其他地方就会出错。

感谢您的帮助。

1 个答案:

答案 0 :(得分:2)

错误是由于缺少收集密钥而导致的。

CollectionTypeResizeListener匹配得很高。它使用子表单填充集合表单:

public function preSetData(FormEvent $event)
{
    $form = $event->getForm();
    $data = $event->getData();

    ...

    // Then add all rows again in the correct order
    foreach ($data as $name => $value) {
        $form->add($name, $this->type, array_replace(array(
            'property_path' => '['.$name.']',
        ), $this->options));
    }
}

因此,每个子表单都映射到集合对象(底层数据),并具有适用于集合索引的名称,例如: '[0]','[1]'。当您从集合中删除元素时,ResizeListener会删除多余的子表单。

public function preSubmit(FormEvent $event)
{
    $form = $event->getForm();
    $data = $event->getData();
    ...

    // Remove all empty rows
    if ($this->allowDelete) {
        foreach ($form as $name => $child) {
            if (!isset($data[$name])) {
                $form->remove($name);
            }
        }
    }
}

假设有data[columns][0][id]=1, data[columns][1][id]=2, data[columns][2][id]=3

从最后删除元素时 - 一切都很好。 data[columns][0][id]=1, data[columns][1][id]=2有相应的内容。然后将删除子表单[2],然后将从集合中删除索引为2的元素。

当您删除不在末尾的元素而您没有保留键时 - 会发生错误。例如,您发送data[columns][0][id]=2, data[columns][1][id]=3ResizeListener将删除索引为[2]的子表单。对于其余子表单([0][1])及其子表单(id),将覆盖基础数据。首先处理大多数嵌套的子表单。

 [0] (Column)
    [id]
        1 => 2
 [1] (Column)
    [id]
        2 => 3

然后PropertyPathMapper会检测到id子表单的数据不等于列的id属性值(这是[0]的基础数据):

public function mapFormsToData($forms, &$data)
{
    ...
            if (!is_object($data) || !$config->getByReference() || $form->getData() !== $this->propertyAccessor->getValue($data, $propertyPath)) {
                $this->propertyAccessor->setValue($data, $propertyPath, $form->getData());
            }
    ...
}

它会PropertyAccessor将新的id值设置为Column个对象。最后一个会抛出异常,因为无法将新的id设置为Column(没有setter,属性不公开等)。

解决方案:保留关键订单。如果您获得data[columns][0][id]=1, data[columns][1][id]=2, data[columns][2][id]=3并删除了第一个元素,则应发送data[columns][1][id]=2, data[columns][2][id]=3

PS 保留表格的关键顺序是所有情况下的良好做法。它会阻止您进行冗余的UPDATE查询和循环。