Symfony2更新“子表单”中的项目

时间:2015-05-18 19:15:01

标签: php symfony doctrine-orm symfony-forms

我的问题的简短版本:

如何在Symfony2中编辑子表单的实体?

= - = - = - = - = - = - =长而详细的版本= - = - = - = - = - = - = - =

我有一个实体订单

<?php

class Order
{
    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="integer")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

    /**
     * @ORM\ManyToOne(targetEntity="Customer")
     * @ORM\JoinColumn(name="customer_id", referencedColumnName="id", nullable=false)
     **/
    private $customer;

    /**
     * @var \DateTime
     *
     * @ORM\Column(name="date", type="date")
     */
    private $date;

    /**
     * @ORM\ManyToOne(targetEntity="\AppBundle\Entity\OrderStatus")
     * @ORM\JoinColumn(name="order_status_id", referencedColumnName="id", nullable=false)
     **/
    private $orderStatus;

    /**
     * @var string
     *
     * @ORM\Column(name="reference", type="string", length=64)
     */
    private $reference;

    /**
     * @var string
     *
     * @ORM\Column(name="comments", type="text")
     */
    private $comments;

    /**
     * @var array
     *
     * @ORM\OneToMany(targetEntity="OrderRow", mappedBy="Order", cascade={"persist"})
     */
    private $orderRows;

    ...
}

的MySQL

_____________________________________________________________
|id                           | order id                    |
|customer_id                  | fk customer.id NOT NULL     |
|date                         | order date                  |
|order_status_id              | fk order_status.id NOT NULL |
|reference                    | varchar order reference     |
|comments                     | text comments               |
|___________________________________________________________|

实体OrderRow(订单可以包含一行或多行)

<?php

class OrderRow
{
    /**
     * @var integer
     *
     * @ORM\Column(name="id", type="integer")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="AUTO")
     */
    private $id;

    /**
     * @ORM\ManyToOne(targetEntity="Order", inversedBy="orderRows", cascade={"persist"})
     * @ORM\JoinColumn(name="order_id, referencedColumnName="id", nullable=false)
     **/
    private $order;

    /**
     * @ORM\ManyToOne(targetEntity="[MyShop\Bundle\ProductBundle\Entity\Product")
     * @ORM\JoinColumn(name="product_id", referencedColumnName="id", nullable=true)
     **/
    private $product;

    /**
     * @var string
     *
     * @ORM\Column(name="description", type="string", length=255)
     */
    private $description;

    /**
     * @var integer
     *
     * @ORM\Column(name="count", type="integer")
     */
    private $count = 1;

    /**
     * @var \DateTime
     *
     * @ORM\Column(name="date", type="date")
     */
    private $date;

    /**
     * @var decimal
     *
     * @ORM\Column(name="amount", type="decimal", precision=5, scale=2)
     */
    private $amount;

    /**
     * @var string
     *
     * @ORM\Column(name="tax_amount", type="decimal", precision=5, scale=2)
     */
    private $taxAmount;

    /**
     * @var string
     *
     * @ORM\Column(name="discount_amount", type="decimal", precision=5, scale=2)
     */
    private $discountAmount;

    ...
}

的MySQL

_____________________________________________________________
|id                           | order id                    |
|order_id                     | fk order.id NOT NULL        |
|product_id                   | fk product.id               |
|description                  | varchar product description |
|count                        | int count                   |
|date                         | date                        |
|amount                       | amount                      |
|taxAmount                    | tax amount                  |
|discountAmount               | discount amount             |
|___________________________________________________________|

我想创建一个允许编辑一个订单及其行的表单。

OrderType.php     

class OrderType extends AbstractType
{
    /**
     * @param FormBuilderInterface $builder
     * @param array $options
     */
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder
            ->add('customer', 'entity', array(
                'class' => 'Customer',
                'multiple' => false
            ))
            ->add('orderStatus', 'entity', array(
                'class' => 'AppBundle\Entity\OrderStatus',
                'multiple' => false
            ))
            ->add('date')
            ->add('reference')
            ->add('comments')
            ->add('orderRows', 'collection', [
                'type' => new OrderRowType(),
                'allow_add' => true,
                'by_reference' => false,
            ])
        ;
    }

    ...
}

OrderRowType.php     

class OrderRowType extends AbstractType
{
    /**
     * @param FormBuilderInterface $builder
     * @param array $options
     */
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder
            ->add('order', 'entity', array(
                'class' => 'MyShop\Bundle\OrderBundle\Entity\Order',
                'multiple' => false
            ))
            ->add('product', 'product_selector') // service
            ->add('orderRowStatus', 'entity', array(
                'class' => 'AppBundle\Entity\OrderRowStatus',
                'multiple' => false
            ))
            ->add('description')
            ->add('count')
            ->add('startDate')
            ->add('endDate')
            ->add('amount')
            ->add('taxAmount')
            ->add('discountAmount')
        ;
    }

    ...
}

