mirror of
https://github.com/nasa/openmct.git
synced 2025-05-07 19:18:36 +00:00
Independent time conductor related handling for plot synchronization. (#7956)
Some checks failed
CodeQL / Analyze (push) Has been cancelled
Some checks failed
CodeQL / Analyze (push) Has been cancelled
* Ensure that the mode set when independent time conductor is enabled/disabled is propagated correctly. Also ensure that global time conductor changes are not picked up by the independent time conductor when the user has enabled it at least once before * Use structuredClone instead of deep copy * Add e2e test * Assert that you're in fixed mode after sync time conductor * Comment explaining new time context test * Change test to be a little less complicated * Fix linting errors
This commit is contained in:
parent
a6517bb33e
commit
ecd120387c
@ -108,4 +108,42 @@ test.describe('Plot Controls', () => {
|
|||||||
// Expect before and after plot points to match
|
// Expect before and after plot points to match
|
||||||
await expect(plotPixelSizeAtPause).toEqual(plotPixelSizeAfterWait);
|
await expect(plotPixelSizeAtPause).toEqual(plotPixelSizeAfterWait);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
/*
|
||||||
|
Test to verify that switching a plot's time context from global to
|
||||||
|
its own independent time context and then back to global context works correctly.
|
||||||
|
|
||||||
|
After switching from fixed time mode (ITC) to real time mode (global context),
|
||||||
|
the pause control for the plot should be available, indicating that it is following the right context.
|
||||||
|
*/
|
||||||
|
test('Plots follow the right time context', async ({ page }) => {
|
||||||
|
// Set global time conductor to real-time mode
|
||||||
|
await setRealTimeMode(page);
|
||||||
|
|
||||||
|
// hover over plot for plot controls
|
||||||
|
await page.getByLabel('Plot Canvas').hover();
|
||||||
|
// Ensure pause control is visible since global time conductor is in Real time mode.
|
||||||
|
await expect(page.getByTitle('Pause incoming real-time data')).toBeVisible();
|
||||||
|
|
||||||
|
// Toggle independent time conductor ON
|
||||||
|
await page.getByLabel('Enable Independent Time Conductor').click();
|
||||||
|
|
||||||
|
// Bring up the independent time conductor popup and switch to fixed time mode
|
||||||
|
await page.getByLabel('Independent Time Conductor Settings').click();
|
||||||
|
await page.getByLabel('Independent Time Conductor Mode Menu').click();
|
||||||
|
await page.getByRole('menuitem', { name: /Fixed Timespan/ }).click();
|
||||||
|
|
||||||
|
// hover over plot for plot controls
|
||||||
|
await page.getByLabel('Plot Canvas').hover();
|
||||||
|
// Ensure pause control is no longer visible since the plot is following the independent time context
|
||||||
|
await expect(page.getByTitle('Pause incoming real-time data')).toBeHidden();
|
||||||
|
|
||||||
|
// Toggle independent time conductor OFF - Note that the global time conductor is still in Real time mode
|
||||||
|
await page.getByLabel('Disable Independent Time Conductor').click();
|
||||||
|
|
||||||
|
// hover over plot for plot controls
|
||||||
|
await page.getByLabel('Plot Canvas').hover();
|
||||||
|
// Ensure pause control is visible since the global time conductor is in real time mode
|
||||||
|
await expect(page.getByTitle('Pause incoming real-time data')).toBeVisible();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -359,6 +359,18 @@ class IndependentTimeContext extends TimeContext {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @returns {boolean}
|
||||||
|
* @override
|
||||||
|
*/
|
||||||
|
isFixed() {
|
||||||
|
if (this.upstreamTimeContext) {
|
||||||
|
return this.upstreamTimeContext.isFixed(...arguments);
|
||||||
|
} else {
|
||||||
|
return super.isFixed(...arguments);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @returns {number}
|
* @returns {number}
|
||||||
* @override
|
* @override
|
||||||
@ -400,7 +412,7 @@ class IndependentTimeContext extends TimeContext {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reset the time context to the global time context
|
* Reset the time context from the global time context
|
||||||
*/
|
*/
|
||||||
resetContext() {
|
resetContext() {
|
||||||
if (this.upstreamTimeContext) {
|
if (this.upstreamTimeContext) {
|
||||||
@ -428,6 +440,10 @@ class IndependentTimeContext extends TimeContext {
|
|||||||
// Emit bounds so that views that are changing context get the upstream bounds
|
// Emit bounds so that views that are changing context get the upstream bounds
|
||||||
this.emit('bounds', this.getBounds());
|
this.emit('bounds', this.getBounds());
|
||||||
this.emit(TIME_CONTEXT_EVENTS.boundsChanged, this.getBounds());
|
this.emit(TIME_CONTEXT_EVENTS.boundsChanged, this.getBounds());
|
||||||
|
// Also emit the mode in case it's different from previous time context
|
||||||
|
if (this.getMode()) {
|
||||||
|
this.emit(TIME_CONTEXT_EVENTS.modeChanged, this.#copy(this.getMode()));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -502,6 +518,10 @@ class IndependentTimeContext extends TimeContext {
|
|||||||
// Emit bounds so that views that are changing context get the upstream bounds
|
// Emit bounds so that views that are changing context get the upstream bounds
|
||||||
this.emit('bounds', this.getBounds());
|
this.emit('bounds', this.getBounds());
|
||||||
this.emit(TIME_CONTEXT_EVENTS.boundsChanged, this.getBounds());
|
this.emit(TIME_CONTEXT_EVENTS.boundsChanged, this.getBounds());
|
||||||
|
// Also emit the mode in case it's different from the global time context
|
||||||
|
if (this.getMode()) {
|
||||||
|
this.emit(TIME_CONTEXT_EVENTS.modeChanged, this.#copy(this.getMode()));
|
||||||
|
}
|
||||||
// now that the view's context is set, tell others to check theirs in case they were following this view's context.
|
// now that the view's context is set, tell others to check theirs in case they were following this view's context.
|
||||||
this.globalTimeContext.emit('refreshContext', viewKey);
|
this.globalTimeContext.emit('refreshContext', viewKey);
|
||||||
}
|
}
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
import { FIXED_MODE_KEY, REALTIME_MODE_KEY } from '@/api/time/constants';
|
import { FIXED_MODE_KEY, REALTIME_MODE_KEY } from '@/api/time/constants';
|
||||||
import IndependentTimeContext from '@/api/time/IndependentTimeContext';
|
import IndependentTimeContext from '@/api/time/IndependentTimeContext';
|
||||||
|
|
||||||
|
import { TIME_CONTEXT_EVENTS } from './constants';
|
||||||
import GlobalTimeContext from './GlobalTimeContext.js';
|
import GlobalTimeContext from './GlobalTimeContext.js';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -142,7 +143,7 @@ class TimeAPI extends GlobalTimeContext {
|
|||||||
addIndependentContext(key, value, clockKey) {
|
addIndependentContext(key, value, clockKey) {
|
||||||
let timeContext = this.getIndependentContext(key);
|
let timeContext = this.getIndependentContext(key);
|
||||||
|
|
||||||
//stop following upstream time context since the view has it's own
|
//stop following upstream time context since the view has its own
|
||||||
timeContext.resetContext();
|
timeContext.resetContext();
|
||||||
|
|
||||||
if (clockKey) {
|
if (clockKey) {
|
||||||
@ -152,6 +153,9 @@ class TimeAPI extends GlobalTimeContext {
|
|||||||
timeContext.setMode(FIXED_MODE_KEY, value);
|
timeContext.setMode(FIXED_MODE_KEY, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Also emit the mode in case it's different from the previous time context
|
||||||
|
timeContext.emit(TIME_CONTEXT_EVENTS.modeChanged, structuredClone(timeContext.getMode()));
|
||||||
|
|
||||||
// Notify any nested views to update, pass in the viewKey so that particular view can skip getting an upstream context
|
// Notify any nested views to update, pass in the viewKey so that particular view can skip getting an upstream context
|
||||||
this.emit('refreshContext', key);
|
this.emit('refreshContext', key);
|
||||||
|
|
||||||
|
@ -539,6 +539,7 @@ export default {
|
|||||||
this.followTimeContext();
|
this.followTimeContext();
|
||||||
},
|
},
|
||||||
followTimeContext() {
|
followTimeContext() {
|
||||||
|
this.updateMode();
|
||||||
this.updateDisplayBounds(this.timeContext.getBounds());
|
this.updateDisplayBounds(this.timeContext.getBounds());
|
||||||
this.timeContext.on('modeChanged', this.updateMode);
|
this.timeContext.on('modeChanged', this.updateMode);
|
||||||
this.timeContext.on('boundsChanged', this.updateDisplayBounds);
|
this.timeContext.on('boundsChanged', this.updateDisplayBounds);
|
||||||
|
@ -243,12 +243,20 @@ export default {
|
|||||||
this.timeContext.off(TIME_CONTEXT_EVENTS.modeChanged, this.setTimeOptionsMode);
|
this.timeContext.off(TIME_CONTEXT_EVENTS.modeChanged, this.setTimeOptionsMode);
|
||||||
},
|
},
|
||||||
setTimeOptionsClock(clock) {
|
setTimeOptionsClock(clock) {
|
||||||
|
// If the user has persisted any time options, then don't override them with global settings.
|
||||||
|
if (this.independentTCEnabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
this.setTimeOptionsOffsets();
|
this.setTimeOptionsOffsets();
|
||||||
this.timeOptions.clock = clock.key;
|
this.timeOptions.clock = clock.key;
|
||||||
},
|
},
|
||||||
setTimeOptionsMode(mode) {
|
setTimeOptionsMode(mode) {
|
||||||
this.setTimeOptionsOffsets();
|
// If the user has persisted any time options, then don't override them with global settings.
|
||||||
this.timeOptions.mode = mode;
|
if (this.independentTCEnabled) {
|
||||||
|
this.setTimeOptionsOffsets();
|
||||||
|
this.timeOptions.mode = mode;
|
||||||
|
this.isFixed = this.timeOptions.mode === FIXED_MODE_KEY;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
setTimeOptionsOffsets() {
|
setTimeOptionsOffsets() {
|
||||||
this.timeOptions.clockOffsets =
|
this.timeOptions.clockOffsets =
|
||||||
|
Loading…
x
Reference in New Issue
Block a user