Symfony 3.4: Проблемы при сохранении объекта ArrayCollection OneToMany - PullRequest
0 голосов
/ 28 апреля 2018

Мне нужна ваша помощь по моей проблеме для сохранения моих данных в таблице.
Я использую fosUserBundle и имею разные типы пользователей (Admin, Pro, Client), которые являются классами, расширяющими базовую сущность User.
У меня также есть объект Language, связанный с Пользователем ассоциацией «один ко многим».
Чтобы подписаться, пользовательский клиент должен поставить галочку напротив того языка, на котором он говорит.

Страница формы в порядке, она отображает форму UserClient (+ родительская форма пользователя) со списком языков в виде флажков.

Проблема в том, что данные сохранены в таблице. У меня такие же значения в столбце language:

O:43:"Doctrine\Common\Collections\ArrayCollection":1:{s:53:"Doctrine\Common\Collections\ArrayCollectionelements";a:2:{i:0;O:32:"LanguagesBundle\Entity\Languages":10:{s:36:"LanguagesBundle\Entity\Languagesid";i:2;s:38:"LanguagesBundle\Entity\Languagesname";s:7:"English";s:44:"LanguagesBundle\Entity\Languagesshort_slug";s:2:"en";s:43:"LanguagesBundle\Entity\Languageslong_slug";s:5:"en_EN";s:43:"LanguagesBundle\Entity\LanguagesisDefault";b:0;s:43:"LanguagesBundle\Entity\LanguagesupdatedAt";O:8:"DateTime":3:{s:4:"date";s:26:"2018-04-20 22:46:48.000000";s:13:"timezone_type";i:3;s:8:"timezone";s:12:"Europe/Paris";}s:43:"LanguagesBundle\Entity\LanguagesimageFile";N;s:39:"LanguagesBundle\Entity\Languagesimage";s:39:"Flag_of_United_Kingdom_-_Circle-512.png";s:43:"LanguagesBundle\Entity\LanguagesimageSize";i:37621;s:44:"LanguagesBundle\Entity\LanguagesuserClient";N;}i:1;O:32:"LanguagesBundle\Entity\Languages":10:{s:36:"LanguagesBundle\Entity\Languagesid";i:1;s:38:"LanguagesBundle\Entity\Languagesname";s:9:"Français";s:44:"LanguagesBundle\Entity\Languagesshort_slug";s:2:"fr";s:43:"LanguagesBundle\Entity\Languageslong_slug";s:5:"fr_FR";s:43:"LanguagesBundle\Entity\LanguagesisDefault";b:1;s:43:"LanguagesBundle\Entity\LanguagesupdatedAt";O:8:"DateTime":3:{s:4:"date";s:26:"2018-04-20 22:45:50.000000";s:13:"timezone_type";i:3;s:8:"timezone";s:12:"Europe/Paris";}s:43:"LanguagesBundle\Entity\LanguagesimageFile";N;s:39:"LanguagesBundle\Entity\Languagesimage";s:31:"Flag_of_France_-_Circle-512.png";s:43:"LanguagesBundle\Entity\LanguagesimageSize";i:23352;s:44:"LanguagesBundle\Entity\LanguagesuserClient";N;}}}

Вот мой код:

