J'ai une table pour mes catégories, chaque catégorie a et id, nom et parent_id.
Select IF(a.parent_id IS NULL, a.name, CONCAT((SELECT b.name FROM category b WHERE b.id = a.parent_id), " / ", a.name) ) as n, a.id, a.parent_id
FROM category a
ORDER BY n
Je veux le convertir en mon constructeur de requêtes Doctrine2
$em = $this->getDoctrine()->getEntityManager();
$qb = $em->createQueryBuilder();
$q = $qb
->select("c.id")
->addSelect(
"IF(c.parent_id IS NULL, c.name, CONCAT((" .
$em->createQueryBuilder()
->select("t.name")
->from("MyBundle:Category", "t")
->getQuery()->getDQL() .
"), \" / \", c.name) )"
)
->from("MyBundle:Category", "c");
echo $q->getQuery()->getSQL();
exit;
Quelque chose comme ça, mais je ne peux pas utiliser IF
, et CONCAT
.
Ok j'ai trouvé la solution.
Vous pouvez utiliser CASE
au lieu de IF
. Regardez ceci, mais quand j'utilise CASE
je ne peux pas CONCAT
mes champs:
$em = $this->getDoctrine()->getEntityManager();
$qb = $em->createQueryBuilder();
$q = $qb
->select("c.id")
->addSelect("CASE WHEN (c.parent IS NULL) THEN c.name ELSE 'something' END")
->from("MyBundle:Category", "c")
->leftJoin("c.parent", "t");
echo $q->getQuery()->getSQL();
Une autre solution est de créer votre propre fonction DQL, telle que IF
et de l’utiliser comme ceci:
$em = $this->getDoctrine()->getEntityManager();
$qb = $em->createQueryBuilder();
$q = $qb
->select("c.id")
->addSelect("IF(c.parent IS NULL, c.name, CONCAT(CONCAT(t.name, ' / '), c.name))")
->from("MyBundle:Category", "c")
->leftJoin("c.parent", "t");
echo $q->getQuery()->getSQL();
Pour créer ce SI, vous pouvez aller sur ce lien et apprendre: http://docs.doctrine-project.org/projects/doctrine-orm/en/latest/reference/dql-doctrine-query-language.html#adding -votre-fonctions-propres-au-langage-dql
Je vais poster ici ma classe pour cet IF et le fichier config.yml pour aider d’autres personnes . Voici la classe IfFunction (que j’ai eu à partir de https://github.com/beberlei/DoctrineExtensions/blob/master/ src/Query/Mysql/IfElse.php ):
<?php
namespace MyName\MiscBundle\Doctrine\ORM\Query\AST\Functions;
use Doctrine\ORM\Query\AST\Functions\FunctionNode;
use Doctrine\ORM\Query\Lexer;
/**
* Usage: IF(expr1, expr2, expr3)
*
* If expr1 is TRUE (expr1 <> 0 and expr1 <> NULL) then IF() returns expr2;
* otherwise it returns expr3. IF() returns a numeric or string value,
* depending on the context in which it is used.
*
* @author Andrew Mackrodt <[email protected]>
* @version 2011.06.19
*/
class IfFunction extends FunctionNode
{
private $expr = array();
public function parse(\Doctrine\ORM\Query\Parser $parser)
{
$parser->match(Lexer::T_IDENTIFIER);
$parser->match(Lexer::T_OPEN_PARENTHESIS);
$this->expr[] = $parser->ConditionalExpression();
for ($i = 0; $i < 2; $i++)
{
$parser->match(Lexer::T_COMMA);
$this->expr[] = $parser->ArithmeticExpression();
}
$parser->match(Lexer::T_CLOSE_PARENTHESIS);
}
public function getSql(\Doctrine\ORM\Query\SqlWalker $sqlWalker)
{
return sprintf('IF(%s, %s, %s)',
$sqlWalker->walkConditionalExpression($this->expr[0]),
$sqlWalker->walkArithmeticPrimary($this->expr[1]),
$sqlWalker->walkArithmeticPrimary($this->expr[2]));
}
}
Après cela, vous devez mettre à jour votre fichier config.yml comme ceci (vient d’ajouter les 3 dernières lignes):
doctrine:
dbal:
driver: "%database_driver%"
Host: "%database_Host%"
port: "%database_port%"
dbname: "%database_name%"
user: "%database_user%"
password: "%database_password%"
charset: UTF8
orm:
auto_generate_proxy_classes: "%kernel.debug%"
auto_mapping: true
dql: #ADDED THIS LINE
string_functions: #ADDED THIS LINE
IF: MyName\MiscBundle\Doctrine\ORM\Query\AST\Functions\IfFunction #ADDED THIS LINE
Merci
Vous pouvez également combiner CASE
et CONCAT
:
$q = $this->createQueryBuilder('a')
->select('a.id')
->addSelect('CASE WHEN(a.parent IS NULL) THEN \'\' else CONCAT(:variable, a.name, \'string\') END as name')
->setParameter('variable', $variable)
... ;
En supposant que vous ayez une relation nommée "parent" dans la catégorie, cela fonctionne-t-il mieux? :
$em = $this->getDoctrine()->getEntityManager();
$qb = $em->createQueryBuilder();
$q = $qb
->select("c.id")
->addSelect("IF (c.parent_id IS NULL, c.name, CONCAT(t.name, ' / ', c.name))"
->from("MyBundle:Category", "c")
->leftJoin("c.parent t"));
echo $q->getQuery()->getSQL();