Turn filament default radio button into a selectable card with icons, title and description.
- FilamentPHP v4.x
- PHP 8.2+
- Laravel v11.28+
- Tailwind CSS v4.0+
You can install the package via composer:
composer require jaocero/radio-deck
To adhere to Filament's theming approach, you'll be required to employ a personalized theme in order to utilize this plugin.
Custom Theme Installation Filament v4 Docs - Creating a Custom Theme
Instead of adding the plugin's views to your tailwind.config.js
file, add the following source directive to your custom theme's CSS file (usually resources/css/filament/admin/theme.css
):
@source '../../../../vendor/jaocero/radio-deck/resources/views';
This will include the plugin's styles during the compilation process.
If you're upgrading from Radio Deck v3 to v4, please follow these steps:
composer require jaocero/radio-deck:^2.0
Since FilamentPHP v4 requires custom themes for plugins, you need to create one:
php artisan make:filament-theme
Remove the old configuration from your tailwind.config.js
:
// Remove this from tailwind.config.js
content: [
...
'./vendor/jaocero/radio-deck/resources/views/**/*.blade.php', // Remove this line
]
Add the source directive to your theme's CSS file instead:
/* Add this to resources/css/filament/admin/theme.css */
@source '../../../../vendor/jaocero/radio-deck/resources/views';
Update your import statements to use the new namespace structure:
// Old (v1.x)
use JaOcero\RadioDeck\Forms\Components\RadioDeck;
// New (v2.x) - Same import, but make sure you're using v2.x
use JaOcero\RadioDeck\Forms\Components\RadioDeck;
Some method names have been updated for better consistency:
// Old method
->gap('gap-4')
// New method (renamed for clarity & to avoid conflicts with Filament’s built-in gap)
->optionsGap('gap-4')
use JaOcero\RadioDeck\Forms\Components\RadioDeck;
use Filament\Support\Enums\IconSize;
use Filament\Support\Enums\Alignment;
use Filament\Support\Enums\IconPosition;
public static function form(Form $form): Form
{
return $form
->schema([
RadioDeck::make('name')
->options([
'ios' => 'iOS',
'android' => 'Android',
'web' => 'Web',
'windows' => 'Windows',
'mac' => 'Mac',
'linux' => 'Linux',
])
->descriptions([
'ios' => 'iOS Mobile App',
'android' => 'Android Mobile App',
'web' => 'Web App',
'windows' => 'Windows Desktop App',
'mac' => 'Mac Desktop App',
'linux' => 'Linux Desktop App',
])
->icons([
'ios' => 'heroicon-m-device-phone-mobile',
'android' => 'heroicon-m-device-phone-mobile',
'web' => 'heroicon-m-globe-alt',
'windows' => 'heroicon-m-computer-desktop',
'mac' => 'heroicon-m-computer-desktop',
'linux' => 'heroicon-m-computer-desktop',
])
->required()
->iconSizes(IconSize::Medium) // Medium | Small | Large | ExtraLarge | TwoExtraLarge
->iconPosition(IconPosition::Before) // Before | After
->alignment(Alignment::Center) // Start | Center | End
->optionGap('gap-5') // Gap between Options and Descriptions between the Icon
->padding('px-4 py-6') // Padding around the deck
->extraCardsAttributes([ // Extra attributes for card elements
'class' => 'rounded-xl'
])
->extraOptionsAttributes([ // Extra attributes for option elements
'class' => 'text-3xl leading-none w-full flex flex-col items-center justify-center p-4'
])
->extraDescriptionsAttributes([ // Extra attributes for description elements
'class' => 'text-sm font-light text-center'
])
->multiple() // Enable multiple selection (returns array)
->colors('primary')
// or you can use an array of colors per option or you can use one color for all options
// ->colors([
// 'ios' => 'blue',
// 'android' => 'green',
// 'web' => 'purple',
// ])
->columns(3)
// or you can use how many columns every screen size
// ->columns([
// 'sm' => 1,
// 'md' => 2,
// 'lg' => 3,
// ])
->columnSpanFull()
]);
}
You can also utilize an Enum class for ->options()
, ->descriptions()
, and ->icons()
. Here's an example:
<?php
namespace App\Filament\Enums;
use Filament\Support\Contracts\HasLabel;
use JaOcero\RadioDeck\Contracts\HasDescriptions;
use JaOcero\RadioDeck\Contracts\HasIcons;
enum AssetType: string implements HasLabel, HasDescriptions, HasIcons
{
case iOs = 'ios';
case Android = 'android';
case Web = 'web';
case Windows = 'windows';
case Mac = 'mac';
case Linux = 'linux';
public function getLabel(): ?string
{
return match ($this) {
self::iOs => 'iOS',
self::Android => 'Android',
self::Web => 'Web',
self::Windows => 'Windows',
self::Mac => 'Mac',
self::Linux => 'Linux',
};
}
public function getDescriptions(): ?string
{
return match ($this) {
self::iOs => 'iOS Mobile App',
self::Android => 'Android Mobile App',
self::Web => 'Web App',
self::Windows => 'Windows Desktop App',
self::Mac => 'Mac Desktop App',
self::Linux => 'Linux Desktop App',
};
}
public function getIcons(): ?string
{
return match ($this) {
self::iOs => 'heroicon-m-device-phone-mobile',
self::Android => 'heroicon-m-device-phone-mobile',
self::Web => 'heroicon-m-globe-alt',
self::Windows => 'heroicon-m-computer-desktop',
self::Mac => 'heroicon-m-computer-desktop',
self::Linux => 'heroicon-m-computer-desktop',
};
}
}
Usage with Enum:
public static function form(Form $form): Form
{
return $form
->schema([
RadioDeck::make('name')
->options(AssetType::class)
->descriptions(AssetType::class)
->icons(AssetType::class)
->required()
->iconPosition(IconPosition::Before)
->alignment(Alignment::Center)
->colors('primary')
->columns(3),
])
->columns('full');
}
Please see CHANGELOG for more information on what has changed recently.
Please see CONTRIBUTING for details.
Please review our security policy on how to report security vulnerabilities.
The MIT License (MIT). Please see License File for more information.