User.php

    use Doctrine\Common\Collections\ArrayCollection;
    use Symfony\Component\HttpFoundation\File\UploadedFile;
    use UserBundle\Entity\UserAdmin;
    use UserBundle\Entity\UserProfessional;
    use UserBundle\Entity\UserClient;
    use UserBundle\Entity\Adress;

    /**
     * @ORM\Entity
     * @ORM\InheritanceType("JOINED")
     * @ORM\DiscriminatorColumn(name="discr", type="string")
     * @ORM\DiscriminatorMap({"client" = "UserClient", "professional"="UserProfessional", "admin"="UserAdmin"})
     */
    abstract class User extends BaseUser {

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

        /**
         * @ORM\Column(type="string", length=255)
         */
        protected $firstname;

//[...] getters and setters

Сущность: UserClient.php

<?php

// src/UserBundle/Entity/UserGuest.php

namespace UserBundle\Entity;

use Doctrine\ORM\Mapping as ORM;
use Doctrine\Common\Collections\ArrayCollection;
/**
 * @ORM\Entity
 */
class UserClient extends User {

    /**
     * @ORM\Column(type="array")
     * @ORM\OneToMany(targetEntity="LanguagesBundle\Entity\Languages", mappedBy="userClient")
     */
    private $languages;

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

    /**
     * Set languages
     *
     * @param array $languages
     *
     * @return UserClient
     */
    public function setLanguages($languages)
    {
        $this->languages = $languages;

    }

    /**
     * Get languages
     *
     * @return array
     */
    public function getLanguages()
    {
        return $this->languages;
    }

    /**
     * Add language
     *
     * @param \LanguagesBundle\Entity\Languages $language
     *
     * @return UserClient
     */
    public function addLanguage(LanguagesBundle\Entity\Languages $language)
    {
        $this->languages[] = $language;
        return $this;
    }

    /**
     * Remove language
     *
     * @param \LanguagesBundle\Entity\Languages $language
     */
    public function removeLanguage(\LanguagesBundle\Entity\Languages $language)
    {
        $this->languages->removeElement($language);
    }
}

Entity Languages.php

<?php

namespace LanguagesBundle\Entity;

use Doctrine\ORM\Mapping as ORM;
use Symfony\Component\Validator\Constraints as Assert;
use Doctrine\Common\Collections\ArrayCollection;
use Symfony\Component\HttpFoundation\File\File;
use Vich\UploaderBundle\Mapping\Annotation as Vich;
/**
 * @ORM\Entity
 *
 * @Vich\Uploadable
 */

class Languages
{
    /**
     * @ORM\Id
     * @ORM\GeneratedValue
     * @ORM\Column(type="integer")
     */
    private $id;
    /**
     * @var string
     *
     * @ORM\Column(type="string")
     * @Assert\NotBlank()
     */
    private $name;
    /**
     * @ORM\Column(type="string", length=2, unique=true)
     */
    private $short_slug;
    /**
     * @var string
     *
     * @ORM\Column(type="string", length=5, unique=true)
     * @Assert\NotBlank()
     */
    private $long_slug;
    /**
     * @var string
     *
     * @ORM\Column(type="boolean", nullable=true)
     */
    private $isDefault;

    /**
     * @ORM\Column(type="datetime")
     * @var \DateTime
     */

    private $updatedAt;

    /**
     * NOTE: This is not a mapped field of entity metadata, just a simple property.
     * 
     * @Assert\File(
     *     maxSize="1M",
     *     mimeTypes={"image/png", "image/jpeg", "image/pjpeg"}
     * )
     * @Vich\UploadableField(mapping="language_images", fileNameProperty="image", size="imageSize")
     * @var File
     */
    private $imageFile;
    /**
     * @ORM\Column(type="string", length=255, nullable = true)
     * @var string
     */
    private $image;
/**
     * @ORM\Column(type="integer")
     *
     * @var integer
     */
    private $imageSize;


    /**
     * @ORM\ManyToOne(targetEntity="UserBundle\Entity\UserClient", inversedBy="languages")
     * @ORM\JoinColumn(name="user_id", referencedColumnName="id")
     */
    private $userClient;
    /**
     * If manually uploading a file (i.e. not using Symfony Form) ensure an instance
     * of 'UploadedFile' is injected into this setter to trigger the  update. If this
     * bundle's configuration parameter 'inject_on_load' is set to 'true' this setter
     * must be able to accept an instance of 'File' as the bundle will inject one here
     * during Doctrine hydration.
     *
     * @param File|\Symfony\Component\HttpFoundation\File\UploadedFile $image
     */
    public function setImageFile(?File $image = null): void
    {
        $this->imageFile = $image;

        if (null !== $image) {
            // It is required that at least one field changes if you are using doctrine
            // otherwise the event listeners won't be called and the file is lost
            $this->updatedAt = new \DateTimeImmutable();
        }
    }
    public function getImageFile()
    {
        return $this->imageFile;
    }
    /**
     * Constructor
     */
    public function __construct()
    {
        $this->updatedAt = new \DateTime();
    }

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

    /**
     * Set name
     *
     * @param string $name
     *
     * @return Languages
     */
    public function setName($name)
    {
        $this->name = $name;

        return $this;
    }

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

    /**
     * Set shortSlug
     *
     * @param string $shortSlug
     *
     * @return Languages
     */
    public function setShortSlug($shortSlug)
    {
        $this->short_slug = $shortSlug;

        return $this;
    }

    /**
     * Get shortSlug
     *
     * @return string
     */
    public function getShortSlug()
    {
        return $this->short_slug;
    }

    /**
     * Set longSlug
     *
     * @param string $longSlug
     *
     * @return Languages
     */
    public function setLongSlug($longSlug)
    {
        $this->long_slug = $longSlug;

        return $this;
    }

    /**
     * Get longSlug
     *
     * @return string
     */
    public function getLongSlug()
    {
        return $this->long_slug;
    }

    /**
     * Set isDefault
     *
     * @param boolean $isDefault
     *
     * @return Languages
     */
    public function setIsDefault($isDefault)
    {
        $this->isDefault = $isDefault;

        return $this;
    }

    /**
     * Get isDefault
     *
     * @return boolean
     */
    public function getIsDefault()
    {
        return $this->isDefault;
    }

    /**
     * Set updatedAt
     *
     * @param \DateTime $updatedAt
     *
     * @return Languages
     */
    public function setUpdatedAt($updatedAt)
    {
        $this->updatedAt = $updatedAt;

        return $this;
    }

    /**
     * Get updatedAt
     *
     * @return \DateTime
     */
    public function getUpdatedAt()
    {
        return $this->updatedAt;
    }

    /**
     * Set image
     *
     * @param string $image
     *
     * @return Languages
     */
    public function setImage($image)
    {
        $this->image = $image;

        return $this;
    }

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

    /**
     * Set imageSize
     *
     * @param integer $imageSize
     *
     * @return Languages
     */
    public function setImageSize($imageSize)
    {
        $this->imageSize = $imageSize;

        return $this;
    }

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


    /**
     * Set userClient
     *
     * @param \UserBundle\Entity\UserClient $userClient
     *
     * @return Languages
     */
    public function setUserClient(\UserBundle\Entity\UserClient $userClient = null)
    {
        $this->userClient = $userClient;

        return $this;
    }

    /**
     * Get userClient
     *
     * @return \UserBundle\Entity\UserClient
     */
    public function getUserClient()
    {
        return $this->userClient;
    }

    /**
     * {@inheritdoc}
     */
    public function __toString()
    {
        return $this->getName() ?: '-';
    }
}

My FormType: RegistrationTypeClient.php

<?php

// src/UserBundle/Form/UserContactFormType.php

namespace UserBundle\Form;

use Symfony\Component\Form\AbstractType;
use Symfony\Component\Form\FormBuilderInterface;
use Symfony\Component\Form\Extension\Core\Type\ChoiceType;
use Symfony\Component\OptionsResolver\OptionsResolver;
use LanguagesBundle\Services\LanguagesService;
use LanguagesBundle\Entity\Languages ;
use Symfony\Bridge\Doctrine\Form\Type\EntityType;
use Doctrine\ORM\EntityRepository;

class RegistrationClientType extends AbstractType {

    protected $service;

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

        parent::buildForm($builder, $options);

        $builder
                ->add('languages', EntityType::class, [
                'class' => 'LanguagesBundle:Languages',
                'choice_label' => 'name',
                'query_builder' => function (EntityRepository $er) {
        return $er->createQueryBuilder('u')
            ->orderBy('u.name', 'ASC');
    },
                'label'     => 'Langues',
                'expanded'  => true,
                'multiple'  => true,
            ])                 
                ;
    }

    public function getParent() {
        // on inclut le formulaire de base d'inscription utilisateur
        return 'UserBundle\Form\RegistrationType';
    }

    public function getBlockPrefix() {
        return 'app_user_registration_client';
    }

    // For Symfony 2.x
    public function getName() {
        return $this->getBlockPrefix();
    }

    public function configureOptions(OptionsResolver $resolver) {
        $resolver
                ->setDefaults(array(
            'data_class' => \UserBundle\Entity\UserClient::class,
                ))
                //->setRequired('service_languages');
                ;
    }

}

