Laravel Search Query Builder is a Illuminate\Database\Query\Builder
macro to easily search on multiple database columns and model relationships.
Let's imagine a case where you want to return all users with something in their name
or bio
field.
$search = 'Cesar';
// vanilla laravel
User::where('name', 'LIKE', '%'. $search .'%')->orWhere('bio', 'LIKE', '%'. $search .'%')->get();
// laravel with laravel-search-query-builder package
User::search(['name', 'bio'], $search)->get();
This example is not very difficult to achieve in Laravel, but it can grow very quickly, especially if you also want to search in model relationships like posts or comments for example.
Let's see how the Laravel Search Query Builder package can improve this situation.
Laravel Search Query Builder requires PHP 8+ and Laravel 9+.
You can install the package via composer:
composer require djl997/laravel-search-query-builder
The most basic call to the search
method requires two arguments:
- an array of columns,
- the search value.
For example, the following query retrieves users where the name
column or the value of the bio
column is like "Cesar".
$search = 'Cesar';
User::search(['name', 'bio'], $search)->get();
This will generate a query where "Cesar" is LIKE the name
or LIKE bio
column. For clarification, the search term is wrapped between %...%
to check if (a part of) the string is present.
You can also perform multiple searches at once by separating them with a komma.
$search = 'Cesar, Victoria';
User::search(['name', 'bio'], $search)->get();
This will generate a query for each part of the search term, exploded on a ,
by default.
If you want to change the delimiter, you can do so by changing the third argument.
User::search(['name', 'bio'], 'Cesar Victoria', '|')->get();
This will generate a query for each part of the search term, exploded on a |
.
As you probably know, the whereHas
method gives you a lot of power to define additional query constraints. And you guessed it, you can use the search
method here as well.
For example, the following query will return all users with "Laravel is cool" in their bios + all users who write posts that include this phrase in the title or body.
$search = 'Laravel is cool';
User::search(['bio'], $search)
->orWhereHas('posts', function($query) use ($search) {
$query->search(['title', 'body'], $search);
})
->get();
Since the search
method is developed to be a macro and extend the Illuminate\Database\Query\Builder
it can be used inline with any other Builder methods such as whereIn
, withTrashed
or orderBy
, or your very own macros.
$search = 'Laravel is cool';
User::whereIn('role', ['author', 'reviewer'])
->where(function($query) { //see note below
->search(['bio'], $search)
->orWhereHas('posts', function($query) use ($search) {
$query
->search(['title', 'body'], $search)
->whereNotNull('published_at');
})
->orWhereHas('comments', function($query) use ($search) {
$query->search(['body'], $search);
})
})
->withTrashed()
->orderBy('name')
->get();
This will return a ordered list of users with author-, or reviewer roles (deleted AND non-deleted), with "Laravel is cool" in their bios, published posts or comments, ordered by name.
Note! The search query is wrapped in a where query, because you most likely don't want the 'orWheres' to bypass any other filters.
This is a short-sighted example of how you can also make the columns dynamic:
<input type="text" name="search" value="Laravel is cool"/>
<input type="checkbox" name="userColumns[]" value="name"/>
<input type="checkbox" name="userColumns[]" value="bio"/>
$search = $request->input('search');
$columns = $request->input('userColumns');
User::search($columns, $search)->get();
Please see the GitHub releases for more information on what has changed recently.
Contributions or ideas are welcome.
The MIT License (MIT). Please see License File for more information.