我希望能够使用symfony2表单从集合中删除实体。
我可以向集合添加新实体,并将其删除,只要添加或删除的实体位于集合的末尾即可。只要我从开头或中间删除一个,我就会收到以下错误:
当我尝试这样做时,我收到此错误:
属性“id”和其中一种方法都没有 “addId()”/“removeId()”,“setId()”,“id()”,“__ set()”或“__call()” 在课堂上存在并具有公共访问权限 “ApiBundle \实体\数据\列”。
以下是所有相关代码。
数据
/**
* 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';
}
}
为清晰起见,我已从这些代码段中删除了一些代码
就像我说的那样,从集合的末尾添加或删除时我没有遇到任何问题。但是只要它在任何其他地方就会出错。
感谢您的帮助。
答案 0 :(得分:2)
错误是由于缺少收集密钥而导致的。
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));
}
}
因此,每个子表单都映射到集合对象(底层数据),并具有适用于集合索引的名称,例如: '[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]=3
。 ResizeListener
将删除索引为[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
查询和循环。