Components

ActionButton

Inherits MoonShineComponent.

* has the same capabilities.

Basics

When you need to add a button with a specific action, ActionButton comes to the rescue. In MoonShine, they are already used - in forms, tables, and on pages.

ActionButton::make( Closure|string $label, Closure|string $url = '#', ?DataWrapperContract $data = null, )
ActionButton::make(
Closure|string $label,
Closure|string $url = '#',
?DataWrapperContract $data = null,
)
  • label - button text,
  • url - button link URL,
  • data - optional button data, available in closures.
use MoonShine\UI\Components\ActionButton; protected function components(): iterable { return [ ActionButton::make( label: 'Button Title', url: 'https://moonshine-laravel.com', ) ]; }
use MoonShine\UI\Components\ActionButton;
 
protected function components(): iterable
{
return [
ActionButton::make(
label: 'Button Title',
url: 'https://moonshine-laravel.com',
)
];
}

Open in new window

The blank() method allows opening a URL in a new window. The attribute target="_blank" will be added.

ActionButton::make( label: 'Click me', url: '/', )->blank()
ActionButton::make(
label: 'Click me',
url: '/',
)->blank()

Icon

The icon() method allows specifying an icon for the button.

ActionButton::make( label: fn() => 'Click me', url: 'https://moonshine-laravel.com', )->icon('pencil')
ActionButton::make(
label: fn() => 'Click me',
url: 'https://moonshine-laravel.com',
)->icon('pencil')

For more detailed information, refer to the Icons section.

Color

For ActionButton, there is a set of methods to set the button color: primary(), secondary(), warning(), success(), and error().

ActionButton::make( label: 'Click me', url: fn() => 'https://moonshine-laravel.com', )->primary()
ActionButton::make(
label: 'Click me',
url: fn() => 'https://moonshine-laravel.com',
)->primary()

Badge

The badge() method allows adding a badge to the button.

badge(Closure|string|int|float|null $value)
badge(Closure|string|int|float|null $value)
ActionButton::make('Button')->badge(fn() => Comment::count()) // ...
ActionButton::make('Button')->badge(fn() => Comment::count())
// ...

onClick

The onClick method allows executing js code upon clicking:

ActionButton::make( label: 'Click me', url: 'https://moonshine-laravel.com', )->onClick(fn() => "alert('Example')", 'prevent')
ActionButton::make(
label: 'Click me',
url: 'https://moonshine-laravel.com',
)->onClick(fn() => "alert('Example')", 'prevent')

If you need to get data in the onClick method, use the onAfterSet method:

ActionButton::make('Alert') ->onAfterSet(function (?DataWrapperContract $data, ActionButton $button) { return $button->onClick(fn() => 'alert('.$data?->getKey().')'); })
ActionButton::make('Alert')
->onAfterSet(function (?DataWrapperContract $data, ActionButton $button) {
return $button->onClick(fn() => 'alert('.$data?->getKey().')');
})

Basics

To trigger a modal window when the button is clicked, use the inModal() method.

For more detailed information on modal methods, refer to the Modal section.

use MoonShine\UI\Components\Modal; ActionButton::make( label: 'Click me', url: 'https://moonshine-laravel.com', ) ->inModal( title: fn() => 'Modal Window Title', content: fn() => 'Modal Window Content', name: 'my-modal', builder: fn(Modal $modal, ActionButton $ctx) => $modal )
use MoonShine\UI\Components\Modal;
 
ActionButton::make(
label: 'Click me',
url: 'https://moonshine-laravel.com',
)
->inModal(
title: fn() => 'Modal Window Title',
content: fn() => 'Modal Window Content',
name: 'my-modal',
builder: fn(Modal $modal, ActionButton $ctx) => $modal
)
  • title - modal window title,
  • content - modal window content,
  • name - unique modal window name for event dispatching,
  • builder - closure with access to the Modal component.

If you are using multiple similar modal windows, such as in tables for each item, you need to specify a unique name for each:

