Components

TableBuilder

Basics

TableBuilder is a tool in MoonShine for creating customizable tables for displaying data. It is used on index and detail CRUD pages, as well as for relationship fields such as HasMany, BelongsToMany, RelationRepeater, and Json fields.

use MoonShine\UI\Components\Table\TableBuilder;TableBuilder::make(iterable $fields = [], iterable $items = [])
use MoonShine\UI\Components\Table\TableBuilder;
 
TableBuilder::make(iterable $fields = [], iterable $items = [])
<x-moonshine::table
:columns="[
'#', 'First', 'Last', 'Email'
]"
:values="[
[1, fake()->firstName(), fake()->lastName(), fake()->safeEmail()],
[2, fake()->firstName(), fake()->lastName(), fake()->safeEmail()],
[3, fake()->firstName(), fake()->lastName(), fake()->safeEmail()]
]"
/>

Basic Usage

Example of using TableBuilder:

TableBuilder::make() ->items([ ['id' => 1, 'title' => 'Hello world'] ]) ->fields([ ID::make()->sortable(), Text::make('Title', 'title'), ])
TableBuilder::make()
->items([
['id' => 1, 'title' => 'Hello world']
])
->fields([
ID::make()->sortable(),
Text::make('Title', 'title'),
])

Basic Methods

Fields

Fields for TableBuilder simplify the filling of data and displaying table cells. By default, fields are displayed in preview mode. The fields method defines the table fields, each field is a table cell (td):

->fields([ ID::make()->sortable(), Text::make('Title', 'title'), ])
->fields([
ID::make()->sortable(),
Text::make('Title', 'title'),
])

If you need to specify attributes for td, use the customWrapperAttributes method:

->fields([ ID::make()->sortable(), Text::make('Title', 'title')->customWrapperAttributes(['class' => 'my-class']), ])
->fields([
ID::make()->sortable(),
Text::make('Title', 'title')->customWrapperAttributes(['class' => 'my-class']),
])

Items

The items() method sets the data for the table:

->items($this->getCollection())
->items($this->getCollection())

Paginator

The paginator method sets the paginator for the table. You need to pass an object that implements the MoonShine\Contracts\Core\Paginator\PaginatorContract interface:

If you need to specify a paginator for QueryBuilder, you can use the built-in ModelCaster, as in the example below:

->paginator( (new ModelCaster(Article::class)) ->paginatorCast( Article::query()->paginate() ) )
->paginator(
(new ModelCaster(Article::class))
->paginatorCast(
Article::query()->paginate()
)
)

The paginator can also be specified through the items() method.

Simple Paginator

The simple() method applies a simplified pagination style to the table:

->simple()
->simple()

Buttons

The buttons method adds action buttons:

->buttons([ ActionButton::make('Delete', fn() => route('name.delete')), ActionButton::make('Edit', fn() => route('name.edit'))->showInDropdown(), ActionButton::make('Go to home', fn() => route('home'))->blank()->canSee(fn($data) => $data->active), ActionButton::make('Mass Delete', fn() => route('name.mass_delete'))->bulk(), ])
->buttons([
ActionButton::make('Delete', fn() => route('name.delete')),
ActionButton::make('Edit', fn() => route('name.edit'))->showInDropdown(),
ActionButton::make('Go to home', fn() => route('home'))->blank()->canSee(fn($data) => $data->active),
ActionButton::make('Mass Delete', fn() => route('name.mass_delete'))->bulk(),
])

To specify bulk actions on table items, the bulk() method should be set on ActionButton:

->buttons([ ActionButton::make('Mass Delete', fn() => route('name.mass_delete'))->bulk(), ])
->buttons([
ActionButton::make('Mass Delete', fn() => route('name.mass_delete'))->bulk(),
])

If you need to stick buttons, then use the stickyButtons() method:

->stickyButtons()
->stickyButtons()

View Methods

Vertical Display

The vertical() method displays the table in vertical format (used on DetailPage):

->vertical()
->vertical()

If you want to change the attributes of the columns in vertical mode, use the title or value parameters:

/** @param TableBuilder $component */ public function modifyDetailComponent(ComponentContract $component): ComponentContract { return $component->vertical( title: fn(FieldContract $field, Column $default, TableBuilder $ctx): ComponentContract => $default->columnSpan(2), value: fn(FieldContract $field, Column $default, TableBuilder $ctx): ComponentContract => $default->columnSpan(10), ); }
/** @param TableBuilder $component */
public function modifyDetailComponent(ComponentContract $component): ComponentContract
{
return $component->vertical(
title: fn(FieldContract $field, Column $default, TableBuilder $ctx): ComponentContract => $default->columnSpan(2),
value: fn(FieldContract $field, Column $default, TableBuilder $ctx): ComponentContract => $default->columnSpan(10),
);
}
  • title - Column with the header
  • value - Column with the value

You can also pass an integer value to specify the columns:

$component->vertical( title: 2, value: 10, )
$component->vertical(
title: 2,
value: 10,
)

Editable Table

The editable() method makes the table editable, switching all fields to defaultMode (form mode):

->editable()
->editable()

Preview Mode

The preview() method disables the display of buttons and sorting for the table:

->preview()
->preview()

With "Not Found" Notification

By default, if the table has no data, it will be empty, but you can display a message saying "No records found yet." To do this, use the withNotFound() method:

TableBuilder::make() ->withNotFound()
TableBuilder::make()
->withNotFound()

Row Customization

Fields accelerate the process and fill the table independently, constructing the table header with field headers and sorts, the body of the table with data output through fields, and the footer of the table with bulk actions. However, sometimes there may be a need to specify rows manually or add additional ones. For this task, methods are provided for the corresponding sections of the table: headRows (thead), rows (tbody), footRows (tfoot).

// tbody TableBuilder::make() ->rows( static fn(TableRowsContract $default) => $default->pushRow( TableCells::make()->pushCell( 'td content' ) ) ) // thead TableBuilder::make() ->headRows( static fn(TableRowContract $default) => TableRows::make([$default])->pushRow( TableCells::make()->pushCell( 'td content' ) ) ) // tfoot TableBuilder::make() ->footRows( static fn(?TableRowContract $default) => TableRows::make([$default])->pushRow( TableCells::make()->pushCell( 'td content' ) ) )
// tbody
TableBuilder::make()
->rows(
static fn(TableRowsContract $default) => $default->pushRow(
TableCells::make()->pushCell(
'td content'
)
)
)
 
 
// thead
TableBuilder::make()
->headRows(
static fn(TableRowContract $default) => TableRows::make([$default])->pushRow(
TableCells::make()->pushCell(
'td content'
)
)
)
 
// tfoot
TableBuilder::make()
->footRows(
static fn(?TableRowContract $default) => TableRows::make([$default])->pushRow(
TableCells::make()->pushCell(
'td content'
)
)
)

Note that for footRows, a ?TableRowContract is passed, and the value of $default will be passed as null if there are no bulk action buttons. The null value can be specified in the $items list in TableRows::make, and it will be ignored.

TableRows and TableCells are collections of components with additional functionality for quickly adding a row or cell to the table.

TableRows::make()->pushRow( TableCellsContract $cells, int|string|null $key = null, ?Closure $builder = null )
TableRows::make()->pushRow(
TableCellsContract $cells,
int|string|null $key = null,
?Closure $builder = null
)
  • $cells - a collection of cells,
  • $key - a unique key for tr for bulk actions and row update events,
  • $builder - access to TableBuilder.
TableCells::make()->pushCell( Closure|string $content, ?int $index = null, ?Closure $builder = null, array $attributes = [] )
TableCells::make()->pushCell(
Closure|string $content,
?int $index = null,
?Closure $builder = null,
array $attributes = []
)
  • $content - content of the cell,
  • $index - ordinal number of the cell,
  • $builder - access to TableBuilder,
  • $attributes - HTML attributes of the cell.

TableCells also has additional helper methods.

pushFields for quick generation of cells based on fields:

