Apex Charts integration for Filament
You can install the package via composer:
composer require leandrocfe/filament-apex-charts:"^3.1"
Register the plugin for the Filament Panels you want to use:
use Leandrocfe\FilamentApexCharts\FilamentApexChartsPlugin;public function panel(Panel $panel): Panel{ return $panel ->plugins([ FilamentApexChartsPlugin::make() ]);}
Filament V2 - if you are using Filament v2.x, you can use this section
Start by creating a widget with the command:
php artisan make:filament-apex-charts BlogPostsChart
You may choose:
You may also create an empty chart by selecting the Empty option.
This command will create the BlogPostsChart.php file in app\Filament\Widgets. Ex:
namespace App\Filament\Widgets;Â use Leandrocfe\FilamentApexCharts\Widgets\ApexChartWidget;Â class BlogPostsChart extends ApexChartWidget{ /** * Chart Id * * @var string */ protected static ?string $chartId = 'blogPostsChart';Â /** * Widget Title * * @var string|null */ protected static ?string $heading = 'BlogPostsChart';Â /** * Chart options (series, labels, types, size, animations...) * https://apexcharts.com/docs/options * * @return array */ protected function getOptions(): array { return [ 'chart' => [ 'type' => 'bar', 'height' => 300, ], 'series' => [ [ 'name' => 'BlogPostsChart', 'data' => [7, 4, 6, 10, 14, 7, 5, 9, 10, 15, 13, 18], ], ], 'xaxis' => [ 'categories' => ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'], 'labels' => [ 'style' => [ 'fontFamily' => 'inherit', 'fontWeight' => 600, ], ], ], 'yaxis' => [ 'labels' => [ 'style' => [ 'fontFamily' => 'inherit', ], ], ], 'colors' => ['#f59e0b'], ]; }}
Now, check out your new widget in the dashboard.
The getOptions()
method is used to return an array of options based on Apex Charts Options. This structure is identical with the Apex Chart library, which Filament Apex Charts
uses to render charts. You may use the Apex Chart documentation to fully understand the possibilities to return from getOptions().
You may set a widget title:
protected static ?string $heading = 'Blog Posts Chart';
Optionally, you can use the getHeading()
method.
You may set a widget subheading:
protected static ?string $subheading = 'This is a subheading';
Optionally, you can use the getSubheading()
method.
You may set a chart id:
protected static string $chartId = 'blogPostsChart';
You may set a widget height:
protected static ?int $contentHeight = 300; //px
Optionally, you can use the getContentHeight()
method.
protected function getContentHeight(): ?int{ return 300;}
You may set a widget footer:
protected static ?string $footer = 'Lorem Ipsum is simply dummy text of the printing and typesetting industry.';
You can also use the getFooter()
method:
Custom view:
use Illuminate\Contracts\Support\Htmlable;use Illuminate\Contracts\View\View;Â protected function getFooter(): null|string|Htmlable|View{ return view('custom-footer', ['text' => 'Lorem Ipsum is simply dummy text of the printing and typesetting industry.']);}
<!--resources/views/custom-footer.blade.php--><div> <p class="text-danger-500">{{ $text }}</p></div>
Html string:
use Illuminate\Contracts\Support\Htmlable;use Illuminate\Contracts\View\View;protected function getFooter(): null|string|Htmlable|View{ return new HtmlString('<p class="text-danger-500">Lorem Ipsum is simply dummy text of the printing and typesetting industry.</p>');}
You can hide header content by NOT providing these
You can set up chart filters to change the data shown on chart. Commonly, this is used to change the time period that chart data is rendered for.
You may use components from the Form Builder to create custom filter forms:
use Filament\Forms\Components\DatePicker;use Filament\Forms\Components\TextInput; protected function getFormSchema(): array{ return [ TextInput::make('title') ->default('My Chart'), DatePicker::make('date_start') ->default('2023-01-01'), DatePicker::make('date_end') ->default('2023-12-31') ];}
The data from the custom filter form is available in the $this->filterFormData array. You can use the active filter form values within your getOptions()
method:
protected function getOptions(): array{ $title = $this->filterFormData['title']; $dateStart = $this->filterFormData['date_start']; $dateEnd = $this->filterFormData['date_end'];Â return [ //chart options ];}
Also, if you want your chart data to update when the value of a filter changes, you have to combine reactive()
with afterStateUpdated()
:
use Filament\Forms\Components\DatePicker;use Filament\Forms\Components\TextInput;Â protected function getFormSchema(): array{ return [ TextInput::make('title') ->default('My Chart') ->live() ->afterStateUpdated(function () { $this->updateChartOptions(); }), ... ];}
You can change the filter form width by setting the $filterFormWidth
property:
use Filament\Support\Enums\MaxWidth;protected static MaxWidth|string $filterFormWidth = MaxWidth::Medium;
To set a default filter value, set the $filter
property:
public ?string $filter = 'today';
Then, define the getFilters()
method to return an array of values and labels for your filter:
protected function getFilters(): ?array{ return [ 'today' => 'Today', 'week' => 'Last week', 'month' => 'Last month', 'year' => 'This year', ];}
You can use the active filter value within your getOptions()
method:
protected function getOptions(): array{ $activeFilter = $this->filter;Â return [ //chart options ];}
By default, chart widgets refresh their data every 5 seconds.
To customize this, you may override the $pollingInterval
property on the class to a new interval:
protected static ?string $pollingInterval = '10s';
Alternatively, you may disable polling altogether:
protected static ?string $pollingInterval = null;
This can be helpful when you have slow queries and you don't want to hold up the entire page load:
protected static bool $deferLoading = true;Â protected function getOptions(): array{ //showing a loading indicator immediately after the page load if (!$this->readyToLoad) { return []; }Â //slow query sleep(2);Â return [ //chart options ];}
You can change the loading indicator:
protected static ?string $loadingIndicator = 'Loading...';
You can also use the getLoadingIndicator()
method:
use Illuminate\Contracts\View\View;protected function getLoadingIndicator(): null|string|View{ return view('custom-loading-indicator');}
<!--resources/views/custom-loading-indicator.blade.php--><div> <p class="text-danger-500">Loading...</p></div>
The dark mode is supported and enabled by default now.
Optionally, you can disable it:
protected static bool $darkMode = false;
You can also set the theme in the getOptions method:
protected function getOptions(): array{ return [ 'theme' => [ 'mode' => 'light' //dark ], 'chart' => [ 'type' => 'bar', ... ], ... ];}
You can use the extraJsOptions
method to add additional options to the chart:
protected function extraJsOptions(): ?RawJs{ return RawJs::make(<<<'JS' { xaxis: { labels: { formatter: function (val, timestamp, opts) { return val + '/24' } } }, yaxis: { labels: { formatter: function (val, index) { return '$' + val } } }, tooltip: { x: { formatter: function (val) { return val + '/24' } } }, dataLabels: { enabled: true, formatter: function (val, opt) { return opt.w.globals.labels[opt.dataPointIndex] + ': $' + val }, dropShadow: { enabled: true }, } } JS);}
Optionally, you can publish the views using:
php artisan vendor:publish --tag="filament-apex-charts-views"
composer test
Please see CHANGELOG for more information on what has changed recently.
Please see CONTRIBUTING for details.
If you discover a security vulnerability within this package, please send an e-mail to leandrocfe@gmail.com.
The MIT License (MIT). Please see License File for more information.
Leandro is a PHP/Laravel developer who has been coding for close to 7+ years. You can learn more about Leandro on his website.