This documentation explains how to use the @extras/_actions.twig
component in Craft CMS to call web controller actions via JavaScript and display success or error notices.
The Actions
component allows you to make asynchronous requests to Craft CMS web controller actions and handle responses within your JavaScript code. Additionally, the component can display success or error notifications to users based on the outcome of the requests.
To include the Actions
component in your Craft CMS templates, use the following Twig code:
{% include '<pathToTemplates>/_actions.twig' with {...} only %}
Ensure that you include this component only on pages where it is necessary to avoid unnecessary JS/CSS loading.
This method allows you to send a POST request to a Craft CMS web controller action.
Syntax:
window.Actions.postAction(action, data = {}, callback = null, options = {})
Parameters:
action
(String): The route to the controller action (e.g.,mymodule/mycontroller/myaction
).data
(Object): Parameters to be passed to the server. Can be anything that can be converted to valid JSON data.callback
(Function): Function to handle the response. If null, a success notice will be displayed. It can be in the form:() => {...}
data => {...}
(data, status, ok) => {...}
options
(Object, Optional): Additional settings for the request:handleFailuresInCallback
(Boolean, default:false
): Set totrue
if you want to handle400
responses in the callback.timeout
(Number, default:20000
): The number of milliseconds after which the request is aborted. Set to0
for no timeout.logLevel
(String, default:'none'
): Set to'info'
to log responses for debugging purposes.indicatorSelector
(?String, default:null
): A unique CSS selector of the HTML element to provide user feedback while the request is in progress, e.g. a spinner.indicatorClass
(String, default:fetch-request
): The class to apply to the indicator element while the request is in progress.
The callback provided to postAction
is invoked with the following parameters:
function callback(data, status, ok) {
// Handle response
}
The content of the data parameter depends on the content type returned from the server:
application/json:
This is the content type expected from the server, if the controller uses return asSuccess()/asModelSuccess()/asFailure()/asModelFailure()
.
The data parameter will be an object with the following properties:
-
data
(Object): Decoded JSON response from the server.data.message
: Success or error message returned from the server.data.<key>
: Additional data returned from the server.data.<modelName>
: Model data returned from server via->asModelSuccess()
.->asModelFailure()
.data.errors
: Validation errors for models (if any).data.cart
: Commerce only: The cart data returned from Commerce actions likecommerce/cart/update-cart
.
-
status
(Number): HTTP status code (e.g.,200
,400
). -
ok
(Boolean): Indicates whether the request was successful (true
for success,false
for errors).
If the controller uses return $this->asJson(...)
, the data parameter will be the raw response from the server.
In this case, the default failure/notice handling will not work, and you will have to handle errors manually.
text/html:
This is the content type returned from the server if the controller uses return $this->renderTemplate()
.
The data parameter will be a string containing the HTML content of the response. data.message
is not present in this case.
other:
No specific handling is provided for other content types. The data parameter will be what response.text()
returns. Gook luck!
By default, the callback will only be called if the server responds with a status code "200", so that you don't have to care about any errors in your client code.
Errors will be (optionally) logged to console and displayed via an error notice:
- Controller runtime errors
- Connection failure (server not running)
- Non-existing controller actions
- Uncaught exceptions thrown in controller action
- Failed 'require...' constraints (like $this->requireLogin())
- Timed out requests
- Non-JSON responses (that should never happen...)
- Responses with status code 400, like failed controller actions (
return $this->asFailure(...)
), if handleFailuresInCallback = false (default)
Note that errors may be different depending on Craft environment (dev, staging, production, devMode=on/off).
User feedback for successful actions is up to you, you may call Action.notice({type:'success', text: data.message)
inside your callback, or use any other method in order to provide visual feedback.
This example demonstrates sending a POST request and handling a success response.
Controller Action:
return $this->asSuccess('Action completed successfully');
JavaScript:
window.Actions.postAction("mymodule/mycontroller/myaction",
{'id': 1234},
(data) => {
window.Actions.success(data.message);
}
);
If you want to handle failures (status 400
) directly in your callback, set handleFailuresInCallback
to true
in the options.
Controller Action:
if (...someErrorCondition...) {
return $this->asFailure('An error occurred');
}
return $this->asSuccess('Action completed successfully');
JavaScript:
window.Actions.postAction("mymodule/mycontroller/myaction",
{'id': 1234},
(data, status, ok) => {
if (!ok) {
// cleanup...
window.Actions.error(data.message);
return;
}
// Do something with the data
window.Actions.success(data.message);
},
{ handleFailuresInCallback: true }
);
If the controller returns additional data, you can access it in the callback.
Controller Action:
return $this->asSuccess('Success message', ['foo' => 'bar']);
JavaScript:
window.Actions.postAction("mymodule/mycontroller/myaction",
{'id': 1234},
(data) => {
// Do somthing with the data
alert(data.message + ': Foo=' + data.foo);
}
);
HTML (Alpine JS Example)
<div x-html="searchResultsHtml"></div>
JavaScript:
window.Actions.postAction("mymodule/mycontroller/myaction",
{
variables: {
q: this.q,
section: this.section
}
},
(html) => {
this.searchResultsHtml = html;
}
);
Controller Action:
return $this->renderTemplate(
'path/to/your-twig-template.twig',
Craft::$app->getRequest()->getRequiredBodyParam('variables')
);
For security reasons, this script does not support calling twig templates directly without using a controller action.
In case you want to write a generic controller action that renders any template passed as a parameter, make sure:
- to pass the template path as a hashed value
- to validate the path using
Craft::$app->security->validateData($templatePath)
You may want to look into Alpine's Morph plugin for more intelligent DOM updates.
Display an indicator while the request is in progress.
Requires an HTML element that can be queried by the CSS selector specified in indicatorSelector
, where the presence of the class specified in indicatorClass
in some way toggles visibility.
document.querySelector()
is used internally to find the element, so technically the selector can be anything that works with this method, however using an id
is best practice.
JavaScript:
window.Actions.postAction("mymodule/mycontroller/myaction",
{'id': 1234},
(data) => {
// Do somthing with the data
Actions.notice({ type: 'success', text: data.message });
},
{indicatorSelector: '#my-indicator', indicatorClass: 'my-indicator-class'}
);
HTML (Example)
<div id="my-indicator" class="styling-the-indicator">Loading...</div>
CSS (Example)
#my-indicator {
display: none;
}
#my-indicator.my-indicator-class {
display: block;
}
The Actions
component includes a built-in system for displaying notifications to the user.
Triggering Notices:
- Use
Actions.notice({ type: 'success', text: 'Your message here' })
to display a notification.
Types of notices:
success
error
Actions.success(data.message)
Actions.error(data.message)