И мой контроллер:

public function registerClientAction(Request $request) {
        /** @var $formFactory FactoryInterface */
        //$formFactory = $this->get('fos_user.registration_pro.form.factory');
        /** @var $userManager UserManagerInterface */
        $userManager = $this->get('fos_user.user_manager');
        /** @var $dispatcher EventDispatcherInterface */
        $dispatcher = $this->get('event_dispatcher');

        $user = new UserClient();

        $event = new GetResponseUserEvent($user, $request);
        $dispatcher->dispatch(FOSUserEvents::REGISTRATION_INITIALIZE, $event);

        if (null !== $event->getResponse()) {
            return $event->getResponse();
        }

        $form = $this->createForm(RegistrationClientType::class, $user);

        $form->setData($user);
        $form->handleRequest($request);
        if ($form->isSubmitted()) {
            if ($form->isValid()) {
                $event = new FormEvent($form, $request);
                $dispatcher->dispatch(FOSUserEvents::REGISTRATION_SUCCESS, $event);

                $userManager->updateUser($user);

                if (null === $response = $event->getResponse()) {
                    $url = $this->generateUrl('fos_user_registration_confirmed');
                    $response = new RedirectResponse($url);
                }

                $dispatcher->dispatch(FOSUserEvents::REGISTRATION_COMPLETED, new FilterUserResponseEvent($user, $request, $response));

                return $response;
            }

            $event = new FormEvent($form, $request);
            $dispatcher->dispatch(FOSUserEvents::REGISTRATION_FAILURE, $event);

            if (null !== $response = $event->getResponse()) {
                return $response;
            }
        }

        return $this->render('UserBundle\Registration\register_client.html.twig', array(
                    'form' => $form->createView(),
        ));
    }

Вот изображение, показывающее данные, сохраненные в таблице

Мне нужно, чтобы языки сохранялись в базе данных в виде массива для userClient. Не могли бы вы помочь мне, пожалуйста.

1 Ответ

0 голосов
/ 28 апреля 2018

Спасибо за повторное редактирование моего вопроса Philip-B- Спасибо за ваше чтение.

Я решил мою проблему, добавив foreach для преобразования моего объекта в массив: Код в моем контроллере:

if ($form->isValid()) {
                $event = new FormEvent($form, $request);
                $dispatcher->dispatch(FOSUserEvents::REGISTRATION_SUCCESS, $event);

                foreach ($user->getLanguages() as $value) {
                    $langue[] = $value->getShortSlug();
                }
                $user->setLanguages($langue);
                $userManager->updateUser($user);

                if (null === $response = $event->getResponse()) {
                    $url = $this->generateUrl('fos_user_registration_confirmed');
                    $response = new RedirectResponse($url);
                }

                $dispatcher->dispatch(FOSUserEvents::REGISTRATION_COMPLETED, new FilterUserResponseEvent($user, $request, $response));

                return $response;
            }
...