diff --git a/core/misc/dialog/dialog.ajax.js b/core/misc/dialog/dialog.ajax.js index dd62949273539a8f784a500488f849534d36302e..c44c031b5154d5c7a8a4781130e3f0fc715fd25e 100644 --- a/core/misc/dialog/dialog.ajax.js +++ b/core/misc/dialog/dialog.ajax.js @@ -3,7 +3,7 @@ * Extends the Drupal AJAX functionality to integrate the dialog API. */ -(function ($, Drupal) { +(function ($, Drupal, { focusable }) { /** * Initialize dialogs for Ajax purposes. * @@ -46,6 +46,27 @@ // Overwrite the close method to remove the dialog on closing. settings.dialog.close = function (event, ...args) { originalClose.apply(settings.dialog, [event, ...args]); + // Check if the opener element is inside an AJAX container. + const $element = $(event.target); + const ajaxContainer = $element.data('uiDialog') + ? $element + .data('uiDialog') + .opener.closest('[data-drupal-ajax-container]') + : []; + + // If the opener element was in an ajax container, and focus is on the + // body element, we can assume focus was lost. To recover, focus is moved + // to the first focusable element in the container. + if ( + ajaxContainer.length && + (document.activeElement === document.body || + $(document.activeElement).not(':visible')) + ) { + const focusableChildren = focusable(ajaxContainer[0]); + if (focusableChildren.length > 0) { + focusableChildren[0].focus(); + } + } $(event.target).remove(); }; }, @@ -246,4 +267,4 @@ $(window).on('dialog:beforeclose', (e, dialog, $element) => { $element.off('.dialog'); }); -})(jQuery, Drupal); +})(jQuery, Drupal, window.tabbable); diff --git a/core/modules/block/tests/src/Functional/Views/DisplayBlockTest.php b/core/modules/block/tests/src/Functional/Views/DisplayBlockTest.php index 9c02828c99479a869bd1bfb6aa093d79bceaf2e3..55f54f822d49b9927dab35affc9e24ac60fe7f44 100644 --- a/core/modules/block/tests/src/Functional/Views/DisplayBlockTest.php +++ b/core/modules/block/tests/src/Functional/Views/DisplayBlockTest.php @@ -393,8 +393,8 @@ public function testBlockContextualLinks() { $cached_id_token = Crypt::hmacBase64($cached_id, Settings::getHashSalt() . $this->container->get('private_key')->get()); // @see \Drupal\contextual\Tests\ContextualDynamicContextTest:assertContextualLinkPlaceHolder() // Check existence of the contextual link placeholders. - $this->assertSession()->responseContains('<div' . new Attribute(['data-contextual-id' => $id, 'data-contextual-token' => $id_token]) . '></div>'); - $this->assertSession()->responseContains('<div' . new Attribute(['data-contextual-id' => $cached_id, 'data-contextual-token' => $cached_id_token]) . '></div>'); + $this->assertSession()->responseContains('<div' . new Attribute(['data-contextual-id' => $id, 'data-contextual-token' => $id_token, 'data-drupal-ajax-container' => '']) . '></div>'); + $this->assertSession()->responseContains('<div' . new Attribute(['data-contextual-id' => $cached_id, 'data-contextual-token' => $cached_id_token, 'data-drupal-ajax-container' => '']) . '></div>'); // Get server-rendered contextual links. // @see \Drupal\contextual\Tests\ContextualDynamicContextTest:renderContextualLinks() diff --git a/core/modules/contextual/src/Element/ContextualLinksPlaceholder.php b/core/modules/contextual/src/Element/ContextualLinksPlaceholder.php index 40b82058ea459cc29bde88bae960bc79832b51e3..b59da5b4ace02df0b63c6c83994f16fc6c5023c1 100644 --- a/core/modules/contextual/src/Element/ContextualLinksPlaceholder.php +++ b/core/modules/contextual/src/Element/ContextualLinksPlaceholder.php @@ -49,6 +49,7 @@ public static function preRenderPlaceholder(array $element) { $attribute = new Attribute([ 'data-contextual-id' => $element['#id'], 'data-contextual-token' => $token, + 'data-drupal-ajax-container' => '', ]); $element['#markup'] = new FormattableMarkup('<div@attributes></div>', ['@attributes' => $attribute]); diff --git a/core/modules/contextual/tests/src/FunctionalJavascript/ContextualLinksTest.php b/core/modules/contextual/tests/src/FunctionalJavascript/ContextualLinksTest.php index 8bcc8a301594c286765fd5e64098b7ec2befa893..5de661be23e4d5d617b6ea60ceb431bcc86cec21 100644 --- a/core/modules/contextual/tests/src/FunctionalJavascript/ContextualLinksTest.php +++ b/core/modules/contextual/tests/src/FunctionalJavascript/ContextualLinksTest.php @@ -88,6 +88,14 @@ public function testContextualLinksClick() { $this->clickContextualLink('#block-branding', 'Test Link with Ajax'); $this->assertNotEmpty($this->assertSession()->waitForElementVisible('css', '#drupal-modal')); $this->assertSession()->elementContains('css', '#drupal-modal', 'Everything is contextual!'); + $this->getSession()->getPage()->pressButton('Close'); + $this->assertSession()->assertNoElementAfterWait('css', 'ui.dialog'); + + // When the dialog is closed, the opening contextual link is now inside a + // collapsed container, so focus should be routed to the contextual link + // toggle button. + $this->assertJsCondition('document.activeElement === document.querySelector("#block-branding button.trigger")'); + // Check to make sure that page was not reloaded. $this->assertSession()->pageTextContains($current_page_string); diff --git a/core/modules/system/tests/modules/dialog_renderer_test/dialog_renderer_test.routing.yml b/core/modules/system/tests/modules/dialog_renderer_test/dialog_renderer_test.routing.yml index c135824810b5f5653c5ccb0e3acd34af2d9cbc74..bbcc9c7f2d8bf132a2f0e9e3c2f4699587b409a6 100644 --- a/core/modules/system/tests/modules/dialog_renderer_test/dialog_renderer_test.routing.yml +++ b/core/modules/system/tests/modules/dialog_renderer_test/dialog_renderer_test.routing.yml @@ -29,3 +29,11 @@ dialog_renderer_test.modal_content_input: _title: 'Thing 3' requirements: _access: 'TRUE' + +dialog_renderer_test.collapsed_opener: + path: '/dialog_renderer-collapsed-opener' + defaults: + _controller: '\Drupal\dialog_renderer_test\Controller\TestController::collapsedOpener' + _title: 'Collapsed Openers' + requirements: + _access: 'TRUE' diff --git a/core/modules/system/tests/modules/dialog_renderer_test/src/Controller/TestController.php b/core/modules/system/tests/modules/dialog_renderer_test/src/Controller/TestController.php index 469e35a187de48c248a0ac3f1e0516e9411b0ee8..4dae862fb95d80857b12285823448e4d9a909669 100644 --- a/core/modules/system/tests/modules/dialog_renderer_test/src/Controller/TestController.php +++ b/core/modules/system/tests/modules/dialog_renderer_test/src/Controller/TestController.php @@ -189,4 +189,39 @@ public function linksDisplay() { ]; } + /** + * Displays a dropbutton with a link that opens in a modal dialog. + * + * @return array + * Render array with links. + */ + public function collapsedOpener() { + return [ + '#markup' => '<h2>Honk</h2>', + 'dropbutton' => [ + '#type' => 'dropbutton', + '#dropbutton_type' => 'small', + '#links' => [ + 'front' => [ + 'title' => 'front!', + 'url' => Url::fromRoute('<front>'), + ], + 'in a dropbutton' => [ + 'title' => 'inside a dropbutton', + 'url' => Url::fromRoute('dialog_renderer_test.modal_content'), + 'attributes' => [ + 'class' => ['use-ajax'], + 'data-dialog-type' => 'modal', + ], + ], + ], + ], + '#attached' => [ + 'library' => [ + 'core/drupal.ajax', + ], + ], + ]; + } + } diff --git a/core/modules/system/tests/src/FunctionalJavascript/ModalRendererTest.php b/core/modules/system/tests/src/FunctionalJavascript/ModalRendererTest.php index 0760f7042b6da48ab7a28c5956e4561d08aae9b5..2cec0b84e2f72476eec9fa35e74f9963e4a297ce 100644 --- a/core/modules/system/tests/src/FunctionalJavascript/ModalRendererTest.php +++ b/core/modules/system/tests/src/FunctionalJavascript/ModalRendererTest.php @@ -76,4 +76,31 @@ public function testModalRenderer() { $this->assertJsCondition('document.activeElement === document.querySelector(".ui-dialog .form-text")'); } + /** + * Confirm focus management of a dialog openers in a dropbutton. + */ + public function testOpenerInDropbutton() { + $assert_session = $this->assertSession(); + $page = $this->getSession()->getPage(); + + $this->drupalGet('dialog_renderer-collapsed-opener'); + + // Open a modal using a link inside a dropbutton. + $page->find('css', '.dropbutton-toggle button')->click(); + $modal_link = $assert_session->waitForElementVisible('css', '.secondary-action a'); + $modal_link->click(); + $assert_session->waitForElementVisible('css', '.ui-dialog'); + $assert_session->assertVisibleInViewport('css', '.ui-dialog .ui-dialog-content'); + $page->pressButton('Close'); + + // When the dialog "closes" it is still present, so wait on it switching to + // `display: none;`. + $assert_session->waitForElement('css', '.ui-dialog[style*="display: none;"]'); + + // Confirm that when the modal closes, focus is moved to the first visible + // and focusable item in the contextual link container, because the original + // opener is not available. + $this->assertJsCondition('document.activeElement === document.querySelector(".dropbutton-action a")'); + } + }