redberry/page-builder-plugin

Page builder plugin for filamentphp admin panel to build pages using blocks.

dev-main 2025-04-03 19:29 UTC

This package is not auto-updated.

Last update: 2025-04-04 05:12:34 UTC


README

Latest Version on Packagist GitHub Tests Action Status Total Downloads

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'
        ]);
    });
})
  1. 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.