table = $table; $name = $table->getAttribute(Doctrine::ATTR_COLL_KEY); if($name !== null) { $this->generator = new Doctrine_IndexGenerator($name); } } /** * @return object Doctrine_Table */ public function getTable() { return $this->table; } /** * whether or not an offset batch has been expanded * @return boolean */ public function isExpanded($offset) { return isset($this->expanded[$offset]); } /** * whether or not this collection is expandable * @return boolean */ public function isExpandable() { return $this->expandable; } /** * @param Doctrine_IndexGenerator $generator * @return void */ public function setGenerator($generator) { $this->generator = $generator; } /** * @return Doctrine_IndexGenerator */ public function getGenerator() { return $this->generator; } /** * @return array */ public function getData() { return $this->data; } /** * @param array $data */ public function addData(array $data) { $this->data[] = $data; } /** * @return mixed */ public function getLast() { return end($this->data); } /** * @return void */ public function setReference(Doctrine_Record $record,Doctrine_Relation $relation) { $this->reference = $record; $this->relation = $relation; if($relation instanceof Doctrine_ForeignKey || $relation instanceof Doctrine_LocalKey) { $this->reference_field = $relation->getForeign(); $value = $record->get($relation->getLocal()); foreach($this->getNormalIterator() as $record) { if($value !== null) { $record->set($this->reference_field, $value); } else { $record->set($this->reference_field, $this->reference); } } } } /** * @return mixed */ public function getReference() { return $this->reference; } /** * @return boolean */ public function expand($key) { $where = array(); $params = array(); $limit = null; $offset = null; switch(get_class($this)): case "Doctrine_Collection_Offset": $limit = $this->getLimit(); $offset = (floor($key / $limit) * $limit); if( ! $this->expandable && isset($this->expanded[$offset])) return false; $fields = implode(", ",$this->table->getColumnNames()); break; default: if( ! $this->expandable) return false; if( ! isset($this->reference)) return false; $id = $this->reference->getID(); if(empty($id)) return false; switch(get_class($this)): case "Doctrine_Collection_Immediate": $fields = implode(", ",$this->table->getColumnNames()); break; default: $fields = implode(", ",$this->table->getPrimaryKeys()); endswitch; endswitch; if(isset($this->relation)) { if($this->relation instanceof Doctrine_ForeignKey) { $params = array($this->reference->getID()); $where[] = $this->reference_field." = ?"; if( ! isset($offset)) { $ids = $this->getPrimaryKeys(); if( ! empty($ids)) { $where[] = $this->table->getIdentifier()." NOT IN (".substr(str_repeat("?, ",count($ids)),0,-2).")"; $params = array_merge($params,$ids); } $this->expandable = false; } } elseif($this->relation instanceof Doctrine_Association) { $asf = $fk->getAssociationFactory(); $query = "SELECT ".$foreign." FROM ".$asf->getTableName()." WHERE ".$local."=".$this->getID(); $table = $fk->getTable(); $graph = new Doctrine_DQL_Parser($table->getSession()); $q = "FROM ".$table->getComponentName()." WHERE ".$table->getComponentName().".".$table->getIdentifier()." IN ($query)"; } } $query = "SELECT ".$fields." FROM ".$this->table->getTableName(); // apply column aggregation inheritance foreach($this->table->getInheritanceMap() as $k => $v) { $where[] = $k." = ?"; $params[] = $v; } if( ! empty($where)) { $query .= " WHERE ".implode(" AND ",$where); } $params = array_merge($params, array_values($this->table->getInheritanceMap())); $coll = $this->table->execute($query, $params, $limit, $offset); if( ! isset($offset)) { foreach($coll as $record) { if(isset($this->reference_field)) $record->rawSet($this->reference_field,$this->reference); $this->reference->addReference($record); } } else { $i = $offset; foreach($coll as $record) { if(isset($this->reference)) { $this->reference->addReference($record,$i); } else $this->data[$i] = $record; $i++; } $this->expanded[$offset] = true; // check if the fetched collection's record count is smaller // than the query limit, if so this collection has been expanded to its max size if(count($coll) < $limit) { $this->expandable = false; } } return $coll; } /** * @return boolean */ public function remove($key) { if( ! isset($this->data[$key])) throw new InvalidKeyException(); $removed = $this->data[$key]; unset($this->data[$key]); return $removed; } /** * @param mixed $key * @return boolean */ public function contains($key) { return isset($this->data[$key]); } /** * @param mixed $key * @return object Doctrine_Record return a specified record */ public function get($key) { if( ! isset($this->data[$key])) { $this->expand($key); if( ! isset($this->data[$key])) $this->data[$key] = $this->table->create(); if(isset($this->reference_field)) { $value = $this->reference->get($this->relation->getLocal()); if($value !== null) { $this->data[$key]->set($this->reference_field, $value); } else { $this->data[$key]->set($this->reference_field, $this->reference); } } } return $this->data[$key]; } /** * @return array an array containing all primary keys */ public function getPrimaryKeys() { $list = array(); $name = $this->table->getIdentifier(); foreach($this->data as $record): if(is_array($record) && isset($record[$name])) { $list[] = $record[$name]; } else { $list[] = $record->getID(); } endforeach; return $list; } /** * returns all keys * @return array */ public function getKeys() { return array_keys($this->data); } /** * count * this class implements interface countable * @return integer number of records in this collection */ public function count() { return count($this->data); } /** * set * @param integer $key * @param Doctrine_Record $record * @return void */ public function set($key,Doctrine_Record $record) { if(isset($this->reference_field)) $record->set($this->reference_field,$this->reference); $this->data[$key] = $record; } /** * adds a record to collection * @param Doctrine_Record $record record to be added * @param string $key optional key for the record * @return boolean */ public function add(Doctrine_Record $record,$key = null) { if(isset($this->reference_field)) $record->rawSet($this->reference_field,$this->reference); if(isset($key)) { if(isset($this->data[$key])) return false; $this->data[$key] = $record; return true; } if(isset($this->generator)) { $key = $this->generator->getIndex($record); $this->data[$key] = $record; } else $this->data[] = $record; return true; } /** * @param Doctrine_DQL_Parser $graph * @param integer $key */ public function populate(Doctrine_DQL_Parser $graph) { $name = $this->table->getComponentName(); if($this instanceof Doctrine_Collection_Immediate || $this instanceof Doctrine_Collection_Offset) { $data = $graph->getData($name); if(is_array($data)) { foreach($data as $k=>$v): $this->table->setData($v); $this->add($this->table->getRecord()); endforeach; } } elseif($this instanceof Doctrine_Collection_Batch) { $this->data = $graph->getData($name); if(isset($this->generator)) { foreach($this->data as $k => $v) { $record = $this->get($k); $i = $this->generator->getIndex($record); $this->data[$i] = $record; unset($this->data[$k]); } } } } /** * @return Doctrine_Iterator_Normal */ public function getNormalIterator() { return new Doctrine_Iterator_Normal($this); } /** * save * saves all records */ public function save() { $this->table->getSession()->saveCollection($this); } /** * single shot delete * deletes all records from this collection * uses only one database query to perform this operation * @return boolean */ public function delete() { $ids = $this->table->getSession()->deleteCollection($this); $this->data = array(); } /** * getIterator * @return object ArrayIterator */ public function getIterator() { $data = $this->data; return new ArrayIterator($data); } /** * returns a string representation of this object */ public function __toString() { return Doctrine_Lib::getCollectionAsString($this); } } ?>