Преглед изворни кода

Relation Realisateur Film en ManyToMany

François пре 6 година
родитељ
комит
e8d6232650

+ 1 - 1
src/AppBundle/Controller/RealisateurController.php

@@ -83,7 +83,7 @@ class RealisateurController extends Controller
             $em = $this->getDoctrine()->getManager();
             foreach ($realisateur->getFilms() as $film)
             {
-                $film->setRealisateur(null);
+                $film->removeRealisateur($realisateur);
             }
             $em->remove($realisateur);
             $em->flush();

+ 0 - 1
src/AppBundle/Controller/VideothequeController.php

@@ -15,7 +15,6 @@ class VideothequeController extends Controller
 	 */
 	public function indexAction()
 	{
-	    
 		return $this->redirectToRoute('videotheque_liste');
 	}
 

+ 30 - 12
src/AppBundle/Entity/Film.php

@@ -50,10 +50,10 @@ class Film
     private $note;
 
     /**
-     * @ORM\ManyToOne(targetEntity="AppBundle\Entity\Realisateur", inversedBy="films", cascade="persist")
-     * @ORM\JoinColumn(nullable=true)
+     * @ORM\ManyToMany(targetEntity="AppBundle\Entity\Realisateur", inversedBy="films", cascade="persist")
+     * @var \Doctrine\Common\Collections\Collection
      */
-    private $realisateur;
+    private $realisateurs;
 
 
     /**
@@ -163,26 +163,44 @@ class Film
     }
 
     /**
-     * Set realisateur
+     * Constructor
+     */
+    public function __construct()
+    {
+        $this->realisateurs = new \Doctrine\Common\Collections\ArrayCollection();
+    }
+
+    /**
+     * Add film
      *
-     * @param \AppBundle\Entity\Realisateur $realisateur
+     * @param \AppBundle\Entity\Film $film
      *
-     * @return Film
+     * @return Realisateurs
      */
-    public function setRealisateur(\AppBundle\Entity\Realisateur $realisateur = null)
+    public function addRealisateur(\AppBundle\Entity\Realisateur $realisateur)
     {
-        $this->realisateur = $realisateur;
+        $this->realisateurs[] = $realisateur;
 
         return $this;
     }
 
     /**
-     * Get realisateur
+     * Remove film
+     *
+     * @param \AppBundle\Entity\Film $film
+     */
+    public function removeRealisateur(\AppBundle\Entity\Realisateur $realisateur)
+    {
+        $this->realisateurs->removeElement($realisateur);
+    }
+
+    /**
+     * Get realisateurs
      *
-     * @return \AppBundle\Entity\Realisateur
+     * @return \Doctrine\Common\Collections\Collection
      */
-    public function getRealisateur()
+    public function getRealisateurs()
     {
-        return $this->realisateur;
+        return $this->realisateurs;
     }
 }

+ 1 - 1
src/AppBundle/Entity/Realisateur.php

@@ -36,7 +36,7 @@ class Realisateur
     private $nom;
 
     /**
-     * @ORM\OneToMany(targetEntity="AppBundle\Entity\Film", mappedBy="realisateur", cascade="persist")
+     * @ORM\ManyToMany(targetEntity="AppBundle\Entity\Film", mappedBy="realisateurs", cascade="persist")
      * @var \Doctrine\Common\Collections\Collection
      */
     private $films;

+ 12 - 1
src/AppBundle/Form/FilmType.php

@@ -6,6 +6,7 @@ use AppBundle\Entity\Realisateur;
 use AppBundle\Repository\RealisateurRepository;
 use Symfony\Bridge\Doctrine\Form\Type\EntityType;
 use Symfony\Component\Form\Extension\Core\Type\ChoiceType;
+use Symfony\Component\Form\Extension\Core\Type\CollectionType;
 use Symfony\Component\Form\Extension\Core\Type\RangeType;
 use Symfony\Component\Form\Extension\Core\Type\TextType;
 use Symfony\Component\Form\AbstractType;
@@ -32,10 +33,20 @@ class FilmType extends AbstractType
                 'format'    =>  'yyyy',
                 'html5'     =>  true,
             ))
