humam-k98/laravel-filters

Laravel 8+ filtering toolkit using builder pattern

v1.1.0 2025-05-03 12:09 UTC

This package is auto-updated.

Last update: 2025-07-03 12:28:47 UTC


README

A powerful Laravel 8+ filtering package that provides a clean and flexible way to filter Eloquent models using the builder pattern. Now with support for Laravel 8, 9, 10, and 11! It enables developers to create robust API filtering systems with minimal code, supporting both manual filter creation and dynamic generation of filter classes based on model properties.

Features

  • Zero Configuration - Works out of the box with your Eloquent models
  • Flexible Filtering - Filter by exact matches, ranges, partial matches, and more
  • Sorting Support - Easy sorting with sort_by and sort_direction parameters
  • Framework Compatibility - Works with Laravel 8, 9, 10, and 11
  • Custom Filters - Create custom filter classes for complex filtering logic
  • Dynamic Filters - Generate filters on-the-fly based on model properties
  • Parameter Aliasing - Map request parameters to filter methods
  • Pagination Support - Seamlessly works with Laravel's pagination

Installation

You can install the package via composer:

composer require humam-k98/laravel-filters

The package will automatically register its service provider.

Basic Usage

1. Add the Filterable trait to your model

<?php

namespace App\Models;

use Illuminate\Database\Eloquent\Model;
use HumamK98\LaravelFilters\Traits\Filterable;

class User extends Model
{
    use Filterable;
    
    /**
     * The attributes that are mass assignable.
     *
     * @var array
     */
    protected $fillable = [
        'name', 'email', 'age', 'created_at',
    ];
    
    /**
     * The attributes that can be filtered.
     *
     * @var array
     */
    protected $filterable = [
        'name', 'email', 'age', 'created_at',
    ];
}

2. Start filtering with zero configuration!

The easiest way to use this package - dynamic filtering with no extra configuration:

<?php

namespace App\Http\Controllers;

use App\Models\User;
use Illuminate\Http\Request;

class UserController extends Controller
{
    /**
     * Display a listing of users with automatic filtering.
     *
     * @param  \Illuminate\Http\Request  $request
     * @return \Illuminate\Http\Response
     */
    public function index(Request $request)
    {
        // No filter class needed - it's created automatically!
        $users = User::filter()->paginate(10);
        
        return response()->json($users);
    }
}

Now you can make API requests with filter parameters:

GET /api/users?name=john&age_min=18&age_max=35&sort_by=created_at&sort_direction=desc

Advanced Usage

Creating a custom filter class for more control

For complex filtering needs, you can create your own filter class:

<?php

namespace App\Filters;

use HumamK98\LaravelFilters\Filters\Filter;

class UserFilter extends Filter
{
    /**
     * Get the allowed filters.
     *
     * @return array
     */
    protected function getAllowedFilters(): array
    {
        return [
            'name', 
            'email', 
            'age', 
            'age_min', 
            'age_max',
            'created_at',
            'sort_by',
            'sort_direction',
            'is_active', // Custom filter parameter
        ];
    }
    
    /**
     * Filter users by name.
     *
     * @param string $name
     * @return \Illuminate\Database\Eloquent\Builder
     */
    protected function name($name)
    {
        return $this->builder->where('name', 'like', "%{$name}%");
    }
    
    /**
     * Filter users by minimum age.
     *
     * @param int $age
     * @return \Illuminate\Database\Eloquent\Builder
     */
    protected function ageMin($age)
    {
        return $this->builder->where('age', '>=', $age);
    }
    
    /**
     * Filter users by maximum age.
     *
     * @param int $age
     * @return \Illuminate\Database\Eloquent\Builder
     */
    protected function ageMax($age)
    {
        return $this->builder->where('age', '<=', $age);
    }
    
    /**
     * Filter by active status (custom complex logic).
     *
     * @param bool $isActive
     * @return \Illuminate\Database\Eloquent\Builder
     */
    protected function isActive($isActive)
    {
        if ($isActive) {
            return $this->builder->whereNotNull('email_verified_at')
                ->where('banned', false);
        }
        
        return $this->builder->whereNull('email_verified_at')
            ->orWhere('banned', true);
    }
}

And in your controller:

public function index(Request $request, UserFilter $filter)
{
    $users = User::filter($filter)->paginate(10);
    
    return response()->json($users);
}

3 Filtering Approaches

1. Dynamic Auto-Generated Filters (Zero Configuration)

// In controller
public function index()
{
    // The filter is generated automatically based on model properties
    return User::filter()->paginate(10);
}

2. Using ModelFilter with Explicit Model

