Generate unique urls for blogs, ecommerce and platforms without prefix.
Supports Laravel 9-12 | PHP 8.1-8.4
- When create or update a model to generate a unique url based on urlStrategy() function inside each model
- Possibility to have different urls for the different languages (not only a prefix in the beginning)
- If the url exists to create a new url with suffix _1, _2, etc.
- If we update the model to create a redirect from the old to the new url
- If there is a multiple redirects to redirect only to the last one
- Possibility to have an url depending on relations (category-name/product-name)
You can install the package via composer:
composer require vlados/laravel-unique-urlsYou can publish and run the migrations with:
php artisan vendor:publish --tag="laravel-unique-urls-migrations"
php artisan migrateYou can publish the config file with:
php artisan vendor:publish --tag="laravel-unique-urls-config"There will create unique-urls.php with:
return [
    // Locale => $language
    'languages' => [
        'bg_BG' => 'bg',
        'en_US' => 'en',
        'de_DE' => 'de',
    ],
    'redirect_http_code' => 301,
];In your Model add these methods:
class MyModel extends Model
{
    use Vlados\LaravelUniqueUrls\HasUniqueUrls;
    public function urlStrategy($language,$locale): string
    {
        return Str::slug($this->getAttribute('name'),"-",$locale);
    }
    
    public function urlHandler(): array
    {
        return [
            // The controller used to handle the request
            'controller' => CategoryController::class,
            // The method
            'method' => 'view',
            // additional arguments sent to this method
            'arguments' => [],
        ];
    }The method for handling the request:
public function view(Request $request, $arguments = [])
{
    dd($arguments);
}And last, add this line at the end of your routes/web.php
Route::get('{urlObj}', [\Vlados\LaravelUniqueUrls\LaravelUniqueUrlsController::class, 'handleRequest'])->where('urlObj', '.*');If for example you have category tree and you need to import all the data before creating the urls, you can disable the automatic generation of the url on model creation
To disable automatically generating the urls on create or update overwrite the method isAutoGenerateUrls in the model:
public function isAutoGenerateUrls(): bool
{
    return false;
}and call generateUrl() later like this:
YourModel::all()->each(function (YourModel $model) {
    $model->generateUrl();
});or if you want to disable it on the go, use
$model = new TestModel();
$model->disableGeneratingUrlsOnCreate();
$model->name = "Test";
$model->save();To use Livewire full-page component to handle the request, first set in urlHandler() function in your model:
public function urlHandler(): array
{
    return [
        // The Livewire controller
        'controller' => CategoryController::class,
        // The method should be empty
        'method' => '',
        // additional arguments sent to the mount() function
        'arguments' => [],
    ];
}Example livewire component:
class LivewireComponentExample extends Component
{
    private Url $urlModel;
    private array $url_arguments;
    public function mount(Url $urlObj, $arguments = [])
    {
        $this->urlModel = $urlObj;
        $this->url_arguments = $arguments;
    }
    public function render()
    {
        return view('livewire.view-category');
    }
}| Methods | Description | Parameters | 
|---|---|---|
| generateUrl() | Generate manually the URL | |
| getSlug() | Get the URL for a specific language in relative or absolute format | ?string $language = '', bool $relative = true | 
| urlStrategy | The strategy for creating the URL for the model | $language, $locale | 
| isAutoGenerateUrls() | Disable generating urls on creation, globally for the model | |
| disableGeneratingUrlsOnCreate() | Disable generating urls on creation | |
| Properties | ||
| relative_url | The url path, relative to the site url | |
| absolute_url | The absolute url, including the domain | |
| Relations | ||
| urls() | All the active urls, related to the current model | 
// Get relative URL for current locale
$model->relative_url; // e.g., "my-product-name"
// Get absolute URL for current locale
$model->absolute_url; // e.g., "https://example.com/my-product-name"
// Get URL for specific language (relative)
$model->getSlug('en', true); // e.g., "my-product-name"
// Get URL for specific language (absolute)
$model->getSlug('en', false); // e.g., "https://example.com/my-product-name"
// Get URL for current locale (defaults to app locale)
$model->getSlug(); // e.g., "my-product-name"This command generates unique URLs for the specified model or all models in the application that implement the HasUniqueUrls trait.
php artisan urls:generate [--model=ModelName] [--fresh] [--only-missing]This command checks if all models in the application have implemented the HasUniqueUrls trait and the required functions correctly.
php artisan urls:doctor [--model=ModelName]- Check if all the methods are implemented with the right parameters
- Check if the urlHandler is returning the correct controller and method
- Check if urlStrategy is generating unique urls for all the languages
composer testPlease see CHANGELOG for more information on what has changed recently.
See how a minor change to your commit message style can make you a better programmer.
Format: <type>(<scope>): <subject>
<scope> is optional
feat: add hat wobble
^--^  ^------------^
|     |
|     +-> Summary in present tense.
|
+-------> Type: chore, docs, feat, fix, refactor, style, or test.
More Examples:
- feat: (new feature for the user, not a new feature for build script)
- fix: (bug fix for the user, not a fix to a build script)
- docs: (changes to the documentation)
- style: (formatting, missing semi colons, etc; no production code change)
- refactor: (refactoring production code, eg. renaming a variable)
- test: (adding missing tests, refactoring tests; no production code change)
- chore: (updating grunt tasks etc; no production code change)
References:
- https://www.conventionalcommits.org/
- https://seesparkbox.com/foundry/semantic_commit_messages
- http://karma-runner.github.io/1.0/dev/git-commit-msg.html
Please review our security policy on how to report security vulnerabilities.
The MIT License (MIT). Please see License File for more information.