Filament Translatable Pro V2 is a powerful FilamentPHP v4 plugin that enhances the Spatie Translatable package with advanced, user-friendly features. This major version introduces significant improvements, including enhanced translation management, better performance, and a more intuitive API design.
You can find the V1 documentation here.
| URL | Username | Password |
|---|---|---|
| https://translatable-pro.afsakar.com | admin@example.com | password |
PS: Demo is reset every day at 05:00 AM UTC+3.
TranslateRecordAction and TranslateContentAction for comprehensive translation workflowsTranslatedColumn and TranslationProgressColumn for better data displayActions Namespace:
// V1
use Afsakar\FilamentTranslatablePro\Actions\TranslateAndCopyAction;
use Afsakar\FilamentTranslatablePro\Actions\TranslateFieldAction;
// V2
use Afsakar\FilamentTranslatablePro\Filament\Actions\TranslateRecordAction;
use Afsakar\FilamentTranslatablePro\Filament\Actions\TranslateContentAction;
Components Namespace:
// V1
use Afsakar\FilamentTranslatablePro\Forms\Components\TranslatableInput;
// V2
use Afsakar\FilamentTranslatablePro\Filament\Components\TranslatableInput;
Columns Namespace:
// V1
use Afsakar\FilamentTranslatablePro\Columns\TranslationProgressColumn;
// V2
use Afsakar\FilamentTranslatablePro\Filament\Columns\TranslationProgressColumn;
use Afsakar\FilamentTranslatablePro\Filament\Columns\TranslatedColumn; // New
Removed Methods:
->renderHook() - Global switcher location is now fixed->suffixLocale() and ->prefixLocale() - Now handled at component level->navigationTitle(), ->navigationGroup(), etc. - Now configured via config fileNew Methods:
->localeSwitcherType() - Choose between 'dropdown' and 'toggle'// V1
TranslateAndCopyAction::make() // Renamed
TranslateFieldAction::make() // Renamed
// V2
TranslateRecordAction::make() // New name
TranslateContentAction::make() // New name
// V1
TranslatableInput::make()
->prefixLocale() // Removed
->suffixLocale() // Removed
->exclude() // Renamed
// V2
TranslatableInput::make()
->prefixLocaleLabel() // New
->suffixLocaleLabel() // New
->excludeFields() // New name
// V1
use Afsakar\FilamentTranslatablePro\Facades\FilamentTranslatablePro;
// V2
use Afsakar\FilamentTranslatablePro\Facades\TranslatableProFacade;
Replace all old namespaces with new ones:
# Find and replace in your codebase:
# Actions
find . -name "*.php" -exec sed -i '' 's/use Afsakar\\FilamentTranslatablePro\\Actions\\/use Afsakar\\FilamentTranslatablePro\\Filament\\Actions\\/g' {} \;
# Components
find . -name "*.php" -exec sed -i '' 's/use Afsakar\\FilamentTranslatablePro\\Forms\\Components\\/use Afsakar\\FilamentTranslatablePro\\Filament\\Components\\/g' {} \;
# Columns
find . -name "*.php" -exec sed -i '' 's/use Afsakar\\FilamentTranslatablePro\\Columns\\/use Afsakar\\FilamentTranslatablePro\\Filament\\Columns\\/g' {} \;
# Facades
find . -name "*.php" -exec sed -i '' 's/use Afsakar\\FilamentTranslatablePro\\Facades\\FilamentTranslatablePro/use Afsakar\\FilamentTranslatablePro\\Facades\\TranslatableProFacade/g' {} \;
// Replace these action calls:
TranslateAndCopyAction::make() // Change to TranslateRecordAction::make()
TranslateFieldAction::make() // Change to TranslateContentAction::make()
// V1 Configuration (Remove these)
FilamentTranslatableProPlugin::make()
->renderHook(PanelsRenderHook::TOPBAR_START) // Remove
->suffixLocale(true) // Remove
->prefixLocale(true) // Remove
->navigationTitle('Content Translations') // Remove
->navigationGroup('Content') // Remove
->navigationSort(1) // Remove
->navigationIcon('heroicon-o-globe') // Remove
->navigationSlug('content-translations'); // Remove
// V2 Configuration (Use this)
FilamentTranslatableProPlugin::make()
->locales([
'tr' => 'Türkçe',
'en' => 'English'
])
->globalSwitcher(true)
->localeSwitcherType('dropdown'); // New: 'dropdown' or 'toggle'
// V1
TranslatableInput::make()
->prefixLocale() // Deprecated
->suffixLocale() // Deprecated
->exclude(['description']);
// V2
TranslatableInput::make()
->excludeFields(['description']);
Publish and update the config file:
php artisan vendor:publish --tag="filament-translatable-pro-config" --force
Update your config file:
// config/filament-translatable-pro.php
return [
'locales' => ['en', 'tr'],
// New configuration section
'translation_status' => [
'navigation_group' => 'Content',
'navigation_icon' => 'heroicon-o-language',
'model_label' => 'Translation Status',
'navigation_label' => 'Translation Status',
'slug' => 'translation-statuses',
'sort' => 99,
],
];
V2 includes new database tables:
php artisan migrate
Filament Translatable Pro uses AnyStack to handle payment, licensing, and distribution.
You can get your license here: FilamentPHP Translatable PRO
To install you'll need to add the repository to your composer.json file:
{
"repositories": [
{
"type": "composer",
"url": "https://satis.afsakar.com"
}
]
}
Once the repository has been added to the composer.json file, you can install Filament Translatable Pro like any other composer package using the composer require command:
composer require afsakar/filament-translatable-pro
You will be prompted to provide your username and password. The username will be the email address and the password will be equal to your license key.
Loading composer repositories with package information
Authentication required (satis.afsakar.com):
Username: [license-email]
Password: [license-key]
Next, add the plugin's views to your custom theme in your theme.css file:
Important
If you haven't set up a custom theme and are using a Panel, follow the instructions in the Filament Docs first. This setup applies to both the Panels and standalone Forms packages.
@import '../your/vendor/path/translatable-pro/resources/css/index.css';
@source '../your/vendor/path/afsakar/filament-translatable-pro/resources/views/**/*';
Afterward, run npm run build or yarn build to compile assets.
Publish the configuration file if needed:
php artisan vendor:publish --tag="filament-translatable-pro-config"
In the published config file, you can define the available locales and translation status settings:
return [
'locales' => ['tr', 'en'],
'translation_status' => [
'navigation_group' => 'Content',
'navigation_icon' => 'heroicon-o-language',
'model_label' => 'Translation Status',
'navigation_label' => 'Translation Status',
'slug' => 'translation-statuses',
'sort' => 99,
],
];
Optionally, publish translations using:
php artisan vendor:publish --tag="filament-translatable-pro-translations"
Run the migrations:
php artisan migrate
Register the plugin in your Filament service provider:
class AdminPanelProvider extends PanelProvider
{
public function panel(Panel $panel): Panel
{
return $panel
->plugins([
\Afsakar\FilamentTranslatablePro\FilamentTranslatableProPlugin::make()
->locales([
'tr' => 'Türkçe',
'en' => 'English'
])
->globalSwitcher(true) // Enable/disable the global switcher
->localeSwitcherType('dropdown') // 'dropdown' or 'toggle'
]);
}
}
To prepare your resources for translatable fields, add the Translatable trait in your Filament Resources:
<?php
namespace App\Filament\Resources;
use Afsakar\FilamentTranslatablePro\Resources\Concerns\Translatable;
use Filament\Resources\Resource;
class PostResource extends Resource
{
use Translatable;
// resource code
}
Add the Translatable trait to your List Page as well:
<?php
namespace App\Filament\Resources\PostResource\Pages;
use Afsakar\FilamentTranslatablePro\Resources\Pages\ListRecords\Concerns\Translatable;
use App\Filament\Resources\PostResource;
use Filament\Resources\Pages\ListRecords;
class ListPosts extends ListRecords
{
use Translatable;
protected static string $resource = PostResource::class;
}
Edit Page:
<?php
namespace App\Filament\Resources\PostResource\Pages;
use Afsakar\FilamentTranslatablePro\Filament\Actions\TranslateRecordAction;
use Afsakar\FilamentTranslatablePro\Resources\Pages\EditRecords\Concerns\Translatable;
use App\Filament\Resources\PostResource;
use Filament\Actions;
use Filament\Resources\Pages\EditRecord;
class EditPost extends EditRecord
{
use Translatable;
protected static string $resource = PostResource::class;
protected function getHeaderActions(): array
{
return [
TranslateRecordAction::make(),
Actions\DeleteAction::make(),
];
}
// You can use Translatable in widgets as well
protected function getHeaderWidgets(): array
{
return [
PostResource\Widgets\PostCommentsWidget::make([
'activeLocale' => $this->activeLocale, // You need to pass the active locale to the widget
]),
];
}
}
View Page:
<?php
namespace App\Filament\Resources\PostResource\Pages;
use Afsakar\FilamentTranslatablePro\Resources\Pages\ViewRecord\Concerns\Translatable;
use App\Filament\Resources\PostResource;
use Filament\Resources\Pages\ViewRecord;
class ViewPost extends ViewRecord
{
use Translatable;
protected static string $resource = PostResource::class;
}
Create Page:
<?php
namespace App\Filament\Resources\PostResource\Pages;
use Afsakar\FilamentTranslatablePro\Resources\Pages\CreateRecord\Concerns\Translatable;
use App\Filament\Resources\PostResource;
use Filament\Resources\Pages\CreateRecord;
class CreatePost extends CreateRecord
{
use Translatable;
protected static string $resource = PostResource::class;
}
PostCommentsWidget class:
<?php
namespace App\Filament\Resources\PostResource\Widgets;
use Afsakar\FilamentTranslatablePro\Facades\TranslatableProFacade;
use Afsakar\FilamentTranslatablePro\Resources\Widgets\Concerns\Translatable;
use Filament\Widgets\StatsOverviewWidget as BaseWidget;
class PostCommentsWidget extends BaseWidget
{
use Translatable;
public ?Model $record = null;
protected function getStats(): array
{
return [
BaseWidget\Stat::make('Total ('.TranslatableProFacade::getLocaleLabel($this->activeLocale).')', $this->record->comments->where('locale', $this->activeLocale)->count())
->icon('heroicon-o-chat-bubble-left-right'),
BaseWidget\Stat::make('Published ('.TranslatableProFacade::getLocaleLabel($this->activeLocale).')', $this->record->comments->where('locale', $this->activeLocale)->whereNotNull('published_at')->count())
->icon('heroicon-o-check-circle'),
BaseWidget\Stat::make('Unpublish ('.TranslatableProFacade::getLocaleLabel($this->activeLocale).')', $this->record->comments->where('locale', $this->activeLocale)->whereNull('published_at')->count())
->icon('heroicon-o-clock'),
];
}
}
For resources with relationships, include the Translatable trait in the related Resource Form Page:
<?php
namespace App\Filament\Resources\PostResource\RelationManagers;
use Afsakar\FilamentTranslatablePro\Resources\RelationManagers\Concerns\Translatable;
use Filament\Resources\RelationManagers\RelationManager;
class CategoriesRelationManager extends RelationManager
{
use Translatable;
protected static string $relationship = 'categories';
}
The TranslatableInput component supports multi-language input fields in Filament forms:
use Afsakar\FilamentTranslatablePro\Filament\Components\TranslatableInput;
use Filament\Forms;
use Filament\Actions\Action;
TranslatableInput::make()
->locales(['tr', 'en'])
->onlyMainLocaleRequired(condition: true, force: false)
->showFlags(true)
->excludeFields(['description'])
->vertical()
->schema([
Forms\Components\TextInput::make('name')->required(fn ($component) => $component->getMeta('locale') === 'tr'),
Forms\Components\Textarea::make('description')->required(fn ($component) => $component->getMeta('locale') === 'tr'),
]);
Available Methods:
excludeFields(): An array of fields to be excluded from the componentonlyMainLocaleRequired(): Make only main locale required if component has required attribute. If force is true, make all components required for main locale. (defaults: condition: true, force: false)showFlags(): Show flags in the tabslocales(): An array of locale codes to be used in the componentvertical(): Display the locale tabs verticallyuse Afsakar\FilamentTranslatablePro\Filament\Components\TranslatableInput;
use Filament\Forms;
TranslatableInput::make()
->schema([
Forms\Components\Section::make([
Forms\Components\TextInput::make('title'),
Forms\Components\Textarea::make('description'),
]),
]);
Get the current locale of a component with getMeta('locale'):
use Afsakar\FilamentTranslatablePro\Facades\TranslatableProFacade;
use Afsakar\FilamentTranslatablePro\Filament\Actions\TranslateContentAction;
use Afsakar\FilamentTranslatablePro\Filament\Components\TranslatableInput;
use Filament\Forms;
public static function form(Form $form): Form
{
return $form
->schema([
TranslatableInput::make()
->schema([
Forms\Components\Section::make('Title Section')
->columnSpanFull()
->columns()
->schema([
Forms\Components\TextInput::make('title')
->live(true)
->hintAction(TranslateContentAction::make())
->afterStateUpdated(function ($state, $set, $context, $component) {
if ($context === 'edit') {
return;
}
$language = $component->getMeta('locale');
$set('slug.' . $language, str($state)->slug());
})
->required(),
Forms\Components\TextInput::make('slug')
->dehydrateStateUsing(function ($state) {
return str($state)->slug();
})->unique(ignoreRecord: true),
]),
Forms\Components\Section::make()
->schema([
Forms\Components\FileUpload::make('image')->lazy(),
Forms\Components\RichEditor::make('content')->hintAction(TranslateContentAction::make()),
Forms\Components\Repeater::make('optional')
->label('Optional')
->addActionLabel(fn($component) => 'Add Optional (' . TranslatableProFacade::getLocaleLabel($component->getMeta('locale')) . ')')
->schema([
Forms\Components\TextInput::make('key'),
Forms\Components\TextInput::make('value'),
]),
]),
]),
]);
}
The TranslateRecordAction allows you to translate an entire record from one locale to another using automated translation services.
<?php
namespace App\Filament\Resources\PostResource\Pages;
use Afsakar\FilamentTranslatablePro\Filament\Actions\TranslateRecordAction;
use App\Filament\Resources\PostResource;
use Filament\Actions;
use Filament\Resources\Pages\EditRecord;
class EditPost extends EditRecord
{
protected static string $resource = PostResource::class;
protected function getHeaderActions(): array
{
return [
TranslateRecordAction::make()
->excepts(['slug']), // Exclude specific fields from translation
Actions\DeleteAction::make(),
];
}
}
The TranslateContentAction allows you to translate individual field content to other locales. This action can be used as a hint action on form fields.
use Afsakar\FilamentTranslatablePro\Filament\Actions\TranslateContentAction;
use Afsakar\FilamentTranslatablePro\Filament\Components\TranslatableInput;
use Filament\Forms;
TranslatableInput::make()
->schema([
Forms\Components\RichEditor::make('content')
->hintAction(TranslateContentAction::make()),
Forms\Components\TextInput::make('title')
->hintAction(TranslateContentAction::make()),
]);
Note: The TranslateContentAction action doesn't work with the Repeater component.
The CheckTranslationsAction allows you to check and update translation status for specific models. This is useful for monitoring translation completeness across your application.
use Afsakar\FilamentTranslatablePro\Filament\Actions\CheckTranslationsAction;
// Use in table header actions
protected function getHeaderActions(): array
{
return [
CheckTranslationsAction::make(),
];
}
The TranslationProgressColumn displays the translation progress of a record. It automatically detects the translatable attributes from the model and calculates the progress for each locale.
use Afsakar\FilamentTranslatablePro\Filament\Columns\TranslationProgressColumn;
public static function table(Table $table): Table
{
return $table
->columns([
Tables\Columns\TextColumn::make('title'),
TranslationProgressColumn::make('translation_progress')
->label('Translation Progress'),
// or use circle style
TranslationProgressColumn::make('translation_progress')
->label('Translation Progress')
->circle(),
]);
}
The TranslatedColumn automatically displays translated content based on the current active locale. It intelligently handles both translatable and non-translatable fields, including relationships.
use Afsakar\FilamentTranslatablePro\Filament\Columns\TranslatedColumn;
public static function table(Table $table): Table
{
return $table
->columns([
TranslatedColumn::make('title'), // Will show translated title based on active locale
TranslatedColumn::make('category.name'), // Works with relationships too
Tables\Columns\TextColumn::make('created_at'),
]);
}
First, you need to add the HasTranslationStatus trait to your models. You can exclude specific attributes from the translation status calculation by using the exceptedTranslationStatusAttributes method.
use Afsakar\FilamentTranslatablePro\Concerns\InteractsWithTranslationStatus;
class Post extends Model
{
use InteractsWithTranslationStatus;
public function exceptedTranslationStatusAttributes(): array
{
return ['slug']; // These fields won't be considered in translation status
}
}
The Translation Status Resource provides a comprehensive view of all translation statuses across your application. You can access it through the navigation menu (configurable in the config file).
Features:
This command calculates the translation status of records. It automatically detects the translatable attributes from the model and calculates the status for each locale. You can run this command from the terminal.
php artisan translations:check --model=Post
Available options:
--model: Specify the model class name to check (e.g., Post, User)Examples:
# Check all models
php artisan translations:check
# Check specific model
php artisan translations:check --model=Post
# Check multiple models (run command multiple times)
php artisan translations:check --model=Post
php artisan translations:check --model=Category
The command will:
If you have a question, bug or feature request, please e-mail me at afsakarr@gmail.com or tag @afsakar on #afsakar-translatable-pro on the Filament Discord. Love to hear from you!
This package is licensed under the MIT License. See the LICENSE file for details.
If you discover any security related issues, please email afsakarr@gmail.com instead of using the issue tracker.
Find the perfect tools to enhance your development workflow.
View All Products