diff --git a/e2e/tests/ui/layout/search/grandsearch.e2e.spec.js b/e2e/tests/ui/layout/search/grandsearch.e2e.spec.js new file mode 100644 index 0000000000..f67b336731 --- /dev/null +++ b/e2e/tests/ui/layout/search/grandsearch.e2e.spec.js @@ -0,0 +1,111 @@ +/***************************************************************************** + * Open MCT, Copyright (c) 2014-2022, United States Government + * as represented by the Administrator of the National Aeronautics and Space + * Administration. All rights reserved. + * + * Open MCT is licensed under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * http://www.apache.org/licenses/LICENSE-2.0. + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + * + * Open MCT includes source code licensed under additional open source + * licenses. See the Open Source Licenses file (LICENSES.md) included with + * this source code distribution or the Licensing information page available + * at runtime from the About dialog for additional information. + *****************************************************************************/ + +/* +This test suite is dedicated to tests which verify search functionality. +*/ + +const { expect } = require('@playwright/test'); +const { test } = require('../../../../fixtures'); + +/** + * Creates a notebook object and adds an entry. + * @param {import('@playwright/test').Page} page + */ +async function createClockAndDisplayLayout(page) { + //Go to baseURL + await page.goto('/', { waitUntil: 'networkidle' }); + + // Click button:has-text("Create") + await page.locator('button:has-text("Create")').click(); + // Click li:has-text("Notebook") + await page.locator('li:has-text("Clock")').click(); + // Click button:has-text("OK") + await Promise.all([ + page.waitForNavigation(), + page.locator('button:has-text("OK")').click() + ]); + + // Click a:has-text("My Items") + await Promise.all([ + page.waitForNavigation(), + page.locator('a:has-text("My Items") >> nth=0').click() + ]); + // Click button:has-text("Create") + await page.locator('button:has-text("Create")').click(); + // Click li:has-text("Notebook") + await page.locator('li:has-text("Display Layout")').click(); + // Click button:has-text("OK") + await Promise.all([ + page.waitForNavigation(), + page.locator('button:has-text("OK")').click() + ]); +} + +test.describe('Grand Search', () => { + test('Can search for objects, and subsequent search dropdown behaves properly', async ({ page }) => { + await createClockAndDisplayLayout(page); + + // Click [aria-label="OpenMCT Search"] input[type="search"] + await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click(); + // Fill [aria-label="OpenMCT Search"] input[type="search"] + await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Cl'); + await expect(page.locator('[aria-label="Search Result"]')).toContainText('Clock'); + // Click text=Elements >> nth=0 + await page.locator('text=Elements').first().click(); + await expect(page.locator('[aria-label="Search Result"]')).not.toBeVisible(); + + // Click [aria-label="OpenMCT Search"] [aria-label="Search Input"] + await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').click(); + // Click [aria-label="Unnamed Clock clock result"] >> text=Unnamed Clock + await page.locator('[aria-label="Unnamed Clock clock result"] >> text=Unnamed Clock').click(); + await expect(page.locator('.js-preview-window')).toBeVisible(); + + // Click [aria-label="Close"] + await page.locator('[aria-label="Close"]').click(); + await expect(page.locator('[aria-label="Search Result"]')).toBeVisible(); + await expect(page.locator('[aria-label="Search Result"]')).toContainText('Cloc'); + + // Click [aria-label="OpenMCT Search"] a >> nth=0 + await page.locator('[aria-label="OpenMCT Search"] a').first().click(); + await expect(page.locator('[aria-label="Search Result"]')).not.toBeVisible(); + + // Fill [aria-label="OpenMCT Search"] input[type="search"] + await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('foo'); + await expect(page.locator('[aria-label="Search Result"]')).not.toBeVisible(); + + // Click text=Snapshot Save and Finish Editing Save and Continue Editing >> button >> nth=1 + await page.locator('text=Snapshot Save and Finish Editing Save and Continue Editing >> button').nth(1).click(); + // Click text=Save and Finish Editing + await page.locator('text=Save and Finish Editing').click(); + // Click [aria-label="OpenMCT Search"] [aria-label="Search Input"] + await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').click(); + // Fill [aria-label="OpenMCT Search"] [aria-label="Search Input"] + await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').fill('Cl'); + // Click text=Unnamed Clock + await Promise.all([ + page.waitForNavigation(), + page.locator('text=Unnamed Clock').click() + ]); + await expect(page.locator('.is-object-type-clock')).toBeVisible(); + }); +}); diff --git a/e2e/tests/visual/search.visual.spec.js b/e2e/tests/visual/search.visual.spec.js new file mode 100644 index 0000000000..654a336dd8 --- /dev/null +++ b/e2e/tests/visual/search.visual.spec.js @@ -0,0 +1,104 @@ +/***************************************************************************** + * Open MCT, Copyright (c) 2014-2022, United States Government + * as represented by the Administrator of the National Aeronautics and Space + * Administration. All rights reserved. + * + * Open MCT is licensed under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * http://www.apache.org/licenses/LICENSE-2.0. + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + * + * Open MCT includes source code licensed under additional open source + * licenses. See the Open Source Licenses file (LICENSES.md) included with + * this source code distribution or the Licensing information page available + * at runtime from the About dialog for additional information. + *****************************************************************************/ + +/* +This test suite is dedicated to tests which verify search functionality. +*/ + +const { test, expect } = require('@playwright/test'); +const percySnapshot = require('@percy/playwright'); + +/** + * Creates a notebook object and adds an entry. + * @param {import('@playwright/test').Page} page + */ +async function createClockAndDisplayLayout(page) { + //Go to baseURL + await page.goto('/', { waitUntil: 'networkidle' }); + + // Click button:has-text("Create") + await page.locator('button:has-text("Create")').click(); + // Click li:has-text("Notebook") + await page.locator('li:has-text("Clock")').click(); + // Click button:has-text("OK") + await Promise.all([ + page.waitForNavigation(), + page.locator('button:has-text("OK")').click() + ]); + + // Click a:has-text("My Items") + await Promise.all([ + page.waitForNavigation(), + page.locator('a:has-text("My Items") >> nth=0').click() + ]); + // Click button:has-text("Create") + await page.locator('button:has-text("Create")').click(); + // Click li:has-text("Notebook") + await page.locator('li:has-text("Display Layout")').click(); + // Click button:has-text("OK") + await Promise.all([ + page.waitForNavigation(), + page.locator('button:has-text("OK")').click() + ]); +} + +test.describe('Grand Search', () => { + test('Can search for objects, and subsequent search dropdown behaves properly', async ({ page }) => { + await createClockAndDisplayLayout(page); + + // Click [aria-label="OpenMCT Search"] input[type="search"] + await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click(); + // Fill [aria-label="OpenMCT Search"] input[type="search"] + await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Cl'); + await expect(page.locator('[aria-label="Search Result"]')).toContainText('Clock'); + await percySnapshot(page, 'Searching for Clocks'); + // Click text=Elements >> nth=0 + await page.locator('text=Elements').first().click(); + await expect(page.locator('[aria-label="Search Result"]')).not.toBeVisible(); + + // Click [aria-label="OpenMCT Search"] [aria-label="Search Input"] + await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').click(); + // Click [aria-label="Unnamed Clock clock result"] >> text=Unnamed Clock + await page.locator('[aria-label="Unnamed Clock clock result"] >> text=Unnamed Clock').click(); + await percySnapshot(page, 'Preview for clock should display when editing enabled and search item clicked'); + + // Click [aria-label="Close"] + await page.locator('[aria-label="Close"]').click(); + await percySnapshot(page, 'Search should still be showing after preview closed'); + + // Click text=Snapshot Save and Finish Editing Save and Continue Editing >> button >> nth=1 + await page.locator('text=Snapshot Save and Finish Editing Save and Continue Editing >> button').nth(1).click(); + // Click text=Save and Finish Editing + await page.locator('text=Save and Finish Editing').click(); + // Click [aria-label="OpenMCT Search"] [aria-label="Search Input"] + await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').click(); + // Fill [aria-label="OpenMCT Search"] [aria-label="Search Input"] + await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').fill('Cl'); + // Click text=Unnamed Clock + await Promise.all([ + page.waitForNavigation(), + page.locator('text=Unnamed Clock').click() + ]); + await percySnapshot(page, 'Clicking on search results should navigate to them if not editing'); + + }); +}); diff --git a/package.json b/package.json index 1ddce586a0..a053839873 100644 --- a/package.json +++ b/package.json @@ -89,7 +89,7 @@ "test": "cross-env NODE_OPTIONS=\"--max_old_space_size=4096\" karma start --single-run", "test:firefox": "cross-env NODE_OPTIONS=\"--max_old_space_size=4096\" karma start --single-run --browsers=FirefoxHeadless", "test:debug": "cross-env NODE_ENV=debug karma start --no-single-run", - "test:e2e:ci": "npx playwright test --config=e2e/playwright-ci.config.js --project=chrome smoke branding default condition timeConductor clock exampleImagery persistence performance", + "test:e2e:ci": "npx playwright test --config=e2e/playwright-ci.config.js --project=chrome smoke branding default condition timeConductor clock exampleImagery persistence performance grandsearch", "test:e2e:local": "npx playwright test --config=e2e/playwright-local.config.js --project=chrome", "test:e2e:updatesnapshots": "npx playwright test --config=e2e/playwright-local.config.js --project=chrome --grep @snapshot --update-snapshots", "test:e2e:visual": "percy exec --config ./e2e/.percy.yml -- npx playwright test --config=e2e/playwright-visual.config.js", diff --git a/src/api/forms/components/controls/AutoCompleteField.vue b/src/api/forms/components/controls/AutoCompleteField.vue index a88a9036d3..bc39105267 100644 --- a/src/api/forms/components/controls/AutoCompleteField.vue +++ b/src/api/forms/components/controls/AutoCompleteField.vue @@ -44,6 +44,7 @@