redberry / page-builder-plugin
Page builder plugin for filamentphp admin panel to build pages using blocks.
Fund package maintenance!
RedberryProducts
Requires
- php: ^8.1
- filament/filament: ^3.0
- spatie/laravel-package-tools: ^1.15.0
Requires (Dev)
- laravel/pint: ^1.0
- nunomaduro/collision: ^7.9
- nunomaduro/larastan: ^2.0.1
- orchestra/testbench: ^8.0
- pestphp/pest: ^2.1
- pestphp/pest-plugin-arch: ^2.0
- pestphp/pest-plugin-laravel: ^2.0
- phpstan/extension-installer: ^1.1
- phpstan/phpstan-deprecation-rules: ^1.0
- phpstan/phpstan-phpunit: ^1.0
- spatie/laravel-ray: ^1.26
This package is not auto-updated.
Last update: 2025-04-04 05:12:34 UTC
README
- Page builder plugin
- Introduction
- Features
- Installation
- Usage
- Credits
- License
Introduction
this filamentphp plugin is aimed at allowing you to seamlessly integrate page builder functionality into your filament admin panel, preview changes in real-time via iframe or view files, and manage your content with ease.
Features
- predefined form and infolist component which can be fully customized
- previewing changes in real-time via iframes or view files
- easily customizable page builder block components
- ability to use every filamentphp field inside component blocks
- ability to fully customize formatting of page builder blocks
- predefined table and trait for easily adding page builder functionality to your resources
Installation
Pre-requisites
- PHP 8.1 or higher
- Laravel 10.x or higher
- Filament 3.x
You can install the package via composer:
composer require redberry/page-builder-plugin
You can publish and run the migrations with:
php artisan vendor:publish --tag="page-builder-plugin-migrations"
php artisan migrate
Optionally, you can publish the config file using:
php artisan vendor:publish --tag="page-builder-plugin-config"
Optionally, you can publish the views using
php artisan vendor:publish --tag="page-builder-plugin-views"
Usage
- add trait to the model you want to have page building functionality to
<?php use Redberry\PageBuilderPlugin\Traits\HasPageBuilder; class Page extends Model { use HasPageBuilder; }
this trait simply adds relationship to the model.
- next add PageBuilder form field to schema which you want to have page builder, like this:
<?php $form->schema([ PageBuilder::make('website_content') ->blocks([]), ]);
when blocks are empty it will not show any blocks, thankfully blocks can be created via command.
php artisan page-builder-plugin:make-block --type=view
this command will create a block class in the app/Filament/{id of admin panel}/Blocks
directory and also create a view file in the resources/views/blocks/
directory.
in block class that you just created you will notice function blockSchema
.
in blockSchema
function you must return schema of the block, this blocks are rendered just like filament forms, so you can use any filament form field and their features inside the block.
for example:
<?php class Description extends Block { public static function blockSchema(): array { return [ RichEditor::make('text') ->required() ]; } }
- now you can add new block that we just created to the page builder field like this:
<?php $form->schema([ PageBuilder::make('website_content') ->blocks([ Description::class, ]), ]);
that its it, now you can add/edit/delete blocks to the page builder field.
command for generating blocks
since in most project there will be many blocks, we decided to write command for generating blocks, this command will create block class and view file if you specify type as an view, otherwise it will create only block class.
php artisan page-builder-plugin:make-block
if you wish to customize any of the files that are created by the command you can do so by publishing the stubs using this command:
php artisan vendor:publish --tag="page-builder-plugin-stubs"
and modify them to your heart content.
additional configuration options
enable reorders
you can enable reordering of blocks by adding reorderable()
method to the page builder field like this:
<?php $form->schema([ PageBuilder::make('website_content') ->reorderable(), ]);
previewing in real time with iframe
by default components in preview are rendered using normal views, but sometimes you might want to preview them in iframe, in cases which components are located in another repository.
to enable iframe previewing call renderPreviewWithIframes
method on the page builder field like this:
<?php $form->schema([ PageBuilder::make('website_content') ->renderPreviewWithIframes( condition: true, createUrl: 'http://localhost:5173', ), ]);
sadly this is not all and some configurations are required to be done on the website that you will be using preview for. considering the fact that many frontend frameworks can not accept data right away from iframe you will need to notify filament about the fact that framework is fully hydrated and ready to load data, to do this after your framework is ready to be used run this code below:
// replace * with the url of your filament admin panel window.parent.postMessage({ type: "readyForPreview", }, "*");
this will send message to admin panel letting it know that page is ready to send data. to accept that data you will need to attach message event listener to the window like this:
window.addEventListener("message", (event) => console.log(event.data));
and that it. now website rendered via iframe will receive data from filament in real time.
iframe resizing
iframe height can not be adjusted based on content of iframe because of CORS issues, because of this there are two ways to size iframe height to not cause components to hide.
one is to just provide default height to iframe like this:
for PageBuilderPreview
and PageBuilderPreviewEntry
:
<?php PageBuilderPreview::make('...') // ... ->iframeAttributes([ 'height' => '500px' ]); ])
or incase of PageBuilder
like this:
<?php PageBuilder::make('...') // ... ->createAction(function (PageBuilder $action) { return $action->pageBuilderPreviewField(function (PageBuilderPreview $field) { return $field->iframeUrl('http://localhost:5173')->autoResizeIframe()->iframeAttributes([ 'height' => '500px' ]); }); })
- considering above solution is not the best, we provide ability to auto resize iframe height based on frontend events, to do this first configure backend to track auto resize
for PageBuilder
field:
<?php PageBuilder::make('...') ->renderPreviewWithIframes( condition: true, autoResize: true, createUrl: 'http://localhost:5173', ),
for PageBuilderPreview
and PageBuilderPreviewEntry
simply add autoResizeIframe
method to the preview field like this:
<?php PageBuilderPreview::make('...') // ... ->autoResizeIframe();
conditional schemas
there will be cases where you will need to change schema of the block based on some condition, of course this can be done normally by using closures and their own parameters, but for sake of convenience we provide same parameter injection for blockSchema
function on a block class, this function gets injected with following parameters: $record
, $action
, $component
, $livewire
. value of parameters work the same as they do in filament forms.
example of how to use this feature:
<?php class Description extends Block { public static function blockSchema($record): array { return [ RichEditor::make('text') ->default($record->text) ->required() ]; } }
all of this parameters are optional and you can use only the ones you need.
rendering page builder items on infolist
outside of form you might want to render page builder items on infolist, for this we provide two prebuilt entries:
PageBuilderEntry
and PageBuilderPreviewEntry
PageBuilderEntry
is used to render page builder items without rendering part itself,
PageBuilderPreviewEntry
functions in same way that PageBuilderPreview
does, but it is used to render page builder items on infolist which is about only difference.
both of these feature same type of confuguration including requirment to provide iframe urls, and blocks list.
example:
<?php $infolist ->schema([ PageBuilderEntry::make('website_content') ->blocks([LongDescription::class]) ->columnSpan(1), PageBuilderPreviewEntry::make('website_content_preview') ->blocks([LongDescription::class]) ->iframeUrl('http://localhost:5173') ->autoResizeIframe() ->columnSpan(2), ]);
rendering page builder item previews on forms
by default preview is rendered for create and edit. the same component that is used in create and edit actions can be used for listing as well, all you have to do is add PageBuilderPreview
to the schema and provide name of PageBuilder
field like so:
<?php PageBuilderPreview::make('website_content_preview') ->pageBuilderField('website_content') ->iframeUrl('http://localhost:5173') ->autoResizeIframe()
this will render preview of items selected in PageBuilder
field and it will update in real time.
customizing actions and button rendering
only component which has actions is PageBuilder
, all of this actions have their own modifier functions and are moved to the own class, so you can easily customize them, here is the list of actions, functions to modify them and their classes:
Action name | Class | Modifier function |
---|---|---|
Create | CreatePageBuilderBlockAction |
createAction |
Edit | EditPageBuilderBlockAction |
editAction |
Delete | DeletePageBuilderBlockAction |
deleteAction |
Reorder | ReorderPageBuilderBlockAction |
reorderAction |
Select block | SelectBlockAction |
selectBlockAction |
customizing buttons for actions
one strange thing about this package is how buttons are customized, because of how filamentphp actions are structured each button render comes with lot baggage to say so, multiple views, many checks and etc. while this is not too much of a problem if you are using couple of actions but due to nature of components for building a page there will be need for many many actions, 3 actions per component, its hard to quanitify exactly how much performance disadvantage this causes but in large project we first used this package in it became a massive problem to a point where removing those actions increase paged speeding 2-3 times, same numbers are replicable in this package on smaller scale as well, for example page which was rendering 65 components took around 500ms on local machine with using normal actions and no additional logic on their part while using simple buttons took around 150ms on average because of this drastic performance deference we decided to opt into using simple button rather than action. most buttons are rendered like this:
<?php return view('filament::components.button.index', [ 'slot' => $deleteAction->getLabel(), 'labelSrOnly' => true, 'icon' => 'heroicon-o-trash', 'color' => 'danger', 'disabled' => $deleteAction->isDisabled(), 'attributes' => collect([ 'wire:click' => "mountFormComponentAction('$statePath', '{$this->getDeleteActionName()}', { item: '$item', index: '$index' } )", ]), ])
as you can see we are using filament button to render our buttons, this gives us advantage of having pretty much same capabilities as filament actions but without all the performance issues that come with using them.
this buttons can also be easily change, even the component view itself using modifier functions provided on PageBuilder
component, this functions are injected with: $action
$item
$index
and $attributes
which can be injected via passing a closure to the modifier function,
this is a list of modifier functions and their corresponding actions:
Action name | Modifier function |
---|---|
Delete | deleteActionButton |
Edit | editActionButton |
Reorder | reorderActionButton |
here is example on how to use this modifier functions:
<?php PageBuilder::make('website_content') ->deleteActionButton(function ($action, $item, $index, $attributes) { return view('filament::components.button.index', [ ...$attributes, 'id' => 'delete-button' ] ); })
Credits
License
The MIT License (MIT). Please see License File for more information.