mirror of
https://github.com/EGroupware/egroupware.git
synced 2025-01-14 18:08:21 +01:00
An attempt to add more functionality to searching
- All words are trimmed - Double quotes require the exact phrase ("Nathan Gray" will not match Nathan Brown or Gray Power) - Modifiers + and - before a word will require or exclude the word (+test -fail), otherwise word is ORed - User language modifiers AND, OR and NOT (uppercase) will be parsed. - Combinations: tracker entry -testing -"fatal error" Note that "entry" will not match "entries", and the results are not sorted by match strength. All applications that use so_sql for searching should gain these benefits.
This commit is contained in:
parent
9127fb5d71
commit
cc0f8f3c28
@ -782,7 +782,8 @@ class so_sql
|
|||||||
// if extending class or instanciator set columns to search, convert string criteria to array
|
// if extending class or instanciator set columns to search, convert string criteria to array
|
||||||
if ($criteria && !is_array($criteria))
|
if ($criteria && !is_array($criteria))
|
||||||
{
|
{
|
||||||
$criteria = $this->search2criteria($criteria,$wildcard,$op);
|
$search = $this->search2criteria($criteria,$wildcard,$op);
|
||||||
|
$criteria = array($search);
|
||||||
}
|
}
|
||||||
if (!is_array($criteria))
|
if (!is_array($criteria))
|
||||||
{
|
{
|
||||||
@ -1017,12 +1018,51 @@ class so_sql
|
|||||||
* @param string $extra_col=null extra column to search
|
* @param string $extra_col=null extra column to search
|
||||||
* @return array or column => value pairs
|
* @return array or column => value pairs
|
||||||
*/
|
*/
|
||||||
protected function search2criteria($pattern,&$wildcard='',&$op='AND',$extra_col=null)
|
public function search2criteria($pattern,&$wildcard='',&$op='AND',$extra_col=null, $search_cols = array())
|
||||||
{
|
{
|
||||||
|
// This function can get called multiple times. Make sure it doesn't re-process.
|
||||||
if (empty($pattern) || is_array($pattern)) return $pattern;
|
if (empty($pattern) || is_array($pattern)) return $pattern;
|
||||||
|
if(strpos($pattern, 'CONCAT') !== false) {
|
||||||
|
return $pattern;
|
||||||
|
}
|
||||||
|
|
||||||
|
$criteria = array();
|
||||||
|
$filter = array();
|
||||||
|
$columns = '';
|
||||||
|
$or_list = array();
|
||||||
|
if(!$search_cols) {
|
||||||
|
$search_cols = is_null($this->columns_to_search) ? $this->db_cols : $this->columns_to_search;
|
||||||
|
}
|
||||||
|
if(!$search_cols) return '';
|
||||||
|
|
||||||
|
// Concat all fields to be searched together, so the conditions operate across the whole record
|
||||||
|
foreach($search_cols as $col)
|
||||||
|
{
|
||||||
|
$columns .= "CAST(COALESCE($col,'') AS char),";
|
||||||
|
}
|
||||||
|
if(strlen($columns) > 0) {
|
||||||
|
$columns = 'CONCAT(' . substr($columns, 0, -1) . ')';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Break the search string into tokens
|
||||||
|
$break = ' ';
|
||||||
|
$token = strtok($pattern, $break);
|
||||||
|
|
||||||
|
while($token) {
|
||||||
|
if($token == strtoupper(lang('AND'))) {
|
||||||
|
$token = '+'.strtok($break);
|
||||||
|
} elseif ($token == strtoupper(lang('OR'))) {
|
||||||
|
continue;
|
||||||
|
} elseif ($token == strtoupper(lang('NOT'))) {
|
||||||
|
$token = '-'.strtok($break);
|
||||||
|
}
|
||||||
|
if ($token[0]=='"') {
|
||||||
|
$token = substr($token, 1,strlen($token));
|
||||||
|
$token .= ' '.strtok('"');
|
||||||
|
}
|
||||||
|
|
||||||
// prepend and append extra wildcard %, if pattern does NOT already contain wildcards
|
// prepend and append extra wildcard %, if pattern does NOT already contain wildcards
|
||||||
if (strpos($pattern,'*') === false && strpos($pattern,'?') === false)
|
if (strpos($token,'*') === false && strpos($token,'?') === false)
|
||||||
{
|
{
|
||||||
$wildcard = '%'; // if pattern contains no wildcards, add them before AND after the pattern
|
$wildcard = '%'; // if pattern contains no wildcards, add them before AND after the pattern
|
||||||
}
|
}
|
||||||
@ -1030,15 +1070,39 @@ class so_sql
|
|||||||
{
|
{
|
||||||
$wildcard = ''; // no extra wildcard, if pattern already contains some
|
$wildcard = ''; // no extra wildcard, if pattern already contains some
|
||||||
}
|
}
|
||||||
if ($pattern[0] != '!') $op = 'OR';
|
switch($token[0]) {
|
||||||
$criteria = array();
|
case '+':
|
||||||
foreach(is_null($this->columns_to_search) ? $this->db_cols : $this->columns_to_search as $col)
|
$op = 'AND';
|
||||||
{
|
$token = substr($token, 1, strlen($token));
|
||||||
$criteria[$col] = $pattern;
|
break;
|
||||||
|
case '-':
|
||||||
|
case '!':
|
||||||
|
$op = 'NOT';
|
||||||
|
$token = substr($token, 1, strlen($token));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
$op = 'OR';
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
if ($extra_col) $criteria[$extra_col] = $pattern;
|
$criteria[$op][] = " $columns LIKE " .
|
||||||
|
$GLOBALS['egw']->db->quote($wildcard.str_replace(array('%','_','*','?'),array('\\%','\\_','%','_'),$token).$wildcard);
|
||||||
|
|
||||||
return $criteria;
|
$token = strtok($break);
|
||||||
|
}
|
||||||
|
|
||||||
|
if($criteria['NOT']) {
|
||||||
|
$filter[] = 'NOT (' . implode(' OR ', $criteria['NOT']) . ') ';
|
||||||
|
}
|
||||||
|
if($criteria['AND']) {
|
||||||
|
$filter[] = implode(' AND ', $criteria['AND']) . ' ';
|
||||||
|
}
|
||||||
|
if($criteria['OR']) {
|
||||||
|
$filter[] = '(' . implode(' OR ', $criteria['OR']) . ') ';
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($extra_col) $filter[] = (strlen($filter) ? ' AND ' : ' ' ) . "$extra_col = $pattern";
|
||||||
|
$op = 'AND';
|
||||||
|
return $filter;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
Loading…
Reference in New Issue
Block a user