->inModal( name: static fn (mixed $item, ActionButtonContract $ctx): string => "delete-button-{$ctx->getData()?->getKey()}" )
->inModal(
name: static fn (mixed $item, ActionButtonContract $ctx): string => "delete-button-{$ctx->getData()?->getKey()}"
)

You can also open a modal window using the toggleModal method, and if the ActionButton is inside a modal window, simply openModal.

use MoonShine\UI\Components\ActionButton; use MoonShine\UI\Components\Modal; protected function components(): iterable { return [ Modal::make( 'Title', fn() => 'Content', )->name('my-modal'), ActionButton::make( label: 'Open modal window', )->toggleModal('my-modal') ]; }
use MoonShine\UI\Components\ActionButton;
use MoonShine\UI\Components\Modal;
 
protected function components(): iterable
{
return [
Modal::make(
'Title',
fn() => 'Content',
)->name('my-modal'),
 
ActionButton::make(
label: 'Open modal window',
)->toggleModal('my-modal')
];
}

Async mode

If you need to load content in the modal window asynchronously, enable the async mode on the ActionButton.

protected function components(): iterable { return [ ActionButton::make( label: 'Click me', url: to_page('action_button', fragment: 'doc-content'), ) ->async() ->inModal( title: fn() => 'Modal Window Title', ) ]; }
protected function components(): iterable
{
return [
ActionButton::make(
label: 'Click me',
url: to_page('action_button', fragment: 'doc-content'),
)
->async()
->inModal(
title: fn() => 'Modal Window Title',
)
];
}

You can find out about Fragment in the "Components" section.

Confirmation

The withConfirm() method allows creating a button with action confirmation.

