Skip to content

Commit 5ee9b66

Browse files
committed
Merge branch 'mct-7442-7959' of https://github.com/nasa/openmct into mct-7442-7959
merging latest master changes
2 parents 8b2d77c + 07eaac2 commit 5ee9b66

File tree

13 files changed

+251
-57
lines changed

13 files changed

+251
-57
lines changed

.github/workflows/e2e-couchdb.yml

+7-3
Original file line numberDiff line numberDiff line change
@@ -51,7 +51,7 @@ jobs:
5151
env:
5252
COMMIT_INFO_SHA: ${{github.event.pull_request.head.sha }}
5353
run: npm run test:e2e:couchdb
54-
54+
5555
- name: Generate Code Coverage Report
5656
run: npm run cov:e2e:report
5757

@@ -66,15 +66,19 @@ jobs:
6666

6767
- name: Archive test results
6868
if: success() || failure()
69-
uses: actions/upload-artifact@v3
69+
uses: actions/upload-artifact@v4
7070
with:
71+
name: e2e-couchdb-test-results
7172
path: test-results
73+
overwrite: true
7274

7375
- name: Archive html test results
7476
if: success() || failure()
75-
uses: actions/upload-artifact@v3
77+
uses: actions/upload-artifact@v4
7678
with:
79+
name: e2e-couchdb-html-test-results
7780
path: html-test-results
81+
overwrite: true
7882

7983
- name: Remove pr:e2e:couchdb label (if present)
8084
if: always()

.github/workflows/e2e-flakefinder.yml

+3-1
Original file line numberDiff line numberDiff line change
@@ -38,9 +38,11 @@ jobs:
3838

3939
- name: Archive test results
4040
if: success() || failure()
41-
uses: actions/upload-artifact@v3
41+
uses: actions/upload-artifact@v4
4242
with:
43+
name: e2e-flakefinder-test-results
4344
path: test-results
45+
overwrite: true
4446

4547
- name: Remove pr:e2e:flakefinder label (if present)
4648
if: always()

.github/workflows/e2e-perf.yml

+3-1
Original file line numberDiff line numberDiff line change
@@ -35,9 +35,11 @@ jobs:
3535
- run: npm run test:perf:memory
3636
- name: Archive test results
3737
if: success() || failure()
38-
uses: actions/upload-artifact@v3
38+
uses: actions/upload-artifact@v4
3939
with:
40+
name: e2e-perf-test-results
4041
path: test-results
42+
overwrite: true
4143

4244
- name: Remove pr:e2e:perf label (if present)
4345
if: always()

.github/workflows/e2e-pr.yml

+3-1
Original file line numberDiff line numberDiff line change
@@ -45,9 +45,11 @@ jobs:
4545
npm run cov:e2e:full:publish
4646
- name: Archive test results
4747
if: success() || failure()
48-
uses: actions/upload-artifact@v3
48+
uses: actions/upload-artifact@v4
4949
with:
50+
name: e2e-pr-test-results
5051
path: test-results
52+
overwrite: true
5153

5254
- name: Remove pr:e2e label (if present)
5355
if: always()

e2e/tests/functional/planning/timestrip.e2e.spec.js

+68-15
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,9 @@ import {
2424
createDomainObjectWithDefaults,
2525
createPlanFromJSON,
2626
navigateToObjectWithFixedTimeBounds,
27-
setFixedIndependentTimeConductorBounds
27+
setFixedIndependentTimeConductorBounds,
28+
setFixedTimeMode,
29+
setTimeConductorBounds
2830
} from '../../../appActions.js';
2931
import { expect, test } from '../../../pluginFixtures.js';
3032

@@ -74,29 +76,22 @@ const testPlan = {
7476
};
7577

