From d55d8323599b30e58074f9385778b852433ebe5c Mon Sep 17 00:00:00 2001 From: Charlie Pearce Date: Tue, 3 Dec 2019 08:42:38 +0000 Subject: [PATCH 1/2] Add case insensitive search modes, by forcing query to lowercase. --- src/Engines/Modes/LikeCaseInsensitive.php | 45 ++++++++++++++++ .../Modes/LikeExpandedCaseInsensitive.php | 53 +++++++++++++++++++ 2 files changed, 98 insertions(+) create mode 100644 src/Engines/Modes/LikeCaseInsensitive.php create mode 100644 src/Engines/Modes/LikeExpandedCaseInsensitive.php diff --git a/src/Engines/Modes/LikeCaseInsensitive.php b/src/Engines/Modes/LikeCaseInsensitive.php new file mode 100644 index 0000000..457ed2e --- /dev/null +++ b/src/Engines/Modes/LikeCaseInsensitive.php @@ -0,0 +1,45 @@ +fields = $this->modelService->setModel($builder->model)->getSearchableFields(); + + $queryString .= $this->buildWheres($builder); + + $queryString .= '('; + + foreach ($this->fields as $field) { + $queryString .= "LCASE(`$field`) LIKE LOWER(?) OR "; + } + + $queryString = trim($queryString, 'OR '); + $queryString .= ')'; + + return $queryString; + } + + public function buildParams(Builder $builder) + { + for ($itr = 0; $itr < count($this->fields); ++$itr) { + $this->whereParams[] = '%'.$builder->query.'%'; + } + + return $this->whereParams; + } + + public function isFullText() + { + return false; + } +} diff --git a/src/Engines/Modes/LikeExpandedCaseInsensitive.php b/src/Engines/Modes/LikeExpandedCaseInsensitive.php new file mode 100644 index 0000000..3c98c14 --- /dev/null +++ b/src/Engines/Modes/LikeExpandedCaseInsensitive.php @@ -0,0 +1,53 @@ +fields = $this->modelService->setModel($builder->model)->getSearchableFields(); + + $queryString .= $this->buildWheres($builder); + + $words = explode(' ', $builder->query); + + $queryString .= '('; + + foreach ($this->fields as $field) { + foreach ($words as $word) { + $queryString .= "LCASE(`$field`) LIKE LOWER(?) OR "; + } + } + + $queryString = trim($queryString, 'OR '); + $queryString .= ')'; + + return $queryString; + } + + public function buildParams(Builder $builder) + { + $words = explode(' ', $builder->query); + + for ($i = 0; $i < count($this->fields); ++$i) { + foreach ($words as $word) { + $this->whereParams[] = '%'.$word.'%'; + } + } + + return $this->whereParams; + } + + public function isFullText() + { + return false; + } +} From a4066e45d4cb056ea8735e9a7a2ecd66b34744bb Mon Sep 17 00:00:00 2001 From: Charlie Pearce Date: Tue, 3 Dec 2019 09:03:31 +0000 Subject: [PATCH 2/2] Update README.md to explain new LIKE_CASE_INSENSITIVE and LIKE_EXPANDED_CASE_INSENSITIVE modes. --- README.md | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/README.md b/README.md index 2806936..1a6d99a 100644 --- a/README.md +++ b/README.md @@ -163,6 +163,31 @@ and the following query in `LIKE_EXPANDED` mode given the search string "John Sm SELECT * FROM `customers` WHERE (`id` LIKE '%John%' OR `id` LIKE '%Smith%' OR `first_name` LIKE '%John%' OR `first_name` LIKE '%Smith%' OR `last_name` LIKE '%John%' OR `last_name` LIKE '%Smith%') ``` +### LIKE_CASE_INSENSITIVE and LIKE_EXPANDED_CASE_INSENSITIVE Modes + +`LIKE_CASE_INSENSITIVE` and `LIKE_EXPANDED_CASE_INSENSITIVE` modes are similar to `LIKE` and `LIKE_EXPANDED` modes except they will force both the field and parameter to lowercase in order +to perform a case-insensitive search. Note that you do not need to use this if you are already using a case-insensitive collation. + +For example running a search on a `Customer` model with the following database structure: + +| column name | type | +|-------------|------------------| +| id | int(10) UN AI PK | +| first_name | VARCHAR(255) | +| last_name | VARCHAR(255) | + +would produce the following query in `LIKE_CASE_INSENSITIVE` mode given the search string "John": + +```sql +SELECT * FROM `customers` WHERE (LCASE(`id`) LIKE LOWER('%John%') OR LCASE(`first_name`) LIKE LOWER('%John%') OR LCASE(`last_name`) LIKE LOWER('%JOHN%')) +``` + +and the following query in `LIKE_EXPANDED_CASE_INSENSITIVE` mode given the search string "John Smith": + +```sql +SELECT * FROM `customers` WHERE (LCASE(`id`) LIKE LOWER('%John%') OR LCASE(`id`) LIKE LOWER('%Smith%') OR LCASE(`first_name`) LIKE LOWER('%John%') OR LCASE(`first_name`) LIKE LOWER('%Smith%') OR LCASE(`last_name`) LIKE LOWER('%John%') OR LCASE(`last_name`) LIKE LOWER('%Smith%')) +``` + Console Command
---------------