2012-01-22 16:35:06 +04:00
< ? php
/**
* Doctrine ORM
*
* LICENSE
*
* This source file is subject to the new BSD license that is bundled
* with this package in the file LICENSE . This license can also be viewed
* at http :// hobodave . com / license . txt
*
* @ category DoctrineExtensions
* @ package DoctrineExtensions\Paginate
* @ author David Abdemoulaie < dave @ hobodave . com >
* @ copyright Copyright ( c ) 2010 David Abdemoulaie ( http :// hobodave . com / )
* @ license http :// hobodave . com / license . txt New BSD License
*/
namespace Doctrine\ORM\Tools\Pagination ;
2012-10-12 15:53:20 +04:00
use Doctrine\DBAL\Types\Type ;
use Doctrine\ORM\Query\TreeWalkerAdapter ;
use Doctrine\ORM\Query\AST\SelectStatement ;
use Doctrine\ORM\Query\AST\SelectExpression ;
use Doctrine\ORM\Query\AST\PathExpression ;
use Doctrine\ORM\Query\AST\AggregateExpression ;
2012-01-22 16:35:06 +04:00
/**
* Replaces the selectClause of the AST with a SELECT DISTINCT root . id equivalent
*
* @ category DoctrineExtensions
* @ package DoctrineExtensions\Paginate
* @ author David Abdemoulaie < dave @ hobodave . com >
* @ copyright Copyright ( c ) 2010 David Abdemoulaie ( http :// hobodave . com / )
* @ license http :// hobodave . com / license . txt New BSD License
*/
class LimitSubqueryWalker extends TreeWalkerAdapter
{
/**
* ID type hint
*/
const IDENTIFIER_TYPE = 'doctrine_paginator.id.type' ;
/**
* @ var int Counter for generating unique order column aliases
*/
private $_aliasCounter = 0 ;
/**
* Walks down a SelectStatement AST node , modifying it to retrieve DISTINCT ids
* of the root Entity
*
* @ param SelectStatement $AST
* @ return void
*/
public function walkSelectStatement ( SelectStatement $AST )
{
$parent = null ;
$parentName = null ;
$selectExpressions = array ();
2012-03-24 14:16:32 +04:00
foreach ( $this -> _getQueryComponents () as $dqlAlias => $qComp ) {
2012-01-22 16:35:06 +04:00
// preserve mixed data in query for ordering
if ( isset ( $qComp [ 'resultVariable' ])) {
$selectExpressions [] = new SelectExpression ( $qComp [ 'resultVariable' ], $dqlAlias );
continue ;
}
if ( $qComp [ 'parent' ] === null && $qComp [ 'nestingLevel' ] == 0 ) {
$parent = $qComp ;
$parentName = $dqlAlias ;
continue ;
}
}
$identifier = $parent [ 'metadata' ] -> getSingleIdentifierFieldName ();
2012-05-27 20:33:35 +04:00
if ( isset ( $parent [ 'metadata' ] -> associationMappings [ $identifier ])) {
throw new \RuntimeException ( " Paginating an entity with foreign key as identifier only works when using the Output Walkers. Call Paginator#setUseOutputWalkers(true) before iterating the paginator. " );
}
2012-01-22 16:35:06 +04:00
$this -> _getQuery () -> setHint (
self :: IDENTIFIER_TYPE ,
Type :: getType ( $parent [ 'metadata' ] -> getTypeOfField ( $identifier ))
);
$pathExpression = new PathExpression (
PathExpression :: TYPE_STATE_FIELD | PathExpression :: TYPE_SINGLE_VALUED_ASSOCIATION ,
$parentName ,
$identifier
);
$pathExpression -> type = PathExpression :: TYPE_STATE_FIELD ;
array_unshift ( $selectExpressions , new SelectExpression ( $pathExpression , '_dctrn_id' ));
$AST -> selectClause -> selectExpressions = $selectExpressions ;
if ( isset ( $AST -> orderByClause )) {
foreach ( $AST -> orderByClause -> orderByItems as $item ) {
if ( $item -> expression instanceof PathExpression ) {
$pathExpression = new PathExpression (
PathExpression :: TYPE_STATE_FIELD | PathExpression :: TYPE_SINGLE_VALUED_ASSOCIATION ,
$item -> expression -> identificationVariable ,
$item -> expression -> field
);
$pathExpression -> type = PathExpression :: TYPE_STATE_FIELD ;
$AST -> selectClause -> selectExpressions [] = new SelectExpression (
$pathExpression ,
2012-06-05 17:40:44 +04:00
'_dctrn_ord' . $this -> _aliasCounter ++
2012-01-22 16:35:06 +04:00
);
}
}
}
$AST -> selectClause -> isDistinct = true ;
}
}