7678
test.describe('Time Strip', () => {
77-
test('Create two Time Strips, add a single Plan to both, and verify they can have separate Independent Time Contexts', async ({
78-
page
79-
}) => {
80-
test.info().annotations.push({
81-
type: 'issue',
82-
description: 'https://github.com/nasa/openmct/issues/5627'
83-
});
84-
85-
// Constant locators
86-
const activityBounds = page.locator('.activity-bounds');
79+
let timestrip;
80+
let plan;
8781

82+
test.beforeEach(async ({ page }) => {
8883
// Goto baseURL
8984
await page.goto('./', { waitUntil: 'domcontentloaded' });
9085

91-
const timestrip = await test.step('Create a Time Strip', async () => {
86+
timestrip = await test.step('Create a Time Strip', async () => {
9287
const createdTimeStrip = await createDomainObjectWithDefaults(page, { type: 'Time Strip' });
9388
const objectName = await page.locator('.l-browse-bar__object-name').innerText();
9489
expect(objectName).toBe(createdTimeStrip.name);
9590

9691
return createdTimeStrip;
9792
});
9893

99-
const plan = await test.step('Create a Plan and add it to the timestrip', async () => {
94+
plan = await test.step('Create a Plan and add it to the timestrip', async () => {
10095
const createdPlan = await createPlanFromJSON(page, {
10196
name: 'Test Plan',
10297
json: testPlan
@@ -110,6 +105,22 @@ test.describe('Time Strip', () => {
110105
.dragTo(page.getByLabel('Object View'));
111106
await page.getByLabel('Save').click();
112107
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
108+
109+
return createdPlan;
110+
});
111+
});
112+
test('Create two Time Strips, add a single Plan to both, and verify they can have separate Independent Time Contexts', async ({
113+
page
114+
}) => {
115+
test.info().annotations.push({
116+
type: 'issue',
117+
description: 'https://github.com/nasa/openmct/issues/5627'
118+
});
119+
120+
// Constant locators
121+
const activityBounds = page.locator('.activity-bounds');
122+
123+
await test.step('Set time strip to fixed timespan mode and verify activities', async () => {
113124
const startBound = testPlan.TEST_GROUP[0].start;
114125
const endBound = testPlan.TEST_GROUP[testPlan.TEST_GROUP.length - 1].end;
115126

@@ -119,8 +130,6 @@ test.describe('Time Strip', () => {
119130
// Verify all events are displayed
120131
const eventCount = await page.locator('.activity-bounds').count();
121132
expect(eventCount).toEqual(testPlan.TEST_GROUP.length);
122-
123-
return createdPlan;
124133
});
125134

126135
await test.step('TimeStrip can use the Independent Time Conductor', async () => {
@@ -177,4 +186,48 @@ test.describe('Time Strip', () => {
177186
expect(await activityBounds.count()).toEqual(1);
178187
});
179188
});
189+
190+
test('Time strip now line', async ({ page }) => {
191+
test.info().annotations.push({
192+
type: 'issue',
193+
description: 'https://github.com/nasa/openmct/issues/7817'
194+
});
195+
196+
await test.step('Is displayed in realtime mode', async () => {
197+
await expect(page.getByLabel('Now Marker')).toBeVisible();
198+
});
199+
200+
await test.step('Is hidden when out of bounds of the time axis', async () => {
201+
// Switch to fixed timespan mode
202+
await setFixedTimeMode(page);
203+
// Get the end bounds
204+
const endBounds = await page.getByLabel('End bounds').textContent();
205+
206+
// Add 2 minutes to end bound datetime and use it as the new end time
207+
let endTimeStamp = new Date(endBounds);
208+
endTimeStamp.setUTCMinutes(endTimeStamp.getUTCMinutes() + 2);
209+
const endDate = endTimeStamp.toISOString().split('T')[0];
210+
const milliseconds = endTimeStamp.getMilliseconds();
211+
const endTime = endTimeStamp.toISOString().split('T')[1].replace(`.${milliseconds}Z`, '');
212+
213+
// Subtract 1 minute from the end bound and use it as the new start time
214+
let startTimeStamp = new Date(endBounds);
215+
startTimeStamp.setUTCMinutes(startTimeStamp.getUTCMinutes() + 1);
216+
const startDate = startTimeStamp.toISOString().split('T')[0];
217+
const startMilliseconds = startTimeStamp.getMilliseconds();
218+
const startTime = startTimeStamp
219+
.toISOString()
220+
.split('T')[1]
221+
.replace(`.${startMilliseconds}Z`, '');
222+
// Set fixed timespan mode to the future so that "now" is out of bounds.
223+
await setTimeConductorBounds(page, {
224+
startDate,
225+
endDate,
226+
startTime,
227+
endTime
228+
});
229+
230+
await expect(page.getByLabel('Now Marker')).toBeHidden();
231+
});
232+
});
180233
});

e2e/tests/functional/plugins/plot/plotControls.e2e.spec.js

+38
Original file line numberDiff line numberDiff line change
@@ -108,4 +108,42 @@ test.describe('Plot Controls', () => {
108108
// Expect before and after plot points to match
109109
await expect(plotPixelSizeAtPause).toEqual(plotPixelSizeAfterWait);
110110
});
111+
112+
/*
113+
Test to verify that switching a plot's time context from global to
114+
its own independent time context and then back to global context works correctly.
115+
116+
After switching from fixed time mode (ITC) to real time mode (global context),
117+
the pause control for the plot should be available, indicating that it is following the right context.
118+
*/
119+
test('Plots follow the right time context', async ({ page }) => {
120+
// Set global time conductor to real-time mode
121+
await setRealTimeMode(page);
122+
123+
// hover over plot for plot controls
124+
await page.getByLabel('Plot Canvas').hover();
125+
// Ensure pause control is visible since global time conductor is in Real time mode.
126+
await expect(page.getByTitle('Pause incoming real-time data')).toBeVisible();
127+
128+
// Toggle independent time conductor ON
129+
await page.getByLabel('Enable Independent Time Conductor').click();
130+
131+
// Bring up the independent time conductor popup and switch to fixed time mode
132+
await page.getByLabel('Independent Time Conductor Settings').click();
133+
await page.getByLabel('Independent Time Conductor Mode Menu').click();
134+
await page.getByRole('menuitem', { name: /Fixed Timespan/ }).click();
135+
136+
// hover over plot for plot controls
137+
await page.getByLabel('Plot Canvas').hover();
138+
// Ensure pause control is no longer visible since the plot is following the independent time context
139+
await expect(page.getByTitle('Pause incoming real-time data')).toBeHidden();
140+
141+
// Toggle independent time conductor OFF - Note that the global time conductor is still in Real time mode
142+
await page.getByLabel('Disable Independent Time Conductor').click();
143+
144+
// hover over plot for plot controls
145+
await page.getByLabel('Plot Canvas').hover();
146+
// Ensure pause control is visible since the global time conductor is in real time mode
147+
await expect(page.getByTitle('Pause incoming real-time data')).toBeVisible();
148+
});
111149
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
/*****************************************************************************
2+
* Open MCT, Copyright (c) 2014-2025, United States Government
3+
* as represented by the Administrator of the National Aeronautics and Space
4+
* Administration. All rights reserved.
5+
*
6+
* Open MCT is licensed under the Apache License, Version 2.0 (the
7+
* "License"); you may not use this file except in compliance with the License.
8+
* You may obtain a copy of the License at
9+
* http://www.apache.org/licenses/LICENSE-2.0.
10+
*
11+
* Unless required by applicable law or agreed to in writing, software
12+
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
13+
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
14+
* License for the specific language governing permissions and limitations
15+
* under the License.
16+
*
17+
* Open MCT includes source code licensed under additional open source
18+
* licenses. See the Open Source Licenses file (LICENSES.md) included with
19+
* this source code distribution or the Licensing information page available
20+
* at runtime from the About dialog for additional information.
21+
*****************************************************************************/
22+
23+
/*
24+
* This test suite is dedicated to testing the rendering and interaction of plots.
25+
*
26+
*/
27+
28+
import { createDomainObjectWithDefaults } from '../../../../appActions.js';
29+
import { expect, test } from '../../../../pluginFixtures.js';
30+
31+
test.describe('Plot Controls in compact mode', () => {
32+
let timeStrip;
33+
34+
test.beforeEach(async ({ page }) => {
35+
// Open a browser, navigate to the main page, and wait until all networkevents to resolve
36+
await page.goto('./', { waitUntil: 'domcontentloaded' });
37+
timeStrip = await createDomainObjectWithDefaults(page, {
38+
type: 'Time Strip'
39+
});
40+
41+
// Create an overlay plot with a sine wave generator
42+
await createDomainObjectWithDefaults(page, {
43+
type: 'Sine Wave Generator',
44+
parent: timeStrip.uuid
45+
});
46+
await page.goto(`${timeStrip.url}`);
47+
});
48+
49+
test('Plots show cursor guides', async ({ page }) => {
50+
// hover over plot for plot controls
51+
await page.getByLabel('Plot Canvas').hover();
52+
// click on cursor guides control
53+
await page.getByTitle('Toggle cursor guides').click();
54+
await page.getByLabel('Plot Canvas').hover();
55+
await expect(page.getByLabel('Vertical cursor guide')).toBeVisible();
56+
await expect(page.getByLabel('Horizontal cursor guide')).toBeVisible();
57+
});
58+
});

src/api/time/IndependentTimeContext.js

+21-1
Original file line numberDiff line numberDiff line change
@@ -359,6 +359,18 @@ class IndependentTimeContext extends TimeContext {
359359
}
360360
}
361361

362+
/**
363+
* @returns {boolean}
364+
* @override
365+
*/
366+
isFixed() {
367+
if (this.upstreamTimeContext) {
368+
return this.upstreamTimeContext.isFixed(...arguments);
369+
} else {
370+
return super.isFixed(...arguments);
371+
}
372+
}
373+
362374
/**
363375
* @returns {number}
364376
* @override
@@ -400,7 +412,7 @@ class IndependentTimeContext extends TimeContext {
400412
}
401413

402414
/**
403-
* Reset the time context to the global time context
415+
* Reset the time context from the global time context
404416
*/
405417
resetContext() {
406418
if (this.upstreamTimeContext) {
@@ -428,6 +440,10 @@ class IndependentTimeContext extends TimeContext {
428440
// Emit bounds so that views that are changing context get the upstream bounds
429441
this.emit('bounds', this.getBounds());
430442
this.emit(TIME_CONTEXT_EVENTS.boundsChanged, this.getBounds());
443+
// Also emit the mode in case it's different from previous time context
444+
if (this.getMode()) {
445+
this.emit(TIME_CONTEXT_EVENTS.modeChanged, this.#copy(this.getMode()));
446+
}
431447
}
432448

433449
/**
@@ -502,6 +518,10 @@ class IndependentTimeContext extends TimeContext {
502518
// Emit bounds so that views that are changing context get the upstream bounds
503519
this.emit('bounds', this.getBounds());
504520
this.emit(TIME_CONTEXT_EVENTS.boundsChanged, this.getBounds());
521+
// Also emit the mode in case it's different from the global time context
522+
if (this.getMode()) {
523+
this.emit(TIME_CONTEXT_EVENTS.modeChanged, this.#copy(this.getMode()));
524+
}
505525
// now that the view's context is set, tell others to check theirs in case they were following this view's context.
506526
this.globalTimeContext.emit('refreshContext', viewKey);
507527
}

src/api/time/TimeAPI.js

+5-1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@
2323
import { FIXED_MODE_KEY, REALTIME_MODE_KEY } from '@/api/time/constants';
2424
import IndependentTimeContext from '@/api/time/IndependentTimeContext';
2525

26+
import { TIME_CONTEXT_EVENTS } from './constants';
2627
import GlobalTimeContext from './GlobalTimeContext.js';
2728

2829
/**
@@ -142,7 +143,7 @@ class TimeAPI extends GlobalTimeContext {
142143
addIndependentContext(key, value, clockKey) {
143144
let timeContext = this.getIndependentContext(key);
144145

145-
//stop following upstream time context since the view has it's own
146+
//stop following upstream time context since the view has its own
146147
timeContext.resetContext();
147148

148149
if (clockKey) {
@@ -152,6 +153,9 @@ class TimeAPI extends GlobalTimeContext {
152153
timeContext.setMode(FIXED_MODE_KEY, value);
153154
}
154155

156+
// Also emit the mode in case it's different from the previous time context
157+
timeContext.emit(TIME_CONTEXT_EVENTS.modeChanged, structuredClone(timeContext.getMode()));
158+
155159
// Notify any nested views to update, pass in the viewKey so that particular view can skip getting an upstream context
156160
this.emit('refreshContext', key);
157161

0 commit comments

Comments
 (0)