Symfony:更新多对多

问题描述 投票:0回答:2

我有

Post
Tag
具有多对多关系的实体。在
Post
创建和编辑表单中有一个文本框,我可以在其中输入与该帖子相关的用逗号分隔的标签。例如,当我为标题为“Post1”的帖子输入 tag1、tag2、tag3 时,表单将创建帖子和标签实体,并将这些标签添加到该帖子的标签列表中。我使用数据转换器来创建标签实体。

class Post{
    /**
     * @ORM\ManyToMany(targetEntity="Tag", mappedBy="posts",cascade={"all"})
     */ 
    protected $tags;

    public function __construct() {
            $this->tags = new ArrayCollection();
        }

    /**
     * @return ArrayCollection
     */
    public function getTags()
    {
        return $this->tags;
    }

    /**
     * @param Tag $tag
     */
    public function addTag(Tag $tag)
    {
        $tag->addPost($this);

        $this->tags->add($tag);
    }

    /**
     * @param Tag $tag
     */
    public function removeTag(Tag $tag)
    {
        $this->tags->removeElement($tag);
    }
}

帖子类型

 public function buildForm(FormBuilderInterface $builder, array $options)
{

    $builder
        ->add('title', TextType::class, array('label' => 'Title'))
        ->add('tags', TextType::class, array('label' => 'Tags'))
    ;

    $builder->get('tags')
        ->addModelTransformer(new TagViewTransformer($this->manager));
}

标签视图转换器

class TagViewTransformer implements DataTransformerInterface
{

public function transform($value)
{
    /...
}

public function reverseTransform($value)
{
    $tags = array();
    if ( $value )
    {
        if( strpos($value, ',') !== false )
        {
            $list = array_unique(explode(",", $value));
        }
        else
        {
            $list = array(trim($value));
        }

        foreach ( $list AS $tagName )
        {
            $tag = $this->em
                ->getRepository('CoreBundle:Tag')
                ->findOneBy(array('name' => trim($tagName)));

            if( !$tag )
            {
                $tag = new Tag();
                $tag->setName(trim($tagName));
                $this->em->persist($tag);
            }

            $tags[] = $tag;
        }
     }
    return $tags;
    }
}

当我尝试创建帖子时,这工作正常,所有标签都转换为实体并添加到帖子的标签列表中。但当我尝试编辑时,我开始遇到问题

 public function editAction(Request $request, Post $post)
{
    $deleteForm = $this->createDeleteForm($post);
    $editForm = $this->createForm(PostType::class, $post);
    $editForm->handleRequest($request);

    $originalTags = $post->getTags();

    if ($editForm->isSubmitted() && $editForm->isValid()) {
        $em = $this->getDoctrine()->getManager();
        $newTags = $editForm->get('tags')->getData();

        foreach ($originalTags as $currentTag) {
            if (!in_array($currentTag, $newTags)) {
                $post->removeTag($currentTag);
            }
        }

        $em->persist($post);
        $em->flush();

        return $this->redirectToRoute('post_show', array('id' => $post->getId()));
    }

    return $this->render('AppBundle:Post:edit.html.twig', array(
        'entity' => $post,
        'form' => $editForm->createView(),
        'delete_form' => $deleteForm->createView(),
    ));
}

假设帖子有标签:

tag1, tag2, tag3
,但我想删除
tag3
并添加
tag4
。所以我将标签文本框更改为
tag1, tag2, tag4
。然而,当我提交表格时,我得到
tag1, tag2, tag3, tag4
。所以 tag3 并没有从 Post 的标签列表中删除。

editAction 代码有什么问题?

php symfony doctrine-orm symfony-forms
2个回答
0
投票

试试这个

public function editAction(Request $request, Post $post)
{
    $deleteForm = $this->createDeleteForm($post);
    $editForm = $this->createForm(PostType::class, $post);
    $editForm->handleRequest($request);



    if ($editForm->isSubmitted() && $editForm->isValid()) {
        $em = $this->getDoctrine()->getManager();

        $em->persist($post);
        $em->flush();

        return $this->redirectToRoute('post_show', array('id' => $post->getId()));
    }

    return $this->render('AppBundle:Post:edit.html.twig', array(
        'entity' => $post,
        'form' => $editForm->createView(),
        'delete_form' => $deleteForm->createView(),
    ));
}

0
投票

使用孤儿删除来执行此操作:

class Post
{
    /**
     * @ORM\ManyToMany(targetEntity="Tag", mappedBy="posts", cascade={"all"}, orphanRemoval=true)
     */ 
    protected $tags;

    /* Rest of your class */

}

做一些测试,也许这里不需要全部级联。

© www.soinside.com 2019 - 2024. All rights reserved.