TableCells::make()->pushFields( FieldsContract $fields, ?Closure $builder = null, int $startIndex = 0 )
TableCells::make()->pushFields(
FieldsContract $fields,
?Closure $builder = null,
int $startIndex = 0
)
  • $fields - collection of fields,
  • $builder - access to TableBuilder,
  • $startIndex - starting index (since there may have already been cells added to the table previously)

Conditional methods pushWhen and pushCellWhen are also available.

Additional Features

Adding New Rows

The creatable() method allows adding new rows, making the table dynamic:

->creatable(reindex: true, limit: 5, label: 'Add', icon: 'plus', attributes: ['class' => 'my-class'])
->creatable(reindex: true, limit: 5, label: 'Add', icon: 'plus', attributes: ['class' => 'my-class'])
creatable( bool $reindex = true, ?int $limit = null, ?string $label = null, ?string $icon = null, array $attributes = [], ?ActionButtonContract $button = null )
creatable(
bool $reindex = true,
?int $limit = null,
?string $label = null,
?string $icon = null,
array $attributes = [],
?ActionButtonContract $button = null
)
  • $reindex - editing mode with dynamic name,
  • $limit - the number of records that can be added,
  • $label - button name,
  • $icon - button icon,
  • $attributes - additional attributes,
  • $button - custom add button.

In add mode, it is necessary for the last element to be empty (skeleton for a new record)!

If there are fields in the table in editing mode with dynamic name, you need to add the method or parameter reindex:

TableBuilder::make() ->creatable(reindex: true) TableBuilder::make() ->creatable() ->reindex()
TableBuilder::make()
->creatable(reindex: true)
 
TableBuilder::make()
->creatable()
->reindex()

Example with specifying a custom add button:

TableBuilder::make() ->creatable( button: ActionButton::make('Foo', '#') )
TableBuilder::make()
->creatable(
button: ActionButton::make('Foo', '#')
)

Reindexing

The reindex() method allows reindexing the table elements, adding an index to all name attributes of form elements. Example: The field Text::make('Title', 'title') in the first row of the table will look like <input name="title[1]">. In creatable or removable mode, when adding/removing a new row, all name attributes will be reindexed considering the ordinal number.

->reindex()
->reindex()

Drag and Drop Sorting

The reorderable() method adds the ability to sort rows by dragging:

->reorderable(url: '/reorder-url', key: 'id', group: 'group-name')
->reorderable(url: '/reorder-url', key: 'id', group: 'group-name')
  • $url - handler URL,
  • $key - item key,
  • $group - grouping (if required).

The sticky() method makes the table header fixed:

->sticky()
->sticky()

Column Selection

The columnSelection() method adds the ability to select displayed columns:

->columnSelection()
->columnSelection()

If you need to disable the display selection for certain fields, use the columnSelection method on the field with the parameter set to false:

TableBuilder::make() ->fields([ Text::make('Title') ->columnSelection(false), Text::make('Text') ]) ->columnSelection()
TableBuilder::make()
->fields([
Text::make('Title')
->columnSelection(false),
Text::make('Text')
])
->columnSelection()

When using columnSelection, the name parameter of the TableBuilder component must be unique across all pages. This is because data is stored in localStorage based on the value of the component's name.

The searchable() method adds the search function for the table:

->searchable()
->searchable()

Click Action

The clickAction() method sets an action to be performed on clicking the row: In the example below, clicking the table row will trigger a click on the edit button.

->clickAction(ClickAction::EDIT)
->clickAction(ClickAction::EDIT)

If you use custom buttons or have overridden the default buttons, you may also need to specify a button selector:

->clickAction(ClickAction::EDIT, '.edit-button')
->clickAction(ClickAction::EDIT, '.edit-button')

Types of ClickAction:

  • ClickAction::SELECT - select a row for bulk actions,
  • ClickAction::EDIT - go to edit,
  • ClickAction::DETAIL - go to detailed view.

Save State in URL

The pushState() method saves the state of the table in the URL:

->pushState()
->pushState()

Modify Row Checkbox

The modifyRowCheckbox() method allows modifying the bulk action checkbox. The example below demonstrates selecting the active checkbox by default:

