The sort
query parameter is used to determine by which property the results collection will be ordered. Sorting is ascending by default and can be reversed by adding a hyphen (-
) to the start of the property name.
All sorts have to be explicitly allowed by passing an array to the allowedSorts()
method. The allowedSorts
method takes an array of column names as strings or instances of AllowedSorts
.
For more advanced use cases, custom sorts can be used.
##Basic usage
$users = QueryBuilder::for(User::class)
->allowedSorts('name')
->get();
To define a default sort parameter that should be applied without explicitly adding it to the request, you can use the defaultSort
method.
$users = QueryBuilder::for(User::class)
->defaultSort('name')
->allowedSorts('name', 'street')
->get();
You can use -
if you want to have the default order sorted descendingly.
$users = QueryBuilder::for(User::class)
->defaultSort('-name')
->allowedSorts('name', 'street')
->get();
You can define multiple default sorts
$users = QueryBuilder::for(User::class)
->defaultSort('-street', 'name')
->allowedSorts('name', 'street')
->get();
You can sort by multiple properties by separating them with a comma:
$users = QueryBuilder::for(User::class)
->allowedSorts(['name', 'street'])
->get();
##Disallowed sorts
When trying to sort by a property that's not specified in allowedSorts()
an InvalidSortQuery
exception will be thrown.
$users = QueryBuilder::for(User::class)
->allowedSorts(['name'])
->get();
##Custom sorts
You can specify custom sorting methods using the AllowedSort::custom()
method. Custom sorts are instances of invokable classes that implement the \Spatie\QueryBuilder\Sorts\Sort
interface. The __invoke
method will receive the current query builder instance, the direction to sort in and the sort's name. This way you can build any sorting query your heart desires.
For example sorting by string column length:
class StringLengthSort implements \Spatie\QueryBuilder\Sorts\Sort
{
public function __invoke(Builder $query, bool $descending, string $property)
{
$direction = $descending ? 'DESC' : 'ASC';
$query->orderByRaw("LENGTH(`{$property}`) {$direction}");
}
}
The custom StringLengthSort
sort class can then be used like this to sort by the length of the users.name
column:
$users = QueryBuilder::for(User::class)
->allowedSorts([
AllowedSort::custom('name-length', new StringLengthSort(), 'name'),
])
->get();
To change the default direction of the a sort you can use defaultDirection
:
$customSort = AllowedSort::custom('custom-sort', new SentSort())->defaultDirection(SortDirection::DESCENDING);
$users = QueryBuilder::for(User::class)
->allowedSorts($customSort)
->defaultSort($customSort)
->get();
##Using an alias for sorting
There may be occasions where it is not appropriate to expose the column name to the user.
Similar to using an alias when filtering, you can do this for sorts as well.
The column name can be passed as optional parameter and defaults to the property string.
$users = QueryBuilder::for(User::class)
->allowedSorts([
AllowedSort::field('street', 'actual_column_street'),
])
->get();