php - Symfony2 : Removing entity from middle of collection

标签 php symfony doctrine-orm

1。概述

我希望能够使用 symfony2 Form 从集合中删除一个实体。

1.1 问题

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

当我尝试执行此操作时出现此错误:

Neither the property "id" nor one of the methods "addId()"/"removeId()", "setId()", "id()", "__set()" or "__call()" exist and have public access in class "ApiBundle\Entity\Data\Column".

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;
    }
}

数据格式类型

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 结论

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

感谢您的帮助。

最佳答案

错误是由于没有保存集合 key 导致的。

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 子表单的数据不等于 Column 的 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());
            }
    ...
}

这将使 PropertyAccessorColumn 对象设置新的 id 值。最后一个将抛出异常,因为无法将新的 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 查询和循环。

关于php - Symfony2 : Removing entity from middle of collection,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27188201/

相关文章:

php - jQuery "waypoints"不起作用

php - mysql。以月为单位的日期差异

php - 有没有办法在 Doctrine ORM 中将 LIMIT 添加到 UPDATE 查询?

php - 使用FOSElastica bundle 一次搜索多种类型

php - 按两列排序 - 使用评分最高的平均评分

php - 如何获取给定结构中的 json 输出?

javascript - GMT 与 UTC 日期

Symfony2 - 我应该把用户上传的文件放在哪里?

php - Symfony 3多对多不将数据保存到ManyToMany表

php - Symfony 2 不生成 web/bundle/framework 结构