thinktomorrow / locale
A Laravel package for route and domain based localization
Installs: 6 781
Dependents: 0
Suggesters: 0
Security: 0
Stars: 2
Watchers: 4
Forks: 0
Open Issues: 12
pkg:composer/thinktomorrow/locale
Requires
- php: >=8.2
- laravel/framework: ^11.31
- thinktomorrow/url: ^4.0
Requires (Dev)
- mockery/mockery: ^1.6.6
- orchestra/testbench: ^9.9
- phpstan/phpstan: ^2.1
- phpunit/phpunit: ^11.5
- 5.0.0
- 4.0.0
- 3.0.2
- 3.0.1
- 3.0.0
- 2.5.3
- 2.5.2
- 2.5.1
- 2.5.0
- 2.4.1
- 2.4.0
- 2.3.4
- 2.3.3
- 2.3.2
- 2.3.1
- 2.3.0
- 2.2.0
- 2.1.2
- 2.1.1
- 2.1.0
- 2.0.x-dev
- 2.0.2
- 2.0.1
- 2.0.1-beta
- 1.5
- 1.4.1
- 1.4.0
- 1.3.2
- 1.3.1
- 1.3
- 1.2.0
- 1.1.1
- 1.1.0
- dev-master / 1.0.x-dev
- 1.0.4
- 1.0.3
- 1.0.2
- 1.0.1
- 1.0.0
- 0.0.6
- 0.0.5
- 0.0.4
- 0.0.3
- 0.0.2
- 0.0.1
- dev-dependabot/composer/guzzlehttp/psr7-2.5.0
- dev-dependabot/composer/symfony/http-kernel-6.2.6
- dev-dependabot/composer/doctrine/dbal-3.1.4
- dev-dependabot/composer/laravel/framework-8.52.0
- dev-dependabot/composer/psalm/plugin-laravel-1.5.1
- dev-dependabot/composer/vimeo/psalm-4.9.2
- dev-dependabot/composer/phpunit/phpunit-9.5.8
- dev-dependabot/composer/friendsofphp/php-cs-fixer-3.0.0
- dev-dependabot/composer/league/flysystem-1.1.4
- dev-dependabot/add-v2-config-file
- dev-dependabot/composer/mockery/mockery-1.4.3
- dev-fix/scope-detection-error
- dev-ft/canonical-check
This package is auto-updated.
Last update: 2025-10-05 07:52:53 UTC
README
Requirements
The package requires a php version of 8.2 or more. Laravel version 5.6.* and upwards is supported.
Installation
The package can be installed via Composer.
$ composer require thinktomorrow/locale
The package service provider will be autodiscovered by laravel.
To publish the config file to /config/thinktomorrow/locale.php, run:
    php artisan vendor:publish --provider="Thinktomorrow\Locale\LocaleServiceProvider"
