Symfony Relationship Bundle
TangoMan Relationship Bundle provides magic methods for OneToOne, OneToMany, ManyToOne, ManyToMany, relationships., (*1)
Open a command console, enter your project directory and execute the following command to download the latest stable version of this bundle:, (*2)
$ composer require tangoman/relationship-bundle
This command requires you to have Composer installed globally, as explained in the installation chapter of the Composer documentation., (*3)
Then, enable the bundle by adding it to the list of registered bundles
in the app/AppKernel.php
file of your project:, (*4)
<?php // app/AppKernel.php // ... class AppKernel extends Kernel { // ... public function registerBundles() { $bundles = array( // ... new TangoMan\FrontBundle\TangoManRelationshipBundle(), ); // ... } }
Use TangoMan\RelationshipBundle\Traits\HasRelationships
trait inside your entities,
and define properties with appropriate doctrine annotations., (*5)
Open a command console, enter your project directory and execute the following command to update your database schema:, (*6)
$ php bin/console schema:update
HasRelationships
trait.cascade={"persist"}
annotation is MANDATORY (will allow bidirectional linking between entities).@method
annotation will allow for correct autocomplete in your IDE (optional).inversedBy
is always on the owning side of the relationship (but I believe you know that already.)cascade={"remove"}
will avoid orphan Item
on Owner
deletion (optional).<?php // src\AppBundle\Entity\Owner.php namespace AppBundle\Entity; use Doctrine\ORM\Mapping as ORM; use TangoMan\RelationshipBundle\Traits\HasRelationships; /** * Class Owner * @ORM\Table(name="owner") * @ORM\Entity(repositoryClass="AppBundle\Repository\OwnerRepository") * * @package AppBundle\Entity * * @method $this setItem(Item $item) * @method Item getItems() */ class Owner { use HasRelationships; // ... /** * @var Item * @ORM\OneToOne(targetEntity="AppBundle\Entity\Item", inversedBy="owner", cascade={"persist", "remove"}) */ private $item; // ...
<?php // src\AppBundle\Entity\Item.php namespace AppBundle\Entity; use Doctrine\ORM\Mapping as ORM; use TangoMan\RelationshipBundle\Traits\HasRelationships; /** * Class Item * @ORM\Table(name="item") * @ORM\Entity(repositoryClass="AppBundle\Repository\ItemRepository") * * @package AppBundle\Entity * * @method $this setOwner(Owner $owner) * @method Owner getOwner() */ class Item { use HasRelationships; // ... /** * @var Owner * @ORM\OneToOne(targetEntity="AppBundle\Entity\Owner", mappedBy="item", cascade={"persist"}) */ private $owner; // ...
@var ArrayCollection
@ORM\OrderBy({"name"="ASC"})
will allow to define custom orderBy when fetching items
(optional).<?php // src\AppBundle\Entity\Owner.php namespace AppBundle\Entity; use Doctrine\Common\Collections\ArrayCollection; use Doctrine\ORM\Mapping as ORM; use TangoMan\RelationshipBundle\Traits\HasRelationships; /** * Class Owner * @ORM\Table(name="owner") * @ORM\Entity(repositoryClass="AppBundle\Repository\OwnerRepository") * * @package AppBundle\Entity * * @method $this setItems(Item[] $items) * @method Item getItems() */ class Owner { use HasRelationships; // ... /** * @var ArrayCollection|Item[] * @ORM\ManyToMany(targetEntity="AppBundle\Entity\Item", inversedBy="owners", cascade={"persist", "remove"}) * @ORM\OrderBy({"name"="ASC"}) */ private $items; // ...
<?php // src\AppBundle\Entity\Item.php namespace AppBundle\Entity; use Doctrine\Common\Collections\ArrayCollection; use Doctrine\ORM\Mapping as ORM; use TangoMan\RelationshipBundle\Traits\HasRelationships; /** * Class Item * @ORM\Table(name="item") * @ORM\Entity(repositoryClass="AppBundle\Repository\ItemRepository") * * @package AppBundle\Entity * * @method $this setOwners(Owner[] $owners) * @method Owner getOwners() */ class Item { use HasRelationships; // ... /** * @var ArrayCollection|Owner[] * @ORM\ManyToMany(targetEntity="AppBundle\Entity\Owner", mappedBy="items", cascade={"persist"}) * @ORM\OrderBy({"name"="ASC"}) */ private $owners; // ...
Constructors MUST initialize properties with ArrayCollection
, (*7)
<?php // src\AppBundle\Entity\Owner.php namespace AppBundle\Entity; use Doctrine\Common\Collections\ArrayCollection; use Doctrine\ORM\Mapping as ORM; use TangoMan\RelationshipBundle\Traits\HasRelationships; // ... /** * Owner constructor. */ public function __construct() { // ... $this->Items = new ArrayCollection(); }
Your formTypes elements from the INVERSE side of relationships MUST own 'by_reference' => false,
attribute
to force use of appropriate setters and getters (i.e. add
and remove
methods)., (*8)
<?php // src\AppBundle\Form\ItemType.php // ... /** * @param FormBuilderInterface $builder * @param array $options */ public function buildForm(FormBuilderInterface $builder, array $options) { $builder // ... ->add( 'owner', EntityType::class, [ 'label' => 'Owner', 'placeholder' => 'Select owner', 'class' => 'AppBundle:Owner', 'by_reference' => false, 'multiple' => true, 'expanded' => false, 'required' => false, 'query_builder' => function (EntityRepository $em) { return $em->createQueryBuilder('o') ->orderBy('o.name'); }, ] ); }
If you find any bug please report here : Issues, (*9)
Copyright (c) 2018 Matthias Morin, (*10)
Distributed under the MIT license., (*11)
If you like TangoMan Relationship Bundle please star! And follow me on GitHub: TangoMan75 ... And check my other cool projects., (*12)
Matthias Morin | LinkedIn, (*13)