-            ->add('realisateur', EntityType::class, array(
+            /*->add('realisateur', EntityType::class, array(
                 'class'         =>  'AppBundle\Entity\Realisateur',
                 'choice_label'  =>  'nomComplet',
                 'required'      =>  false
+            ))*/
+            ->add('realisateurs', CollectionType::class, array(
+                'entry_type'    =>  EntityType::class,
+                'entry_options'    =>  array(
+                        'class'     =>  'AppBundle\Entity\Realisateur',
+                        'choice_label'  =>  'nomComplet',
+                        'required'  =>  false
+                ),
+                'allow_add'     =>  true,
+                'allow_delete'  =>  true
             ))
             ->add('commentaire', TextareaType::class, array(
                 'required'  => false

+ 4 - 4
src/AppBundle/Form/RealisateurType.php

@@ -3,8 +3,8 @@
 namespace AppBundle\Form;
 
 use Symfony\Component\Form\AbstractType;
-use Symfony\Component\Form\Extension\Core\Type\SearchType;
 use Symfony\Component\Form\Extension\Core\Type\SubmitType;
+use Symfony\Component\Form\Extension\Core\Type\TextType;
 use Symfony\Component\Form\FormBuilderInterface;
 use Symfony\Component\OptionsResolver\OptionsResolver;
 
@@ -15,9 +15,9 @@ class RealisateurType extends AbstractType
      */
     public function buildForm(FormBuilderInterface $builder, array $options)
     {
-        $builder->add('prenom', SearchType::class)
-                ->add('nom', SearchType::class)
-                ->add('save', SubmitType::class, array ('label' =>  'Enregistrer'));
+        $builder->add('prenom', TextType::class)
+                ->add('nom', TextType::class);
+                //->add('save', SubmitType::class, array ('label' =>  'Enregistrer'));
     }
 
     /**

+ 0 - 3
src/AppBundle/Form/UserEditType.php

@@ -21,7 +21,4 @@ class UserEditType extends AbstractType
     {
         return UserType::class;
     }
-
-
-
 }

+ 69 - 0
src/AppBundle/Resources/views/videotheque/base.html.twig

@@ -3,3 +3,72 @@
 {% block title %}Videothèque commune{% endblock %}
 {% block titre %}Vidéothèque commune{% endblock %}
 
+{% block javascripts %}
+    <script type="text/javascript">
+        $(document).ready(function() {
+            // On récupère la balise <div> en question qui contient l'attribut « data-prototype » qui nous intéresse.
+            var $container = $('div#appbundle_film_realisateurs');
+
+            // On définit un compteur unique pour nommer les champs qu'on va ajouter dynamiquement
+            var index = $container.find(':input').length;
+
+            // On ajoute un nouveau champ à chaque clic sur le lien d'ajout.
+            $('#add_realisateur').click(function(e) {
+                addRealisateur($container);
+
+                e.preventDefault(); // évite qu'un # apparaisse dans l'URL
+                return false;
+            });
+
+            // On ajoute un premier champ automatiquement s'il n'en existe pas déjà un (cas d'une nouvelle annonce par exemple).
+            if (index == 0) {
+                addRealisateur($container);
+            } else {
+                // S'il existe déjà des catégories, on ajoute un lien de suppression pour chacune d'entre elles
+                $container.children('div').each(function() {
+                    addDeleteLink($(this));
+                });
+            }
+
+            // La fonction qui ajoute un formulaire RealisateurType
+            function addRealisateur($container) {
+                // Dans le contenu de l'attribut « data-prototype », on remplace :
+                // - le texte "__name__label__" qu'il contient par le label du champ
+                // - le texte "__name__" qu'il contient par le numéro du champ
+                var template = $container.attr('data-prototype')
+                    .replace(/__name__label__/g, 'réalisateur n°' + (index+1))
+                    .replace(/__name__/g,        index)
+                ;
+
+                // On crée un objet jquery qui contient ce template
+                var $prototype = $(template);
+
+                // On ajoute au prototype un lien pour pouvoir supprimer la catégorie
+                addDeleteLink($prototype);
+
+                // On ajoute le prototype modifié à la fin de la balise <div>
+                $container.append($prototype);
+
+                // Enfin, on incrémente le compteur pour que le prochain ajout se fasse avec un autre numéro
+                index++;
+            }
+
+            // La fonction qui ajoute un lien de suppression d'une catégorie
+            function addDeleteLink($prototype) {
+                // Création du lien
+                var $deleteLink = $('<a href="#" class="btn btn-danger">Supprimer</a>');
+
+                // Ajout du lien
+                $prototype.append($deleteLink);
+
+                // Ajout du listener sur le clic du lien pour effectivement supprimer la catégorie
+                $deleteLink.click(function(e) {
+                    $prototype.remove();
+
+                    e.preventDefault(); // évite qu'un # apparaisse dans l'URL
+                    return false;
+                });
+            }
+        });
+    </script>
+{% endblock %}

+ 11 - 8
src/AppBundle/Resources/views/videotheque/form.html.twig

@@ -1,18 +1,21 @@
     {{ form_start(form) }}
-    <div class="container-fluid">
-        <div class="row">
-        {{ form_row(form.titre) }}
+    <div class="container">
+        <div class="form-group row">
+            {{ form_row(form.titre) }}
         </div>
-        <div class="row">
-            <div class="col-md-6">
+        <div class="form-group row">
+            <div class="col">
                 {{ form_row(form.annee) }}
-                {{ form_row(form.realisateur) }}
+                {{ form_widget (form.realisateurs) }}
+                <a href="#" id="add_realisateur" class="btn btn-link">Ajouter un réalisateur</a>
             </div>
-            <div class="col-md-6">
+            <div class="col">
                 {{ form_row(form.note) }}
                 {{ form_row(form.commentaire) }}
             </div>
         </div>
     </div>
+
     {{ form_rest(form) }}
-    {{ form_end(form) }}
+    {{ form_end(form) }}
+

+ 7 - 3
src/AppBundle/Resources/views/videotheque/liste.html.twig

@@ -19,8 +19,12 @@
 		<tr>
 			<td>{{ film.titre }}</td>
 			<td>
-			{% if film.realisateur.nomComplet is defined %}
-				{{ film.realisateur.nomComplet }}
+			{% if film.realisateurs is defined %}
+				<ul>
+				{%  for realisateur in film.realisateurs %}
+					<li>{{ realisateur.nomComplet }}</li>
+				{%  endfor %}
+				</ul>
 			{% endif %}
 			</td>
 			<td>{{ film.annee | date('Y') }}</td>
@@ -36,4 +40,4 @@
 		{% endfor %}
 	</tbody>
 </table>
-{% endblock %}
+{% endblock %}