breezy_utility-1.0.x-dev/src/Form/BreezyUtilityAjaxFormTrait.php
src/Form/BreezyUtilityAjaxFormTrait.php
<?php
namespace Drupal\breezy_utility\Form;
use Drupal\breezy_utility\Ajax\BreezyUtilityCloseDialogCommand;
use Drupal\breezy_utility\Ajax\BreezyUtilityRefreshCommand;
use Drupal\breezy_utility\Utility\BreezyUtilityElementHelper;
use Drupal\Component\Utility\Html;
use Drupal\Core\Ajax\AjaxResponse;
use Drupal\Core\Ajax\CloseDialogCommand;
use Drupal\Core\Ajax\HtmlCommand;
use Drupal\Core\EventSubscriber\MainContentViewSubscriber;
use Drupal\Core\Form\FormStateInterface;
use Drupal\Core\Render\Element;
use Drupal\Core\Template\Attribute;
use Drupal\Core\Url;
use Symfony\Component\HttpFoundation\RedirectResponse;
/**
* Trait form ajax support.
*/
trait BreezyUtilityAjaxFormTrait {
/**
* {@inheritdoc}
*/
protected function isAjax() {
return $this->isDialog();
}
/**
* Cancel form #ajax callback.
*
* @param array $form
* An associative array containing the structure of the form.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*
* @return \Drupal\Core\Ajax\AjaxResponse
* An Ajax response that display validation error messages or redirects
* to a URL.
*/
abstract public function cancelAjaxForm(array &$form, FormStateInterface $form_state);
/**
* Get default ajax callback settings.
*
* @return array
* An associative array containing default ajax callback settings.
*/
protected function getDefaultAjaxSettings() {
return [
'disable-refocus' => TRUE,
'effect' => 'fade',
'speed' => 1000,
'progress' => [
'type' => 'throbber',
'message' => '',
],
];
}
/**
* Is the current request for an Ajax modal/dialog.
*
* @return bool
* TRUE if the current request is for an Ajax modal/dialog.
*/
protected function isDialog() {
$wrapper_format = $this->getRequest()
->get(MainContentViewSubscriber::WRAPPER_FORMAT);
return (in_array($wrapper_format, [
'drupal_ajax',
'drupal_modal',
'drupal_dialog',
'drupal_dialog.off_canvas',
])) ? TRUE : FALSE;
}
/**
* Is the current request for an off canvas dialog.
*
* @return bool
* TRUE if the current request is for an off canvas dialog.
*/
protected function isOffCanvasDialog() {
$wrapper_format = $this->getRequest()
->get(MainContentViewSubscriber::WRAPPER_FORMAT);
return (in_array($wrapper_format, ['drupal_dialog.off_canvas'])) ? TRUE : FALSE;
}
/**
* Get the form's Ajax wrapper id.
*
* @return string
* The form's Ajax wrapper id.
*/
protected function getWrapperId() {
$form_id = (method_exists($this, 'getBaseFormId') ? $this->getBaseFormId() : $this->getFormId());
return Html::getId($form_id . '-ajax');
}
/**
* Add Ajax support to a form.
*
* @param array $form
* An associative array containing the structure of the form.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
* @param array $settings
* Ajax settings.
*
* @return array
* The form with Ajax callbacks.
*/
protected function buildAjaxForm(array &$form, FormStateInterface $form_state, array $settings = []) {
if (!$this->isAjax()) {
return $form;
}
// Apply default settings.
$settings += $this->getDefaultAjaxSettings();
// Add Ajax callback to all submit buttons.
foreach (Element::children($form) as $element_key) {
if (!BreezyUtilityElementHelper::isType($form[$element_key], 'actions')) {
continue;
}
$actions = &$form[$element_key];
foreach (Element::children($actions) as $action_key) {
if (BreezyUtilityElementHelper::isType($actions[$action_key], 'submit') && !isset($actions[$action_key]['#ajax'])) {
$actions[$action_key]['#ajax'] = [
'callback' => '::submitAjaxForm',
'event' => 'click',
] + $settings;
}
}
}
// Add Ajax wrapper with wrapper content bookmark around the form.
// @see Drupal.AjaxCommands.prototype.breezyScrollTop
$wrapper_id = $this->getWrapperId();
$wrapper_attributes = [];
$wrapper_attributes['id'] = $wrapper_id;
$wrapper_attributes['class'] = ['breezy-ajax-form-wrapper'];
if (isset($settings['effect'])) {
$wrapper_attributes['data-effect'] = $settings['effect'];
}
if (isset($settings['progress']['type'])) {
$wrapper_attributes['data-progress-type'] = $settings['progress']['type'];
}
$wrapper_attributes = new Attribute($wrapper_attributes);
$form['#form_wrapper_id'] = $wrapper_id;
$form += ['#prefix' => '', '#suffix' => ''];
$form['#prefix'] .= '<span id="' . $wrapper_id . '-content"></span>';
$form['#prefix'] .= '<div' . $wrapper_attributes . '>';
$form['#suffix'] = '</div>' . $form['#suffix'];
// Add Ajax library which contains 'Scroll to top' Ajax command and
// Ajax callback for confirmation back to link.
$form['#attached']['library'][] = 'breezy_utility/breezy_utility.ajax';
// Add validate Ajax form.
$form['#validate'][] = '::validateAjaxForm';
return $form;
}
/**
* Validation form for #ajax callback.
*
* @param array $form
* The form array.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*/
public function validateAjaxForm(array &$form, FormStateInterface $form_state) {
}
/**
* Submit form #ajax callback.
*
* @param array $form
* An associative array containing the structure of the form.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*
* @return \Drupal\Core\Ajax\AjaxResponse
* An Ajax response that display validation error messages or redirects
* to a URL
*/
public function submitAjaxForm(array &$form, FormStateInterface $form_state) {
if ($form_state->hasAnyErrors()) {
// Display validation errors and scroll to the top of the page.
$response = $this->replaceForm($form, $form_state);
}
elseif ($form_state->getResponse() instanceof AjaxResponse) {
// Allow developers via form_alter hooks to set their own Ajax response.
// The custom Ajax response could be used to close modals and refresh
// selected regions and blocks on the page.
$response = $form_state->getResponse();
}
elseif ($form_state->isRebuilding()) {
// Rebuild form.
$response = $this->replaceForm($form, $form_state);
}
elseif ($redirect_url = $this->getFormStateRedirectUrl($form_state)) {
// Redirect to URL.
$response = $this->createAjaxResponse($form, $form_state);
$response->addCommand(new BreezyUtilityCloseDialogCommand());
$response->addCommand(new BreezyUtilityRefreshCommand($redirect_url));
}
else {
$response = $this->cancelAjaxForm($form, $form_state);
}
return $response;
}
/**
* Get redirect URL from the form's state.
*
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*
* @return bool|\Drupal\Core\GeneratedUrl|string
* The redirect URL or FALSE if the form is not redirecting.
*/
protected function getFormStateRedirectUrl(FormStateInterface $form_state) {
// Always check the ?destination which is used by the off-canvas/system
// tray.
if ($this->getRequest()->get('destination')) {
$destination = $this->getRedirectDestination()->get();
return (strpos($destination, $destination) === 0) ? $destination : base_path() . $destination;
}
// ISSUE:
// Can't get the redirect URL from the form state during an AJAX submission.
//
// WORKAROUND:
// Re-enable redirect, grab the URL, and then disable again.
$no_redirect = $form_state->isRedirectDisabled();
$form_state->disableRedirect(FALSE);
$redirect = $form_state->getResponse() ?: $form_state->getRedirect();
$form_state->disableRedirect($no_redirect);
if ($redirect instanceof RedirectResponse) {
return $redirect->getTargetUrl();
}
elseif ($redirect instanceof Url) {
return $redirect->setAbsolute()->toString();
}
else {
return FALSE;
}
}
/**
* Empty submit callback.
*
* Only used to have the submit button to use an #ajax submit callback.
*
* This allows modal dialog to using ::submitCallback to validate and submit
* the form via one ajax request.
*/
public function noSubmit(array &$form, FormStateInterface $form_state) {
// Do nothing.
}
/**
* Create an AjaxResponse object.
*
* @param array $form
* An associative array containing the structure of the form.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*
* @return \Drupal\Core\Ajax\AjaxResponse
* An AjaxResponse object
*/
protected function createAjaxResponse(array $form, FormStateInterface $form_state) {
return new AjaxResponse();
}
/**
* Replace form via an Ajax response.
*
* @param array $form
* An associative array containing the structure of the form.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*
* @return \Drupal\Core\Ajax\AjaxResponse
* An Ajax response that replaces a form.
*/
protected function replaceForm(array $form, FormStateInterface $form_state) {
// Display messages first by prefixing it the form and setting its weight
// to -1000.
$form = [
'status_messages' => [
'#type' => 'status_messages',
'#weight' => -1000,
],
] + $form;
// Remove wrapper.
unset($form['#prefix'], $form['#suffix']);
$response = $this->createAjaxResponse($form, $form_state);
$response->addCommand(new HtmlCommand('#' . $this->getWrapperId(), $form));
return $response;
}
/**
* Close dialog.
*
* @param array $form
* An associative array containing the structure of the form.
* @param \Drupal\Core\Form\FormStateInterface $form_state
* The current state of the form.
*
* @return bool|\Drupal\Core\Ajax\AjaxResponse
* An AJAX response that display validation error messages.
*/
public function closeDialog(array &$form, FormStateInterface $form_state) {
$response = new AjaxResponse();
$response->addCommand(new CloseDialogCommand());
return $response;
}
}
