. */ namespace Doctrine\ORM\Tools\Pagination; use Doctrine\ORM\QueryBuilder; use Doctrine\ORM\Query; use Doctrine\ORM\Query\ResultSetMapping; use Doctrine\ORM\NoResultException; use Doctrine\ORM\Tools\Pagination\WhereInWalker; use Doctrine\ORM\Tools\Pagination\WhereInSqlWalker; use Doctrine\ORM\Tools\Pagination\CountWalker; use Doctrine\ORM\Tools\Pagination\CountSqlWalker; use Countable; use IteratorAggregate; use ArrayIterator; /** * Paginator * * The paginator can handle various complex scenarios with DQL. * * @author Pablo Díez * @author Benjamin Eberlei * @license New BSD */ class Paginator implements \Countable, \IteratorAggregate { /** * @var Query */ private $query; /** * @var bool */ private $fetchJoinCollection; /** * @var bool|null */ private $useSqlWalkers; /** * @var int */ private $count; /** * Constructor. * * @param Query|QueryBuilder $query A Doctrine ORM query or query builder. * @param Boolean $fetchJoinCollection Whether the query joins a collection (true by default). */ public function __construct($query, $fetchJoinCollection = true) { if ($query instanceof QueryBuilder) { $query = $query->getQuery(); } $this->query = $query; $this->fetchJoinCollection = (Boolean) $fetchJoinCollection; } /** * Returns the query * * @return Query */ public function getQuery() { return $this->query; } /** * Returns whether the query joins a collection. * * @return Boolean Whether the query joins a collection. */ public function getFetchJoinCollection() { return $this->fetchJoinCollection; } /** * Returns whether the paginator will use an SQL TreeWalker * * @return bool|null */ public function getUseSqlWalkers() { return $this->useSqlWalkers; } /** * Set whether the paginator will use an SQL TreeWalker * * @param bool|null $useSqlWalkers * @return $this */ public function setUseSqlWalkers($useSqlWalkers) { $this->useSqlWalkers = $useSqlWalkers; return $this; } /** * {@inheritdoc} */ public function count() { if ($this->count === null) { /* @var $countQuery Query */ $countQuery = $this->cloneQuery($this->query); if ( ! $countQuery->getHint(CountWalker::HINT_DISTINCT)) { $countQuery->setHint(CountWalker::HINT_DISTINCT, true); } if ($this->useSqlWalker($countQuery)) { $rsm = new ResultSetMapping(); $rsm->addScalarResult('_dctrn_count', 'count'); $countQuery->setHint(Query::HINT_CUSTOM_OUTPUT_WALKER, 'Doctrine\ORM\Tools\Pagination\CountSqlWalker'); $countQuery->setResultSetMapping($rsm); } else { $countQuery->setHint(Query::HINT_CUSTOM_TREE_WALKERS, array('Doctrine\ORM\Tools\Pagination\CountWalker')); } $countQuery->setFirstResult(null)->setMaxResults(null); try { $data = $countQuery->getScalarResult(); $data = array_map('current', $data); $this->count = array_sum($data); } catch(NoResultException $e) { $this->count = 0; } } return $this->count; } /** * {@inheritdoc} */ public function getIterator() { $offset = $this->query->getFirstResult(); $length = $this->query->getMaxResults(); if ($this->fetchJoinCollection) { $subQuery = $this->cloneQuery($this->query); if ($this->useSqlWalker($subQuery)) { $subQuery->setHint(Query::HINT_CUSTOM_OUTPUT_WALKER, 'Doctrine\ORM\Tools\Pagination\LimitSubquerySqlWalker'); } else { $subQuery->setHint(Query::HINT_CUSTOM_TREE_WALKERS, array('Doctrine\ORM\Tools\Pagination\LimitSubqueryWalker')); } $subQuery->setFirstResult($offset)->setMaxResults($length); $ids = array_map('current', $subQuery->getScalarResult()); $whereInQuery = $this->cloneQuery($this->query); // don't do this for an empty id array if (count($ids) > 0) { $namespace = WhereInWalker::PAGINATOR_ID_ALIAS; $whereInQuery->setHint(Query::HINT_CUSTOM_TREE_WALKERS, array('Doctrine\ORM\Tools\Pagination\WhereInWalker')); $whereInQuery->setHint(WhereInWalker::HINT_PAGINATOR_ID_COUNT, count($ids)); $whereInQuery->setFirstResult(null)->setMaxResults(null); foreach ($ids as $i => $id) { $i++; $whereInQuery->setParameter("{$namespace}_{$i}", $id); } } $result = $whereInQuery->getResult($this->query->getHydrationMode()); } else { $result = $this->cloneQuery($this->query) ->setMaxResults($length) ->setFirstResult($offset) ->getResult($this->query->getHydrationMode()) ; } return new \ArrayIterator($result); } /** * Clones a query. * * @param Query $query The query. * * @return Query The cloned query. */ private function cloneQuery(Query $query) { /* @var $cloneQuery Query */ $cloneQuery = clone $query; $cloneQuery->setParameters($query->getParameters()); foreach ($query->getHints() as $name => $value) { $cloneQuery->setHint($name, $value); } return $cloneQuery; } /** * Determine whether to use an SQL TreeWalker for the query * * @param Query $query The query. * * @return bool */ private function useSqlWalker(Query $query) { if ($this->useSqlWalkers === null) { return (Boolean) $query->getHint(Query::HINT_CUSTOM_OUTPUT_WALKER) == false; } return $this->useSqlWalkers; } }