Form Builder
Getting started
Preparing your Livewire component
Implement the HasForms
interface and use the InteractsWithForms
trait:
<?php namespace App\Http\Livewire; use Filament\Forms;use Illuminate\Contracts\View\View;use Livewire\Component; class EditPost extends Component implements Forms\Contracts\HasForms { use Forms\Concerns\InteractsWithForms; public function render(): View { return view('edit-post'); }}
In your Livewire component's view, render the form:
<form wire:submit.prevent="submit"> {{ $this->form }} <button type="submit"> Submit </button></form>
Finally, add any fields and layout components to the Livewire component's getFormSchema()
method:
<?php namespace App\Http\Livewire; use Filament\Forms;use Illuminate\Contracts\View\View;use Livewire\Component; class EditPost extends Component implements Forms\Contracts\HasForms{ use Forms\Concerns\InteractsWithForms; public Post $post; public $title; public $content; public function mount(): void { $this->form->fill([ 'title' => $this->post->title, 'content' => $this->post->content, ]); } protected function getFormSchema(): array { return [ Forms\Components\TextInput::make('title')->required(), Forms\Components\MarkdownEditor::make('content'), // ... ]; } public function submit(): void { // ... } public function render(): View { return view('edit-post'); }}
Visit your Livewire component in the browser, and you should see the form components from getFormSchema()
.
Initializing forms
You must initialize forms when the Livewire component is first loaded. This is done with the fill()
form method, often called in the mount()
method of the Livewire component.
For your fields to hold data, they should have a corresponding property on your Livewire component, just as in Livewire normally.
<?php namespace App\Http\Livewire; use App\Models\Post;use Filament\Forms;use Illuminate\Contracts\View\View;use Livewire\Component; class CreatePost extends Component implements Forms\Contracts\HasForms{ use Forms\Concerns\InteractsWithForms; public $title = ''; public $content = ''; public function mount(): void { $this->form->fill(); } protected function getFormSchema(): array { return [ Forms\Components\TextInput::make('title') ->default('Status Update') ->required(), Forms\Components\MarkdownEditor::make('content'), ]; } public function render(): View { return view('create-post'); }}
You may customize what happens after fields are filled using the afterStateHydrated()
method.
Filling forms with data
To fill a form with data, call the fill()
method on your form, and pass an array of data to fill it with:
<?php namespace App\Http\Livewire; use App\Models\Post;use Filament\Forms;use Illuminate\Contracts\View\View;use Livewire\Component; class EditPost extends Component implements Forms\Contracts\HasForms{ use Forms\Concerns\InteractsWithForms; public Post $post; public $title; public $content; public function mount(): void { $this->form->fill([ 'title' => $this->post->title, 'content' => $this->post->content, ]); } protected function getFormSchema(): array { return [ Forms\Components\TextInput::make('title')->required(), Forms\Components\MarkdownEditor::make('content'), ]; } public function render(): View { return view('edit-post'); }}
Getting data from forms
To get all form data in an array, call the getState()
method on your form.
<?php namespace App\Http\Livewire; use App\Models\Post;use Filament\Forms;use Illuminate\Contracts\View\View;use Livewire\Component; class CreatePost extends Component implements Forms\Contracts\HasForms{ use Forms\Concerns\InteractsWithForms; public $title = ''; public $content = ''; public function mount(): void { $this->form->fill(); } protected function getFormSchema(): array { return [ Forms\Components\TextInput::make('title')->required(), Forms\Components\MarkdownEditor::make('content'), ]; } public function create(): void { Post::create($this->form->getState()); } public function render(): View { return view('create-post'); }}
When getState()
is run:
- Validation rules are checked, and if errors are present, the form is not submitted.
- Any pending file uploads are stored permanently in the filesystem.
- Field relationships, if they are defined, are saved.
You may transform the value that is dehydrated from a field using the
dehydrateStateUsing()
method.
Registering a model
You may register a model to a form. The form builder is able to use this model to unlock DX features, such as:
- Automatically retrieving the database table name when using database validation rules like
exists
andunique
. - Automatically attaching relationships to the model when the form is saved, when using fields such as the
Select
,Repeater
,SpatieMediaLibraryFileUpload
, orSpatieTagsInput
.
Pass a model instance to a form using the getFormModel()
method:
<?php namespace App\Http\Livewire; use App\Models\Post;use Filament\Forms;use Illuminate\Contracts\View\View;use Livewire\Component; class EditPost extends Component implements Forms\Contracts\HasForms{ use Forms\Concerns\InteractsWithForms; public Post $post; public $title; public $content; public $tags; public function mount(): void { $this->form->fill([ 'title' => $this->post->title, 'content' => $this->post->content, ]); } protected function getFormSchema(): array { return [ Forms\Components\TextInput::make('title')->required(), Forms\Components\MarkdownEditor::make('content'), Forms\Components\SpatieTagsInput::make('tags'), ]; } protected function getFormModel(): Post { return $this->post; } public function render(): View { return view('edit-post'); }}
Alternatively, you may pass the model instance to the field that requires it directly, using the model()
method:
<?php namespace App\Http\Livewire; use App\Models\Post;use Filament\Forms;use Illuminate\Contracts\View\View;use Livewire\Component; class EditPost extends Component implements Forms\Contracts\HasForms{ use Forms\Concerns\InteractsWithForms; public Post $post; public $title; public $content; public $tags; public function mount(): void { $this->form->fill([ 'title' => $this->post->title, 'content' => $this->post->content, ]); } protected function getFormSchema(): array { return [ Forms\Components\TextInput::make('title')->required(), Forms\Components\MarkdownEditor::make('content'), Forms\Components\SpatieTagsInput::make('tags')->model($this->post), ]; } public function render(): View { return view('edit-post'); }}
You may now use field relationships;
Registering a model class
In some cases, the model instance is not available until the form has been submitted. For example, in a form that creates a post, the post model instance cannot be passed to the form before it has been submitted.
You may receive some of the same benefits of registering a model by registering its class instead:
<?php namespace App\Http\Livewire; use App\Models\Post;use Filament\Forms;use Illuminate\Contracts\View\View;use Livewire\Component; class CreatePost extends Component implements Forms\Contracts\HasForms{ use Forms\Concerns\InteractsWithForms; public $title = ''; public $content = ''; public $categories = []; public function mount(): void { $this->form->fill(); } protected function getFormSchema(): array { return [ Forms\Components\TextInput::make('title') ->required(), Forms\Components\MarkdownEditor::make('content'), Forms\Components\Select::make('categories') ->multiple() ->relationship('categories', 'name'), ]; } protected function getFormModel(): string { return Post::class; } public function render(): View { return view('create-post'); }}
You may now use field relationships.
Field relationships
Some fields, such as the Select
, Repeater
, SpatieMediaLibraryFileUpload
, or SpatieTagsInput
are able to interact with model relationships.
For example, Select
can be used to attach multiple records to a BelongstoMany
relationship. When registering a model to the form or component, these relationships will be automatically saved to the pivot table when getState()
is called:
<?php namespace App\Http\Livewire; use App\Models\Post;use Filament\Forms;use Illuminate\Contracts\View\View;use Livewire\Component; class EditPost extends Component implements Forms\Contracts\HasForms{ use Forms\Concerns\InteractsWithForms; public Post $post; public $title; public $content; public $categories; public function mount(): void { $this->form->fill([ 'title' => $this->post->title, 'content' => $this->post->content, ]); } protected function getFormSchema(): array { return [ Forms\Components\TextInput::make('title') ->required(), Forms\Components\MarkdownEditor::make('content'), Forms\Components\Select::make('categories') ->multiple() ->relationship('categories', 'name'), ]; } protected function getFormModel(): Post { return $this->post; } public function save(): void { $this->post->update( $this->form->getState(), ); } public function render(): View { return view('edit-post'); }}
Saving field relationships manually
In some cases, the model instance is not available until the form has been submitted. For example, in a form that creates a post, the post model instance cannot be passed to the form before it has been submitted. In this case, you will pass the model class instead, but any field relationships will need to be saved manually after.
In this situation, you may call the model()
and saveRelationships()
methods on the form after the instance has been created:
<?php namespace App\Http\Livewire; use App\Models\Post;use Filament\Forms;use Illuminate\Contracts\View\View;use Illuminate\Database\Eloquent\Model;use Livewire\Component; class CreatePost extends Component implements Forms\Contracts\HasForms{ use Forms\Concerns\InteractsWithForms; public $title = ''; public $content = ''; public $tags = []; public function mount(): void { $this->form->fill(); } protected function getFormSchema(): array { return [ Forms\Components\TextInput::make('title')->required(), Forms\Components\MarkdownEditor::make('content'), Forms\Components\SpatieTagsInput::make('tags'), ]; } protected function getFormModel(): string { return Post::class; } public function create(): void { $post = Post::create($this->form->getState()); $this->form->model($post)->saveRelationships(); } public function render(): View { return view('create-post'); }}
Saving relationships when the field is hidden
By default, relationships will only be saved if the field is visible. For example, if you have a Repeater
field that is only visible on a certain condition, the relationships will not be saved when it is hidden.
This might cause unexpected behaviour if you still want to save the relationship, even when the field is hidden. To force relationships to be saved, you may call the saveRelationshipsWhenHidden()
method on the form component:
use Filament\Forms\Components\SpatieMediaLibraryFileUpload; SpatieMediaLibraryFileUpload::make('attachments') ->visible(fn (Closure $get): bool => $get('has_attachments')) ->saveRelationshipsWhenHidden();
Using multiple forms
By default, the InteractsWithForms
trait only handles one form per Livewire component. To change this, you can override the getForms()
method to return more than one form, each with a unique name:
<?php namespace App\Http\Livewire; use App\Models\Author;use App\Models\Post;use Filament\Forms;use Illuminate\Contracts\View\View;use Livewire\Component; class EditPost extends Component implements Forms\Contracts\HasForms{ use Forms\Concerns\InteractsWithForms; public Author $author; public Post $post; public $title; public $content; public $name; public $email; public function mount(): void { $this->postForm->fill([ 'title' => $this->post->title, 'content' => $this->post->content, ]); $this->authorForm->fill([ 'name' => $this->author->name, 'email' => $this->author->email, ]); } protected function getPostFormSchema(): array { return [ Forms\Components\TextInput::make('title')->required(), Forms\Components\MarkdownEditor::make('content'), ]; } protected function getAuthorFormSchema(): array { return [ Forms\Components\TextInput::make('name')->required(), Forms\Components\TextInput::make('email')->email()->required(), ]; } public function savePost(): void { $this->post->update( $this->postForm->getState(), ); } public function saveAuthor(): void { $this->author->update( $this->authorForm->getState(), ); } protected function getForms(): array { return [ 'postForm' => $this->makeForm() ->schema($this->getPostFormSchema()) ->model($this->post), 'authorForm' => $this->makeForm() ->schema($this->getAuthorFormSchema()) ->model($this->author), ]; } public function render(): View { return view('edit-post'); }}
Scoping form data to an array property
You may scope the entire form data to a single array property on your Livewire component. This will allow you to avoid having to define a new property for each field:
<?php namespace App\Http\Livewire; use App\Models\Post;use Filament\Forms;use Illuminate\Contracts\View\View;use Livewire\Component; class EditPost extends Component implements Forms\Contracts\HasForms{ use Forms\Concerns\InteractsWithForms; public Post $post; public $data; public function mount(): void { $this->form->fill([ 'title' => $this->post->title, 'content' => $this->post->content, ]); } protected function getFormSchema(): array { return [ Forms\Components\TextInput::make('title')->required(), Forms\Components\MarkdownEditor::make('content'), Forms\Components\SpatieTagsInput::make('tags'), ]; } protected function getFormModel(): Post { return $this->post; } protected function getFormStatePath(): string { return 'data'; } public function render(): View { return view('edit-post'); }}
In this example, all data from your form will be stored in the $data
array.
Still need help? Join our Discord community or open a GitHub discussion