. */ namespace Doctrine\ORM; use Doctrine\Common\DoctrineException; use Doctrine\ORM\Mapping\AssociationMapping; use \Closure; /** * A PersistentCollection represents a collection of elements that have persistent state. * * Collections of entities represent only the associations (links) to those entities. * That means, if the collection is part of a many-many mapping and you remove * entities from the collection, only the links in the relation table are removed (on flush). * Similarly, if you remove entities from a collection that is part of a one-many * mapping this will only result in the nulling out of the foreign keys on flush. * If you want entities in a one-many collection to be removed when * they're removed from the collection, use deleteOrphans => true on the one-many * mapping. * * @license http://www.opensource.org/licenses/lgpl-license.php LGPL * @since 2.0 * @version $Revision: 4930 $ * @author Konsta Vesterinen * @author Roman Borschel * @author Giorgio Sironi */ final class PersistentCollection extends \Doctrine\Common\Collections\Collection { /** * A snapshot of the collection at the moment it was fetched from the database. * This is used to create a diff of the collection at commit time. * * @var array */ private $_snapshot = array(); /** * The entity that owns this collection. * * @var object */ private $_owner; /** * The association mapping the collection belongs to. * This is currently either a OneToManyMapping or a ManyToManyMapping. * * @var Doctrine\ORM\Mapping\AssociationMapping */ private $_association; /** * The EntityManager that manages the persistence of the collection. * * @var Doctrine\ORM\EntityManager */ private $_em; /** * The name of the field on the target entities that points to the owner * of the collection. This is only set if the association is bi-directional. * * @var string */ private $_backRefFieldName; /** * The class descriptor of the owning entity. */ private $_typeClass; /** * Whether the collection is dirty and needs to be synchronized with the database * when the UnitOfWork that manages its persistent state commits. * * @var boolean */ private $_isDirty = false; /** Whether the collection has already been initialized. */ private $_initialized = true; /** * Creates a new persistent collection. */ public function __construct(EntityManager $em, $class, array $data = array()) { parent::__construct($data); $this->_em = $em; $this->_typeClass = $class; } /** * INTERNAL: * Sets the collection owner. Used (only?) during hydration. * * @param object $entity * @param AssociationMapping $assoc */ public function setOwner($entity, AssociationMapping $assoc) { $this->_owner = $entity; $this->_association = $assoc; // Check for bidirectionality if ($assoc->isInverseSide()) { // For sure bi-directional $this->_backRefFieldName = $assoc->mappedByFieldName; } else { $targetClass = $this->_em->getClassMetadata($assoc->targetEntityName); if (isset($targetClass->inverseMappings[$assoc->sourceFieldName])) { // Bi-directional $this->_backRefFieldName = $targetClass->inverseMappings[$assoc->sourceFieldName]->sourceFieldName; } } } /** * INTERNAL: * Gets the collection owner. * * @return object */ public function getOwner() { return $this->_owner; } /** * Gets the class descriptor for the owning entity class. * * @return Doctrine\ORM\Mapping\ClassMetadata */ public function getOwnerClass() { return $this->_typeClass; } /** * INTERNAL: * Adds an element to a collection during hydration. * * @param mixed $element The element to add. */ public function hydrateAdd($element) { parent::add($element); // If _backRefFieldName is set, then the association is bidirectional // and we need to set the back reference. if ($this->_backRefFieldName) { // Set back reference to owner if ($this->_association->isOneToMany()) { // OneToMany $this->_typeClass->reflFields[$this->_backRefFieldName] ->setValue($element, $this->_owner); } else { // ManyToMany $this->_typeClass->reflFields[$this->_backRefFieldName] ->getValue($element)->add($this->_owner); } } } /** * INTERNAL: * Sets a keyed element in the collection during hydration. * * @param mixed $key The key to set. * $param mixed $value The element to set. */ public function hydrateSet($key, $value) { parent::set($key, $value); } /** * Initializes the collection by loading its contents from the database * if the collection is not yet initialized. */ private function _initialize() { if ( ! $this->_initialized) { $this->_association->load($this->_owner, $this, $this->_em); $this->_initialized = true; } } /** * INTERNAL: * Tells this collection to take a snapshot of its current state. */ public function takeSnapshot() { $this->_snapshot = $this->_elements; } /** * INTERNAL: * Returns the last snapshot of the elements in the collection. * * @return array The last snapshot of the elements. */ public function getSnapshot() { return $this->_snapshot; } /** * INTERNAL: * getDeleteDiff * * @return array */ public function getDeleteDiff() { return array_udiff($this->_snapshot, $this->_elements, array($this, '_compareRecords')); } /** * INTERNAL getInsertDiff * * @return array */ public function getInsertDiff() { return array_udiff($this->_elements, $this->_snapshot, array($this, '_compareRecords')); } /** * Compares two records. To be used on _snapshot diffs using array_udiff. * * @return integer */ private function _compareRecords($a, $b) { return $a === $b ? 0 : 1; } /** * INTERNAL: Gets the association mapping of the collection. * * @return Doctrine\ORM\Mapping\AssociationMapping */ public function getMapping() { return $this->_association; } /** * Marks this collection as changed/dirty. */ private function _changed() { $this->_isDirty = true; } /** * Gets a boolean flag indicating whether this colleciton is dirty which means * its state needs to be synchronized with the database. * * @return boolean TRUE if the collection is dirty, FALSE otherwise. */ public function isDirty() { return $this->_isDirty; } /** * Sets a boolean flag, indicating whether this collection is dirty. * * @param boolean $dirty Whether the collection should be marked dirty or not. */ public function setDirty($dirty) { $this->_isDirty = $dirty; } /** * * @param $bool * @return unknown_type */ public function setInitialized($bool) { $this->_initialized = $bool; } /** * Checks whether this collection has been initialized. * * @return boolean */ public function isInitialized() { return $this->_initialized; } /** {@inheritdoc} */ public function first() { $this->_initialize(); return parent::first(); } /** {@inheritdoc} */ public function last() { $this->_initialize(); return parent::last(); } /** * {@inheritdoc} */ public function remove($key) { $this->_initialize(); $removed = parent::remove($key); if ($removed) { $this->_changed(); if ($this->_association->isOneToMany() && $this->_association->orphanRemoval) { $this->_em->getUnitOfWork()->scheduleOrphanRemoval($removed); } } return $removed; } /** * {@inheritdoc} */ public function removeElement($element) { $this->_initialize(); $result = parent::removeElement($element); $this->_changed(); return $result; } /** * {@inheritdoc} */ public function containsKey($key) { $this->_initialize(); return parent::containsKey($key); } /** * {@inheritdoc} */ public function contains($element) { $this->_initialize(); return parent::contains($element); } /** * {@inheritdoc} */ public function exists(Closure $p) { $this->_initialize(); return parent::exists($p); } /** * {@inheritdoc} */ public function search($element) { $this->_initialize(); return parent::search($element); } /** * {@inheritdoc} */ public function get($key) { $this->_initialize(); return parent::get($key); } /** * {@inheritdoc} */ public function getKeys() { $this->_initialize(); return parent::getKeys(); } /** * {@inheritdoc} */ public function getElements() { $this->_initialize(); return parent::getElements(); } /** * {@inheritdoc} */ public function count() { $this->_initialize(); return parent::count(); } /** * {@inheritdoc} */ public function set($key, $value) { parent::set($key, $value); $this->_changed(); } /** * {@inheritdoc} */ public function add($value) { parent::add($value); $this->_changed(); return true; } /** * {@inheritdoc} */ public function isEmpty() { $this->_initialize(); return parent::isEmpty(); } /** * {@inheritdoc} */ public function getIterator() { $this->_initialize(); return parent::getIterator(); } /** * {@inheritdoc} */ public function map(Closure $func) { $this->_initialize(); $result = parent::map($func); $this->_changed(); return $result; } /** * {@inheritdoc} */ public function filter(Closure $p) { $this->_initialize(); return parent::filter($p); } /** * {@inheritdoc} */ public function forAll(Closure $p) { $this->_initialize(); return parent::forAll($p); } /** * {@inheritdoc} */ public function partition(Closure $p) { $this->_initialize(); return parent::partition($p); } /** * {@inheritdoc} */ public function clear() { $this->_initialize(); $result = parent::clear(); if ($this->_association->isOwningSide) { $this->_changed(); $this->_em->getUnitOfWork()->scheduleCollectionDeletion($this); } return $result; } /** * Called by PHP when this collection is serialized. Ensures that only the * elements are properly serialized. * * @internal Tried to implement Serializable first but that did not work well * with circular references. This solution seems simpler and works well. */ public function __sleep() { return array('_elements'); } }