7

Customer是“关键字/客户”关系的反面Keyword

/**
 * @ORM\ManyToMany(targetEntity="Keyword", mappedBy="customers",
 *     cascade={"persist", "remove"}
 * )
 */
protected $keywords;

创建新客户时,应选择一个或多个关键字。实体表单字段为:

$form->add($this->factory->createNamed('entity', 'keywords', null, array(
    'class'    => 'Acme\HelloBundle\Entity\Keyword',
    'property' => 'select_label',
    'multiple' => true,
    'expanded' => true,
)));

在我的控制器代码中,绑定请求并检查表单是否有效后,我需要保留客户和所有客户/关键字关联,即连接表。

但是客户是反面,所以这是行不通的:

if($request->isPost()) {
    $form->bindRequest($request);

    if(!$form->isValid()) {
        return array('form' => $form->createView());
    }

    // Valid form here   
    $em = $this->getEntityManager();

    $em->persist($customer);    
    $em->flush();
}

带有“级联”选项的事件,此代码失败。$customer->getKeywords()将返回Doctrine\ORM\PersistentCollection,其中仅包含选定的关键字。

我是否应该手动检查删除/添加了哪个关键字,然后从拥有方更新?

4

2 回答 2

11

好的,找到了方法,即使我并不完全满意。关键是这个示例表单集合字段类型。基本上,我之前的表单定义发生的情况是:

$customer->getKeywords() = $postData; // $postData is somewhere in form framework

这只是将(选定关键字的)集合分配给客户关键字。Keyword没有在实例(拥有方)上调用任何方法。关键选项是by_reference(对我来说这只是一个坏名字,但无论如何......):

$form
    ->add($this->factory->createNamed('entity', 'keywords', null, array(
        // ...
        'by_reference' => false
    ))
);

这样表单框架将调用 setter,即$customer->setKeywords(Collection $keywords). 在该方法中,您可以“告诉”拥有方存储您的关联:

public function setKeywords(Collection $keywords)
{
    foreach($keywords as $keyword) {
        $keyword->addCustomer($this); // Owning side call!
    }

    $this->keywords = $keywords;

    return $this;
}

(始终使用方法检查拥有方的重复实例contains)。

此时,只会添加选中的关键字($keyword参数)。需要管理删除未经检查的关键字(控制器端):

$originalKeywords = $customer->getKeywords()->toArray(); // When GET or POST

// When POST and form valid
$checkedKeywords = $customer->getKeywords()->toArray(); // Thanks to setKeywords

// Loop over all keywords
foreach($originalKeywords as $keyword) {
    if(!in_array($keyword, $checkedKeywords)) { // Keyword has been unchecked
        $keyword->removeCustomer($customer);
        $manager->persist($keyword);
    }
}

丑陋,但有效。我会将删除代码移到Customer课堂上,但这根本不可能。如果您能找到更好的解决方案,请告诉我!

于 2012-11-17T03:24:00.993 回答
3

我使用的解决方案与@gredmo 略有不同。从学说文档中:当您满足此假设时,您可以使用 Orphan Removal :

当使用orphanRemoval=true选项 Doctrine 时,假设实体是私有的,不会被其他实体重用。如果您忽略此假设,即使您将孤立实体分配给另一个实体,您的实体也会被 Doctrine 删除。

我有这个实体类:

class Contract {
/**
 * @ORM\OneToMany(targetEntity="ContractParticipant", mappedBy="contract", cascade={"all"}, orphanRemoval=true)
 **/
}
protected $participants;

表单处理(伪代码):

    // $_POST carry the Contract entity values

    $received = [];

    foreach ($_POST['participants'] as $key => $participant) {

        if ((!$relation = $collection->get($key))) {
            // new entity
            $collection[$key] = $relation = $relationMeta->newInstance();

        } else {
            // editing existing entity
        }

        $received[] = $key;
        $this->mapper->save($relation, $participant);   // map POST data to entity
    }

    foreach ($collection as $key => $relation) {
        if ($this->isAllowedRemove() && !in_array($key, $received)) {
            // entity has been deleted
            unset($collection[$key]);
        }
    }

不要忘记持久化实体结束刷新。Flush 还会删除已删除的实体。

    $this->em->persist($entity);
    $this->em->flush();
于 2015-07-14T09:21:53.090 回答