Commit 0fa19291 by Qiang Xue

refactored filterWhere().

parent 85bb94a6
...@@ -470,25 +470,6 @@ class Query extends Component implements QueryInterface ...@@ -470,25 +470,6 @@ class Query extends Component implements QueryInterface
} }
/** /**
* Sets the WHERE part of the query ignoring empty parameters.
*
* @param string|array $condition the conditions that should be put in the WHERE part. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see andFilter()
* @see orFilter()
*/
public function filterWhere($condition, $params = [])
{
$condition = $this->filterCondition($condition);
if ($condition !== []) {
$this->where($condition, $params);
}
return $this;
}
/**
* Adds an additional WHERE condition to the existing one. * Adds an additional WHERE condition to the existing one.
* The new condition and the existing one will be joined using the 'AND' operator. * The new condition and the existing one will be joined using the 'AND' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]] * @param string|array $condition the new WHERE condition. Please refer to [[where()]]
...@@ -510,26 +491,6 @@ class Query extends Component implements QueryInterface ...@@ -510,26 +491,6 @@ class Query extends Component implements QueryInterface
} }
/** /**
* Adds an additional WHERE condition to the existing one ignoring empty parameters.
* The new condition and the existing one will be joined using the 'AND' operator.
*
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see filter()
* @see orFilter()
*/
public function andFilterWhere($condition, $params = [])
{
$condition = $this->filterCondition($condition);
if ($condition !== []) {
$this->andWhere($condition, $params);
}
return $this;
}
/**
* Adds an additional WHERE condition to the existing one. * Adds an additional WHERE condition to the existing one.
* The new condition and the existing one will be joined using the 'OR' operator. * The new condition and the existing one will be joined using the 'OR' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]] * @param string|array $condition the new WHERE condition. Please refer to [[where()]]
...@@ -551,26 +512,6 @@ class Query extends Component implements QueryInterface ...@@ -551,26 +512,6 @@ class Query extends Component implements QueryInterface
} }
/** /**
* Adds an additional WHERE condition to the existing one ignoring empty parameters.
* The new condition and the existing one will be joined using the 'OR' operator.
*
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see filter()
* @see andFilter()
*/
public function orFilterWhere($condition, $params = [])
{
$condition = $this->filterCondition($condition);
if ($condition !== []) {
$this->orWhere($condition, $params);
}
return $this;
}
/**
* Sets the GROUP BY part of the query. * Sets the GROUP BY part of the query.
* @param string|array $columns the columns to be grouped by. * @param string|array $columns the columns to be grouped by.
* Columns can be specified in either a string (e.g. "id, name") or an array (e.g. ['id', 'name']). * Columns can be specified in either a string (e.g. "id, name") or an array (e.g. ['id', 'name']).
......
...@@ -573,65 +573,6 @@ class Query extends Component implements QueryInterface ...@@ -573,65 +573,6 @@ class Query extends Component implements QueryInterface
} }
/** /**
* Sets the WHERE part of the query ignoring empty parameters.
*
* @param string|array $condition the conditions that should be put in the WHERE part. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see andFilter()
* @see orFilter()
*/
public function filterWhere($condition, $params = [])
{
$condition = $this->filterCondition($condition);
if ($condition !== []) {
$this->where($condition, $params);
}
return $this;
}
/**
* Adds an additional WHERE condition to the existing one ignoring empty parameters.
* The new condition and the existing one will be joined using the 'AND' operator.
*
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see filter()
* @see orFilter()
*/
public function andFilterWhere($condition, $params = [])
{
$condition = $this->filterCondition($condition);
if ($condition !== []) {
$this->andWhere($condition, $params);
}
return $this;
}
/**
* Adds an additional WHERE condition to the existing one ignoring empty parameters.
* The new condition and the existing one will be joined using the 'OR' operator.
*
* @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter.
* @param array $params the parameters (name => value) to be bound to the query.
* @return static the query object itself
* @see filter()
* @see andFilter()
*/
public function orFilterWhere($condition, $params = [])
{
$condition = $this->filterCondition($condition);
if ($condition !== []) {
$this->orWhere($condition, $params);
}
return $this;
}
/**
* Appends a JOIN part to the query. * Appends a JOIN part to the query.
* The first parameter specifies what type of join it is. * The first parameter specifies what type of join it is.
* @param string $type the type of join, such as INNER JOIN, LEFT JOIN. * @param string $type the type of join, such as INNER JOIN, LEFT JOIN.
......
...@@ -136,7 +136,7 @@ interface QueryInterface ...@@ -136,7 +136,7 @@ interface QueryInterface
* - `or not like`: similar to the `not like` operator except that `OR` is used to concatenate * - `or not like`: similar to the `not like` operator except that `OR` is used to concatenate
* the `NOT LIKE` predicates. * the `NOT LIKE` predicates.
* *
* @param array $condition the conditions that should be put in the WHERE part. * @param string|array $condition the conditions that should be put in the WHERE part.
* @return static the query object itself * @return static the query object itself
* @see andWhere() * @see andWhere()
* @see orWhere() * @see orWhere()
...@@ -144,59 +144,59 @@ interface QueryInterface ...@@ -144,59 +144,59 @@ interface QueryInterface
public function where($condition); public function where($condition);
/** /**
* Sets the WHERE part of the query ignoring empty parameters. * Adds an additional WHERE condition to the existing one.
* * The new condition and the existing one will be joined using the 'AND' operator.
* @param array $condition the conditions that should be put in the WHERE part. Please refer to [[where()]] * @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter. * on how to specify this parameter.
* @return static the query object itself * @return static the query object itself
* @see andFilterWhere() * @see where()
* @see orFilterWhere() * @see orWhere()
*/ */
public function filterWhere($condition); public function andWhere($condition);
/** /**
* Adds an additional WHERE condition to the existing one. * Adds an additional WHERE condition to the existing one.
* The new condition and the existing one will be joined using the 'AND' operator. * The new condition and the existing one will be joined using the 'OR' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]] * @param string|array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter. * on how to specify this parameter.
* @return static the query object itself * @return static the query object itself
* @see where() * @see where()
* @see orWhere() * @see andWhere()
*/ */
public function andWhere($condition); public function orWhere($condition);
/** /**
* Adds an additional WHERE condition to the existing one ignoring empty parameters. * Sets the WHERE part of the query ignoring empty parameters.
* The new condition and the existing one will be joined using the 'AND' operator. *
* @param string|array $condition the new WHERE condition. Please refer to [[where()]] * @param array $condition the conditions that should be put in the WHERE part. Please refer to [[where()]]
* on how to specify this parameter. * on how to specify this parameter.
* @return static the query object itself * @return static the query object itself
* @see filterWhere() * @see andFilterWhere()
* @see orFilterWhere() * @see orFilterWhere()
*/ */
public function andFilterWhere($condition); public function filterWhere(array $condition);
/** /**
* Adds an additional WHERE condition to the existing one. * Adds an additional WHERE condition to the existing one ignoring empty parameters.
* The new condition and the existing one will be joined using the 'OR' operator. * The new condition and the existing one will be joined using the 'AND' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]] * @param array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter. * on how to specify this parameter.
* @return static the query object itself * @return static the query object itself
* @see where() * @see filterWhere()
* @see andWhere() * @see orFilterWhere()
*/ */
public function orWhere($condition); public function andFilterWhere(array $condition);
/** /**
* Adds an additional WHERE condition to the existing one ignoring empty parameters. * Adds an additional WHERE condition to the existing one ignoring empty parameters.
* The new condition and the existing one will be joined using the 'OR' operator. * The new condition and the existing one will be joined using the 'OR' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]] * @param array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter. * on how to specify this parameter.
* @return static the query object itself * @return static the query object itself
* @see filterWhere() * @see filterWhere()
* @see andFilterWhere() * @see andFilterWhere()
*/ */
public function orFilterWhere($condition); public function orFilterWhere(array $condition);
/** /**
* Sets the ORDER BY part of the query. * Sets the ORDER BY part of the query.
......
...@@ -78,7 +78,7 @@ trait QueryTrait ...@@ -78,7 +78,7 @@ trait QueryTrait
* *
* See [[QueryInterface::where()]] for detailed documentation. * See [[QueryInterface::where()]] for detailed documentation.
* *
* @param array $condition the conditions that should be put in the WHERE part. * @param string|array $condition the conditions that should be put in the WHERE part.
* @return static the query object itself * @return static the query object itself
* @see andWhere() * @see andWhere()
* @see orWhere() * @see orWhere()
...@@ -128,10 +128,24 @@ trait QueryTrait ...@@ -128,10 +128,24 @@ trait QueryTrait
} }
/** /**
* Sets the WHERE part of the query but ignores [[isParameterNotEmpty|empty parameters]]. * Sets the WHERE part of the query but ignores [[isEmpty()|empty operands]].
*
* This method is similar to [[where()]]. The main difference is that this method will
* remove [[isEmpty()|empty query operands]]. As a result, this method is best suited
* for building query conditions based on filter values entered by users.
*
* The following code shows the difference between this method and [[where()]]:
* *
* This function can be used to pass fields of a search form directly as search condition * ```php
* by ignoring fields that have not been filled. * // WHERE `age`=:age
* $query->filterWhere(['name' => null, 'age' => 20]);
* // WHERE `age`=:age
* $query->where(['age' => 20]);
* // WHERE `name` IS NULL AND `age`=:age
* $query->where(['name' => null, 'age' => 20]);
* ```
*
* Note that unlike [[where()]], you cannot pass binding parameters to this method.
* *
* @param array $condition the conditions that should be put in the WHERE part. * @param array $condition the conditions that should be put in the WHERE part.
* See [[where()]] on how to specify this parameter. * See [[where()]] on how to specify this parameter.
...@@ -140,7 +154,7 @@ trait QueryTrait ...@@ -140,7 +154,7 @@ trait QueryTrait
* @see andFilterWhere() * @see andFilterWhere()
* @see orFilterWhere() * @see orFilterWhere()
*/ */
public function filterWhere($condition) public function filterWhere(array $condition)
{ {
$condition = $this->filterCondition($condition); $condition = $this->filterCondition($condition);
if ($condition !== []) { if ($condition !== []) {
...@@ -150,15 +164,20 @@ trait QueryTrait ...@@ -150,15 +164,20 @@ trait QueryTrait
} }
/** /**
* Adds an additional WHERE condition to the existing one but ignores [[isParameterNotEmpty|empty parameters]]. * Adds an additional WHERE condition to the existing one but ignores [[isEmpty()|empty operands]].
* The new condition and the existing one will be joined using the 'AND' operator. * The new condition and the existing one will be joined using the 'AND' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]] *
* This method is similar to [[andWhere()]]. The main difference is that this method will
* remove [[isEmpty()|empty query operands]]. As a result, this method is best suited
* for building query conditions based on filter values entered by users.
*
* @param array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter. * on how to specify this parameter.
* @return static the query object itself * @return static the query object itself
* @see filterWhere() * @see filterWhere()
* @see orFilterWhere() * @see orFilterWhere()
*/ */
public function andFilterWhere($condition) public function andFilterWhere(array $condition)
{ {
$condition = $this->filterCondition($condition); $condition = $this->filterCondition($condition);
if ($condition !== []) { if ($condition !== []) {
...@@ -168,15 +187,20 @@ trait QueryTrait ...@@ -168,15 +187,20 @@ trait QueryTrait
} }
/** /**
* Adds an additional WHERE condition to the existing one but ignores [[isParameterNotEmpty|empty parameters]]. * Adds an additional WHERE condition to the existing one but ignores [[isEmpty()|empty operands]].
* The new condition and the existing one will be joined using the 'OR' operator. * The new condition and the existing one will be joined using the 'OR' operator.
* @param string|array $condition the new WHERE condition. Please refer to [[where()]] *
* This method is similar to [[orWhere()]]. The main difference is that this method will
* remove [[isEmpty()|empty query operands]]. As a result, this method is best suited
* for building query conditions based on filter values entered by users.
*
* @param array $condition the new WHERE condition. Please refer to [[where()]]
* on how to specify this parameter. * on how to specify this parameter.
* @return static the query object itself * @return static the query object itself
* @see filterWhere() * @see filterWhere()
* @see andFilterWhere() * @see andFilterWhere()
*/ */
public function orFilterWhere($condition) public function orFilterWhere(array $condition)
{ {
$condition = $this->filterCondition($condition); $condition = $this->filterCondition($condition);
if ($condition !== []) { if ($condition !== []) {
...@@ -186,88 +210,92 @@ trait QueryTrait ...@@ -186,88 +210,92 @@ trait QueryTrait
} }
/** /**
* Returns a new condition with [[isParameterNotEmpty|empty parameters]] removed. * Removes [[isEmpty()|empty operands]] from the given query condition.
* *
* @param array $condition original condition * @param array $condition the original condition
* @return array condition with [[isParameterNotEmpty|empty parameters]] removed. * @return array the condition with [[isEmpty()|empty operands]] removed.
* @throws NotSupportedException if the condition operator is not supported * @throws NotSupportedException if the condition operator is not supported
*/ */
protected function filterCondition($condition) protected function filterCondition($condition)
{ {
if (is_array($condition) && isset($condition[0])) { if (!is_array($condition)) {
$operator = strtoupper($condition[0]); return $condition;
}
switch ($operator) { if (!isset($condition[0])) {
case 'NOT': // hash format: 'column1' => 'value1', 'column2' => 'value2', ...
case 'AND': foreach ($condition as $name => $value) {
case 'OR': if ($this->isEmpty($value)) {
for ($i = 1, $operandsCount = count($condition); $i < $operandsCount; $i++) { unset($condition[$name]);
$subCondition = $this->filterCondition($condition[$i]); }
if ($this->isParameterNotEmpty($subCondition)) { }
$condition[$i] = $subCondition; return $condition;
} else { }
unset($condition[$i]);
} // operator format: operator, operand 1, operand 2, ...
}
$operandsCount = count($condition) - 1; $operator = array_shift($condition);
if ($operator === 'NOT' && $operandsCount === 0) {
$condition = []; switch (strtoupper($operator)) {
case 'NOT':
case 'AND':
case 'OR':
foreach ($condition as $i => $operand) {
$subCondition = $this->filterCondition($operand);
if ($this->isEmpty($subCondition)) {
unset($condition[$i]);
} else { } else {
// reindex array $condition[$i] = $subCondition;
array_splice($condition, 0, 0);
if ($operandsCount === 1) {
$condition = $condition[1];
}
} }
}
if (empty($condition)) {
return [];
}
break; break;
case 'IN': case 'IN':
case 'NOT IN': case 'NOT IN':
case 'LIKE': case 'LIKE':
case 'OR LIKE': case 'OR LIKE':
case 'NOT LIKE': case 'NOT LIKE':
case 'OR NOT LIKE': case 'OR NOT LIKE':
if (!$this->isParameterNotEmpty($condition[2])) { if (array_key_exists(1, $condition) && $this->isEmpty($condition[1])) {
$condition = []; return [];
} }
break; break;
case 'BETWEEN': case 'BETWEEN':
case 'NOT BETWEEN': case 'NOT BETWEEN':
if (!$this->isParameterNotEmpty($condition[2]) && !$this->isParameterNotEmpty($condition[3])) { if (array_key_exists(1, $condition) && array_key_exists(2, $condition)) {
$condition = []; if ($this->isEmpty($condition[2]) || $this->isEmpty($condition[3])) {
return [];
} }
}
break; break;
default: default:
throw new NotSupportedException("filterWhere() does not support the '$operator' operator."); throw new NotSupportedException("Operator not supported: $operator");
}
} elseif (is_array($condition)) {
// hash format: 'column1' => 'value1', 'column2' => 'value2', ...
return array_filter($condition, [$this, 'isParameterNotEmpty']);
} else {
return $condition;
} }
array_unshift($condition, $operator);
return $condition; return $condition;
} }
/** /**
* Returns `true` if value passed is not "empty". * Returns a value indicating whether the give value is "empty".
* *
* The value is considered "empty", if * The value is considered "empty", if one of the following conditions is satisfied:
* *
* - it is `null`, * - it is `null`,
* - an empty string (`''`), * - an empty string (`''`),
* - a string containing only whitespace characters, * - a string containing only whitespace characters,
* - or an empty array. * - or an empty array.
* *
* @param $value * @param mixed $value
* @return boolean if parameter is empty * @return boolean if the value is empty
*/ */
protected function isParameterNotEmpty($value) protected function isEmpty($value)
{ {
if (is_string($value)) { return $value === '' || $value === [] || $value === null || is_string($value) && trim($value) === '';
$value = trim($value);
}
return $value !== '' && $value !== [] && $value !== null;
} }
/** /**
......
...@@ -115,7 +115,7 @@ class QueryTest extends SphinxTestCase ...@@ -115,7 +115,7 @@ class QueryTest extends SphinxTestCase
{ {
$query = new Query(); $query = new Query();
$query->filterWhere(['and', ['like', 'name', ''], ['like', 'title', ''], ['id' => 1], ['not', ['like', 'name', '']]]); $query->filterWhere(['and', ['like', 'name', ''], ['like', 'title', ''], ['id' => 1], ['not', ['like', 'name', '']]]);
$this->assertEquals(['id' => 1], $query->where); $this->assertEquals(['and', ['id' => 1]], $query->where);
} }
public function testGroup() public function testGroup()
......
...@@ -104,7 +104,7 @@ class QueryTest extends DatabaseTestCase ...@@ -104,7 +104,7 @@ class QueryTest extends DatabaseTestCase
{ {
$query = new Query(); $query = new Query();
$query->filterWhere(['and', ['like', 'name', ''], ['like', 'title', ''], ['id' => 1], ['not', ['like', 'name', '']]]); $query->filterWhere(['and', ['like', 'name', ''], ['like', 'title', ''], ['id' => 1], ['not', ['like', 'name', '']]]);
$this->assertEquals(['id' => 1], $query->where); $this->assertEquals(['and', ['id' => 1]], $query->where);
} }
public function testJoin() public function testJoin()
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment