noS*_*owP 5 php symfony doctrine-orm
我希望能够使用symfony2表单从集合中删除实体.
我可以向集合中添加新实体,并将其删除,只要添加或删除的实体位于集合的末尾即可.只要我从开头或中间删除一个,我就会收到以下错误:
当我尝试这样做时,我收到此错误:
属性"id"和方法"addId()"/"removeId()","setId()","id()","__ set()"或"__call()"之一都不存在并且公开在"ApiBundle\Entity\Data\Column"类中访问.
这是所有相关的代码.
数据
/**
* 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);
}
}
Run Code Online (Sandbox Code Playgroud)
柱
/**
* 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;
}
}
Run Code Online (Sandbox Code Playgroud)
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';
}
}
Run Code Online (Sandbox Code Playgroud)
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';
}
}
Run Code Online (Sandbox Code Playgroud)
为清楚起见,我已从这些代码段中删除了一些代码
就像我说的那样,从集合的末尾添加或删除时我没有遇到任何问题.但是,只要它在任何其他地方,它就会出错.
谢谢你的帮助.
该错误是由于缺少集合密钥保存而导致的。
CollectionType是高坚韧的ResizeListener。它用子表单填充集合表单:
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));
}
}
Run Code Online (Sandbox Code Playgroud)
因此,每个子表单都映射到集合对象(底层数据),并具有适用于集合索引的名称,例如“[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);
}
}
}
}
Run Code Online (Sandbox Code Playgroud)
可以说有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]=3. ResizeListener将删除索引为的子表单[2]。其余子表单 ( [0]、[1]) 及其子表单 ( id) 的基础数据将被覆盖。大多数嵌套子表单都会首先处理。
[0] (Column)
[id]
1 => 2
[1] (Column)
[id]
2 => 3
Run Code Online (Sandbox Code Playgroud)
然后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());
}
...
}
Run Code Online (Sandbox Code Playgroud)
它将为对象PropertyAccessor设置新id值Column。最后一个将抛出异常,因为无法id为 Column 设置 new (没有 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查询和循环。