Introduction
All packages in the Filament ecosystem share an asset management system. This allows both official plugins and third-party plugins to register CSS and JavaScript files that can then be consumed by Blade views.The FilamentAsset facade
The FilamentAsset facade is used to register files into the asset system. These files may be sourced from anywhere in the filesystem, but are then copied into the /public directory of the application when the php artisan filament:assets command is run. By copying them into the /public directory for you, we can predictably load them in Blade views, and also ensure that third party packages are able to load their assets without having to worry about where they are located.
Assets always have a unique ID chosen by you, which is used as the file name when the asset is copied into the /public directory. This ID is also used to reference the asset in Blade views. While the ID is unique, if you are registering assets for a plugin, then you do not need to worry about IDs clashing with other plugins, since the asset will be copied into a directory named after your plugin.
The FilamentAsset facade should be used in the boot() method of a service provider. It can be used inside an application service provider such as AppServiceProvider, or inside a plugin service provider.
The FilamentAsset facade has one main method, register(), which accepts an array of assets to register:
Registering assets for a plugin
When registering assets for a plugin, you should pass the name of the Composer package as the second argument of theregister() method:
/public, to avoid the possibility of clashing with other plugins’ files with the same names.
Registering CSS files
To register a CSS file with the asset system, use theFilamentAsset::register() method in the boot() method of a service provider. You must pass in an array of Css objects, which each represents a CSS file that should be registered in the asset system.
Each Css object has a unique ID and a path to the CSS file:
__DIR__ to generate a relative path to the asset from the current file. For instance, if you were adding this code to /app/Providers/AppServiceProvider.php, then the CSS file should exist in /resources/css/custom.css.
Now, when the php artisan filament:assets command is run, this CSS file is copied into the /public directory. In addition, it is now loaded into all Blade views that use Filament. If you’re interested in only loading the CSS when it is required by an element on the page, check out the Lazy loading CSS section.
Using Tailwind CSS in plugins
Typically, registering CSS files is used to register custom stylesheets for your application. If you want to process these files using Tailwind CSS, you need to consider the implications of that, especially if you are a plugin developer. Tailwind builds are unique to every application - they contain a minimal set of utility classes, only the ones that you are actually using in your application. This means that if you are a plugin developer, you probably should not be building your Tailwind CSS files into your plugin. Instead, you should provide the raw CSS files and instruct the user that they should build the Tailwind CSS file themselves. To do this, they need to add your vendor directory to their custom theme’s CSS file using the@source directive. In their custom theme CSS file (e.g., resources/css/filament/admin/theme.css), they should add:
Lazy loading CSS
By default, all CSS files registered with the asset system are loaded in the<head> of every Filament page. This is the simplest way to load CSS files, but sometimes they may be quite heavy and not required on every page. In this case, you can leverage the Alpine.js Lazy Load Assets package that comes bundled with Filament. It allows you to easily load CSS files on-demand using Alpine.js. The premise is very simple, you use the x-load-css directive on an element, and when that element is loaded onto the page, the specified CSS files are loaded into the <head> of the page. This is perfect for both small UI elements and entire pages that require a CSS file:
loadedOnRequest() method:
FilamentAsset::getStyleHref() method:
Registering CSS files from a URL
If you want to register a CSS file from a URL, you may do so. These assets will be loaded on every page as normal, but not copied into the/public directory when the php artisan filament:assets command is run. This is useful for registering external stylesheets from a CDN, or stylesheets that you are already compiling directly into the /public directory:
Registering CSS variables
Sometimes, you may wish to use dynamic data from the backend in CSS files. To do this, you can use theFilamentAsset::registerCssVariables() method in the boot() method of a service provider:
Registering JavaScript files
To register a JavaScript file with the asset system, use theFilamentAsset::register() method in the boot() method of a service provider. You must pass in an array of Js objects, which each represents a JavaScript file that should be registered in the asset system.
Each Js object has a unique ID and a path to the JavaScript file:
__DIR__ to generate a relative path to the asset from the current file. For instance, if you were adding this code to /app/Providers/AppServiceProvider.php, then the JavaScript file should exist in /resources/js/custom.js.
Now, when the php artisan filament:assets command is run, this JavaScript file is copied into the /public directory. In addition, it is now loaded into all Blade views that use Filament. If you’re interested in only loading the JavaScript when it is required by an element on the page, check out the Lazy loading JavaScript section.
Lazy loading JavaScript
By default, all JavaScript files registered with the asset system are loaded at the bottom of every Filament page. This is the simplest way to load JavaScript files, but sometimes they may be quite heavy and not required on every page. In this case, you can leverage the Alpine.js Lazy Load Assets package that comes bundled with Filament. It allows you to easily load JavaScript files on-demand using Alpine.js. The premise is very simple, you use thex-load-js directive on an element, and when that element is loaded onto the page, the specified JavaScript files are loaded at the bottom of the page. This is perfect for both small UI elements and entire pages that require a JavaScript file:
loadedOnRequest() method:
FilamentAsset::getScriptSrc() method:
Asynchronous Alpine.js components
Sometimes, you may want to load external JavaScript libraries for your Alpine.js-based components. The best way to do this is by storing the compiled JavaScript and Alpine component in a separate file, and letting us load it whenever the component is rendered. Firstly, you should install esbuild via NPM, which we will use to create a single JavaScript file containing your external library and Alpine component:bin/build.js:
resources/js/components/test-component.js into resources/js/dist/components/test-component.js. You can change these paths to suit your needs. You can compile as many components as you want.
Now, create a new file called resources/js/components/test-component.js:
resources/js/dist/components/test-component.js by running the following command:
/public directory of the Laravel application, so it is accessible to the browser. To do this, you can use the FilamentAsset::register() method in the boot() method of a service provider, passing in an AlpineComponent object:
php artisan filament:assets, the compiled file will be copied into the /public directory.
Finally, you can load this asynchronous Alpine component in your view using x-load attributes and the FilamentAsset::getAlpineComponentSrc() method:
state in as a parameter to the testComponent() function, which is entangled with a Livewire component property. You can pass in any parameters you want, and access them in the testComponent() function. If you’re not using a custom form field, you can ignore the state parameter in this example.
The x-load attributes come from the Async Alpine package, and any features of that package can be used here.
Registering script data
Sometimes, you may wish to make data from the backend available to JavaScript files. To do this, you can use theFilamentAsset::registerScriptData() method in the boot() method of a service provider:
window.filamentData object:
Registering JavaScript files from a URL
If you want to register a JavaScript file from a URL, you may do so. These assets will be loaded on every page as normal, but not copied into the/public directory when the php artisan filament:assets command is run. This is useful for registering external scripts from a CDN, or scripts that you are already compiling directly into the /public directory:
Using Vite-compiled JavaScript files
Thephp artisan filament:assets command copies files as-is into the /public directory without bundling or resolving dependencies. This means that if your JavaScript file uses import statements to pull in npm packages, the browser will not be able to resolve them. To use JavaScript files that require bundling, you should compile them with Vite first, and then register the compiled output as a URL-based asset.
First, add your JavaScript file as an entry point in your vite.config.js:
Vite::asset() to resolve the versioned URL:
Vite::asset() returns a URL, the asset will not be copied by php artisan filament:assets — it is served directly from Vite’s build output.
If you need to bundle JavaScript for an asynchronous Alpine.js component, consider using esbuild instead, as documented in that section.