通过向我的API发送请求来更新订单:

  • 请求网址:https://api.example.net/admin/orders/update/37
  • 请求方法:POST
  • 状态代码:200

    Params: {
    
    "order[customer]": "3",
    "order[orderStatus]": "1",
    "order[date][month]:": "5",
    "order[date][day]": "18",
    "order[date][year]": "2015",
    "order[reference]": "Testing",
    "order[comments]": "I have nothing to say!",
    "order[orderRows][0][order]": "32",
    "order[orderRows][0][product]": "16721",
    "order[orderRows][0][orderRowStatus]:1": "1",
    "order[orderRows][0][description]": "8 GB memory",
    "order[orderRows][0][count]": "12",
    "order[orderRows][0][startDate][month]": "5",
    "order[orderRows][0][startDate][day]": "18",
    "order[orderRows][0][startDate][year]": "2015",
    "order[orderRows][0][endDate][month]": "5",
    "order[orderRows][0][endDate][day]": "18",
    "order[orderRows][0][endDate][year]": "2015",
    "order[orderRows][0][amount]": "122.03",
    "order[orderRows][0][taxAmount]": "25.63",
    "order[orderRows][0][discountAmount]": "0",
    "order[orderRows][1][order]": "32",
    "order[orderRows][1][product]": "10352",
    "order[orderRows][1][orderRowStatus]": "2",
    "order[orderRows][1][description]": "12 GB MEMORY",
    "order[orderRows][1][count]": "1",
    "order[orderRows][1][startDate][month]": "5",
    "order[orderRows][1][startDate][day]": "18",
    "order[orderRows][1][startDate][year]": "2015",
    "order[orderRows][1][endDate][month]": "5",
    "order[orderRows][1][endDate][day]": "18",
    "order[orderRows][1][endDate][year]": "2015",
    "order[orderRows][1][amount]": "30.8",
    "order[orderRows][1][taxAmount]": "6.47",
    "order[orderRows][1][discountAmount]": "0",
    "order[orderRows][2][order]": "32",
    "order[orderRows][2][product]": "2128",
    "order[orderRows][2][orderRowStatus]": "3",
    "order[orderRows][2][description]": "4GB MEMORY",
    "order[orderRows][2][count]": "5",
    "order[orderRows][2][startDate][month]": "5",
    "order[orderRows][2][startDate][day]": "18",
    "order[orderRows][2][startDate][year]": "2015",
    "order[orderRows][2][endDate][month]": "5",
    "order[orderRows][2][endDate][day]": "18",
    "order[orderRows][2][endDate][year]": "2015",
    "order[orderRows][2][amount]": "35.5",
    "order[orderRows][2][taxAmount]": "7.46",
    "order[orderRows][2][discountAmount]": "0"
    }
    

上面的请求编辑订单详细信息并创建新的order_rows,因为没有提供order_row_id。现在在Symfony2中,我发现我应该只将$ builder-&gt; add('id')添加到我的OrderRowType中,我的实体也不会有列ID的设置者。

经过大量的信息后,我的问题非常简短。我该如何更新此表单中的order_rows记录?

3 个答案:

答案 0 :(得分:7)

如果您不了解内部情况,处理收集和学说有时会很复杂。我将首先向您提供有关内部结构的一些信息,以便您更清楚地了解幕后的内容。

很难从您提供的详细信息中估计实际问题,但我给您一些建议,可以帮助您调试问题。我给出了一个广泛的答案,因此它可以帮助其他人。

TL; DR版本

以下是我的猜测:您正在通过引用修改实体,即使您将by_reference设置为false也是如此。这可能是因为您还没有定义addOrderRowremoveOrderRow方法(两者都有)或者因为您没有使用教条集合对象

一些内部

表格

在控制器中创建Form对象时,绑定与从数据库中检索到的实体(即带有ID),或者您刚创建的实体:这意味着表单不要需要主实体的id,也不需要集合对象的id。为方便起见,您可以将其添加到表单中,但如果确实它们是不可变的(例如hidden键入disabled => true选项)。

创建Collection表单时,Symfony会自动为实体集合中已存在的每个实体创建一个子表单;这就是为什么在entity/<id>/edit动作中你应该总是看到已经存在的集合元素的可编辑形式。

allow_addallow_delete选项控制是否可以动态调整生成的子表单的大小,删除集合的某些元素,或者添加新元素(请参阅ResizeFormListener类) 。请注意,当您使用带有javascript的prototype时,必须谨慎使用__prototype__占位符:这是用于重新映射对象服务器端的实际key,因此如果您更改它表单将在集合中创建一个新元素。

学说

在Doctrine中,您需要好好照顾owning sideinverse side映射。 owning侧是将持久关联到数据库的实体,而反面是另一个实体。持久化时,owning侧是唯一触发保存关系的方。在对象修改期间,保持两个关系同步是一种模范职责。

在处理一对多关系时,owning方是many(例如OrderRow},而one是{ {1}}方。

最后,应用程序需要明确标记要保留的实体。关系的两侧都可以标记为inverse,因此通过关系的所有可到达实体也会被持久化。在此过程中,所有新实体都会自动保留,并且(在标准配置中)全部&#34;脏&#34;实体已更新。

