2016-03-09 113 views
2

我需要關於Symfony2表單一對一關係更新的幫助。我有實體:User和的UserInfoSymfony2表單更新一對一關係

用戶實體:

class User extends BaseUser 
{ 
    /** 
    * @ORM\Id 
    * @ORM\Column(type="integer") 
    * @ORM\GeneratedValue(strategy="AUTO") 
    */ 
    protected $id; 

    /** 
    * @ORM\ManyToMany(targetEntity="AppBundle\Entity\Group") 
    * @ORM\JoinTable(name="fos_user_user_group", 
    *  joinColumns={@ORM\JoinColumn(name="user_id", referencedColumnName="id")}, 
    *  inverseJoinColumns={@ORM\JoinColumn(name="group_id", referencedColumnName="id")} 
    *) 
    */ 
    protected $groups; 

    /** 
    * @ORM\OneToOne(targetEntity="UserInfo", mappedBy="user", cascade={"all"}) 
    */ 
    private $info; 

    /** 
    * @ORM\OneToMany(targetEntity="Log", mappedBy="user") 
    */ 
    private $logs; 

    public function __construct() 
    { 
     parent::__construct(); 
     $this->logs = new ArrayCollection(); 
    } 

    /** 
    * Set info 
    * 
    * @param \AppBundle\Entity\UserInfo $info 
    * @return User 
    */ 
    public function setInfo(\AppBundle\Entity\UserInfo $info = null) 
    { 
     $this->info = $info; 
     $info->setUser($this); 

     return $this; 
    } 

    /** 
    * Get info 
    * 
    * @return \AppBundle\Entity\UserInfo 
    */ 
    public function getInfo() 
    { 
     return $this->info; 
    } 

    /** 
    * Add logs 
    * 
    * @param \AppBundle\Entity\Log $logs 
    * @return User 
    */ 
    public function addLog(\AppBundle\Entity\Log $logs) 
    { 
     $this->logs[] = $logs; 

     return $this; 
    } 

    /** 
    * Remove logs 
    * 
    * @param \AppBundle\Entity\Log $logs 
    */ 
    public function removeLog(\AppBundle\Entity\Log $logs) 
    { 
     $this->logs->removeElement($logs); 
    } 

    /** 
    * Get logs 
    * 
    * @return \Doctrine\Common\Collections\Collection 
    */ 
    public function getLogs() 
    { 
     return $this->logs; 
    } 
} 

的UserInfo實體:

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

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

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

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

    /** 
    * @ORM\OneToOne(targetEntity="User", inversedBy="info") 
    */ 
    private $user; 

    /** 
    * Get id 
    * 
    * @return integer 
    */ 
    public function getId() 
    { 
     return $this->id; 
    } 

    /** 
    * Set position 
    * 
    * @param string $position 
    * @return UserInfo 
    */ 
    public function setPosition($position) 
    { 
     $this->position = $position; 

     return $this; 
    } 

    /** 
    * Get position 
    * 
    * @return string 
    */ 
    public function getPosition() 
    { 
     return $this->position; 
    } 

    /** 
    * Set info 
    * 
    * @param string $info 
    * @return UserInfo 
    */ 
    public function setInfo($info) 
    { 
     $this->info = $info; 

     return $this; 
    } 

    /** 
    * Get info 
    * 
    * @return string 
    */ 
    public function getInfo() 
    { 
     return $this->info; 
    } 

    /** 
    * Set fullname 
    * 
    * @param string $fullname 
    * @return UserInfo 
    */ 
    public function setFullname($fullname) 
    { 
     $this->fullname = $fullname; 

     return $this; 
    } 

    /** 
    * Get fullname 
    * 
    * @return string 
    */ 
    public function getFullname() 
    { 
     return $this->fullname; 
    } 

    /** 
    * Set user 
    * 
    * @param \AppBundle\Entity\User $user 
    * @return UserInfo 
    */ 
    public function setUser(\AppBundle\Entity\User $user = null) 
    { 
     $this->user = $user; 

     return $this; 
    } 

    /** 
    * Get user 
    * 
    * @return \AppBundle\Entity\User 
    */ 
    public function getUser() 
    { 
     return $this->user; 
    } 
} 