->modifyRowCheckbox( fn(Checkbox $checkbox, DataWrapperContract $data, TableBuilder $ctx) => $data->getKey() === 2 ? $checkbox->customAttributes(['checked' => true]) : $checkbox )
->modifyRowCheckbox(
fn(Checkbox $checkbox, DataWrapperContract $data, TableBuilder $ctx) => $data->getKey() === 2 ? $checkbox->customAttributes(['checked' => true]) : $checkbox
)

Attribute Configuration

TableBuilder provides methods for configuring HTML attributes:

->trAttributes(fn(?DataWrapperContract $data, int $row): array => ['class' => $row % 2 ? 'bg-gray-100' : '']) ->tdAttributes(fn(?DataWrapperContract $data, int $row, int $cell): array => ['class' => $cell === 0 ? 'font-bold' : '']) ->headAttributes(['class' => 'bg-blue-500 text-white']) ->bodyAttributes(['class' => 'text-sm']) ->footAttributes(['class' => 'bg-gray-200']) ->customAttributes(['class' => 'custom-table'])
->trAttributes(fn(?DataWrapperContract $data, int $row): array => ['class' => $row % 2 ? 'bg-gray-100' : ''])
->tdAttributes(fn(?DataWrapperContract $data, int $row, int $cell): array => ['class' => $cell === 0 ? 'font-bold' : ''])
->headAttributes(['class' => 'bg-blue-500 text-white'])
->bodyAttributes(['class' => 'text-sm'])
->footAttributes(['class' => 'bg-gray-200'])
->customAttributes(['class' => 'custom-table'])

Async Loading

The async() method configures asynchronous loading of the table:

The async method must be after the name method

->async( Closure|string|null $url = null, string|array|null $events = null, ?AsyncCallback $callback = null, )
->async(
Closure|string|null $url = null,
string|array|null $events = null,
?AsyncCallback $callback = null,
)
  • $url - URL of the asynchronous request (the response must return TableBuilder),
  • $events - events that will be triggered after a successful response,
  • $callback - JS callback that can be added as a wrapper for the response.

After a successful request, you can trigger events by adding the events parameter.

use MoonShine\Support\AlpineJs; use MoonShine\Support\Enums\JsEvent; TableBuilder::make() ->name('crud') ->async(events: [ AlpineJs::event(JsEvent::FORM_RESET, 'main-form'), AlpineJs::event(JsEvent::TOAST, params: ['text' => 'Success', 'type' => 'success']), ])
use MoonShine\Support\AlpineJs;
use MoonShine\Support\Enums\JsEvent;
 
TableBuilder::make()
->name('crud')
->async(events: [
AlpineJs::event(JsEvent::FORM_RESET, 'main-form'),
AlpineJs::event(JsEvent::TOAST, params: ['text' => 'Success', 'type' => 'success']),
])

Event list for TableBuilder:

  • JsEvent::TABLE_UPDATED - table update,
  • JsEvent::TABLE_REINDEX - table reindexing (see reindex())
  • JsEvent::TABLE_ROW_UPDATED - table row update (AlpineJs::event(JsEvent::TABLE_ROW_UPDATED, "{component-name}-{row-id}"))

For more information on js events, refer to the Events section.

All parameters of the async method are optional, and by default, TableBuilder will automatically set URL based on the current page.

In the process of using TableBuilder in async mode, there may arise a task where you use it outside the admin panel on pages that are not declared in the MoonShine system. Then you will need to specify your own URL and implement a response with the HTML table. Let's consider an implementation example:

TableBuilder::make()->name('my-table')->async(route('undefined-page.component', [ '_namespace' => self::class, '_component_name' => 'my-table' ]))
TableBuilder::make()->name('my-table')->async(route('undefined-page.component', [
'_namespace' => self::class,
'_component_name' => 'my-table'
]))

Controller

namespace App\MoonShine\Controllers; use Illuminate\Contracts\View\View; use MoonShine\Laravel\MoonShineRequest; use MoonShine\Laravel\Http\Controllers\MoonShineController; final class UndefinedPageController extends MoonShineController { public function component(MoonShineRequest $request): View { $page = app($request->input('_namespace')); $component = $page->getComponents()->findByName( $request->getComponentName() ); return $component->render(); } }
namespace App\MoonShine\Controllers;
 
