mindtwo / laravel-translatable
This package is a Laravel extension for easy translation of model attributes, enabling seamless multi-language support in your application.
Fund package maintenance!
mindtwo GmbH
Installs: 7
Dependents: 0
Suggesters: 0
Security: 0
Stars: 0
Watchers: 5
Forks: 0
Open Issues: 0
pkg:composer/mindtwo/laravel-translatable
Requires
- php: ^8.2||^8.3||^8.4
- illuminate/contracts: ^10.18||^11.0||^12.0
- illuminate/database: ^10.18||^11.0||^12.0
- mindtwo/laravel-auto-create-uuid: ^2.6
- spatie/laravel-package-tools: ^1.14.0
Requires (Dev)
- larastan/larastan: ^2.9||^3.0
- laravel/pint: ^1.0
- nunomaduro/collision: ^7.8||^8.0
- orchestra/testbench: ^10.0.0||^9.0.0||^8.22.0
- pestphp/pest: ^3.0
- pestphp/pest-plugin-arch: ^3.0
- pestphp/pest-plugin-laravel: ^3.0
- phpstan/extension-installer: ^1.3||^2.0
- phpstan/phpstan-deprecation-rules: ^1.1||^2.0
- phpstan/phpstan-phpunit: ^1.3||^2.0
- dev-main
- 2.1.0
- 2.0.0
- 1.5.0
- 1.4.1
- 1.4.0
- 1.3.1
- 1.3.0
- 1.2.2
- 1.2.1
- 1.2.0
- 1.1.2
- 1.1.1
- 1.1.0
- 1.0.3
- 1.0.2
- 1.0.1
- 1.0
- dev-feature/handle-default-locale-on-model
- dev-feature/extend-queries-to-allow-or-for-searches
- dev-feature/v1
- dev-update/composer-dependencies
- dev-remove-fields-for-good
- dev-optimize-get-translation
- dev-remove-nova-field
- dev-update-translatable-nova-field-for-dependables
- dev-add-rules-for-to-field
- dev-add-scope-helper
- dev-add-without-translations-to-model
- dev-add-fallback-locale
- dev-fix/field-update
- dev-allow-empty-translation-in-field
This package is auto-updated.
Last update: 2025-10-15 15:39:22 UTC
README
Overview
The mindtwo/laravel-translatable package provides a simple and effective way to manage multilingual models in a Laravel application. It allows you to easily translate Eloquent model attributes into multiple languages without the need for separate tables for each language.
Features
- ✅ Automatic Translation Override: Translated fields automatically override base model attributes
- ✅ Fallback Locale Support: Uses Laravel's app locale and fallback locale
- ✅ Query Scope Methods: Search, filter, and order by translated content
- ✅ Performance Optimized: Efficient database queries with proper indexing
- ✅ IDE Autocomplete: Full PHPDoc support for all methods
- ✅ Flexible Configuration: Customizable locale resolution and auto-translation
- ✅ Laravel Integration: Works seamlessly with Eloquent relationships and collections
Installation
To install the package, run the following command in your Laravel project:
composer require mindtwo/laravel-translatable
You can publish and run the migrations with:
php artisan vendor:publish --tag="translatable-migrations"
php artisan migrate
You can publish the config file with:
php artisan vendor:publish --tag="translatable-config"
This is the contents of the published config file:
<?php use mindtwo\LaravelTranslatable\Models\Translatable; use mindtwo\LaravelTranslatable\Resolvers\LocaleResolver; return [ /* |-------------------------------------------------------------------------- | Translatable Model |-------------------------------------------------------------------------- | | The model class to use for storing translations. | */ 'model' => Translatable::class, /* |-------------------------------------------------------------------------- | Locale Resolver |-------------------------------------------------------------------------- | | The resolver class to use for determining locale fallback chains. | */ 'resolver' => LocaleResolver::class, /* |-------------------------------------------------------------------------- | Auto Translate Attributes |-------------------------------------------------------------------------- | | When enabled, translatable attributes will be automatically translated | when accessed via magic attribute accessors (e.g., $model->title). | This can be overridden per model by implementing the autoTranslateAttributes() method. | */ 'auto_translate_attributes' => true, /* |-------------------------------------------------------------------------- | Default Locale on Model |-------------------------------------------------------------------------- | | When enabled, the default locale returned by the locale resolver is considered | to be stored on the model itself, i.e. not available in the translatable table | but stored directly in fields in the model table. | */ 'default_locale_on_model' => false, ];
Quick Start
1. Create a Translatable Model
<?php use Illuminate\Database\Eloquent\Model; use mindtwo\LaravelTranslatable\Traits\HasTranslations; class Product extends Model { use HasTranslations; protected $fillable = ['title', 'description', 'price']; /** * Define which fields are translatable */ protected $translatable = ['title', 'description']; }
2. Add Translations
$product = Product::create([ 'title' => 'Default Title', 'description' => 'Default Description', 'price' => 99.99 ]); // Add individual translations $product->setTranslation('title', 'English Product', 'en'); $product->setTranslation('title', 'Deutsches Produkt', 'de'); $product->setTranslation('title', 'Produit Français', 'fr'); // Or add multiple translations at once for a locale $product->setTranslations([ 'title' => 'English Product', 'description' => 'English Description', ], 'en');
3. Automatic Translation Override
// Set application locale app()->setLocale('de'); // Load model with translations $product = Product::withTranslations()->first(); echo $product->title; // "Deutsches Produkt" (automatically translated!) echo $product->description; // "Deutsche Beschreibung" echo $product->price; // 99.99 (non-translated field remains unchanged) // Get original value if needed echo $product->getUntranslated('title'); // "Default Title" // Get specific translation echo $product->getTranslation('title', 'en'); // "English Product" // Get all translations for a key $titles = $product->getAllTranslations('title'); // ['en' => 'English Product', 'de' => 'Deutsches Produkt', 'fr' => 'Produit Français']
Advanced Usage
Query Scope Methods
All scope methods support IDE autocomplete and have full parameter type hints:
// Eager load translations for current locale $products = Product::withTranslations()->get(); // Eager load translations for specific locales $products = Product::withTranslations(['en', 'de'])->get(); // Search in translated fields (with fallback locale support) $products = Product::searchByTranslation('title', 'Product')->get(); // Search in specific locales $products = Product::searchByTranslation('title', 'Product', ['en', 'de'])->get(); // Exact match search $exact = Product::searchByTranslationExact('title', 'Deutsches Produkt')->get(); // Prefix/suffix search $startsWith = Product::searchByTranslationStartsWith('title', 'German')->get(); $endsWith = Product::searchByTranslationEndsWith('description', 'warranty')->get(); // Multiple field search $multiField = Product::searchByTranslation(['title', 'description'], 'search term')->get(); // Filter models that have a translation $hasTitle = Product::whereHasTranslation('title')->get(); $hasGermanTitle = Product::whereHasTranslation('title', 'de')->get(); // Filter by exact translation value $products = Product::whereTranslation('title', 'Exact Title')->get(); // Order by translated field $sorted = Product::orderByTranslation('title', 'asc')->get(); // Method chaining works perfectly $results = Product::searchByTranslationStartsWith('title', 'Premium') ->whereHasTranslation('description') ->orderByTranslation('title') ->limit(10) ->get();
Locale Resolution
By default, the package uses Laravel's built-in locale configuration:
// The LocaleResolver provides locales in this order: // 1. Current application locale: app()->getLocale() // 2. Fallback locale: app()->getFallbackLocale()
Custom Locale Resolution
You can customize locale resolution by extending the LocaleResolver:
use mindtwo\LaravelTranslatable\Resolvers\LocaleResolver; class CustomLocaleResolver extends LocaleResolver { public function getLocales(): array { // Return custom locale chain return ['de', 'en', 'fr']; } }
Then register it in your config/translatable.php:
'resolver' => \App\Services\CustomLocaleResolver::class,
Or set locales dynamically:
use mindtwo\LaravelTranslatable\Resolvers\LocaleResolver; $resolver = app(LocaleResolver::class); $resolver->setLocales(['de', 'en', 'fr']); // Now queries will use this locale chain $products = Product::withTranslations()->get();
Working with Translation Data
// Check if translation exists if ($product->hasTranslation('title', 'de')) { echo "German translation available"; } // Get specific translation (returns translated text) $translation = $product->getTranslation('title', 'en'); echo $translation; // "English Product" // Get translations for specific locales $translations = $product->getTranslations('title', ['en', 'de']); // ['en' => 'English Product', 'de' => 'Deutsches Produkt'] // Get all translations for a key $allTitles = $product->getAllTranslations('title'); // ['en' => 'English Product', 'de' => 'Deutsches Produkt', 'fr' => 'Produit Français'] // Access the translations relationship $translationModels = $product->translations; foreach ($translationModels as $translation) { echo "{$translation->locale}: {$translation->key} = {$translation->text}"; }
Performance Features
- Efficient Eager Loading: Use
withTranslations()to eager load translations and avoid N+1 queries - Optimized Subqueries: Order by translation uses database-agnostic subqueries
- Index-Friendly Queries: Uses
whereIn()andwhereColumn()for better database performance - Cached Locale Resolution: Locale chain resolved once per request
- Translation Map Indexing: O(1) attribute access after initial load
Configuration Examples
E-commerce Setup
// Product.php class Product extends Model { use HasTranslations; protected $translatable = ['title', 'description', 'features']; } // Category.php class Category extends Model { use HasTranslations; protected $translatable = ['name', 'description']; }
Multi-Region Content
// Article.php class Article extends Model { use HasTranslations; protected $translatable = ['title', 'content', 'excerpt', 'meta_description']; // Disable auto-translation for this model if needed protected function autoTranslateAttributes(): bool { return false; } }
API Reference
Instance Methods
| Method | Description | Parameters |
|---|---|---|
setTranslation($key, $value, $locale) |
Set translation for a field | string $key, string $value, ?string $locale |
setTranslations($translations, $locale) |
Set multiple translations at once | array $translations, ?string $locale |
getTranslation($key, $locales) |
Get translated text with fallback | string $key, string|array|null $locales |
getTranslations($key, $locales) |
Get translations as array | string $key, string|array|null $locales |
getAllTranslations($key) |
Get all translations for a key | string $key |
hasTranslation($key, $locale) |
Check if translation exists | string $key, ?string $locale |
getUntranslated($key) |
Get original table value | string $key |
translations() |
Get translations relationship | - |
Query Scope Methods (Static)
| Method | Description | Parameters |
|---|---|---|
withTranslations($locales) |
Eager load translations | ?array $locales |
searchByTranslation($key, $search, $locales, $operator) |
Search in translations | string|array $key, string $search, string|array|null $locales, string $operator |
searchByTranslationExact($key, $search, $locales) |
Exact match search | string|array $key, string $search, string|array|null $locales |
searchByTranslationStartsWith($key, $search, $locales) |
Prefix search | string|array $key, string $search, string|array|null $locales |
searchByTranslationEndsWith($key, $search, $locales) |
Suffix search | string|array $key, string $search, string|array|null $locales |
whereHasTranslation($key, $locales) |
Filter models with translation | string $key, string|array|null $locales |
whereTranslation($key, $value, $locales, $operator) |
Filter by translation value | string $key, string $value, string|array|null $locales, string $operator |
orderByTranslation($key, $direction) |
Order by translated field | string $key, string $direction |
Required Configuration
Models using HasTranslations must define translatable fields:
// Property approach (recommended) protected $translatable = ['title', 'description']; // Or method approach (for backwards compatibility) public function translatedKeys(): array { return ['title', 'description']; }
Optional Configuration
// Disable auto-translation for specific model protected function autoTranslateAttributes(): bool { return false; }
IDE Support
The package includes comprehensive PHPDoc annotations for full IDE support:
- ✅ Autocomplete for all scope methods
- ✅ Parameter hints with proper types (
string|array,?string) - ✅ Return type information for method chaining
- ✅ Inline documentation on method hover
- ✅ Static analysis support (PHPStan compatible)
Testing
composer test
Changelog
Please see CHANGELOG for more information on what has changed recently.
Contributing
Please see CONTRIBUTING for details.
Security Vulnerabilities
Please review our security policy on how to report security vulnerabilities.
Credits
License
The MIT License (MIT). Please see License File for more information.