Ajax in the backend
An Ajax endpoint in the TYPO3 backend is usually implemented as a method in a
regular controller. The method receives a request object implementing the
\Psr\
, which allows to access all
aspects of the requests and returns an appropriate response in a normalized way.
This approach is standardized as PSR-7.
Create a controller
By convention, a controller is placed within the extension's Controller/
directory, optionally in a subdirectory. To have such controller, create a new
Example
in Classes/
inside your extension.
The controller needs not that much logic right now. We create a method called
do
which will be our Ajax endpoint.
<?php
declare(strict_types=1);
namespace MyVendor\MyExtension\Controller;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
final class ExampleController
{
public function doSomethingAction(ServerRequestInterface $request): ResponseInterface
{
// TODO: return ResponseInterface
}
}
In its current state, the method does nothing yet. We can add a very generic
handling that exponentiates an incoming number by 2. The incoming value will be
passed as a query string argument named input
.
<?php
declare(strict_types=1);
namespace MyVendor\MyExtension\Controller;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
final class ExampleController
{
public function doSomethingAction(ServerRequestInterface $request): ResponseInterface
{
$input = $request->getQueryParams()['input']
?? throw new \InvalidArgumentException(
'Please provide a number',
1580585107,
);
$result = $input ** 2;
// TODO: return ResponseInterface
}
}
Note
This is a really simple example. Something like this should not be used in production, as such a feature is available via JavaScript as well.
We have computed our result by using the exponentiation operator, but we do nothing with it yet. It is time to build a proper response:
<?php
declare(strict_types=1);
namespace MyVendor\MyExtension\Controller;
use Psr\Http\Message\ResponseFactoryInterface;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
final class ExampleController
{
public function __construct(
private readonly ResponseFactoryInterface $responseFactory,
) {}
public function doSomethingAction(ServerRequestInterface $request): ResponseInterface
{
$input = $request->getQueryParams()['input']
?? throw new \InvalidArgumentException(
'Please provide a number',
1580585107,
);
$result = $input ** 2;
$response = $this->responseFactory->createResponse()
->withHeader('Content-Type', 'application/json; charset=utf-8');
$response->getBody()->write(
json_encode(['result' => $result], JSON_THROW_ON_ERROR),
);
return $response;
}
}
Register the endpoint
The endpoint must be registered as route. Create a
file called Configuration/
in your extension. The
file basically just returns an array of route definitions. Every route in this
file will be exposed to JavaScript automatically. Let us register our endpoint
now:
<?php
use MyVendor\MyExtension\Controller\ExampleController;
return [
'myextension_example_dosomething' => [
'path' => '/my-extension/example/do-something',
'target' => ExampleController::class . '::doSomethingAction',
],
];
The naming of the key myextension_
and path
/my-
are up to you, but should contain the
extension name, controller name and action name to avoid potential conflicts
with other existing routes.
Attention
Flushing caches is mandatory after modifying any route definition.
Use in Ajax
Since the route is registered in Ajax
it is exposed to
JavaScript now and stored in the global TYPO3.
object
identified by the used key in the registration. In this example it is
TYPO3.
.
Now you are free to use the endpoint in any of your Ajax calls. To complete this example, we will ask the server to compute our input and write the result into the console.
import AjaxRequest from "@typo3/core/ajax/ajax-request.js";
// Generate a random number between 1 and 32
const randomNumber = Math.ceil(Math.random() * 32);
new AjaxRequest(TYPO3.settings.ajaxUrls.myextension_example_dosomething)
.withQueryArguments({input: randomNumber})
.get()
.then(async function (response) {
const resolved = await response.resolve();
console.log(resolved.result);
});
See also