the official docs中很好地解释了脏实体的概念。默认情况下,Doctrine会通过将每个属性与原始状态进行比较来自动检测更新的实体,并在刷新期间生成persist cascading语句。如果明确提高性能(即UPDATE),则必须手动保留所有实体,即使关系标记为级联。

另请注意,当从DB重新加载实体时,Doctrine使用@ChangeTrackingPolicy("DEFERRED_EXPLICIT")实例来处理集合,因此您需要使用doctrine集合接口来处理实体集合。

检查什么

总而言之,这是一个(希望完整的)列表,可以检查正确的集合更新。

学说关系的两个方面都设置正确

  1. 拥有方和反方都应标记为级联持久(如果不是控制器必须手动级联...不推荐,除非太慢);
  2. 集合属性必须是PersistenCollection的实现,而不是简单的数组;
  3. 模型必须在每次更改时相互更新,因此这意味着
  4. 集合对象不应该按原样返回,以避免通过引用进行修改。
  5. 在你的情况下:

    Doctrine\Common\Collection

    表格集合已正确配置

    1. 确保表单未公开订单<?php class Order { /** * @var integer * * @ORM\Column(name="id", type="integer") * @ORM\Id * @ORM\GeneratedValue(strategy="AUTO") */ private $id; /** * @var \Doctrine\Common\Collections\Collection * @ORM\OneToMany(targetEntity="OrderRow", mappedBy="Order", cascade={"persist"}) */ private $orderRows; public function __construct() { // this is required, as Doctrine will replace it by a PersistenCollection on load $this->orderRows = new \Doctrine\Common\Collections\ArrayCollection(); } /** * Add order row * * @param OrderRow $row */ public function addOrderRow(OrderRow $row) { if (! $this->orderRows->contains($row)) $this->orderRows[] = $row; $row->setOrder($this); } /** * Remove order row * * @param OrderRow $row */ public function removeOrderRow(OrderRow $row) { $removed = $this->orderRows->removeElement($row); /* // you may decide to allow your domain to have spare rows, with order set to null if ($removed) $row->setOrder(null); */ return $removed; } /** * Get order rows * @return OrderRow[] */ public function getOrders() { // toArray prevent edit by reference, which breaks encapsulation return $this->orderRows->toArray(); } } class OrderRows { /** * @var integer * * @ORM\Column(name="id", type="integer") * @ORM\Id * @ORM\GeneratedValue(strategy="AUTO") */ private $id; /** * @var Order * @ORM\ManyToOne(targetEntity="Order", inversedBy="orderRows", cascade={"persist"}) * @ORM\JoinColumn(name="order_id, referencedColumnName="id", nullable=false) */ private $order; /** * Set order * * @param Order $order */ public function setOrder(Order $order) { // avoid infinite loops addOrderRow -> setOrder -> addOrderRow if ($this->order === $order) { return; } if (null !== $this->order) { // see the comment above about spare order rows $this->order->removeOrderRow($this); } $this->order = $order; } /** * Get order * * @return Order */ public function getOrder() { return $this->order; } } (但在模板中包含路由器操作的正确id参数)
    2. 确保OrderRow GET不存在,因为这将由模型类自动更新
    3. 确保order设置为by_reference
    4. 确保在false
    5. 中定义addOrderRowremoveOrderRow
    6. 加快调试速度,确保Order不直接返回集合
    7. 这里是片段:

      Order::getOrderRows

      控制器必须正确更新实体

      1. 确保正确创建表单;
      2. 如果使用class OrderType extends AbstractType { public function buildForm(FormBuilderInterface $builder, array $options) { $builder ->add('orderRows', 'collection', [ 'type' => new OrderRowType(), 'allow_add' => true, // without, new elements are ignored 'allow_delete' => true, // without, deleted elements are not updated 'by_reference' => false, // hint Symfony to use addOrderRow and removeOrderRow // NOTE: both method MUST exist, or Symfony will ignore the option ]) ; } } class OrderRowType extends AbstractType { public function buildForm(FormBuilderInterface $builder, array $options) { $builder // ->add('order') NOT required, the model will handle the setting ->add('product', 'product_selector') // service ; } } ,请确保HTTP方法与Form方法属性匹配
      3. 如果表单有效,请处理集合中已删除的元素
      4. 如果表单有效,则保留实体然后刷新
      5. 在你的情况下你应该有这样的行动:

        Form::handleRequest

        希望这有帮助!

答案 1 :(得分:1)

我不相信这是可能的,原因如下:

OrderRows仅由其ID标识,因此为了让Doctrine知道实际更新了哪个实体,需要知道id。但是你需要将OrderRow id作为一个字段添加,你不想这样做,因为这样可以改变&#39; foreign&#39; OrderRows不属于订单。 (没有复杂的权限检查)

解决方案是完全删除旧的OrderRows并插入新的OrderRows。 插入已经有效: - )。

Doctrine:确保数据库持久性

中的the cookbook中描述了

删除实体

只有一个小缺点:当订单更新时,OrderRows会获得新的ID。

答案 2 :(得分:1)

mappedBy应该是 o rder而不是 O rder,因为它指向属性而不是类名。

...