這些都是他們的類型:

class UserType extends AbstractType 
{ 
    public function buildForm(FormBuilderInterface $builder, array $options) 
    { 
     $builder 
      ->add('username') 
      ->add('email') 
      ->add('password') 
      ->add('info', new UserInfoType(), [ 
       'by_reference' => false, 
      ]) 
      ->add('Submit', 'submit') 
     ; 
    } 

    public function configureOptions(OptionsResolver $resolver) 
    { 
     $resolver->setDefaults([ 
      'data_class' => 'AppBundle\Entity\User', 
     ]); 
    } 

    public function getName() 
    { 
     return 'user'; 
    } 
} 

class UserInfoType extends AbstractType 
{ 
    public function buildForm(FormBuilderInterface $builder, array $options) 
    { 
     $builder 
      ->add('position') 
      ->add('info') 
      ->add('fullname') 
     ; 
    } 

    public function configureOptions(OptionsResolver $resolver) 
    { 
     $resolver->setDefaults([ 
      'data_class' => 'AppBundle\Entity\UserInfo', 
      'allow_delete' => true, 
     ]); 
    } 

    public function getName() 
    { 
     return 'userinfo'; 
    } 
} 

新記錄插入沒有任何問題,但是當我要更新這個紀錄,這個動作:

public function editAction(Request $request, $id) 
{ 
    $em = $this->getDoctrine()->getManager(); 

    $userRepository = $em->getRepository('AppBundle:User'); 

    $user = $userRepository->findOneBy(['id' => $id]); 

    $form = $this->createForm(new UserType(), $user); 

    $form->handleRequest($request); 

    if ($form->isSubmitted() && $form->isValid()) { 
     $em->persist($user); 
     $em->flush(); 

     return $this->redirectToRoute('user_list'); 
    } 

    return $this->render('@App/User/form.html.twig', [ 
     'form' => $form->createView(), 
    ]); 
} 

我得到這個錯誤:

An exception occurred while executing 'INSERT INTO user_info (position, info, fullname, user_id) VALUES (?, ?, ?, ?)' with params ["321", "321", "321", 14]: 

SQLSTATE[23000]: Integrity constraint violation: 1062 Duplicate entry '14' for key 'UNIQ_B1087D9EA76ED395' 

好吧,我明白,他想插入新記錄,但我該怎麼做才能讓他更新或刪除現有的記錄,並插入新的記錄。

+0

如何將'fetch =「EAGER」'添加到'User'類'$ info'屬性? –

+0

@PawełMikołajczukNop,仍然一樣。 – Justinas

回答

3

嘗試使User擁有方和UserInfo逆之一(見docs)。

class User extends BaseUser 
{ 
    /** 
    * @ORM\OneToOne(targetEntity="UserInfo", inversedBy="user", cascade={"all"}) 
    */ 
    private $info; 
} 

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

/** 
    * @ORM\OneToOne(targetEntity="User", mappedBy="info") 
    */ 
    private $user; 
} 

還與this SO question

編輯:要在更新後刪除舊的,未引用UserInfo記錄,設置在User類的info財產orphanRemoval選項。

class User extends BaseUser 
{ 
    /** 
    * @ORM\OneToOne(targetEntity="UserInfo", inversedBy="user", cascade={"all"}, orphanRemoval=true) 
    */ 
    private $info; 
} 
+0

現在它將信息插入'UserInfo'表並更新'User'表中的'info_id'列..舊記錄只是掛在表中..如何刪除舊記錄? – Justinas

+0

嘗試Pawel的評論,並將'fetch =「EAGER」'添加到'User'類的'info'屬性中。 –

+0

仍然沒有刪除 – Justinas

0

您需要使用JoinColumn註釋在User類:

/** 
* @ORM\OneToOne(targetEntity="UserInfo", inversedBy="user", cascade={"all"}) 
* @ORM\JoinColumn(name="info", referencedColumnName="id") 
*/ 
private $info; 

的UserInfo:

/** 
* @ORM\OneToOne(targetEntity="User", mappedBy="info") 
*/ 
private $user; 
+0

仍然是相同的錯誤 – Justinas