use Illuminate\Contracts\View\View;
use MoonShine\Laravel\MoonShineRequest;
use MoonShine\Laravel\Http\Controllers\MoonShineController;
 
final class UndefinedPageController extends MoonShineController
{
public function component(MoonShineRequest $request): View
{
$page = app($request->input('_namespace'));
 
$component = $page->getComponents()->findByName(
$request->getComponentName()
);
 
return $component->render();
}
}

Lazy and whenAsync Methods

If you need to send a request to update the TableBuilder component immediately upon page load, you must add the lazy() method. Additionally, the lazy() and whenAsync() methods in combination can solve the problem of lazy loading data or loading data from an external source.

TableBuilder::make() ->name('dashboard-table') ->fields([ ID::make(), Slug::make('Slug'), Text::make('Title'), Preview::make('Image')->image() ]) ->async() ->lazy() ->whenAsync( fn(TableBuilder $table) => $table->items( Http::get('https://jsonplaceholder.org/posts')->json() ) ),
TableBuilder::make()
->name('dashboard-table')
->fields([
ID::make(),
Slug::make('Slug'),
Text::make('Title'),
Preview::make('Image')->image()
])
->async()
->lazy()
->whenAsync(
fn(TableBuilder $table) => $table->items(
Http::get('https://jsonplaceholder.org/posts')->json()
)
),

The whenAsync() method checks if the current request is asynchronous to get the current TableBuilder component. An example interaction with the methods where the loading of the table occurs by clicking a button:

ActionButton::make('Reload') ->async(events: [AlpineJs::event(JsEvent::TABLE_UPDATED, 'my-table')]), TableBuilder::make() ->name('my-table') ->fields([ ID::make(), Slug::make('Slug'), Text::make('Title'), Preview::make('Image')->image() ]) ->async() ->lazy() ->whenAsync( fn(TableBuilder $table) => $table->items( Http::get('https://jsonplaceholder.org/posts')->json() ) ), ->withNotFound()
ActionButton::make('Reload')
->async(events: [AlpineJs::event(JsEvent::TABLE_UPDATED, 'my-table')]),
 
TableBuilder::make()
->name('my-table')
->fields([
ID::make(),
Slug::make('Slug'),
Text::make('Title'),
Preview::make('Image')->image()
])
->async()
->lazy()
->whenAsync(
fn(TableBuilder $table) => $table->items(
Http::get('https://jsonplaceholder.org/posts')->json()
)
),
->withNotFound()

Type Cast

If you use data in the table without cast, you must specify what your data has as a key. Otherwise, some features, such as bulk operations, will not work.

Example:

TableBuilder::make() ->castKeyName('id') ->name('my-table') ->fields([ ID::make(), Text::make('Title') ]) ->items([ ['id' => 3,'title' => 'Hello world'] ]) ->buttons([ ActionButton::make('Mass Delete') ->bulk() ]),
TableBuilder::make()
->castKeyName('id')
->name('my-table')
->fields([
ID::make(),
Text::make('Title')
])
->items([
['id' => 3,'title' => 'Hello world']
])
->buttons([
ActionButton::make('Mass Delete')
->bulk()
]),

The cast method is used to cast values in the table to a certain type. Since by default, fields work with primitive types:

use MoonShine\Laravel\TypeCasts\ModelCaster; TableBuilder::make() ->cast(new ModelCaster(User::class))
use MoonShine\Laravel\TypeCasts\ModelCaster;
 
TableBuilder::make()
->cast(new ModelCaster(User::class))

In this example, we cast the data to the model format of User using ModelCaster.

For more detailed information, refer to the TypeCasts section.

Using in Blade

Basics

Styled tables can be created using the moonshine::table component.