These are the config defaults:
Setup
Let's say you want to support two locales: nl and en, where nl is the default locale. Here's how you would configure this:
# config/thinktomorrow/locale.php 'locales' => [ '*' => [ 'en' => 'en', '/' => 'nl', ], ],
Some important things to note here:
- The key of each entry represents the uri segment whereas the value is the application locale.
- The * acts as a wildcard group, which means any host will match. This is called the default scope but more on scopes later on.
- The default locale has a forward slash '/' as its key. This is a required item. It is the locale when no specific uri segment matches.
Segments
The package allows for two ways to identify the locale in a request url. Either via path segments or via the host. Let's take a moment to introduce these concepts. A locale segment is the uri path identifier of a specific locale. E.g. example.com/nl has nl as a locale segment since it identifies the locale in the given request uri.. This is the most common way of determining locales via the incoming request. This is also sufficient when your application only has to deal with one domain root.
The configuration for this is the most basic setup where you give a list of locales under the default scope, like the config example from above.
Scopes
A scope is a higher level identifier for a group of locales. Generally, you can compare a scope with the host part of the request url. A scope bundles one or more of these segments together.
Let's say we want to allow example.com and fr.example.com, where the first host localises in nl and the latter in fr. We can provide the following configuration:
# config/thinktomorrow/locale.php 'locales' => [ 'fr.example.com' => 'fr', '*' => 'nl', ],
- If the scope detects only one locale, it can be entered as a string, instead of an array.
- The more specific scope should be placed first, because matching is performed from top to bottom.
- The default scope '*' is always required and because of the first match first serve rule, should be placed at the bottom of the list.
If your application is visitable by only one domain root, which is the case for most apps, you are good to go with the default scope. All available application locales are grouped by so called scopes. Each scope has its own set * of locales. A scope which can be compared with domains. Each scope * could represent a domain and its supported locales. Each scope entry consists of a key as the * pattern identifier and an array of locales. Matches are done from top to bottom so declare * the more specific hosts above general ones.
Usage
To make your routes localized, place them inside a Route::group() with a following prefix:
    
    Route::group(['prefix' => localeRoutePrefix()],function(){
        
        // Routes registered within this group will be localized
        
    });
    
Generating a localized url
Localisation of your routes is done automatically when named routes are being used. Creation of all named routes will be localized based on current locale. Quick non-obtrusive integration.
route('pages.about'); // prints out http://example.com/en/about (if en is the active locale)
To create an url with a specific locale other than the active one, you can use the Thinktomorrow\Locale\LocaleUrl class.
    
    // Generate localized url from uri (resolves as laravel url() function)
    localeroute('about','en'); // http://example.com/en/about
    
    // Generate localized url from named route (resolves as laravel route() function)
    localeroute('pages.about','en'); // http://example.com/en/about  
    
    // Add additional parameters as third parameter
    localeroute('products.show','en',['slug' => 'tablet'])); // http://example/en/products/tablet
    
Note: Passing the locale as 'lang' query parameter will force the locale example.com/en/about?lang=nl makes sure the request will deal with a 'nl' locale.
Configuration
- 
locales: Whitelist of locales available for usage inside your application. Basic usage: 'locales' => [ '*' => [ 'nl', 'en', ] ], Multi-domain usage: 'locales' => [ 'https://awesome-domain-nl.com' => [ '/' => 'nl', ], 'https://awesome-domain-en.com' => [ '/' => 'en', ] ], Each multi-domain can have multiple locale as well: Multi-domain usage: 'locales' => [ 'https://awesome-domain.com' => [ 'en' => 'en', '/' => 'nl', ] ], 
- 
hidden_locale: You can set one of the available locales as 'hidden' which means any request without a locale in its uri, should be localized as this hidden locale. For example if the hidden locale is 'nl' and the request uri is /foo/bar, this request is interpreted with the 'nl' locale. Note that this is best used for your main / default locale. 
- 
placeholder: Explicit route placeholder for the locale. Must be used for the LocaleUrl::route()` method when multiple parameters need to be injected. 
Locale API
Set a new locale for current request
Locale::set('en'); // Sets a new application locale and returns the locale slug
Get the current locale
Locale::get(); // returns the current locale e.g. 'en'; // You can pass it a locale that will only be returned if it's a valid locale Locale::get('fr'); // returns 'fr' is fr is an accepted locale value Locale::get('foobar'); // ignores the invalid locale and returns the default locale
Get the locale slug to be used for url injection
Locale::getSlug(); // returns 'en' or null if the current locale is set to be hidden
Check if current locale is hidden
Locale::isHidden(); // checks current or passed locale and returns boolean
Testing
$ vendor/bin/phpunit
For more details check out our full documentation https://thinktomorrow.github.io/package-docs/src/locale/
Security
If you discover any security related issues, please email ben@thinktomorrow.be instead of using the issue tracker.
Credits
- Ben Cavens ben@thinktomorrow.be
License
The MIT License (MIT). Please see License File for more information.