Compare commits
123 Commits
fix-openmc
...
fix-duplic
Author | SHA1 | Date | |
---|---|---|---|
d3825fc554 | |||
d42b2dd023 | |||
59ae7cddc5 | |||
321c7a3af5 | |||
609cf72bd1 | |||
a447b0ada8 | |||
5788f4cc69 | |||
f94b4e53c7 | |||
faf71f1e67 | |||
23310f85ae | |||
d80819634b | |||
483b62c152 | |||
1254279635 | |||
c768a71656 | |||
678a92bd29 | |||
34b488944a | |||
4d1dd2f51d | |||
080f7b8f4b | |||
483f2feac8 | |||
7ec2c4475b | |||
8f59b16465 | |||
36cfb1d515 | |||
2ff7132e90 | |||
d0ca398e01 | |||
59278e8a06 | |||
c8377f392b | |||
29df748f2b | |||
665ba6dae1 | |||
f39f8df4e2 | |||
4aa572d489 | |||
0b24c4f2c5 | |||
e4657f79cd | |||
f2059406e0 | |||
3e3dc7dd83 | |||
50742c4f82 | |||
2f04add2a3 | |||
0ce5060246 | |||
00353cdccf | |||
a1ac209d74 | |||
bdd8477b54 | |||
f690f36bfb | |||
e174f075df | |||
8cf12db104 | |||
453b1f3009 | |||
201c669328 | |||
1b7fb9b952 | |||
a3c5450205 | |||
8831b75c5d | |||
8fe0472af2 | |||
6cb5c47f3a | |||
eff0cc96b9 | |||
6ac7f24c63 | |||
39463c515f | |||
25c0dab346 | |||
3714958627 | |||
270a3d4f49 | |||
1dc137f95e | |||
ff3a20e446 | |||
0b3e0e7efd | |||
22cc28d733 | |||
006fa0bcc7 | |||
817d8da3e4 | |||
8df81f0ea9 | |||
1f30706d27 | |||
600890c4a6 | |||
b5002e166a | |||
39cff51db0 | |||
73734d99ea | |||
1d4cf1ff06 | |||
f388d9a548 | |||
8040b275fc | |||
0dd12bce85 | |||
9c9e0442f1 | |||
d49f057698 | |||
c74ad1279c | |||
470a451956 | |||
fa6cbb6f4d | |||
52c00cfaef | |||
96d723a424 | |||
fb4b80862e | |||
bb2c8cfa63 | |||
ceffee9f22 | |||
a08ccd80dc | |||
3509eacdec | |||
d4496cba41 | |||
64f300d466 | |||
8de24a109a | |||
6d62e0e73c | |||
5da1c9c0d7 | |||
4fa9a9697b | |||
bf48a6e306 | |||
00ad452930 | |||
8df1f6406b | |||
a50960d66c | |||
e3a69c8856 | |||
672cb7e621 | |||
7dcccee1ae | |||
302dbe7359 | |||
b4df01965e | |||
5a8f1d542e | |||
10decda94e | |||
5b1f8d0eac | |||
2f6e1b703a | |||
5384022a59 | |||
b57974b462 | |||
3c36ba9a71 | |||
2ac463de90 | |||
be38c3e654 | |||
0f312a88bb | |||
422b7f3e09 | |||
800062d37e | |||
c1e8c7915c | |||
c1c1d87953 | |||
0382d22f7f | |||
f570424357 | |||
393c801426 | |||
6d63339b23 | |||
66d7c626e1 | |||
2246f33023 | |||
871362d469 | |||
cc1bf47f5a | |||
9c784398b3 | |||
21ce013df2 |
@ -23,5 +23,5 @@ module.exports = merge(common, {
|
|||||||
__OPENMCT_ROOT_RELATIVE__: '""'
|
__OPENMCT_ROOT_RELATIVE__: '""'
|
||||||
})
|
})
|
||||||
],
|
],
|
||||||
devtool: "source-map"
|
devtool: "eval-source-map"
|
||||||
});
|
});
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Open MCT License
|
# Open MCT License
|
||||||
|
|
||||||
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, Copyright (c) 2014-2023, 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.
|
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.
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
# Open MCT [](http://www.apache.org/licenses/LICENSE-2.0) [](https://lgtm.com/projects/g/nasa/openmct/context:javascript) [](https://codecov.io/gh/nasa/openmct) [](https://percy.io/b2e34b17/openmct) [](https://www.npmjs.com/package/openmct)
|
# Open MCT [](http://www.apache.org/licenses/LICENSE-2.0) [](https://codecov.io/gh/nasa/openmct) [](https://percy.io/b2e34b17/openmct) [](https://www.npmjs.com/package/openmct)
|
||||||
|
|
||||||
Open MCT (Open Mission Control Technologies) is a next-generation mission control framework for visualization of data on desktop and mobile devices. It is developed at NASA's Ames Research Center, and is being used by NASA for data analysis of spacecraft missions, as well as planning and operation of experimental rover systems. As a generalizable and open source framework, Open MCT could be used as the basis for building applications for planning, operation, and analysis of any systems producing telemetry data.
|
Open MCT (Open Mission Control Technologies) is a next-generation mission control framework for visualization of data on desktop and mobile devices. It is developed at NASA's Ames Research Center, and is being used by NASA for data analysis of spacecraft missions, as well as planning and operation of experimental rover systems. As a generalizable and open source framework, Open MCT could be used as the basis for building applications for planning, operation, and analysis of any systems producing telemetry data.
|
||||||
|
|
||||||
@ -98,7 +98,7 @@ To run the performance tests:
|
|||||||
The test suite is configured to all tests localed in `e2e/tests/` ending in `*.e2e.spec.js`. For more about the e2e test suite, please see the [README](./e2e/README.md)
|
The test suite is configured to all tests localed in `e2e/tests/` ending in `*.e2e.spec.js`. For more about the e2e test suite, please see the [README](./e2e/README.md)
|
||||||
|
|
||||||
### Security Tests
|
### Security Tests
|
||||||
Each commit is analyzed for known security vulnerabilities using [CodeQL](https://codeql.github.com/docs/codeql-language-guides/codeql-library-for-javascript/) and our overall security report is available on [LGTM](https://lgtm.com/projects/g/nasa/openmct/). The list of CWE coverage items is avaiable in the [CodeQL docs](https://codeql.github.com/codeql-query-help/javascript-cwe/). The CodeQL workflow is specified in the [CodeQL analysis file](./.github/workflows/codeql-analysis.yml) and the custom [CodeQL config](./.github/codeql/codeql-config.yml).
|
Each commit is analyzed for known security vulnerabilities using [CodeQL](https://codeql.github.com/docs/codeql-language-guides/codeql-library-for-javascript/). The list of CWE coverage items is avaiable in the [CodeQL docs](https://codeql.github.com/codeql-query-help/javascript-cwe/). The CodeQL workflow is specified in the [CodeQL analysis file](./.github/workflows/codeql-analysis.yml) and the custom [CodeQL config](./.github/codeql/codeql-config.yml).
|
||||||
|
|
||||||
### Test Reporting and Code Coverage
|
### Test Reporting and Code Coverage
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
#*****************************************************************************
|
#*****************************************************************************
|
||||||
#* Open MCT, Copyright (c) 2014-2022, United States Government
|
#* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
#* as represented by the Administrator of the National Aeronautics and Space
|
#* as represented by the Administrator of the National Aeronautics and Space
|
||||||
#* Administration. All rights reserved.
|
#* Administration. All rights reserved.
|
||||||
#*
|
#*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<!--
|
<!--
|
||||||
Open MCT, Copyright (c) 2014-2022, United States Government
|
Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
as represented by the Administrator of the National Aeronautics and Space
|
as represented by the Administrator of the National Aeronautics and Space
|
||||||
Administration. All rights reserved.
|
Administration. All rights reserved.
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -89,17 +89,37 @@ Read more about [Playwright Snapshots](https://playwright.dev/docs/test-snapshot
|
|||||||
#### Open MCT's implementation
|
#### Open MCT's implementation
|
||||||
|
|
||||||
- Our Snapshot tests receive a `@snapshot` tag.
|
- Our Snapshot tests receive a `@snapshot` tag.
|
||||||
- Snapshots need to be executed within the official Playwright container to ensure we're using the exact rendering platform in CI and locally.
|
- Snapshots need to be executed within the official Playwright container to ensure we're using the exact rendering platform in CI and locally. To do a valid comparison locally:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
docker run --rm --network host -v $(pwd):/work/ -w /work/ -it mcr.microsoft.com/playwright:[GET THIS VERSION FROM OUR CIRCLECI CONFIG FILE]-focal /bin/bash
|
// Replace {X.X.X} with the current Playwright version
|
||||||
|
// from our package.json or circleCI configuration file
|
||||||
|
docker run --rm --network host -v $(pwd):/work/ -w /work/ -it mcr.microsoft.com/playwright:v{X.X.X}-focal /bin/bash
|
||||||
npm install
|
npm install
|
||||||
npx playwright test --config=e2e/playwright-ci.config.js --project=chrome --grep @snapshot
|
npx playwright test --config=e2e/playwright-ci.config.js --project=chrome --grep @snapshot
|
||||||
```
|
```
|
||||||
|
|
||||||
### (WIP) Updating Snapshots
|
### Updating Snapshots
|
||||||
|
|
||||||
When the `@snapshot` tests fail, they will need to be evaluated to see if the failure is an acceptable change or
|
When the `@snapshot` tests fail, they will need to be evaluated to determine if the failure is an acceptable and desireable or an unintended regression.
|
||||||
|
|
||||||
|
To compare a snapshot, run a test and open the html report with the 'Expected' vs 'Actual' screenshot. If the actual screenshot is preferred, then the source-controlled 'Expected' snapshots will need to be updated with the following scripts.
|
||||||
|
|
||||||
|
MacOS
|
||||||
|
|
||||||
|
```
|
||||||
|
npm run test:e2e:updatesnapshots
|
||||||
|
```
|
||||||
|
|
||||||
|
Linux/CI
|
||||||
|
|
||||||
|
```sh
|
||||||
|
// Replace {X.X.X} with the current Playwright version
|
||||||
|
// from our package.json or circleCI configuration file
|
||||||
|
docker run --rm --network host -v $(pwd):/work/ -w /work/ -it mcr.microsoft.com/playwright:v{X.X.X}-focal /bin/bash
|
||||||
|
npm install
|
||||||
|
npm run test:e2e:updatesnapshots
|
||||||
|
```
|
||||||
|
|
||||||
## Performance Testing
|
## Performance Testing
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -159,24 +159,26 @@ async function expandTreePaneItemByName(page, name) {
|
|||||||
* @returns {Promise<CreatedObjectInfo>} An object containing information about the newly created domain object.
|
* @returns {Promise<CreatedObjectInfo>} An object containing information about the newly created domain object.
|
||||||
*/
|
*/
|
||||||
async function createPlanFromJSON(page, { name, json, parent = 'mine' }) {
|
async function createPlanFromJSON(page, { name, json, parent = 'mine' }) {
|
||||||
|
if (!name) {
|
||||||
|
name = `Plan:${genUuid()}`;
|
||||||
|
}
|
||||||
|
|
||||||
const parentUrl = await getHashUrlToDomainObject(page, parent);
|
const parentUrl = await getHashUrlToDomainObject(page, parent);
|
||||||
|
|
||||||
// Navigate to the parent object. This is necessary to create the object
|
// Navigate to the parent object. This is necessary to create the object
|
||||||
// in the correct location, such as a folder, layout, or plot.
|
// in the correct location, such as a folder, layout, or plot.
|
||||||
await page.goto(`${parentUrl}?hideTree=true`);
|
await page.goto(`${parentUrl}?hideTree=true`);
|
||||||
|
|
||||||
//Click the Create button
|
// Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
// Click 'Plan' menu option
|
// Click 'Plan' menu option
|
||||||
await page.click(`li:text("Plan")`);
|
await page.click(`li:text("Plan")`);
|
||||||
|
|
||||||
// Modify the name input field of the domain object to accept 'name'
|
// Modify the name input field of the domain object to accept 'name'
|
||||||
if (name) {
|
const nameInput = page.locator('form[name="mctForm"] .first input[type="text"]');
|
||||||
const nameInput = page.locator('form[name="mctForm"] .first input[type="text"]');
|
await nameInput.fill("");
|
||||||
await nameInput.fill("");
|
await nameInput.fill(name);
|
||||||
await nameInput.fill(name);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Upload buffer from memory
|
// Upload buffer from memory
|
||||||
await page.locator('input#fileElem').setInputFiles({
|
await page.locator('input#fileElem').setInputFiles({
|
||||||
@ -194,7 +196,7 @@ async function createPlanFromJSON(page, { name, json, parent = 'mine' }) {
|
|||||||
]);
|
]);
|
||||||
|
|
||||||
// Wait until the URL is updated
|
// Wait until the URL is updated
|
||||||
await page.waitForURL(`**/mine/*`);
|
await page.waitForURL(`**/${parent}/*`);
|
||||||
const uuid = await getFocusedObjectUuid(page);
|
const uuid = await getFocusedObjectUuid(page);
|
||||||
const objectUrl = await getHashUrlToDomainObject(page, uuid);
|
const objectUrl = await getHashUrlToDomainObject(page, uuid);
|
||||||
|
|
||||||
@ -235,6 +237,12 @@ async function expandEntireTree(page, treeName = "Main Tree") {
|
|||||||
|
|
||||||
while (await collapsedTreeItems.count() > 0) {
|
while (await collapsedTreeItems.count() > 0) {
|
||||||
await collapsedTreeItems.nth(0).click();
|
await collapsedTreeItems.nth(0).click();
|
||||||
|
|
||||||
|
// FIXME: Replace hard wait with something event-driven.
|
||||||
|
// Without the wait, this fails periodically due to a race condition
|
||||||
|
// with Vue rendering (loop exits prematurely).
|
||||||
|
// eslint-disable-next-line playwright/no-wait-for-timeout
|
||||||
|
await page.waitForTimeout(200);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -377,6 +385,25 @@ async function setEndOffset(page, offset) {
|
|||||||
await setTimeConductorOffset(page, offset, endOffsetButton);
|
await setTimeConductorOffset(page, offset, endOffsetButton);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Selects an inspector tab based on the provided tab name
|
||||||
|
*
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {String} name the name of the tab
|
||||||
|
*/
|
||||||
|
async function selectInspectorTab(page, name) {
|
||||||
|
const inspectorTabs = page.getByRole('tablist');
|
||||||
|
const inspectorTab = inspectorTabs.getByTitle(name);
|
||||||
|
const inspectorTabClass = await inspectorTab.getAttribute('class');
|
||||||
|
const isSelectedInspectorTab = inspectorTabClass.includes('is-current');
|
||||||
|
|
||||||
|
// do not click a tab that is already selected or it will timeout your test
|
||||||
|
// do to a { pointer-events: none; } on selected tabs
|
||||||
|
if (!isSelectedInspectorTab) {
|
||||||
|
await inspectorTab.click();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// eslint-disable-next-line no-undef
|
// eslint-disable-next-line no-undef
|
||||||
module.exports = {
|
module.exports = {
|
||||||
createDomainObjectWithDefaults,
|
createDomainObjectWithDefaults,
|
||||||
@ -390,5 +417,6 @@ module.exports = {
|
|||||||
setFixedTimeMode,
|
setFixedTimeMode,
|
||||||
setRealTimeMode,
|
setRealTimeMode,
|
||||||
setStartOffset,
|
setStartOffset,
|
||||||
setEndOffset
|
setEndOffset,
|
||||||
|
selectInspectorTab
|
||||||
};
|
};
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
/* eslint-disable no-undef */
|
/* eslint-disable no-undef */
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
32
e2e/helper/addInitNotebookWithUrls.js
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2023, 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 should be used to install the re-instal default Notebook plugin with a simple url whitelist.
|
||||||
|
// e.g.
|
||||||
|
// await page.addInitScript({ path: path.join(__dirname, 'addInitNotebookWithUrls.js') });
|
||||||
|
const NOTEBOOK_NAME = 'Notebook';
|
||||||
|
const URL_WHITELIST = ['google.com'];
|
||||||
|
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
const openmct = window.openmct;
|
||||||
|
openmct.install(openmct.plugins.Notebook(NOTEBOOK_NAME, URL_WHITELIST));
|
||||||
|
});
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -32,9 +32,8 @@ async function enterTextEntry(page, text) {
|
|||||||
await page.locator(NOTEBOOK_DROP_AREA).click();
|
await page.locator(NOTEBOOK_DROP_AREA).click();
|
||||||
|
|
||||||
// enter text
|
// enter text
|
||||||
await page.locator('div.c-ne__text').click();
|
await page.locator('[aria-label="Notebook Entry"].is-selected div.c-ne__text').fill(text);
|
||||||
await page.locator('div.c-ne__text').fill(text);
|
await commitEntry(page);
|
||||||
await page.locator('div.c-ne__text').press('Enter');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -51,6 +50,15 @@ async function dragAndDropEmbed(page, notebookObject) {
|
|||||||
await page.click('button[title="Show selected item in tree"]');
|
await page.click('button[title="Show selected item in tree"]');
|
||||||
// Drag and drop the SWG into the notebook
|
// Drag and drop the SWG into the notebook
|
||||||
await page.dragAndDrop(`text=${swg.name}`, NOTEBOOK_DROP_AREA);
|
await page.dragAndDrop(`text=${swg.name}`, NOTEBOOK_DROP_AREA);
|
||||||
|
await commitEntry(page);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function commitEntry(page) {
|
||||||
|
await page.locator('.c-ne__save-button > button').click();
|
||||||
}
|
}
|
||||||
|
|
||||||
// eslint-disable-next-line no-undef
|
// eslint-disable-next-line no-undef
|
||||||
|
92
e2e/helper/planningUtils.js
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2023, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
import { expect } from '../pluginFixtures';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Asserts that the number of activities in the plan view matches the number of
|
||||||
|
* activities in the plan data within the specified time bounds. Performs an assertion
|
||||||
|
* for each activity in the plan data per group, using the earliest activity's
|
||||||
|
* start time as the start bound and the current activity's end time as the end bound.
|
||||||
|
* @param {import('@playwright/test').Page} page the page
|
||||||
|
* @param {object} plan The raw plan json to assert against
|
||||||
|
* @param {string} objectUrl The URL of the object to assert against (plan or gantt chart)
|
||||||
|
*/
|
||||||
|
export async function assertPlanActivities(page, plan, objectUrl) {
|
||||||
|
const groups = Object.keys(plan);
|
||||||
|
for (const group of groups) {
|
||||||
|
for (let i = 0; i < plan[group].length; i++) {
|
||||||
|
// Set the startBound to the start time of the first activity in the group
|
||||||
|
const startBound = plan[group][0].start;
|
||||||
|
// Set the endBound to the end time of the current activity
|
||||||
|
let endBound = plan[group][i].end;
|
||||||
|
if (endBound === startBound) {
|
||||||
|
// Prevent oddities with setting start and end bound equal
|
||||||
|
// via URL params
|
||||||
|
endBound += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Switch to fixed time mode with all plan events within the bounds
|
||||||
|
await page.goto(`${objectUrl}?tc.mode=fixed&tc.startBound=${startBound}&tc.endBound=${endBound}&tc.timeSystem=utc&view=plan.view`);
|
||||||
|
|
||||||
|
// Assert that the number of activities in the plan view matches the number of
|
||||||
|
// activities in the plan data within the specified time bounds
|
||||||
|
const eventCount = await page.locator('.activity-bounds').count();
|
||||||
|
expect(eventCount).toEqual(Object.values(plan)
|
||||||
|
.flat()
|
||||||
|
.filter(event =>
|
||||||
|
activitiesWithinTimeBounds(event.start, event.end, startBound, endBound)).length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns true if the activities time bounds overlap, false otherwise.
|
||||||
|
* @param {number} start1 the start time of the first activity
|
||||||
|
* @param {number} end1 the end time of the first activity
|
||||||
|
* @param {number} start2 the start time of the second activity
|
||||||
|
* @param {number} end2 the end time of the second activity
|
||||||
|
* @returns {boolean} true if the activities overlap, false otherwise
|
||||||
|
*/
|
||||||
|
function activitiesWithinTimeBounds(start1, end1, start2, end2) {
|
||||||
|
return (start1 >= start2 && start1 <= end2)
|
||||||
|
|| (end1 >= start2 && end1 <= end2)
|
||||||
|
|| (start2 >= start1 && start2 <= end1)
|
||||||
|
|| (end2 >= start1 && end2 <= end1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Navigate to the plan view, switch to fixed time mode,
|
||||||
|
* and set the bounds to span all activities.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {object} planJson
|
||||||
|
* @param {string} planObjectUrl
|
||||||
|
*/
|
||||||
|
export async function setBoundsToSpanAllActivities(page, planJson, planObjectUrl) {
|
||||||
|
const activities = Object.values(planJson).flat();
|
||||||
|
// Get the earliest start value
|
||||||
|
const start = Math.min(...activities.map(activity => activity.start));
|
||||||
|
// Get the latest end value
|
||||||
|
const end = Math.max(...activities.map(activity => activity.end));
|
||||||
|
// Set the start and end bounds to the earliest start and latest end
|
||||||
|
await page.goto(`${planObjectUrl}?tc.mode=fixed&tc.startBound=${start}&tc.endBound=${end}&tc.timeSystem=utc&view=plan.view`);
|
||||||
|
}
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -73,7 +73,7 @@ const config = {
|
|||||||
open: 'never',
|
open: 'never',
|
||||||
outputFolder: '../html-test-results' //Must be in different location due to https://github.com/microsoft/playwright/issues/12840
|
outputFolder: '../html-test-results' //Must be in different location due to https://github.com/microsoft/playwright/issues/12840
|
||||||
}],
|
}],
|
||||||
['junit', { outputFile: 'test-results/results.xml' }],
|
['junit', { outputFile: '../test-results/results.xml' }],
|
||||||
['github']
|
['github']
|
||||||
]
|
]
|
||||||
};
|
};
|
||||||
|
@ -35,8 +35,8 @@ const config = {
|
|||||||
],
|
],
|
||||||
reporter: [
|
reporter: [
|
||||||
['list'],
|
['list'],
|
||||||
['junit', { outputFile: 'test-results/results.xml' }],
|
['junit', { outputFile: '../test-results/results.xml' }],
|
||||||
['json', { outputFile: 'test-results/results.json' }]
|
['json', { outputFile: '../test-results/results.json' }]
|
||||||
]
|
]
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -40,7 +40,7 @@ const config = {
|
|||||||
],
|
],
|
||||||
reporter: [
|
reporter: [
|
||||||
['list'],
|
['list'],
|
||||||
['junit', { outputFile: 'test-results/results.xml' }],
|
['junit', { outputFile: '../test-results/results.xml' }],
|
||||||
['html', {
|
['html', {
|
||||||
open: 'on-failure',
|
open: 'on-failure',
|
||||||
outputFolder: '../html-test-results' //Must be in different location due to https://github.com/microsoft/playwright/issues/12840
|
outputFolder: '../html-test-results' //Must be in different location due to https://github.com/microsoft/playwright/issues/12840
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
/* eslint-disable no-undef */
|
/* eslint-disable no-undef */
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
1080
e2e/test-data/examplePlans/ExamplePlan_Large.json
Normal file
44
e2e/test-data/examplePlans/ExamplePlan_Small1.json
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
{
|
||||||
|
"Group 1": [
|
||||||
|
{
|
||||||
|
"name": "Past event 1",
|
||||||
|
"start": 1660320408000,
|
||||||
|
"end": 1660343797000,
|
||||||
|
"type": "Group 1",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Past event 2",
|
||||||
|
"start": 1660406808000,
|
||||||
|
"end": 1660429160000,
|
||||||
|
"type": "Group 1",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Past event 3",
|
||||||
|
"start": 1660493208000,
|
||||||
|
"end": 1660503981000,
|
||||||
|
"type": "Group 1",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Past event 4",
|
||||||
|
"start": 1660579608000,
|
||||||
|
"end": 1660624108000,
|
||||||
|
"type": "Group 1",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Past event 5",
|
||||||
|
"start": 1660666008000,
|
||||||
|
"end": 1660681529000,
|
||||||
|
"type": "Group 1",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
38
e2e/test-data/examplePlans/ExamplePlan_Small2.json
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
{
|
||||||
|
"Group 1": [
|
||||||
|
{
|
||||||
|
"name": "Group 1 event 1",
|
||||||
|
"start": 1650320408000,
|
||||||
|
"end": 1660343797000,
|
||||||
|
"type": "Group 1",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Group 1 event 2",
|
||||||
|
"start": 1660005808000,
|
||||||
|
"end": 1660429160000,
|
||||||
|
"type": "Group 1",
|
||||||
|
"color": "yellow",
|
||||||
|
"textColor": "white"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Group 2": [
|
||||||
|
{
|
||||||
|
"name": "Group 2 event 1",
|
||||||
|
"start": 1660320408000,
|
||||||
|
"end": 1660420408000,
|
||||||
|
"type": "Group 2",
|
||||||
|
"color": "green",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Group 2 event 2",
|
||||||
|
"start": 1660406808000,
|
||||||
|
"end": 1690429160000,
|
||||||
|
"type": "Group 2",
|
||||||
|
"color": "blue",
|
||||||
|
"textColor": "white"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -24,18 +24,51 @@
|
|||||||
This test suite is dedicated to tests which verify Open MCT's Notification functionality
|
This test suite is dedicated to tests which verify Open MCT's Notification functionality
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// FIXME: Remove this eslint exception once tests are implemented
|
const { createDomainObjectWithDefaults, createNotification } = require('../../appActions');
|
||||||
// eslint-disable-next-line no-unused-vars
|
|
||||||
const { createDomainObjectWithDefaults } = require('../../appActions');
|
|
||||||
const { test, expect } = require('../../pluginFixtures');
|
const { test, expect } = require('../../pluginFixtures');
|
||||||
|
|
||||||
test.describe('Notifications List', () => {
|
test.describe('Notifications List', () => {
|
||||||
test.fixme('Notifications can be dismissed individually', async ({ page }) => {
|
test('Notifications can be dismissed individually', async ({ page }) => {
|
||||||
// Create some persistent notifications
|
test.info().annotations.push({
|
||||||
// Verify that they are present in the notifications list
|
type: 'issue',
|
||||||
// Dismiss one of the notifications
|
description: 'https://github.com/nasa/openmct/issues/6122'
|
||||||
// Verify that it is no longer present in the notifications list
|
});
|
||||||
// Verify that the other notifications are still present in the notifications list
|
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Create an error notification with the message "Error message"
|
||||||
|
await createNotification(page, {
|
||||||
|
severity: 'error',
|
||||||
|
message: 'Error message'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Create an alert notification with the message "Alert message"
|
||||||
|
await createNotification(page, {
|
||||||
|
severity: 'alert',
|
||||||
|
message: 'Alert message'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Verify that there is a button with aria-label "Review 2 Notifications"
|
||||||
|
expect(await page.locator('button[aria-label="Review 2 Notifications"]').count()).toBe(1);
|
||||||
|
|
||||||
|
// Click on button with aria-label "Review 2 Notifications"
|
||||||
|
await page.click('button[aria-label="Review 2 Notifications"]');
|
||||||
|
|
||||||
|
// Click on button with aria-label="Dismiss notification of Error message"
|
||||||
|
await page.click('button[aria-label="Dismiss notification of Error message"]');
|
||||||
|
|
||||||
|
// Verify there is no a notification (listitem) with the text "Error message" since it was dismissed
|
||||||
|
expect(await page.locator('div[role="dialog"] div[role="listitem"]').innerText()).not.toContain('Error message');
|
||||||
|
|
||||||
|
// Verify there is still a notification (listitem) with the text "Alert message"
|
||||||
|
expect(await page.locator('div[role="dialog"] div[role="listitem"]').innerText()).toContain('Alert message');
|
||||||
|
|
||||||
|
// Click on button with aria-label="Dismiss notification of Alert message"
|
||||||
|
await page.click('button[aria-label="Dismiss notification of Alert message"]');
|
||||||
|
|
||||||
|
// Verify that there is no dialog since the notification overlay was closed automatically after all notifications were dismissed
|
||||||
|
expect(await page.locator('div[role="dialog"]').count()).toBe(0);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
85
e2e/tests/functional/planning/ganttChart.e2e.spec.js
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2023, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
const { test, expect } = require('../../../pluginFixtures');
|
||||||
|
const { createPlanFromJSON, createDomainObjectWithDefaults, selectInspectorTab } = require('../../../appActions');
|
||||||
|
const testPlan1 = require('../../../test-data/examplePlans/ExamplePlan_Small1.json');
|
||||||
|
const testPlan2 = require('../../../test-data/examplePlans/ExamplePlan_Small2.json');
|
||||||
|
const { assertPlanActivities, setBoundsToSpanAllActivities } = require('../../../helper/planningUtils');
|
||||||
|
const { getPreciseDuration } = require('../../../../src/utils/duration');
|
||||||
|
|
||||||
|
test.describe("Gantt Chart", () => {
|
||||||
|
let ganttChart;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
ganttChart = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Gantt Chart'
|
||||||
|
});
|
||||||
|
await createPlanFromJSON(page, {
|
||||||
|
json: testPlan1,
|
||||||
|
parent: ganttChart.uuid
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Displays all plan events", async ({ page }) => {
|
||||||
|
await page.goto(ganttChart.url);
|
||||||
|
|
||||||
|
await assertPlanActivities(page, testPlan1, ganttChart.url);
|
||||||
|
});
|
||||||
|
test("Replaces a plan with a new plan", async ({ page }) => {
|
||||||
|
await assertPlanActivities(page, testPlan1, ganttChart.url);
|
||||||
|
await createPlanFromJSON(page, {
|
||||||
|
json: testPlan2,
|
||||||
|
parent: ganttChart.uuid
|
||||||
|
});
|
||||||
|
const replaceModal = page.getByRole('dialog').filter({ hasText: "This action will replace the current Plan. Do you want to continue?" });
|
||||||
|
await expect(replaceModal).toBeVisible();
|
||||||
|
await page.getByRole('button', { name: 'OK' }).click();
|
||||||
|
|
||||||
|
await assertPlanActivities(page, testPlan2, ganttChart.url);
|
||||||
|
});
|
||||||
|
test("Can select a single activity and display its details in the inspector", async ({ page }) => {
|
||||||
|
test.slow();
|
||||||
|
await page.goto(ganttChart.url);
|
||||||
|
|
||||||
|
await setBoundsToSpanAllActivities(page, testPlan1, ganttChart.url);
|
||||||
|
|
||||||
|
const activities = Object.values(testPlan1).flat();
|
||||||
|
const activity = activities[0];
|
||||||
|
await page.locator('g').filter({ hasText: new RegExp(activity.name) }).click();
|
||||||
|
await selectInspectorTab(page, 'Activity');
|
||||||
|
|
||||||
|
const startDateTime = await page.locator('.c-inspect-properties__label:has-text("Start DateTime")+.c-inspect-properties__value').innerText();
|
||||||
|
const endDateTime = await page.locator('.c-inspect-properties__label:has-text("End DateTime")+.c-inspect-properties__value').innerText();
|
||||||
|
const duration = await page.locator('.c-inspect-properties__label:has-text("duration")+.c-inspect-properties__value').innerText();
|
||||||
|
|
||||||
|
const expectedStartDate = new Date(activity.start).toISOString();
|
||||||
|
const actualStartDate = new Date(startDateTime).toISOString();
|
||||||
|
const expectedEndDate = new Date(activity.end).toISOString();
|
||||||
|
const actualEndDate = new Date(endDateTime).toISOString();
|
||||||
|
const expectedDuration = getPreciseDuration(activity.end - activity.start);
|
||||||
|
const actualDuration = duration;
|
||||||
|
|
||||||
|
expect(expectedStartDate).toEqual(actualStartDate);
|
||||||
|
expect(expectedEndDate).toEqual(actualEndDate);
|
||||||
|
expect(expectedDuration).toEqual(actualDuration);
|
||||||
|
});
|
||||||
|
});
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -19,69 +19,21 @@
|
|||||||
* this source code distribution or the Licensing information page available
|
* this source code distribution or the Licensing information page available
|
||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
const { test, expect } = require('../../../pluginFixtures');
|
const { test } = require('../../../pluginFixtures');
|
||||||
const { createPlanFromJSON } = require('../../../appActions');
|
const { createPlanFromJSON } = require('../../../appActions');
|
||||||
|
const testPlan1 = require('../../../test-data/examplePlans/ExamplePlan_Small1.json');
|
||||||
const testPlan = {
|
const { assertPlanActivities } = require('../../../helper/planningUtils');
|
||||||
"TEST_GROUP": [
|
|
||||||
{
|
|
||||||
"name": "Past event 1",
|
|
||||||
"start": 1660320408000,
|
|
||||||
"end": 1660343797000,
|
|
||||||
"type": "TEST-GROUP",
|
|
||||||
"color": "orange",
|
|
||||||
"textColor": "white"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "Past event 2",
|
|
||||||
"start": 1660406808000,
|
|
||||||
"end": 1660429160000,
|
|
||||||
"type": "TEST-GROUP",
|
|
||||||
"color": "orange",
|
|
||||||
"textColor": "white"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "Past event 3",
|
|
||||||
"start": 1660493208000,
|
|
||||||
"end": 1660503981000,
|
|
||||||
"type": "TEST-GROUP",
|
|
||||||
"color": "orange",
|
|
||||||
"textColor": "white"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "Past event 4",
|
|
||||||
"start": 1660579608000,
|
|
||||||
"end": 1660624108000,
|
|
||||||
"type": "TEST-GROUP",
|
|
||||||
"color": "orange",
|
|
||||||
"textColor": "white"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "Past event 5",
|
|
||||||
"start": 1660666008000,
|
|
||||||
"end": 1660681529000,
|
|
||||||
"type": "TEST-GROUP",
|
|
||||||
"color": "orange",
|
|
||||||
"textColor": "white"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
};
|
|
||||||
|
|
||||||
test.describe("Plan", () => {
|
test.describe("Plan", () => {
|
||||||
test("Create a Plan and display all plan events @unstable", async ({ page }) => {
|
let plan;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
await page.goto('./', { waitUntil: 'networkidle' });
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
plan = await createPlanFromJSON(page, {
|
||||||
const plan = await createPlanFromJSON(page, {
|
json: testPlan1
|
||||||
name: 'Test Plan',
|
|
||||||
json: testPlan
|
|
||||||
});
|
});
|
||||||
const startBound = testPlan.TEST_GROUP[0].start;
|
});
|
||||||
const endBound = testPlan.TEST_GROUP[testPlan.TEST_GROUP.length - 1].end;
|
|
||||||
|
|
||||||
// Switch to fixed time mode with all plan events within the bounds
|
test("Displays all plan events", async ({ page }) => {
|
||||||
await page.goto(`${plan.url}?tc.mode=fixed&tc.startBound=${startBound}&tc.endBound=${endBound}&tc.timeSystem=utc&view=plan.view`);
|
await assertPlanActivities(page, testPlan1, plan.url);
|
||||||
const eventCount = await page.locator('.activity-bounds').count();
|
|
||||||
expect(eventCount).toEqual(testPlan.TEST_GROUP.length);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -52,10 +52,9 @@ test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
|||||||
|
|
||||||
//Set object identifier from url
|
//Set object identifier from url
|
||||||
conditionSetUrl = page.url();
|
conditionSetUrl = page.url();
|
||||||
console.log('conditionSetUrl ' + conditionSetUrl);
|
|
||||||
|
|
||||||
getConditionSetIdentifierFromUrl = conditionSetUrl.split('/').pop().split('?')[0];
|
getConditionSetIdentifierFromUrl = conditionSetUrl.split('/').pop().split('?')[0];
|
||||||
console.debug('getConditionSetIdentifierFromUrl ' + getConditionSetIdentifierFromUrl);
|
console.debug(`getConditionSetIdentifierFromUrl: ${getConditionSetIdentifierFromUrl}`);
|
||||||
await page.close();
|
await page.close();
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -181,10 +180,11 @@ test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test.describe('Basic Condition Set Use', () => {
|
test.describe('Basic Condition Set Use', () => {
|
||||||
test('Can add a condition', async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
//Navigate to baseURL
|
// Open a browser, navigate to the main page, and wait until all network events to resolve
|
||||||
await page.goto('./', { waitUntil: 'networkidle' });
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
});
|
||||||
|
test('Can add a condition', async ({ page }) => {
|
||||||
// Create a new condition set
|
// Create a new condition set
|
||||||
await createDomainObjectWithDefaults(page, {
|
await createDomainObjectWithDefaults(page, {
|
||||||
type: 'Condition Set',
|
type: 'Condition Set',
|
||||||
@ -199,4 +199,127 @@ test.describe('Basic Condition Set Use', () => {
|
|||||||
const numOfUnnamedConditions = await page.locator('text=Unnamed Condition').count();
|
const numOfUnnamedConditions = await page.locator('text=Unnamed Condition').count();
|
||||||
expect(numOfUnnamedConditions).toEqual(1);
|
expect(numOfUnnamedConditions).toEqual(1);
|
||||||
});
|
});
|
||||||
|
test('ConditionSet should display appropriate view options', async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/5924'
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Sine Wave Generator',
|
||||||
|
name: "Alpha Sine Wave Generator"
|
||||||
|
});
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Sine Wave Generator',
|
||||||
|
name: "Beta Sine Wave Generator"
|
||||||
|
});
|
||||||
|
const conditionSet1 = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Condition Set',
|
||||||
|
name: "Test Condition Set"
|
||||||
|
});
|
||||||
|
|
||||||
|
// Change the object to edit mode
|
||||||
|
await page.locator('[title="Edit"]').click();
|
||||||
|
|
||||||
|
// Expand the 'My Items' folder in the left tree
|
||||||
|
await page.goto(conditionSet1.url);
|
||||||
|
page.click('button[title="Show selected item in tree"]');
|
||||||
|
// Add the Alpha & Beta Sine Wave Generator to the Condition Set and save changes
|
||||||
|
const treePane = page.getByRole('tree', {
|
||||||
|
name: 'Main Tree'
|
||||||
|
});
|
||||||
|
const alphaGeneratorTreeItem = treePane.getByRole('treeitem', { name: "Alpha Sine Wave Generator"});
|
||||||
|
const betaGeneratorTreeItem = treePane.getByRole('treeitem', { name: "Beta Sine Wave Generator"});
|
||||||
|
const conditionCollection = page.locator('#conditionCollection');
|
||||||
|
|
||||||
|
await alphaGeneratorTreeItem.dragTo(conditionCollection);
|
||||||
|
await betaGeneratorTreeItem.dragTo(conditionCollection);
|
||||||
|
|
||||||
|
const saveButtonLocator = page.locator('button[title="Save"]');
|
||||||
|
await saveButtonLocator.click();
|
||||||
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
await page.click('button[title="Change the current view"]');
|
||||||
|
|
||||||
|
await expect(page.getByRole('menuitem', { name: /Lad Table/ })).toBeHidden();
|
||||||
|
await expect(page.getByRole('menuitem', { name: /Conditions View/ })).toBeVisible();
|
||||||
|
await expect(page.getByRole('menuitem', { name: /Plot/ })).toBeVisible();
|
||||||
|
await expect(page.getByRole('menuitem', { name: /Telemetry Table/ })).toBeVisible();
|
||||||
|
});
|
||||||
|
test('ConditionSet should output blank instead of the default value', async ({ page }) => {
|
||||||
|
//Navigate to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
//Click the Create button
|
||||||
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
|
// Click the object specified by 'type'
|
||||||
|
await page.click(`li[role='menuitem']:text("Sine Wave Generator")`);
|
||||||
|
await page.getByRole('spinbutton', { name: 'Loading Delay (ms)' }).fill('8000');
|
||||||
|
const nameInput = page.locator('form[name="mctForm"] .first input[type="text"]');
|
||||||
|
await nameInput.fill("Delayed Sine Wave Generator");
|
||||||
|
|
||||||
|
// Click OK button and wait for Navigate event
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForLoadState(),
|
||||||
|
page.click('[aria-label="Save"]'),
|
||||||
|
// Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Create a new condition set
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Condition Set',
|
||||||
|
name: "Test Blank Output of Condition Set"
|
||||||
|
});
|
||||||
|
// Change the object to edit mode
|
||||||
|
await page.locator('[title="Edit"]').click();
|
||||||
|
|
||||||
|
// Click Add Condition button twice
|
||||||
|
await page.locator('#addCondition').click();
|
||||||
|
await page.locator('#addCondition').click();
|
||||||
|
await page.locator('#conditionCollection').getByRole('textbox').nth(0).fill('First Condition');
|
||||||
|
await page.locator('#conditionCollection').getByRole('textbox').nth(1).fill('Second Condition');
|
||||||
|
|
||||||
|
// Expand the 'My Items' folder in the left tree
|
||||||
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').first().click();
|
||||||
|
// Add the Sine Wave Generator to the Condition Set and save changes
|
||||||
|
const treePane = page.getByRole('tree', {
|
||||||
|
name: 'Main Tree'
|
||||||
|
});
|
||||||
|
const sineWaveGeneratorTreeItem = treePane.getByRole('treeitem', { name: "Delayed Sine Wave Generator"});
|
||||||
|
const conditionCollection = await page.locator('#conditionCollection');
|
||||||
|
|
||||||
|
await sineWaveGeneratorTreeItem.dragTo(conditionCollection);
|
||||||
|
|
||||||
|
const firstCriterionTelemetry = await page.locator('[aria-label="Criterion Telemetry Selection"] >> nth=0');
|
||||||
|
firstCriterionTelemetry.selectOption({ label: 'Delayed Sine Wave Generator' });
|
||||||
|
|
||||||
|
const secondCriterionTelemetry = await page.locator('[aria-label="Criterion Telemetry Selection"] >> nth=1');
|
||||||
|
secondCriterionTelemetry.selectOption({ label: 'Delayed Sine Wave Generator' });
|
||||||
|
|
||||||
|
const firstCriterionMetadata = await page.locator('[aria-label="Criterion Metadata Selection"] >> nth=0');
|
||||||
|
firstCriterionMetadata.selectOption({ label: 'Sine' });
|
||||||
|
|
||||||
|
const secondCriterionMetadata = await page.locator('[aria-label="Criterion Metadata Selection"] >> nth=1');
|
||||||
|
secondCriterionMetadata.selectOption({ label: 'Sine' });
|
||||||
|
|
||||||
|
const firstCriterionComparison = await page.locator('[aria-label="Criterion Comparison Selection"] >> nth=0');
|
||||||
|
firstCriterionComparison.selectOption({ label: 'is greater than or equal to' });
|
||||||
|
|
||||||
|
const secondCriterionComparison = await page.locator('[aria-label="Criterion Comparison Selection"] >> nth=1');
|
||||||
|
secondCriterionComparison.selectOption({ label: 'is less than' });
|
||||||
|
|
||||||
|
const firstCriterionInput = await page.locator('[aria-label="Criterion Input"] >> nth=0');
|
||||||
|
await firstCriterionInput.fill("0");
|
||||||
|
|
||||||
|
const secondCriterionInput = await page.locator('[aria-label="Criterion Input"] >> nth=1');
|
||||||
|
await secondCriterionInput.fill("0");
|
||||||
|
|
||||||
|
const saveButtonLocator = page.locator('button[title="Save"]');
|
||||||
|
await saveButtonLocator.click();
|
||||||
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
|
const outputValue = await page.locator('[aria-label="Current Output Value"]');
|
||||||
|
await expect(outputValue).toHaveText('---');
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -29,8 +29,8 @@ const { waitForAnimations } = require('../../../../baseFixtures');
|
|||||||
const { test, expect } = require('../../../../pluginFixtures');
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
||||||
const backgroundImageSelector = '.c-imagery__main-image__background-image';
|
const backgroundImageSelector = '.c-imagery__main-image__background-image';
|
||||||
const panHotkey = process.platform === 'linux' ? ['Control', 'Alt'] : ['Alt'];
|
const panHotkey = process.platform === 'linux' ? ['Shift', 'Alt'] : ['Alt'];
|
||||||
const expectedAltText = process.platform === 'linux' ? 'Ctrl+Alt drag to pan' : 'Alt drag to pan';
|
const expectedAltText = process.platform === 'linux' ? 'Shift+Alt drag to pan' : 'Alt drag to pan';
|
||||||
const thumbnailUrlParamsRegexp = /\?w=100&h=100/;
|
const thumbnailUrlParamsRegexp = /\?w=100&h=100/;
|
||||||
|
|
||||||
//The following block of tests verifies the basic functionality of example imagery and serves as a template for Imagery objects embedded in other objects.
|
//The following block of tests verifies the basic functionality of example imagery and serves as a template for Imagery objects embedded in other objects.
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -21,7 +21,120 @@
|
|||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
const { createDomainObjectWithDefaults, setStartOffset, setFixedTimeMode, setRealTimeMode } = require('../../../../appActions');
|
const { createDomainObjectWithDefaults, setStartOffset, setFixedTimeMode, setRealTimeMode, selectInspectorTab } = require('../../../../appActions');
|
||||||
|
|
||||||
|
test.describe('Testing LAD table configuration', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Create LAD table
|
||||||
|
const ladTable = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'LAD Table',
|
||||||
|
name: "Test LAD Table"
|
||||||
|
});
|
||||||
|
|
||||||
|
// Create Sine Wave Generator
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Sine Wave Generator',
|
||||||
|
name: "Test Sine Wave Generator",
|
||||||
|
parent: ladTable.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(ladTable.url);
|
||||||
|
});
|
||||||
|
test('in edit mode, LAD Tables provide ability to hide columns', async ({ page }) => {
|
||||||
|
// Edit LAD table
|
||||||
|
await page.locator('[title="Edit"]').click();
|
||||||
|
|
||||||
|
// // Expand the 'My Items' folder in the left tree
|
||||||
|
// await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
||||||
|
// // Add the Sine Wave Generator to the LAD table and save changes
|
||||||
|
// await page.dragAndDrop('role=treeitem[name=/Test Sine Wave Generator/]', '.c-lad-table-wrapper');
|
||||||
|
// select configuration tab in inspector
|
||||||
|
await selectInspectorTab(page, 'LAD Table Configuration');
|
||||||
|
|
||||||
|
// make sure headers are visible initially
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Type' })).toBeVisible();
|
||||||
|
|
||||||
|
// hide timestamp column
|
||||||
|
await page.getByLabel('Timestamp').uncheck();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Type' })).toBeVisible();
|
||||||
|
|
||||||
|
// hide units & type column
|
||||||
|
await page.getByLabel('Units').uncheck();
|
||||||
|
await page.getByLabel('Type').uncheck();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Type' })).toBeHidden();
|
||||||
|
|
||||||
|
// save and reload and verify they columns are still hidden
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
await page.reload();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Type' })).toBeHidden();
|
||||||
|
|
||||||
|
// Edit LAD table
|
||||||
|
await page.locator('[title="Edit"]').click();
|
||||||
|
await selectInspectorTab(page, 'LAD Table Configuration');
|
||||||
|
|
||||||
|
// show timestamp column
|
||||||
|
await page.getByLabel('Timestamp').check();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Type' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeVisible();
|
||||||
|
|
||||||
|
// save and reload and make sure only timestamp is still visible
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
await page.reload();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Type' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeVisible();
|
||||||
|
|
||||||
|
// Edit LAD table
|
||||||
|
await page.locator('[title="Edit"]').click();
|
||||||
|
await selectInspectorTab(page, 'LAD Table Configuration');
|
||||||
|
|
||||||
|
// show units and type columns
|
||||||
|
await page.getByLabel('Units').check();
|
||||||
|
await page.getByLabel('Type').check();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Type' })).toBeVisible();
|
||||||
|
|
||||||
|
// save and reload and make sure all columns are still visible
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
await page.reload();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Type' })).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('LAD Tables don\'t allow selection of rows but does show context click menus', async ({ page }) => {
|
||||||
|
const cell = await page.locator('.js-first-data');
|
||||||
|
const userSelectable = await cell.evaluate((el) => {
|
||||||
|
return window.getComputedStyle(el).getPropertyValue('user-select');
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(userSelectable).toBe('none');
|
||||||
|
// Right-click on the LAD table row
|
||||||
|
await cell.click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
const menuOptions = page.locator('.c-menu ul');
|
||||||
|
await expect.soft(menuOptions).toContainText('View Full Datum');
|
||||||
|
await expect.soft(menuOptions).toContainText('View Historical Data');
|
||||||
|
await expect.soft(menuOptions).toContainText('Remove');
|
||||||
|
// await page.locator('li[title="Remove this object from its containing object."]').click();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
test.describe('Testing LAD table @unstable', () => {
|
test.describe('Testing LAD table @unstable', () => {
|
||||||
let sineWaveObject;
|
let sineWaveObject;
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -25,8 +25,11 @@ This test suite is dedicated to tests which verify the basic operations surround
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
const { expandTreePaneItemByName, createDomainObjectWithDefaults } = require('../../../../appActions');
|
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
||||||
const nbUtils = require('../../../../helper/notebookUtils');
|
const nbUtils = require('../../../../helper/notebookUtils');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
const NOTEBOOK_NAME = 'Notebook';
|
||||||
|
|
||||||
test.describe('Notebook CRUD Operations', () => {
|
test.describe('Notebook CRUD Operations', () => {
|
||||||
test.fixme('Can create a Notebook Object', async ({ page }) => {
|
test.fixme('Can create a Notebook Object', async ({ page }) => {
|
||||||
@ -73,8 +76,7 @@ test.describe('Notebook section tests', () => {
|
|||||||
|
|
||||||
// Create Notebook
|
// Create Notebook
|
||||||
await createDomainObjectWithDefaults(page, {
|
await createDomainObjectWithDefaults(page, {
|
||||||
type: 'Notebook',
|
type: NOTEBOOK_NAME
|
||||||
name: "Test Notebook"
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
test('Default and new sections are automatically named Unnamed Section with Unnamed Page', async ({ page }) => {
|
test('Default and new sections are automatically named Unnamed Section with Unnamed Page', async ({ page }) => {
|
||||||
@ -135,8 +137,7 @@ test.describe('Notebook page tests', () => {
|
|||||||
|
|
||||||
// Create Notebook
|
// Create Notebook
|
||||||
await createDomainObjectWithDefaults(page, {
|
await createDomainObjectWithDefaults(page, {
|
||||||
type: 'Notebook',
|
type: NOTEBOOK_NAME
|
||||||
name: "Test Notebook"
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
//Test will need to be implemented after a refactor in #5713
|
//Test will need to be implemented after a refactor in #5713
|
||||||
@ -207,24 +208,30 @@ test.describe('Notebook search tests', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test.describe('Notebook entry tests', () => {
|
test.describe('Notebook entry tests', () => {
|
||||||
|
// Create Notebook with URL Whitelist
|
||||||
|
let notebookObject;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
await page.addInitScript({ path: path.join(__dirname, '../../../../helper/', 'addInitNotebookWithUrls.js') });
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
notebookObject = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: NOTEBOOK_NAME
|
||||||
|
});
|
||||||
|
});
|
||||||
test.fixme('When a new entry is created, it should be focused', async ({ page }) => {});
|
test.fixme('When a new entry is created, it should be focused', async ({ page }) => {});
|
||||||
test('When an object is dropped into a notebook, a new entry is created and it should be focused @unstable', async ({ page }) => {
|
test('When an object is dropped into a notebook, a new entry is created and it should be focused @unstable', async ({ page }) => {
|
||||||
await page.goto('./#/browse/mine', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
// Create Notebook
|
|
||||||
const notebook = await createDomainObjectWithDefaults(page, {
|
|
||||||
type: 'Notebook',
|
|
||||||
name: "Embed Test Notebook"
|
|
||||||
});
|
|
||||||
// Create Overlay Plot
|
// Create Overlay Plot
|
||||||
await createDomainObjectWithDefaults(page, {
|
await createDomainObjectWithDefaults(page, {
|
||||||
type: 'Overlay Plot',
|
type: 'Overlay Plot'
|
||||||
name: "Dropped Overlay Plot"
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await expandTreePaneItemByName(page, 'My Items');
|
// Navigate to the notebook object
|
||||||
|
await page.goto(notebookObject.url);
|
||||||
|
|
||||||
|
// Reveal the notebook in the tree
|
||||||
|
await page.getByTitle('Show selected item in tree').click();
|
||||||
|
|
||||||
await page.goto(notebook.url);
|
|
||||||
await page.dragAndDrop('role=treeitem[name=/Dropped Overlay Plot/]', '.c-notebook__drag-area');
|
await page.dragAndDrop('role=treeitem[name=/Dropped Overlay Plot/]', '.c-notebook__drag-area');
|
||||||
|
|
||||||
const embed = page.locator('.c-ne__embed__link');
|
const embed = page.locator('.c-ne__embed__link');
|
||||||
@ -234,22 +241,16 @@ test.describe('Notebook entry tests', () => {
|
|||||||
expect(embedName).toBe('Dropped Overlay Plot');
|
expect(embedName).toBe('Dropped Overlay Plot');
|
||||||
});
|
});
|
||||||
test('When an object is dropped into a notebooks existing entry, it should be focused @unstable', async ({ page }) => {
|
test('When an object is dropped into a notebooks existing entry, it should be focused @unstable', async ({ page }) => {
|
||||||
await page.goto('./#/browse/mine', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
// Create Notebook
|
|
||||||
const notebook = await createDomainObjectWithDefaults(page, {
|
|
||||||
type: 'Notebook',
|
|
||||||
name: "Embed Test Notebook"
|
|
||||||
});
|
|
||||||
// Create Overlay Plot
|
// Create Overlay Plot
|
||||||
await createDomainObjectWithDefaults(page, {
|
await createDomainObjectWithDefaults(page, {
|
||||||
type: 'Overlay Plot',
|
type: 'Overlay Plot'
|
||||||
name: "Dropped Overlay Plot"
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await expandTreePaneItemByName(page, 'My Items');
|
// Navigate to the notebook object
|
||||||
|
await page.goto(notebookObject.url);
|
||||||
|
|
||||||
await page.goto(notebook.url);
|
// Reveal the notebook in the tree
|
||||||
|
await page.getByTitle('Show selected item in tree').click();
|
||||||
|
|
||||||
await nbUtils.enterTextEntry(page, 'Entry to drop into');
|
await nbUtils.enterTextEntry(page, 'Entry to drop into');
|
||||||
await page.dragAndDrop('role=treeitem[name=/Dropped Overlay Plot/]', 'text=Entry to drop into');
|
await page.dragAndDrop('role=treeitem[name=/Dropped Overlay Plot/]', 'text=Entry to drop into');
|
||||||
@ -263,19 +264,14 @@ test.describe('Notebook entry tests', () => {
|
|||||||
});
|
});
|
||||||
test.fixme('new entries persist through navigation events without save', async ({ page }) => {});
|
test.fixme('new entries persist through navigation events without save', async ({ page }) => {});
|
||||||
test.fixme('previous and new entries can be deleted', async ({ page }) => {});
|
test.fixme('previous and new entries can be deleted', async ({ page }) => {});
|
||||||
test.fixme('when a valid link is entered into a notebook entry, it becomes clickable when viewing', async ({ page }) => {
|
test('when a valid link is entered into a notebook entry, it becomes clickable when viewing', async ({ page }) => {
|
||||||
const TEST_LINK = 'http://www.google.com';
|
const TEST_LINK = 'http://www.google.com';
|
||||||
await page.goto('./#/browse/mine', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
// Create Notebook
|
// Navigate to the notebook object
|
||||||
const notebook = await createDomainObjectWithDefaults(page, {
|
await page.goto(notebookObject.url);
|
||||||
type: 'Notebook',
|
|
||||||
name: "Entry Link Test"
|
|
||||||
});
|
|
||||||
|
|
||||||
await expandTreePaneItemByName(page, 'My Items');
|
// Reveal the notebook in the tree
|
||||||
|
await page.getByTitle('Show selected item in tree').click();
|
||||||
await page.goto(notebook.url);
|
|
||||||
|
|
||||||
await nbUtils.enterTextEntry(page, `This should be a link: ${TEST_LINK} is it?`);
|
await nbUtils.enterTextEntry(page, `This should be a link: ${TEST_LINK} is it?`);
|
||||||
|
|
||||||
@ -293,19 +289,14 @@ test.describe('Notebook entry tests', () => {
|
|||||||
|
|
||||||
expect(await validLink.count()).toBe(1);
|
expect(await validLink.count()).toBe(1);
|
||||||
});
|
});
|
||||||
test.fixme('when an invalid link is entered into a notebook entry, it does not become clickable when viewing', async ({ page }) => {
|
test('when an invalid link is entered into a notebook entry, it does not become clickable when viewing', async ({ page }) => {
|
||||||
const TEST_LINK = 'www.google.com';
|
const TEST_LINK = 'www.google.com';
|
||||||
await page.goto('./#/browse/mine', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
// Create Notebook
|
// Navigate to the notebook object
|
||||||
const notebook = await createDomainObjectWithDefaults(page, {
|
await page.goto(notebookObject.url);
|
||||||
type: 'Notebook',
|
|
||||||
name: "Entry Link Test"
|
|
||||||
});
|
|
||||||
|
|
||||||
await expandTreePaneItemByName(page, 'My Items');
|
// Reveal the notebook in the tree
|
||||||
|
await page.getByTitle('Show selected item in tree').click();
|
||||||
await page.goto(notebook.url);
|
|
||||||
|
|
||||||
await nbUtils.enterTextEntry(page, `This should NOT be a link: ${TEST_LINK} is it?`);
|
await nbUtils.enterTextEntry(page, `This should NOT be a link: ${TEST_LINK} is it?`);
|
||||||
|
|
||||||
@ -313,20 +304,70 @@ test.describe('Notebook entry tests', () => {
|
|||||||
|
|
||||||
expect(await invalidLink.count()).toBe(0);
|
expect(await invalidLink.count()).toBe(0);
|
||||||
});
|
});
|
||||||
test.fixme('when a nefarious link is entered into a notebook entry, it is sanitized when viewing', async ({ page }) => {
|
test('when a link is entered, but it is not in the whitelisted urls, it does not become clickable when viewing', async ({ page }) => {
|
||||||
|
const TEST_LINK = 'http://www.bing.com';
|
||||||
|
|
||||||
|
// Navigate to the notebook object
|
||||||
|
await page.goto(notebookObject.url);
|
||||||
|
|
||||||
|
// Reveal the notebook in the tree
|
||||||
|
await page.getByTitle('Show selected item in tree').click();
|
||||||
|
|
||||||
|
await nbUtils.enterTextEntry(page, `This should NOT be a link: ${TEST_LINK} is it?`);
|
||||||
|
|
||||||
|
const invalidLink = page.locator(`a[href="${TEST_LINK}"]`);
|
||||||
|
|
||||||
|
expect(await invalidLink.count()).toBe(0);
|
||||||
|
});
|
||||||
|
test('when a valid link with a subdomain and a valid domain in the whitelisted urls is entered into a notebook entry, it becomes clickable when viewing', async ({ page }) => {
|
||||||
|
const INVALID_TEST_LINK = 'http://bing.google.com';
|
||||||
|
|
||||||
|
// Navigate to the notebook object
|
||||||
|
await page.goto(notebookObject.url);
|
||||||
|
|
||||||
|
// Reveal the notebook in the tree
|
||||||
|
await page.getByTitle('Show selected item in tree').click();
|
||||||
|
|
||||||
|
await nbUtils.enterTextEntry(page, `This should be a link: ${INVALID_TEST_LINK} is it?`);
|
||||||
|
|
||||||
|
const validLink = page.locator(`a[href="${INVALID_TEST_LINK}"]`);
|
||||||
|
|
||||||
|
expect(await validLink.count()).toBe(1);
|
||||||
|
});
|
||||||
|
test('when a valid secure link is entered into a notebook entry, it becomes clickable when viewing', async ({ page }) => {
|
||||||
|
const TEST_LINK = 'https://www.google.com';
|
||||||
|
|
||||||
|
// Navigate to the notebook object
|
||||||
|
await page.goto(notebookObject.url);
|
||||||
|
|
||||||
|
// Reveal the notebook in the tree
|
||||||
|
await page.getByTitle('Show selected item in tree').click();
|
||||||
|
|
||||||
|
await nbUtils.enterTextEntry(page, `This should be a link: ${TEST_LINK} is it?`);
|
||||||
|
|
||||||
|
const validLink = page.locator(`a[href="${TEST_LINK}"]`);
|
||||||
|
|
||||||
|
// Start waiting for popup before clicking. Note no await.
|
||||||
|
const popupPromise = page.waitForEvent('popup');
|
||||||
|
|
||||||
|
await validLink.click();
|
||||||
|
const popup = await popupPromise;
|
||||||
|
|
||||||
|
// Wait for the popup to load.
|
||||||
|
await popup.waitForLoadState();
|
||||||
|
expect.soft(popup.url()).toContain('www.google.com');
|
||||||
|
|
||||||
|
expect(await validLink.count()).toBe(1);
|
||||||
|
});
|
||||||
|
test('when a nefarious link is entered into a notebook entry, it is sanitized when viewing', async ({ page }) => {
|
||||||
const TEST_LINK = 'http://www.google.com?bad=';
|
const TEST_LINK = 'http://www.google.com?bad=';
|
||||||
const TEST_LINK_BAD = `http://www.google.com?bad=<script>alert('gimme your cookies')</script>`;
|
const TEST_LINK_BAD = `http://www.google.com?bad=<script>alert('gimme your cookies')</script>`;
|
||||||
await page.goto('./#/browse/mine', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
// Create Notebook
|
// Navigate to the notebook object
|
||||||
const notebook = await createDomainObjectWithDefaults(page, {
|
await page.goto(notebookObject.url);
|
||||||
type: 'Notebook',
|
|
||||||
name: "Entry Link Test"
|
|
||||||
});
|
|
||||||
|
|
||||||
await expandTreePaneItemByName(page, 'My Items');
|
// Reveal the notebook in the tree
|
||||||
|
await page.getByTitle('Show selected item in tree').click();
|
||||||
await page.goto(notebook.url);
|
|
||||||
|
|
||||||
await nbUtils.enterTextEntry(page, `This should be a link, BUT not a bad link: ${TEST_LINK_BAD} is it?`);
|
await nbUtils.enterTextEntry(page, `This should be a link, BUT not a bad link: ${TEST_LINK_BAD} is it?`);
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -41,6 +41,7 @@ test.describe('Notebook Tests with CouchDB @couchdb', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test('Inspect Notebook Entry Network Requests', async ({ page }) => {
|
test('Inspect Notebook Entry Network Requests', async ({ page }) => {
|
||||||
|
await page.getByText('Annotations').click();
|
||||||
// Expand sidebar
|
// Expand sidebar
|
||||||
await page.locator('.c-notebook__toggle-nav-button').click();
|
await page.locator('.c-notebook__toggle-nav-button').click();
|
||||||
|
|
||||||
@ -130,13 +131,13 @@ test.describe('Notebook Tests with CouchDB @couchdb', () => {
|
|||||||
// This happens for 3 tags so 12 requests
|
// This happens for 3 tags so 12 requests
|
||||||
addingNotebookElementsRequests = [];
|
addingNotebookElementsRequests = [];
|
||||||
await page.hover('[aria-label="Tag"]:has-text("Driving")');
|
await page.hover('[aria-label="Tag"]:has-text("Driving")');
|
||||||
await page.locator('[aria-label="Tag"]:has-text("Driving") ~ .c-completed-tag-deletion').click();
|
await page.locator('[aria-label="Remove tag Driving"]').click();
|
||||||
await page.waitForSelector('[aria-label="Tag"]:has-text("Driving")', {state: 'hidden'});
|
await page.waitForSelector('[aria-label="Tag"]:has-text("Driving")', {state: 'hidden'});
|
||||||
await page.hover('[aria-label="Tag"]:has-text("Drilling")');
|
await page.hover('[aria-label="Tag"]:has-text("Drilling")');
|
||||||
await page.locator('[aria-label="Tag"]:has-text("Drilling") ~ .c-completed-tag-deletion').click();
|
await page.locator('[aria-label="Remove tag Drilling"]').click();
|
||||||
await page.waitForSelector('[aria-label="Tag"]:has-text("Drilling")', {state: 'hidden'});
|
await page.waitForSelector('[aria-label="Tag"]:has-text("Drilling")', {state: 'hidden'});
|
||||||
page.hover('[aria-label="Tag"]:has-text("Science")');
|
page.hover('[aria-label="Tag"]:has-text("Science")');
|
||||||
await page.locator('[aria-label="Tag"]:has-text("Science") ~ .c-completed-tag-deletion').click();
|
await page.locator('[aria-label="Remove tag Science"]').click();
|
||||||
await page.waitForSelector('[aria-label="Tag"]:has-text("Science")', {state: 'hidden'});
|
await page.waitForSelector('[aria-label="Tag"]:has-text("Science")', {state: 'hidden'});
|
||||||
page.waitForLoadState('networkidle');
|
page.waitForLoadState('networkidle');
|
||||||
expect(filterNonFetchRequests(addingNotebookElementsRequests).length).toBeLessThanOrEqual(12);
|
expect(filterNonFetchRequests(addingNotebookElementsRequests).length).toBeLessThanOrEqual(12);
|
||||||
@ -162,20 +163,20 @@ test.describe('Notebook Tests with CouchDB @couchdb', () => {
|
|||||||
await page.locator('[aria-label="Notebook Entry Input"] >> nth=2').press('Enter');
|
await page.locator('[aria-label="Notebook Entry Input"] >> nth=2').press('Enter');
|
||||||
|
|
||||||
// Add three tags
|
// Add three tags
|
||||||
await page.hover(`button:has-text("Add Tag") >> nth=2`);
|
await page.hover(`button:has-text("Add Tag")`);
|
||||||
await page.locator(`button:has-text("Add Tag") >> nth=2`).click();
|
await page.locator(`button:has-text("Add Tag")`).click();
|
||||||
await page.locator('[placeholder="Type to select tag"]').click();
|
await page.locator('[placeholder="Type to select tag"]').click();
|
||||||
await page.locator('[aria-label="Autocomplete Options"] >> text=Science').click();
|
await page.locator('[aria-label="Autocomplete Options"] >> text=Science').click();
|
||||||
await page.waitForSelector('[aria-label="Tag"]:has-text("Science")');
|
await page.waitForSelector('[aria-label="Tag"]:has-text("Science")');
|
||||||
|
|
||||||
await page.hover(`button:has-text("Add Tag") >> nth=2`);
|
await page.hover(`button:has-text("Add Tag")`);
|
||||||
await page.locator(`button:has-text("Add Tag") >> nth=2`).click();
|
await page.locator(`button:has-text("Add Tag")`).click();
|
||||||
await page.locator('[placeholder="Type to select tag"]').click();
|
await page.locator('[placeholder="Type to select tag"]').click();
|
||||||
await page.locator('[aria-label="Autocomplete Options"] >> text=Drilling').click();
|
await page.locator('[aria-label="Autocomplete Options"] >> text=Drilling').click();
|
||||||
await page.waitForSelector('[aria-label="Tag"]:has-text("Drilling")');
|
await page.waitForSelector('[aria-label="Tag"]:has-text("Drilling")');
|
||||||
|
|
||||||
await page.hover(`button:has-text("Add Tag") >> nth=2`);
|
await page.hover(`button:has-text("Add Tag")`);
|
||||||
await page.locator(`button:has-text("Add Tag") >> nth=2`).click();
|
await page.locator(`button:has-text("Add Tag")`).click();
|
||||||
await page.locator('[placeholder="Type to select tag"]').click();
|
await page.locator('[placeholder="Type to select tag"]').click();
|
||||||
await page.locator('[aria-label="Autocomplete Options"] >> text=Driving').click();
|
await page.locator('[aria-label="Autocomplete Options"] >> text=Driving').click();
|
||||||
await page.waitForSelector('[aria-label="Tag"]:has-text("Driving")');
|
await page.waitForSelector('[aria-label="Tag"]:has-text("Driving")');
|
||||||
@ -231,6 +232,7 @@ test.describe('Notebook Tests with CouchDB @couchdb', () => {
|
|||||||
type: 'issue',
|
type: 'issue',
|
||||||
description: 'https://github.com/akhenry/openmct-yamcs/issues/69'
|
description: 'https://github.com/akhenry/openmct-yamcs/issues/69'
|
||||||
});
|
});
|
||||||
|
await page.getByText('Annotations').click();
|
||||||
await page.locator('text=To start a new entry, click here or drag and drop any object').click();
|
await page.locator('text=To start a new entry, click here or drag and drop any object').click();
|
||||||
await page.locator('[aria-label="Notebook Entry Input"]').click();
|
await page.locator('[aria-label="Notebook Entry Input"]').click();
|
||||||
await page.locator('[aria-label="Notebook Entry Input"]').fill(`First Entry`);
|
await page.locator('[aria-label="Notebook Entry Input"]').fill(`First Entry`);
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -25,7 +25,8 @@ This test suite is dedicated to tests which verify form functionality.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
const { createDomainObjectWithDefaults, selectInspectorTab } = require('../../../../appActions');
|
||||||
|
const nbUtils = require('../../../../helper/notebookUtils');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a notebook object and adds an entry.
|
* Creates a notebook object and adds an entry.
|
||||||
@ -39,12 +40,7 @@ async function createNotebookAndEntry(page, iterations = 1) {
|
|||||||
const notebook = createDomainObjectWithDefaults(page, { type: 'Notebook' });
|
const notebook = createDomainObjectWithDefaults(page, { type: 'Notebook' });
|
||||||
|
|
||||||
for (let iteration = 0; iteration < iterations; iteration++) {
|
for (let iteration = 0; iteration < iterations; iteration++) {
|
||||||
// Create an entry
|
await nbUtils.enterTextEntry(page, `Entry ${iteration}`);
|
||||||
await page.locator('text=To start a new entry, click here or drag and drop any object').click();
|
|
||||||
const entryLocator = `[aria-label="Notebook Entry Input"] >> nth = ${iteration}`;
|
|
||||||
await page.locator(entryLocator).click();
|
|
||||||
await page.locator(entryLocator).fill(`Entry ${iteration}`);
|
|
||||||
await page.locator(entryLocator).press('Enter');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return notebook;
|
return notebook;
|
||||||
@ -57,7 +53,7 @@ async function createNotebookAndEntry(page, iterations = 1) {
|
|||||||
*/
|
*/
|
||||||
async function createNotebookEntryAndTags(page, iterations = 1) {
|
async function createNotebookEntryAndTags(page, iterations = 1) {
|
||||||
const notebook = await createNotebookAndEntry(page, iterations);
|
const notebook = await createNotebookAndEntry(page, iterations);
|
||||||
await page.locator('text=Annotations').click();
|
await selectInspectorTab(page, 'Annotations');
|
||||||
|
|
||||||
for (let iteration = 0; iteration < iterations; iteration++) {
|
for (let iteration = 0; iteration < iterations; iteration++) {
|
||||||
// Hover and click "Add Tag" button
|
// Hover and click "Add Tag" button
|
||||||
@ -88,7 +84,10 @@ test.describe('Tagging in Notebooks @addInit', () => {
|
|||||||
test('Can load tags', async ({ page }) => {
|
test('Can load tags', async ({ page }) => {
|
||||||
await createNotebookAndEntry(page);
|
await createNotebookAndEntry(page);
|
||||||
|
|
||||||
await page.locator('text=Annotations').click();
|
// TODO can be removed with fix for https://github.com/nasa/openmct/issues/6411
|
||||||
|
await page.locator('[aria-label="Notebook Entry"].is-selected div.c-ne__text').click();
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Annotations');
|
||||||
|
|
||||||
await page.locator('button:has-text("Add Tag")').click();
|
await page.locator('button:has-text("Add Tag")').click();
|
||||||
|
|
||||||
@ -111,7 +110,31 @@ test.describe('Tagging in Notebooks @addInit', () => {
|
|||||||
await expect(page.locator('[aria-label="Autocomplete Options"]')).not.toContainText("Driving");
|
await expect(page.locator('[aria-label="Autocomplete Options"]')).not.toContainText("Driving");
|
||||||
await expect(page.locator('[aria-label="Autocomplete Options"]')).toContainText("Drilling");
|
await expect(page.locator('[aria-label="Autocomplete Options"]')).toContainText("Drilling");
|
||||||
});
|
});
|
||||||
test('Can search for tags', async ({ page }) => {
|
test('Can cancel adding tags', async ({ page }) => {
|
||||||
|
await createNotebookAndEntry(page);
|
||||||
|
|
||||||
|
// TODO can be removed with fix for https://github.com/nasa/openmct/issues/6411
|
||||||
|
await page.locator('[aria-label="Notebook Entry"].is-selected div.c-ne__text').click();
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Annotations');
|
||||||
|
|
||||||
|
// Test canceling adding a tag after we click "Type to select tag"
|
||||||
|
await page.locator('button:has-text("Add Tag")').click();
|
||||||
|
|
||||||
|
await page.locator('[placeholder="Type to select tag"]').click();
|
||||||
|
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
|
||||||
|
await expect(page.locator('button:has-text("Add Tag")')).toBeVisible();
|
||||||
|
|
||||||
|
// Test canceling adding a tag after we just click "Add Tag"
|
||||||
|
await page.locator('button:has-text("Add Tag")').click();
|
||||||
|
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
|
||||||
|
await expect(page.locator('button:has-text("Add Tag")')).toBeVisible();
|
||||||
|
});
|
||||||
|
test('Can search for tags and preview works properly', async ({ page }) => {
|
||||||
await createNotebookEntryAndTags(page);
|
await createNotebookEntryAndTags(page);
|
||||||
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('sc');
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('sc');
|
||||||
@ -126,13 +149,26 @@ test.describe('Tagging in Notebooks @addInit', () => {
|
|||||||
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Xq');
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Xq');
|
||||||
await expect(page.locator('text=No results found')).toBeVisible();
|
await expect(page.locator('text=No results found')).toBeVisible();
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Display Layout'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Go back into edit mode for the display layout
|
||||||
|
await page.locator('button[title="Edit"]').click();
|
||||||
|
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Sc');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"]')).toContainText("Science");
|
||||||
|
await page.getByText('Entry 0').click();
|
||||||
|
await expect(page.locator('.js-preview-window')).toBeVisible();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('Can delete tags', async ({ page }) => {
|
test('Can delete tags', async ({ page }) => {
|
||||||
await createNotebookEntryAndTags(page);
|
await createNotebookEntryAndTags(page);
|
||||||
// Delete Driving
|
// Delete Driving
|
||||||
await page.hover('[aria-label="Tag"]:has-text("Driving")');
|
await page.hover('[aria-label="Tag"]:has-text("Driving")');
|
||||||
await page.locator('[aria-label="Tag"]:has-text("Driving") ~ .c-completed-tag-deletion').click();
|
await page.locator('[aria-label="Remove tag Driving"]').click();
|
||||||
|
|
||||||
await expect(page.locator('[aria-label="Tags Inspector"]')).toContainText("Science");
|
await expect(page.locator('[aria-label="Tags Inspector"]')).toContainText("Science");
|
||||||
await expect(page.locator('[aria-label="Tags Inspector"]')).not.toContainText("Driving");
|
await expect(page.locator('[aria-label="Tags Inspector"]')).not.toContainText("Driving");
|
||||||
@ -231,4 +267,27 @@ test.describe('Tagging in Notebooks @addInit', () => {
|
|||||||
await expect(page.locator(entryLocator)).toContainText("Driving");
|
await expect(page.locator(entryLocator)).toContainText("Driving");
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
test('Can cancel adding a tag', async ({ page }) => {
|
||||||
|
await createNotebookAndEntry(page);
|
||||||
|
|
||||||
|
// TODO can be removed with fix for https://github.com/nasa/openmct/issues/6411
|
||||||
|
await page.locator('[aria-label="Notebook Entry"].is-selected div.c-ne__text').click();
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Annotations');
|
||||||
|
|
||||||
|
// Click on the "Add Tag" button
|
||||||
|
await page.locator('button:has-text("Add Tag")').click();
|
||||||
|
|
||||||
|
// Click inside the AutoComplete field
|
||||||
|
await page.locator('[placeholder="Type to select tag"]').click();
|
||||||
|
|
||||||
|
// Click on the "Tags" header (simulating a click outside the autocomplete)
|
||||||
|
await page.locator('div.c-inspect-properties__header:has-text("Tags")').click();
|
||||||
|
|
||||||
|
// Verify there is a button with text "Add Tag"
|
||||||
|
await expect(page.locator('button:has-text("Add Tag")')).toBeVisible();
|
||||||
|
|
||||||
|
// Verify the AutoComplete field is hidden
|
||||||
|
await expect(page.locator('[placeholder="Type to select tag"]')).toBeHidden();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -24,6 +24,7 @@
|
|||||||
Testsuite for plot autoscale.
|
Testsuite for plot autoscale.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
const { selectInspectorTab } = require('../../../../appActions');
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
test.use({
|
test.use({
|
||||||
viewport: {
|
viewport: {
|
||||||
@ -32,7 +33,7 @@ test.use({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
test.fixme('ExportAsJSON', () => {
|
test.describe('Autoscale', () => {
|
||||||
test('User can set autoscale with a valid range @snapshot', async ({ page, openmctConfig }) => {
|
test('User can set autoscale with a valid range @snapshot', async ({ page, openmctConfig }) => {
|
||||||
const { myItemsFolderName } = openmctConfig;
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
|
||||||
@ -47,16 +48,33 @@ test.fixme('ExportAsJSON', () => {
|
|||||||
|
|
||||||
await testYTicks(page, ['-1.00', '-0.50', '0.00', '0.50', '1.00']);
|
await testYTicks(page, ['-1.00', '-0.50', '0.00', '0.50', '1.00']);
|
||||||
|
|
||||||
|
// enter edit mode
|
||||||
|
await page.click('button[title="Edit"]');
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Config');
|
||||||
await turnOffAutoscale(page);
|
await turnOffAutoscale(page);
|
||||||
|
|
||||||
// Make sure that after turning off autoscale, the user selected range values start at the same values the plot had prior.
|
await setUserDefinedMinAndMax(page, '-2', '2');
|
||||||
await testYTicks(page, ['-1.00', '-0.50', '0.00', '0.50', '1.00']);
|
|
||||||
|
// save
|
||||||
|
await page.click('button[title="Save"]');
|
||||||
|
await Promise.all([
|
||||||
|
page.locator('li[title = "Save and Finish Editing"]').click(),
|
||||||
|
//Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
//Wait until Save Banner is gone
|
||||||
|
await page.locator('.c-message-banner__close-button').click();
|
||||||
|
await page.waitForSelector('.c-message-banner__message', { state: 'detached'});
|
||||||
|
|
||||||
|
// Make sure that after turning off autoscale, the user entered range values are reflexted in the ticks.
|
||||||
|
await testYTicks(page, ['-2.00', '-1.50', '-1.00', '-0.50', '0.00', '0.50', '1.00', '1.50', '2.00']);
|
||||||
|
|
||||||
const canvas = page.locator('canvas').nth(1);
|
const canvas = page.locator('canvas').nth(1);
|
||||||
|
|
||||||
await canvas.hover({trial: true});
|
await canvas.hover({trial: true});
|
||||||
|
|
||||||
expect(await canvas.screenshot()).toMatchSnapshot('autoscale-canvas-prepan.png', { animations: 'disabled' });
|
expect.soft(await canvas.screenshot()).toMatchSnapshot('autoscale-canvas-prepan.png', { animations: 'disabled' });
|
||||||
|
|
||||||
//Alt Drag Start
|
//Alt Drag Start
|
||||||
await page.keyboard.down('Alt');
|
await page.keyboard.down('Alt');
|
||||||
@ -76,11 +94,12 @@ test.fixme('ExportAsJSON', () => {
|
|||||||
await page.keyboard.up('Alt');
|
await page.keyboard.up('Alt');
|
||||||
|
|
||||||
// Ensure the drag worked.
|
// Ensure the drag worked.
|
||||||
await testYTicks(page, ['0.00', '0.50', '1.00', '1.50', '2.00']);
|
await testYTicks(page, ['0.00', '0.50', '1.00', '1.50', '2.00', '2.50', '3.00', '3.50']);
|
||||||
|
|
||||||
|
//Wait for canvas to stablize.
|
||||||
await canvas.hover({trial: true});
|
await canvas.hover({trial: true});
|
||||||
|
|
||||||
expect(await canvas.screenshot()).toMatchSnapshot('autoscale-canvas-panned.png', { animations: 'disabled' });
|
expect.soft(await canvas.screenshot()).toMatchSnapshot('autoscale-canvas-panned.png', { animations: 'disabled' });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -152,22 +171,25 @@ async function createSinewaveOverlayPlot(page, myItemsFolderName) {
|
|||||||
* @param {import('@playwright/test').Page} page
|
* @param {import('@playwright/test').Page} page
|
||||||
*/
|
*/
|
||||||
async function turnOffAutoscale(page) {
|
async function turnOffAutoscale(page) {
|
||||||
// enter edit mode
|
|
||||||
await page.locator('text=Unnamed Overlay Plot Snapshot >> button').nth(3).click();
|
|
||||||
|
|
||||||
// uncheck autoscale
|
// uncheck autoscale
|
||||||
await page.getByRole('listitem').filter({ hasText: 'Auto scale' }).getByRole('checkbox').uncheck();
|
await page.getByRole('listitem').filter({ hasText: 'Auto scale' }).getByRole('checkbox').uncheck();
|
||||||
|
}
|
||||||
|
|
||||||
// save
|
/**
|
||||||
await page.locator('text=Snapshot Save and Finish Editing Save and Continue Editing >> button').nth(1).click();
|
* @param {import('@playwright/test').Page} page
|
||||||
await Promise.all([
|
* @param {string} min
|
||||||
page.locator('text=Save and Finish Editing').click(),
|
* @param {string} max
|
||||||
//Wait for Save Banner to appear
|
*/
|
||||||
page.waitForSelector('.c-message-banner__message')
|
async function setUserDefinedMinAndMax(page, min, max) {
|
||||||
]);
|
// set minimum value
|
||||||
//Wait until Save Banner is gone
|
const minRangeInput = page.getByRole('listitem').filter({ hasText: 'Minimum Value' }).locator('input[type="number"]');
|
||||||
await page.locator('.c-message-banner__close-button').click();
|
await minRangeInput.click();
|
||||||
await page.waitForSelector('.c-message-banner__message', { state: 'detached'});
|
await minRangeInput.fill(min);
|
||||||
|
|
||||||
|
// set maximum value
|
||||||
|
const maxRangeInput = page.getByRole('listitem').filter({ hasText: 'Maximum Value' }).locator('input[type="number"]');
|
||||||
|
await maxRangeInput.click();
|
||||||
|
await maxRangeInput.fill(max);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -179,7 +201,7 @@ async function testYTicks(page, values) {
|
|||||||
let promises = [yTicks.count().then(c => expect(c).toBe(values.length))];
|
let promises = [yTicks.count().then(c => expect(c).toBe(values.length))];
|
||||||
|
|
||||||
for (let i = 0, l = values.length; i < l; i += 1) {
|
for (let i = 0, l = values.length; i < l; i += 1) {
|
||||||
promises.push(expect(yTicks.nth(i)).toHaveText(values[i])); // eslint-disable-line
|
promises.push(expect.soft(yTicks.nth(i)).toHaveText(values[i])); // eslint-disable-line
|
||||||
}
|
}
|
||||||
|
|
||||||
await Promise.all(promises);
|
await Promise.all(promises);
|
||||||
|
Before Width: | Height: | Size: 16 KiB After Width: | Height: | Size: 19 KiB |
Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 17 KiB |
Before Width: | Height: | Size: 18 KiB After Width: | Height: | Size: 19 KiB |
Before Width: | Height: | Size: 18 KiB After Width: | Height: | Size: 19 KiB |
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -26,6 +26,8 @@ necessarily be used for reference when writing new tests in this area.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
const { selectInspectorTab } = require('../../../../appActions');
|
||||||
|
|
||||||
test.describe('Log plot tests', () => {
|
test.describe('Log plot tests', () => {
|
||||||
test('Log Plot ticks are functionally correct in regular and log mode and after refresh', async ({ page, openmctConfig }) => {
|
test('Log Plot ticks are functionally correct in regular and log mode and after refresh', async ({ page, openmctConfig }) => {
|
||||||
const { myItemsFolderName } = openmctConfig;
|
const { myItemsFolderName } = openmctConfig;
|
||||||
@ -36,6 +38,7 @@ test.describe('Log plot tests', () => {
|
|||||||
await makeOverlayPlot(page, myItemsFolderName);
|
await makeOverlayPlot(page, myItemsFolderName);
|
||||||
await testRegularTicks(page);
|
await testRegularTicks(page);
|
||||||
await enableEditMode(page);
|
await enableEditMode(page);
|
||||||
|
await selectInspectorTab(page, 'Config');
|
||||||
await enableLogMode(page);
|
await enableLogMode(page);
|
||||||
await testLogTicks(page);
|
await testLogTicks(page);
|
||||||
await disableLogMode(page);
|
await disableLogMode(page);
|
||||||
@ -160,35 +163,16 @@ async function testRegularTicks(page) {
|
|||||||
*/
|
*/
|
||||||
async function testLogTicks(page) {
|
async function testLogTicks(page) {
|
||||||
const yTicks = await page.locator('.gl-plot-y-tick-label');
|
const yTicks = await page.locator('.gl-plot-y-tick-label');
|
||||||
expect(await yTicks.count()).toBe(28);
|
expect(await yTicks.count()).toBe(9);
|
||||||
await expect(yTicks.nth(0)).toHaveText('-2.98');
|
await expect(yTicks.nth(0)).toHaveText('-2.98');
|
||||||
await expect(yTicks.nth(1)).toHaveText('-2.50');
|
await expect(yTicks.nth(1)).toHaveText('-1.51');
|
||||||
await expect(yTicks.nth(2)).toHaveText('-2.00');
|
await expect(yTicks.nth(2)).toHaveText('-0.58');
|
||||||
await expect(yTicks.nth(3)).toHaveText('-1.51');
|
await expect(yTicks.nth(3)).toHaveText('-0.00');
|
||||||
await expect(yTicks.nth(4)).toHaveText('-1.20');
|
await expect(yTicks.nth(4)).toHaveText('0.58');
|
||||||
await expect(yTicks.nth(5)).toHaveText('-1.00');
|
await expect(yTicks.nth(5)).toHaveText('1.51');
|
||||||
await expect(yTicks.nth(6)).toHaveText('-0.80');
|
await expect(yTicks.nth(6)).toHaveText('2.98');
|
||||||
await expect(yTicks.nth(7)).toHaveText('-0.58');
|
await expect(yTicks.nth(7)).toHaveText('5.31');
|
||||||
await expect(yTicks.nth(8)).toHaveText('-0.40');
|
await expect(yTicks.nth(8)).toHaveText('9.00');
|
||||||
await expect(yTicks.nth(9)).toHaveText('-0.20');
|
|
||||||
await expect(yTicks.nth(10)).toHaveText('-0.00');
|
|
||||||
await expect(yTicks.nth(11)).toHaveText('0.20');
|
|
||||||
await expect(yTicks.nth(12)).toHaveText('0.40');
|
|
||||||
await expect(yTicks.nth(13)).toHaveText('0.58');
|
|
||||||
await expect(yTicks.nth(14)).toHaveText('0.80');
|
|
||||||
await expect(yTicks.nth(15)).toHaveText('1.00');
|
|
||||||
await expect(yTicks.nth(16)).toHaveText('1.20');
|
|
||||||
await expect(yTicks.nth(17)).toHaveText('1.51');
|
|
||||||
await expect(yTicks.nth(18)).toHaveText('2.00');
|
|
||||||
await expect(yTicks.nth(19)).toHaveText('2.50');
|
|
||||||
await expect(yTicks.nth(20)).toHaveText('2.98');
|
|
||||||
await expect(yTicks.nth(21)).toHaveText('3.50');
|
|
||||||
await expect(yTicks.nth(22)).toHaveText('4.00');
|
|
||||||
await expect(yTicks.nth(23)).toHaveText('4.50');
|
|
||||||
await expect(yTicks.nth(24)).toHaveText('5.31');
|
|
||||||
await expect(yTicks.nth(25)).toHaveText('7.00');
|
|
||||||
await expect(yTicks.nth(26)).toHaveText('8.00');
|
|
||||||
await expect(yTicks.nth(27)).toHaveText('9.00');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -205,6 +189,7 @@ async function enableEditMode(page) {
|
|||||||
*/
|
*/
|
||||||
async function enableLogMode(page) {
|
async function enableLogMode(page) {
|
||||||
// turn on log mode
|
// turn on log mode
|
||||||
|
await expect(page.getByRole('listitem').filter({ hasText: 'Log mode' }).getByRole('checkbox')).not.toBeChecked();
|
||||||
await page.getByRole('listitem').filter({ hasText: 'Log mode' }).getByRole('checkbox').check();
|
await page.getByRole('listitem').filter({ hasText: 'Log mode' }).getByRole('checkbox').check();
|
||||||
// await page.locator('text=Y Axis Label Log mode Auto scale Padding >> input[type="checkbox"]').first().check();
|
// await page.locator('text=Y Axis Label Log mode Auto scale Padding >> input[type="checkbox"]').first().check();
|
||||||
}
|
}
|
||||||
@ -214,6 +199,7 @@ async function enableLogMode(page) {
|
|||||||
*/
|
*/
|
||||||
async function disableLogMode(page) {
|
async function disableLogMode(page) {
|
||||||
// turn off log mode
|
// turn off log mode
|
||||||
|
await expect(page.getByRole('listitem').filter({ hasText: 'Log mode' }).getByRole('checkbox')).toBeChecked();
|
||||||
await page.getByRole('listitem').filter({ hasText: 'Log mode' }).getByRole('checkbox').uncheck();
|
await page.getByRole('listitem').filter({ hasText: 'Log mode' }).getByRole('checkbox').uncheck();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -26,11 +26,14 @@ necessarily be used for reference when writing new tests in this area.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
const { createDomainObjectWithDefaults, selectInspectorTab } = require('../../../../appActions');
|
||||||
|
|
||||||
test.describe('Overlay Plot', () => {
|
test.describe('Overlay Plot', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
});
|
||||||
|
|
||||||
test('Plot legend color is in sync with plot series color', async ({ page }) => {
|
test('Plot legend color is in sync with plot series color', async ({ page }) => {
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
|
||||||
const overlayPlot = await createDomainObjectWithDefaults(page, {
|
const overlayPlot = await createDomainObjectWithDefaults(page, {
|
||||||
type: "Overlay Plot"
|
type: "Overlay Plot"
|
||||||
});
|
});
|
||||||
@ -42,6 +45,8 @@ test.describe('Overlay Plot', () => {
|
|||||||
|
|
||||||
await page.goto(overlayPlot.url);
|
await page.goto(overlayPlot.url);
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Config');
|
||||||
|
|
||||||
// navigate to plot series color palette
|
// navigate to plot series color palette
|
||||||
await page.click('.l-browse-bar__actions__edit');
|
await page.click('.l-browse-bar__actions__edit');
|
||||||
await page.locator('li.c-tree__item.menus-to-left .c-disclosure-triangle').click();
|
await page.locator('li.c-tree__item.menus-to-left .c-disclosure-triangle').click();
|
||||||
@ -56,69 +61,221 @@ test.describe('Overlay Plot', () => {
|
|||||||
|
|
||||||
expect(color).toBe('rgb(255, 166, 61)');
|
expect(color).toBe('rgb(255, 166, 61)');
|
||||||
});
|
});
|
||||||
test('The elements pool supports dragging series into multiple y-axis buckets', async ({ page }) => {
|
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
test('Limit lines persist when series is moved to another Y Axis and on refresh', async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/6338'
|
||||||
|
});
|
||||||
|
// Create an Overlay Plot with a default SWG
|
||||||
const overlayPlot = await createDomainObjectWithDefaults(page, {
|
const overlayPlot = await createDomainObjectWithDefaults(page, {
|
||||||
type: "Overlay Plot"
|
type: "Overlay Plot"
|
||||||
});
|
});
|
||||||
|
|
||||||
await createDomainObjectWithDefaults(page, {
|
const swgA = await createDomainObjectWithDefaults(page, {
|
||||||
type: "Sine Wave Generator",
|
type: "Sine Wave Generator",
|
||||||
name: 'swg a',
|
|
||||||
parent: overlayPlot.uuid
|
parent: overlayPlot.uuid
|
||||||
});
|
});
|
||||||
await createDomainObjectWithDefaults(page, {
|
|
||||||
|
await page.goto(overlayPlot.url);
|
||||||
|
|
||||||
|
// Assert that no limit lines are shown by default
|
||||||
|
await page.waitForSelector('.js-limit-area', { state: 'attached' });
|
||||||
|
expect(await page.locator('.c-plot-limit-line').count()).toBe(0);
|
||||||
|
|
||||||
|
// Enter edit mode
|
||||||
|
await page.click('button[title="Edit"]');
|
||||||
|
|
||||||
|
// Expand the "Sine Wave Generator" plot series options and enable limit lines
|
||||||
|
await selectInspectorTab(page, 'Config');
|
||||||
|
await page.getByRole('list', { name: 'Plot Series Properties' }).locator('span').first().click();
|
||||||
|
await page.getByRole('list', { name: 'Plot Series Properties' }).locator('[title="Display limit lines"]~div input').check();
|
||||||
|
|
||||||
|
await assertLimitLinesExistAndAreVisible(page);
|
||||||
|
|
||||||
|
// Save (exit edit mode)
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('li[title="Save and Finish Editing"]').click();
|
||||||
|
|
||||||
|
await assertLimitLinesExistAndAreVisible(page);
|
||||||
|
|
||||||
|
await page.reload();
|
||||||
|
|
||||||
|
await assertLimitLinesExistAndAreVisible(page);
|
||||||
|
|
||||||
|
// Enter edit mode
|
||||||
|
await page.click('button[title="Edit"]');
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Elements');
|
||||||
|
|
||||||
|
// Drag Sine Wave Generator series from Y Axis 1 into Y Axis 2
|
||||||
|
await page.locator(`#inspector-elements-tree >> text=${swgA.name}`).dragTo(page.locator('[aria-label="Element Item Group Y Axis 2"]'));
|
||||||
|
|
||||||
|
await assertLimitLinesExistAndAreVisible(page);
|
||||||
|
|
||||||
|
// Save (exit edit mode)
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('li[title="Save and Finish Editing"]').click();
|
||||||
|
|
||||||
|
await assertLimitLinesExistAndAreVisible(page);
|
||||||
|
|
||||||
|
await page.reload();
|
||||||
|
|
||||||
|
await assertLimitLinesExistAndAreVisible(page);
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
test('The elements pool supports dragging series into multiple y-axis buckets', async ({ page }) => {
|
||||||
|
const overlayPlot = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Overlay Plot"
|
||||||
|
});
|
||||||
|
|
||||||
|
const swgA = await createDomainObjectWithDefaults(page, {
|
||||||
type: "Sine Wave Generator",
|
type: "Sine Wave Generator",
|
||||||
name: 'swg b',
|
|
||||||
parent: overlayPlot.uuid
|
parent: overlayPlot.uuid
|
||||||
});
|
});
|
||||||
await createDomainObjectWithDefaults(page, {
|
const swgB = await createDomainObjectWithDefaults(page, {
|
||||||
type: "Sine Wave Generator",
|
type: "Sine Wave Generator",
|
||||||
name: 'swg c',
|
|
||||||
parent: overlayPlot.uuid
|
parent: overlayPlot.uuid
|
||||||
});
|
});
|
||||||
await createDomainObjectWithDefaults(page, {
|
const swgC = await createDomainObjectWithDefaults(page, {
|
||||||
type: "Sine Wave Generator",
|
type: "Sine Wave Generator",
|
||||||
name: 'swg d',
|
|
||||||
parent: overlayPlot.uuid
|
parent: overlayPlot.uuid
|
||||||
});
|
});
|
||||||
await createDomainObjectWithDefaults(page, {
|
const swgD = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Sine Wave Generator",
|
||||||
|
parent: overlayPlot.uuid
|
||||||
|
});
|
||||||
|
const swgE = await createDomainObjectWithDefaults(page, {
|
||||||
type: "Sine Wave Generator",
|
type: "Sine Wave Generator",
|
||||||
name: 'swg e',
|
|
||||||
parent: overlayPlot.uuid
|
parent: overlayPlot.uuid
|
||||||
});
|
});
|
||||||
|
|
||||||
await page.goto(overlayPlot.url);
|
await page.goto(overlayPlot.url);
|
||||||
await page.click('button[title="Edit"]');
|
await page.click('button[title="Edit"]');
|
||||||
|
|
||||||
// Expand the elements pool vertically
|
await selectInspectorTab(page, 'Elements');
|
||||||
await page.locator('.l-pane__handle').nth(2).hover({ trial: true });
|
|
||||||
await page.mouse.down();
|
|
||||||
await page.mouse.move(0, 100);
|
|
||||||
await page.mouse.up();
|
|
||||||
|
|
||||||
// Drag swg a, c, e into Y Axis 2
|
// Drag swg a, c, e into Y Axis 2
|
||||||
await page.locator('#inspector-elements-tree >> text=swg a').dragTo(page.locator('[aria-label="Element Item Group Y Axis 2"]'));
|
await page.locator(`#inspector-elements-tree >> text=${swgA.name}`).dragTo(page.locator('[aria-label="Element Item Group Y Axis 2"]'));
|
||||||
await page.locator('#inspector-elements-tree >> text=swg c').dragTo(page.locator('[aria-label="Element Item Group Y Axis 2"]'));
|
await page.locator(`#inspector-elements-tree >> text=${swgC.name}`).dragTo(page.locator('[aria-label="Element Item Group Y Axis 2"]'));
|
||||||
await page.locator('#inspector-elements-tree >> text=swg e').dragTo(page.locator('[aria-label="Element Item Group Y Axis 2"]'));
|
await page.locator(`#inspector-elements-tree >> text=${swgE.name}`).dragTo(page.locator('[aria-label="Element Item Group Y Axis 2"]'));
|
||||||
|
|
||||||
// Drag swg b into Y Axis 3
|
// Assert that Y Axis 1 and Y Axis 2 property groups are visible only
|
||||||
await page.locator('#inspector-elements-tree >> text=swg b').dragTo(page.locator('[aria-label="Element Item Group Y Axis 3"]'));
|
await selectInspectorTab(page, 'Config');
|
||||||
|
|
||||||
|
const yAxis1PropertyGroup = page.locator('[aria-label="Y Axis Properties"]');
|
||||||
|
const yAxis2PropertyGroup = page.locator('[aria-label="Y Axis 2 Properties"]');
|
||||||
|
const yAxis3PropertyGroup = page.locator('[aria-label="Y Axis 3 Properties"]');
|
||||||
|
|
||||||
|
await expect(yAxis1PropertyGroup).toBeVisible();
|
||||||
|
await expect(yAxis2PropertyGroup).toBeVisible();
|
||||||
|
await expect(yAxis3PropertyGroup).toBeHidden();
|
||||||
|
|
||||||
const yAxis1Group = page.getByLabel("Y Axis 1");
|
const yAxis1Group = page.getByLabel("Y Axis 1");
|
||||||
const yAxis2Group = page.getByLabel("Y Axis 2");
|
const yAxis2Group = page.getByLabel("Y Axis 2");
|
||||||
const yAxis3Group = page.getByLabel("Y Axis 3");
|
const yAxis3Group = page.getByLabel("Y Axis 3");
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Elements');
|
||||||
|
|
||||||
|
// Drag swg b into Y Axis 3
|
||||||
|
await page.locator(`#inspector-elements-tree >> text=${swgB.name}`).dragTo(page.locator('[aria-label="Element Item Group Y Axis 3"]'));
|
||||||
|
|
||||||
|
// Assert that all Y Axis property groups are visible
|
||||||
|
await selectInspectorTab(page, 'Config');
|
||||||
|
|
||||||
|
await expect(yAxis1PropertyGroup).toBeVisible();
|
||||||
|
await expect(yAxis2PropertyGroup).toBeVisible();
|
||||||
|
await expect(yAxis3PropertyGroup).toBeVisible();
|
||||||
|
|
||||||
// Verify that the elements are in the correct buckets and in the correct order
|
// Verify that the elements are in the correct buckets and in the correct order
|
||||||
expect(yAxis1Group.getByRole('listitem', { name: 'swg d' })).toBeTruthy();
|
await selectInspectorTab(page, 'Elements');
|
||||||
expect(yAxis1Group.getByRole('listitem').nth(0).getByText('swg d')).toBeTruthy();
|
|
||||||
expect(yAxis2Group.getByRole('listitem', { name: 'swg e' })).toBeTruthy();
|
expect(yAxis1Group.getByRole('listitem', { name: swgD.name })).toBeTruthy();
|
||||||
expect(yAxis2Group.getByRole('listitem').nth(0).getByText('swg e')).toBeTruthy();
|
expect(yAxis1Group.getByRole('listitem').nth(0).getByText(swgD.name)).toBeTruthy();
|
||||||
expect(yAxis2Group.getByRole('listitem', { name: 'swg c' })).toBeTruthy();
|
expect(yAxis2Group.getByRole('listitem', { name: swgE.name })).toBeTruthy();
|
||||||
expect(yAxis2Group.getByRole('listitem').nth(1).getByText('swg c')).toBeTruthy();
|
expect(yAxis2Group.getByRole('listitem').nth(0).getByText(swgE.name)).toBeTruthy();
|
||||||
expect(yAxis2Group.getByRole('listitem', { name: 'swg a' })).toBeTruthy();
|
expect(yAxis2Group.getByRole('listitem', { name: swgC.name })).toBeTruthy();
|
||||||
expect(yAxis2Group.getByRole('listitem').nth(2).getByText('swg a')).toBeTruthy();
|
expect(yAxis2Group.getByRole('listitem').nth(1).getByText(swgC.name)).toBeTruthy();
|
||||||
expect(yAxis3Group.getByRole('listitem', { name: 'swg b' })).toBeTruthy();
|
expect(yAxis2Group.getByRole('listitem', { name: swgA.name })).toBeTruthy();
|
||||||
expect(yAxis3Group.getByRole('listitem').nth(0).getByText('swg b')).toBeTruthy();
|
expect(yAxis2Group.getByRole('listitem').nth(2).getByText(swgA.name)).toBeTruthy();
|
||||||
|
expect(yAxis3Group.getByRole('listitem', { name: swgB.name })).toBeTruthy();
|
||||||
|
expect(yAxis3Group.getByRole('listitem').nth(0).getByText(swgB.name)).toBeTruthy();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Clicking on an item in the elements pool brings up the plot preview with data points', async ({ page }) => {
|
||||||
|
const overlayPlot = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Overlay Plot"
|
||||||
|
});
|
||||||
|
|
||||||
|
const swgA = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Sine Wave Generator",
|
||||||
|
parent: overlayPlot.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(overlayPlot.url);
|
||||||
|
await page.click('button[title="Edit"]');
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Elements');
|
||||||
|
|
||||||
|
await page.locator(`#inspector-elements-tree >> text=${swgA.name}`).click();
|
||||||
|
|
||||||
|
const plotPixelSize = await getCanvasPixelsWithData(page);
|
||||||
|
expect(plotPixelSize).toBeGreaterThan(0);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function getCanvasPixelsWithData(page) {
|
||||||
|
const getTelemValuePromise = new Promise(resolve => page.exposeFunction('getCanvasValue', resolve));
|
||||||
|
|
||||||
|
await page.evaluate(() => {
|
||||||
|
// The document canvas is where the plot points and lines are drawn.
|
||||||
|
// The only way to access the canvas is using document (using page.evaluate)
|
||||||
|
let data;
|
||||||
|
let canvas;
|
||||||
|
let ctx;
|
||||||
|
canvas = document.querySelector('.js-overlay canvas');
|
||||||
|
ctx = canvas.getContext('2d');
|
||||||
|
data = ctx.getImageData(0, 0, canvas.width, canvas.height).data;
|
||||||
|
const imageDataValues = Object.values(data);
|
||||||
|
let plotPixels = [];
|
||||||
|
// Each pixel consists of four values within the ImageData.data array. The for loop iterates by multiples of four.
|
||||||
|
// The values associated with each pixel are R (red), G (green), B (blue), and A (alpha), in that order.
|
||||||
|
for (let i = 0; i < imageDataValues.length;) {
|
||||||
|
if (imageDataValues[i] > 0) {
|
||||||
|
plotPixels.push({
|
||||||
|
startIndex: i,
|
||||||
|
endIndex: i + 3,
|
||||||
|
value: `rgb(${imageDataValues[i]}, ${imageDataValues[i + 1]}, ${imageDataValues[i + 2]}, ${imageDataValues[i + 3]})`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
i = i + 4;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
window.getCanvasValue(plotPixels.length);
|
||||||
|
});
|
||||||
|
|
||||||
|
return getTelemValuePromise;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function assertLimitLinesExistAndAreVisible(page) {
|
||||||
|
// Wait for plot series data to load
|
||||||
|
await expect(page.locator('.js-series-data-loaded')).toBeVisible();
|
||||||
|
// Wait for limit lines to be created
|
||||||
|
await page.waitForSelector('.js-limit-area', { state: 'attached' });
|
||||||
|
const limitLineCount = await page.locator('.c-plot-limit-line').count();
|
||||||
|
// There should be 10 limit lines created by default
|
||||||
|
expect(await page.locator('.c-plot-limit-line').count()).toBe(10);
|
||||||
|
for (let i = 0; i < limitLineCount; i++) {
|
||||||
|
await expect(page.locator('.c-plot-limit-line').nth(i)).toBeVisible();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -25,6 +25,7 @@ Tests to verify log plot functionality. Note this test suite if very much under
|
|||||||
necessarily be used for reference when writing new tests in this area.
|
necessarily be used for reference when writing new tests in this area.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
const { selectInspectorTab } = require('../../../../appActions');
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
|
||||||
test.describe('Legend color in sync with plot color', () => {
|
test.describe('Legend color in sync with plot color', () => {
|
||||||
@ -33,6 +34,8 @@ test.describe('Legend color in sync with plot color', () => {
|
|||||||
|
|
||||||
// navigate to plot series color palette
|
// navigate to plot series color palette
|
||||||
await page.click('.l-browse-bar__actions__edit');
|
await page.click('.l-browse-bar__actions__edit');
|
||||||
|
await selectInspectorTab(page, 'Config');
|
||||||
|
|
||||||
await page.locator('li.c-tree__item.menus-to-left .c-disclosure-triangle').click();
|
await page.locator('li.c-tree__item.menus-to-left .c-disclosure-triangle').click();
|
||||||
await page.locator('.c-click-swatch--menu').click();
|
await page.locator('.c-click-swatch--menu').click();
|
||||||
await page.locator('.c-palette__item[style="background: rgb(255, 166, 61);"]').click();
|
await page.locator('.c-palette__item[style="background: rgb(255, 166, 61);"]').click();
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -25,7 +25,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
const { createDomainObjectWithDefaults, selectInspectorTab } = require('../../../../appActions');
|
||||||
const uuid = require('uuid').v4;
|
const uuid = require('uuid').v4;
|
||||||
|
|
||||||
test.describe('Scatter Plot', () => {
|
test.describe('Scatter Plot', () => {
|
||||||
@ -40,8 +40,8 @@ test.describe('Scatter Plot', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test('Can add and remove telemetry sources', async ({ page }) => {
|
test('Can add and remove telemetry sources', async ({ page }) => {
|
||||||
const editButtonLocator = page.locator('button[title="Edit"]');
|
const editButton = page.locator('button[title="Edit"]');
|
||||||
const saveButtonLocator = page.locator('button[title="Save"]');
|
const saveButton = page.locator('button[title="Save"]');
|
||||||
|
|
||||||
// Create a sine wave generator within the scatter plot
|
// Create a sine wave generator within the scatter plot
|
||||||
const swg1 = await createDomainObjectWithDefaults(page, {
|
const swg1 = await createDomainObjectWithDefaults(page, {
|
||||||
@ -53,9 +53,10 @@ test.describe('Scatter Plot', () => {
|
|||||||
// Navigate to the scatter plot and verify that
|
// Navigate to the scatter plot and verify that
|
||||||
// the SWG appears in the elements pool
|
// the SWG appears in the elements pool
|
||||||
await page.goto(scatterPlot.url);
|
await page.goto(scatterPlot.url);
|
||||||
await editButtonLocator.click();
|
await editButton.click();
|
||||||
|
await selectInspectorTab(page, 'Elements');
|
||||||
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg1.name}`)).toBeVisible();
|
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg1.name}`)).toBeVisible();
|
||||||
await saveButtonLocator.click();
|
await saveButton.click();
|
||||||
await page.locator('li[title="Save and Finish Editing"]').click();
|
await page.locator('li[title="Save and Finish Editing"]').click();
|
||||||
|
|
||||||
// Create another sine wave generator within the scatter plot
|
// Create another sine wave generator within the scatter plot
|
||||||
@ -72,10 +73,13 @@ test.describe('Scatter Plot', () => {
|
|||||||
// Navigate to the scatter plot and verify that the new SWG
|
// Navigate to the scatter plot and verify that the new SWG
|
||||||
// appears in the elements pool and the old one is gone
|
// appears in the elements pool and the old one is gone
|
||||||
await page.goto(scatterPlot.url);
|
await page.goto(scatterPlot.url);
|
||||||
await editButtonLocator.click();
|
await editButton.click();
|
||||||
|
|
||||||
|
// Click the "Elements" tab
|
||||||
|
await selectInspectorTab(page, 'Elements');
|
||||||
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg1.name}`)).toBeHidden();
|
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg1.name}`)).toBeHidden();
|
||||||
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg2.name}`)).toBeVisible();
|
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg2.name}`)).toBeVisible();
|
||||||
await saveButtonLocator.click();
|
await saveButton.click();
|
||||||
|
|
||||||
// Right click on the new SWG in the elements pool and delete it
|
// Right click on the new SWG in the elements pool and delete it
|
||||||
await page.locator(`#inspector-elements-tree >> text=${swg2.name}`).click({
|
await page.locator(`#inspector-elements-tree >> text=${swg2.name}`).click({
|
||||||
|
189
e2e/tests/functional/plugins/plot/stackedPlot.e2e.spec.js
Normal file
@ -0,0 +1,189 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2023, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/*
|
||||||
|
Tests to verify log plot functionality. Note this test suite if very much under active development and should not
|
||||||
|
necessarily be used for reference when writing new tests in this area.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults, selectInspectorTab } = require('../../../../appActions');
|
||||||
|
|
||||||
|
test.describe('Stacked Plot', () => {
|
||||||
|
let stackedPlot;
|
||||||
|
let swgA;
|
||||||
|
let swgB;
|
||||||
|
let swgC;
|
||||||
|
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// Open a browser, navigate to the main page, and wait until all networkevents to resolve
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
stackedPlot = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Stacked Plot"
|
||||||
|
});
|
||||||
|
|
||||||
|
swgA = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Sine Wave Generator",
|
||||||
|
parent: stackedPlot.uuid
|
||||||
|
});
|
||||||
|
swgB = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Sine Wave Generator",
|
||||||
|
parent: stackedPlot.uuid
|
||||||
|
});
|
||||||
|
swgC = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Sine Wave Generator",
|
||||||
|
parent: stackedPlot.uuid
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Using the remove action removes the correct plot', async ({ page }) => {
|
||||||
|
const swgAElementsPoolItem = page.locator('#inspector-elements-tree').locator('.c-object-label', { hasText: swgA.name });
|
||||||
|
const swgBElementsPoolItem = page.locator('#inspector-elements-tree').locator('.c-object-label', { hasText: swgB.name });
|
||||||
|
const swgCElementsPoolItem = page.locator('#inspector-elements-tree').locator('.c-object-label', { hasText: swgC.name });
|
||||||
|
|
||||||
|
await page.goto(stackedPlot.url);
|
||||||
|
|
||||||
|
await page.click('button[title="Edit"]');
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Elements');
|
||||||
|
|
||||||
|
await swgBElementsPoolItem.click({ button: 'right' });
|
||||||
|
await page.getByRole('menuitem').filter({ hasText: /Remove/ }).click();
|
||||||
|
await page.getByRole('button').filter({ hasText: "OK" }).click();
|
||||||
|
|
||||||
|
await expect(page.locator('#inspector-elements-tree .js-elements-pool__item')).toHaveCount(2);
|
||||||
|
|
||||||
|
// Confirm that the elements pool contains the items we expect
|
||||||
|
await expect(swgAElementsPoolItem).toHaveCount(1);
|
||||||
|
await expect(swgBElementsPoolItem).toHaveCount(0);
|
||||||
|
await expect(swgCElementsPoolItem).toHaveCount(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can reorder Stacked Plot items', async ({ page }) => {
|
||||||
|
const swgAElementsPoolItem = page.locator('#inspector-elements-tree').locator('.c-object-label', { hasText: swgA.name });
|
||||||
|
const swgBElementsPoolItem = page.locator('#inspector-elements-tree').locator('.c-object-label', { hasText: swgB.name });
|
||||||
|
const swgCElementsPoolItem = page.locator('#inspector-elements-tree').locator('.c-object-label', { hasText: swgC.name });
|
||||||
|
|
||||||
|
await page.goto(stackedPlot.url);
|
||||||
|
|
||||||
|
await page.click('button[title="Edit"]');
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Elements');
|
||||||
|
|
||||||
|
const stackedPlotItem1 = page.locator('.c-plot--stacked-container').nth(0);
|
||||||
|
const stackedPlotItem2 = page.locator('.c-plot--stacked-container').nth(1);
|
||||||
|
const stackedPlotItem3 = page.locator('.c-plot--stacked-container').nth(2);
|
||||||
|
|
||||||
|
// assert initial plot order - [swgA, swgB, swgC]
|
||||||
|
await expect(stackedPlotItem1).toHaveAttribute('aria-label', `Stacked Plot Item ${swgA.name}`);
|
||||||
|
await expect(stackedPlotItem2).toHaveAttribute('aria-label', `Stacked Plot Item ${swgB.name}`);
|
||||||
|
await expect(stackedPlotItem3).toHaveAttribute('aria-label', `Stacked Plot Item ${swgC.name}`);
|
||||||
|
|
||||||
|
// Drag and drop to reorder - [swgB, swgA, swgC]
|
||||||
|
await swgBElementsPoolItem.dragTo(swgAElementsPoolItem);
|
||||||
|
|
||||||
|
// assert plot order after reorder - [swgB, swgA, swgC]
|
||||||
|
await expect(stackedPlotItem1).toHaveAttribute('aria-label', `Stacked Plot Item ${swgB.name}`);
|
||||||
|
await expect(stackedPlotItem2).toHaveAttribute('aria-label', `Stacked Plot Item ${swgA.name}`);
|
||||||
|
await expect(stackedPlotItem3).toHaveAttribute('aria-label', `Stacked Plot Item ${swgC.name}`);
|
||||||
|
|
||||||
|
// Drag and drop to reorder - [swgB, swgC, swgA]
|
||||||
|
await swgCElementsPoolItem.dragTo(swgAElementsPoolItem);
|
||||||
|
|
||||||
|
// assert plot order after second reorder - [swgB, swgC, swgA]
|
||||||
|
await expect(stackedPlotItem1).toHaveAttribute('aria-label', `Stacked Plot Item ${swgB.name}`);
|
||||||
|
await expect(stackedPlotItem2).toHaveAttribute('aria-label', `Stacked Plot Item ${swgC.name}`);
|
||||||
|
await expect(stackedPlotItem3).toHaveAttribute('aria-label', `Stacked Plot Item ${swgA.name}`);
|
||||||
|
|
||||||
|
// collapse inspector
|
||||||
|
await page.locator('.l-shell__pane-inspector .l-pane__collapse-button').click();
|
||||||
|
|
||||||
|
// Save (exit edit mode)
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('li[title="Save and Finish Editing"]').click();
|
||||||
|
|
||||||
|
// assert plot order persists after save - [swgB, swgC, swgA]
|
||||||
|
await expect(stackedPlotItem1).toHaveAttribute('aria-label', `Stacked Plot Item ${swgB.name}`);
|
||||||
|
await expect(stackedPlotItem2).toHaveAttribute('aria-label', `Stacked Plot Item ${swgC.name}`);
|
||||||
|
await expect(stackedPlotItem3).toHaveAttribute('aria-label', `Stacked Plot Item ${swgA.name}`);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Selecting a child plot while in browse and edit modes shows its properties in the inspector', async ({ page }) => {
|
||||||
|
await page.goto(stackedPlot.url);
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Config');
|
||||||
|
|
||||||
|
// Click on the 1st plot
|
||||||
|
await page.locator(`[aria-label="Stacked Plot Item ${swgA.name}"] canvas`).nth(1).click();
|
||||||
|
|
||||||
|
// Assert that the inspector shows the Y Axis properties for swgA
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] >> h2')).toContainText("Plot Series");
|
||||||
|
await expect(page.getByRole('list', { name: "Y Axis Properties" }).locator("h2")).toContainText("Y Axis");
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] .c-object-label')).toContainText(swgA.name);
|
||||||
|
|
||||||
|
// Click on the 2nd plot
|
||||||
|
await page.locator(`[aria-label="Stacked Plot Item ${swgB.name}"] canvas`).nth(1).click();
|
||||||
|
|
||||||
|
// Assert that the inspector shows the Y Axis properties for swgB
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] >> h2')).toContainText("Plot Series");
|
||||||
|
await expect(page.getByRole('list', { name: "Y Axis Properties" }).locator("h2")).toContainText("Y Axis");
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] .c-object-label')).toContainText(swgB.name);
|
||||||
|
|
||||||
|
// Click on the 3rd plot
|
||||||
|
await page.locator(`[aria-label="Stacked Plot Item ${swgC.name}"] canvas`).nth(1).click();
|
||||||
|
|
||||||
|
// Assert that the inspector shows the Y Axis properties for swgC
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] >> h2')).toContainText("Plot Series");
|
||||||
|
await expect(page.getByRole('list', { name: "Y Axis Properties" }).locator("h2")).toContainText("Y Axis");
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] .c-object-label')).toContainText(swgC.name);
|
||||||
|
|
||||||
|
// Go into edit mode
|
||||||
|
await page.click('button[title="Edit"]');
|
||||||
|
|
||||||
|
await selectInspectorTab(page, 'Config');
|
||||||
|
|
||||||
|
// Click on canvas for the 1st plot
|
||||||
|
await page.locator(`[aria-label="Stacked Plot Item ${swgA.name}"]`).click();
|
||||||
|
|
||||||
|
// Assert that the inspector shows the Y Axis properties for swgA
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] >> h2')).toContainText("Plot Series");
|
||||||
|
await expect(page.getByRole('list', { name: "Y Axis Properties" }).locator("h2")).toContainText("Y Axis");
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] .c-object-label')).toContainText(swgA.name);
|
||||||
|
|
||||||
|
//Click on canvas for the 2nd plot
|
||||||
|
await page.locator(`[aria-label="Stacked Plot Item ${swgB.name}"]`).click();
|
||||||
|
|
||||||
|
// Assert that the inspector shows the Y Axis properties for swgB
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] >> h2')).toContainText("Plot Series");
|
||||||
|
await expect(page.getByRole('list', { name: "Y Axis Properties" }).locator("h2")).toContainText("Y Axis");
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] .c-object-label')).toContainText(swgB.name);
|
||||||
|
|
||||||
|
//Click on canvas for the 3rd plot
|
||||||
|
await page.locator(`[aria-label="Stacked Plot Item ${swgC.name}"]`).click();
|
||||||
|
|
||||||
|
// Assert that the inspector shows the Y Axis properties for swgC
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] >> h2')).toContainText("Plot Series");
|
||||||
|
await expect(page.getByRole('list', { name: "Y Axis Properties" }).locator("h2")).toContainText("Y Axis");
|
||||||
|
await expect(page.locator('[aria-label="Plot Series Properties"] .c-object-label')).toContainText(swgC.name);
|
||||||
|
});
|
||||||
|
});
|
267
e2e/tests/functional/plugins/plot/tagging.e2e.spec.js
Normal file
@ -0,0 +1,267 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2023, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/*
|
||||||
|
Tests to verify plot tagging functionality.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults, setRealTimeMode, setFixedTimeMode } = require('../../../../appActions');
|
||||||
|
|
||||||
|
test.describe('Plot Tagging', () => {
|
||||||
|
/**
|
||||||
|
* Given a canvas and a set of points, tags the points on the canvas.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {HTMLCanvasElement} canvas a telemetry item with a plot
|
||||||
|
* @param {Number} xEnd a telemetry item with a plot
|
||||||
|
* @param {Number} yEnd a telemetry item with a plot
|
||||||
|
* @returns {Promise}
|
||||||
|
*/
|
||||||
|
async function createTags({page, canvas, xEnd, yEnd}) {
|
||||||
|
await canvas.hover({trial: true});
|
||||||
|
|
||||||
|
//Alt+Shift Drag Start to select some points to tag
|
||||||
|
await page.keyboard.down('Alt');
|
||||||
|
await page.keyboard.down('Shift');
|
||||||
|
|
||||||
|
await canvas.dragTo(canvas, {
|
||||||
|
sourcePosition: {
|
||||||
|
x: 1,
|
||||||
|
y: 1
|
||||||
|
},
|
||||||
|
targetPosition: {
|
||||||
|
x: xEnd,
|
||||||
|
y: yEnd
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
//Alt Drag End
|
||||||
|
await page.keyboard.up('Alt');
|
||||||
|
await page.keyboard.up('Shift');
|
||||||
|
|
||||||
|
//Wait for canvas to stablize.
|
||||||
|
await canvas.hover({trial: true});
|
||||||
|
|
||||||
|
// add some tags
|
||||||
|
await page.getByText('Annotations').click();
|
||||||
|
await page.getByRole('button', { name: /Add Tag/ }).click();
|
||||||
|
await page.getByPlaceholder('Type to select tag').click();
|
||||||
|
await page.getByText('Driving').click();
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: /Add Tag/ }).click();
|
||||||
|
await page.getByPlaceholder('Type to select tag').click();
|
||||||
|
await page.getByText('Science').click();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given a telemetry item (e.g., a Sine Wave Generator) with a plot, tests that the plot can be tagged.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {import('../../../../appActions').CreatedObjectInfo} telemetryItem a telemetry item with a plot
|
||||||
|
* @returns {Promise}
|
||||||
|
*/
|
||||||
|
async function testTelemetryItem(page, telemetryItem) {
|
||||||
|
// Check that telemetry item also received the tag
|
||||||
|
await page.goto(telemetryItem.url);
|
||||||
|
|
||||||
|
await expect(page.getByText('No tags to display for this item')).toBeVisible();
|
||||||
|
|
||||||
|
const canvas = page.locator('canvas').nth(1);
|
||||||
|
|
||||||
|
//Wait for canvas to stablize.
|
||||||
|
await canvas.hover({trial: true});
|
||||||
|
|
||||||
|
// click on the tagged plot point
|
||||||
|
await canvas.click({
|
||||||
|
position: {
|
||||||
|
x: 325,
|
||||||
|
y: 377
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
await expect(page.getByText('Science')).toBeVisible();
|
||||||
|
await expect(page.getByText('Driving')).toBeHidden();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given a page, tests that tags are searchable, deletable, and persist across reloads.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @returns {Promise}
|
||||||
|
*/
|
||||||
|
async function basicTagsTests(page) {
|
||||||
|
// Search for Driving
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
|
||||||
|
// Clicking elsewhere should cause annotation selection to be cleared
|
||||||
|
await expect(page.getByText('No tags to display for this item')).toBeVisible();
|
||||||
|
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('driv');
|
||||||
|
// click on the search result
|
||||||
|
await page.getByRole('searchbox', { name: 'OpenMCT Search' }).getByText(/Sine Wave/).first().click();
|
||||||
|
|
||||||
|
// Delete Driving
|
||||||
|
await page.hover('[aria-label="Tag"]:has-text("Driving")');
|
||||||
|
await page.locator('[aria-label="Remove tag Driving"]').click();
|
||||||
|
|
||||||
|
// Search for Science
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('sc');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"]').nth(0)).toContainText("Science");
|
||||||
|
await expect(page.locator('[aria-label="Search Result"]').nth(0)).not.toContainText("Drilling");
|
||||||
|
|
||||||
|
// Search for Driving
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('driv');
|
||||||
|
await expect(page.getByText('No results found')).toBeVisible();
|
||||||
|
|
||||||
|
//Reload Page
|
||||||
|
await Promise.all([
|
||||||
|
page.reload(),
|
||||||
|
page.waitForLoadState('networkidle')
|
||||||
|
]);
|
||||||
|
// wait for plots to load
|
||||||
|
await expect(page.locator('.js-series-data-loaded')).toBeVisible();
|
||||||
|
|
||||||
|
await page.getByText('Annotations').click();
|
||||||
|
await expect(page.getByText('No tags to display for this item')).toBeVisible();
|
||||||
|
|
||||||
|
const canvas = page.locator('canvas').nth(1);
|
||||||
|
// click on the tagged plot point
|
||||||
|
await canvas.click({
|
||||||
|
position: {
|
||||||
|
x: 100,
|
||||||
|
y: 100
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
await expect(page.getByText('Science')).toBeVisible();
|
||||||
|
await expect(page.getByText('Driving')).toBeHidden();
|
||||||
|
}
|
||||||
|
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Tags work with Overlay Plots', async ({ page }) => {
|
||||||
|
//Test.slow decorator is currently broken. Needs to be fixed in https://github.com/nasa/openmct/issues/5374
|
||||||
|
test.slow();
|
||||||
|
|
||||||
|
const overlayPlot = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Overlay Plot"
|
||||||
|
});
|
||||||
|
|
||||||
|
const alphaSineWave = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Sine Wave Generator",
|
||||||
|
name: "Alpha Sine Wave",
|
||||||
|
parent: overlayPlot.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Sine Wave Generator",
|
||||||
|
name: "Beta Sine Wave",
|
||||||
|
parent: overlayPlot.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(overlayPlot.url);
|
||||||
|
|
||||||
|
let canvas = page.locator('canvas').nth(1);
|
||||||
|
|
||||||
|
// Switch to real-time mode
|
||||||
|
// Adding tags should pause the plot
|
||||||
|
await setRealTimeMode(page);
|
||||||
|
|
||||||
|
await createTags({
|
||||||
|
page,
|
||||||
|
canvas,
|
||||||
|
xEnd: 700,
|
||||||
|
yEnd: 480
|
||||||
|
});
|
||||||
|
|
||||||
|
await setFixedTimeMode(page);
|
||||||
|
|
||||||
|
// changing to fixed time mode rebuilds canvas?
|
||||||
|
canvas = page.locator('canvas').nth(1);
|
||||||
|
|
||||||
|
await basicTagsTests(page);
|
||||||
|
await testTelemetryItem(page, alphaSineWave);
|
||||||
|
|
||||||
|
// set to real time mode
|
||||||
|
await setRealTimeMode(page);
|
||||||
|
|
||||||
|
// Search for Science
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('sc');
|
||||||
|
// click on the search result
|
||||||
|
await page.getByRole('searchbox', { name: 'OpenMCT Search' }).getByText('Alpha Sine Wave').first().click();
|
||||||
|
// wait for plots to load
|
||||||
|
await expect(page.locator('.js-series-data-loaded')).toBeVisible();
|
||||||
|
// expect plot to be paused
|
||||||
|
await expect(page.locator('[title="Resume displaying real-time data"]')).toBeVisible();
|
||||||
|
|
||||||
|
await setFixedTimeMode(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Tags work with Plot View of telemetry items', async ({ page }) => {
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Sine Wave Generator"
|
||||||
|
});
|
||||||
|
|
||||||
|
const canvas = page.locator('canvas').nth(1);
|
||||||
|
await createTags({
|
||||||
|
page,
|
||||||
|
canvas,
|
||||||
|
xEnd: 700,
|
||||||
|
yEnd: 480
|
||||||
|
});
|
||||||
|
await basicTagsTests(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Tags work with Stacked Plots', async ({ page }) => {
|
||||||
|
const stackedPlot = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Stacked Plot"
|
||||||
|
});
|
||||||
|
|
||||||
|
const alphaSineWave = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Sine Wave Generator",
|
||||||
|
name: "Alpha Sine Wave",
|
||||||
|
parent: stackedPlot.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Sine Wave Generator",
|
||||||
|
name: "Beta Sine Wave",
|
||||||
|
parent: stackedPlot.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(stackedPlot.url);
|
||||||
|
|
||||||
|
const canvas = page.locator('canvas').nth(1);
|
||||||
|
|
||||||
|
await createTags({
|
||||||
|
page,
|
||||||
|
canvas,
|
||||||
|
xEnd: 700,
|
||||||
|
yEnd: 215
|
||||||
|
});
|
||||||
|
await basicTagsTests(page);
|
||||||
|
await testTelemetryItem(page, alphaSineWave);
|
||||||
|
});
|
||||||
|
});
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -22,37 +22,46 @@
|
|||||||
|
|
||||||
const { test, expect } = require('../../pluginFixtures.js');
|
const { test, expect } = require('../../pluginFixtures.js');
|
||||||
const { createDomainObjectWithDefaults } = require('../../appActions.js');
|
const { createDomainObjectWithDefaults } = require('../../appActions.js');
|
||||||
|
const { waitForAnimations } = require('../../baseFixtures.js');
|
||||||
|
|
||||||
test.describe('Recent Objects', () => {
|
test.describe('Recent Objects', () => {
|
||||||
test('Recent Objects CRUD operations', async ({ page }) => {
|
/** @type {import('@playwright/test').Locator} */
|
||||||
|
let recentObjectsList;
|
||||||
|
/** @type {import('@playwright/test').Locator} */
|
||||||
|
let clock;
|
||||||
|
/** @type {import('@playwright/test').Locator} */
|
||||||
|
let folderA;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
await page.goto('./', { waitUntil: 'networkidle' });
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Set Recent Objects List locator for subsequent tests
|
||||||
|
recentObjectsList = page.getByRole('list', {
|
||||||
|
name: 'Recent Objects'
|
||||||
|
});
|
||||||
|
|
||||||
// Create a folder and nest a Clock within it
|
// Create a folder and nest a Clock within it
|
||||||
const recentObjectsList = page.locator('[aria-label="Recent Objects"]');
|
folderA = await createDomainObjectWithDefaults(page, {
|
||||||
const folderA = await createDomainObjectWithDefaults(page, {
|
|
||||||
type: 'Folder'
|
type: 'Folder'
|
||||||
});
|
});
|
||||||
const clock = await createDomainObjectWithDefaults(page, {
|
clock = await createDomainObjectWithDefaults(page, {
|
||||||
type: 'Clock',
|
type: 'Clock',
|
||||||
parent: folderA.uuid
|
parent: folderA.uuid
|
||||||
});
|
});
|
||||||
|
|
||||||
// Drag the Recent Objects panel up a bit
|
// Drag the Recent Objects panel up a bit
|
||||||
await page.locator('div:nth-child(2) > .l-pane__handle').hover();
|
await page.locator('.l-pane.l-pane--vertical-handle-before', {
|
||||||
|
hasText: 'Recently Viewed'
|
||||||
|
}).locator('.l-pane__handle').hover();
|
||||||
await page.mouse.down();
|
await page.mouse.down();
|
||||||
await page.mouse.move(0, 100);
|
await page.mouse.move(0, 100);
|
||||||
await page.mouse.up();
|
await page.mouse.up();
|
||||||
|
});
|
||||||
|
test('Navigated objects show up in recents, object renames and deletions are reflected', async ({ page }) => {
|
||||||
// Verify that both created objects appear in the list and are in the correct order
|
// Verify that both created objects appear in the list and are in the correct order
|
||||||
expect(recentObjectsList.getByRole('listitem', { name: clock.name })).toBeTruthy();
|
assertInitialRecentObjectsListState();
|
||||||
expect(recentObjectsList.getByRole('listitem', { name: folderA.name })).toBeTruthy();
|
|
||||||
expect(recentObjectsList.getByRole('listitem', { name: clock.name }).locator('a').getByText(folderA.name)).toBeTruthy();
|
|
||||||
expect(recentObjectsList.getByRole('listitem').nth(0).getByText(clock.name)).toBeTruthy();
|
|
||||||
expect(recentObjectsList.getByRole('listitem', { name: clock.name }).locator('a').getByText(folderA.name)).toBeTruthy();
|
|
||||||
expect(recentObjectsList.getByRole('listitem').nth(1).getByText(folderA.name)).toBeTruthy();
|
|
||||||
|
|
||||||
// Navigate to the folder by clicking on the main object name in the recent objects list item
|
// Navigate to the folder by clicking on the main object name in the recent objects list item
|
||||||
await recentObjectsList.getByRole('listitem', { name: folderA.name }).getByText(folderA.name).click();
|
await page.getByRole('listitem', { name: folderA.name }).getByText(folderA.name).click();
|
||||||
await page.waitForURL(`**/${folderA.uuid}?*`);
|
await page.waitForURL(`**/${folderA.uuid}?*`);
|
||||||
expect(recentObjectsList.getByRole('listitem').nth(0).getByText(folderA.name)).toBeTruthy();
|
expect(recentObjectsList.getByRole('listitem').nth(0).getByText(folderA.name)).toBeTruthy();
|
||||||
|
|
||||||
@ -63,7 +72,11 @@ test.describe('Recent Objects', () => {
|
|||||||
await page.keyboard.press('Enter');
|
await page.keyboard.press('Enter');
|
||||||
|
|
||||||
// Verify rename has been applied in recent objects list item and objects paths
|
// Verify rename has been applied in recent objects list item and objects paths
|
||||||
expect(page.getByRole('listitem', { name: clock.name }).locator('a').getByText(folderA.name)).toBeTruthy();
|
expect(await page.getByRole('navigation', {
|
||||||
|
name: clock.name
|
||||||
|
}).locator('a').filter({
|
||||||
|
hasText: folderA.name
|
||||||
|
}).count()).toBeGreaterThan(0);
|
||||||
expect(recentObjectsList.getByRole('listitem', { name: folderA.name })).toBeTruthy();
|
expect(recentObjectsList.getByRole('listitem', { name: folderA.name })).toBeTruthy();
|
||||||
|
|
||||||
// Delete
|
// Delete
|
||||||
@ -79,7 +92,173 @@ test.describe('Recent Objects', () => {
|
|||||||
await expect(recentObjectsList.getByRole('listitem', { name: folderA.name })).toBeHidden();
|
await expect(recentObjectsList.getByRole('listitem', { name: folderA.name })).toBeHidden();
|
||||||
await expect(recentObjectsList.getByRole('listitem', { name: clock.name })).toBeHidden();
|
await expect(recentObjectsList.getByRole('listitem', { name: clock.name })).toBeHidden();
|
||||||
});
|
});
|
||||||
test.fixme("Clicking on the 'target button' scrolls the object into view in the tree and highlights it");
|
test("Clicking on an object in the path of a recent object navigates to the object", async ({ page, openmctConfig }) => {
|
||||||
test.fixme("Clicking on an object in the path of a recent object navigates to the object");
|
const { myItemsFolderName } = openmctConfig;
|
||||||
test.fixme("Tests for context menu actions from recent objects");
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/6151'
|
||||||
|
});
|
||||||
|
await page.goto('./#/browse/mine');
|
||||||
|
|
||||||
|
// Navigate to the folder by clicking on its entry in the Clock's breadcrumb
|
||||||
|
const waitForFolderNavigation = page.waitForURL(`**/${folderA.uuid}?*`);
|
||||||
|
await page.getByRole('navigation', {
|
||||||
|
name: clock.name
|
||||||
|
}).locator('a').filter({
|
||||||
|
hasText: folderA.name
|
||||||
|
}).click();
|
||||||
|
|
||||||
|
// Verify that the hash URL updates correctly
|
||||||
|
await waitForFolderNavigation;
|
||||||
|
expect(page.url()).toMatch(new RegExp(`.*${folderA.uuid}?.*`));
|
||||||
|
|
||||||
|
// Navigate to My Items by clicking on its entry in the Clock's breadcrumb
|
||||||
|
const waitForMyItemsNavigation = page.waitForURL(`**/mine?*`);
|
||||||
|
await page.getByRole('navigation', {
|
||||||
|
name: clock.name
|
||||||
|
}).locator('a').filter({
|
||||||
|
hasText: myItemsFolderName
|
||||||
|
}).click();
|
||||||
|
|
||||||
|
// Verify that the hash URL updates correctly
|
||||||
|
await waitForMyItemsNavigation;
|
||||||
|
expect(page.url()).toMatch(new RegExp(`.*mine?.*`));
|
||||||
|
});
|
||||||
|
test("Clicking on the 'target button' scrolls the object into view in the tree and highlights it", async ({ page }) => {
|
||||||
|
const clockTreeItem = page.getByRole('tree', { name: 'Main Tree'}).getByRole('treeitem', { name: clock.name });
|
||||||
|
const folderTreeItem = page.getByRole('tree', { name: 'Main Tree'})
|
||||||
|
.getByRole('treeitem', {
|
||||||
|
name: folderA.name,
|
||||||
|
expanded: true
|
||||||
|
});
|
||||||
|
|
||||||
|
// Click the "Target" button for the Clock which is nested in a folder
|
||||||
|
await page.getByRole('button', { name: `Open and scroll to ${clock.name}`}).click();
|
||||||
|
|
||||||
|
// Assert that the Clock parent folder has expanded and the Clock is visible)
|
||||||
|
await expect(folderTreeItem.locator('.c-disclosure-triangle')).toHaveClass(/--expanded/);
|
||||||
|
await expect(clockTreeItem).toBeVisible();
|
||||||
|
|
||||||
|
// Assert that the Clock treeitem is highlighted
|
||||||
|
await expect(clockTreeItem.locator('.c-tree__item')).toHaveClass(/is-targeted-item/);
|
||||||
|
|
||||||
|
// Wait for highlight animation to end
|
||||||
|
await waitForAnimations(clockTreeItem.locator('.c-tree__item'));
|
||||||
|
|
||||||
|
// Assert that the Clock treeitem is no longer highlighted
|
||||||
|
await expect(clockTreeItem.locator('.c-tree__item')).not.toHaveClass(/is-targeted-item/);
|
||||||
|
});
|
||||||
|
test("Persists on refresh", async ({ page }) => {
|
||||||
|
assertInitialRecentObjectsListState();
|
||||||
|
await page.reload();
|
||||||
|
assertInitialRecentObjectsListState();
|
||||||
|
});
|
||||||
|
test("Displays objects and aliases uniquely", async ({ page }) => {
|
||||||
|
const mainTree = page.getByRole('tree', { name: 'Main Tree'});
|
||||||
|
|
||||||
|
// Navigate to the clock and reveal it in the tree
|
||||||
|
await page.goto(clock.url);
|
||||||
|
await page.getByTitle('Show selected item in tree').click();
|
||||||
|
|
||||||
|
// Right click the clock and create an alias using the "link" context menu action
|
||||||
|
const clockTreeItem = page.getByRole('tree', {
|
||||||
|
name: 'Main Tree'
|
||||||
|
}).getByRole('treeitem', {
|
||||||
|
name: clock.name
|
||||||
|
});
|
||||||
|
await clockTreeItem.click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
await page.getByRole('menuitem', {
|
||||||
|
name: /Create Link/
|
||||||
|
}).click();
|
||||||
|
await page.getByRole('tree', { name: 'Create Modal Tree'}).getByRole('treeitem').first().click();
|
||||||
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
|
|
||||||
|
// Click the newly created object alias in the tree
|
||||||
|
await mainTree.getByRole('treeitem', {
|
||||||
|
name: new RegExp(clock.name)
|
||||||
|
}).filter({
|
||||||
|
has: page.locator('.is-alias')
|
||||||
|
}).click();
|
||||||
|
|
||||||
|
// Assert that two recent objects are displayed and one of them is an alias
|
||||||
|
expect(await recentObjectsList.getByRole('listitem', { name: clock.name }).count()).toBe(2);
|
||||||
|
expect(await recentObjectsList.locator('.is-alias').count()).toBe(1);
|
||||||
|
|
||||||
|
// Assert that the alias and the original's breadcrumbs are different
|
||||||
|
const clockBreadcrumbs = recentObjectsList.getByRole('listitem', {name: clock.name}).getByRole('navigation');
|
||||||
|
expect(await clockBreadcrumbs.count()).toBe(2);
|
||||||
|
expect(await clockBreadcrumbs.nth(0).innerText()).not.toEqual(await clockBreadcrumbs.nth(1).innerText());
|
||||||
|
});
|
||||||
|
test("Enforces a limit of 20 recent objects and clears the recent objects", async ({ page }) => {
|
||||||
|
// Creating 21 objects takes a while, so increase the timeout
|
||||||
|
test.slow();
|
||||||
|
|
||||||
|
// Assert that the list initially contains 3 objects (clock, folder, my items)
|
||||||
|
expect(await recentObjectsList.locator('.c-recentobjects-listitem').count()).toBe(3);
|
||||||
|
|
||||||
|
let lastFolder;
|
||||||
|
let lastClock;
|
||||||
|
// Create 19 more objects (3 in beforeEach() + 18 new = 21 total)
|
||||||
|
for (let i = 0; i < 9; i++) {
|
||||||
|
lastFolder = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Folder",
|
||||||
|
parent: lastFolder?.uuid
|
||||||
|
});
|
||||||
|
lastClock = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: "Clock",
|
||||||
|
parent: lastFolder?.uuid
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Assert that the list contains 20 objects
|
||||||
|
expect(await recentObjectsList.locator('.c-recentobjects-listitem').count()).toBe(20);
|
||||||
|
|
||||||
|
// Collapse the tree
|
||||||
|
await page.getByTitle("Collapse all tree items").click();
|
||||||
|
const lastFolderTreeItem = page.getByRole('tree', { name: 'Main Tree'})
|
||||||
|
.getByRole('treeitem', {
|
||||||
|
name: lastFolder.name,
|
||||||
|
expanded: true
|
||||||
|
});
|
||||||
|
const lastClockTreeItem = page.getByRole('tree', { name: 'Main Tree'})
|
||||||
|
.getByRole('treeitem', {
|
||||||
|
name: lastClock.name
|
||||||
|
});
|
||||||
|
|
||||||
|
// Test "Open and Scroll To" in a deeply nested tree, while we're here
|
||||||
|
await page.getByRole('button', { name: `Open and scroll to ${lastClock.name}`}).click();
|
||||||
|
|
||||||
|
// Assert that the Clock parent folder has expanded and the Clock is visible)
|
||||||
|
await expect(lastFolderTreeItem.locator('.c-disclosure-triangle')).toHaveClass(/--expanded/);
|
||||||
|
await expect(lastClockTreeItem).toBeVisible();
|
||||||
|
|
||||||
|
// Assert that the Clock treeitem is highlighted
|
||||||
|
await expect(lastClockTreeItem.locator('.c-tree__item')).toHaveClass(/is-targeted-item/);
|
||||||
|
|
||||||
|
// Wait for highlight animation to end
|
||||||
|
await waitForAnimations(lastClockTreeItem.locator('.c-tree__item'));
|
||||||
|
|
||||||
|
// Assert that the Clock treeitem is no longer highlighted
|
||||||
|
await expect(lastClockTreeItem.locator('.c-tree__item')).not.toHaveClass(/is-targeted-item/);
|
||||||
|
|
||||||
|
// Click the aria-label="Clear Recently Viewed" button
|
||||||
|
await page.getByRole('button', { name: 'Clear Recently Viewed' }).click();
|
||||||
|
|
||||||
|
// Click on the "OK" button in the confirmation dialog
|
||||||
|
await page.getByRole('button', { name: 'OK' }).click();
|
||||||
|
|
||||||
|
// Assert that the list is empty
|
||||||
|
expect(await recentObjectsList.locator('.c-recentobjects-listitem').count()).toBe(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
function assertInitialRecentObjectsListState() {
|
||||||
|
expect(recentObjectsList.getByRole('listitem', { name: clock.name })).toBeTruthy();
|
||||||
|
expect(recentObjectsList.getByRole('listitem', { name: folderA.name })).toBeTruthy();
|
||||||
|
expect(recentObjectsList.getByRole('listitem', { name: clock.name }).locator('a').getByText(folderA.name)).toBeTruthy();
|
||||||
|
expect(recentObjectsList.getByRole('listitem').nth(0).getByText(clock.name)).toBeTruthy();
|
||||||
|
expect(recentObjectsList.getByRole('listitem', { name: clock.name }).locator('a').getByText(folderA.name)).toBeTruthy();
|
||||||
|
expect(recentObjectsList.getByRole('listitem').nth(1).getByText(folderA.name)).toBeTruthy();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -24,10 +24,18 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../pluginFixtures');
|
const { test, expect } = require('../../pluginFixtures');
|
||||||
const { createDomainObjectWithDefaults } = require('../../appActions');
|
const { createDomainObjectWithDefaults, selectInspectorTab } = require('../../appActions');
|
||||||
const { v4: uuid } = require('uuid');
|
const { v4: uuid } = require('uuid');
|
||||||
|
|
||||||
test.describe('Grand Search', () => {
|
test.describe('Grand Search', () => {
|
||||||
|
const searchResultSelector = '.c-gsearch-result__title';
|
||||||
|
const searchResultDropDownSelector = '.c-gsearch__results';
|
||||||
|
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto("./", { waitUntil: "networkidle" });
|
||||||
|
});
|
||||||
|
|
||||||
test('Can search for objects, and subsequent search dropdown behaves properly', async ({ page, openmctConfig }) => {
|
test('Can search for objects, and subsequent search dropdown behaves properly', async ({ page, openmctConfig }) => {
|
||||||
const { myItemsFolderName } = openmctConfig;
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
|
||||||
@ -42,7 +50,7 @@ test.describe('Grand Search', () => {
|
|||||||
await expect(page.locator('[aria-label="Search Result"] >> nth=2')).toContainText(`Clock C ${myItemsFolderName} Red Folder Blue Folder`);
|
await expect(page.locator('[aria-label="Search Result"] >> nth=2')).toContainText(`Clock C ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
await expect(page.locator('[aria-label="Search Result"] >> nth=3')).toContainText(`Clock D ${myItemsFolderName} Red Folder Blue Folder`);
|
await expect(page.locator('[aria-label="Search Result"] >> nth=3')).toContainText(`Clock D ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
// Click the Elements pool to dismiss the search menu
|
// Click the Elements pool to dismiss the search menu
|
||||||
await page.locator('.l-pane__label:has-text("Elements")').click();
|
await selectInspectorTab(page, 'Elements');
|
||||||
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toBeHidden();
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toBeHidden();
|
||||||
|
|
||||||
await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').click();
|
await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').click();
|
||||||
@ -55,7 +63,7 @@ test.describe('Grand Search', () => {
|
|||||||
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toContainText(`Clock A ${myItemsFolderName} Red Folder Blue Folder`);
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toContainText(`Clock A ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
|
|
||||||
// Click [aria-label="OpenMCT Search"] a >> nth=0
|
// Click [aria-label="OpenMCT Search"] a >> nth=0
|
||||||
await page.locator('[aria-label="OpenMCT Search"] a').first().click();
|
await page.locator('[aria-label="Search Result"] >> nth=0').click();
|
||||||
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toBeHidden();
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toBeHidden();
|
||||||
|
|
||||||
// Fill [aria-label="OpenMCT Search"] input[type="search"]
|
// Fill [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
@ -89,15 +97,8 @@ test.describe('Grand Search', () => {
|
|||||||
await expect(page.locator('[aria-label="Search Result"] >> nth=2')).toContainText(`Clock C ${myItemsFolderName} Red Folder Blue Folder`);
|
await expect(page.locator('[aria-label="Search Result"] >> nth=2')).toContainText(`Clock C ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
await expect(page.locator('[aria-label="Search Result"] >> nth=3')).toContainText(`Clock D ${myItemsFolderName} Red Folder Blue Folder`);
|
await expect(page.locator('[aria-label="Search Result"] >> nth=3')).toContainText(`Clock D ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
});
|
});
|
||||||
});
|
|
||||||
|
|
||||||
test.describe("Search Tests @unstable", () => {
|
|
||||||
const searchResultSelector = '.c-gsearch-result__title';
|
|
||||||
|
|
||||||
test('Validate empty search result', async ({ page }) => {
|
test('Validate empty search result', async ({ page }) => {
|
||||||
// Go to baseURL
|
|
||||||
await page.goto("./", { waitUntil: "networkidle" });
|
|
||||||
|
|
||||||
// Invalid search for objects
|
// Invalid search for objects
|
||||||
await page.type("input[type=search]", 'not found');
|
await page.type("input[type=search]", 'not found');
|
||||||
|
|
||||||
@ -105,7 +106,7 @@ test.describe("Search Tests @unstable", () => {
|
|||||||
await waitForSearchCompletion(page);
|
await waitForSearchCompletion(page);
|
||||||
|
|
||||||
// Get the search results
|
// Get the search results
|
||||||
const searchResults = await page.locator(searchResultSelector);
|
const searchResults = page.locator(searchResultSelector);
|
||||||
|
|
||||||
// Verify that no results are found
|
// Verify that no results are found
|
||||||
expect(await searchResults.count()).toBe(0);
|
expect(await searchResults.count()).toBe(0);
|
||||||
@ -115,9 +116,6 @@ test.describe("Search Tests @unstable", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test('Validate single object in search result @couchdb', async ({ page }) => {
|
test('Validate single object in search result @couchdb', async ({ page }) => {
|
||||||
//Go to baseURL
|
|
||||||
await page.goto("./", { waitUntil: "networkidle" });
|
|
||||||
|
|
||||||
// Create a folder object
|
// Create a folder object
|
||||||
const folderName = uuid();
|
const folderName = uuid();
|
||||||
await createDomainObjectWithDefaults(page, {
|
await createDomainObjectWithDefaults(page, {
|
||||||
@ -135,25 +133,61 @@ test.describe("Search Tests @unstable", () => {
|
|||||||
const searchResults = page.locator(searchResultSelector);
|
const searchResults = page.locator(searchResultSelector);
|
||||||
|
|
||||||
// Verify that one result is found
|
// Verify that one result is found
|
||||||
|
await expect(searchResults).toBeVisible();
|
||||||
expect(await searchResults.count()).toBe(1);
|
expect(await searchResults.count()).toBe(1);
|
||||||
await expect(searchResults).toHaveText(folderName);
|
await expect(searchResults).toHaveText(folderName);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('Search results are debounced @couchdb', async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/6179'
|
||||||
|
});
|
||||||
|
await createObjectsForSearch(page);
|
||||||
|
|
||||||
|
let networkRequests = [];
|
||||||
|
page.on('request', (request) => {
|
||||||
|
const searchRequest = request.url().endsWith('_find');
|
||||||
|
const fetchRequest = request.resourceType() === 'fetch';
|
||||||
|
if (searchRequest && fetchRequest) {
|
||||||
|
networkRequests.push(request);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Full search for object
|
||||||
|
await page.type("input[type=search]", 'Clock', { delay: 100 });
|
||||||
|
|
||||||
|
// Wait for search to finish
|
||||||
|
await waitForSearchCompletion(page);
|
||||||
|
|
||||||
|
// Network requests for the composite telemetry with multiple items should be:
|
||||||
|
// 1. batched request for latest telemetry using the bulk API
|
||||||
|
expect(networkRequests.length).toBe(1);
|
||||||
|
|
||||||
|
const searchResultDropDown = await page.locator(searchResultDropDownSelector);
|
||||||
|
|
||||||
|
await expect(searchResultDropDown).toContainText('Clock A');
|
||||||
|
});
|
||||||
|
|
||||||
test("Validate multiple objects in search results return partial matches", async ({ page }) => {
|
test("Validate multiple objects in search results return partial matches", async ({ page }) => {
|
||||||
test.info().annotations.push({
|
test.info().annotations.push({
|
||||||
type: 'issue',
|
type: 'issue',
|
||||||
description: 'https://github.com/nasa/openmct/issues/4667'
|
description: 'https://github.com/nasa/openmct/issues/4667'
|
||||||
});
|
});
|
||||||
|
|
||||||
// Go to baseURL
|
|
||||||
await page.goto("/", { waitUntil: "networkidle" });
|
|
||||||
|
|
||||||
// Create folder objects
|
// Create folder objects
|
||||||
const folderName = "e928a26e-e924-4ea0";
|
const folderName1 = "e928a26e-e924-4ea0";
|
||||||
const folderName2 = "e928a26e-e924-4001";
|
const folderName2 = "e928a26e-e924-4001";
|
||||||
|
|
||||||
await createFolderObject(page, folderName);
|
await createDomainObjectWithDefaults(page, {
|
||||||
await createFolderObject(page, folderName2);
|
type: 'Folder',
|
||||||
|
name: folderName1
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: folderName2
|
||||||
|
});
|
||||||
|
|
||||||
// Partial search for objects
|
// Partial search for objects
|
||||||
await page.type("input[type=search]", 'e928a26e');
|
await page.type("input[type=search]", 'e928a26e');
|
||||||
@ -161,36 +195,22 @@ test.describe("Search Tests @unstable", () => {
|
|||||||
// Wait for search to finish
|
// Wait for search to finish
|
||||||
await waitForSearchCompletion(page);
|
await waitForSearchCompletion(page);
|
||||||
|
|
||||||
// Get the search results
|
const searchResultDropDown = page.locator(searchResultDropDownSelector);
|
||||||
const searchResults = await page.locator(searchResultSelector);
|
|
||||||
|
|
||||||
// Verify that the search result/s correctly match the search query
|
// Verify that the search result/s correctly match the search query
|
||||||
|
await expect(searchResultDropDown).toContainText(folderName1);
|
||||||
|
await expect(searchResultDropDown).toContainText(folderName2);
|
||||||
|
|
||||||
|
// Get the search results
|
||||||
|
const searchResults = page.locator(searchResultSelector);
|
||||||
|
// Verify that two results are found
|
||||||
expect(await searchResults.count()).toBe(2);
|
expect(await searchResults.count()).toBe(2);
|
||||||
await expect(await searchResults.first()).toHaveText(folderName);
|
|
||||||
await expect(await searchResults.last()).toHaveText(folderName2);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
async function createFolderObject(page, folderName) {
|
|
||||||
// Open Create menu
|
|
||||||
await page.locator('button:has-text("Create")').click();
|
|
||||||
|
|
||||||
// Select Folder object
|
|
||||||
await page.locator('text=Folder').nth(1).click();
|
|
||||||
|
|
||||||
// Click folder title to enter edit mode
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').click();
|
|
||||||
|
|
||||||
// Enter folder name
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').fill(folderName);
|
|
||||||
|
|
||||||
// Create folder object
|
|
||||||
await page.locator('button:has-text("OK")').click();
|
|
||||||
}
|
|
||||||
|
|
||||||
async function waitForSearchCompletion(page) {
|
async function waitForSearchCompletion(page) {
|
||||||
// Wait loading spinner to disappear
|
// Wait loading spinner to disappear
|
||||||
await page.waitForSelector('.c-tree-and-search__loading', { state: 'detached' });
|
await page.waitForSelector('.search-finished');
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -198,9 +218,6 @@ async function waitForSearchCompletion(page) {
|
|||||||
* @param {import('@playwright/test').Page} page
|
* @param {import('@playwright/test').Page} page
|
||||||
*/
|
*/
|
||||||
async function createObjectsForSearch(page) {
|
async function createObjectsForSearch(page) {
|
||||||
//Go to baseURL
|
|
||||||
await page.goto('./', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
const redFolder = await createDomainObjectWithDefaults(page, {
|
const redFolder = await createDomainObjectWithDefaults(page, {
|
||||||
type: 'Folder',
|
type: 'Folder',
|
||||||
name: 'Red Folder'
|
name: 'Red Folder'
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -26,10 +26,80 @@ const {
|
|||||||
openObjectTreeContextMenu
|
openObjectTreeContextMenu
|
||||||
} = require('../../appActions.js');
|
} = require('../../appActions.js');
|
||||||
|
|
||||||
test.describe('Tree operations', () => {
|
test.describe('Main Tree', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Creating a child object within a folder and immediately opening it shows the created object in the tree @couchdb', async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/5975'
|
||||||
|
});
|
||||||
|
|
||||||
|
const folder = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder'
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.getByTitle('Show selected item in tree').click();
|
||||||
|
|
||||||
|
const clock = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Clock',
|
||||||
|
parent: folder.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await expandTreePaneItemByName(page, folder.name);
|
||||||
|
await assertTreeItemIsVisible(page, clock.name);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Creating a child object on one tab and expanding its parent on the other shows the correct composition @2p', async ({ page, openmctConfig }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/6391'
|
||||||
|
});
|
||||||
|
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
const page2 = await page.context().newPage();
|
||||||
|
|
||||||
|
// Both pages: Go to baseURL
|
||||||
|
await Promise.all([
|
||||||
|
page.goto('./', { waitUntil: 'networkidle' }),
|
||||||
|
page2.goto('./', { waitUntil: 'networkidle' })
|
||||||
|
]);
|
||||||
|
|
||||||
|
const page1Folder = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder'
|
||||||
|
});
|
||||||
|
|
||||||
|
await expandTreePaneItemByName(page2, myItemsFolderName);
|
||||||
|
await assertTreeItemIsVisible(page2, page1Folder.name);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Creating a child object on one tab and expanding its parent on the other shows the correct composition @couchdb @2p', async ({ page, openmctConfig }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/6391'
|
||||||
|
});
|
||||||
|
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
const page2 = await page.context().newPage();
|
||||||
|
|
||||||
|
// Both pages: Go to baseURL
|
||||||
|
await Promise.all([
|
||||||
|
page.goto('./', { waitUntil: 'networkidle' }),
|
||||||
|
page2.goto('./', { waitUntil: 'networkidle' })
|
||||||
|
]);
|
||||||
|
|
||||||
|
const page1Folder = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder'
|
||||||
|
});
|
||||||
|
|
||||||
|
await expandTreePaneItemByName(page2, myItemsFolderName);
|
||||||
|
await assertTreeItemIsVisible(page2, page1Folder.name);
|
||||||
|
});
|
||||||
|
|
||||||
test('Renaming an object reorders the tree @unstable', async ({ page, openmctConfig }) => {
|
test('Renaming an object reorders the tree @unstable', async ({ page, openmctConfig }) => {
|
||||||
const { myItemsFolderName } = openmctConfig;
|
const { myItemsFolderName } = openmctConfig;
|
||||||
await page.goto('./', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
await createDomainObjectWithDefaults(page, {
|
await createDomainObjectWithDefaults(page, {
|
||||||
type: 'Folder',
|
type: 'Folder',
|
||||||
@ -111,17 +181,30 @@ async function getAndAssertTreeItems(page, expected) {
|
|||||||
expect(allTexts).toEqual(expected);
|
expect(allTexts).toEqual(expected);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function assertTreeItemIsVisible(page, name) {
|
||||||
|
const mainTree = page.getByRole('tree', {
|
||||||
|
name: 'Main Tree'
|
||||||
|
});
|
||||||
|
const treeItem = mainTree.getByRole('treeitem', {
|
||||||
|
name
|
||||||
|
});
|
||||||
|
|
||||||
|
await expect(treeItem).toBeVisible();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {import('@playwright/test').Page} page
|
* @param {import('@playwright/test').Page} page
|
||||||
* @param {string} name
|
* @param {string} name
|
||||||
*/
|
*/
|
||||||
async function expandTreePaneItemByName(page, name) {
|
async function expandTreePaneItemByName(page, name) {
|
||||||
const treePane = page.getByRole('tree', {
|
const mainTree = page.getByRole('tree', {
|
||||||
name: 'Main Tree'
|
name: 'Main Tree'
|
||||||
});
|
});
|
||||||
const treeItem = treePane.locator(`role=treeitem[expanded=false][name=/${name}/]`);
|
const treeItem = mainTree.getByRole('treeitem', {
|
||||||
const expandTriangle = treeItem.locator('.c-disclosure-triangle');
|
name,
|
||||||
await expandTriangle.click();
|
expanded: false
|
||||||
|
});
|
||||||
|
await treeItem.locator('.c-disclosure-triangle').click();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -138,6 +138,7 @@ test.describe('Performance tests', () => {
|
|||||||
await page.evaluate(() => window.performance.mark("notebook-search-processed"));
|
await page.evaluate(() => window.performance.mark("notebook-search-processed"));
|
||||||
|
|
||||||
//Clear Search
|
//Clear Search
|
||||||
|
await page.locator('.c-search.c-notebook__search .c-search__input').hover();
|
||||||
await page.locator('.c-search.c-notebook__search .c-search__clear-input').click();
|
await page.locator('.c-search.c-notebook__search .c-search__clear-input').click();
|
||||||
await page.evaluate(() => window.performance.mark("notebook-search-processed"));
|
await page.evaluate(() => window.performance.mark("notebook-search-processed"));
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
/* eslint-disable no-undef */
|
/* eslint-disable no-undef */
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
52
e2e/tests/visual/plan.visual.spec.js
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2023, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test } = require('../../pluginFixtures');
|
||||||
|
const { setBoundsToSpanAllActivities } = require('../../helper/planningUtils');
|
||||||
|
const { createDomainObjectWithDefaults, createPlanFromJSON } = require('../../appActions');
|
||||||
|
const percySnapshot = require('@percy/playwright');
|
||||||
|
const examplePlanLarge = require('../../test-data/examplePlans/ExamplePlan_Large.json');
|
||||||
|
|
||||||
|
test.describe('Visual - Planning', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
});
|
||||||
|
test('Plan View', async ({ page, theme }) => {
|
||||||
|
const plan = await createPlanFromJSON(page, {
|
||||||
|
json: examplePlanLarge
|
||||||
|
});
|
||||||
|
|
||||||
|
await setBoundsToSpanAllActivities(page, examplePlanLarge, plan.url);
|
||||||
|
await percySnapshot(page, `Plan View (theme: ${theme})`);
|
||||||
|
});
|
||||||
|
test('Gantt Chart View', async ({ page, theme }) => {
|
||||||
|
const ganttChart = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Gantt Chart'
|
||||||
|
});
|
||||||
|
await createPlanFromJSON(page, {
|
||||||
|
json: examplePlanLarge,
|
||||||
|
parent: ganttChart.uuid
|
||||||
|
});
|
||||||
|
await setBoundsToSpanAllActivities(page, examplePlanLarge, ganttChart.url);
|
||||||
|
await percySnapshot(page, `Gantt Chart View (theme: ${theme})`);
|
||||||
|
});
|
||||||
|
});
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -20,11 +20,23 @@
|
|||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
import availableTags from './tags.json';
|
import availableTags from './tags.json';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@typedef {{
|
||||||
|
namespaceToSaveAnnotations: string
|
||||||
|
}} TagsPluginOptions
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef {TagsPluginOptions} options
|
||||||
* @returns {function} The plugin install function
|
* @returns {function} The plugin install function
|
||||||
*/
|
*/
|
||||||
export default function exampleTagsPlugin() {
|
export default function exampleTagsPlugin(options) {
|
||||||
return function install(openmct) {
|
return function install(openmct) {
|
||||||
|
if (options?.namespaceToSaveAnnotations) {
|
||||||
|
openmct.annotation.setNamespaceToSaveAnnotations(options?.namespaceToSaveAnnotations);
|
||||||
|
}
|
||||||
|
|
||||||
Object.keys(availableTags.tags).forEach(tagKey => {
|
Object.keys(availableTags.tags).forEach(tagKey => {
|
||||||
const tagDefinition = availableTags.tags[tagKey];
|
const tagDefinition = availableTags.tags[tagKey];
|
||||||
openmct.annotation.defineTag(tagKey, tagDefinition);
|
openmct.annotation.defineTag(tagKey, tagDefinition);
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2023, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|