<x-moonshine::table
:columns="[
'#', 'First', 'Last', 'Email'
]"
:values="[
[1, fake()->firstName(), fake()->lastName(), fake()->safeEmail()],
[2, fake()->firstName(), fake()->lastName(), fake()->safeEmail()],
[3, fake()->firstName(), fake()->lastName(), fake()->safeEmail()]
]"
/>

Simple View

The simple parameter allows creating a simplified view of the table.

<x-moonshine::table
:simple="true"
:columns="[
'#', 'First', 'Last', 'Email'
]"
:values="[
[1, fake()->firstName(), fake()->lastName(), fake()->safeEmail()],
[2, fake()->firstName(), fake()->lastName(), fake()->safeEmail()],
[3, fake()->firstName(), fake()->lastName(), fake()->safeEmail()]
]"
/>

Sticky Header

If the table contains a large number of items, you can sticky the head while scrolling the table.

<x-moonshine::table
:sticky="true"
:columns="[
'#', 'First', 'Last', 'Email'
]"
:values="[
[1, fake()->firstName(), fake()->lastName(), fake()->safeEmail()],
[2, fake()->firstName(), fake()->lastName(), fake()->safeEmail()],
[3, fake()->firstName(), fake()->lastName(), fake()->safeEmail()]
]"
/>

With "Not Found" Notification

The notfound parameter allows displaying a message when there are no items in the table.

<x-moonshine::table
:columns="[
'#', 'First', 'Last', 'Email'
]"
:notfound="true"
/>

Slots

The table can be formed using slots.

Header 1 {{ fake()->firstName() }} {{ fake()->lastName() }} {{ fake()->safeEmail() }} 2 {{ fake()->firstName() }} {{ fake()->lastName() }} {{ fake()->safeEmail() }} 3 {{ fake()->firstName() }} {{ fake()->lastName() }} {{ fake()->safeEmail() }} Footer
<x-moonshine::table>
<x-slot:thead class="text-center">
<th colspan="4">Header</th>
</x-slot:thead>
<x-slot:tbody>
<tr>
<th>1</th>
<th>{{ fake()->firstName() }}</th>
<th>{{ fake()->lastName() }}</th>
<th>{{ fake()->safeEmail() }}</th>
</tr>
<tr>
<th>2</th>
<th>{{ fake()->firstName() }}</th>
<th>{{ fake()->lastName() }}</th>
<th>{{ fake()->safeEmail() }}</th>
</tr>
<tr>
<th>3</th>
<th>{{ fake()->firstName() }}</th>
<th>{{ fake()->lastName() }}</th>
<th>{{ fake()->safeEmail() }}</th>
</tr>
</x-slot:tbody>
<x-slot:tfoot class="text-center">
<td colspan="4">Footer</td>
</x-slot:tfoot>
</x-moonshine::table>

Styling

For styling the table, there are pre-defined classes that can be used for tr / td.

Available classes:

  • bgc-purple
  • bgc-pink
  • bgc-blue
  • bgc-green
  • bgc-yellow
  • bgc-red
  • bgc-gray
  • bgc-primary
  • bgc-secondary
  • bgc-success
  • bgc-warning
  • bgc-error
  • bgc-info
Header {{ fake()->firstName() }} {{ fake()->lastName() }} {{ fake()->safeEmail() }} {{ fake()->firstName() }} {{ fake()->lastName() }} {{ fake()->safeEmail() }}
<x-moonshine::table>
<x-slot:thead class="bgc-secondary text-center">
<th colspan="3">Header</th>
</x-slot:thead>
<x-slot:tbody>
<tr>
<th class="bgc-pink">{{ fake()->firstName() }}</th>
<th class="bgc-gray">{{ fake()->lastName() }}</th>
<th class="bgc-purple">{{ fake()->safeEmail() }}</th>
</tr>
<tr>
<th class="bgc-green">{{ fake()->firstName() }}</th>
<th class="bgc-red">{{ fake()->lastName() }}</th>
<th class="bgc-yellow">{{ fake()->safeEmail() }}</th>
</tr>
</x-slot:tbody>
</x-moonshine::table>

TableBuilder in MoonShine offers a wide range of capabilities for creating flexible and functional tables in the admin panel.