ActionButton::make( label: 'Click me', url: 'https://moonshine-laravel.com', ) ->withConfirm( title: 'Confirmation Modal Window Title', content: 'Confirmation Modal Window Content', button: 'Confirmation Modal Window Button', // optionally - additional form fields fields: null, method: HttpMethod::POST, // optionally - closure with FormBuilder formBuilder: null, // optionally - closure with Modal modalBuilder: null, name: 'my-modal', )
ActionButton::make(
label: 'Click me',
url: 'https://moonshine-laravel.com',
)
->withConfirm(
title: 'Confirmation Modal Window Title',
content: 'Confirmation Modal Window Content',
button: 'Confirmation Modal Window Button',
// optionally - additional form fields
fields: null,
method: HttpMethod::POST,
// optionally - closure with FormBuilder
formBuilder: null,
// optionally - closure with Modal
modalBuilder: null,
name: 'my-modal',
)

If you are using multiple similar modal windows, such as in tables for each item, you need to specify a unique name for each:

->inModal( name: static fn (mixed $item, ActionButtonContract $ctx): string => "delete-button-{$ctx->getData()?->getKey()}" )
->inModal(
name: static fn (mixed $item, ActionButtonContract $ctx): string => "delete-button-{$ctx->getData()?->getKey()}"
)

Offcanvas

To trigger an offcanvas panel when clicking the button, use the inOffCanvas() method.

use MoonShine\UI\Components\OffCanvas; protected function components(): iterable { return [ ActionButton::make( label: 'Click me', url: 'https://moonshine-laravel.com', ) ->inOffCanvas( title: fn() => 'Offcanvas Title', content: fn() => 'Content', name: false, builder: fn(OffCanvas $offCanvas, ActionButton $ctx) => $offCanvas->left() // optionally - necessary for components to be available for searching in the system, as content is just HTML components: [] ) ]; }
use MoonShine\UI\Components\OffCanvas;
 
protected function components(): iterable
{
return [
ActionButton::make(
label: 'Click me',
url: 'https://moonshine-laravel.com',
)
->inOffCanvas(
title: fn() => 'Offcanvas Title',
content: fn() => 'Content',
name: false,
builder: fn(OffCanvas $offCanvas, ActionButton $ctx) => $offCanvas->left()
// optionally - necessary for components to be available for searching in the system, as content is just HTML
components: []
)
];
}

Grouping

If you need to organize logic with multiple ActionButton, with some of them needing to be hidden or displayed in a dropdown menu, use the ActionGroup component.

use MoonShine\UI\Components\ActionGroup; protected function components(): iterable { return [ ActionGroup::make([ ActionButton::make('Button 1', '/')->canSee(fn() => false), ActionButton::make('Button 2', '/', $model)->canSee(fn($model) => $model->active) ]) ]; }
use MoonShine\UI\Components\ActionGroup;
 
protected function components(): iterable
{
return [
ActionGroup::make([
ActionButton::make('Button 1', '/')->canSee(fn() => false),
ActionButton::make('Button 2', '/', $model)->canSee(fn($model) => $model->active)
])
];
}

Display

With ActionGroup, you can also change the display of buttons, showing them inline or in a dropdown for space-saving.

use MoonShine\UI\Components\ActionGroup; protected function components(): iterable { return [ ActionGroup::make([ ActionButton::make('Button 1', '/')->showInLine(), ActionButton::make('Button 2', '/')->showInDropdown() ]) ]; }
use MoonShine\UI\Components\ActionGroup;
 
protected function components(): iterable
{
return [
ActionGroup::make([
ActionButton::make('Button 1', '/')->showInLine(),
ActionButton::make('Button 2', '/')->showInDropdown()
])
];
}

Bulk actions

The bulk() method allows creating a bulk action button for ModelResource.

protected function indexButtons(): ListOf { return parent::indexButtons()->add(ActionButton::make('Link', '/endpoint')->bulk()); }
protected function indexButtons(): ListOf
{
return parent::indexButtons()->add(ActionButton::make('Link', '/endpoint')->bulk());
}

The bulk() method is only used within ModelResource.

Async mode

The async() method allows implementing asynchronous functionality for ActionButton.

async( HttpMethod $method = HttpMethod::GET, ?string $selector = null, array $events = [], ?AsyncCallback $callback = null )
async(
HttpMethod $method = HttpMethod::GET,
?string $selector = null,
array $events = [],
?AsyncCallback $callback = null
)
  • $method - the method of the asynchronous request,
  • $selector - the selector of the element whose content will change according to the response,
  • $events - events that will be triggered after a successful request,
  • $callback - js callback function after receiving the response.

You can learn more about Events in the "Frontend" section.

You can learn more about Callback in the "Frontend" section.

protected function components(): iterable { return [ ActionButton::make( 'Click me', '/endpoint' ) ->async() ]; }
protected function components(): iterable
{
return [
ActionButton::make(
'Click me',
'/endpoint'
)
->async()
];
}

Notifications

If you need to display a notification or redirect after clicking, simply implement a json response according to the structure:

{message: 'Toast', messageType: 'success', redirect: '/url'}
{message: 'Toast', messageType: 'success', redirect: '/url'}

The redirect parameter is optional.

HTML content

If you need to replace an HTML area upon clicking, you can return HTML content or json with the html key in the response:

{html: 'Html content'}
{html: 'Html content'}
protected function components(): iterable { return [ ActionButton::make( 'Click me', '/endpoint' ) ->async(selector: '#my-selector') ]; }
protected function components(): iterable
{
return [
ActionButton::make(
'Click me',
'/endpoint'
)
->async(selector: '#my-selector')
];
}

Events

After a successful request, you can trigger events:

protected function components(): iterable { return [ ActionButton::make( 'Click me', '/endpoint' ) ->async(events: [AlpineJs::event(JsEvent::TABLE_UPDATED, $this->getListComponentName())]) ]; }
protected function components(): iterable
{
return [
ActionButton::make(
'Click me',
'/endpoint'
)
->async(events: [AlpineJs::event(JsEvent::TABLE_UPDATED, $this->getListComponentName())])
];
}

For the JsEvent::TABLE_UPDATED event to work, the table must have async mode enabled.

Callback

If you need to handle the response differently, you must implement a handler function and specify it in the async() method.

protected function components(): iterable { return [ ActionButton::make( 'Click me', '/endpoint' ) ->async(callback: AsyncCallback::with(responseHandler: 'afterResponseFunction')) ]; }
protected function components(): iterable
{
return [
ActionButton::make(
'Click me',
'/endpoint'
)
->async(callback: AsyncCallback::with(responseHandler: 'afterResponseFunction'))
];
}
document.addEventListener("moonshine:init", () => { MoonShine.onCallback('myFunction', function(response, element, events, component) { if(response.confirmed === true) { component.$dispatch('toast', {type: 'success', text: 'Success'}) } else { component.$dispatch('toast', {type: 'error', text: 'Error'}) } }) })
document.addEventListener("moonshine:init", () => {
MoonShine.onCallback('myFunction', function(response, element, events, component) {
if(response.confirmed === true) {
component.$dispatch('toast', {type: 'success', text: 'Success'})
} else {
component.$dispatch('toast', {type: 'error', text: 'Error'})
}
})
})

Details in the Js section

Method calls

method() allows specifying a method name in the resource and call it asynchronously when the ActionButton is clicked, without the need to create additional controllers.

method( string $method, array|Closure $params = [], ?string $message = null, ?string $selector = null, array $events = [], ?AsyncCallback $callback = null, ?PageContract $page = null, ?ResourceContract $resource = null )
method(
string $method,
array|Closure $params = [],
?string $message = null,
?string $selector = null,
array $events = [],
?AsyncCallback $callback = null,
?PageContract $page = null,
?ResourceContract $resource = null
)
  • $method - the method name,
  • $params - optionally - parameters for the request,
  • $message - optionally - message upon successful execution,
  • $selector - optionally - the selector of the element whose content will change,
  • $events - optionally - events that will be triggered after a successful query,
  • $callback - optionally - js callback function after receiving the response,
  • $page - optionally - the page containing the method (if the button is outside the page and resource),
  • $resource - optionally - the resource containing the method (if the button is outside the resource).
protected function components(): iterable { return [ ActionButton::make('Click me') ->method('updateSomething'), ]; }
protected function components(): iterable
{
return [
ActionButton::make('Click me')
->method('updateSomething'),
];
}
// With notification public function updateSomething(MoonShineRequest $request): MoonShineJsonResponse { // $request->getResource(); // $request->getResource()->getItem(); // $request->getPage(); return MoonShineJsonResponse::make()->toast('My message', ToastType::SUCCESS); } // Redirect public function updateSomething(MoonShineRequest $request): MoonShineJsonResponse { return MoonShineJsonResponse::make()->redirect('/'); } // Redirect public function updateSomething(MoonShineRequest $request): RedirectResponse { return back(); } // Exception public function updateSomething(MoonShineRequest $request): void { throw new \Exception('My message'); } // Custom JSON response public function updateSomething(MoonShineRequest $request) { return MoonShineJsonResponse::make()->html('Content'); }
// With notification
public function updateSomething(MoonShineRequest $request): MoonShineJsonResponse
{
// $request->getResource();
// $request->getResource()->getItem();
// $request->getPage();
 
return MoonShineJsonResponse::make()->toast('My message', ToastType::SUCCESS);
}
 
// Redirect
public function updateSomething(MoonShineRequest $request): MoonShineJsonResponse
{
return MoonShineJsonResponse::make()->redirect('/');
}
 
// Redirect
public function updateSomething(MoonShineRequest $request): RedirectResponse
{
return back();
}
 
// Exception
public function updateSomething(MoonShineRequest $request): void
{
throw new \Exception('My message');
}
 
// Custom JSON response
public function updateSomething(MoonShineRequest $request)
{
return MoonShineJsonResponse::make()->html('Content');
}

Methods called via ActionButton in the resource must be public!

For access to data from the request, you must pass them as parameters.

Passing the current item

If the request contains resourceItem, you can access the current item in the resource through the getItem() method.

  • When the data contains a model and the button is created in the indexButtons() method or detailButtons or formButtons of TableBuilder, CardsBuilder or FormBuilder, it automatically gets the data, and the parameters will contain resourceItem.
  • When the button is on the form page of ModelResource, you can pass the id of the current item.
ActionButton::make('Click me') ->method( 'updateSomething', params: ['resourceItem' => $this->getResource()->getItemID()] )
ActionButton::make('Click me')
->method(
'updateSomething',
params: ['resourceItem' => $this->getResource()->getItemID()]
)
  • When the button is in the index table of ModelResource, you need to use a closure.
ActionButton::make('Click me') ->method( 'updateSomething', params: fn(Model $item) => ['resourceItem' => $item->getKey()] )
ActionButton::make('Click me')
->method(
'updateSomething',
params: fn(Model $item) => ['resourceItem' => $item->getKey()]
)

Field values

The withSelectorsParams() method allows passing field values with the request using element selectors.

ActionButton::make('Async method') ->method('updateSomething') ->withSelectorsParams([ 'alias' => '[data-column="title"]', 'slug' => '#slug' ]),
ActionButton::make('Async method')
->method('updateSomething')
->withSelectorsParams([
'alias' => '[data-column="title"]',
'slug' => '#slug'
]),
use MoonShine\Laravel\Http\Responses\MoonShineJsonResponse; use MoonShine\Laravel\MoonShineRequest; public function updateSomething(MoonShineRequest $request): MoonShineJsonResponse { return MoonShineJsonResponse::make() ->toast($request->get('slug', 'Error')); }
use MoonShine\Laravel\Http\Responses\MoonShineJsonResponse;
use MoonShine\Laravel\MoonShineRequest;
 
public function updateSomething(MoonShineRequest $request): MoonShineJsonResponse
{
return MoonShineJsonResponse::make()
->toast($request->get('slug', 'Error'));
}

When using the withSelectorsParams() method, requests will be sent via POST.

Download

The invoked method can return BinaryFileResponse, allowing a file download.

ActionButton::make('Download')->method('download')
ActionButton::make('Download')->method('download')
public function download(): BinaryFileResponse { // ... return response()->download($file); }
public function download(): BinaryFileResponse
{
// ...
 
return response()->download($file);
}

Event dispatching

To dispatch JavaScript events, you can use the dispatchEvent() method.

dispatchEvent(array|string $events)
dispatchEvent(array|string $events)
ActionButton::make('Refresh') ->dispatchEvent(AlpineJs::event(JsEvent::TABLE_UPDATED, 'index-table')),
ActionButton::make('Refresh')
->dispatchEvent(AlpineJs::event(JsEvent::TABLE_UPDATED, 'index-table')),

By default, when an event is triggered with a request, all query parameters (e.g., ?param=value) from the url (specified when creating the ActionButton) will be sent.

You can exclude unnecessary ones through the exclude parameter:

->dispatchEvent( AlpineJs::event(JsEvent::TABLE_UPDATED, 'index-table'), exclude: ['something'] )
->dispatchEvent(
AlpineJs::event(JsEvent::TABLE_UPDATED, 'index-table'),
exclude: ['something']
)

You can also completely exclude the sending of withoutPayload:

->dispatchEvent( AlpineJs::event(JsEvent::TABLE_UPDATED, 'index-table'), withoutPayload: true )
->dispatchEvent(
AlpineJs::event(JsEvent::TABLE_UPDATED, 'index-table'),
withoutPayload: true
)

URL query parameters

You can include the current request URL parameters (e.g., ?param=value) in the request:

->withQueryParams()
->withQueryParams()

Data filling

When working with ModelResource, the action buttons ActionButton are usually automatically filled with the necessary data. This process happens "under the hood" using the setData method. Let’s examine this mechanism in more detail.

ActionButton::make('Button')->setData(?DataWrapperContract $data = null)
ActionButton::make('Button')->setData(?DataWrapperContract $data = null)

For more information about DataWrapperContract, read the TypeCasts section.

Methods with callbacks before and after filling the button are also available.

ActionButton::make('Button')->onBeforeSet(fn(?DataWrapperContract $data, ActionButton $ctx) => $data)
ActionButton::make('Button')->onBeforeSet(fn(?DataWrapperContract $data, ActionButton $ctx) => $data)
ActionButton::make('Button')->onAfterSet(function(?DataWrapperContract $data, ActionButton $ctx): void { // logic })
ActionButton::make('Button')->onAfterSet(function(?DataWrapperContract $data, ActionButton $ctx): void {
// logic
})