19

我正在尝试将 SonataMediaBundle 与另一个实体相关联:具有多对多关系的产品。

架构和关系创建得很好。

但是,当我编辑或创建新产品时,我尝试添加一个按钮,我可以在其中通过媒体库搜索媒体文件,并添加一个按钮来上传新文件。

对于关系 OneToMany,这很容易Admin\ProductAdmin::configureFormFields通过添加:

->add('image', 'sonata_type_model_list', array(
                    'required' => false
                ), array(
                    'link_parameters' => array(
                        'context'  => 'default',
                        'provider' => 'sonata.media.provider.image'
                     )
                ))

所以我得到了与 SonataMediaBundle 画廊中已经使用的相同的 3 个图标(从库中添加上传删除

但在多对多关系上,这是不可能的!因为每次我选择一种媒体时,它都会取代之前的媒体。所以我不能选择多种媒体类型。

我想过使用与画廊相同的方式(galleryHasMedia

->add('galleryHasMedias', 'sonata_type_collection', array(
            'by_reference' => false
        ), array(
            'edit'     => 'inline',
            'inline'   => 'table',
            'sortable' => 'position',
            'link_parameters' => array('context' => $context)
        ))

然而,它真的很复杂。如何通过多对多关系在另一个实体上选择或上传多个媒体文件?

4

2 回答 2

20

我和你有同样的问题,但我已经解决了。

首先,您可能希望选择一对多/多对一关系(使用中间实体)而不是多对多关系。为什么?因为这允许额外的列,例如position列。这样,您可以以任何您想要的方式重新排序图像。在多对多关系中,链接表只有两列:关联表的 id。

教义文档

(...) 您经常希望将附加属性与关联关联,在这种情况下您需要引入关联类。因此,直接的多对多关联消失了,取而代之的是三个参与类之间的一对多/多对一关联。

所以我将此添加到我的产品映射文件中:(如您所见,我使用 YAML 作为我的配置文件格式)

oneToMany:
    images:
        targetEntity: MyBundle\Entity\ProductImage
        mappedBy: product
        orderBy:
            position: ASC

我创建了一个新的 ProductImage 映射文件:

MyBundle\Entity\ProductImage:
    type: entity
    table: product_images
    id:
        id:
            type: integer
            generator: { strategy: AUTO }
    fields:
        position:
            type: integer
    manyToOne:
        product:
            targetEntity: MyBundle\Entity\Product
            inversedBy: images
        image:
            targetEntity: Application\Sonata\MediaBundle\Entity\Media

使用命令行 ( php app/console doctrine:generate:entities MyBundle) 我创建/更新了相应的实体 (ProductProductImage)。

接下来,我创建/更新了 Admin 类。产品管理员.php:

class ProductAdmin extends Admin
{
    protected function configureFormFields(FormMapper $formMapper)
    {
        $formMapper
            // define other form fields
            ->add('images', 'sonata_type_collection', array(
                'required' => false
            ), array(
                'edit' => 'inline',
                'inline' => 'table',
                'sortable'  => 'position',
            ))
        ;
    }

ProductImageAdmin.php:

class ProductImageAdmin extends Admin
{
    protected function configureFormFields(FormMapper $formMapper)
    {
        $formMapper
            ->add('image', 'sonata_type_model_list', array(
                'required' => false
            ), array(
                'link_parameters' => array(
                    'context' => 'product_image'
                )
            ))
            ->add('position', 'hidden')
        ;
    }

不要忘记将它们都添加为服务。如果您不想在仪表板上显示 ProductImage 表单的链接,请添加show_in_dashboard: false标记。(您如何执行此操作取决于您使用的配置格式(yaml/xml/php))

在此之后,我的管理表单正常工作,但是我在尝试保存产品时仍然遇到了一些问题。我必须执行以下步骤才能解决所有问题:

首先,我必须为 Product 实体配置级联持久化操作。同样,如何执行此操作取决于您的配置格式。我使用的是 yaml,所以在images一对多的关系中,我添加了级联属性:

oneToMany:
    images:
        targetEntity: MyBundle\Entity\ProductImage
        mappedBy: product
        orderBy:
            position: ASC
        cascade: ["persist"]

这让它工作(或者我认为),但我注意到product_id数据库中的设置为NULL. 我通过在类中添加prePersist()preUpdate()方法解决了这个问题ProductAdmin

public function prePersist($object)
{
    foreach ($object->getImages() as $image) {
        $image->setProduct($object);
    }
}

public function preUpdate($object)
{
    foreach ($object->getImages() as $image) {
        $image->setProduct($object);
    }
}

...并在实体的addImages()方法中添加了一行:Product

public function addImage(\MyBundle\Entity\ProductImage $images)
{
    $images->setProduct($this);
    $this->images[] = $images;

    return $this;
}

这对我有用,现在我可以在我的产品中添加、更改、重新排序、删除等图像。

于 2014-02-25T10:14:16.063 回答
4

您需要依赖 MediaBundle Gallery。在您的实体中,您类似于:

/**
 * @ORM\ManyToOne(targetEntity="Application\Sonata\MediaBundle\Entity\Gallery")
 * @ORM\JoinColumn(name="image", referencedColumnName="id")
 */
private $images;

然后在您的表单中,您将能够使用以下内容将图库链接到您的对象:

->add('images', 'sonata_type_model_list', array('required' => false), array('link_parameters' => array('context' => $context)))
于 2014-02-18T11:04:33.373 回答