use HumamK98\LaravelFilters\Filters\ModelFilter;

public function index(Request $request)
{
    $filter = app(ModelFilter::class)->forModel(User::class);
    return User::filter($filter)->paginate(10);
}

3. Custom Filter Class for Complex Logic

use App\Filters\UserFilter;

public function index(Request $request, UserFilter $filter)
{
    return User::filter($filter)->paginate(10);
}

Filter Parameter Types

The package supports various filter parameter formats:

Parameter Format Example Description
{column} name=John Exact match
{column}_min age_min=18 Greater than or equal
{column}_max age_max=65 Less than or equal
{column}_like name_like=John LIKE search with % wildcards
sort_by sort_by=created_at Column to sort by
sort_direction sort_direction=desc Sort direction (asc/desc)

Camel Case Method Names

When creating custom filter classes, method names should be in camelCase format. For example:

  • For parameter age_min, create a method named ageMin
  • For parameter name_like, create a method named nameLike

The filter system will automatically convert snake_case parameters from the request to camelCase method names when calling your filter methods.

class UserFilter extends Filter
{
    protected function getAllowedFilters(): array
    {
        return ['name', 'email', 'age_min', 'age_max'];
    }
    
    // Method to handle 'age_min' parameter
    protected function ageMin($value)
    {
        return $this->builder->where('age', '>=', $value);
    }
    
    // Method to handle 'age_max' parameter
    protected function ageMax($value)
    {
        return $this->builder->where('age', '<=', $value);
    }
}

Filtering with Static Values

Using Static Conditions with Filters

You can combine the filter functionality with static conditions:

// In your controller
public function getAdminUsers(Request $request)
{
    // Apply both dynamic filters from request and a static condition
    $users = User::filter()
        ->where('type', 'Admin')
        ->paginate(10);
        
    return response()->json($users);
}

Using Query Scopes with Filters

Laravel query scopes work seamlessly with the filters:

// In your User model
class User extends Model
{
    use Filterable;
    
    // Define a query scope
    public function scopeOnlyAdmins($query)
    {
        return $query->where('type', 'Admin');
    }
}

// In your controller
public function getAdminUsers(Request $request)
{
    // Apply both the scope and dynamic filters
    $users = User::onlyAdmins()->filter()->paginate(10);
    
    return response()->json($users);
}

Custom Filter Method for Static Values

Create a dedicated filter method in your custom filter class:

class UserFilter extends Filter
{
    protected function getAllowedFilters(): array
    {
        return [
            'name', 
            'email',
            'only_admins', // Custom filter parameter for admins
            // other filters...
        ];
    }
    
    // Filter by admin type (static value)
    protected function onlyAdmins($value = true)
    {
        if ($value) {
            return $this->builder->where('type', 'Admin');
        }
        return $this->builder;
    }
}

Then in your API endpoint:

GET /api/users?only_admins=true

Adding Static Conditions to a Dynamic Filter

use HumamK98\LaravelFilters\Filters\DynamicFilterResolver;

public function index(Request $request, DynamicFilterResolver $resolver)
{
    $filter = $resolver->resolveFilterForModel(User::class);
    
    // Add a static condition to the filter
    $filter->addFilterClosure(function($builder) {
        return $builder->where('type', 'Admin');
    });
    
    $users = User::filter($filter)->paginate(10);
    return response()->json($users);
}

Examples

Simple Filtering

/api/users?name=John

Range Filtering

/api/users?age_min=18&age_max=35

Combined Filters with Sorting

/api/users?name_like=Smith&created_at_min=2023-01-01&sort_by=created_at&sort_direction=desc

Using DynamicFilterResolver Directly

For advanced cases, you can use the resolver directly:

use HumamK98\LaravelFilters\Filters\DynamicFilterResolver;

public function index(Request $request, DynamicFilterResolver $resolver)
{
    $filter = $resolver->resolveFilterForModel(User::class, ['request' => $request]);
    
    // Apply additional conditions to the filter if needed
    
    $users = User::filter($filter)->paginate(10);
    return response()->json($users);
}

Package Configuration

You can publish the configuration file with:

php artisan vendor:publish --tag=laravel-filters-config

This will create a config/laravel-filters.php file where you can adjust settings:

return [
    // Default operator for "like" searches
    'default_like_operator' => 'like', // Options: 'like', 'ilike' (PostgreSQL)
    
    // Whether to enable automatic query parameter aliasing
    'enable_param_aliasing' => true,
    
    // Common parameter mappings
    'param_mappings' => [
        'created_after' => 'created_at_min',
        'created_before' => 'created_at_max',
        // Add your own aliases here
    ],
];

License

The MIT License (MIT).