From f4a1c0201ed7d87502bbcf3eb10beccecae67f10 Mon Sep 17 00:00:00 2001 From: Adam G-H <32250-phenaproxima@users.noreply.drupalcode.org> Date: Fri, 21 Feb 2025 19:11:45 +0000 Subject: [PATCH] Issue #3508090 by phenaproxima, tim.plunkett: Add assertElementIsVisible() as a more assertive waitForElementVisible() --- .../ProjectBrowserExamplePluginTest.php | 2 +- .../ProjectBrowserInstallerUiTest.php | 14 +- .../ProjectBrowserPluginTest.php | 10 +- .../ProjectBrowserUiTest.php | 124 ++++++++---------- .../ProjectBrowserUiTestJsonApi.php | 22 ++-- .../ProjectBrowserUiTestTrait.php | 15 +++ 6 files changed, 92 insertions(+), 95 deletions(-) diff --git a/tests/src/FunctionalJavascript/ProjectBrowserExamplePluginTest.php b/tests/src/FunctionalJavascript/ProjectBrowserExamplePluginTest.php index 8265c2911..29c01642e 100644 --- a/tests/src/FunctionalJavascript/ProjectBrowserExamplePluginTest.php +++ b/tests/src/FunctionalJavascript/ProjectBrowserExamplePluginTest.php @@ -51,7 +51,7 @@ class ProjectBrowserExamplePluginTest extends WebDriverTestBase { $this->drupalGet('admin/modules/browse/project_browser_source_example'); $this->svelteInitHelper('css', '#project-browser .pb-project--grid'); $this->assertEquals('Grid', $this->getElementText('#project-browser .pb-display__button[value="Grid"]')); - $assert_session->waitForElementVisible('css', '#project-browser .pb-project'); + $this->assertElementIsVisible('css', '#project-browser .pb-project'); $this->assertTrue($assert_session->waitForText('Project 1')); $assert_session->pageTextNotContains('No modules found'); $this->svelteInitHelper('css', '.pb-filter__checkbox'); diff --git a/tests/src/FunctionalJavascript/ProjectBrowserInstallerUiTest.php b/tests/src/FunctionalJavascript/ProjectBrowserInstallerUiTest.php index 9673f0e00..eaf989a41 100644 --- a/tests/src/FunctionalJavascript/ProjectBrowserInstallerUiTest.php +++ b/tests/src/FunctionalJavascript/ProjectBrowserInstallerUiTest.php @@ -93,13 +93,10 @@ class ProjectBrowserInstallerUiTest extends WebDriverTestBase { * such as via the terminal with Compose or a direct file addition. */ public function testInstallModuleAlreadyInFilesystem(): void { - $assert_session = $this->assertSession(); - $this->drupalGet('admin/modules/browse/project_browser_test_mock'); $this->waitForProject('Pinky and the Brain') ->pressButton('Install Pinky and the Brain'); - $popup = $assert_session->waitForElementVisible('css', '.project-browser-popup'); - $this->assertNotEmpty($popup); + $popup = $this->assertElementIsVisible('css', '.project-browser-popup'); // The Pinky and the Brain module doesn't actually exist in the filesystem, // but the test activator pretends it does, in order to test the presence // of the "Install" button as opposed vs. the default "Add and Install" @@ -123,7 +120,7 @@ class ProjectBrowserInstallerUiTest extends WebDriverTestBase { $this->drupalGet('admin/modules/browse/recipes'); $this->svelteInitHelper('css', '.pb-projects-list'); $this->inputSearchField('image', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); // Apply a recipe that ships with core. $card = $this->waitForProject('Image media type'); @@ -133,8 +130,7 @@ class ProjectBrowserInstallerUiTest extends WebDriverTestBase { // If we reload, the installation status should be remembered. $this->getSession()->reload(); $this->inputSearchField('image', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit") - ?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $card = $this->waitForProject('Image media type'); $this->waitForProjectToBeInstalled($card); @@ -144,7 +140,7 @@ class ProjectBrowserInstallerUiTest extends WebDriverTestBase { $this->markTestSkipped('This test cannot continue because this version of Drupal does not support collecting recipe input.'); } $this->inputSearchField('test', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->waitForProject('Test Recipe')->pressButton('Install'); $field = $assert_session->waitForField('test_recipe[new_name]'); $this->assertNotEmpty($field); @@ -152,7 +148,7 @@ class ProjectBrowserInstallerUiTest extends WebDriverTestBase { $page->pressButton('Continue'); $this->checkForMetaRefresh(); $this->inputSearchField('test', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->waitForProjectToBeInstalled('Test Recipe'); $this->assertSame('Y halo thar!', $this->config('system.site')->get('name')); } diff --git a/tests/src/FunctionalJavascript/ProjectBrowserPluginTest.php b/tests/src/FunctionalJavascript/ProjectBrowserPluginTest.php index 6976e05ff..d0ea21375 100644 --- a/tests/src/FunctionalJavascript/ProjectBrowserPluginTest.php +++ b/tests/src/FunctionalJavascript/ProjectBrowserPluginTest.php @@ -59,7 +59,7 @@ class ProjectBrowserPluginTest extends WebDriverTestBase { $this->drupalGet('admin/modules/browse/random_data'); $this->svelteInitHelper('css', '#project-browser .pb-project--grid'); $this->assertEquals('Grid', $this->getElementText('#project-browser .pb-display__button[value="Grid"]')); - $assert_session->waitForElementVisible('css', '#project-browser .pb-project'); + $this->assertElementIsVisible('css', '#project-browser .pb-project'); $this->assertTrue($assert_session->waitForText('Results')); $assert_session->pageTextNotContains('No modules found'); } @@ -155,15 +155,13 @@ class ProjectBrowserPluginTest extends WebDriverTestBase { */ public function testDetailPageRandomDataPlugin(): void { $assert_session = $this->assertSession(); - $page = $this->getSession()->getPage(); $this->drupalGet('admin/modules/browse/random_data'); - $this->assertNotEmpty($assert_session->waitForElementVisible('css', '#project-browser .pb-project')); + $this->assertElementIsVisible('css', '#project-browser .pb-project'); $this->assertTrue($assert_session->waitForText('Results')); - $assert_session->waitForElementVisible('css', '.pb-project .pb-project__title'); - $first_project_selector = $page->find('css', '.pb-project .pb-project__title .pb-project__link'); - $first_project_selector?->click(); + $this->assertElementIsVisible('css', '.pb-project .pb-project__title .pb-project__link') + ->click(); $this->assertTrue($assert_session->waitForText('sites report using this module')); } diff --git a/tests/src/FunctionalJavascript/ProjectBrowserUiTest.php b/tests/src/FunctionalJavascript/ProjectBrowserUiTest.php index fa833827f..5966812df 100644 --- a/tests/src/FunctionalJavascript/ProjectBrowserUiTest.php +++ b/tests/src/FunctionalJavascript/ProjectBrowserUiTest.php @@ -75,16 +75,16 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->assertTrue($assert_session->waitForText('Results')); $assert_session->pageTextNotContains('No modules found'); $page->pressButton('List'); - $this->assertNotNull($assert_session->waitForElementVisible('css', '#project-browser .pb-project.pb-project--list')); + $this->assertElementIsVisible('css', '#project-browser .pb-project.pb-project--list'); $assert_session->elementsCount('css', '#project-browser .pb-project.pb-project--list', 10); $page->pressButton('Grid'); - $this->assertNotNull($assert_session->waitForElementVisible('css', '#project-browser .pb-project.pb-project--grid')); + $this->assertElementIsVisible('css', '#project-browser .pb-project.pb-project--grid'); $this->getSession()->resizeWindow(1100, 1000); $assert_session->assertNoElementAfterWait('css', '.pb-display__button[value="List"]'); - $this->assertNotNull($assert_session->waitForElementVisible('css', '#project-browser .pb-project.pb-project--list')); + $this->assertElementIsVisible('css', '#project-browser .pb-project.pb-project--list'); $assert_session->elementsCount('css', '#project-browser .pb-project.pb-project--list', 10); $this->getSession()->resizeWindow(1210, 1210); - $this->assertNotNull($assert_session->waitForElementVisible('css', '#project-browser .pb-project.pb-project--grid')); + $this->assertElementIsVisible('css', '#project-browser .pb-project.pb-project--grid'); $assert_session->elementsCount('css', '#project-browser .pb-project.pb-project--grid', 10); } @@ -131,9 +131,9 @@ class ProjectBrowserUiTest extends WebDriverTestBase { // @todo Move this into a trait for testing Project Browser's UI in a // stable, consistent fashion. - $assert_category_filters_applied = function (array $expected_categories) use ($assert_session): void { + $assert_category_filters_applied = function (array $expected_categories): void { $selector = '.filter-applied__label'; - $this->assertNotEmpty($assert_session->waitForElementVisible('css', $selector)); + $this->assertElementIsVisible('css', $selector); $applied_categories = array_map( fn ($element) => $element->getText(), $this->getSession()->getPage()->findAll('css', $selector), @@ -210,12 +210,11 @@ class ProjectBrowserUiTest extends WebDriverTestBase { */ public function testReadonlyFields(): void { $page = $this->getSession()->getPage(); - $assert_session = $this->assertSession(); $this->drupalGet('admin/modules/browse/project_browser_test_mock'); $this->svelteInitHelper('text', 'Helvetica'); - $assert_session->waitForElementVisible('css', '.project__action_button'); - $page->pressButton('View Commands for Helvetica'); + $this->assertElementIsVisible('named', ['button', 'View Commands for Helvetica']) + ->press(); $command_boxes = $page->waitFor(10, fn ($page) => $page->findAll('css', '.command-box textarea[readonly]')); $this->assertCount(2, $command_boxes); @@ -337,7 +336,7 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $assert_session->elementsCount('css', '#project-browser .pb-project.pb-project--list', 12); $assert_session->waitForText('Modules per page'); $page->selectFieldOption('num-projects', '24'); - $assert_session->waitForElementVisible('css', '#project-browser .pb-project.pb-project--list'); + $this->assertElementIsVisible('css', '#project-browser .pb-project.pb-project--list'); $assert_session->elementsCount('css', '#project-browser .pb-project.pb-project--list', 24); } @@ -543,8 +542,6 @@ class ProjectBrowserUiTest extends WebDriverTestBase { * Tests search with strings that need URI encoding. */ public function testSearchForSpecialChar(): void { - $assert_session = $this->assertSession(); - // Clear filters. $this->drupalGet('admin/modules/browse/project_browser_test_mock'); $this->svelteInitHelper('text', '10 Results'); @@ -553,7 +550,7 @@ class ProjectBrowserUiTest extends WebDriverTestBase { // Fill in the search field. $this->inputSearchField('', TRUE); $this->inputSearchField('&', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->assertProjectsVisible([ 'Vitamin&C;$?', 'Unwritten&:/', @@ -562,7 +559,7 @@ class ProjectBrowserUiTest extends WebDriverTestBase { // Fill in the search field. $this->inputSearchField('', TRUE); $this->inputSearchField('n&', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->assertProjectsVisible([ 'Vitamin&C;$?', 'Unwritten&:/', @@ -570,28 +567,28 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->inputSearchField('', TRUE); $this->inputSearchField('$', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->assertProjectsVisible([ 'Vitamin&C;$?', ]); $this->inputSearchField('', TRUE); $this->inputSearchField('?', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->assertProjectsVisible([ 'Vitamin&C;$?', ]); $this->inputSearchField('', TRUE); $this->inputSearchField('&:', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->assertProjectsVisible([ 'Unwritten&:/', ]); $this->inputSearchField('', TRUE); $this->inputSearchField('$?', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->assertProjectsVisible([ 'Vitamin&C;$?', ]); @@ -606,7 +603,7 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->svelteInitHelper('text', 'Helvetica'); $assert_session->waitForButton('Helvetica')?->click(); // Check the detail modal displays. - $assert_session->waitForElementVisible('xpath', '//span[contains(@class, "ui-dialog-title") and text()="Helvetica"]'); + $this->assertElementIsVisible('xpath', '//span[contains(@class, "ui-dialog-title") and text()="Helvetica"]'); $assert_session->elementExists('css', 'button.pb__action_button'); // Close the modal. $assert_session->waitForButton('Close')?->click(); @@ -622,18 +619,19 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->svelteInitHelper('text', 'Helvetica'); $assert_session->waitForButton('Helvetica')?->click(); // Check the detail modal displays. - $assert_session->waitForElementVisible('xpath', '//span[contains(@class, "ui-dialog-title") and text()="Helvetica"]'); + $this->assertElementIsVisible('xpath', '//span[contains(@class, "ui-dialog-title") and text()="Helvetica"]'); $assert_session->elementExists('css', 'button.pb__action_button'); // Close the modal and check it no longer exists. $assert_session->waitForButton('Close')?->click(); $assert_session->elementNotExists('xpath', '//span[contains(@class, "ui-dialog-title") and text()="Helvetica"]'); // Check that a different module modal can be opened. $assert_session->waitForButton('Octopus')?->click(); - $assert_session->waitForElementVisible('xpath', '//span[contains(@class, "ui-dialog-title") and text()="Octopus"]'); + $this->assertElementIsVisible('xpath', '//span[contains(@class, "ui-dialog-title") and text()="Octopus"]'); $assert_session->waitForButton('Close')?->click(); $assert_session->elementNotExists('xpath', '//span[contains(@class, "ui-dialog-title") and text()="Octopus"]'); // Check that first detail modal can be reopened. - $assert_session->waitForElementVisible('xpath', '//span[contains(@class, "ui-dialog-title") and text()="Helvetica"]'); + $assert_session->waitForButton('Helvetica')?->click(); + $this->assertElementIsVisible('xpath', '//span[contains(@class, "ui-dialog-title") and text()="Helvetica"]'); $assert_session->elementExists('css', 'button.pb__action_button'); } @@ -652,18 +650,18 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->sortBy('z_a'); // Select the active development status filter. - $assert_session->waitForElementVisible('css', self::DEVELOPMENT_OPTION_SELECTOR); + $this->assertElementIsVisible('css', self::DEVELOPMENT_OPTION_SELECTOR); $this->clickWithWait(self::DEVELOPMENT_OPTION_SELECTOR . self::OPTION_FIRST_CHILD); // Open category drop-down. $assert_session->elementExists('css', '.pb-filter__multi-dropdown')?->click(); // Select the E-commerce filter. - $assert_session->waitForElementVisible('css', '#104'); + $this->assertElementIsVisible('css', '#104'); $this->clickWithWait('#104', '', TRUE); // Select the Media filter. - $assert_session->waitForElementVisible('css', '#67'); + $this->assertElementIsVisible('css', '#67'); $this->clickWithWait('#67', '', TRUE); $this->assertTrue($assert_session->waitForText('15 Results')); @@ -789,7 +787,7 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->pressWithWait('random_data'); $this->svelteInitHelper('css', '.pb-filter__checkbox'); $assert_session->elementsCount('css', '.pb-filter__multi-dropdown__items > div input', 20); - $assert_session->waitForElementVisible('css', '#project-browser .pb-project'); + $this->assertElementIsVisible('css', '#project-browser .pb-project'); $this->assertNotEquals('9 Results Sorted by Active installs', $this->getElementText('.pb-search-results')); // Switching tab will not change result count. $this->assertEquals($second_tab_text . ' (active tab)', $page->findButton('random_data')->getText()); @@ -836,7 +834,7 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->pressWithWait('project_browser_test_mock'); // Filter by search text. $this->inputSearchField('Number', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->assertTrue($assert_session->waitForText('2 Results')); $this->assertProjectsVisible([ '9 Starts With a Higher Number', @@ -1030,7 +1028,7 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->svelteInitHelper('css', '.pb-project.pb-project--list'); $this->inputSearchField('inline form errors', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->svelteInitHelper('text', 'Inline Form Errors'); $install_link = $page->find('css', '.pb-layout__main .pb-actions a'); @@ -1039,7 +1037,7 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->assertIsString($href); $this->assertStringEndsWith('/admin/modules#module-inline-form-errors', $href); $this->drupalGet($href); - $assert_session->waitForElementVisible('css', "#edit-modules-inline-form-errors-enable"); + $this->assertElementIsVisible('css', "#edit-modules-inline-form-errors-enable"); $assert_session->assertVisibleInViewport('css', '#edit-modules-inline-form-errors-enable'); } @@ -1080,7 +1078,7 @@ class ProjectBrowserUiTest extends WebDriverTestBase { // Search for something to change it. $this->inputSearchField('abcdefghijklmnop', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->assertTrue($results->waitFor(10, fn (NodeElement $element) => $element->getText() !== $original_text)); // Remove the search text and make sure it auto-updates. @@ -1096,13 +1094,12 @@ class ProjectBrowserUiTest extends WebDriverTestBase { */ public function testSearchClearNoTabIndex(): void { $page = $this->getSession()->getPage(); - $assert_session = $this->assertSession(); $this->drupalGet('admin/modules/browse/project_browser_test_mock'); $this->svelteInitHelper('css', '.pb-search-results'); // Search and confirm clear button has no focus after tabbing. $this->inputSearchField('abcdefghijklmnop', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->getSession()->getDriver()->keyPress($page->getXpath(), '9'); $has_focus_id = $this->getSession()->evaluateScript('document.activeElement.id'); @@ -1113,8 +1110,6 @@ class ProjectBrowserUiTest extends WebDriverTestBase { * Tests that recipes show instructions for applying them. */ public function testRecipeInstructions(): void { - $assert_session = $this->assertSession(); - $this->config('project_browser.admin_settings') ->set('enabled_sources', ['recipes']) ->save(); @@ -1122,15 +1117,14 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->drupalGet('admin/modules/browse/recipes'); $this->svelteInitHelper('css', '.pb-projects-list'); $this->inputSearchField('image', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); // Look for a recipe that ships with core. - $card = $assert_session->waitForElementVisible('css', '.pb-project:contains("Image media type")'); - $this->assertNotEmpty($card); - $assert_session->buttonExists('View Commands', $card)->press(); - $input = $assert_session->waitForElementVisible('css', '.command-box textarea'); - $this->assertNotEmpty($input); - $command = $input->getValue(); + $this->assertElementIsVisible('css', '.pb-project:contains("Image media type")') + ->pressButton('View Commands'); + $command = $this->assertElementIsVisible('css', '.command-box textarea') + ->getValue(); + assert(is_string($command)); // A full path to the PHP executable should be in the command. $this->assertMatchesRegularExpression('/[^\s]+\/php /', $command); $drupal_root = $this->getDrupalRoot(); @@ -1142,14 +1136,12 @@ class ProjectBrowserUiTest extends WebDriverTestBase { * Test that items with 0 active installs don't show, and >0 do. */ public function testActiveInstallVisibility(): void { - $page = $this->getSession()->getPage(); - $assert_session = $this->assertSession(); $this->drupalGet('admin/modules/browse/project_browser_test_mock'); $this->svelteInitHelper('css', '.pb-search-results'); - $assert_session->waitForElementVisible('css', '.pb-project'); + $this->assertElementIsVisible('css', '.pb-project'); // Find the first and last .pb-project elements. - $projects = $page->findAll('css', '.pb-project'); + $projects = $this->getSession()->getPage()->findAll('css', '.pb-project'); // Assert that there are pb-project elements on the page. $this->assertNotEmpty($projects, 'No .pb-project elements found on the page.'); @@ -1171,8 +1163,6 @@ class ProjectBrowserUiTest extends WebDriverTestBase { * Tests that each source plugin has its own dedicated route. */ public function testSourcePluginRoutes(): void { - $assert_session = $this->assertSession(); - // Enable module for extra source plugin. $this->container->get('module_installer')->install(['project_browser_devel'], TRUE); $this->rebuildContainer(); @@ -1182,7 +1172,7 @@ class ProjectBrowserUiTest extends WebDriverTestBase { foreach (array_keys($current_sources) as $plugin_id) { $this->drupalGet("/admin/modules/browse/{$plugin_id}"); - $this->assertNotNull($assert_session->waitForElementVisible('css', '#project-browser .pb-project.pb-project--list')); + $this->assertElementIsVisible('css', '#project-browser .pb-project.pb-project--list'); } } @@ -1197,11 +1187,11 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->drupalGet('admin/modules/browse/project_browser_test_mock'); $this->svelteInitHelper('text', 'Helvetica'); // This asserts that status icon is present on the cards. - $this->assertNotNull($assert_session->waitForElementVisible('css', '.pb-project__maintenance-icon .pb-project__status-icon-btn')); + $this->assertElementIsVisible('css', '.pb-project__maintenance-icon .pb-project__status-icon-btn'); $assert_session->waitForButton('Helvetica')?->click(); $this->assertTrue($assert_session->waitForText('The module is actively maintained by the maintainers')); // This asserts that status icon is present in detail's modal. - $this->assertNotNull($assert_session->waitForElementVisible('css', '.pb-detail-modal__sidebar .pb-project__status-icon-btn')); + $this->assertElementIsVisible('css', '.pb-detail-modal__sidebar .pb-project__status-icon-btn'); $page->find('css', '.ui-dialog-titlebar-close')?->click(); $this->clickWithWait(self::MAINTENANCE_OPTION_SELECTOR . self::OPTION_LAST_CHILD); @@ -1221,11 +1211,11 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->drupalGet('admin/modules/browse/project_browser_test_mock'); // Ensure the project list is loaded. - $this->assertNotEmpty($assert_session->waitForElementVisible('css', '#project-browser .pb-project')); + $this->assertElementIsVisible('css', '#project-browser .pb-project'); $this->assertTrue($assert_session->waitForText('Results')); // Expect Grapefruit to have 1 install. - $assert_session->waitForElementVisible('xpath', '//span[contains(@class, "pb-project__install-count") and text()="1 install"]'); + $this->assertElementIsVisible('xpath', '//span[contains(@class, "pb-project__install-count") and text()="1 install"]'); // Locate and click the Grapefruit project link. $grapefruit_link = $page->find('xpath', '//button[contains(@class, "pb-project__link") and contains(text(), "Grapefruit")]'); @@ -1255,10 +1245,9 @@ class ProjectBrowserUiTest extends WebDriverTestBase { public function testEnterDoesNotReloadThePage(): void { $this->drupalGet('admin/modules/browse/project_browser_test_mock'); $assert_session = $this->assertSession(); - $search_box = $assert_session->waitForElementVisible('css', '#pb-text'); - $this->assertNotEmpty($search_box); - $session = $this->getSession(); - $session->executeScript('document.body.classList.add("same-page")'); + $search_box = $this->assertElementIsVisible('css', '#pb-text'); + $this->getSession() + ->executeScript('document.body.classList.add("same-page")'); // Enter some nonsense in the search box and press Enter ("\r\n" in PHP). $search_box->focus(); $search_box->setValue("foo\r\n"); @@ -1278,36 +1267,37 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->svelteInitHelper('text', '10 Results'); // Locate the search box and verify it is visible. - $assert_session = $this->assertSession(); - $search_box = $assert_session->waitForElementVisible('css', '#pb-text'); - $this->assertNotEmpty($search_box, 'Search box is visible.'); + $this->assertElementIsVisible('css', '#pb-text'); // Fill in the search field. $this->inputSearchField('', TRUE); // Set the search term to "Astronaut Simulator" to narrow the results. $this->inputSearchField('Astronaut Simulator', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); // Verify the singular result count text is displayed correctly. - $result_count_text = $assert_session->waitForElementVisible('css', '.pb-search-results')?->getText(); - $this->assertSame('1 Result', $result_count_text); + $result_count = $this->assertElementIsVisible('css', '.pb-search-results'); + $this->assertTrue( + $result_count->waitFor( + 10, + fn (NodeElement $element) => $element->getText() === '1 Result', + ), + ); } /** * Tests clicking the X next to search, or clear filters resets search. */ public function testClearSearch(): void { - $assert_session = $this->assertSession(); - // Clear filters. $this->drupalGet('admin/modules/browse/project_browser_test_mock'); $this->svelteInitHelper('text', '10 Results'); $this->pressWithWait('Clear filters', '25 Results'); // Fill in the search field. - $search_field = $assert_session->waitForElementVisible('css', '#pb-text'); + $search_field = $this->assertElementIsVisible('css', '#pb-text'); $this->inputSearchField('Tooth Fairy', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->assertProjectsVisible([ 'Tooth Fairy', ]); @@ -1320,7 +1310,7 @@ class ProjectBrowserUiTest extends WebDriverTestBase { // Run search again. $this->inputSearchField('Tooth Fairy', TRUE); - $assert_session->waitForElementVisible('css', ".search__search-submit")?->click(); + $this->assertElementIsVisible('css', ".search__search-submit")->click(); $this->assertProjectsVisible([ 'Tooth Fairy', ]); diff --git a/tests/src/FunctionalJavascript/ProjectBrowserUiTestJsonApi.php b/tests/src/FunctionalJavascript/ProjectBrowserUiTestJsonApi.php index 323b1d466..33d802b69 100644 --- a/tests/src/FunctionalJavascript/ProjectBrowserUiTestJsonApi.php +++ b/tests/src/FunctionalJavascript/ProjectBrowserUiTestJsonApi.php @@ -74,22 +74,22 @@ class ProjectBrowserUiTestJsonApi extends WebDriverTestBase { $this->getSession()->resizeWindow(1250, 1000); $this->drupalGet('admin/modules/browse/drupalorg_jsonapi'); $this->svelteInitHelper('css', '.pb-project.pb-project--grid'); - $assert_session->waitForElementVisible('css', '#project-browser .pb-display__button[value="Grid"]'); + $this->assertElementIsVisible('css', '#project-browser .pb-display__button[value="Grid"]'); $grid_text = $this->getElementText('#project-browser .pb-display__button[value="Grid"]'); $this->assertEquals('Grid', $grid_text); $this->assertTrue($assert_session->waitForText('Results')); $assert_session->pageTextNotContains('No records available'); $page->pressButton('List'); - $this->assertNotNull($assert_session->waitForElementVisible('css', '#project-browser .pb-project.pb-project--list')); + $this->assertElementIsVisible('css', '#project-browser .pb-project.pb-project--list'); $assert_session->elementsCount('css', '#project-browser .pb-project.pb-project--list', 12); $page->pressButton('Grid'); - $this->assertNotNull($assert_session->waitForElementVisible('css', '#project-browser .pb-project.pb-project--grid')); + $this->assertElementIsVisible('css', '#project-browser .pb-project.pb-project--grid'); $this->getSession()->resizeWindow(1100, 1000); $assert_session->assertNoElementAfterWait('css', '.toggle.list-button'); - $this->assertNotNull($assert_session->waitForElementVisible('css', '#project-browser .pb-project.pb-project--list')); + $this->assertElementIsVisible('css', '#project-browser .pb-project.pb-project--list'); $assert_session->elementsCount('css', '#project-browser .pb-project.pb-project--list', 12); $this->getSession()->resizeWindow(1210, 1210); - $this->assertNotNull($assert_session->waitForElementVisible('css', '#project-browser .pb-project.pb-project--grid')); + $this->assertElementIsVisible('css', '#project-browser .pb-project.pb-project--grid'); $assert_session->elementsCount('css', '#project-browser .pb-project.pb-project--grid', 12); } @@ -249,7 +249,7 @@ class ProjectBrowserUiTestJsonApi extends WebDriverTestBase { $assert_session->pageTextNotContains(' 0 Results'); // Click the Active filter. - $assert_session->waitForElementVisible('css', self::DEVELOPMENT_OPTION_SELECTOR); + $this->assertElementIsVisible('css', self::DEVELOPMENT_OPTION_SELECTOR); $this->clickWithWait(self::DEVELOPMENT_OPTION_SELECTOR . self::OPTION_FIRST_CHILD); // Make sure the correct filter was applied. @@ -353,11 +353,11 @@ class ProjectBrowserUiTestJsonApi extends WebDriverTestBase { $this->drupalGet('admin/modules/browse/project_browser_test_mock'); // Drupal.org test mock defines only two filters (actively maintained filter // and security coverage filter). - $assert_session->waitForElementVisible('css', '.search__form-filters-container'); + $this->assertElementIsVisible('css', '.search__form-filters-container'); $this->assertTrue($assert_session->waitForText('Maintenance status')); - $assert_session->waitForElementVisible('css', self::MAINTENANCE_OPTION_SELECTOR); + $this->assertElementIsVisible('css', self::MAINTENANCE_OPTION_SELECTOR); $this->assertTrue($assert_session->waitForText('Security advisory coverage')); - $assert_session->waitForElementVisible('css', self::SECURITY_OPTION_SELECTOR); + $this->assertElementIsVisible('css', self::SECURITY_OPTION_SELECTOR); // Make sure no other filters are displayed. $this->assertFalse($assert_session->waitForText('Development status')); $this->assertNull($assert_session->waitForElementVisible('css', self::DEVELOPMENT_OPTION_SELECTOR)); @@ -412,7 +412,6 @@ class ProjectBrowserUiTestJsonApi extends WebDriverTestBase { $this->assertCount(2, $local_tasks); // Verify that the mocked source is first tab. $this->assertSame('Contrib modules', $local_tasks[0]->getText()); - $assert_session->waitForElementVisible('css', '.pb-display__button'); // Re-order plugins. $this->drupalGet('admin/config/development/project_browser'); @@ -426,8 +425,7 @@ class ProjectBrowserUiTestJsonApi extends WebDriverTestBase { // Verify that Random data is first tab. $this->drupalGet('admin/modules/browse/drupalorg_jsonapi'); - $assert_session->waitForElementVisible('css', '#project-browser .pb-project'); - $first_tab = $page->find('css', '.pb-tabs__link:nth-child(1)'); + $this->assertElementIsVisible('css', '#project-browser .pb-project'); $this->assertSame('Random data', $local_tasks[0]->getText()); // Disable the mock plugin. diff --git a/tests/src/FunctionalJavascript/ProjectBrowserUiTestTrait.php b/tests/src/FunctionalJavascript/ProjectBrowserUiTestTrait.php index b7f40a120..0816594aa 100644 --- a/tests/src/FunctionalJavascript/ProjectBrowserUiTestTrait.php +++ b/tests/src/FunctionalJavascript/ProjectBrowserUiTestTrait.php @@ -11,6 +11,21 @@ use Behat\Mink\Element\NodeElement; */ trait ProjectBrowserUiTestTrait { + /** + * Waits for an element to be visible, and returns it. + * + * @param mixed ...$arguments + * Arguments to pass to JSWebAssert::waitForElementVisible(). + * + * @return \Behat\Mink\Element\NodeElement + * The element we were waiting for. + */ + protected function assertElementIsVisible(mixed ...$arguments): NodeElement { + $element = $this->assertSession()->waitForElementVisible(...$arguments); + $this->assertInstanceOf(NodeElement::class, $element); + return $element; + } + /** * Installs a specific project. * -- GitLab