-
Notifications
You must be signed in to change notification settings - Fork 412
/
Copy pathModelCriteria.php
2275 lines (2066 loc) · 79.3 KB
/
ModelCriteria.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<?php
/**
* This file is part of the Propel package.
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*
* @license MIT License
*/
/**
* This class extends the Criteria by adding runtime introspection abilities
* in order to ease the building of queries.
*
* A ModelCriteria requires additional information to be initialized.
* Using a model name and tablemaps, a ModelCriteria can do more powerful things than a simple Criteria
*
* magic methods:
*
* @method ModelCriteria leftJoin($relation) Adds a LEFT JOIN clause to the query
* @method ModelCriteria rightJoin($relation) Adds a RIGHT JOIN clause to the query
* @method ModelCriteria innerJoin($relation) Adds a INNER JOIN clause to the query
*
* @author François Zaninotto
* @version $Revision$
* @package propel.runtime.query
*/
class ModelCriteria extends Criteria
{
const MODEL_CLAUSE = "MODEL CLAUSE";
const MODEL_CLAUSE_ARRAY = "MODEL CLAUSE ARRAY";
const MODEL_CLAUSE_LIKE = "MODEL CLAUSE LIKE";
const MODEL_CLAUSE_SEVERAL = "MODEL CLAUSE SEVERAL";
const MODEL_CLAUSE_RAW = "MODEL CLAUSE RAW";
const FORMAT_STATEMENT = 'PropelStatementFormatter';
const FORMAT_ARRAY = 'PropelArrayFormatter';
const FORMAT_OBJECT = 'PropelObjectFormatter';
const FORMAT_ON_DEMAND = 'PropelOnDemandFormatter';
protected $modelName;
protected $modelPeerName;
protected $modelAlias;
protected $useAliasInSQL = false;
protected $tableMap;
protected $primaryCriteria;
protected $formatter;
protected $defaultFormatterClass = ModelCriteria::FORMAT_OBJECT;
protected $with = array();
protected $isWithOneToMany = false;
protected $previousJoin = null; // this is introduced to prevent useQuery->join from going wrong
protected $isKeepQuery = true; // whether to clone the current object before termination methods
protected $select = null; // this is for the select method
/**
* Creates a new instance with the default capacity which corresponds to
* the specified database.
*
* @param string $dbName The dabase name
* @param string $modelName The phpName of a model, e.g. 'Book'
* @param string $modelAlias The alias for the model in this query, e.g. 'b'
*/
public function __construct($dbName = null, $modelName, $modelAlias = null)
{
$this->setDbName($dbName);
$this->originalDbName = $dbName;
$this->modelName = $modelName;
$this->modelPeerName = constant($this->modelName . '::PEER');
$this->modelAlias = $modelAlias;
$this->tableMap = Propel::getDatabaseMap($this->getDbName())->getTableByPhpName($this->modelName);
}
/**
* Returns the name of the class for this model criteria
*
* @return string
*/
public function getModelName()
{
return $this->modelName;
}
/**
* Sets the alias for the model in this query
*
* @param string $modelAlias The model alias
* @param boolean $useAliasInSQL Whether to use the alias in the SQL code (false by default)
*
* @return ModelCriteria The current object, for fluid interface
*/
public function setModelAlias($modelAlias, $useAliasInSQL = false)
{
if ($useAliasInSQL) {
$this->addAlias($modelAlias, $this->tableMap->getName());
$this->useAliasInSQL = true;
}
$this->modelAlias = $modelAlias;
return $this;
}
/**
* Returns the alias of the main class for this model criteria
*
* @return string The model alias
*/
public function getModelAlias()
{
return $this->modelAlias;
}
/**
* Return the string to use in a clause as a model prefix for the main model
*
* @return string The model alias if it exists, the model name if not
*/
public function getModelAliasOrName()
{
return $this->modelAlias ? $this->modelAlias : $this->modelName;
}
/**
* Returns the name of the Peer class for this model criteria
*
* @return string
*/
public function getModelPeerName()
{
return $this->modelPeerName;
}
/**
* Returns the TabkleMap object for this Criteria
*
* @return TableMap
*/
public function getTableMap()
{
return $this->tableMap;
}
/**
* Sets the formatter to use for the find() output
* Formatters must extend PropelFormatter
* Use the ModelCriteria constants for class names:
* <code>
* $c->setFormatter(ModelCriteria::FORMAT_ARRAY);
* </code>
*
* @param string|PropelFormatter $formatter a formatter class name, or a formatter instance
* @return ModelCriteria The current object, for fluid interface
*
* @throws PropelException
*/
public function setFormatter($formatter)
{
if (is_string($formatter)) {
$formatter = new $formatter();
}
if (!$formatter instanceof PropelFormatter) {
throw new PropelException('setFormatter() only accepts classes extending PropelFormatter');
}
$this->formatter = $formatter;
return $this;
}
/**
* Gets the formatter to use for the find() output
* Defaults to an instance of ModelCriteria::$defaultFormatterClass, i.e. PropelObjectsFormatter
*
* @return PropelFormatter
*/
public function getFormatter()
{
if (null === $this->formatter) {
$formatterClass = $this->defaultFormatterClass;
$this->formatter = new $formatterClass();
}
return $this->formatter;
}
/**
* Adds a condition on a column based on a pseudo SQL clause
* but keeps it for later use with combine()
* Until combine() is called, the condition is not added to the query
* Uses introspection to translate the column phpName into a fully qualified name
* <code>
* $c->condition('cond1', 'b.Title = ?', 'foo');
* </code>
*
* @see Criteria::add()
*
* @param string $conditionName A name to store the condition for a later combination with combine()
* @param string $clause The pseudo SQL clause, e.g. 'AuthorId = ?'
* @param mixed $value A value for the condition
* @param mixed $bindingType A value for the condition
*
* @return ModelCriteria The current object, for fluid interface
*/
public function condition($conditionName, $clause, $value = null, $bindingType = null)
{
$this->addCond($conditionName, $this->getCriterionForClause($clause, $value, $bindingType), null, $bindingType);
return $this;
}
/**
* Adds a condition on a column based on a column phpName and a value
* Uses introspection to translate the column phpName into a fully qualified name
* Warning: recognizes only the phpNames of the main Model (not joined tables)
* <code>
* $c->filterBy('Title', 'foo');
* </code>
*
* @see Criteria::add()
*
* @param string $column A string representing thecolumn phpName, e.g. 'AuthorId'
* @param mixed $value A value for the condition
* @param string $comparison What to use for the column comparison, defaults to Criteria::EQUAL
*
* @return ModelCriteria The current object, for fluid interface
*/
public function filterBy($column, $value, $comparison = Criteria::EQUAL)
{
return $this->add($this->getRealColumnName($column), $value, $comparison);
}
/**
* Adds a list of conditions on the columns of the current model
* Uses introspection to translate the column phpName into a fully qualified name
* Warning: recognizes only the phpNames of the main Model (not joined tables)
* <code>
* $c->filterByArray(array(
* 'Title' => 'War And Peace',
* 'Publisher' => $publisher
* ));
* </code>
*
* @see filterBy()
*
* @param mixed $conditions An array of conditions, using column phpNames as key
*
* @return ModelCriteria The current object, for fluid interface
*/
public function filterByArray($conditions)
{
foreach ($conditions as $column => $args) {
call_user_func_array(array($this, 'filterBy' . $column), is_array($args) ? $args : array($args));
}
return $this;
}
/**
* Adds a condition on a column based on a pseudo SQL clause
* Uses introspection to translate the column phpName into a fully qualified name
* <code>
* // simple clause
* $c->where('b.Title = ?', 'foo');
* // named conditions
* $c->condition('cond1', 'b.Title = ?', 'foo');
* $c->condition('cond2', 'b.ISBN = ?', 12345);
* $c->where(array('cond1', 'cond2'), Criteria::LOGICAL_OR);
* </code>
*
* @see Criteria::add()
*
* @param mixed $clause A string representing the pseudo SQL clause, e.g. 'Book.AuthorId = ?'
* Or an array of condition names
* @param mixed $value A value for the condition
* @param string $bindingType
*
* @return ModelCriteria The current object, for fluid interface
*/
public function where($clause, $value = null, $bindingType = null)
{
if (is_array($clause)) {
// where(array('cond1', 'cond2'), Criteria::LOGICAL_OR)
$criterion = $this->getCriterionForConditions($clause, $value);
} else {
// where('Book.AuthorId = ?', 12)
$criterion = $this->getCriterionForClause($clause, $value, $bindingType);
}
$this->addUsingOperator($criterion, null, null);
return $this;
}
/**
* Adds a condition on a column based on a pseudo SQL clause
* Uses introspection to translate the column phpName into a fully qualified name
* <code>
* // simple clause
* $c->orWhere('b.Title = ?', 'foo');
* // named conditions
* $c->condition('cond1', 'b.Title = ?', 'foo');
* $c->condition('cond2', 'b.ISBN = ?', 12345);
* $c->orWhere(array('cond1', 'cond2'), Criteria::LOGICAL_OR);
* </code>
*
* @see Criteria::addOr()
* @deprecated Use _or()->where() instead
*
* @param string $clause The pseudo SQL clause, e.g. 'AuthorId = ?'
* @param mixed $value A value for the condition
* @param string $bindingType
*
* @return ModelCriteria The current object, for fluid interface
*/
public function orWhere($clause, $value = null, $bindingType = null)
{
return $this
->_or()
->where($clause, $value, $bindingType);
}
/**
* Adds a having condition on a column based on a pseudo SQL clause
* Uses introspection to translate the column phpName into a fully qualified name
* <code>
* // simple clause
* $c->having('b.Title = ?', 'foo');
* // named conditions
* $c->condition('cond1', 'b.Title = ?', 'foo');
* $c->condition('cond2', 'b.ISBN = ?', 12345);
* $c->having(array('cond1', 'cond2'), Criteria::LOGICAL_OR);
* </code>
*
* @see Criteria::addHaving()
*
* @param mixed $clause A string representing the pseudo SQL clause, e.g. 'Book.AuthorId = ?'
* Or an array of condition names
* @param mixed $value A value for the condition
* @param string $bindingType
*
* @return ModelCriteria The current object, for fluid interface
*/
public function having($clause, $value = null, $bindingType = null)
{
if (is_array($clause)) {
// having(array('cond1', 'cond2'), Criteria::LOGICAL_OR)
$criterion = $this->getCriterionForConditions($clause, $value);
} else {
// having('Book.AuthorId = ?', 12)
$criterion = $this->getCriterionForClause($clause, $value, $bindingType);
}
$this->addHaving($criterion);
return $this;
}
/**
* Adds an ORDER BY clause to the query
* Usability layer on top of Criteria::addAscendingOrderByColumn() and Criteria::addDescendingOrderByColumn()
* Infers $column and $order from $columnName and some optional arguments
* Examples:
* $c->orderBy('Book.CreatedAt')
* => $c->addAscendingOrderByColumn(BookPeer::CREATED_AT)
* $c->orderBy('Book.CategoryId', 'desc')
* => $c->addDescendingOrderByColumn(BookPeer::CATEGORY_ID)
*
* @param string $columnName The column to order by
* @param string $order The sorting order. Criteria::ASC by default, also accepts Criteria::DESC
*
* @return ModelCriteria The current object, for fluid interface
*
* @throws PropelException
*/
public function orderBy($columnName, $order = Criteria::ASC)
{
list($column, $realColumnName) = $this->getColumnFromName($columnName, false);
$order = strtoupper($order);
switch ($order) {
case Criteria::ASC:
$this->addAscendingOrderByColumn($realColumnName);
break;
case Criteria::DESC:
$this->addDescendingOrderByColumn($realColumnName);
break;
default:
throw new PropelException('ModelCriteria::orderBy() only accepts Criteria::ASC or Criteria::DESC as argument');
}
return $this;
}
/**
* Adds a GROUB BY clause to the query
* Usability layer on top of Criteria::addGroupByColumn()
* Infers $column $columnName
* Examples:
* $c->groupBy('Book.AuthorId')
* => $c->addGroupByColumn(BookPeer::AUTHOR_ID)
*
* @param string $columnName The column to group by
*
* @return ModelCriteria The current object, for fluid interface
*/
public function groupBy($columnName)
{
list($column, $realColumnName) = $this->getColumnFromName($columnName, false);
$this->addGroupByColumn($realColumnName);
return $this;
}
/**
* Adds a GROUB BY clause for all columns of a model to the query
* Examples:
* $c->groupBy('Book');
* => $c->addGroupByColumn(BookPeer::ID);
* => $c->addGroupByColumn(BookPeer::TITLE);
* => $c->addGroupByColumn(BookPeer::AUTHOR_ID);
* => $c->addGroupByColumn(BookPeer::PUBLISHER_ID);
*
* @param string $class The class name or alias
*
* @return ModelCriteria The current object, for fluid interface
*
* @throws PropelException
*/
public function groupByClass($class)
{
if ($class == $this->getModelAliasOrName()) {
// column of the Criteria's model
$tableMap = $this->getTableMap();
} elseif (isset($this->joins[$class])) {
// column of a relations's model
$tableMap = $this->joins[$class]->getTableMap();
} else {
throw new PropelException('Unknown model or alias ' . $class);
}
foreach ($tableMap->getColumns() as $column) {
if (isset($this->aliases[$class])) {
$this->addGroupByColumn($class . '.' . $column->getName());
} else {
$this->addGroupByColumn($column->getFullyQualifiedName());
}
}
return $this;
}
/**
* Adds a DISTINCT clause to the query
* Alias for Criteria::setDistinct()
*
* @return ModelCriteria The current object, for fluid interface
*/
public function distinct()
{
$this->setDistinct();
return $this;
}
/**
* Adds a LIMIT clause (or its subselect equivalent) to the query
* Alias for Criteria:::setLimit()
*
* @param int $limit Maximum number of results to return by the query
*
* @return ModelCriteria The current object, for fluid interface
*/
public function limit($limit)
{
$this->setLimit($limit);
return $this;
}
/**
* Adds an OFFSET clause (or its subselect equivalent) to the query
* Alias for of Criteria::setOffset()
*
* @param int $offset Offset of the first result to return
*
* @return ModelCriteria The current object, for fluid interface
*/
public function offset($offset)
{
$this->setOffset($offset);
return $this;
}
/**
* Makes the ModelCriteria return a string, array, or PropelArrayCollection
* Examples:
* ArticleQuery::create()->select('Name')->find();
* => PropelArrayCollection Object ('Foo', 'Bar')
*
* ArticleQuery::create()->select('Name')->findOne();
* => string 'Foo'
*
* ArticleQuery::create()->select(array('Id', 'Name'))->find();
* => PropelArrayCollection Object (
* array('Id' => 1, 'Name' => 'Foo'),
* array('Id' => 2, 'Name' => 'Bar')
* )
*
* ArticleQuery::create()->select(array('Id', 'Name'))->findOne();
* => array('Id' => 1, 'Name' => 'Foo')
*
* @param mixed $columnArray A list of column names (e.g. array('Title', 'Category.Name', 'c.Content')) or a single column name (e.g. 'Name')
*
* @return ModelCriteria The current object, for fluid interface
*
* @throws PropelException
*/
public function select($columnArray)
{
if (empty($columnArray)) {
throw new PropelException('You must ask for at least one column');
}
if ($columnArray == '*') {
$columnArray = array();
foreach (call_user_func(array($this->modelPeerName, 'getFieldNames'), BasePeer::TYPE_PHPNAME) as $column) {
$columnArray []= $this->modelName . '.' . $column;
}
}
$this->select = $columnArray;
return $this;
}
/**
* Retrieves the columns defined by a previous call to select().
* @see select()
*
* @return array|string A list of column names (e.g. array('Title', 'Category.Name', 'c.Content')) or a single column name (e.g. 'Name')
*/
public function getSelect()
{
return $this->select;
}
protected function configureSelectColumns()
{
if (is_null($this->select)) {
// leave early
return;
}
// select() needs the PropelSimpleArrayFormatter if no formatter given
if (is_null($this->formatter)) {
$this->setFormatter('PropelSimpleArrayFormatter');
}
// clear only the selectColumns, clearSelectColumns() clears asColumns too
$this->selectColumns = array();
// We need to set the primary table name, since in the case that there are no WHERE columns
// it will be impossible for the BasePeer::createSelectSql() method to determine which
// tables go into the FROM clause.
if (!$this->selectQueries) {
$this->setPrimaryTableName(constant($this->modelPeerName . '::TABLE_NAME'));
}
// Add requested columns which are not withColumns
$columnNames = is_array($this->select) ? $this->select : array($this->select);
// temporary store columns Alias or withColumn
$asColumns = $this->getAsColumns();
$this->asColumns = array();
foreach ($columnNames as $columnName) {
// check if the column was added by a withColumn, if not add it
if (!array_key_exists($columnName, $asColumns)) {
$column = $this->getColumnFromName($columnName);
// always put quotes around the columnName to be safe, we strip them in the formatter
$this->addAsColumn('"' . $columnName . '"', $column[1]);
} else {
$this->addAsColumn($columnName, $asColumns[$columnName]);
}
}
}
/**
* This method returns the previousJoin for this ModelCriteria,
* by default this is null, but after useQuery this is set the to the join of that use
*
* @return Join the previousJoin for this ModelCriteria
*/
public function getPreviousJoin()
{
return $this->previousJoin;
}
/**
* This method sets the previousJoin for this ModelCriteria,
* by default this is null, but after useQuery this is set the to the join of that use
*
* @param Join $previousJoin The previousJoin for this ModelCriteria
*/
public function setPreviousJoin(Join $previousJoin)
{
$this->previousJoin = $previousJoin;
}
/**
* This method returns an already defined join clause from the query
*
* @param string $name The name of the join clause
*
* @return Join A join object
*/
public function getJoin($name)
{
return $this->joins[$name];
}
/**
* Adds a JOIN clause to the query
* Infers the ON clause from a relation name
* Uses the Propel table maps, based on the schema, to guess the related columns
* Beware that the default JOIN operator is INNER JOIN, while Criteria defaults to WHERE
* Examples:
* <code>
* $c->join('Book.Author');
* => $c->addJoin(BookPeer::AUTHOR_ID, AuthorPeer::ID, Criteria::INNER_JOIN);
* $c->join('Book.Author', Criteria::RIGHT_JOIN);
* => $c->addJoin(BookPeer::AUTHOR_ID, AuthorPeer::ID, Criteria::RIGHT_JOIN);
* $c->join('Book.Author a', Criteria::RIGHT_JOIN);
* => $c->addAlias('a', AuthorPeer::TABLE_NAME);
* => $c->addJoin(BookPeer::AUTHOR_ID, 'a.ID', Criteria::RIGHT_JOIN);
* </code>
*
* @param string $relation Relation to use for the join
* @param string $joinType Accepted values are null, 'left join', 'right join', 'inner join'
*
* @return ModelCriteria The current object, for fluid interface
*
* @throws PropelException
*/
public function join($relation, $joinType = Criteria::INNER_JOIN)
{
// relation looks like '$leftName.$relationName $relationAlias'
list($fullName, $relationAlias) = self::getClassAndAlias($relation);
if (strpos($fullName, '.') === false) {
// simple relation name, refers to the current table
$leftName = $this->getModelAliasOrName();
$relationName = $fullName;
$previousJoin = $this->getPreviousJoin();
$tableMap = $this->getTableMap();
} else {
list($leftName, $relationName) = explode('.', $fullName);
// find the TableMap for the left table using the $leftName
if ($leftName == $this->getModelAliasOrName()) {
$previousJoin = $this->getPreviousJoin();
$tableMap = $this->getTableMap();
} elseif (isset($this->joins[$leftName])) {
$previousJoin = $this->joins[$leftName];
$tableMap = $previousJoin->getTableMap();
} else {
throw new PropelException('Unknown table or alias ' . $leftName);
}
}
$leftTableAlias = isset($this->aliases[$leftName]) ? $leftName : null;
// find the RelationMap in the TableMap using the $relationName
if (!$tableMap->hasRelation($relationName)) {
throw new PropelException('Unknown relation ' . $relationName . ' on the ' . $leftName .' table');
}
$relationMap = $tableMap->getRelation($relationName);
// create a ModelJoin object for this join
$join = new ModelJoin();
$join->setJoinType($joinType);
if (null !== $previousJoin) {
$join->setPreviousJoin($previousJoin);
}
$join->setRelationMap($relationMap, $leftTableAlias, $relationAlias);
// add the ModelJoin to the current object
if ($relationAlias !== null) {
$this->addAlias($relationAlias, $relationMap->getRightTable()->getName());
$this->addJoinObject($join, $relationAlias);
} else {
$this->addJoinObject($join, $relationName);
}
return $this;
}
/**
* Add another condition to an already added join
* @example
* <code>
* $query->join('Book.Author');
* $query->addJoinCondition('Author', 'Book.Title LIKE ?', 'foo%');
* </code>
*
* @param string $name The relation name or alias on which the join was created
* @param string $clause SQL clause, may contain column and table phpNames
* @param mixed $value An optional value to bind to the clause
* @param string $operator The operator to use to add the condition. Defaults to 'AND'
* @param string $bindingType
*
* @return ModelCriteria The current object, for fluid interface
*
* @throws PropelException
*/
public function addJoinCondition($name, $clause, $value = null, $operator = null, $bindingType = null)
{
if (!isset($this->joins[$name])) {
throw new PropelException(sprintf('Adding a condition to a nonexistent join, %s. Try calling join() first.', $name));
}
$join = $this->joins[$name];
if (!$join->getJoinCondition() instanceof Criterion) {
$join->buildJoinCondition($this);
}
$criterion = $this->getCriterionForClause($clause, $value, $bindingType);
$method = $operator === Criteria::LOGICAL_OR ? 'addOr' : 'addAnd';
$join->getJoinCondition()->$method($criterion);
return $this;
}
/**
* Replace the condition of an already added join
* @example
* <code>
* $query->join('Book.Author');
* $query->condition('cond1', 'Book.AuthorId = Author.Id')
* $query->condition('cond2', 'Book.Title LIKE ?', 'War%')
* $query->combine(array('cond1', 'cond2'), 'and', 'cond3')
* $query->setJoinCondition('Author', 'cond3');
* </code>
*
* @param string $name The relation name or alias on which the join was created
* @param mixed $condition A Criterion object, or a condition name
*
* @return ModelCriteria The current object, for fluid interface
*
* @throws PropelException
*/
public function setJoinCondition($name, $condition)
{
if (!isset($this->joins[$name])) {
throw new PropelException(sprintf('Setting a condition to a nonexistent join, %s. Try calling join() first.', $name));
}
if ($condition instanceof Criterion) {
$this->getJoin($name)->setJoinCondition($condition);
} elseif (isset($this->namedCriterions[$condition])) {
$this->getJoin($name)->setJoinCondition($this->namedCriterions[$condition]);
} else {
throw new PropelException(sprintf('Cannot add condition %s on join %s. setJoinCondition() expects either a Criterion, or a condition added by way of condition()', $condition, $name));
}
return $this;
}
/**
* Add a join object to the Criteria
* @see Criteria::addJoinObject()
* @param Join $join A join object
* @param string $name
*
* @return ModelCriteria The current object, for fluid interface
*/
public function addJoinObject(Join $join, $name = null)
{
if (!in_array($join, $this->joins)) { // compare equality, NOT identity
if (null === $name) {
$this->joins[] = $join;
} else {
$this->joins[$name] = $join;
}
}
return $this;
}
/**
* Adds a JOIN clause to the query and hydrates the related objects
* Shortcut for $c->join()->with()
* <code>
* $c->joinWith('Book.Author');
* => $c->join('Book.Author');
* => $c->with('Author');
* $c->joinWith('Book.Author a', Criteria::RIGHT_JOIN);
* => $c->join('Book.Author a', Criteria::RIGHT_JOIN);
* => $c->with('a');
* </code>
*
* @param string $relation Relation to use for the join
* @param string $joinType Accepted values are null, 'left join', 'right join', 'inner join'
*
* @return ModelCriteria The current object, for fluid interface
*/
public function joinWith($relation, $joinType = Criteria::INNER_JOIN)
{
$this->join($relation, $joinType);
$this->with(self::getRelationName($relation));
return $this;
}
/**
* Adds a relation to hydrate together with the main object
* The relation must be initialized via a join() prior to calling with()
* Examples:
* <code>
* $c->join('Book.Author');
* $c->with('Author');
*
* $c->join('Book.Author a', Criteria::RIGHT_JOIN);
* $c->with('a');
* </code>
* WARNING: on a one-to-many relationship, the use of with() combined with limit()
* will return a wrong number of results for the related objects
*
* @param string $relation Relation to use for the join
*
* @return ModelCriteria The current object, for fluid interface
*
* @throws PropelException
*/
public function with($relation)
{
if (!isset($this->joins[$relation])) {
throw new PropelException('Unknown relation name or alias ' . $relation);
}
$join = $this->joins[$relation];
if ($join->getRelationMap()->getType() == RelationMap::MANY_TO_MANY) {
throw new PropelException('with() does not allow hydration for many-to-many relationships');
} elseif ($join->getRelationMap()->getType() == RelationMap::ONE_TO_MANY) {
// For performance reasons, the formatters will use a special routine in this case
$this->isWithOneToMany = true;
}
// check that the columns of the main class are already added (but only if this isn't a useQuery)
if (!$this->hasSelectClause() && !$this->getPrimaryCriteria()) {
$this->addSelfSelectColumns();
}
// add the columns of the related class
$this->addRelationSelectColumns($relation);
// list the join for later hydration in the formatter
$this->with[$relation] = new ModelWith($join);
return $this;
}
/**
* Gets the array of ModelWith specifying which objects must be hydrated
* together with the main object.
*
* @see with()
* @return array
*/
public function getWith()
{
return $this->with;
}
/**
* Sets the array of ModelWith specifying which objects must be hydrated
* together with the main object.
*
* @param array
*
* @return ModelCriteria The current object, for fluid interface
*/
public function setWith($with)
{
$this->with = $with;
return $this;
}
public function isWithOneToMany()
{
return $this->isWithOneToMany;
}
/**
* Adds a supplementary column to the select clause
* These columns can later be retrieved from the hydrated objects using getVirtualColumn()
*
* @param string $clause The SQL clause with object model column names
* e.g. 'UPPER(Author.FirstName)'
* @param string $name Optional alias for the added column
* If no alias is provided, the clause is used as a column alias
* This alias is used for retrieving the column via BaseObject::getVirtualColumn($alias)
*
* @return ModelCriteria The current object, for fluid interface
*/
public function withColumn($clause, $name = null)
{
if (null === $name) {
$name = str_replace(array('.', '(', ')'), '', $clause);
}
$clause = trim($clause);
$this->replaceNames($clause);
// check that the columns of the main class are already added (if this is the primary ModelCriteria)
if (!$this->hasSelectClause() && !$this->getPrimaryCriteria()) {
$this->addSelfSelectColumns();
}
$this->addAsColumn($name, $clause);
return $this;
}
/**
* Initializes a secondary ModelCriteria object, to be later merged with the current object
*
* @see ModelCriteria::endUse()
* @param string $relationName Relation name or alias
* @param string $secondaryCriteriaClass Classname for the ModelCriteria to be used
*
* @return ModelCriteria The secondary criteria object
*
* @throws PropelException
*/
public function useQuery($relationName, $secondaryCriteriaClass = null)
{
if (!isset($this->joins[$relationName])) {
throw new PropelException('Unknown class or alias ' . $relationName);
}
$className = $this->joins[$relationName]->getTableMap()->getPhpName();
if (null === $secondaryCriteriaClass) {
$secondaryCriteria = PropelQuery::from($className);
} else {
$secondaryCriteria = new $secondaryCriteriaClass();
}
if ($className != $relationName) {
$secondaryCriteria->setModelAlias($relationName, $relationName == $this->joins[$relationName]->getRelationMap()->getName() ? false : true);
}
$secondaryCriteria->setPrimaryCriteria($this, $this->joins[$relationName]);
return $secondaryCriteria;
}
/**
* Finalizes a secondary criteria and merges it with its primary Criteria
*
* @see Criteria::mergeWith()
*
* @return ModelCriteria The primary criteria object
*/
public function endUse()
{
if (isset($this->aliases[$this->modelAlias])) {
$this->removeAlias($this->modelAlias);
}
$primaryCriteria = $this->getPrimaryCriteria();
$primaryCriteria->mergeWith($this);
return $primaryCriteria;
}
/**
* Add the content of a Criteria to the current Criteria
* In case of conflict, the current Criteria keeps its properties
* @see Criteria::mergeWith()
*
* @param Criteria $criteria The criteria to read properties from
* @param string $operator The logical operator used to combine conditions
* Defaults to Criteria::LOGICAL_AND, also accapts Criteria::LOGICAL_OR
*
* @return ModelCriteria The primary criteria object
*/
public function mergeWith(Criteria $criteria, $operator = null)
{
parent::mergeWith($criteria, $operator);
// merge with
if ($criteria instanceof ModelCriteria) {
$this->with = array_merge($this->getWith(), $criteria->getWith());
}
return $this;
}
/**
* Clear the conditions to allow the reuse of the query object.
* The ModelCriteria's Model and alias 'all the properties set by construct) will remain.
*
* @return ModelCriteria The primary criteria object
*/
public function clear()
{
parent::clear();
$this->with = array();
$this->primaryCriteria = null;
$this->formatter=null;
return $this;
}
/**
* Sets the primary Criteria for this secondary Criteria
*
* @param ModelCriteria $criteria The primary criteria
* @param Join $previousJoin The previousJoin for this ModelCriteria
*/