From e3fcbe1a3525720c0494c0b4463f73581899916d Mon Sep 17 00:00:00 2001 From: Jesse Mazzella Date: Thu, 25 Jul 2024 16:55:50 -0700 Subject: [PATCH 1/2] fix(#7791): tc form shouldn't submit bounds changes on dismiss (#7792) * fix(#7791): tc form shouldn't submit bounds changes on dismiss * test(e2e): add tests for validating time conductor popup - update appAction for setting time conductor in fixed mode - add a11y to time conductor in fixed mode - update tests using `setTimeConductorBounds` * fix(#7791): actually fix the problem. Also, add a test. * test: add annotation to regression test * docs: comments * test: fix the reset image button flake ONCE AND FOR ALL - wait for the rightmost image thumbnail to be in the viewport :D * test: add tests for `setTimeConductorMode` and `setTimeConductorBounds` --- e2e/appActions.js | 73 ++++--- e2e/tests/framework/appActions.e2e.spec.js | 35 +++- .../generateLocalStorageData.e2e.spec.js | 25 ++- .../imagery/exampleImagery.e2e.spec.js | 3 + .../plugins/plot/logPlot.e2e.spec.js | 59 +++--- .../telemetryTable/telemetryTable.e2e.spec.js | 13 +- .../timeConductor/timeConductor.e2e.spec.js | 185 +++++++++++++----- e2e/tests/visual-a11y/imagery.visual.spec.js | 7 + .../timeConductor/ConductorInputsFixed.vue | 4 +- src/plugins/timeConductor/TimePopupFixed.vue | 70 +++---- 10 files changed, 302 insertions(+), 172 deletions(-) diff --git a/e2e/appActions.js b/e2e/appActions.js index dea7257a47..c14bceb020 100644 --- a/e2e/appActions.js +++ b/e2e/appActions.js @@ -436,61 +436,67 @@ async function setRealTimeMode(page) { /** * Set the values (hours, mins, secs) for the TimeConductor offsets when in realtime mode * @param {import('@playwright/test').Page} page - * @param {OffsetValues} offset - * @param {import('@playwright/test').Locator} offsetButton + * @param {OffsetValues} offset - Object containing offset values + * @param {boolean} [offset.submitChanges=true] - If true, submit the offset changes; otherwise, discard them */ async function setTimeConductorOffset( page, - { startHours, startMins, startSecs, endHours, endMins, endSecs } + { startHours, startMins, startSecs, endHours, endMins, endSecs, submitChanges = true } ) { if (startHours) { - await page.getByRole('spinbutton', { name: 'Start offset hours' }).fill(startHours); + await page.getByLabel('Start offset hours').fill(startHours); } if (startMins) { - await page.getByRole('spinbutton', { name: 'Start offset minutes' }).fill(startMins); + await page.getByLabel('Start offset minutes').fill(startMins); } if (startSecs) { - await page.getByRole('spinbutton', { name: 'Start offset seconds' }).fill(startSecs); + await page.getByLabel('Start offset seconds').fill(startSecs); } if (endHours) { - await page.getByRole('spinbutton', { name: 'End offset hours' }).fill(endHours); + await page.getByLabel('End offset hours').fill(endHours); } if (endMins) { - await page.getByRole('spinbutton', { name: 'End offset minutes' }).fill(endMins); + await page.getByLabel('End offset minutes').fill(endMins); } if (endSecs) { - await page.getByRole('spinbutton', { name: 'End offset seconds' }).fill(endSecs); + await page.getByLabel('End offset seconds').fill(endSecs); } // Click the check button - await page.locator('.pr-time-input--buttons .icon-check').click(); + if (submitChanges) { + await page.getByLabel('Submit time offsets').click(); + } else { + await page.getByLabel('Discard changes and close time popup').click(); + } } /** * Set the values (hours, mins, secs) for the start time offset when in realtime mode * @param {import('@playwright/test').Page} page * @param {OffsetValues} offset + * @param {boolean} [submit=true] If true, submit the offset changes; otherwise, discard them */ -async function setStartOffset(page, offset) { +async function setStartOffset(page, { submitChanges = true, ...offset }) { // Click 'mode' button await page.getByRole('button', { name: 'Time Conductor Mode', exact: true }).click(); - await setTimeConductorOffset(page, offset); + await setTimeConductorOffset(page, { submitChanges, ...offset }); } /** * Set the values (hours, mins, secs) for the end time offset when in realtime mode * @param {import('@playwright/test').Page} page * @param {OffsetValues} offset + * @param {boolean} [submit=true] If true, submit the offset changes; otherwise, discard them */ -async function setEndOffset(page, offset) { +async function setEndOffset(page, { submitChanges = true, ...offset }) { // Click 'mode' button await page.getByRole('button', { name: 'Time Conductor Mode', exact: true }).click(); - await setTimeConductorOffset(page, offset); + await setTimeConductorOffset(page, { submitChanges, ...offset }); } /** @@ -499,17 +505,40 @@ async function setEndOffset(page, offset) { * NOTE: Unless explicitly testing the Time Conductor itself, it is advised to instead * navigate directly to the object with the desired time bounds using `navigateToObjectWithFixedTimeBounds()`. * @param {import('@playwright/test').Page} page - * @param {string} startDate - * @param {string} endDate + * @param {Object} bounds - The time conductor bounds + * @param {string} [bounds.startDate] - The start date in YYYY-MM-DD format + * @param {string} [bounds.startTime] - The start time in HH:mm:ss format + * @param {string} [bounds.endDate] - The end date in YYYY-MM-DD format + * @param {string} [bounds.endTime] - The end time in HH:mm:ss format + * @param {boolean} [bounds.submitChanges=true] - If true, submit the changes; otherwise, discard them. */ -async function setTimeConductorBounds(page, startDate, endDate) { - // Bring up the time conductor popup - expect(await page.locator('.l-shell__time-conductor.c-compact-tc').count()).toBe(1); - await page.click('.l-shell__time-conductor.c-compact-tc'); +async function setTimeConductorBounds(page, { submitChanges = true, ...bounds }) { + const { startDate, endDate, startTime, endTime } = bounds; - await setTimeBounds(page, startDate, endDate); + // Open the time conductor popup + await page.getByRole('button', { name: 'Time Conductor Mode', exact: true }).click(); - await page.keyboard.press('Enter'); + if (startDate) { + await page.getByLabel('Start date').fill(startDate); + } + + if (startTime) { + await page.getByLabel('Start time').fill(startTime); + } + + if (endDate) { + await page.getByLabel('End date').fill(endDate); + } + + if (endTime) { + await page.getByLabel('End time').fill(endTime); + } + + if (submitChanges) { + await page.getByLabel('Submit time bounds').click(); + } else { + await page.getByLabel('Discard changes and close time popup').click(); + } } /** diff --git a/e2e/tests/framework/appActions.e2e.spec.js b/e2e/tests/framework/appActions.e2e.spec.js index cc93247951..d58d73968c 100644 --- a/e2e/tests/framework/appActions.e2e.spec.js +++ b/e2e/tests/framework/appActions.e2e.spec.js @@ -24,14 +24,18 @@ import { createDomainObjectWithDefaults, createNotification, expandEntireTree, - openObjectTreeContextMenu + openObjectTreeContextMenu, + setFixedTimeMode, + setRealTimeMode, + setTimeConductorBounds } from '../../appActions.js'; import { expect, test } from '../../pluginFixtures.js'; test.describe('AppActions', () => { - test('createDomainObjectsWithDefaults', async ({ page }) => { + test.beforeEach(async ({ page }) => { await page.goto('./', { waitUntil: 'domcontentloaded' }); - + }); + test('createDomainObjectsWithDefaults', async ({ page }) => { const e2eFolder = await createDomainObjectWithDefaults(page, { type: 'Folder', name: 'e2e folder' @@ -91,7 +95,6 @@ test.describe('AppActions', () => { }); }); test('createNotification', async ({ page }) => { - await page.goto('./', { waitUntil: 'domcontentloaded' }); await createNotification(page, { message: 'Test info notification', severity: 'info' @@ -115,8 +118,6 @@ test.describe('AppActions', () => { await page.locator('[aria-label="Dismiss"]').click(); }); test('expandEntireTree', async ({ page }) => { - await page.goto('./', { waitUntil: 'domcontentloaded' }); - const rootFolder = await createDomainObjectWithDefaults(page, { type: 'Folder' }); @@ -168,12 +169,30 @@ test.describe('AppActions', () => { expect(await locatorTreeCollapsedItems.count()).toBe(0); }); test('openObjectTreeContextMenu', async ({ page }) => { - await page.goto('./', { waitUntil: 'domcontentloaded' }); - const folder = await createDomainObjectWithDefaults(page, { type: 'Folder' }); await openObjectTreeContextMenu(page, folder.url); await expect(page.getByLabel(`${folder.name} Context Menu`)).toBeVisible(); }); + test('setTimeConductorMode', async ({ page }) => { + await setFixedTimeMode(page); + await expect(page.getByLabel('Start bounds:')).toBeVisible(); + await expect(page.getByLabel('End bounds:')).toBeVisible(); + await setRealTimeMode(page); + await expect(page.getByLabel('Start offset')).toBeVisible(); + await expect(page.getByLabel('End offset')).toBeVisible(); + }); + test('setTimeConductorBounds', async ({ page }) => { + // Assume in real-time mode by default + await setFixedTimeMode(page); + await setTimeConductorBounds(page, { + startDate: '2024-01-01', + endDate: '2024-01-02', + startTime: '00:00:00', + endTime: '23:59:59' + }); + await expect(page.getByLabel('Start bounds: 2024-01-01 00:00:00')).toBeVisible(); + await expect(page.getByLabel('End bounds: 2024-01-02 23:59:59')).toBeVisible(); + }); }); diff --git a/e2e/tests/framework/generateLocalStorageData.e2e.spec.js b/e2e/tests/framework/generateLocalStorageData.e2e.spec.js index de8741cc88..ee9ff5d73b 100644 --- a/e2e/tests/framework/generateLocalStorageData.e2e.spec.js +++ b/e2e/tests/framework/generateLocalStorageData.e2e.spec.js @@ -117,18 +117,25 @@ test.describe('Generate Visual Test Data @localStorage @generatedata @clock', () end: '2024-11-12 20:11:11.000Z' }); - const NEW_GLOBAL_START_BOUNDS = '2024-11-11 19:11:11.000Z'; - const NEW_GLOBAL_END_BOUNDS = '2024-11-11 20:11:11.000Z'; + const NEW_GLOBAL_START_DATE = '2024-11-11'; + const NEW_GLOBAL_START_TIME = '19:11:11'; + const NEW_GLOBAL_END_DATE = '2024-11-11'; + const NEW_GLOBAL_END_TIME = '20:11:11'; - await setTimeConductorBounds(page, NEW_GLOBAL_START_BOUNDS, NEW_GLOBAL_END_BOUNDS); + await setTimeConductorBounds(page, { + startDate: NEW_GLOBAL_START_DATE, + startTime: NEW_GLOBAL_START_TIME, + endDate: NEW_GLOBAL_END_DATE, + endTime: NEW_GLOBAL_END_TIME + }); // Verify that the global time conductor bounds have been updated - expect( - await page.getByLabel('Global Time Conductor').getByLabel('Start bounds').textContent() - ).toEqual(NEW_GLOBAL_START_BOUNDS); - expect( - await page.getByLabel('Global Time Conductor').getByLabel('End bounds').textContent() - ).toEqual(NEW_GLOBAL_END_BOUNDS); + await expect( + page.getByLabel(`Start bounds: ${NEW_GLOBAL_START_DATE} ${NEW_GLOBAL_START_TIME}.000Z`) + ).toBeVisible(); + await expect( + page.getByLabel(`End bounds: ${NEW_GLOBAL_END_DATE} ${NEW_GLOBAL_END_TIME}.000Z`) + ).toBeVisible(); //Save localStorage for future test execution await context.storageState({ diff --git a/e2e/tests/functional/plugins/imagery/exampleImagery.e2e.spec.js b/e2e/tests/functional/plugins/imagery/exampleImagery.e2e.spec.js index c3e920960c..224bc519ba 100644 --- a/e2e/tests/functional/plugins/imagery/exampleImagery.e2e.spec.js +++ b/e2e/tests/functional/plugins/imagery/exampleImagery.e2e.spec.js @@ -53,6 +53,9 @@ test.describe('Example Imagery Object', () => { // Verify that the created object is focused await expect(page.locator('.l-browse-bar__object-name')).toContainText(exampleImagery.name); await page.getByLabel('Focused Image Element').hover({ trial: true }); + + // Wait for image thumbnail auto-scroll to complete + await expect(page.getByLabel('Image Thumbnail from').last()).toBeInViewport(); }); test('Can use Mouse Wheel to zoom in and out of latest image', async ({ page }) => { diff --git a/e2e/tests/functional/plugins/plot/logPlot.e2e.spec.js b/e2e/tests/functional/plugins/plot/logPlot.e2e.spec.js index 20e4443034..e327b3d093 100644 --- a/e2e/tests/functional/plugins/plot/logPlot.e2e.spec.js +++ b/e2e/tests/functional/plugins/plot/logPlot.e2e.spec.js @@ -25,7 +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. */ -import { setTimeConductorBounds } from '../../../../appActions.js'; +import { createDomainObjectWithDefaults, setTimeConductorBounds } from '../../../../appActions.js'; import { expect, test } from '../../../../pluginFixtures.js'; test.describe('Log plot tests', () => { @@ -86,51 +86,36 @@ async function makeOverlayPlot(page, myItemsFolderName) { // Set a specific time range for consistency, otherwise it will change // on every test to a range based on the current time. - const start = '2022-03-29 22:00:00.000Z'; - const end = '2022-03-29 22:00:30.000Z'; + const startDate = '2022-03-29'; + const startTime = '22:00:00'; + const endDate = '2022-03-29'; + const endTime = '22:00:30'; - await setTimeConductorBounds(page, start, end); + await setTimeConductorBounds(page, { startDate, startTime, endDate, endTime }); - // create overlay plot - - await page.locator('button.c-create-button').click(); - await page.locator('li[role="menuitem"]:has-text("Overlay Plot")').click(); - // Click OK button and wait for Navigate event - await Promise.all([ - page.waitForLoadState(), - await page.getByRole('button', { name: 'Save' }).click(), - // Wait for Save Banner to appear - page.waitForSelector('.c-message-banner__message') - ]); - - // save the overlay plot - await saveOverlayPlot(page); + const overlayPlot = await createDomainObjectWithDefaults(page, { + type: 'Overlay Plot', + name: 'Unnamed Overlay Plot' + }); // create a sinewave generator + await createDomainObjectWithDefaults(page, { + type: 'Sine Wave Generator', + name: 'Unnamed Sine Wave Generator', + parent: overlayPlot.uuid + }); - await page.locator('button.c-create-button').click(); - await page.locator('li[role="menuitem"]:has-text("Sine Wave Generator")').click(); + await page.getByLabel('More actions').click(); + await page.getByLabel('Edit Properties...').click(); // set amplitude to 6, offset 4, data rate 2 hz + await page.getByLabel('Amplitude', { exact: true }).fill('6'); + await page.getByLabel('Offset', { exact: true }).fill('4'); + await page.getByLabel('Data Rate (hz)', { exact: true }).fill('2'); - await page.getByLabel('Amplitude').fill('6'); - await page.getByLabel('Offset').fill('4'); - await page.getByLabel('Data Rate (hz)').fill('2'); + await page.getByLabel('Save').click(); - // Click OK button and wait for Navigate event - await Promise.all([ - page.waitForLoadState(), - await page.getByRole('button', { name: 'Save' }).click(), - // Wait for Save Banner to appear - page.waitForSelector('.c-message-banner__message') - ]); - - // click on overlay plot - await page.locator(`text=Open MCT ${myItemsFolderName} >> span`).nth(3).click(); - await Promise.all([ - page.waitForLoadState(), - page.locator('text=Unnamed Overlay Plot').first().click() - ]); + await page.goto(overlayPlot.url); } /** diff --git a/e2e/tests/functional/plugins/telemetryTable/telemetryTable.e2e.spec.js b/e2e/tests/functional/plugins/telemetryTable/telemetryTable.e2e.spec.js index 688bf50c84..3c94b06952 100644 --- a/e2e/tests/functional/plugins/telemetryTable/telemetryTable.e2e.spec.js +++ b/e2e/tests/functional/plugins/telemetryTable/telemetryTable.e2e.spec.js @@ -112,13 +112,14 @@ test.describe('Telemetry Table', () => { // Subtract 5 minutes from the current end bound datetime and set it // Bring up the time conductor popup - let endDate = await page.locator('[aria-label="End bounds"]').textContent(); - endDate = new Date(endDate); + let endTimeStamp = await page.getByLabel('End bounds').textContent(); + endTimeStamp = new Date(endTimeStamp); - endDate.setUTCMinutes(endDate.getUTCMinutes() - 5); - endDate = endDate.toISOString().replace(/T/, ' '); + endTimeStamp.setUTCMinutes(endTimeStamp.getUTCMinutes() - 5); + const endDate = endTimeStamp.toISOString().split('T')[0]; + const endTime = endTimeStamp.toISOString().split('T')[1]; - await setTimeConductorBounds(page, undefined, endDate); + await setTimeConductorBounds(page, { endDate, endTime }); await expect(tableWrapper).not.toHaveClass(/is-paused/); @@ -131,7 +132,7 @@ test.describe('Telemetry Table', () => { // Verify that it is <= our new end bound const latestMilliseconds = Date.parse(latestTelemetryDate); - const endBoundMilliseconds = Date.parse(endDate); + const endBoundMilliseconds = Date.parse(endTimeStamp); expect(latestMilliseconds).toBeLessThanOrEqual(endBoundMilliseconds); }); diff --git a/e2e/tests/functional/plugins/timeConductor/timeConductor.e2e.spec.js b/e2e/tests/functional/plugins/timeConductor/timeConductor.e2e.spec.js index 8d3822852f..a873b7c306 100644 --- a/e2e/tests/functional/plugins/timeConductor/timeConductor.e2e.spec.js +++ b/e2e/tests/functional/plugins/timeConductor/timeConductor.e2e.spec.js @@ -30,68 +30,80 @@ import { import { expect, test } from '../../../../pluginFixtures.js'; test.describe('Time conductor operations', () => { - test('validate start time does not exceeds end time', async ({ page }) => { + test('validate start time does not exceed end time', async ({ page }) => { // Go to baseURL await page.goto('./', { waitUntil: 'domcontentloaded' }); const year = new Date().getFullYear(); - let startDate = 'xxxx-01-01 01:00:00.000Z'; - startDate = year + startDate.substring(4); + // Set initial valid time bounds + const startDate = `${year}-01-01`; + const startTime = '01:00:00'; + const endDate = `${year}-01-01`; + const endTime = '02:00:00'; + await setTimeConductorBounds(page, { startDate, startTime, endDate, endTime }); - let endDate = 'xxxx-01-01 02:00:00.000Z'; - endDate = year + endDate.substring(4); + // Open the time conductor popup + await page.getByRole('button', { name: 'Time Conductor Mode', exact: true }).click(); - await setTimeConductorBounds(page, startDate, endDate); + // Test invalid start date + const invalidStartDate = `${year}-01-02`; + await page.getByLabel('Start date').fill(invalidStartDate); + await expect(page.getByLabel('Submit time bounds')).toBeDisabled(); + await page.getByLabel('Start date').fill(startDate); + await expect(page.getByLabel('Submit time bounds')).toBeEnabled(); - // invalid start date - startDate = year + 1 + startDate.substring(4); - await setTimeConductorBounds(page, startDate); + // Test invalid end date + const invalidEndDate = `${year - 1}-12-31`; + await page.getByLabel('End date').fill(invalidEndDate); + await expect(page.getByLabel('Submit time bounds')).toBeDisabled(); + await page.getByLabel('End date').fill(endDate); + await expect(page.getByLabel('Submit time bounds')).toBeEnabled(); - // Bring up the time conductor popup - const timeConductorMode = page.locator('.c-compact-tc'); - await timeConductorMode.click(); - const startDateLocator = page.locator('input[type="text"]').first(); - const endDateLocator = page.locator('input[type="text"]').nth(2); + // Test invalid start time + const invalidStartTime = '42:00:00'; + await page.getByLabel('Start time').fill(invalidStartTime); + await expect(page.getByLabel('Submit time bounds')).toBeDisabled(); + await page.getByLabel('Start time').fill(startTime); + await expect(page.getByLabel('Submit time bounds')).toBeEnabled(); - await endDateLocator.click(); + // Test invalid end time + const invalidEndTime = '43:00:00'; + await page.getByLabel('End time').fill(invalidEndTime); + await expect(page.getByLabel('Submit time bounds')).toBeDisabled(); + await page.getByLabel('End time').fill(endTime); + await expect(page.getByLabel('Submit time bounds')).toBeEnabled(); - const startDateValidityStatus = await startDateLocator.evaluate((element) => - element.checkValidity() + // Submit valid time bounds + await page.getByLabel('Submit time bounds').click(); + + // Verify the submitted time bounds + await expect(page.getByLabel('Start bounds')).toHaveText( + new RegExp(`${startDate} ${startTime}.000Z`) ); - expect(startDateValidityStatus).not.toBeTruthy(); - - // fix to valid start date - startDate = year - 1 + startDate.substring(4); - await setTimeConductorBounds(page, startDate); - - // invalid end date - endDate = year - 2 + endDate.substring(4); - await setTimeConductorBounds(page, undefined, endDate); - - await startDateLocator.click(); - - const endDateValidityStatus = await endDateLocator.evaluate((element) => - element.checkValidity() + await expect(page.getByLabel('End bounds')).toHaveText( + new RegExp(`${endDate} ${endTime}.000Z`) ); - expect(endDateValidityStatus).not.toBeTruthy(); }); }); -// Testing instructions: -// Try to change the realtime offsets when in realtime (local clock) mode. -test.describe('Time conductor input fields real-time mode', () => { - test('validate input fields in real-time mode', async ({ page }) => { +test.describe('Global Time Conductor', () => { + test.beforeEach(async ({ page }) => { + await page.goto('./', { waitUntil: 'domcontentloaded' }); + }); + + test('Input field validation: real-time mode', async ({ page }) => { const startOffset = { + startHours: '01', + startMins: '29', startSecs: '23' }; const endOffset = { + endHours: '01', + endMins: '30', endSecs: '31' }; - // Go to baseURL - await page.goto('./', { waitUntil: 'domcontentloaded' }); - // Switch to real-time mode await setRealTimeMode(page); @@ -99,13 +111,95 @@ test.describe('Time conductor input fields real-time mode', () => { await setStartOffset(page, startOffset); // Verify time was updated on time offset button - await expect(page.locator('.c-compact-tc__setting-value.icon-minus')).toContainText('00:30:23'); + await expect(page.getByLabel('Start offset: 01:29:23')).toBeVisible(); // Set end time offset await setEndOffset(page, endOffset); // Verify time was updated on preceding time offset button - await expect(page.locator('.c-compact-tc__setting-value.icon-plus')).toContainText('00:00:31'); + await expect(page.getByLabel('End offset: 01:30:31')).toBeVisible(); + + // Discard changes and verify that offsets remain unchanged + await setStartOffset(page, { + startHours: '00', + startMins: '30', + startSecs: '00', + submitChanges: false + }); + + await expect(page.getByLabel('Start offset: 01:29:23')).toBeVisible(); + await expect(page.getByLabel('End offset: 01:30:31')).toBeVisible(); + }); + + test('Input field validation: fixed time mode', async ({ page }) => { + test.info().annotations.push({ + type: 'issue', + description: 'https://github.com/nasa/openmct/issues/7791' + }); + // Switch to fixed time mode + await setFixedTimeMode(page); + + // Define valid time bounds for testing + const validBounds = { + startDate: '2024-04-20', + startTime: '00:04:20', + endDate: '2024-04-20', + endTime: '16:04:20' + }; + // Set valid time conductor bounds ✌️ + await setTimeConductorBounds(page, validBounds); + + // Verify that the time bounds are set correctly + await expect(page.getByLabel(`Start bounds: 2024-04-20 00:04:20.000Z`)).toBeVisible(); + await expect(page.getByLabel(`End bounds: 2024-04-20 16:04:20.000Z`)).toBeVisible(); + + // Open the Time Conductor Mode popup + await page.getByLabel('Time Conductor Mode').click(); + + // Test invalid start date + const invalidStartDate = '2024-04-21'; + await page.getByLabel('Start date').fill(invalidStartDate); + await expect(page.getByLabel('Submit time bounds')).toBeDisabled(); + await page.getByLabel('Start date').fill(validBounds.startDate); + await expect(page.getByLabel('Submit time bounds')).toBeEnabled(); + + // Test invalid end date + const invalidEndDate = '2024-04-19'; + await page.getByLabel('End date').fill(invalidEndDate); + await expect(page.getByLabel('Submit time bounds')).toBeDisabled(); + await page.getByLabel('End date').fill(validBounds.endDate); + await expect(page.getByLabel('Submit time bounds')).toBeEnabled(); + + // Test invalid start time + const invalidStartTime = '16:04:21'; + await page.getByLabel('Start time').fill(invalidStartTime); + await expect(page.getByLabel('Submit time bounds')).toBeDisabled(); + await page.getByLabel('Start time').fill(validBounds.startTime); + await expect(page.getByLabel('Submit time bounds')).toBeEnabled(); + + // Test invalid end time + const invalidEndTime = '00:04:19'; + await page.getByLabel('End time').fill(invalidEndTime); + await expect(page.getByLabel('Submit time bounds')).toBeDisabled(); + await page.getByLabel('End time').fill(validBounds.endTime); + await expect(page.getByLabel('Submit time bounds')).toBeEnabled(); + + // Verify that the time bounds remain unchanged after invalid inputs + await expect(page.getByLabel(`Start bounds: 2024-04-20 00:04:20.000Z`)).toBeVisible(); + await expect(page.getByLabel(`End bounds: 2024-04-20 16:04:20.000Z`)).toBeVisible(); + + // Discard changes and verify that bounds remain unchanged + await setTimeConductorBounds(page, { + startDate: validBounds.startDate, + startTime: '04:20:00', + endDate: validBounds.endDate, + endTime: '04:20:20', + submitChanges: false + }); + + // Verify that the original time bounds are still displayed after discarding changes + await expect(page.getByLabel(`Start bounds: 2024-04-20 00:04:20.000Z`)).toBeVisible(); + await expect(page.getByLabel(`End bounds: 2024-04-20 16:04:20.000Z`)).toBeVisible(); }); /** @@ -147,14 +241,15 @@ test.describe('Time conductor input fields real-time mode', () => { await setRealTimeMode(page); // Verify updated start time offset persists after mode switch - await expect(page.locator('.c-compact-tc__setting-value.icon-minus')).toContainText('00:30:23'); + await expect(page.getByLabel('Start offset: 00:30:23')).toBeVisible(); // Verify updated end time offset persists after mode switch - await expect(page.locator('.c-compact-tc__setting-value.icon-plus')).toContainText('00:00:01'); + await expect(page.getByLabel('End offset: 00:00:01')).toBeVisible(); // Verify url parameters persist after mode switch - expect(page.url()).toContain(`startDelta=${startDelta}`); - expect(page.url()).toContain(`endDelta=${endDelta}`); + // eslint-disable-next-line no-useless-escape + const urlRegex = new RegExp(`.*tc\.startDelta=${startDelta}&tc\.endDelta=${endDelta}.*`); + await page.waitForURL(urlRegex); }); test.fixme( diff --git a/e2e/tests/visual-a11y/imagery.visual.spec.js b/e2e/tests/visual-a11y/imagery.visual.spec.js index c09e4a0c3e..b063dee0f7 100644 --- a/e2e/tests/visual-a11y/imagery.visual.spec.js +++ b/e2e/tests/visual-a11y/imagery.visual.spec.js @@ -64,6 +64,10 @@ test.describe('Visual - Example Imagery', () => { test('Example Imagery in Fixed Time', async ({ page, theme }) => { await page.goto(exampleImagery.url, { waitUntil: 'domcontentloaded' }); + // Wait for the thumbnails to finish their scroll animation + // (Wait until the rightmost thumbnail is in view) + await expect(page.getByLabel('Image Thumbnail from').last()).toBeInViewport(); + await expect(page.getByLabel('Image Wrapper')).toBeVisible(); await percySnapshot(page, `Example Imagery in Fixed Time (theme: ${theme})`); @@ -76,6 +80,9 @@ test.describe('Visual - Example Imagery', () => { test('Example Imagery in Real Time', async ({ page, theme }) => { await page.goto(exampleImagery.url, { waitUntil: 'domcontentloaded' }); + // Wait for the thumbnails to finish their scroll animation + // (Wait until the rightmost thumbnail is in view) + await expect(page.getByLabel('Image Thumbnail from').last()).toBeInViewport(); await setRealTimeMode(page, true); await expect(page.getByLabel('Image Wrapper')).toBeVisible(); diff --git a/src/plugins/timeConductor/ConductorInputsFixed.vue b/src/plugins/timeConductor/ConductorInputsFixed.vue index 8911b91817..ee5309f25f 100644 --- a/src/plugins/timeConductor/ConductorInputsFixed.vue +++ b/src/plugins/timeConductor/ConductorInputsFixed.vue @@ -32,7 +32,7 @@
{{ formattedBounds.start }}
@@ -40,7 +40,7 @@
{{ formattedBounds.end }}
diff --git a/src/plugins/timeConductor/TimePopupFixed.vue b/src/plugins/timeConductor/TimePopupFixed.vue index 74c5c2fb86..7c098e4462 100644 --- a/src/plugins/timeConductor/TimePopupFixed.vue +++ b/src/plugins/timeConductor/TimePopupFixed.vue @@ -88,7 +88,7 @@ @@ -119,25 +119,21 @@ export default { }, emits: ['update', 'dismiss'], data() { - let timeSystem = this.openmct.time.getTimeSystem(); - let durationFormatter = this.getFormatter( - timeSystem.durationFormat || DEFAULT_DURATION_FORMATTER - ); - let timeFormatter = this.getFormatter(timeSystem.timeFormat); - let bounds = this.bounds || this.openmct.time.getBounds(); + const timeSystem = this.openmct.time.getTimeSystem(); + const bounds = this.openmct.time.getBounds(); return { - timeFormatter, - durationFormatter, + timeFormatter: this.getFormatter(timeSystem.timeFormat), + durationFormatter: this.getFormatter(timeSystem.durationFormat || DEFAULT_DURATION_FORMATTER), bounds: { start: bounds.start, end: bounds.end }, formattedBounds: { - start: timeFormatter.format(bounds.start).split(' ')[0], - end: timeFormatter.format(bounds.end).split(' ')[0], - startTime: durationFormatter.format(Math.abs(bounds.start)), - endTime: durationFormatter.format(Math.abs(bounds.end)) + start: '', + end: '', + startTime: '', + endTime: '' }, isUTCBased: timeSystem.isUTCBased, isDisabled: false @@ -157,9 +153,12 @@ export default { deep: true } }, - mounted() { + created() { this.handleNewBounds = _.throttle(this.handleNewBounds, 300); + }, + mounted() { this.setTimeSystem(JSON.parse(JSON.stringify(this.openmct.time.getTimeSystem()))); + this.setViewFromBounds(this.bounds); }, beforeUnmount() { this.clearAllValidation(); @@ -173,8 +172,10 @@ export default { [this.$refs.startDate, this.$refs.endDate].forEach(this.clearValidationForInput); }, clearValidationForInput(input) { - input.setCustomValidity(''); - input.title = ''; + if (input) { + input.setCustomValidity(''); + input.title = ''; + } }, setBounds(bounds) { this.bounds = bounds; @@ -186,7 +187,6 @@ export default { this.formattedBounds.endTime = this.durationFormatter.format(Math.abs(bounds.end)); }, setTimeSystem(timeSystem) { - this.timeSystem = timeSystem; this.timeFormatter = this.getFormatter(timeSystem.timeFormat); this.durationFormatter = this.getFormatter( timeSystem.durationFormat || DEFAULT_DURATION_FORMATTER @@ -207,39 +207,31 @@ export default { `${this.formattedBounds.end} ${this.formattedBounds.endTime}` ); - this.$emit('update', { - start: start, - end: end - }); + this.$emit('update', { start, end }); } if (dismiss) { this.$emit('dismiss'); - return false; } }, handleFormSubmission(shouldDismiss) { - // Validate bounds before submission this.validateAllBounds('startDate'); this.validateAllBounds('endDate'); - // Submit the form if it's valid if (!this.isDisabled) { this.setBoundsFromView(shouldDismiss); } }, validateAllBounds(ref) { - this.isDisabled = false; // Reset isDisabled at the start of validation + this.isDisabled = false; if (!this.areBoundsFormatsValid()) { this.isDisabled = true; return false; } - let validationResult = { - valid: true - }; + let validationResult = { valid: true }; const currentInput = this.$refs[ref]; return [this.$refs.startDate, this.$refs.endDate].every((input) => { @@ -255,38 +247,30 @@ export default { // const limit = this.getBoundsLimit(); const limit = false; - if (this.timeSystem.isUTCBased && limit && boundsValues.end - boundsValues.start > limit) { + if (this.isUTCBased && limit && boundsValues.end - boundsValues.start > limit) { if (input === currentInput) { validationResult = { valid: false, message: 'Start and end difference exceeds allowable limit' }; } - } else { - if (input === currentInput) { - validationResult = this.openmct.time.validateBounds(boundsValues); - } + } else if (input === currentInput) { + validationResult = this.openmct.time.validateBounds(boundsValues); } return this.handleValidationResults(input, validationResult); }); }, areBoundsFormatsValid() { - let validationResult = { - valid: true - }; - return [this.$refs.startDate, this.$refs.endDate].every((input) => { const formattedDate = input === this.$refs.startDate ? `${this.formattedBounds.start} ${this.formattedBounds.startTime}` : `${this.formattedBounds.end} ${this.formattedBounds.endTime}`; - if (!this.timeFormatter.validate(formattedDate)) { - validationResult = { - valid: false, - message: 'Invalid date' - }; - } + + const validationResult = this.timeFormatter.validate(formattedDate) + ? { valid: true } + : { valid: false, message: 'Invalid date' }; return this.handleValidationResults(input, validationResult); }); From 4ee68cccd66105d068337f2ba50772651dcdb1c2 Mon Sep 17 00:00:00 2001 From: Jesse Mazzella Date: Wed, 31 Jul 2024 10:46:16 -0700 Subject: [PATCH 2/2] docs: better docs and types for the API (#7796) * docs: fix type imports in openmct.js * docs: fix type imports * docs: fix types for eventHelpers * docs: types for TypeRegistry * docs: types for StatusAPI * docs: fix ObjectAPI types and docs * docs: more types * docs: improved types for main entry * docs: improved types * fix: unbreak the linting * chore: remove EventEmitter webpack alias as it hide types * fix: return type * fix: parameter type * fix: types for composables * chore: add webpack consts to eslintrc * fix: remove usage of deprecated timeAPI methods and add a ton of docs and types * docs: update README.md * lint: clean up API.md * chore: upgrade eventemitter to v5.0.2 * refactor: update imports for EventEmitter to remove alias * format: lint * docs: update types for Views and ViewProviders * docs: expose common types at the base import level * docs(types): remove unnecessary tsconfig options * docs: ActionAPI * docs: AnnotationAPI * docs: import common types from the same origin * docs: FormsAPI & TelemetryAPI types * docs: FormController, IndicatorAPI * docs: MenuAPI, ActionsAPI * docs: `@memberof` is not supported by `tsc` and JSDoc generation so remove it * docs: RootRegistry and RootObjectProvider * docs: Transaction + Overlay * lint: words for the word god * fix: review comments --- .cspell.json | 17 +- .eslintrc.cjs | 7 +- .webpack/webpack.common.mjs | 1 - API.md | 61 +++--- README.md | 112 +++++----- docs/src/index.md | 10 +- e2e/appActions.js | 2 +- example/exampleUser/ExampleUserProvider.js | 2 +- .../generator/SinewaveStalenessProvider.js | 2 +- openmct.js | 70 +++--- src/MCT.js | 148 ++++++++----- src/api/Editor.js | 2 +- src/api/actions/ActionCollection.js | 73 ++++++- src/api/actions/ActionsAPI.js | 89 +++++++- src/api/annotation/AnnotationAPI.js | 201 ++++++++++++------ src/api/composition/CompositionAPI.js | 4 +- src/api/composition/CompositionCollection.js | 7 +- src/api/composition/CompositionProvider.js | 19 +- .../composition/DefaultCompositionProvider.js | 4 +- src/api/faultmanagement/FaultManagementAPI.js | 4 +- src/api/forms/FormController.js | 50 +++++ src/api/forms/FormsAPI.js | 32 ++- src/api/forms/toggle-check-box-mixin.js | 22 ++ src/api/indicators/IndicatorAPI.js | 52 +++-- src/api/indicators/SimpleIndicator.js | 2 +- src/api/menu/MenuAPI.js | 78 ++++--- src/api/menu/menu.js | 32 ++- src/api/notifications/NotificationAPI.js | 185 ++++++++-------- src/api/objects/ConflictError.js | 27 +++ src/api/objects/InterceptorRegistry.js | 4 - src/api/objects/MutableDomainObject.js | 3 +- src/api/objects/ObjectAPI.js | 200 +++++++---------- src/api/objects/RootObjectProvider.js | 36 ++++ src/api/objects/RootRegistry.js | 35 +++ src/api/objects/Transaction.js | 43 ++++ src/api/overlays/Overlay.js | 2 +- src/api/overlays/OverlayAPI.js | 88 ++++---- src/api/status/StatusAPI.js | 45 +++- src/api/telemetry/BatchingWebSocket.js | 1 - src/api/telemetry/TelemetryAPI.js | 56 ++--- src/api/telemetry/TelemetryCollection.js | 4 +- .../telemetry/TelemetryRequestInterceptor.js | 4 - src/api/time/IndependentTimeContext.js | 4 - src/api/time/TimeAPI.js | 1 - src/api/time/TimeContext.js | 10 +- src/api/tooltips/ToolTip.js | 2 +- src/api/tooltips/ToolTipAPI.js | 1 - src/api/types/Type.js | 31 ++- src/api/types/TypeRegistry.js | 19 +- src/api/user/StatusAPI.js | 2 +- src/api/user/UserAPI.js | 35 +-- .../DeviceClassifier/src/DeviceMatchers.js | 1 - src/plugins/LADTable/LADTableConfiguration.js | 2 +- .../activityStatesInterceptor.js | 2 +- src/plugins/charts/bar/pluginSpec.js | 2 +- src/plugins/charts/scatter/pluginSpec.js | 2 +- src/plugins/clock/pluginSpec.js | 2 +- src/plugins/condition/Condition.js | 2 +- src/plugins/condition/ConditionManager.js | 2 +- src/plugins/condition/StyleRuleManager.js | 2 +- .../condition/criterion/TelemetryCriterion.js | 2 +- src/plugins/displayLayout/LayoutDrag.js | 1 - .../exportAsJSONAction/ExportAsJSONAction.js | 2 +- src/plugins/flexibleLayout/pluginSpec.js | 2 +- src/plugins/imagery/lib/eventHelpers.js | 8 + .../inspectorViews/styles/StylesManager.js | 2 +- .../localTimeSystem/LocalTimeFormat.js | 1 - src/plugins/localTimeSystem/pluginSpec.js | 2 +- src/plugins/notebook/snapshot-container.js | 2 +- .../persistence/couch/CouchDocument.js | 1 - src/plugins/plan/PlanViewConfiguration.js | 2 +- .../components/PlanViewConfiguration.vue | 2 +- src/plugins/plot/configuration/Model.js | 2 +- .../configuration/PlotConfigurationModel.js | 2 +- src/plugins/plot/draw/Draw2D.js | 3 +- src/plugins/plot/draw/DrawWebGL.js | 3 +- src/plugins/plot/lib/eventHelpers.js | 6 +- src/plugins/plot/overlayPlot/pluginSpec.js | 2 +- src/plugins/plot/pluginSpec.js | 2 +- src/plugins/plot/stackedPlot/pluginSpec.js | 2 +- src/plugins/plugins.js | 3 + src/plugins/remoteClock/RemoteClock.js | 2 - src/plugins/summaryWidget/src/Condition.js | 2 +- .../summaryWidget/src/ConditionManager.js | 2 +- src/plugins/summaryWidget/src/Rule.js | 2 +- src/plugins/summaryWidget/src/TestDataItem.js | 2 +- src/plugins/summaryWidget/src/WidgetDnD.js | 2 +- .../summaryWidget/src/input/Palette.js | 2 +- src/plugins/summaryWidget/src/input/Select.js | 2 +- src/plugins/tabs/pluginSpec.js | 2 +- .../src/MeanTelemetryProviderSpec.js | 2 +- src/plugins/telemetryTable/TelemetryTable.js | 2 +- .../TelemetryTableConfiguration.js | 2 +- .../collections/TableRowCollection.js | 2 +- src/plugins/timeline/pluginSpec.js | 2 +- src/plugins/timelist/pluginSpec.js | 2 +- src/plugins/timer/plugin.js | 1 + src/plugins/utcTimeSystem/DurationFormat.js | 1 - src/plugins/utcTimeSystem/UTCTimeFormat.js | 1 - src/selection/Selection.js | 7 +- src/ui/composables/edit.js | 2 +- src/ui/composables/resize.js | 2 +- src/ui/layout/RecentObjectsList.vue | 2 +- src/ui/preview/PreviewAction.js | 2 +- src/ui/registries/InspectorViewRegistry.js | 86 ++------ src/ui/registries/ToolbarRegistry.js | 5 - src/ui/registries/ViewRegistry.js | 200 ++++++----------- src/ui/router/ApplicationRouter.js | 35 +-- src/ui/router/Browse.js | 27 ++- src/utils/agent/Agent.js | 5 +- src/utils/clock/DefaultClock.js | 45 ++-- src/utils/duration.js | 16 +- src/utils/mount.js | 14 ++ src/utils/staleness.js | 28 +-- src/utils/visibility/VisibilityObserver.js | 51 +++-- src/utils/vueWrapHtmlElement.js | 1 + tsconfig.json | 12 +- 117 files changed, 1494 insertions(+), 1089 deletions(-) diff --git a/.cspell.json b/.cspell.json index b085166829..cc3394b05e 100644 --- a/.cspell.json +++ b/.cspell.json @@ -481,9 +481,20 @@ "specced", "composables", "countup", - "darkmatter" + "darkmatter", + "Undeletes" + ], + "dictionaries": [ + "npm", + "softwareTerms", + "node", + "html", + "css", + "bash", + "en_US", + "en-gb", + "misc" ], - "dictionaries": ["npm", "softwareTerms", "node", "html", "css", "bash", "en_US", "en-gb", "misc"], "ignorePaths": [ "package.json", "dist/**", @@ -494,4 +505,4 @@ "html-test-results", "test-results" ] -} +} \ No newline at end of file diff --git a/.eslintrc.cjs b/.eslintrc.cjs index 6715df7121..8511e79a31 100644 --- a/.eslintrc.cjs +++ b/.eslintrc.cjs @@ -10,7 +10,12 @@ const config = { serviceworker: true }, globals: { - _: 'readonly' + _: 'readonly', + __webpack_public_path__: 'writeable', + __OPENMCT_VERSION__: 'readonly', + __OPENMCT_BUILD_DATE__: 'readonly', + __OPENMCT_REVISION__: 'readonly', + __OPENMCT_BUILD_BRANCH__: 'readonly' }, plugins: ['prettier', 'unicorn', 'simple-import-sort'], extends: [ diff --git a/.webpack/webpack.common.mjs b/.webpack/webpack.common.mjs index fab1106dfa..7290ce999e 100644 --- a/.webpack/webpack.common.mjs +++ b/.webpack/webpack.common.mjs @@ -70,7 +70,6 @@ const config = { '@': path.join(projectRootDir, 'src'), legacyRegistry: path.join(projectRootDir, 'src/legacyRegistry'), csv: 'comma-separated-values', - EventEmitter: 'eventemitter3', bourbon: 'bourbon.scss', 'plotly-basic': 'plotly.js-basic-dist-min', 'plotly-gl2d': 'plotly.js-gl2d-dist-min', diff --git a/API.md b/API.md index f7f7b71a62..40d8b7a178 100644 --- a/API.md +++ b/API.md @@ -381,6 +381,7 @@ openmct.composition.addProvider({ The `addProvider` function accepts a Composition Provider object as its sole argument. A Composition Provider is a javascript object exposing two functions: + - `appliesTo`: A `function` that accepts a `domainObject` argument, and returns a `boolean` value indicating whether this composition provider applies to the given object. @@ -618,9 +619,10 @@ interface Formatter { Open MCT on its own defines a handful of built-in formats: -###### **Number Format (default):** +###### **Number Format (default):** Applied to data with `format: 'number'` + ```js valueMetadata = { format: 'number' @@ -635,15 +637,18 @@ interface NumberFormatter extends Formatter { validate: (value: any) => boolean; } ``` -###### **String Format**: + +###### **String Format** Applied to data with `format: 'string'` + ```js valueMetadata = { format: 'string' // ... }; ``` + ```ts interface StringFormatter extends Formatter { parse: (value: any) => string; @@ -652,8 +657,10 @@ interface StringFormatter extends Formatter { } ``` -###### **Enum Format**: +###### **Enum Format** + Applied to data with `format: 'enum'` + ```js valueMetadata = { format: 'enum', @@ -676,6 +683,7 @@ valueMetadata = { Creates a two-way mapping between enum string and value to be used in the `parse` and `format` methods. Ex: + - `formatter.parse('APPLE') === 1;` - `formatter.format(1) === 'APPLE';` @@ -691,7 +699,6 @@ interface EnumFormatter extends Formatter { Formats implement the following interface (provided here as TypeScript for simplicity): - Formats are registered with the Telemetry API using the `addFormat` function. eg. ```javascript @@ -763,8 +770,8 @@ state of the application, and emits events to inform listeners when the state ch Because the data displayed tends to be time domain data, Open MCT must always have at least one time system installed and activated. When you download Open -MCT, it will be pre-configured to use the UTC time system, which is installed and activated, -along with other default plugins, in `index.html`. Installing and activating a time system +MCT, it will be pre-configured to use the UTC time system, which is installed and activated, +along with other default plugins, in `index.html`. Installing and activating a time system is simple, and is covered [in the next section](#defining-and-registering-time-systems). ### Time Systems and Bounds @@ -817,7 +824,7 @@ numbers in UTC terrestrial time. #### Getting and Setting the Active Time System Once registered, a time system can be activated by calling `setTimeSystem` with -the timeSystem `key` or an instance of the time system. You can also specify +the timeSystem `key` or an instance of the time system. You can also specify valid [bounds](#time-bounds) for the timeSystem. ```javascript @@ -841,10 +848,9 @@ Setting the active time system will trigger a [`'timeSystemChanged'`](#time-even event. If you supplied bounds, a [`'boundsChanged'`](#time-events) event will be triggered afterwards with your newly supplied bounds. > ⚠️ **Deprecated** +> > - The method `timeSystem()` is deprecated. Please use `getTimeSystem()` and `setTimeSystem()` as a replacement. - - #### Time Bounds The TimeAPI provides a getter and setter for querying and setting time bounds. Time @@ -875,15 +881,16 @@ To respond to bounds change events, listen for the [`'boundsChanged'`](#time-eve event. > ⚠️ **Deprecated** +> > - The method `bounds()` is deprecated and will be removed in a future release. Please use `getBounds()` and `setBounds()` as a replacement. ### Clocks -The Time API requires a clock source which will cause the bounds to be updated -automatically whenever the clock source "ticks". A clock is simply an object that -supports registration of listeners and periodically invokes its listeners with a -number. Open MCT supports registration of new clock sources that tick on almost -anything. A tick occurs when the clock invokes callback functions registered by its +The Time API requires a clock source which will cause the bounds to be updated +automatically whenever the clock source "ticks". A clock is simply an object that +supports registration of listeners and periodically invokes its listeners with a +number. Open MCT supports registration of new clock sources that tick on almost +anything. A tick occurs when the clock invokes callback functions registered by its listeners with a new time value. An example of a clock source is the [LocalClock](https://github.com/nasa/openmct/blob/master/src/plugins/utcTimeSystem/LocalClock.js) @@ -972,6 +979,7 @@ openmct.time.getClock(); ``` > ⚠️ **Deprecated** +> > - The method `clock()` is deprecated and will be removed in a future release. Please use `getClock()` and `setClock()` as a replacement. #### ⚠️ [DEPRECATED] Stopping an active clock @@ -986,12 +994,13 @@ openmct.time.stopClock(); ``` > ⚠️ **Deprecated** +> > - The method `stopClock()` is deprecated and will be removed in a future release. #### Clock Offsets -When in Real-time [mode](#time-modes), the time bounds of the application will be updated automatically each time the -clock "ticks". The bounds are calculated based on the current value provided by +When in Real-time [mode](#time-modes), the time bounds of the application will be updated automatically each time the +clock "ticks". The bounds are calculated based on the current value provided by the active clock (via its `tick` event, or its `currentValue()` method). Unlike bounds, which represent absolute time values, clock offsets represent @@ -1026,13 +1035,14 @@ new bounds will be calculated based on the `currentValue()` of the active clock. Clock offsets are only relevant when in Real-time [mode](#time-modes). > ⚠️ **Deprecated** +> > - The method `clockOffsets()` is deprecated and will be removed in a future release. Please use `getClockOffsets()` and `setClockOffsets()` as a replacement. ### Time Modes -There are two time modes in Open MCT, "Fixed" and "Real-time". In Real-time mode the +There are two time modes in Open MCT, "Fixed" and "Real-time". In Real-time mode the time bounds of the application will be updated automatically each time the clock "ticks". -The bounds are calculated based on the current value provided by the active clock. In +The bounds are calculated based on the current value provided by the active clock. In Fixed mode, the time bounds are set for a specified time range. When Open MCT is first initialized, it will be in Real-time mode. @@ -1120,6 +1130,7 @@ The events emitted by the Time API are: - `mode`: A string representation of the current time mode, either `'realtime'` or `'fixed'`. > ⚠️ **Deprecated Events** (These will be removed in a future release): +> > - `bounds` → `boundsChanged` > - `timeSystem` → `timeSystemChanged` > - `clock` → `clockChanged` @@ -1262,7 +1273,7 @@ Returns the currently set text as a `string`. [the built-in glyphs](https://nasa.github.io/openmct/style-guide/#/browse/styleguide:home/glyphs?view=styleguide.glyphs) may be used here, or a custom CSS class can be provided. Returns the currently defined CSS class as a `string`. -- `.statusClass([className])`: Gets or sets the CSS class used to determine status. Accepts an __optional__ +- `.statusClass([className])`: Gets or sets the CSS class used to determine status. Accepts an **optional** `string` parameter to be used to set a status class applied to the indicator. May be used to apply different colors to indicate status. @@ -1312,7 +1323,7 @@ can be used to manage user information and roles. ### Example -Open MCT provides an example [user](example/exampleUser/exampleUserCreator.js) and [user provider](example/exampleUser/ExampleUserProvider.js) which +Open MCT provides an example [user](example/exampleUser/exampleUserCreator.js) and [user provider](example/exampleUser/ExampleUserProvider.js) which can be used as a starting point for creating a custom user provider. ## Visibility-Based Rendering in View Providers @@ -1335,10 +1346,10 @@ Here’s the signature for the show function: `show(element, isEditing, viewOptions)` - * `element` (HTMLElement) - The DOM element where the view should be rendered. - * `isEditing` (boolean) - Indicates whether the view is in editing mode. - * `viewOptions` (Object) - An object with configuration options for the view, including: - * `renderWhenVisible` (Function) - This function wraps the `requestAnimationFrame` and only triggers the provided render logic when the view is visible in the viewport. +- `element` (HTMLElement) - The DOM element where the view should be rendered. +- `isEditing` (boolean) - Indicates whether the view is in editing mode. +- `viewOptions` (Object) - An object with configuration options for the view, including: + - `renderWhenVisible` (Function) - This function wraps the `requestAnimationFrame` and only triggers the provided render logic when the view is visible in the viewport. ### Example @@ -1367,8 +1378,6 @@ const myViewProvider = { }; ``` - Note that `renderWhenVisible` defers rendering while the view is not visible and caters to the latest execution call. This provides responsiveness for dynamic content while ensuring performance optimizations. Ensure your view logic is prepared to handle potentially multiple deferrals if using this API, as only the last call to renderWhenVisible will be queued for execution upon the view becoming visible. - diff --git a/README.md b/README.md index 78335fbb99..054c73722f 100644 --- a/README.md +++ b/README.md @@ -5,12 +5,10 @@ Open MCT (Open Mission Control Technologies) is a next-generation mission contro > [!NOTE] > Please visit our [Official Site](https://nasa.github.io/openmct/) and [Getting Started Guide](https://nasa.github.io/openmct/getting-started/) - Once you've created something amazing with Open MCT, showcase your work in our GitHub Discussions [Show and Tell](https://github.com/nasa/openmct/discussions/categories/show-and-tell) section. We love seeing unique and wonderful implementations of Open MCT! ![Screen Shot 2022-11-23 at 9 51 36 AM](https://user-images.githubusercontent.com/4215777/203617422-4d912bfc-766f-4074-8324-409d9bbe7c05.png) - ## Building and Running Open MCT Locally Building and running Open MCT in your local dev environment is very easy. Be sure you have [Git](https://git-scm.com/downloads) and [Node.js](https://nodejs.org/) installed, then follow the directions below. Need additional information? Check out the [Getting Started](https://nasa.github.io/openmct/getting-started/) page on our website. @@ -18,19 +16,19 @@ Building and running Open MCT in your local dev environment is very easy. Be sur 1. Clone the source code: -``` +```sh git clone https://github.com/nasa/openmct.git ``` 2. (Optional) Install the correct node version using [nvm](https://github.com/nvm-sh/nvm): -``` +```sh nvm install ``` -3. Install development dependencies (Note: Check the `package.json` engine for our tested and supported node versions): +3. Install development dependencies (Note: Check the `package.json` engine for our tested and supported node versions): -``` +```sh npm install ``` @@ -57,9 +55,9 @@ our documentation. > [!NOTE] > We want Open MCT to be as easy to use, install, run, and develop for as -> possible, and your feedback will help us get there! -> Feedback can be provided via [GitHub issues](https://github.com/nasa/openmct/issues/new/choose), -> [Starting a GitHub Discussion](https://github.com/nasa/openmct/discussions), +> possible, and your feedback will help us get there! +> Feedback can be provided via [GitHub issues](https://github.com/nasa/openmct/issues/new/choose), +> [Starting a GitHub Discussion](https://github.com/nasa/openmct/discussions), > or by emailing us at [arc-dl-openmct@mail.nasa.gov](mailto:arc-dl-openmct@mail.nasa.gov). ## Developing Applications With Open MCT @@ -68,28 +66,29 @@ For more on developing with Open MCT, see our documentation for a guide on [Deve ## Compatibility -This is a fast moving project and we do our best to test and support the widest possible range of browsers, operating systems, and nodejs APIs. We have a published list of support available in our package.json's `browserslist` key. +This is a fast moving project and we do our best to test and support the widest possible range of browsers, operating systems, and NodeJS APIs. We have a published list of support available in our package.json's `browserslist` key. -The project uses `nvm` to ensure the node and npm version used, is coherent in all projects. Install nvm (non-windows), [here](https://github.com/nvm-sh/nvm) or the windows equivalent [here](https://github.com/coreybutler/nvm-windows) +The project utilizes `nvm` to maintain consistent node and npm versions across all projects. For UNIX, MacOS, Windows WSL, and other POSIX-compliant shell environments, click [here](https://github.com/nvm-sh/nvm). For Windows, check out [nvm-windows](https://github.com/coreybutler/nvm-windows). -If you encounter an issue with a particular browser, OS, or nodejs API, please file a [GitHub issue](https://github.com/nasa/openmct/issues/new/choose) +If you encounter an issue with a particular browser, OS, or NodeJS API, please [file an issue](https://github.com/nasa/openmct/issues/new/choose). ## Plugins -Open MCT can be extended via plugins that make calls to the Open MCT API. A plugin is a group +Open MCT can be extended via plugins that make calls to the Open MCT API. A plugin is a group of software components (including source code and resources such as images and HTML templates) that is intended to be added or removed as a single unit. -As well as providing an extension mechanism, most of the core Open MCT codebase is also +As well as providing an extension mechanism, most of the core Open MCT codebase is also written as plugins. For information on writing plugins, please see [our API documentation](./API.md#plugins). ## Tests -Our automated test coverage comes in the form of unit, e2e, visual, performance, and security tests. +Our automated test coverage comes in the form of unit, e2e, visual, performance, and security tests. ### Unit Tests + Unit Tests are written for [Jasmine](https://jasmine.github.io/api/edge/global) and run by [Karma](http://karma-runner.github.io). To run: @@ -101,24 +100,34 @@ in the `src` hierarchy. Full configuration details are found in alongside the units that they test; for example, `src/foo/Bar.js` would be tested by `src/foo/BarSpec.js`. -### e2e, Visual, and Performance tests -The e2e, Visual, and Performance tests are written for playwright and run by playwright's new test runner [@playwright/test](https://playwright.dev/). +### e2e, Visual, and Performance Testing -To run the e2e tests which are part of every commit: +Our e2e (end-to-end), Visual, and Performance tests leverage the Playwright framework and are executed using Playwright's test runner, [@playwright/test](https://playwright.dev/). -`npm run test:e2e:stable` +#### How to Run Tests -To run the visual test suite: +- **e2e Tests**: These tests are run on every commit. To run the tests locally, use: -`npm run test:e2e:visual` + ```sh + npm run test:e2e:stable + ``` -To run the performance tests: +- **Visual Tests**: For running the visual test suite, use: -`npm run test:perf` + ```sh + npm run test:e2e:visual + ``` -The test suite is configured to all tests located in `e2e/tests/` ending in `*.e2e.spec.js`. For more about the e2e test suite, please see the [README](./e2e/README.md) +- **Performance Tests**: To initiate the performance tests, enter: + + ```sh + npm run test:perf + ``` + +All tests are located within the `e2e/tests/` directory and are identified by the `*.e2e.spec.js` filename pattern. For more information about the e2e test suite, refer to the [README](./e2e/README.md). ### Security Tests + 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 available 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 @@ -129,60 +138,41 @@ Our code coverage is generated during the runtime of our unit, e2e, and visual t For more on the specifics of our code coverage setup, [see](TESTING.md#code-coverage) -# Glossary +## Glossary Certain terms are used throughout Open MCT with consistent meanings or conventions. Any deviations from the below are issues and should be addressed (either by updating this glossary or changing code to reflect correct usage.) Other developer documentation, particularly in-line documentation, may presume an understanding of these terms. - -* _plugin_: A plugin is a removable, reusable grouping of software elements. - The application is composed of plugins. -* _composition_: In the context of a domain object, this refers to the set of - other domain objects that compose or are contained by that object. A domain - object's composition is the set of domain objects that should appear - immediately beneath it in a tree hierarchy. A domain object's composition is - described in its model as an array of id's; its composition capability - provides a means to retrieve the actual domain object instances associated - with these identifiers asynchronously. -* _description_: When used as an object property, this refers to the human-readable - description of a thing; usually a single sentence or short paragraph. - (Most often used in the context of extensions, domain - object models, or other similar application-specific objects.) -* _domain object_: A meaningful object to the user; a distinct thing in - the work support by Open MCT. Anything that appears in the left-hand - tree is a domain object. -* _identifier_: A tuple consisting of a namespace and a key, which together uniquely - identifies a domain object. -* _model_: The persistent state associated with a domain object. A domain - object's model is a JavaScript object which can be converted to JSON - without losing information (that is, it contains no methods.) -* _name_: When used as an object property, this refers to the human-readable - name for a thing. (Most often used in the context of extensions, domain - object models, or other similar application-specific objects.) -* _navigation_: Refers to the current state of the application with respect - to the user's expressed interest in a specific domain object; e.g. when - a user clicks on a domain object in the tree, they are _navigating_ to - it, and it is thereafter considered the _navigated_ object (until the - user makes another such choice.) -* _namespace_: A name used to identify a persistence store. A running open MCT -application could potentially use multiple persistence stores, with the +| Term | Definition | +|---------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| _plugin_ | A removable, reusable grouping of software elements. The application is composed of plugins. | +| _composition_ | In the context of a domain object, this term refers to the set of other domain objects that compose or are contained by that object. A domain object's composition is the set of domain objects that should appear immediately beneath it in a tree hierarchy. It is described in its model as an array of ids, providing a means to asynchronously retrieve the actual domain object instances associated with these identifiers. | +| _description_ | When used as an object property, this term refers to the human-readable description of a thing, usually a single sentence or short paragraph. It is most often used in the context of extensions, domain object models, or other similar application-specific objects. | +| _domain object_ | A meaningful object to the user and a distinct thing in the work supported by Open MCT. Anything that appears in the left-hand tree is a domain object. | +| _identifier_ | A tuple consisting of a namespace and a key, which together uniquely identifies a domain object. | +| _model_ | The persistent state associated with a domain object. A domain object's model is a JavaScript object that can be converted to JSON without losing information, meaning it contains no methods. | +| _name_ | When used as an object property, this term refers to the human-readable name for a thing. It is most often used in the context of extensions, domain object models, or other similar application-specific objects. | +| _navigation_ | This term refers to the current state of the application with respect to the user's expressed interest in a specific domain object. For example, when a user clicks on a domain object in the tree, they are navigating to it, and it is thereafter considered the navigated object until the user makes another such choice. | +| _namespace_ | A name used to identify a persistence store. A running Open MCT application could potentially use multiple persistence stores. | ## Open MCT v2.0.0 + Support for our legacy bundle-based API, and the libraries that it was built on (like Angular 1.x), have now been removed entirely from this repository. For now if you have an Open MCT application that makes use of the legacy API, [a plugin](https://github.com/nasa/openmct-legacy-plugin) is provided that bootstraps the legacy bundling mechanism and API. This plugin will not be maintained over the long term however, and the legacy support plugin will not be tested for compatibility with future versions of Open MCT. It is provided for convenience only. ### How do I know if I am using legacy API? + You might still be using legacy API if your source code -* Contains files named bundle.js, or bundle.json, -* Makes calls to `openmct.$injector()`, or `openmct.$angular`, -* Makes calls to `openmct.legacyRegistry`, `openmct.legacyExtension`, or `openmct.legacyBundle`. - +- Contains files named bundle.js, or bundle.json, +- Makes calls to `openmct.$injector()`, or `openmct.$angular`, +- Makes calls to `openmct.legacyRegistry`, `openmct.legacyExtension`, or `openmct.legacyBundle`. ### What should I do if I am using legacy API? + Please refer to [the modern Open MCT API](https://nasa.github.io/openmct/documentation/). Post any questions to the [Discussions section](https://github.com/nasa/openmct/discussions) of the Open MCT GitHub repository. ## Related Repos diff --git a/docs/src/index.md b/docs/src/index.md index 52781a3739..0e21453734 100644 --- a/docs/src/index.md +++ b/docs/src/index.md @@ -14,13 +14,13 @@ extend the platform are provided in the following documentation. ## Sections - - * The [API](api/) uses inline documentation + +* The [API](api/) uses inline documentation. using [TypeScript](https://www.typescriptlang.org) and some legacy [JSDoc](https://jsdoc.app/). It describes the JavaScript objects and functions that make up the software platform. - * The [Development Process](process/) document describes the +* The [Development Process](process/) document describes the Open MCT software development cycle. - * The [Tutorials](https://github.com/nasa/openmct-tutorial) give examples of extending the platform to add - functionality, and integrate with data sources. +* The [tutorial](https://github.com/nasa/openmct-tutorial) and [plugin template](https://github.com/nasa/openmct-hello) give examples of extending the platform to add + functionality and integrate with data sources. diff --git a/e2e/appActions.js b/e2e/appActions.js index c14bceb020..35b238843e 100644 --- a/e2e/appActions.js +++ b/e2e/appActions.js @@ -35,7 +35,7 @@ * @property {string} type the type of domain object to create (e.g.: "Sine Wave Generator"). * @property {string} [name] the desired name of the created domain object. * @property {string | import('../src/api/objects/ObjectAPI').Identifier} [parent] the Identifier or uuid of the parent object. - * @property {Object} [customParameters] any additional parameters to be passed to the domain object's form. E.g. '[aria-label="Data Rate (hz)"]': {'0.1'} + * @property {Record} [customParameters] any additional parameters to be passed to the domain object's form. E.g. '[aria-label="Data Rate (hz)"]': {'0.1'} */ /** diff --git a/example/exampleUser/ExampleUserProvider.js b/example/exampleUser/ExampleUserProvider.js index d80735abd0..712b18868a 100644 --- a/example/exampleUser/ExampleUserProvider.js +++ b/example/exampleUser/ExampleUserProvider.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { v4 as uuid } from 'uuid'; import createExampleUser from './exampleUserCreator.js'; diff --git a/example/generator/SinewaveStalenessProvider.js b/example/generator/SinewaveStalenessProvider.js index 06783dada3..0a54022ed4 100644 --- a/example/generator/SinewaveStalenessProvider.js +++ b/example/generator/SinewaveStalenessProvider.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; export default class SinewaveLimitProvider extends EventEmitter { #openmct; diff --git a/openmct.js b/openmct.js index ecf82b35ad..aebb13a077 100644 --- a/openmct.js +++ b/openmct.js @@ -22,13 +22,38 @@ const matcher = /\/openmct.js$/; if (document.currentScript) { + // @ts-ignore let src = document.currentScript.src; if (src && matcher.test(src)) { - // eslint-disable-next-line no-undef + // @ts-ignore __webpack_public_path__ = src.replace(matcher, '') + '/'; } } +import { MCT } from './src/MCT.js'; + +const openmct = new MCT(); + +export default openmct; + +/** + * @typedef {MCT} OpenMCT + * @typedef {import('./src/api/objects/ObjectAPI.js').DomainObject} DomainObject + * @typedef {import('./src/api/objects/ObjectAPI.js').Identifier} Identifier + * @typedef {import('./src/api/objects/Transaction.js').default} Transaction + * @typedef {import('./src/api/actions/ActionsAPI.js').Action} Action + * @typedef {import('./src/api/actions/ActionCollection.js').default} ActionCollection + * @typedef {import('./src/api/composition/CompositionCollection.js').default} CompositionCollection + * @typedef {import('./src/api/composition/CompositionProvider.js').default} CompositionProvider + * @typedef {import('./src/ui/registries/ViewRegistry.js').ViewProvider} ViewProvider + * @typedef {import('./src/ui/registries/ViewRegistry.js').View} View + * + * @typedef {DomainObject[]} ObjectPath + * @typedef {(...args: any[]) => (openmct: OpenMCT) => void} OpenMCTPlugin + * An OpenMCT Plugin returns a function that receives an instance of + * the OpenMCT API and uses it to install itself. + */ + /** * @typedef {Object} BuildInfo * @property {string} version @@ -36,46 +61,3 @@ if (document.currentScript) { * @property {string} revision * @property {string} branch */ -/** - * @typedef {Object} OpenMCT - * @property {BuildInfo} buildInfo - * @property {import('./src/selection/Selection').default} selection - * @property {import('./src/api/time/TimeAPI').default} time - * @property {import('./src/api/composition/CompositionAPI').default} composition - * @property {import('./src/ui/registries/ViewRegistry').default} objectViews - * @property {import('./src/ui/registries/InspectorViewRegistry').default} inspectorViews - * @property {import('./src/ui/registries/ViewRegistry').default} propertyEditors - * @property {import('./src/ui/registries/ToolbarRegistry').default} toolbars - * @property {import('./src/api/types/TypeRegistry').default} types - * @property {import('./src/api/objects/ObjectAPI').default} objects - * @property {import('./src/api/telemetry/TelemetryAPI').default} telemetry - * @property {import('./src/api/indicators/IndicatorAPI').default} indicators - * @property {import('./src/api/user/UserAPI').default} user - * @property {import('./src/api/notifications/NotificationAPI').default} notifications - * @property {import('./src/api/Editor').default} editor - * @property {import('./src/api/overlays/OverlayAPI')} overlays - * @property {import('./src/api/tooltips/ToolTipAPI')} tooltips - * @property {import('./src/api/menu/MenuAPI').default} menus - * @property {import('./src/api/actions/ActionsAPI').default} actions - * @property {import('./src/api/status/StatusAPI').default} status - * @property {import('./src/api/priority/PriorityAPI').default} priority - * @property {import('./src/ui/router/ApplicationRouter')} router - * @property {import('./src/api/faultmanagement/FaultManagementAPI').default} faults - * @property {import('./src/api/forms/FormsAPI').default} forms - * @property {import('./src/api/Branding').default} branding - * @property {import('./src/api/annotation/AnnotationAPI').default} annotation - * @property {{(plugin: OpenMCTPlugin) => void}} install - * @property {{() => string}} getAssetPath - * @property {{(assetPath: string) => void}} setAssetPath - * @property {{(domElement: HTMLElement, isHeadlessMode: boolean) => void}} start - * @property {{() => void}} startHeadless - * @property {{() => void}} destroy - * @property {OpenMCTPlugin[]} plugins - * @property {OpenMCTComponent[]} components - */ -import { MCT } from './src/MCT.js'; - -/** @type {OpenMCT} */ -const openmct = new MCT(); - -export default openmct; diff --git a/src/MCT.js b/src/MCT.js index 75215abefd..6c16c3e8c2 100644 --- a/src/MCT.js +++ b/src/MCT.js @@ -19,8 +19,7 @@ * this source code distribution or the Licensing information page available * at runtime from the About dialog for additional information. *****************************************************************************/ -/* eslint-disable no-undef */ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { createApp, markRaw } from 'vue'; import ActionsAPI from './api/actions/ActionsAPI.js'; @@ -73,10 +72,25 @@ import Browse from './ui/router/Browse.js'; * The Open MCT application. This may be configured by installing plugins * or registering extensions before the application is started. * @constructor - * @memberof module:openmct - * @extends EventEmitter */ export class MCT extends EventEmitter { + /** + * @type {import('openmct.js').BuildInfo} + */ + buildInfo; + /** + * @type {string} + */ + defaultClock; + /** + * @type {Record} + */ + plugins; + /** + * Tracks current selection state of the application. + * @type {Selection} + */ + selection; constructor() { super(); @@ -89,21 +103,11 @@ export class MCT extends EventEmitter { this.destroy = this.destroy.bind(this); this.defaultClock = 'local'; - this.plugins = plugins; - - /** - * Tracks current selection state of the application. - * @private - */ this.selection = new Selection(this); /** - * MCT's time conductor, which may be used to synchronize view contents - * for telemetry- or time-based views. - * @type {module:openmct.TimeConductor} - * @memberof module:openmct.MCT# - * @name conductor + * @type {TimeAPI} */ this.time = new TimeAPI(this); @@ -116,9 +120,7 @@ export class MCT extends EventEmitter { * `composition` may be called as a function, in which case it acts * as [`composition.get`]{@link module:openmct.CompositionAPI#get}. * - * @type {module:openmct.CompositionAPI} - * @memberof module:openmct.MCT# - * @name composition + * @type {CompositionAPI} */ this.composition = new CompositionAPI(this); @@ -126,9 +128,7 @@ export class MCT extends EventEmitter { * Registry for views of domain objects which should appear in the * main viewing area. * - * @type {module:openmct.ViewRegistry} - * @memberof module:openmct.MCT# - * @name objectViews + * @type {ViewRegistry} */ this.objectViews = new ViewRegistry(); @@ -136,9 +136,7 @@ export class MCT extends EventEmitter { * Registry for views which should appear in the Inspector area. * These views will be chosen based on the selection state. * - * @type {module:openmct.InspectorViewRegistry} - * @memberof module:openmct.MCT# - * @name inspectorViews + * @type {InspectorViewRegistry} */ this.inspectorViews = new InspectorViewRegistry(); @@ -147,9 +145,7 @@ export class MCT extends EventEmitter { * dialogs, and similar user interface elements used for * modifying domain objects external to its regular views. * - * @type {module:openmct.ViewRegistry} - * @memberof module:openmct.MCT# - * @name propertyEditors + * @type {ViewRegistry} */ this.propertyEditors = new ViewRegistry(); @@ -157,9 +153,7 @@ export class MCT extends EventEmitter { * Registry for views which should appear in the toolbar area while * editing. These views will be chosen based on the selection state. * - * @type {module:openmct.ToolbarRegistry} - * @memberof module:openmct.MCT# - * @name toolbars + * @type {ToolbarRegistry} */ this.toolbars = new ToolbarRegistry(); @@ -167,9 +161,7 @@ export class MCT extends EventEmitter { * Registry for domain object types which may exist within this * instance of Open MCT. * - * @type {module:openmct.TypeRegistry} - * @memberof module:openmct.MCT# - * @name types + * @type {TypeRegistry} */ this.types = new TypeRegistry(); @@ -177,9 +169,7 @@ export class MCT extends EventEmitter { * An interface for interacting with domain objects and the domain * object hierarchy. * - * @type {module:openmct.ObjectAPI} - * @memberof module:openmct.MCT# - * @name objects + * @type {ObjectAPI} */ this.objects = new ObjectAPI(this.types, this); @@ -187,49 +177,100 @@ export class MCT extends EventEmitter { * An interface for retrieving and interpreting telemetry data associated * with a domain object. * - * @type {module:openmct.TelemetryAPI} - * @memberof module:openmct.MCT# - * @name telemetry + * @type {TelemetryAPI} */ this.telemetry = new TelemetryAPI(this); /** * An interface for creating new indicators and changing them dynamically. * - * @type {module:openmct.IndicatorAPI} - * @memberof module:openmct.MCT# - * @name indicators + * @type {IndicatorAPI} */ this.indicators = new IndicatorAPI(this); /** * MCT's user awareness management, to enable user and * role specific functionality. - * @type {module:openmct.UserAPI} - * @memberof module:openmct.MCT# - * @name user + * @type {UserAPI} */ this.user = new UserAPI(this); + /** + * An interface for managing notifications and alerts. + * @type {NotificationAPI} + */ this.notifications = new NotificationAPI(); + + /** + * An interface for editing domain objects. + * @type {EditorAPI} + */ this.editor = new EditorAPI(this); + + /** + * An interface for managing overlays. + * @type {OverlayAPI} + */ this.overlays = new OverlayAPI(); + + /** + * An interface for managing tooltips. + * @type {ToolTipAPI} + */ this.tooltips = new ToolTipAPI(); + + /** + * An interface for managing menus. + * @type {MenuAPI} + */ this.menus = new MenuAPI(this); + + /** + * An interface for managing menu actions. + * @type {ActionsAPI} + */ this.actions = new ActionsAPI(this); + + /** + * An interface for managing statuses. + * @type {StatusAPI} + */ this.status = new StatusAPI(this); + + /** + * An object defining constants for priority levels. + * @type {PriorityAPI} + */ this.priority = PriorityAPI; + + /** + * An interface for routing application traffic. + * @type {ApplicationRouter} + */ this.router = new ApplicationRouter(this); + + /** + * An interface for managing faults. + * @type {FaultManagementAPI} + */ this.faults = new FaultManagementAPI(this); + + /** + * An interface for managing forms. + * @type {FormsAPI} + */ this.forms = new FormsAPI(this); + + /** + * An interface for branding the application. + * @type {BrandingAPI} + */ this.branding = BrandingAPI; /** * MCT's annotation API that enables * human-created comments and categorization linked to data products - * @type {module:openmct.AnnotationAPI} - * @memberof module:openmct.MCT# - * @name annotation + * @type {AnnotationAPI} */ this.annotation = new AnnotationAPI(this); @@ -268,7 +309,6 @@ export class MCT extends EventEmitter { } /** * Set path to where assets are hosted. This should be the path to main.js. - * @memberof module:openmct.MCT# * @method setAssetPath */ setAssetPath(assetPath) { @@ -276,7 +316,6 @@ export class MCT extends EventEmitter { } /** * Get path to where assets are hosted. - * @memberof module:openmct.MCT# * @method getAssetPath */ getAssetPath() { @@ -295,9 +334,8 @@ export class MCT extends EventEmitter { * Start running Open MCT. This should be called only after any plugins * have been installed. * @fires module:openmct.MCT~start - * @memberof module:openmct.MCT# * @method start - * @param {HTMLElement} [domElement] the DOM element in which to run + * @param {Element?} domElement the DOM element in which to run * MCT; if undefined, MCT will be run in the body of the document */ start(domElement = document.body.firstElementChild, isHeadlessMode = false) { @@ -330,7 +368,6 @@ export class MCT extends EventEmitter { * Fired by [MCT]{@link module:openmct.MCT} when the application * is started. * @event start - * @memberof module:openmct.MCT~ */ if (!isHeadlessMode) { const appLayout = createApp(Layout); @@ -361,7 +398,6 @@ export class MCT extends EventEmitter { * * @param {Function} plugin a plugin install function which will be * invoked with the mct instance. - * @memberof module:openmct.MCT# */ install(plugin) { plugin(this); @@ -372,3 +408,7 @@ export class MCT extends EventEmitter { this.emit('destroy'); } } + +/** + * @typedef {import('../openmct.js').OpenMCTPlugin} OpenMCTPlugin + */ diff --git a/src/api/Editor.js b/src/api/Editor.js index 218ec4dd04..7bf981e179 100644 --- a/src/api/Editor.js +++ b/src/api/Editor.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; export default class Editor extends EventEmitter { constructor(openmct) { diff --git a/src/api/actions/ActionCollection.js b/src/api/actions/ActionCollection.js index d099edb25d..73a1d690ee 100644 --- a/src/api/actions/ActionCollection.js +++ b/src/api/actions/ActionCollection.js @@ -20,10 +20,22 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; +/** + * A collection of actions applicable to a domain object. + * @extends EventEmitter + */ class ActionCollection extends EventEmitter { + /** + * Creates an instance of ActionCollection. + * @param {Object.} applicableActions - The actions applicable to the domain object. + * @param {import('openmct').ObjectPath} objectPath - The path to the domain object. + * @param {import('openmct').View} view - The view displaying the domain object. + * @param {import('openmct').OpenMCT} openmct - The Open MCT API. + * @param {boolean} skipEnvironmentObservers - Whether to skip setting up environment observers. + */ constructor(applicableActions, objectPath, view, openmct, skipEnvironmentObservers) { super(); @@ -48,6 +60,10 @@ class ActionCollection extends EventEmitter { } } + /** + * Disables the specified actions. + * @param {string[]} actionKeys - The keys of the actions to disable. + */ disable(actionKeys) { actionKeys.forEach((actionKey) => { if (this.applicableActions[actionKey]) { @@ -57,6 +73,10 @@ class ActionCollection extends EventEmitter { this._update(); } + /** + * Enables the specified actions. + * @param {string[]} actionKeys - The keys of the actions to enable. + */ enable(actionKeys) { actionKeys.forEach((actionKey) => { if (this.applicableActions[actionKey]) { @@ -66,6 +86,10 @@ class ActionCollection extends EventEmitter { this._update(); } + /** + * Hides the specified actions. + * @param {string[]} actionKeys - The keys of the actions to hide. + */ hide(actionKeys) { actionKeys.forEach((actionKey) => { if (this.applicableActions[actionKey]) { @@ -75,6 +99,10 @@ class ActionCollection extends EventEmitter { this._update(); } + /** + * Shows the specified actions. + * @param {string[]} actionKeys - The keys of the actions to show. + */ show(actionKeys) { actionKeys.forEach((actionKey) => { if (this.applicableActions[actionKey]) { @@ -84,6 +112,9 @@ class ActionCollection extends EventEmitter { this._update(); } + /** + * Destroys the action collection, removing all listeners and observers. + */ destroy() { if (!this.skipEnvironmentObservers) { this.objectUnsubscribes.forEach((unsubscribe) => { @@ -97,6 +128,10 @@ class ActionCollection extends EventEmitter { this.removeAllListeners(); } + /** + * Gets all visible actions. + * @returns {Action[]} An array of visible actions. + */ getVisibleActions() { let actionsArray = Object.keys(this.applicableActions); let visibleActions = []; @@ -112,6 +147,10 @@ class ActionCollection extends EventEmitter { return visibleActions; } + /** + * Gets all actions that should be shown in the status bar. + * @returns {Action[]} An array of status bar actions. + */ getStatusBarActions() { let actionsArray = Object.keys(this.applicableActions); let statusBarActions = []; @@ -127,17 +166,34 @@ class ActionCollection extends EventEmitter { return statusBarActions; } + /** + * Gets the object containing all applicable actions. + * @returns {Object.} The object of applicable actions. + */ getActionsObject() { return this.applicableActions; } + /** + * Emits an update event with the current applicable actions. + * @private + */ _update() { this.emit('update', this.applicableActions); } + /** + * Sets up observers for the object path. + * @private + */ _observeObjectPath() { let actionCollection = this; + /** + * Updates an object with new properties. + * @param {Object} oldObject - The object to update. + * @param {Object} newObject - The object containing new properties. + */ function updateObject(oldObject, newObject) { Object.assign(oldObject, newObject); @@ -157,6 +213,10 @@ class ActionCollection extends EventEmitter { }); } + /** + * Updates the applicable actions. + * @private + */ _updateActions() { let newApplicableActions = this.openmct.actions._applicableActions(this.objectPath, this.view); @@ -167,6 +227,13 @@ class ActionCollection extends EventEmitter { this._update(); } + /** + * Merges old and new actions, preserving existing action states. + * @param {Object.} oldActions - The existing actions. + * @param {Object.} newActions - The new actions. + * @returns {Object.} The merged actions. + * @private + */ _mergeOldAndNewActions(oldActions, newActions) { let mergedActions = {}; Object.keys(newActions).forEach((key) => { @@ -182,3 +249,7 @@ class ActionCollection extends EventEmitter { } export default ActionCollection; + +/** + * @typedef {import('openmct').Action} Action + */ diff --git a/src/api/actions/ActionsAPI.js b/src/api/actions/ActionsAPI.js index fdb984a328..9ca37194f6 100644 --- a/src/api/actions/ActionsAPI.js +++ b/src/api/actions/ActionsAPI.js @@ -19,19 +19,30 @@ * this source code distribution or the Licensing information page available * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; import ActionCollection from './ActionCollection.js'; +/** + * The ActionsAPI manages the registration and retrieval of actions in Open MCT. + * @extends EventEmitter + */ class ActionsAPI extends EventEmitter { + /** + * @param {import('openmct').OpenMCT} openmct - The Open MCT instance + */ constructor(openmct) { super(); + /** @type {Object} */ this._allActions = {}; + /** @type {WeakMap} */ this._actionCollections = new WeakMap(); + /** @type {import('openmct').OpenMCT} */ this._openmct = openmct; + /** @type {string[]} */ this._groupOrder = ['windowing', 'undefined', 'view', 'action', 'export', 'import']; this.register = this.register.bind(this); @@ -40,14 +51,29 @@ class ActionsAPI extends EventEmitter { this._updateCachedActionCollections = this._updateCachedActionCollections.bind(this); } + /** + * Register an action with the API. + * @param {Action} actionDefinition - The definition of the action to register + */ register(actionDefinition) { this._allActions[actionDefinition.key] = actionDefinition; } + /** + * Get an action by its key. + * @param {string} key - The key of the action to retrieve + * @returns {Action|undefined} The action definition, or undefined if not found + */ getAction(key) { return this._allActions[key]; } + /** + * Get or create an ActionCollection for a given object path and view. + * @param {import('openmct').ObjectPath} objectPath - The path of the object + * @param {import('openmct').View} [view] - The view object + * @returns {ActionCollection} The ActionCollection for the given object path and view + */ getActionsCollection(objectPath, view) { if (view) { return ( @@ -59,14 +85,31 @@ class ActionsAPI extends EventEmitter { } } + /** + * Update the order in which action groups are displayed. + * @param {string[]} groupArray - An array of group names in the desired order + */ updateGroupOrder(groupArray) { this._groupOrder = groupArray; } + /** + * Get a cached ActionCollection for a given view. + * @param {import('openmct').ObjectPath} objectPath - The path of the object + * @param {Object} view - The view object + * @returns {ActionCollection|undefined} The cached ActionCollection, or undefined if not found + */ _getCachedActionCollection(objectPath, view) { return this._actionCollections.get(view); } + /** + * Create a new ActionCollection. + * @param {import('openmct').ObjectPath} objectPath - The path of the object + * @param {import('openmct').View} [view] - The view object + * @param {boolean} skipEnvironmentObservers - Whether to skip environment observers + * @returns {ActionCollection} The new ActionCollection + */ _newActionCollection(objectPath, view, skipEnvironmentObservers) { let applicableActions = this._applicableActions(objectPath, view); @@ -84,20 +127,35 @@ class ActionsAPI extends EventEmitter { return actionCollection; } + /** + * Cache an ActionCollection for a given view. + * @param {import('openmct').View} view - The view object + * @param {ActionCollection} actionCollection - The ActionCollection to cache + */ _cacheActionCollection(view, actionCollection) { this._actionCollections.set(view, actionCollection); actionCollection.on('destroy', this._updateCachedActionCollections); } - _updateCachedActionCollections(key) { - if (this._actionCollections.has(key)) { - let actionCollection = this._actionCollections.get(key); + /** + * Update cached ActionCollections when destroyed. + * @param {import('openmct').View} view - The key (View object)of the destroyed ActionCollection + */ + _updateCachedActionCollections(view) { + if (this._actionCollections.has(view)) { + let actionCollection = this._actionCollections.get(view); actionCollection.off('destroy', this._updateCachedActionCollections); delete actionCollection.applicableActions; - this._actionCollections.delete(key); + this._actionCollections.delete(view); } } + /** + * Get applicable actions for a given object path and view. + * @param {import('openmct').ObjectPath} objectPath - The path of the object + * @param {import('openmct').View} [view] - The view object + * @returns {Object} A dictionary of applicable actions keyed by action key + */ _applicableActions(objectPath, view) { let actionsObject = {}; @@ -120,6 +178,11 @@ class ActionsAPI extends EventEmitter { return actionsObject; } + /** + * Group and sort actions based on their group and priority. + * @param {Action[]|Object} actionsArray - An array or object of actions to group and sort + * @returns {Action[][]} An array of grouped and sorted action arrays + */ _groupAndSortActions(actionsArray = []) { if (!Array.isArray(actionsArray) && typeof actionsArray === 'object') { actionsArray = Object.keys(actionsArray).map((key) => actionsArray[key]); @@ -153,3 +216,19 @@ class ActionsAPI extends EventEmitter { } export default ActionsAPI; + +/** + * @typedef {Object} Action + * @property {string} name - The display name of the action. + * @property {string} key - A unique identifier for the action. + * @property {string} description - A brief description of what the action does. + * @property {string} cssClass - The CSS class for the action's icon. + * @property {string} [group] - The group this action belongs to (e.g., 'action', 'import'). + * @property {number} [priority] - The priority of the action within its group (controls the order of the actions in the menu). + * @property {boolean} [isHidden] - Whether the action should be hidden from menus. + * @property {(objectPath: ObjectPath, view: View) => void} invoke - Executes the action. + * @property {(objectPath: ObjectPath, view: View) => boolean} appliesTo - Determines if the action is applicable to the given object path. + */ + +/** @typedef {import('openmct').ObjectPath} ObjectPath */ +/** @typedef {import('openmct').View} View */ diff --git a/src/api/annotation/AnnotationAPI.js b/src/api/annotation/AnnotationAPI.js index 7f2ba5a99e..3ca5889b1b 100644 --- a/src/api/annotation/AnnotationAPI.js +++ b/src/api/annotation/AnnotationAPI.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; import { v4 as uuid } from 'uuid'; @@ -41,8 +41,14 @@ const ANNOTATION_TYPES = Object.freeze({ PLOT_SPATIAL: 'PLOT_SPATIAL' }); +/** + * @type {string} + */ const ANNOTATION_TYPE = 'annotation'; +/** + * @type {string} + */ const ANNOTATION_LAST_CREATED = 'annotationLastCreated'; /** @@ -53,15 +59,15 @@ const ANNOTATION_LAST_CREATED = 'annotationLastCreated'; */ /** - * @typedef {import('../objects/ObjectAPI').DomainObject} DomainObject + * @typedef {import('openmct').DomainObject} DomainObject */ /** - * @typedef {import('../objects/ObjectAPI').Identifier} Identifier + * @typedef {import('openmct').Identifier} Identifier */ /** - * @typedef {import('../../../openmct').OpenMCT} OpenMCT + * @typedef {import('openmct').OpenMCT} OpenMCT */ /** @@ -73,7 +79,8 @@ const ANNOTATION_LAST_CREATED = 'annotationLastCreated'; * about rationals behind why the robot has taken a certain path. * Annotations are discoverable using search, and are typically rendered in OpenMCT views to bring attention * to other users. - * @constructor + * @class AnnotationAPI + * @extends {EventEmitter} */ export default class AnnotationAPI extends EventEmitter { /** @type {Map boolean >>} */ @@ -120,10 +127,9 @@ export default class AnnotationAPI extends EventEmitter { * @property {Array} targets The targets ID keystrings and their specific properties. * For plots, this will be a bounding box, e.g.: {keyString: "d8385009-789d-457b-acc7-d50ba2fd55ea", maxY: 100, minY: 0, maxX: 100, minX: 0} * For notebooks, this will be an entry ID, e.g.: {entryId: "entry-ecb158f5-d23c-45e1-a704-649b382622ba"} - * @property {DomainObject>[]} targetDomainObjects the domain objects this annotation points to (e.g., telemetry objects for a plot) + * @property {DomainObject[]} targetDomainObjects the domain objects this annotation points to (e.g., telemetry objects for a plot) */ /** - * @method create * @param {CreateAnnotationOptions} options * @returns {Promise} a promise which will resolve when the domain object * has been created, or be rejected if it cannot be saved @@ -195,6 +201,10 @@ export default class AnnotationAPI extends EventEmitter { } } + /** + * Updates the annotation modified timestamp for a target domain object + * @param {DomainObject} targetDomainObject The target domain object to update + */ #updateAnnotationModified(targetDomainObject) { // As certain telemetry objects are immutable, we'll need to check here first // to see if we can add the annotation last created property. @@ -207,8 +217,8 @@ export default class AnnotationAPI extends EventEmitter { } /** - * @method defineTag - * @param {string} key a unique identifier for the tag + * Defines a new tag + * @param {string} tagKey a unique identifier for the tag * @param {Tag} tagsDefinition the definition of the tag to add */ defineTag(tagKey, tagsDefinition) { @@ -216,7 +226,7 @@ export default class AnnotationAPI extends EventEmitter { } /** - * @method setNamespaceToSaveAnnotations + * Sets the namespace to save new annotations to * @param {string} namespace the namespace to save new annotations to */ setNamespaceToSaveAnnotations(namespace) { @@ -224,7 +234,7 @@ export default class AnnotationAPI extends EventEmitter { } /** - * @method isAnnotation + * Checks if a domain object is an annotation * @param {DomainObject} domainObject the domainObject in question * @returns {boolean} Returns true if the domain object is an annotation */ @@ -233,7 +243,7 @@ export default class AnnotationAPI extends EventEmitter { } /** - * @method getAvailableTags + * Gets the available tags that have been loaded * @returns {Tag[]} Returns an array of the available tags that have been loaded */ getAvailableTags() { @@ -252,10 +262,10 @@ export default class AnnotationAPI extends EventEmitter { } /** - * @method getAnnotations + * Gets annotations for a given domain object identifier * @param {Identifier} domainObjectIdentifier - The domain object identifier to use to search for annotations. For example, a telemetry object identifier. - * @param {AbortSignal} abortSignal - An abort signal to cancel the search - * @returns {DomainObject[]} Returns an array of annotations that match the search query + * @param {AbortSignal} [abortSignal] - An abort signal to cancel the search + * @returns {Promise} Returns a promise that resolves to an array of annotations that match the search query */ async getAnnotations(domainObjectIdentifier, abortSignal = null) { const keyStringQuery = this.openmct.objects.makeKeyString(domainObjectIdentifier); @@ -273,8 +283,8 @@ export default class AnnotationAPI extends EventEmitter { } /** - * @method deleteAnnotations - * @param {DomainObject[]} existingAnnotation - An array of annotations to delete (set _deleted to true) + * Deletes (marks as deleted) the given annotations + * @param {DomainObject[]} annotations - An array of annotations to delete (set _deleted to true) */ deleteAnnotations(annotations) { if (!annotations) { @@ -289,7 +299,7 @@ export default class AnnotationAPI extends EventEmitter { } /** - * @method deleteAnnotations + * Undeletes (marks as not deleted) the given annotation * @param {DomainObject} annotation - An annotation to undelete (set _deleted to false) */ unDeleteAnnotation(annotation) { @@ -300,6 +310,12 @@ export default class AnnotationAPI extends EventEmitter { this.openmct.objects.mutate(annotation, '_deleted', false); } + /** + * Gets tags from the given annotations + * @param {DomainObject[]} annotations - The annotations to get tags from + * @param {boolean} [filterDuplicates=true] - Whether to filter out duplicate tags + * @returns {Tag[]} An array of tags from the given annotations + */ getTagsFromAnnotations(annotations, filterDuplicates = true) { if (!annotations) { return []; @@ -324,6 +340,11 @@ export default class AnnotationAPI extends EventEmitter { return fullTagModels; } + /** + * Adds meta information to the given tags + * @param {string[]} tags - The tags to add meta information to + * @returns {Tag[]} An array of tags with meta information added + */ #addTagMetaInformationToTags(tags) { // Convert to Set and back to Array to remove duplicates const uniqueTags = [...new Set(tags)]; @@ -336,6 +357,11 @@ export default class AnnotationAPI extends EventEmitter { }); } + /** + * Gets tags that match the given query + * @param {string} query - The query to match tags against + * @returns {string[]} An array of tag keys that match the query + */ #getMatchingTags(query) { if (!query) { return []; @@ -352,6 +378,88 @@ export default class AnnotationAPI extends EventEmitter { return matchingTags; } + /** + * @typedef {Object} AnnotationTarget + * @property {string} keyString - The key string of the target + * @property {*} [additionalProperties] - Additional properties depending on the annotation type + */ + + /** + * @typedef {Object} TargetModel + * @property {import('openmct').DomainObject[]} originalPath - The original path of the target object + * @property {*} [additionalProperties] - Additional properties of the target domain object + */ + + /** + * @typedef {Object} AnnotationResult + * @property {string} name - The name of the annotation + * @property {string} type - The type of the object (always 'annotation') + * @property {{key: string, namespace: string}} identifier - The identifier of the annotation + * @property {string[]} tags - Array of tag keys associated with the annotation + * @property {boolean} _deleted - Whether the annotation is marked as deleted + * @property {ANNOTATION_TYPES} annotationType - The type of the annotation + * @property {string} contentText - The text content of the annotation + * @property {string} originalContextPath - The original context path of the annotation + * @property {AnnotationTarget[]} targets - Array of targets for the annotation + * @property {Tag[]} fullTagModels - Full tag models including metadata + * @property {string[]} matchingTagKeys - Array of tag keys that matched the search query + * @property {TargetModel[]} targetModels - Array of target models with additional information + */ + + /** + * Combines annotations with the same targets + * @param {AnnotationResult[]} results - The results to combine + * @returns {AnnotationResult[]} The combined results + */ + #combineSameTargets(results) { + const combinedResults = []; + results.forEach((currentAnnotation) => { + const existingAnnotation = combinedResults.find((annotationToFind) => { + const { annotationType, targets } = currentAnnotation; + return this.areAnnotationTargetsEqual(annotationType, targets, annotationToFind.targets); + }); + if (!existingAnnotation) { + combinedResults.push(currentAnnotation); + } else { + existingAnnotation.tags.push(...currentAnnotation.tags); + } + }); + + return combinedResults; + } + + /** + * Breaks apart annotations with multiple targets into separate results + * @param {AnnotationResult[]} results - The results to break apart + * @returns {AnnotationResult[]} The separated results + */ + #breakApartSeparateTargets(results) { + const separateResults = []; + results.forEach((result) => { + result.targets.forEach((target) => { + const targetID = target.keyString; + const separatedResult = { + ...result + }; + separatedResult.targets = [target]; + separatedResult.targetModels = result.targetModels.filter((targetModel) => { + const targetKeyString = this.openmct.objects.makeKeyString(targetModel.identifier); + + return targetKeyString === targetID; + }); + separateResults.push(separatedResult); + }); + }); + + return separateResults; + } + + /** + * Adds tag meta information to the given results + * @param {AnnotationResult[]} results - The results to add tag meta information to + * @param {string[]} matchingTagKeys - The matching tag keys + * @returns {AnnotationResult[]} The results with tag meta information added + */ #addTagMetaInformationToResults(results, matchingTagKeys) { const tagsAddedToResults = results.map((result) => { const fullTagModels = this.#addTagMetaInformationToTags(result.tags); @@ -366,6 +474,12 @@ export default class AnnotationAPI extends EventEmitter { return tagsAddedToResults; } + /** + * Adds target models to the results + * @param {AnnotationResult[]} results - The results to add target models to + * @param {AbortSignal} abortSignal - The abort signal + * @returns {Promise} The results with target models added + */ async #addTargetModelsToResults(results, abortSignal) { const modelAddedToResults = await Promise.all( results.map(async (result) => { @@ -397,54 +511,11 @@ export default class AnnotationAPI extends EventEmitter { return modelAddedToResults; } - #combineSameTargets(results) { - const combinedResults = []; - results.forEach((currentAnnotation) => { - const existingAnnotation = combinedResults.find((annotationToFind) => { - const { annotationType, targets } = currentAnnotation; - return this.areAnnotationTargetsEqual(annotationType, targets, annotationToFind.targets); - }); - if (!existingAnnotation) { - combinedResults.push(currentAnnotation); - } else { - existingAnnotation.tags.push(...currentAnnotation.tags); - } - }); - - return combinedResults; - } - /** - * @method #breakApartSeparateTargets - * @param {Array} results A set of search results that could have the multiple targets for the same result - * @returns {Array} The same set of results, but with each target separated out into its own result - */ - #breakApartSeparateTargets(results) { - const separateResults = []; - results.forEach((result) => { - result.targets.forEach((target) => { - const targetID = target.keyString; - const separatedResult = { - ...result - }; - separatedResult.targets = [target]; - separatedResult.targetModels = result.targetModels.filter((targetModel) => { - const targetKeyString = this.openmct.objects.makeKeyString(targetModel.identifier); - - return targetKeyString === targetID; - }); - separateResults.push(separatedResult); - }); - }); - - return separateResults; - } - - /** - * @method searchForTags - * @param {string} query A query to match against tags. E.g., "dr" will match the tags "drilling" and "driving" - * @param {Object} [abortController] An optional abort method to stop the query - * @returns {Promise} returns a model of matching tags with their target domain objects attached + * Searches for tags matching the given query + * @param {string} query - A query to match against tags + * @param {AbortSignal} [abortSignal] - An optional abort signal to stop the query + * @returns {Promise} A promise that resolves to an array of matching annotation results */ async searchForTags(query, abortSignal) { const matchingTagKeys = this.#getMatchingTags(query); diff --git a/src/api/composition/CompositionAPI.js b/src/api/composition/CompositionAPI.js index 3420e94c0f..cfe3f8ab7f 100644 --- a/src/api/composition/CompositionAPI.js +++ b/src/api/composition/CompositionAPI.js @@ -28,7 +28,7 @@ import DefaultCompositionProvider from './DefaultCompositionProvider.js'; */ /** - * @typedef {import('../objects/ObjectAPI').DomainObject} DomainObject + * @typedef {import('openmct').DomainObject} DomainObject */ /** @@ -72,7 +72,7 @@ export default class CompositionAPI { * * @method get * @param {DomainObject} domainObject - * @returns {CompositionCollection} + * @returns {CompositionCollection | undefined} */ get(domainObject) { const provider = this.registry.find((p) => { diff --git a/src/api/composition/CompositionCollection.js b/src/api/composition/CompositionCollection.js index 6699845059..e4da621344 100644 --- a/src/api/composition/CompositionCollection.js +++ b/src/api/composition/CompositionCollection.js @@ -21,7 +21,7 @@ *****************************************************************************/ /** - * @typedef {import('../objects/ObjectAPI').DomainObject} DomainObject + * @typedef {import('openmct').DomainObject} DomainObject */ /** @@ -200,10 +200,9 @@ export default class CompositionCollection { /** * Load the domain objects in this composition. * - * @param {AbortSignal} abortSignal + * @param {AbortSignal} [abortSignal] * @returns {Promise.>} a promise for * the domain objects in this composition - * @memberof {module:openmct.CompositionCollection#} * @name load */ async load(abortSignal) { @@ -280,7 +279,7 @@ export default class CompositionCollection { /** * Handle adds from provider. * @private - * @param {import('../objects/ObjectAPI').Identifier} childId + * @param {import('openmct').Identifier} childId * @returns {DomainObject} */ #onProviderAdd(childId) { diff --git a/src/api/composition/CompositionProvider.js b/src/api/composition/CompositionProvider.js index ec68648a68..c663352198 100644 --- a/src/api/composition/CompositionProvider.js +++ b/src/api/composition/CompositionProvider.js @@ -24,11 +24,11 @@ import _ from 'lodash'; import { makeKeyString, parseKeyString } from '../objects/object-utils.js'; /** - * @typedef {import('../objects/ObjectAPI').DomainObject} DomainObject + * @typedef {import('openmct').DomainObject} DomainObject */ /** - * @typedef {import('../objects/ObjectAPI').Identifier} Identifier + * @typedef {import('openmct').Identifier} Identifier */ /** @@ -36,7 +36,7 @@ import { makeKeyString, parseKeyString } from '../objects/object-utils.js'; */ /** - * @typedef {import('../../../openmct').OpenMCT} OpenMCT + * @typedef {import('openmct').OpenMCT} OpenMCT */ /** @@ -84,7 +84,7 @@ export default class CompositionProvider { * Check if this provider should be used to load composition for a * particular domain object. * @method appliesTo - * @param {import('../objects/ObjectAPI').DomainObject} domainObject the domain object + * @param {DomainObject} domainObject the domain object * to check * @returns {boolean} true if this provider can provide composition for a given domain object */ @@ -98,7 +98,6 @@ export default class CompositionProvider { * for which to load composition * @returns {Promise} a promise for * the Identifiers in this composition - * @method load */ load(domainObject) { throw new Error('This method must be implemented by a subclass.'); @@ -137,7 +136,6 @@ export default class CompositionProvider { * @param {DomainObject} domainObject the domain object * which should have its composition modified * @param {Identifier} childId the domain object to remove - * @method remove */ remove(domainObject, childId) { throw new Error('This method must be implemented by a subclass.'); @@ -151,7 +149,6 @@ export default class CompositionProvider { * @param {DomainObject} parent the domain object * which should have its composition modified * @param {Identifier} childId the domain object to add - * @method add */ add(parent, childId) { throw new Error('This method must be implemented by a subclass.'); @@ -179,7 +176,6 @@ export default class CompositionProvider { /** * Listens on general mutation topic, using injector to fetch to avoid * circular dependencies. - * @private */ #establishTopicListener() { if (this.topicListener) { @@ -194,7 +190,6 @@ export default class CompositionProvider { } /** - * @private * @param {DomainObject} parent * @param {DomainObject} child * @returns {boolean} @@ -207,7 +202,6 @@ export default class CompositionProvider { } /** - * @private * @param {DomainObject} parent * @returns {boolean} */ @@ -219,7 +213,6 @@ export default class CompositionProvider { * Handles mutation events. If there are active listeners for the mutated * object, detects changes to composition and triggers necessary events. * - * @private * @param {DomainObject} oldDomainObject */ #onMutation(newDomainObject, oldDomainObject) { @@ -230,6 +223,10 @@ export default class CompositionProvider { return; } + if (oldDomainObject.composition === undefined || newDomainObject.composition === undefined) { + return; + } + const oldComposition = oldDomainObject.composition.map(makeKeyString); const newComposition = newDomainObject.composition.map(makeKeyString); diff --git a/src/api/composition/DefaultCompositionProvider.js b/src/api/composition/DefaultCompositionProvider.js index 05a6603505..91567c2fe8 100644 --- a/src/api/composition/DefaultCompositionProvider.js +++ b/src/api/composition/DefaultCompositionProvider.js @@ -25,11 +25,11 @@ import { makeKeyString } from '../objects/object-utils.js'; import CompositionProvider from './CompositionProvider.js'; /** - * @typedef {import('../objects/ObjectAPI').DomainObject} DomainObject + * @typedef {import('openmct').DomainObject} DomainObject */ /** - * @typedef {import('../objects/ObjectAPI').Identifier} Identifier + * @typedef {import('openmct').Identifier} Identifier */ /** diff --git a/src/api/faultmanagement/FaultManagementAPI.js b/src/api/faultmanagement/FaultManagementAPI.js index 3be3e3be71..07eef4f7cb 100644 --- a/src/api/faultmanagement/FaultManagementAPI.js +++ b/src/api/faultmanagement/FaultManagementAPI.js @@ -45,7 +45,7 @@ export default class FaultManagementAPI { } /** - * @param {import("../objects/ObjectAPI").DomainObject} domainObject + * @param {import('openmct').DomainObject} domainObject * @returns {Promise.} */ request(domainObject) { @@ -57,7 +57,7 @@ export default class FaultManagementAPI { } /** - * @param {import("../objects/ObjectAPI").DomainObject} domainObject + * @param {import('openmct').DomainObject} domainObject * @param {Function} callback * @returns {Function} unsubscribe */ diff --git a/src/api/forms/FormController.js b/src/api/forms/FormController.js index 4c946573c9..f69970c1dc 100644 --- a/src/api/forms/FormController.js +++ b/src/api/forms/FormController.js @@ -1,3 +1,25 @@ +/***************************************************************************** + * Open MCT, Copyright (c) 2014-2024, 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 mount from 'utils/mount'; import AutoCompleteField from './components/controls/AutoCompleteField.vue'; @@ -11,6 +33,8 @@ import SelectField from './components/controls/SelectField.vue'; import TextAreaField from './components/controls/TextAreaField.vue'; import TextField from './components/controls/TextField.vue'; import ToggleSwitchField from './components/controls/ToggleSwitchField.vue'; + +/** @type {Record} */ export const DEFAULT_CONTROLS_MAP = { autocomplete: AutoCompleteField, checkbox: CheckBoxField, @@ -26,6 +50,12 @@ export const DEFAULT_CONTROLS_MAP = { }; export default class FormControl { + /** @type {Record} */ + controls; + + /** + * @param {OpenMCT} openmct + */ constructor(openmct) { this.openmct = openmct; this.controls = {}; @@ -33,6 +63,10 @@ export default class FormControl { this._addDefaultFormControls(); } + /** + * @param {string} controlName + * @param {ControlViewProvider} controlViewProvider + */ addControl(controlName, controlViewProvider) { const control = this.controls[controlName]; if (control) { @@ -44,6 +78,10 @@ export default class FormControl { this.controls[controlName] = controlViewProvider; } + /** + * @param {string} controlName + * @returns {ControlViewProvider | undefined} + */ getControl(controlName) { const control = this.controls[controlName]; if (!control) { @@ -65,6 +103,8 @@ export default class FormControl { /** * @private + * @param {string} control + * @returns {ControlViewProvider} */ _getControlViewProvider(control) { const self = this; @@ -106,3 +146,13 @@ export default class FormControl { }; } } + +/** + * @typedef {import('openmct')} OpenMCT + */ + +/** + * @typedef {Object} ControlViewProvider + * @property {(element: HTMLElement, model: any, onChange: Function) => any} show + * @property {() => void} destroy + */ diff --git a/src/api/forms/FormsAPI.js b/src/api/forms/FormsAPI.js index f43c18b625..aba7b32286 100644 --- a/src/api/forms/FormsAPI.js +++ b/src/api/forms/FormsAPI.js @@ -26,7 +26,14 @@ import mount from 'utils/mount'; import FormProperties from './components/FormProperties.vue'; import FormController from './FormController.js'; +/** + * The FormsAPI provides methods for creating and managing forms in Open MCT. + */ export default class FormsAPI { + /** + * Creates an instance of FormsAPI. + * @param {import('openmct').OpenMCT} openmct - The Open MCT API + */ constructor(openmct) { this.openmct = openmct; this.formController = new FormController(openmct); @@ -47,7 +54,6 @@ export default class FormsAPI { * Create a new form control definition with a formControlViewProvider * this formControlViewProvider is used inside form overlay to show/render a form row * - * @public * @param {string} controlName a form structure, array of section * @param {ControlViewProvider} controlViewProvider */ @@ -58,7 +64,6 @@ export default class FormsAPI { /** * Get a ControlViewProvider for a given/stored form controlName * - * @public * @param {string} controlName a form structure, array of section * @return {ControlViewProvider} */ @@ -80,20 +85,20 @@ export default class FormsAPI { * @property {string} control represents type of row to render * eg:autocomplete,composite,datetime,file-input,locator,numberfield,select,textarea,textfield * @property {string} cssClass class name for styling this row - * @property {module:openmct.DomainObject} domainObject object to be used by row + * @property {import('openmct').DomainObject} domainObject object to be used by row * @property {string} key id for this row * @property {string} name Name of the row to display on Form - * @property {module:openmct.DomainObject} parent parent object to be used by row + * @property {import('openmct').DomainObject} parent parent object to be used by row * @property {boolean} required is this row mandatory * @property {function} validate a function to validate this row on any changes */ /** * Show form inside an Overlay dialog with given form structure - * @public * @param {Array
} formStructure a form structure, array of section * @param {Object} options - * @property {function} onChange a callback function when any changes detected + * @param {() => void} [options.onChange] a callback function when any changes detected + * @returns {Promise} A promise that resolves with the form data when saved, or rejects when cancelled */ showForm(formStructure, { onChange } = {}) { let overlay; @@ -134,11 +139,11 @@ export default class FormsAPI { /** * Show form as a child of the element provided with given form structure * - * @public * @param {Array
} formStructure a form structure, array of section * @param {Object} options - * @property {HTMLElement} element Parent Element to render a Form - * @property {function} onChange a callback function when any changes detected + * @param {HTMLElement} options.element Parent Element to render a Form + * @param {() => void} [options.onChange] a callback function when any changes detected + * @returns {Promise} A promise that resolves with the form data when saved, or rejects when cancelled */ showCustomForm(formStructure, { element, onChange } = {}) { if (element === undefined) { @@ -179,6 +184,10 @@ export default class FormsAPI { } ); + /** + * Handles form property changes + * @param {Object} data - The changed form data + */ function onFormPropertyChange(data) { if (onChange) { onChange(data); @@ -196,6 +205,11 @@ export default class FormsAPI { } } + /** + * Creates a form action handler + * @param {() => void} callback - The callback to be called when the form action is triggered + * @returns {(...args: any[]) => void} The form action handler + */ function onFormAction(callback) { return () => { destroy(); diff --git a/src/api/forms/toggle-check-box-mixin.js b/src/api/forms/toggle-check-box-mixin.js index bb3429bcba..9b3c8f9be0 100644 --- a/src/api/forms/toggle-check-box-mixin.js +++ b/src/api/forms/toggle-check-box-mixin.js @@ -1,3 +1,25 @@ +/***************************************************************************** + * Open MCT, Copyright (c) 2014-2024, 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. + *****************************************************************************/ + export default { emits: ['on-change'], data() { diff --git a/src/api/indicators/IndicatorAPI.js b/src/api/indicators/IndicatorAPI.js index e5a3e2f30e..24eebc1158 100644 --- a/src/api/indicators/IndicatorAPI.js +++ b/src/api/indicators/IndicatorAPI.js @@ -20,11 +20,18 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import vueWrapHtmlElement from '../../utils/vueWrapHtmlElement.js'; import SimpleIndicator from './SimpleIndicator.js'; +/** + * The Indicator API is used to add indicators to the Open MCT UI. + * An indicator appears in the top navigation bar and can be used to + * display information or trigger actions. + * + * @extends EventEmitter + */ class IndicatorAPI extends EventEmitter { /** @type {import('../../../openmct.js').OpenMCT} */ openmct; @@ -51,38 +58,39 @@ class IndicatorAPI extends EventEmitter { /** * @typedef {Object} Indicator - * @property {HTMLElement} [element] - * @property {VueComponent|Promise} [vueComponent] - * @property {string} key - * @property {number} priority + * @property {HTMLElement} [element] - The HTML element of the indicator. Optional if using vueComponent. + * @property {VueComponent|Promise} [vueComponent] - The Vue component for the indicator. Optional if using element. + * @property {string} key - The unique key for the indicator. + * @property {number} priority - The priority of the indicator (default: -1). */ /** - * Accepts an indicator object, which is a simple object - * with a two attributes: 'element' which has an HTMLElement - * as its value, and 'priority' with an integer that specifies its order in the layout. - * The lower the priority, the further to the right the element is placed. - * If undefined, the priority will be assigned -1. + * Adds an indicator to the API. * - * We provide .simpleIndicator() as a convenience function - * which will create a default Open MCT indicator that can - * be passed to .add(indicator). This indicator also exposes - * functions for changing its appearance to support customization - * and dynamic behavior. + * @param {Indicator} indicator - The indicator object to add. * - * Eg. + * @description + * The indicator object is a simple object with two main attributes: + * - 'element': An HTMLElement (optional if using vueComponent). + * - 'priority': An integer specifying its order in the layout. Lower priority + * places the element further to the right. If undefined, defaults to -1. + * + * A convenience function `.simpleIndicator()` is provided to create a default + * Open MCT indicator that can be passed to `.add(indicator)`. This indicator + * exposes functions for customizing its appearance and behavior. + * + * Example usage: + * ``` * const myIndicator = openmct.indicators.simpleIndicator(); * openmct.indicators.add(myIndicator); * * myIndicator.text("Hello World!"); * myIndicator.iconClass("icon-info"); + * ``` * - * If you would like to use a Vue component, you can pass it in - * directly as the 'vueComponent' attribute of the indicator object. - * This accepts a Vue component or a promise that resolves to a Vue component (for asynchronous - * rendering). - * - * @param {Indicator} indicator + * For Vue components, pass the component directly as the 'vueComponent' + * attribute. This can be a Vue component or a promise resolving to a + * Vue component for asynchronous rendering. */ add(indicator) { if (!indicator.priority) { diff --git a/src/api/indicators/SimpleIndicator.js b/src/api/indicators/SimpleIndicator.js index 45d2af18d2..c1e18c10e0 100644 --- a/src/api/indicators/SimpleIndicator.js +++ b/src/api/indicators/SimpleIndicator.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { convertTemplateToHTML } from '@/utils/template/templateHelpers'; diff --git a/src/api/menu/MenuAPI.js b/src/api/menu/MenuAPI.js index 317eeba8ce..40da1c0b58 100644 --- a/src/api/menu/MenuAPI.js +++ b/src/api/menu/MenuAPI.js @@ -22,32 +22,14 @@ import Menu, { MENU_PLACEMENT } from './menu.js'; -/** - * Popup Menu options - * @typedef {Object} MenuOptions - * @property {string} menuClass Class for popup menu - * @property {MENU_PLACEMENT} placement Placement for menu relative to click - * @property {Function} onDestroy callback function: invoked when menu is destroyed - */ - -/** - * Popup Menu Item/action - * @typedef {Object} Action - * @property {string} cssClass Class for menu item - * @property {boolean} isDisabled adds disable class if true - * @property {string} name Menu item text - * @property {string} description Menu item description - * @property {Function} onItemClicked callback function: invoked when item is clicked - */ - /** * The MenuAPI allows the addition of new context menu actions, and for the context menu to be launched from * custom HTML elements. - * @interface MenuAPI - * @memberof module:openmct */ - class MenuAPI { + /** + * @param {import('openmct').OpenMCT} openmct + */ constructor(openmct) { this.openmct = openmct; @@ -61,10 +43,10 @@ class MenuAPI { /** * Show popup menu - * @param {number} x x-coordinates for popup - * @param {number} y x-coordinates for popup - * @param {Array.|Array.>} actions collection of actions{@link Action} or collection of groups of actions {@link Action} - * @param {MenuOptions} [menuOptions] [Optional] The {@link MenuOptions} options for Menu + * @param {number} x - x-coordinates for popup + * @param {number} y - y-coordinates for popup + * @param {Action[]|Action[][]} items - collection of actions or collection of groups of actions + * @param {MenuOptions} [menuOptions] - The options for Menu */ showMenu(x, y, items, menuOptions) { this._createMenuComponent(x, y, items, menuOptions); @@ -72,6 +54,13 @@ class MenuAPI { this.menuComponent.showMenu(); } + /** + * Convert actions to menu items + * @param {Action[]} actions - collection of actions + * @param {import('openmct').ObjectPath} objectPath - The object path + * @param {import('openmct').ViewProvider} view - The view provider + * @returns {Action[]} + */ actionsToMenuItems(actions, objectPath, view) { return actions.map((action) => { const isActionGroup = Array.isArray(action); @@ -87,10 +76,10 @@ class MenuAPI { /** * Show popup menu with description of item on hover - * @param {number} x x-coordinates for popup - * @param {number} y x-coordinates for popup - * @param {Array.|Array.>} actions collection of actions {@link Action} or collection of groups of actions {@link Action} - * @param {MenuOptions} [menuOptions] [Optional] The {@link MenuOptions} options for Menu + * @param {number} x - x-coordinates for popup + * @param {number} y - y-coordinates for popup + * @param {Action[]|Action[][]} actions - collection of actions or collection of groups of actions + * @param {MenuOptions} [menuOptions] - The options for Menu */ showSuperMenu(x, y, actions, menuOptions) { this._createMenuComponent(x, y, actions, menuOptions); @@ -98,11 +87,23 @@ class MenuAPI { this.menuComponent.showSuperMenu(); } + /** + * Clear the menu component + * @private + */ _clearMenuComponent() { this.menuComponent = undefined; delete this.menuComponent; } + /** + * Create a menu component + * @param {number} x - x-coordinates for popup + * @param {number} y - y-coordinates for popup + * @param {Action[]|Action[][]} actions - collection of actions or collection of groups of actions + * @param {MenuOptions} menuOptions - The options for Menu + * @private + */ _createMenuComponent(x, y, actions, menuOptions = {}) { if (this.menuComponent) { this.menuComponent.dismiss(); @@ -119,6 +120,14 @@ class MenuAPI { this.menuComponent.once('destroy', this._clearMenuComponent); } + /** + * Show object menu + * @param {import('openmct').ObjectPath} objectPath - The object path + * @param {number} x - x-coordinates for popup + * @param {number} y - y-coordinates for popup + * @param {string[]} actionsToBeIncluded - Actions to be included in the menu + * @private + */ _showObjectMenu(objectPath, x, y, actionsToBeIncluded) { let applicableActions = this.openmct.actions._groupedAndSortedObjectActions( objectPath, @@ -129,3 +138,14 @@ class MenuAPI { } } export default MenuAPI; + +/** + * @typedef {Object} MenuOptions + * @property {string} [menuClass] - Class for popup menu + * @property {MENU_PLACEMENT} [placement] - Placement for menu relative to click + * @property {() => void} [onDestroy] - callback function: invoked when menu is destroyed + */ + +/** + * @typedef {import('openmct').Action} Action + */ diff --git a/src/api/menu/menu.js b/src/api/menu/menu.js index f6c5fd3c47..da7de0d336 100644 --- a/src/api/menu/menu.js +++ b/src/api/menu/menu.js @@ -19,13 +19,18 @@ * this source code distribution or the Licensing information page available * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import mount from 'utils/mount'; import { h } from 'vue'; import MenuComponent from './components/MenuComponent.vue'; import SuperMenuComponent from './components/SuperMenu.vue'; +/** + * Enum for menu placement options. + * @readonly + * @enum {string} + */ export const MENU_PLACEMENT = { TOP: 'top', TOP_LEFT: 'top-left', @@ -37,7 +42,15 @@ export const MENU_PLACEMENT = { RIGHT: 'right' }; +/** + * Class representing a menu. + * @extends EventEmitter + */ class Menu extends EventEmitter { + /** + * Create a menu. + * @param {MenuOptions} options - The options for the menu. + */ constructor(options) { super(); @@ -52,6 +65,9 @@ class Menu extends EventEmitter { this.showSuperMenu = this.showSuperMenu.bind(this); } + /** + * Dismiss the menu. + */ dismiss() { if (this.destroy) { this.destroy(); @@ -61,6 +77,9 @@ class Menu extends EventEmitter { this.emit('destroy'); } + /** + * Show the menu component. + */ showMenu() { if (this.destroy) { return; @@ -80,6 +99,9 @@ class Menu extends EventEmitter { this.show(); } + /** + * Show the super menu component. + */ showSuperMenu() { const { vNode, destroy } = mount({ data() { @@ -102,6 +124,9 @@ class Menu extends EventEmitter { this.show(); } + /** + * Show the menu. + */ show() { document.body.appendChild(this.el); document.addEventListener('click', this.dismiss); @@ -109,3 +134,8 @@ class Menu extends EventEmitter { } export default Menu; + +/** + * @typedef {Object} MenuOptions + * @property {() => void} [onDestroy] - Callback function to be called when the menu is destroyed. + */ diff --git a/src/api/notifications/NotificationAPI.js b/src/api/notifications/NotificationAPI.js index 6074d2bd12..1776e7531f 100644 --- a/src/api/notifications/NotificationAPI.js +++ b/src/api/notifications/NotificationAPI.js @@ -27,65 +27,22 @@ * much as possible, notifications share a model with blocking * dialogs so that the same information can be provided in a dialog * and then minimized to a banner notification if needed. - * - * @namespace platform/api/notifications */ -import EventEmitter from 'eventemitter3'; +import { EventEmitter } from 'eventemitter3'; import moment from 'moment'; -/** - * @typedef {Object} NotificationProperties - * @property {function} dismiss Dismiss the notification - * @property {NotificationModel} model The Notification model - * @property {(progressPerc: number, progressText: string) => void} [progress] Update the progress of the notification - */ - -/** - * @typedef {EventEmitter & NotificationProperties} Notification - */ - -/** - * @typedef {Object} NotificationLink - * @property {function} onClick The function to be called when the link is clicked - * @property {string} cssClass A CSS class name to style the link - * @property {string} text The text to be displayed for the link - */ - -/** - * @typedef {Object} NotificationOptions - * @property {number} [autoDismissTimeout] Milliseconds to wait before automatically dismissing the notification - * @property {boolean} [minimized] Allows for a notification to be minimized into the indicator by default - * @property {NotificationLink} [link] A link for the notification - */ - -/** - * A representation of a banner notification. Banner notifications - * are used to inform users of events in a non-intrusive way. As - * much as possible, notifications share a model with blocking - * dialogs so that the same information can be provided in a dialog - * and then minimized to a banner notification if needed, or vice-versa. - * - * @see DialogModel - * @typedef {Object} NotificationModel - * @property {string} message The message to be displayed by the notification - * @property {number | 'unknown'} [progress] The progress of some ongoing task. Should be a number between 0 and 100, or - * with the string literal 'unknown'. - * @property {string} [progressText] A message conveying progress of some ongoing task. - * @property {string} [severity] The severity of the notification. Should be one of 'info', 'alert', or 'error'. - * @property {string} [timestamp] The time at which the notification was created. Should be a string in ISO 8601 format. - * @property {boolean} [minimized] Whether or not the notification has been minimized - * @property {boolean} [autoDismiss] Whether the notification should be automatically dismissed after a short period of time. - * @property {NotificationOptions} options The notification options - */ - const DEFAULT_AUTO_DISMISS_TIMEOUT = 3000; const MINIMIZE_ANIMATION_TIMEOUT = 300; /** * The notification service is responsible for informing the user of * events via the use of banner notifications. + * @extends EventEmitter */ export default class NotificationAPI extends EventEmitter { + /** + * @constructor + */ constructor() { super(); /** @type {Notification[]} */ @@ -123,12 +80,7 @@ export default class NotificationAPI extends EventEmitter { /** * Present an alert to the user. * @param {string} message The message to display to the user. - * @param {NotificationOptions} [options] object with following properties - * autoDismissTimeout: {number} in milliseconds to automatically dismisses notification - * link: {Object} Add a link to notifications for navigation - * onClick: callback function - * cssClass: css class name to add style on link - * text: text to display for link + * @param {NotificationOptions} [options] The notification options * @returns {Notification} */ alert(message, options = {}) { @@ -143,13 +95,8 @@ export default class NotificationAPI extends EventEmitter { /** * Present an error message to the user - * @param {string} message - * @param {Object} [options] object with following properties - * autoDismissTimeout: {number} in milliseconds to automatically dismisses notification - * link: {Object} Add a link to notifications for navigation - * onClick: callback function - * cssClass: css class name to add style on link - * text: text to display for link + * @param {string} message The error message to display + * @param {NotificationOptions} [options] The notification options * @returns {Notification} */ error(message, options = {}) { @@ -164,9 +111,10 @@ export default class NotificationAPI extends EventEmitter { /** * Create a new progress notification. These notifications will contain a progress bar. - * @param {string} message + * @param {string} message The message to display * @param {number | null} progressPerc A value between 0 and 100, or null. * @param {string} [progressText] Text description of progress (eg. "10 of 20 objects copied"). + * @returns {Notification} */ progress(message, progressPerc, progressText) { let notificationModel = { @@ -180,6 +128,9 @@ export default class NotificationAPI extends EventEmitter { return this._notify(notificationModel); } + /** + * Dismiss all active notifications. + */ dismissAllNotifications() { this.notifications = []; this.emit('dismiss-all'); @@ -192,7 +143,7 @@ export default class NotificationAPI extends EventEmitter { * dismissed. * * @private - * @param {Notification | undefined} notification + * @param {Notification | undefined} notification The notification to minimize */ _minimize(notification) { if (!notification) { @@ -204,13 +155,13 @@ export default class NotificationAPI extends EventEmitter { if (this.activeTimeout) { /* - Method can be called manually (clicking dismiss) or - automatically from an auto-timeout. this.activeTimeout - acts as a semaphore to prevent race conditions. Cancel any - timeout in progress (for the case where a manual dismiss - has shortcut an active auto-dismiss), and clear the - semaphore. - */ + * Method can be called manually (clicking dismiss) or + * automatically from an auto-timeout. this.activeTimeout + * acts as a semaphore to prevent race conditions. Cancel any + * timeout in progress (for the case where a manual dismiss + * has shortcut an active auto-dismiss), and clear the + * semaphore. + */ clearTimeout(this.activeTimeout); delete this.activeTimeout; } @@ -232,11 +183,10 @@ export default class NotificationAPI extends EventEmitter { * message banner and remove it from the list of notifications. * Typically only notifications with a severity of info should be * dismissed. If you're not sure whether to dismiss or minimize a - * notification, use {@link Notification#dismissOrMinimize}. - * dismiss + * notification, use {@link NotificationAPI#_dismissOrMinimize}. * * @private - * @param {Notification | undefined} notification + * @param {Notification | undefined} notification The notification to dismiss */ _dismiss(notification) { if (!notification) { @@ -247,14 +197,14 @@ export default class NotificationAPI extends EventEmitter { let index = this.notifications.indexOf(notification); if (this.activeTimeout) { - /* Method can be called manually (clicking dismiss) or + /* + * Method can be called manually (clicking dismiss) or * automatically from an auto-timeout. this.activeTimeout * acts as a semaphore to prevent race conditions. Cancel any * timeout in progress (for the case where a manual dismiss * has shortcut an active auto-dismiss), and clear the * semaphore. */ - clearTimeout(this.activeTimeout); delete this.activeTimeout; } @@ -273,7 +223,7 @@ export default class NotificationAPI extends EventEmitter { * dismiss or minimize where appropriate. * * @private - * @param {Notification | undefined} notification + * @param {Notification | undefined} notification The notification to dismiss or minimize */ _dismissOrMinimize(notification) { let model = notification?.model; @@ -285,6 +235,7 @@ export default class NotificationAPI extends EventEmitter { } /** + * Sets the highest severity notification. * @private */ _setHighestSeverity() { @@ -308,10 +259,10 @@ export default class NotificationAPI extends EventEmitter { * already active, then it will be dismissed or minimized automatically, * and the provided notification displayed in its place. * - * @param {NotificationModel} notificationModel The notification to - * display + * @private + * @param {NotificationModel} notificationModel The notification to display * @returns {Notification} the provided notification decorated with - * functions to {@link Notification#dismiss} or {@link Notification#minimize} + * functions to dismiss or minimize */ _notify(notificationModel) { let notification; @@ -326,21 +277,21 @@ export default class NotificationAPI extends EventEmitter { this._setHighestSeverity(); /* - Check if there is already an active (ie. visible) notification - */ + * Check if there is already an active (ie. visible) notification + */ if (!this.activeNotification && !notification?.model?.options?.minimized) { this._setActiveNotification(notification); } else if (!this.activeTimeout) { /* - If there is already an active notification, time it out. If it's - already got a timeout in progress (either because it has had - timeout forced because of a queue of messages, or it had an - autodismiss specified), leave it to run. Otherwise force a - timeout. - - This notification has been added to queue and will be - serviced as soon as possible. - */ + * If there is already an active notification, time it out. If it's + * already got a timeout in progress (either because it has had + * timeout forced because of a queue of messages, or it had an + * autodismiss specified), leave it to run. Otherwise force a + * timeout. + * + * This notification has been added to queue and will be + * serviced as soon as possible. + */ this.activeTimeout = setTimeout(() => { this._dismissOrMinimize(activeNotification); }, DEFAULT_AUTO_DISMISS_TIMEOUT); @@ -350,8 +301,9 @@ export default class NotificationAPI extends EventEmitter { } /** + * Creates a new notification object. * @private - * @param {NotificationModel} notificationModel + * @param {NotificationModel} notificationModel The model for the notification * @returns {Notification} */ _createNotification(notificationModel) { @@ -374,8 +326,9 @@ export default class NotificationAPI extends EventEmitter { } /** + * Sets the active notification. * @private - * @param {Notification | undefined} notification + * @param {Notification | undefined} notification The notification to set as active */ _setActiveNotification(notification) { this.activeNotification = notification; @@ -400,18 +353,18 @@ export default class NotificationAPI extends EventEmitter { } /** - * Used internally by the NotificationService - * + * Selects the next notification to be displayed. * @private + * @returns {Notification | undefined} */ _selectNextNotification() { let notification; let i = 0; /* - Loop through the notifications queue and find the first one that - has not already been minimized (manually or otherwise). - */ + * Loop through the notifications queue and find the first one that + * has not already been minimized (manually or otherwise). + */ for (; i < this.notifications.length; i++) { notification = this.notifications[i]; @@ -424,3 +377,41 @@ export default class NotificationAPI extends EventEmitter { } } } + +/** + * @typedef {Object} NotificationProperties + * @property {() => void} dismiss Dismiss the notification + * @property {NotificationModel} model The Notification model + * @property {(progressPerc: number, progressText: string) => void} [progress] Update the progress of the notification + */ + +/** + * @typedef {EventEmitter & NotificationProperties} Notification + */ + +/** + * @typedef {Object} NotificationLink + * @property {() => void} onClick The function to be called when the link is clicked + * @property {string} cssClass A CSS class name to style the link + * @property {string} text The text to be displayed for the link + */ + +/** + * @typedef {Object} NotificationOptions + * @property {number} [autoDismissTimeout] Milliseconds to wait before automatically dismissing the notification + * @property {boolean} [minimized] Allows for a notification to be minimized into the indicator by default + * @property {NotificationLink} [link] A link for the notification + */ + +/** + * A representation of a banner notification. + * @typedef {Object} NotificationModel + * @property {string} message The message to be displayed by the notification + * @property {number | 'unknown'} [progress] The progress of some ongoing task. Should be a number between 0 and 100, or 'unknown'. + * @property {string} [progressText] A message conveying progress of some ongoing task. + * @property {'info' | 'alert' | 'error'} [severity] The severity of the notification. + * @property {string} [timestamp] The time at which the notification was created. Should be a string in ISO 8601 format. + * @property {boolean} [minimized] Whether or not the notification has been minimized + * @property {boolean} [autoDismiss] Whether the notification should be automatically dismissed after a short period of time. + * @property {NotificationOptions} options The notification options + */ diff --git a/src/api/objects/ConflictError.js b/src/api/objects/ConflictError.js index bf996e96e6..63d2845f5d 100644 --- a/src/api/objects/ConflictError.js +++ b/src/api/objects/ConflictError.js @@ -1 +1,28 @@ +/***************************************************************************** + * Open MCT, Copyright (c) 2014-2024, 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. + *****************************************************************************/ + +/** + * Represents an error that occurs when there is a conflict (409) while trying to + * persist an object. + * This class extends the built-in Error class. + */ export default class ConflictError extends Error {} diff --git a/src/api/objects/InterceptorRegistry.js b/src/api/objects/InterceptorRegistry.js index 9f88d54456..57faab9fb6 100644 --- a/src/api/objects/InterceptorRegistry.js +++ b/src/api/objects/InterceptorRegistry.js @@ -24,7 +24,6 @@ export default class InterceptorRegistry { /** * A InterceptorRegistry maintains the definitions for different interceptors that may be invoked on domain objects. * @interface InterceptorRegistry - * @memberof module:openmct */ constructor() { this.interceptors = []; @@ -35,7 +34,6 @@ export default class InterceptorRegistry { * @property {function} appliesTo function that determines if this interceptor should be called for the given identifier/object * @property {function} invoke function that transforms the provided domain object and returns the transformed domain object * @property {function} priority the priority for this interceptor. A higher number returned has more weight than a lower number - * @memberof module:openmct InterceptorRegistry# */ /** @@ -43,7 +41,6 @@ export default class InterceptorRegistry { * * @param {module:openmct.InterceptorDef} interceptorDef the interceptor to add * @method addInterceptor - * @memberof module:openmct.InterceptorRegistry# */ addInterceptor(interceptorDef) { this.interceptors.push(interceptorDef); @@ -53,7 +50,6 @@ export default class InterceptorRegistry { * Retrieve all interceptors applicable to a domain object. * @method getInterceptors * @returns [module:openmct.InterceptorDef] the registered interceptors for this identifier/object - * @memberof module:openmct.InterceptorRegistry# */ getInterceptors(identifier, object) { function byPriority(interceptorA, interceptorB) { diff --git a/src/api/objects/MutableDomainObject.js b/src/api/objects/MutableDomainObject.js index 80c46598ff..9bccfc8308 100644 --- a/src/api/objects/MutableDomainObject.js +++ b/src/api/objects/MutableDomainObject.js @@ -19,7 +19,7 @@ * this source code distribution or the Licensing information page available * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; import { makeKeyString, refresh } from './object-utils.js'; @@ -38,7 +38,6 @@ const ANY_OBJECT_EVENT = 'mutation'; * (via openmct.objects.destroy) when you're done with it. * * @typedef MutableDomainObject - * @memberof module:openmct */ class MutableDomainObject { constructor(eventEmitter) { diff --git a/src/api/objects/ObjectAPI.js b/src/api/objects/ObjectAPI.js index a109f63706..c1be1581f6 100644 --- a/src/api/objects/ObjectAPI.js +++ b/src/api/objects/ObjectAPI.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { identifierEquals, makeKeyString, parseKeyString, refresh } from 'objectUtils'; import ConflictError from './ConflictError.js'; @@ -33,39 +33,21 @@ import Transaction from './Transaction.js'; /** * Uniquely identifies a domain object. - * * @typedef {Object} Identifier - * @property {string} namespace the namespace to/from which this domain - * object should be loaded/stored. - * @property {string} key a unique identifier for the domain object - * within that namespace - * @memberof module:openmct.ObjectAPI~ + * @property {string} namespace the namespace to/from which this domain object should be loaded/stored. + * @property {string} key a unique identifier for the domain object within that namespace */ /** - * A domain object is an entity of relevance to a user's workflow, that - * should appear as a distinct and meaningful object within the user - * interface. Examples of domain objects are folders, telemetry sensors, - * and so forth. - * - * A few common properties are defined for domain objects. Beyond these, - * individual types of domain objects may add more as they see fit. - * + * A domain object is an entity of relevance to a user's workflow, that should appear as a distinct and meaningful object within the user interface. * @typedef {Object} DomainObject - * @property {Identifier} identifier a key/namespace pair which - * uniquely identifies this domain object + * @property {Identifier} identifier a key/namespace pair which uniquely identifies this domain object * @property {string} type the type of domain object * @property {string} name the human-readable name for this domain object - * @property {string} [creator] the user name of the creator of this domain - * object - * @property {number} [modified] the time, in milliseconds since the UNIX - * epoch, at which this domain object was last modified - * @property {Identifier[]} [composition] if - * present, this will be used by the default composition provider - * to load domain objects - * @property {Object.} [configuration] A key-value map containing configuration - * settings for this domain object. - * @memberof module:openmct.ObjectAPI~ + * @property {string} [creator] the user name of the creator of this domain object + * @property {number} [modified] the time, in milliseconds since the UNIX epoch, at which this domain object was last modified + * @property {Identifier[]} [composition] if present, this will be used by the default composition provider to load domain objects + * @property {Record} [configuration] A key-value map containing configuration settings for this domain object. */ /** @@ -78,8 +60,6 @@ import Transaction from './Transaction.js'; /** * Utilities for loading, saving, and manipulating domain objects. - * @interface ObjectAPI - * @memberof module:openmct */ export default class ObjectAPI { #makeKeyString; @@ -88,6 +68,10 @@ export default class ObjectAPI { #refresh; #openmct; + /** + * @param {any} typeRegistry + * @param {any} openmct + */ constructor(typeRegistry, openmct) { this.#makeKeyString = makeKeyString; this.#parseKeyString = parseKeyString; @@ -125,6 +109,8 @@ export default class ObjectAPI { /** * Retrieve the provider for a given identifier. + * @param {Identifier} identifier + * @returns {ObjectProvider | RootObjectProvider} */ getProvider(identifier) { if (identifier.key === 'ROOT') { @@ -144,7 +130,7 @@ export default class ObjectAPI { /** * Get the root-level object. - * @returns {Promise.} a promise for the root object + * @returns {Promise} a promise for the root object */ getRoot() { return this.rootProvider.get(); @@ -154,63 +140,17 @@ export default class ObjectAPI { * Register a new object provider for a particular namespace. * * @param {string} namespace the namespace for which to provide objects - * @param {module:openmct.ObjectProvider} provider the provider which - * will handle loading domain objects from this namespace - * @memberof {module:openmct.ObjectAPI#} - * @name addProvider + * @param {ObjectProvider} provider the provider which will handle loading domain objects from this namespace */ addProvider(namespace, provider) { this.providers[namespace] = provider; } - /** - * Provides the ability to read, write, and delete domain objects. - * - * When registering a new object provider, all methods on this interface - * are optional. - * - * @interface ObjectProvider - * @memberof module:openmct - */ - - /** - * Create the given domain object in the corresponding persistence store - * - * @method create - * @memberof module:openmct.ObjectProvider# - * @param {module:openmct.DomainObject} domainObject the domain object to - * create - * @returns {Promise} a promise which will resolve when the domain object - * has been created, or be rejected if it cannot be saved - */ - - /** - * Update this domain object in its persistence store - * - * @method update - * @memberof module:openmct.ObjectProvider# - * @param {module:openmct.DomainObject} domainObject the domain object to - * update - * @returns {Promise} a promise which will resolve when the domain object - * has been updated, or be rejected if it cannot be saved - */ - - /** - * Delete this domain object. - * - * @method delete - * @memberof module:openmct.ObjectProvider# - * @param {module:openmct.DomainObject} domainObject the domain object to - * delete - * @returns {Promise} a promise which will resolve when the domain object - * has been deleted, or be rejected if it cannot be deleted - */ - /** * Get a domain object. * - * @param {string} key the key for the domain object to load - * @param {AbortSignal} abortSignal (optional) signal to abort fetch requests + * @param {Identifier | string} identifier the identifier for the domain object to load + * @param {AbortSignal} [abortSignal] (optional) signal to abort fetch requests * @param {boolean} [forceRemote=false] defaults to false. If true, will skip cached and * dirty/in-transaction objects use and the provider.get method * @returns {Promise} a promise which will resolve when the domain object @@ -289,14 +229,10 @@ export default class ObjectAPI { * and will be searched using the fallback in-memory search. * Search results are asynchronous and resolve in parallel. * - * @method search - * @memberof module:openmct.ObjectAPI# * @param {string} query the term to search for - * @param {AbortController.signal} abortSignal (optional) signal to cancel downstream fetch requests - * @param {string} searchType the type of search as defined by SEARCH_TYPES - * @returns {Array.>} - * an array of promises returned from each object provider's search function - * each resolving to domain objects matching provided search query and options. + * @param {AbortController.signal} [abortSignal] (optional) signal to cancel downstream fetch requests + * @param {string} [searchType=this.SEARCH_TYPES.OBJECTS] the type of search as defined by SEARCH_TYPES + * @returns {Promise[]} an array of promises returned from each object provider's search function, each resolving to domain objects matching the provided search query and options */ search(query, abortSignal, searchType = this.SEARCH_TYPES.OBJECTS) { if (!Object.keys(this.SEARCH_TYPES).includes(searchType.toUpperCase())) { @@ -330,9 +266,8 @@ export default class ObjectAPI { * platform will manage the lifecycle of any mutable objects that it provides. If you use `getMutable` you are * committing to managing that lifecycle yourself. `.destroy` should be called when the object is no longer needed. * - * @memberof {module:openmct.ObjectAPI#} - * @returns {Promise.} a promise that will resolve with a MutableDomainObject if - * the object can be mutated. + * @param {Identifier} identifier the identifier of the object to fetch + * @returns {Promise} a promise that will resolve with a MutableDomainObject if the object can be mutated */ getMutable(identifier) { if (!this.supportsMutation(identifier)) { @@ -348,7 +283,7 @@ export default class ObjectAPI { * This function is for cleaning up a mutable domain object when you're done with it. * You only need to use this if you retrieved the object using `getMutable()`. If the object was provided by the * platform (eg. passed into a `view()` function) then the platform is responsible for its lifecycle. - * @param {MutableDomainObject} domainObject + * @param {MutableDomainObject} domainObject the mutable domain object to destroy */ destroyMutable(domainObject) { if (domainObject.isMutable) { @@ -382,11 +317,8 @@ export default class ObjectAPI { /** * Save this domain object in its current state. * - * @memberof module:openmct.ObjectAPI# - * @param {module:openmct.DomainObject} domainObject the domain object to - * save - * @returns {Promise} a promise which will resolve when the domain object - * has been saved, or be rejected if it cannot be saved + * @param {DomainObject} domainObject the domain object to save + * @returns {Promise} a promise which will resolve when the domain object has been saved, or be rejected if it cannot be saved */ async save(domainObject) { const provider = this.getProvider(domainObject.identifier); @@ -517,7 +449,6 @@ export default class ObjectAPI { * this item(s) position in the root object's composition (example: order in object tree). * For arrays, they are treated as blocks. * @method addRoot - * @memberof module:openmct.ObjectAPI# */ addRoot(identifier, priority) { this.rootRegistry.addRoot(identifier, priority); @@ -530,7 +461,6 @@ export default class ObjectAPI { * * @param {module:openmct.InterceptorDef} interceptorDef the interceptor definition to add * @method addGetInterceptor - * @memberof module:openmct.InterceptorRegistry# */ addGetInterceptor(interceptorDef) { this.interceptorRegistry.addInterceptor(interceptorDef); @@ -538,7 +468,6 @@ export default class ObjectAPI { /** * Retrieve the interceptors for a given domain object. - * @private */ #listGetInterceptors(identifier, object) { return this.interceptorRegistry.getInterceptors(identifier, object); @@ -560,7 +489,7 @@ export default class ObjectAPI { /** * Return relative url path from a given object path * eg: #/browse/mine/cb56f6bf-c900-43b7-b923-2e3b64b412db/6e89e858-77ce-46e4-a1ad-749240286497/.... - * @param {Array} objectPath + * @param {Array} objectPath * @returns {string} relative url for object */ getRelativePath(objectPath) { @@ -612,13 +541,10 @@ export default class ObjectAPI { /** * Modify a domain object. Internal to ObjectAPI, won't call save after. - * @private * - * @param {module:openmct.DomainObject} object the object to mutate + * @param {DomainObject} domainObject the object to mutate * @param {string} path the property to modify * @param {*} value the new value for this property - * @method mutate - * @memberof module:openmct.ObjectAPI# */ #mutate(domainObject, path, value) { if (!this.supportsMutation(domainObject.identifier)) { @@ -628,28 +554,26 @@ export default class ObjectAPI { if (domainObject.isMutable) { domainObject.$set(path, value); } else { - //Creating a temporary mutable domain object allows other mutable instances of the - //object to be kept in sync. + // Creating a temporary mutable domain object allows other mutable instances of the + // object to be kept in sync. let mutableDomainObject = this.toMutable(domainObject); - //Mutate original object + // Mutate original object MutableDomainObject.mutateObject(domainObject, path, value); - //Mutate temporary mutable object, in the process informing any other mutable instances + // Mutate temporary mutable object, in the process informing any other mutable instances mutableDomainObject.$set(path, value); - //Destroy temporary mutable object + // Destroy temporary mutable object this.destroyMutable(mutableDomainObject); } } /** * Modify a domain object and save. - * @param {module:openmct.DomainObject} object the object to mutate + * @param {DomainObject} domainObject the object to mutate * @param {string} path the property to modify * @param {*} value the new value for this property - * @method mutate - * @memberof module:openmct.ObjectAPI# */ mutate(domainObject, path, value) { this.#mutate(domainObject, path, value); @@ -662,11 +586,9 @@ export default class ObjectAPI { } /** - * Create a mutable domain object from an existing domain object - * @param {module:openmct.DomainObject} domainObject the object to make mutable + * Create a mutable domain object from an existing domain object. + * @param {DomainObject} domainObject the object to make mutable * @returns {MutableDomainObject} a mutable domain object that will automatically sync - * @method toMutable - * @memberof module:openmct.ObjectAPI# */ toMutable(domainObject) { let mutableObject; @@ -689,8 +611,8 @@ export default class ObjectAPI { // modified can sometimes be undefined, so make it 0 in this case const mutableObjectModification = mutableObject.modified ?? Number.MIN_SAFE_INTEGER; if (updatedModel.persisted > mutableObjectModification) { - //Don't replace with a stale model. This can happen on slow connections when multiple mutations happen - //in rapid succession and intermediate persistence states are returned by the observe function. + // Don't replace with a stale model. This can happen on slow connections when multiple mutations happen + // in rapid succession and intermediate persistence states are returned by the observe function. updatedModel = this.applyGetInterceptors(identifier, updatedModel); mutableObject.$refresh(updatedModel); } @@ -706,10 +628,10 @@ export default class ObjectAPI { /** * Updates a domain object based on its latest persisted state. Note that this will mutate the provided object. - * @param {module:openmct.DomainObject} domainObject an object to refresh from its persistence store + * @param {DomainObject} domainObject an object to refresh from its persistence store * @param {boolean} [forceRemote=false] defaults to false. If true, will skip cached and * dirty/in-transaction objects use and the provider.get method - * @returns {Promise} the provided object, updated to reflect the latest persisted state of the object. + * @returns {Promise} the provided object, updated to reflect the latest persisted state of the object. */ async refresh(domainObject, forceRemote = false) { const refreshedObject = await this.get(domainObject.identifier, null, forceRemote); @@ -724,7 +646,8 @@ export default class ObjectAPI { } /** - * @param module:openmct.ObjectAPI~Identifier identifier An object identifier + * Determine if the object can be mutated. + * @param {Identifier} identifier An object identifier * @returns {boolean} true if the object can be mutated, otherwise returns false */ supportsMutation(identifier) { @@ -733,12 +656,10 @@ export default class ObjectAPI { /** * Observe changes to a domain object. - * @param {module:openmct.DomainObject} object the object to observe + * @param {DomainObject} domainObject the object to observe * @param {string} path the property to observe - * @param {Function} callback a callback to invoke when new values for - * this property are observed. - * @method observe - * @memberof module:openmct.ObjectAPI# + * @param {Function} callback a callback to invoke when new values for this property are observed. + * @returns {() => void} a function to unsubscribe from the updates */ observe(domainObject, path, callback) { if (domainObject.isMutable) { @@ -785,7 +706,7 @@ export default class ObjectAPI { /** * Given an original path check if the path is reachable via root - * @param {Array} originalPath an array of path objects to check + * @param {Array} originalPath an array of path objects to check * @returns {boolean} whether the domain object is reachable */ isReachable(originalPath) { @@ -796,6 +717,12 @@ export default class ObjectAPI { return false; } + /** + * Check if a path contains a domain object with a given key string + * @param {string} keyStringToCheck the keystring to check for + * @param {Array} path the path to check + * @returns {boolean} true if the path contains a DomainObject with the given keystring, otherwise false + */ #pathContainsDomainObject(keyStringToCheck, path) { if (!keyStringToCheck) { return false; @@ -810,10 +737,10 @@ export default class ObjectAPI { /** * Given an identifier, constructs the original path by walking up its parents - * @param {module:openmct.ObjectAPI~Identifier} identifier - * @param {Array} path an array of path objects + * @param {Identifier} identifier + * @param {Array} path an array of path objects * @param {AbortSignal} abortSignal (optional) signal to abort fetch requests - * @returns {Promise>} a promise containing an array of domain objects + * @returns {Promise>} a promise containing an array of domain objects */ async getOriginalPath(identifier, path = [], abortSignal = null) { const domainObject = await this.get(identifier, abortSignal); @@ -873,6 +800,12 @@ export default class ObjectAPI { return objectPath; } + /** + * Check if the object is a link based on its path + * @param {DomainObject} domainObject the DomainObject to check + * @param {Array} objectPath the object path to check + * @returns {boolean} true if the object path is a link, otherwise false + */ isObjectPathToALink(domainObject, objectPath) { return ( objectPath !== undefined && @@ -881,10 +814,19 @@ export default class ObjectAPI { ); } + /** + * Check if a transaction is active + * @returns {boolean} true if a transaction is active, otherwise false + */ isTransactionActive() { return this.transaction !== undefined && this.transaction !== null; } + /** + * Check if a domain object has already been persisted + * @param {DomainObject} domainObject the domain object to check + * @returns {boolean} true if the domain object has already been persisted, otherwise false + */ #hasAlreadyBeenPersisted(domainObject) { // modified can sometimes be undefined, so make it 0 in this case const modified = domainObject.modified ?? Number.MIN_SAFE_INTEGER; diff --git a/src/api/objects/RootObjectProvider.js b/src/api/objects/RootObjectProvider.js index 91e3195cd3..33de30a2e1 100644 --- a/src/api/objects/RootObjectProvider.js +++ b/src/api/objects/RootObjectProvider.js @@ -20,7 +20,13 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ +/** + * Provides the root object for the Open MCT application. + */ class RootObjectProvider { + /** + * @param {RootRegistry} rootRegistry - The registry containing root objects. + */ constructor(rootRegistry) { if (!RootObjectProvider.instance) { this.rootRegistry = rootRegistry; @@ -42,10 +48,18 @@ class RootObjectProvider { return RootObjectProvider.instance; // eslint-disable-line no-constructor-return } + /** + * Updates the name of the root object. + * @param {string} name - The new name for the root object. + */ updateName(name) { this.rootObject.name = name; } + /** + * Retrieves the root object with updated composition. + * @returns {Promise} A promise that resolves to the root object. + */ async get() { let roots = await this.rootRegistry.getRoots(); this.rootObject.composition = roots; @@ -54,8 +68,30 @@ class RootObjectProvider { } } +/** + * Creates or returns an instance of RootObjectProvider. + * @param {RootRegistry} rootRegistry - The registry containing root objects. + * @returns {RootObjectProvider} An instance of RootObjectProvider. + */ function instance(rootRegistry) { return new RootObjectProvider(rootRegistry); } export default instance; + +/** + * @typedef {import('openmct').Identifier} Identifier + */ + +/** + * @typedef {Object} RootObject + * @property {Identifier} identifier - The identifier of the root object. + * @property {string} name - The name of the root object. + * @property {string} type - The type of the root object. + * @property {Identifier[]} composition - The composition of the root object. + */ + +/** + * @typedef {Object} RootRegistry + * @property {() => Promise} getRoots - A method that returns a promise resolving to an array of root identifiers. + */ diff --git a/src/api/objects/RootRegistry.js b/src/api/objects/RootRegistry.js index 1f1b652bc4..2cc52dc58c 100644 --- a/src/api/objects/RootRegistry.js +++ b/src/api/objects/RootRegistry.js @@ -22,12 +22,24 @@ import { isIdentifier } from './object-utils.js'; +/** + * Registry for managing root items in Open MCT. + */ export default class RootRegistry { + /** + * @param {OpenMCT} openmct - The Open MCT instance. + */ constructor(openmct) { + /** @type {Array} */ this._rootItems = []; + /** @type {OpenMCT} */ this._openmct = openmct; } + /** + * Get all registered root items. + * @returns {Promise>} A promise that resolves to an array of root item identifiers. + */ getRoots() { const sortedItems = this._rootItems.sort((a, b) => b.priority - a.priority); const promises = sortedItems.map((rootItem) => rootItem.provider()); @@ -35,6 +47,11 @@ export default class RootRegistry { return Promise.all(promises).then((rootItems) => rootItems.flat()); } + /** + * Add a root item to the registry. + * @param {RootItemInput} rootItem - The root item to add. + * @param {number} [priority] - The priority of the root item. + */ addRoot(rootItem, priority) { if (!this._isValid(rootItem)) { return; @@ -46,6 +63,12 @@ export default class RootRegistry { }); } + /** + * Validate a root item. + * @param {RootItemInput} rootItem - The root item to validate. + * @returns {boolean} True if the root item is valid, false otherwise. + * @private + */ _isValid(rootItem) { if (isIdentifier(rootItem) || typeof rootItem === 'function') { return true; @@ -58,3 +81,15 @@ export default class RootRegistry { return false; } } + +/** + * @typedef {Object} RootItemEntry + * @property {number} priority - The priority of the root item. + * @property {() => Promise} provider - A function that returns a promise resolving to a root item or an array of root items. + */ + +/** + * @typedef {import('openmct').Identifier} Identifier + * @typedef {Identifier | Identifier[] | (() => Promise)} RootItemInput + * @typedef {import('openmct').OpenMCT} OpenMCT + */ diff --git a/src/api/objects/Transaction.js b/src/api/objects/Transaction.js index 4a51b20b78..6b41d0f034 100644 --- a/src/api/objects/Transaction.js +++ b/src/api/objects/Transaction.js @@ -20,22 +20,42 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ +/** + * Represents a transaction for managing changes to domain objects. + */ export default class Transaction { + /** + * @param {import('./ObjectAPI').default} objectAPI - The object API instance. + */ constructor(objectAPI) { + /** @type {Record} */ this.dirtyObjects = {}; + /** @type {import('./ObjectAPI').default} */ this.objectAPI = objectAPI; } + /** + * Adds an object to the transaction. + * @param {DomainObject} object - The object to add. + */ add(object) { const key = this.objectAPI.makeKeyString(object.identifier); this.dirtyObjects[key] = object; } + /** + * Cancels the transaction and reverts changes. + * @returns {Promise} + */ cancel() { return this._clear(); } + /** + * Commits the transaction and saves changes. + * @returns {Promise} + */ commit() { const promiseArray = []; const save = this.objectAPI.save.bind(this.objectAPI); @@ -47,6 +67,14 @@ export default class Transaction { return Promise.all(promiseArray); } + /** + * Creates a promise for handling a dirty object. + * @template T + * @param {DomainObject} object - The dirty object. + * @param {(object: DomainObject, ...args: any[]) => Promise} action - The action to perform. + * @param {...any} args - Additional arguments for the action. + * @returns {Promise} + */ createDirtyObjectPromise(object, action, ...args) { return new Promise((resolve, reject) => { action(object, ...args) @@ -60,6 +88,11 @@ export default class Transaction { }); } + /** + * Retrieves a dirty object by its identifier. + * @param {Identifier} identifier - The object identifier. + * @returns {DomainObject | undefined} + */ getDirtyObject(identifier) { let dirtyObject; @@ -73,6 +106,11 @@ export default class Transaction { return dirtyObject; } + /** + * Clears the transaction and refreshes objects. + * @returns {Promise} + * @private + */ _clear() { const promiseArray = []; const action = (obj) => this.objectAPI.refresh(obj, true); @@ -84,3 +122,8 @@ export default class Transaction { return Promise.all(promiseArray); } } + +/** + * @typedef {import('openmct').DomainObject} DomainObject + * @typedef {import('openmct').Identifier} Identifier + */ diff --git a/src/api/overlays/Overlay.js b/src/api/overlays/Overlay.js index 7b81f7ddbb..4d506494f4 100644 --- a/src/api/overlays/Overlay.js +++ b/src/api/overlays/Overlay.js @@ -1,4 +1,4 @@ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import mount from 'utils/mount'; import OverlayComponent from './components/OverlayComponent.vue'; diff --git a/src/api/overlays/OverlayAPI.js b/src/api/overlays/OverlayAPI.js index 08a0b3f09b..a1f4f8b215 100644 --- a/src/api/overlays/OverlayAPI.js +++ b/src/api/overlays/OverlayAPI.js @@ -29,11 +29,7 @@ import Selection from './Selection.js'; * The OverlayAPI is responsible for pre-pending templates to * the body of the document, which is useful for displaying templates * which need to block the full screen. - * - * @memberof api/overlays - * @constructor */ - class OverlayAPI { constructor() { this.activeOverlays = []; @@ -48,7 +44,9 @@ class OverlayAPI { } /** - * private + * Shows an overlay + * @private + * @param {Overlay} overlay - The overlay to show */ showOverlay(overlay) { if (this.activeOverlays.length) { @@ -72,7 +70,8 @@ class OverlayAPI { } /** - * private + * Dismisses the last overlay + * @private */ dismissLastOverlay() { let lastOverlay = this.activeOverlays[this.activeOverlays.length - 1]; @@ -83,14 +82,6 @@ class OverlayAPI { /** * Creates and displays an overlay with the specified options. - * - * @typedef {Object} OverlayOptions - * @property {HTMLElement} element The DOM Element to be inserted or shown in the overlay. - * @property {'large'|'small'|'fit'} size The preferred size of the overlay. - * @property {Array<{label: string, callback: Function}>} [buttons] Optional array of button objects, each with 'label' and 'callback' properties. - * @property {Function} onDestroy Callback to be called when the overlay is destroyed. - * @property {boolean} [dismissible=true] Whether the overlay can be dismissed by pressing 'esc' or clicking outside of it. Defaults to true. - * * @param {OverlayOptions} options - The configuration options for the overlay. * @returns {Overlay} An instance of the Overlay class. */ @@ -104,23 +95,9 @@ class OverlayAPI { /** * Displays a blocking (modal) dialog. This dialog can be used for - * displaying messages that require the user's - * immediate attention. - * @param {model} options defines options for the dialog - * @returns {Object} with an object with a dismiss function that can be called from the calling code - * to dismiss/destroy the dialog - * - * A description of the model options that may be passed to the - * dialog method. Note that the DialogModel described - * here is shared with the Notifications framework. - * @see NotificationService - * - * @typedef options - * @property {string} title the title to use for the dialog - * @property {string} iconClass class to apply to icon that is shown on dialog - * @property {string} message text that indicates a current message, - * @property {buttons[]} buttons a list of buttons with title and callback properties that will - * be added to the dialog. + * displaying messages that require the user's immediate attention. + * @param {DialogOptions} options - Defines options for the dialog + * @returns {Dialog} An object with a dismiss function that can be called from the calling code to dismiss/destroy the dialog */ dialog(options) { let dialog = new Dialog(options); @@ -133,21 +110,10 @@ class OverlayAPI { /** * Displays a blocking (modal) progress dialog. This dialog can be used for * displaying messages that require the user's attention, and show progress - * @param {model} options defines options for the dialog - * @returns {Object} with an object with a dismiss function that can be called from the calling code + * @param {ProgressDialogOptions} options - Defines options for the dialog + * @returns {ProgressDialog} An object with a dismiss function that can be called from the calling code * to dismiss/destroy the dialog and an updateProgress function that takes progressPercentage(Number 0-100) * and progressText (string) - * - * A description of the model options that may be passed to the - * dialog method. Note that the DialogModel described - * here is shared with the Notifications framework. - * @see NotificationService - * - * @typedef options - * @property {number | null} progressPerc the initial progress value (0-100) or null for anonymous progress - * @property {string} progressText the initial text to be shown under the progress bar - * @property {buttons[]} buttons a list of buttons with title and callback properties that will - * be added to the dialog. */ progressDialog(options) { let progressDialog = new ProgressDialog(options); @@ -157,6 +123,11 @@ class OverlayAPI { return progressDialog; } + /** + * Creates and displays a selection overlay + * @param {SelectionOptions} options - The options for the selection overlay + * @returns {Selection} The created Selection instance + */ selection(options) { let selection = new Selection(options); this.showOverlay(selection); @@ -166,3 +137,32 @@ class OverlayAPI { } export default OverlayAPI; + +/** + * @typedef {Object} OverlayOptions + * @property {HTMLElement} element - The DOM Element to be inserted or shown in the overlay. + * @property {'large'|'small'|'fit'} size - The preferred size of the overlay. + * @property {Array<{label: string, callback: Function}>} [buttons] - Optional array of button objects, each with 'label' and 'callback' properties. + * @property {Function} onDestroy - Callback to be called when the overlay is destroyed. + * @property {boolean} [dismissible=true] - Whether the overlay can be dismissed by pressing 'esc' or clicking outside of it. Defaults to true. + */ + +/** + * @typedef {Object} DialogOptions + * @property {string} title - The title to use for the dialog + * @property {string} iconClass - Class to apply to icon that is shown on dialog + * @property {string} message - Text that indicates a current message + * @property {Array<{label: string, callback: Function}>} buttons - A list of buttons with label and callback properties that will be added to the dialog. + */ + +/** + * @typedef {Object} ProgressDialogOptions + * @property {number | null} progressPerc - The initial progress value (0-100) or null for anonymous progress + * @property {string} progressText - The initial text to be shown under the progress bar + * @property {Array<{label: string, callback: Function}>} buttons - A list of buttons with label and callback properties that will be added to the dialog. + */ + +/** + * @typedef {Object} SelectionOptions + * @property {any} options - The options for the selection overlay + */ diff --git a/src/api/status/StatusAPI.js b/src/api/status/StatusAPI.js index ed4968d470..3a7c482033 100644 --- a/src/api/status/StatusAPI.js +++ b/src/api/status/StatusAPI.js @@ -20,13 +20,29 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +/** + * @typedef {import('openmct').OpenMCT} OpenMCT + * @typedef {import('openmct').Identifier} Identifier + * @typedef {string} Status + */ +import { EventEmitter } from 'eventemitter3'; +/** + * Get, set, and observe statuses for Open MCT objects. A status is a string + * that represents the current state of an object. + * + * @extends EventEmitter + */ export default class StatusAPI extends EventEmitter { + /** + * Constructs a new instance of the StatusAPI class. + * @param {OpenMCT} openmct - The Open MCT application instance. + */ constructor(openmct) { super(); this._openmct = openmct; + /** @type {Record} */ this._statusCache = {}; this.get = this.get.bind(this); @@ -34,19 +50,33 @@ export default class StatusAPI extends EventEmitter { this.observe = this.observe.bind(this); } + /** + * Retrieves the status of the object with the given identifier. + * @param {Identifier} identifier - The identifier of the object. + * @returns {Status | undefined} The status of the object, or undefined if the object's status is not cached. + */ get(identifier) { let keyString = this._openmct.objects.makeKeyString(identifier); return this._statusCache[keyString]; } - set(identifier, value) { + /** + * Sets the status of the object with the given identifier. + * @param {Identifier} identifier - The identifier of the object. + * @param {Status} status - The new status value for the object. + */ + set(identifier, status) { let keyString = this._openmct.objects.makeKeyString(identifier); - this._statusCache[keyString] = value; - this.emit(keyString, value); + this._statusCache[keyString] = status; + this.emit(keyString, status); } + /** + * Deletes the status of the object with the given identifier. + * @param {Identifier} identifier - The identifier of the object. + */ delete(identifier) { let keyString = this._openmct.objects.makeKeyString(identifier); @@ -55,6 +85,13 @@ export default class StatusAPI extends EventEmitter { delete this._statusCache[keyString]; } + /** + * Observes the status of the object with the given identifier, and calls the provided callback + * function whenever the status changes. + * @param {Identifier} identifier - The identifier of the object. + * @param {(value: any) => void} callback - The function to be called whenever the status changes. + * @returns {() => void} A function that can be called to stop observing the status. + */ observe(identifier, callback) { let key = this._openmct.objects.makeKeyString(identifier); diff --git a/src/api/telemetry/BatchingWebSocket.js b/src/api/telemetry/BatchingWebSocket.js index 992a7d3ca7..a0687751b4 100644 --- a/src/api/telemetry/BatchingWebSocket.js +++ b/src/api/telemetry/BatchingWebSocket.js @@ -48,7 +48,6 @@ import installWorker from './WebSocketWorker.js'; * concerns are not handled on the main event loop. This allows for performant receipt * and batching of messages without blocking either the UI or server. * - * @memberof module:openmct.telemetry */ // Shim for Internet Explorer, I mean Safari. It doesn't support requestIdleCallback, but it's in a tech preview, so it will be dropping soon. const requestIdleCallback = diff --git a/src/api/telemetry/TelemetryAPI.js b/src/api/telemetry/TelemetryAPI.js index 6f2f0eae74..c9c83b195b 100644 --- a/src/api/telemetry/TelemetryAPI.js +++ b/src/api/telemetry/TelemetryAPI.js @@ -52,7 +52,6 @@ import TelemetryValueFormatter from './TelemetryValueFormatter.js'; * to cancel a telemetry request * @property {string} [domain] the domain key of the request * @property {TimeContext} [timeContext] the time context to use for this request - * @memberof module:openmct.TelemetryAPI~ */ /** @@ -69,7 +68,6 @@ import TelemetryValueFormatter from './TelemetryValueFormatter.js'; * rendering a telemetry plot or table. If `batch` is specified, the subscription * callback will be invoked with an Array. * - * @memberof module:openmct.TelemetryAPI~ */ const SUBSCRIBE_STRATEGY = { @@ -80,7 +78,6 @@ const SUBSCRIBE_STRATEGY = { /** * Utilities for telemetry * @interface TelemetryAPI - * @memberof module:openmct */ export default class TelemetryAPI { #isGreedyLAD; @@ -133,7 +130,7 @@ export default class TelemetryAPI { * object is any object which has telemetry metadata-- regardless of whether * the telemetry object has an available telemetry provider. * - * @param {module:openmct.DomainObject} domainObject + * @param {import('openmct').DomainObject} domainObject * @returns {boolean} true if the object is a telemetry object. */ isTelemetryObject(domainObject) { @@ -145,10 +142,9 @@ export default class TelemetryAPI { * this domain object. * * @method canProvideTelemetry - * @param {module:openmct.DomainObject} domainObject the object for + * @param {import('openmct').DomainObject} domainObject the object for * which telemetry would be provided * @returns {boolean} true if telemetry can be provided - * @memberof module:openmct.TelemetryAPI~TelemetryProvider# */ canProvideTelemetry(domainObject) { return ( @@ -161,7 +157,6 @@ export default class TelemetryAPI { * Register a telemetry provider with the telemetry service. This * allows you to connect alternative telemetry sources. * @method addProvider - * @memberof module:openmct.TelemetryAPI# * @param {module:openmct.TelemetryAPI~TelemetryProvider} provider the new * telemetry provider */ @@ -268,7 +263,6 @@ export default class TelemetryAPI { * * @param {module:openmct.RequestInterceptorDef} requestInterceptorDef the request interceptor definition to add * @method addRequestInterceptor - * @memberof module:openmct.TelemetryRequestInterceptorRegistry# */ addRequestInterceptor(requestInterceptorDef) { this.requestInterceptorRegistry.addInterceptor(requestInterceptorDef); @@ -312,7 +306,6 @@ export default class TelemetryAPI { * * @method greedyLAD * @returns {boolean} if greedyLAD is active or not - * @memberof module:openmct.TelemetryAPI# */ greedyLAD(isGreedy) { if (arguments.length > 0) { @@ -333,8 +326,7 @@ export default class TelemetryAPI { * telemetry (aggregation, latest available, etc.). * * @method requestCollection - * @memberof module:openmct.TelemetryAPI~TelemetryProvider# - * @param {module:openmct.DomainObject} domainObject the object + * @param {import('openmct').DomainObject} domainObject the object * which has associated telemetry * @param {TelemetryRequestOptions} options * options for this telemetry collection request @@ -351,8 +343,7 @@ export default class TelemetryAPI { * telemetry (aggregation, latest available, etc.). * * @method request - * @memberof module:openmct.TelemetryAPI~TelemetryProvider# - * @param {module:openmct.DomainObject} domainObject the object + * @param {import('openmct').DomainObject} domainObject the object * which has associated telemetry * @param {TelemetryRequestOptions} options * options for this historical request @@ -410,8 +401,7 @@ export default class TelemetryAPI { * realtime provider. * * @method subscribe - * @memberof module:openmct.TelemetryAPI~TelemetryProvider# - * @param {module:openmct.DomainObject} domainObject the object + * @param {import('openmct').DomainObject} domainObject the object * which has associated telemetry * @param {TelemetrySubscriptionOptions} options configuration items for subscription * @param {Function} callback the callback to invoke with new data, as @@ -527,8 +517,7 @@ export default class TelemetryAPI { * The callback will be called whenever staleness changes. * * @method subscribeToStaleness - * @memberof module:openmct.TelemetryAPI~StalenessProvider# - * @param {module:openmct.DomainObject} domainObject the object + * @param {import('openmct').DomainObject} domainObject the object * to watch for staleness updates * @param {Function} callback the callback to invoke with staleness data, * as it is received: ex. @@ -586,8 +575,7 @@ export default class TelemetryAPI { * limit provider. * * @method subscribeToLimits - * @memberof module:openmct.TelemetryAPI~TelemetryProvider# - * @param {module:openmct.DomainObject} domainObject the object + * @param {import('openmct').DomainObject} domainObject the object * which has associated limits * @param {Function} callback the callback to invoke with new data, as * it becomes available @@ -640,8 +628,7 @@ export default class TelemetryAPI { * Request telemetry staleness for a domain object. * * @method isStale - * @memberof module:openmct.TelemetryAPI~StalenessProvider# - * @param {module:openmct.DomainObject} domainObject the object + * @param {import('openmct').DomainObject} domainObject the object * which has associated telemetry staleness * @returns {Promise.} a promise for a StalenessResponseObject * or undefined if no provider exists @@ -727,7 +714,7 @@ export default class TelemetryAPI { * Get a format map of all value formatters for a given piece of telemetry * metadata. * - * @returns {Object} + * @returns {Record} */ getFormatMap(metadata) { if (!metadata) { @@ -801,11 +788,10 @@ export default class TelemetryAPI { * If a provider does not implement this method, it is presumed * that no limits are defined for this domain object's telemetry. * - * @param {module:openmct.DomainObject} domainObject the domain + * @param {import('openmct').DomainObject} domainObject the domain * object for which to evaluate limits * @returns {module:openmct.TelemetryAPI~LimitEvaluator} * @method limitEvaluator - * @memberof module:openmct.TelemetryAPI~TelemetryProvider# */ limitEvaluator(domainObject) { return this.getLimitEvaluator(domainObject); @@ -821,11 +807,10 @@ export default class TelemetryAPI { * If a provider does not implement this method, it is presumed * that no limits are defined for this domain object's telemetry. * - * @param {module:openmct.DomainObject} domainObject the domain + * @param {import('openmct').DomainObject} domainObject the domain * object for which to get limits * @returns {LimitsResponseObject} * @method limits - * @memberof module:openmct.TelemetryAPI~TelemetryProvider# */ limitDefinition(domainObject) { return this.getLimits(domainObject); @@ -841,11 +826,10 @@ export default class TelemetryAPI { * If a provider does not implement this method, it is presumed * that no limits are defined for this domain object's telemetry. * - * @param {module:openmct.DomainObject} domainObject the domain + * @param {import('openmct').DomainObject} domainObject the domain * object for which to evaluate limits * @returns {module:openmct.TelemetryAPI~LimitEvaluator} * @method limitEvaluator - * @memberof module:openmct.TelemetryAPI~TelemetryProvider# */ getLimitEvaluator(domainObject) { const provider = this.#findLimitEvaluator(domainObject); @@ -868,12 +852,11 @@ export default class TelemetryAPI { * If a provider does not implement this method, it is presumed * that no limits are defined for this domain object's telemetry. * - * @param {module:openmct.DomainObject} domainObject the domain + * @param {import('openmct').DomainObject} domainObject the domain * object for which to display limits * @returns {LimitsResponseObject} * @method limits returns a limits object of type {LimitsResponseObject} * supported colors are purple, red, orange, yellow and cyan - * @memberof module:openmct.TelemetryAPI~TelemetryProvider# */ getLimits(domainObject) { const provider = this.#findLimitEvaluator(domainObject); @@ -911,7 +894,6 @@ export default class TelemetryAPI { * have exceeded nominal conditions. * * @interface LimitEvaluator - * @memberof module:openmct.TelemetryAPI~ */ /** @@ -919,7 +901,6 @@ export default class TelemetryAPI { * @method evaluate * @param {*} datum the telemetry datum to evaluate * @param {TelemetryProperty} the property to check for limit violations - * @memberof module:openmct.TelemetryAPI~LimitEvaluator * @returns {LimitViolation} metadata about * the limit violation, or undefined if a value is within limits */ @@ -927,7 +908,6 @@ export default class TelemetryAPI { /** * A violation of limits defined for a telemetry property. * @typedef LimitViolation - * @memberof {module:openmct.TelemetryAPI~} * @property {string} cssClass the class (or space-separated classes) to * apply to display elements for values which violate this limit * @property {string} name the human-readable name for the limit violation @@ -937,7 +917,6 @@ export default class TelemetryAPI { /** * @typedef {Object} LimitsResponseObject - * @memberof {module:openmct.TelemetryAPI~} * @property {LimitDefinition} limitLevel the level name and it's limit definition * @example { * [limitLevel]: { @@ -956,7 +935,6 @@ export default class TelemetryAPI { /** * Limit defined for a telemetry property. * @typedef LimitDefinition - * @memberof {module:openmct.TelemetryAPI~} * @property {LimitDefinitionValue} low a lower limit * @property {LimitDefinitionValue} high a higher limit */ @@ -964,7 +942,6 @@ export default class TelemetryAPI { /** * Limit definition for a Limit of a telemetry property. * @typedef LimitDefinitionValue - * @memberof {module:openmct.TelemetryAPI~} * @property {string} color color to represent this limit * @property {number} value the limit value */ @@ -974,7 +951,6 @@ export default class TelemetryAPI { * display as text. * * @interface TelemetryFormatter - * @memberof module:openmct.TelemetryAPI~ */ /** @@ -982,7 +958,6 @@ export default class TelemetryAPI { * telemetry metadata in domain object. * * @method format - * @memberof module:openmct.TelemetryAPI~TelemetryFormatter# */ /** @@ -990,7 +965,6 @@ export default class TelemetryAPI { * associated with a particular domain object. * * @typedef TelemetryProperty - * @memberof module:openmct.TelemetryAPI~ * @property {string} key the name of the property in the datum which * contains this telemetry value * @property {string} name the human-readable name for this property @@ -1010,7 +984,6 @@ export default class TelemetryAPI { * Describes and bounds requests for telemetry data. * * @typedef TelemetryRequest - * @memberof module:openmct.TelemetryAPI~ * @property {string} sort the key of the property to sort by. This may * be prefixed with a "+" or a "-" sign to sort in ascending * or descending order respectively. If no prefix is present, @@ -1029,7 +1002,6 @@ export default class TelemetryAPI { * [registered]{@link module:openmct.TelemetryAPI#addProvider}. * * @interface TelemetryProvider - * @memberof module:openmct.TelemetryAPI~ */ /** @@ -1046,7 +1018,6 @@ export default class TelemetryAPI { * and an options object which currently has an abort signal, ex. * { signal: } * this method should return a current StalenessResponseObject - * @memberof module:openmct.TelemetryAPI~ */ /** @@ -1061,5 +1032,4 @@ export default class TelemetryAPI { * * @interface TelemetryAPI * @augments module:openmct.TelemetryAPI~TelemetryProvider - * @memberof module:openmct */ diff --git a/src/api/telemetry/TelemetryCollection.js b/src/api/telemetry/TelemetryCollection.js index eb2462aa36..78e48dc89b 100644 --- a/src/api/telemetry/TelemetryCollection.js +++ b/src/api/telemetry/TelemetryCollection.js @@ -20,14 +20,14 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; import { TIME_CONTEXT_EVENTS } from '../time/constants.js'; import { LOADED_ERROR, TIMESYSTEM_KEY_NOTIFICATION, TIMESYSTEM_KEY_WARNING } from './constants.js'; /** - * @typedef {import('../objects/ObjectAPI').DomainObject} DomainObject + * @typedef {import('openmct').DomainObject} DomainObject */ /** diff --git a/src/api/telemetry/TelemetryRequestInterceptor.js b/src/api/telemetry/TelemetryRequestInterceptor.js index b80ea1f0af..6a6e42a754 100644 --- a/src/api/telemetry/TelemetryRequestInterceptor.js +++ b/src/api/telemetry/TelemetryRequestInterceptor.js @@ -25,7 +25,6 @@ export default class TelemetryRequestInterceptorRegistry { * A TelemetryRequestInterceptorRegistry maintains the definitions for different interceptors that may be invoked on telemetry * requests. * @interface TelemetryRequestInterceptorRegistry - * @memberof module:openmct */ constructor() { this.interceptors = []; @@ -36,7 +35,6 @@ export default class TelemetryRequestInterceptorRegistry { * @property {function} appliesTo function that determines if this interceptor should be called for the given identifier/request * @property {function} invoke function that transforms the provided request and returns the transformed request * @property {function} priority the priority for this interceptor. A higher number returned has more weight than a lower number - * @memberof module:openmct TelemetryRequestInterceptorRegistry# */ /** @@ -44,7 +42,6 @@ export default class TelemetryRequestInterceptorRegistry { * * @param {module:openmct.RequestInterceptorDef} requestInterceptorDef the interceptor to add * @method addInterceptor - * @memberof module:openmct.TelemetryRequestInterceptorRegistry# */ addInterceptor(interceptorDef) { //TODO: sort by priority @@ -55,7 +52,6 @@ export default class TelemetryRequestInterceptorRegistry { * Retrieve all interceptors applicable to a domain object/request. * @method getInterceptors * @returns [module:openmct.RequestInterceptorDef] the registered interceptors for this identifier/request - * @memberof module:openmct.TelemetryRequestInterceptorRegistry# */ getInterceptors(identifier, request) { return this.interceptors.filter((interceptor) => { diff --git a/src/api/time/IndependentTimeContext.js b/src/api/time/IndependentTimeContext.js index 5f5b3b139b..9f9edbcc4d 100644 --- a/src/api/time/IndependentTimeContext.js +++ b/src/api/time/IndependentTimeContext.js @@ -165,7 +165,6 @@ class IndependentTimeContext extends TimeContext { /** * Get the time system of the TimeAPI. * @returns {TimeSystem} The currently applied time system - * @memberof module:openmct.TimeAPI# * @method getTimeSystem * @override */ @@ -214,7 +213,6 @@ class IndependentTimeContext extends TimeContext { /** * The active clock has changed. * @event clock - * @memberof module:openmct.TimeAPI~ * @property {Clock} clock The newly activated clock, or undefined * if the system is no longer following a clock source */ @@ -286,7 +284,6 @@ class IndependentTimeContext extends TimeContext { /** * The active clock has changed. * @event clock - * @memberof module:openmct.TimeAPI~ * @property {Clock} clock The newly activated clock, or undefined * if the system is no longer following a clock source */ @@ -333,7 +330,6 @@ class IndependentTimeContext extends TimeContext { /** * The active mode has changed. * @event modeChanged - * @memberof module:openmct.TimeAPI~ * @property {Mode} mode The newly activated mode */ this.emit(TIME_CONTEXT_EVENTS.modeChanged, this.#copy(this.mode)); diff --git a/src/api/time/TimeAPI.js b/src/api/time/TimeAPI.js index b85bafc1ca..3cc2d8b6e1 100644 --- a/src/api/time/TimeAPI.js +++ b/src/api/time/TimeAPI.js @@ -107,7 +107,6 @@ class TimeAPI extends GlobalTimeContext { * automatically update the time bounds of the data displayed in Open MCT. * * @typedef {Object} Clock - * @memberof openmct.timeAPI * @property {string} key A unique identifier * @property {string} name A human-readable name. The name will be used to * represent this clock in the Time Conductor UI diff --git a/src/api/time/TimeContext.js b/src/api/time/TimeContext.js index c4583e47a2..6bdd6ec204 100644 --- a/src/api/time/TimeContext.js +++ b/src/api/time/TimeContext.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'eventemitter3'; +import { EventEmitter } from 'eventemitter3'; import { FIXED_MODE_KEY, MODES, REALTIME_MODE_KEY, TIME_CONTEXT_EVENTS } from './constants.js'; @@ -34,7 +34,7 @@ import { FIXED_MODE_KEY, MODES, REALTIME_MODE_KEY, TIME_CONTEXT_EVENTS } from '. /** * @typedef {Object} TimeConductorBounds - * @property {number} start The start time displayed by the time conductor + * @property {number } start The start time displayed by the time conductor * in ms since epoch. Epoch determined by currently active time system * @property {number} end The end time displayed by the time conductor in ms * since epoch. @@ -301,7 +301,6 @@ class TimeContext extends EventEmitter { /** * Event that is triggered when clock offsets change. * @event clockOffsets - * @memberof module:openmct.TimeAPI~ * @property {ClockOffsets} clockOffsets The newly activated clock * offsets. */ @@ -426,8 +425,8 @@ class TimeContext extends EventEmitter { /** * Set the time system of the TimeAPI. * Emits a "timeSystem" event with the new time system. - * @param {TimeSystem | string} timeSystemOrKey The time system to set, or its key - * @param {TimeConductorBounds} [bounds] Optional bounds to set + * @param {TimeSystem | string} timeSystemOrKey + * @param {TimeConductorBounds} bounds */ setTimeSystem(timeSystemOrKey, bounds) { if (timeSystemOrKey === undefined) { @@ -546,7 +545,6 @@ class TimeContext extends EventEmitter { /** * The active clock has changed. * @event clock - * @memberof module:openmct.TimeAPI~ * @property {TimeContext} clock The newly activated clock, or undefined * if the system is no longer following a clock source */ diff --git a/src/api/tooltips/ToolTip.js b/src/api/tooltips/ToolTip.js index 5162d21c39..fa61a207f9 100644 --- a/src/api/tooltips/ToolTip.js +++ b/src/api/tooltips/ToolTip.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import mount from 'utils/mount'; import TooltipComponent from './components/TooltipComponent.vue'; diff --git a/src/api/tooltips/ToolTipAPI.js b/src/api/tooltips/ToolTipAPI.js index 66b857e006..14ce9f6740 100644 --- a/src/api/tooltips/ToolTipAPI.js +++ b/src/api/tooltips/ToolTipAPI.js @@ -43,7 +43,6 @@ const TOOLTIP_LOCATIONS = Object.freeze({ * The TooltipAPI is responsible for adding custom tooltips to * the desired elements on the screen * - * @memberof api/tooltips * @constructor */ diff --git a/src/api/types/Type.js b/src/api/types/Type.js index 4f74d2bbb9..d530186167 100644 --- a/src/api/types/Type.js +++ b/src/api/types/Type.js @@ -23,12 +23,11 @@ /** * A Type describes a kind of domain object that may appear or be * created within Open MCT. - * - * @param {module:openmct.TypeRegistry~TypeDefinition} definition - * @class Type - * @memberof module:openmct */ export default class Type { + /** + * @param {TypeDefinition} definition + */ constructor(definition) { this.definition = definition; if (definition.key) { @@ -36,7 +35,9 @@ export default class Type { } } /** - * Create a type definition from a legacy definition. + * Convert a legacy type definition to the new format. + * @param {LegacyTypeDefinition} legacyDefinition + * @returns {TypeDefinition} */ static definitionFromLegacyDefinition(legacyDefinition) { let definition = {}; @@ -86,10 +87,8 @@ export default class Type { } /** * Check if a domain object is an instance of this type. - * @param domainObject + * @param {DomainObject} domainObject * @returns {boolean} true if the domain object is of this type - * @memberof module:openmct.Type# - * @method check */ check(domainObject) { // Depends on assignment from MCT. @@ -119,3 +118,19 @@ export default class Type { return def; } } + +/** + * @typedef {Object} TypeDefinition + * @property {string} [key] + * @property {string} name + * @property {string} cssClass + * @property {string} description + * @property {Form} form + * @property {Telemetry} telemetry + * @property {function(Object): void} initialize + * @property {boolean} creatable + */ + +/** + * @typedef {import('openmct').DomainObject} DomainObject + */ diff --git a/src/api/types/TypeRegistry.js b/src/api/types/TypeRegistry.js index 7a3786e00b..9506992edb 100644 --- a/src/api/types/TypeRegistry.js +++ b/src/api/types/TypeRegistry.js @@ -29,12 +29,11 @@ const UNKNOWN_TYPE = new Type({ /** * @typedef TypeDefinition - * @memberof module:openmct.TypeRegistry~ * @property {string} label the name for this type of object * @property {string} description a longer-form description of this type - * @property {function (object)} [initialize] a function which initializes + * @property {function(domainObject:DomainObject): void} [initialize] a function which initializes * the model for new domain objects of this type - * @property {boolean} [creatable] true if users should be allowed to + * @property {boolean} [creatable=false] true if users should be allowed to * create this type (default: false) * @property {string} [cssClass] the CSS class to apply for icons */ @@ -43,19 +42,19 @@ const UNKNOWN_TYPE = new Type({ * A TypeRegistry maintains the definitions for different types * that domain objects may have. * @interface TypeRegistry - * @memberof module:openmct */ export default class TypeRegistry { constructor() { + /** + * @type {Record} + */ this.types = {}; } /** * Register a new object type. * * @param {string} typeKey a string identifier for this type - * @param {module:openmct.Type} type the type to add - * @method addType - * @memberof module:openmct.TypeRegistry# + * @param {TypeDefinition} typeDef the type to add */ addType(typeKey, typeDef) { this.standardizeType(typeDef); @@ -77,8 +76,6 @@ export default class TypeRegistry { } /** * List keys for all registered types. - * @method listKeys - * @memberof module:openmct.TypeRegistry# * @returns {string[]} all registered type keys */ listKeys() { @@ -86,10 +83,8 @@ export default class TypeRegistry { } /** * Retrieve a registered type by its key. - * @method get * @param {string} typeKey the key for this type - * @memberof module:openmct.TypeRegistry# - * @returns {module:openmct.Type} the registered type + * @returns {Type} the registered type */ get(typeKey) { return this.types[typeKey] || UNKNOWN_TYPE; diff --git a/src/api/user/StatusAPI.js b/src/api/user/StatusAPI.js index befc60a41a..326d690e2c 100644 --- a/src/api/user/StatusAPI.js +++ b/src/api/user/StatusAPI.js @@ -19,7 +19,7 @@ * this source code distribution or the Licensing information page available * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; /** * The StatusAPI is used to get and set various statuses linked to the current logged in user. diff --git a/src/api/user/UserAPI.js b/src/api/user/UserAPI.js index 27849c57be..1de45c1b3e 100644 --- a/src/api/user/UserAPI.js +++ b/src/api/user/UserAPI.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { MULTIPLE_PROVIDER_ERROR, NO_PROVIDER_ERROR } from './constants.js'; import StatusAPI from './StatusAPI.js'; @@ -28,27 +28,27 @@ import StoragePersistence from './StoragePersistence.js'; import User from './User.js'; class UserAPI extends EventEmitter { - /** @type {OpenMCT} */ + /** + * @type {OpenMCT} + */ #openmct; /** * @param {OpenMCT} openmct - * @param {UserAPIConfiguration} config */ - constructor(openmct, config) { + constructor(openmct) { super(); this.#openmct = openmct; this._provider = undefined; this.User = User; - this.status = new StatusAPI(this, openmct, config); + this.status = new StatusAPI(this, openmct); } /** * Set the user provider for the user API. This allows you * to specify ONE user provider to be used with Open MCT. * @method setProvider - * @memberof module:openmct.UserAPI# * @param {module:openmct.UserAPI~UserProvider} provider the new * user provider */ @@ -68,7 +68,6 @@ class UserAPI extends EventEmitter { /** * Return true if the user provider has been set. * - * @memberof module:openmct.UserAPI# * @returns {boolean} true if the user provider exists */ hasProvider() { @@ -79,7 +78,6 @@ class UserAPI extends EventEmitter { * If a user provider is set, it will return a copy of a user object from * the provider. If the user is not logged in, it will return undefined; * - * @memberof module:openmct.UserAPI# * @returns {Function|Promise} user provider 'getCurrentUser' method * @throws Will throw an error if no user provider is set */ @@ -93,7 +91,6 @@ class UserAPI extends EventEmitter { /** * If a user provider is set, it will return an array of possible roles * that can be selected by the current user - * @memberof module:openmct.UserAPI# * @returns {Array} * @throws Will throw an error if no user provider is set */ @@ -106,7 +103,6 @@ class UserAPI extends EventEmitter { } /** * If a user provider is set, it will return the active role or null - * @memberof module:openmct.UserAPI# * @returns {string|null} */ getActiveRole() { @@ -121,7 +117,6 @@ class UserAPI extends EventEmitter { } /** * Set the active role in session storage - * @memberof module:openmct.UserAPI# * @returns {undefined} */ setActiveRole(role) { @@ -135,7 +130,6 @@ class UserAPI extends EventEmitter { /** * Will return if a role can provide a operator status response - * @memberof module:openmct.UserApi# * @returns {boolean} */ canProvideStatusForRole() { @@ -151,7 +145,6 @@ class UserAPI extends EventEmitter { * If a user provider is set, it will return the user provider's * 'isLoggedIn' method * - * @memberof module:openmct.UserAPI# * @returns {Function|Boolean} user provider 'isLoggedIn' method * @throws Will throw an error if no user provider is set */ @@ -167,7 +160,6 @@ class UserAPI extends EventEmitter { * If a user provider is set, it will return a call to it's * 'hasRole' method * - * @memberof module:openmct.UserAPI# * @returns {Function|boolean} user provider 'isLoggedIn' method * @param {string} roleId id of role to check for * @throws Will throw an error if no user provider is set @@ -206,14 +198,9 @@ export default UserAPI; /** * @typedef {string} Role - */ - -/** - * @typedef {import('../../../openmct.js').OpenMCT} OpenMCT - */ - -/** - * @typedef {{statusStyles: Object.}} UserAPIConfiguration + * @typedef {import('../../MCT.js').MCT} OpenMCT + * @typedef {{statusStyles: Record}} UserAPIConfiguration + * @typedef {Object} UserProvider */ /** @@ -224,7 +211,3 @@ export default UserAPI; * @property {string} statusBgColor The background color to apply in the status summary section of the poll question popup for this status eg."#9900cc" * @property {string} statusFgColor The foreground color to apply in the status summary section of the poll question popup for this status eg. "#fff" */ - -/** - * @typedef {Object} UserProvider - */ diff --git a/src/plugins/DeviceClassifier/src/DeviceMatchers.js b/src/plugins/DeviceClassifier/src/DeviceMatchers.js index fe9f100275..335f588a32 100644 --- a/src/plugins/DeviceClassifier/src/DeviceMatchers.js +++ b/src/plugins/DeviceClassifier/src/DeviceMatchers.js @@ -28,7 +28,6 @@ * * For internal use by the mobile support bundle. * - * @memberof src/plugins/DeviceClassifier * @private */ diff --git a/src/plugins/LADTable/LADTableConfiguration.js b/src/plugins/LADTable/LADTableConfiguration.js index b6bb1b2dd4..fac6ba63d9 100644 --- a/src/plugins/LADTable/LADTableConfiguration.js +++ b/src/plugins/LADTable/LADTableConfiguration.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { markRaw } from 'vue'; export default class LADTableConfiguration extends EventEmitter { constructor(domainObject, openmct) { diff --git a/src/plugins/activityStates/activityStatesInterceptor.js b/src/plugins/activityStates/activityStatesInterceptor.js index 94b0a30a0e..e914d59c61 100644 --- a/src/plugins/activityStates/activityStatesInterceptor.js +++ b/src/plugins/activityStates/activityStatesInterceptor.js @@ -24,7 +24,7 @@ import { ACTIVITY_STATES_KEY } from './createActivityStatesIdentifier.js'; /** * @typedef {Object} ActivityStatesInterceptorOptions - * @property {import('../../api/objects/ObjectAPI').Identifier} identifier the {namespace, key} to use for the activity states object. + * @property {import('openmct').Identifier} identifier the {namespace, key} to use for the activity states object. * @property {string} name The name of the activity states model. * @property {number} priority the priority of the interceptor. By default, it is low. */ diff --git a/src/plugins/charts/bar/pluginSpec.js b/src/plugins/charts/bar/pluginSpec.js index e8b809bcfd..bb8db150a4 100644 --- a/src/plugins/charts/bar/pluginSpec.js +++ b/src/plugins/charts/bar/pluginSpec.js @@ -21,7 +21,7 @@ *****************************************************************************/ // import BarGraph from './BarGraphPlot.vue'; -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { createOpenMct, resetApplicationState } from 'utils/testing'; import { nextTick } from 'vue'; diff --git a/src/plugins/charts/scatter/pluginSpec.js b/src/plugins/charts/scatter/pluginSpec.js index 18cdd93b81..0f5d7cd6dd 100644 --- a/src/plugins/charts/scatter/pluginSpec.js +++ b/src/plugins/charts/scatter/pluginSpec.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { createOpenMct, resetApplicationState } from 'utils/testing'; import { nextTick } from 'vue'; diff --git a/src/plugins/clock/pluginSpec.js b/src/plugins/clock/pluginSpec.js index bd70e0b276..542c87dc72 100644 --- a/src/plugins/clock/pluginSpec.js +++ b/src/plugins/clock/pluginSpec.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { createOpenMct, resetApplicationState } from 'utils/testing'; import { nextTick } from 'vue'; diff --git a/src/plugins/condition/Condition.js b/src/plugins/condition/Condition.js index 81e7f22614..a0039b3f1a 100644 --- a/src/plugins/condition/Condition.js +++ b/src/plugins/condition/Condition.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { v4 as uuid } from 'uuid'; import AllTelemetryCriterion from './criterion/AllTelemetryCriterion.js'; diff --git a/src/plugins/condition/ConditionManager.js b/src/plugins/condition/ConditionManager.js index ec48a59417..838730883d 100644 --- a/src/plugins/condition/ConditionManager.js +++ b/src/plugins/condition/ConditionManager.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { v4 as uuid } from 'uuid'; import Condition from './Condition.js'; diff --git a/src/plugins/condition/StyleRuleManager.js b/src/plugins/condition/StyleRuleManager.js index 059175cd60..097dd196f3 100644 --- a/src/plugins/condition/StyleRuleManager.js +++ b/src/plugins/condition/StyleRuleManager.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; export default class StyleRuleManager extends EventEmitter { constructor(styleConfiguration, openmct, callback, suppressSubscriptionOnEdit) { diff --git a/src/plugins/condition/criterion/TelemetryCriterion.js b/src/plugins/condition/criterion/TelemetryCriterion.js index 28e94e3621..ae377f4da5 100644 --- a/src/plugins/condition/criterion/TelemetryCriterion.js +++ b/src/plugins/condition/criterion/TelemetryCriterion.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import StalenessUtils from '@/utils/staleness'; diff --git a/src/plugins/displayLayout/LayoutDrag.js b/src/plugins/displayLayout/LayoutDrag.js index ceeb4355c8..c493ccba25 100644 --- a/src/plugins/displayLayout/LayoutDrag.js +++ b/src/plugins/displayLayout/LayoutDrag.js @@ -45,7 +45,6 @@ * @param {number[]} dimFactor the dimensions factor * @param {number[]} the size of each grid element, in pixels * @constructor - * @memberof platform/features/layout */ export default function LayoutDrag(rawPosition, posFactor, dimFactor, gridSize) { this.rawPosition = rawPosition; diff --git a/src/plugins/exportAsJSONAction/ExportAsJSONAction.js b/src/plugins/exportAsJSONAction/ExportAsJSONAction.js index 93f17b3765..0e283dba68 100644 --- a/src/plugins/exportAsJSONAction/ExportAsJSONAction.js +++ b/src/plugins/exportAsJSONAction/ExportAsJSONAction.js @@ -108,7 +108,7 @@ class ExportAsJSONAction { /** * @private - * @param {import('../../api/objects/ObjectAPI').DomainObject} parent + * @param {import('openmct').DomainObject} parent */ async #write(parent) { this.totalToExport++; diff --git a/src/plugins/flexibleLayout/pluginSpec.js b/src/plugins/flexibleLayout/pluginSpec.js index 4fcacd21c1..719ed0eadc 100644 --- a/src/plugins/flexibleLayout/pluginSpec.js +++ b/src/plugins/flexibleLayout/pluginSpec.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { createOpenMct, resetApplicationState } from 'utils/testing'; import { nextTick } from 'vue'; diff --git a/src/plugins/imagery/lib/eventHelpers.js b/src/plugins/imagery/lib/eventHelpers.js index 9d46dca54d..c4a2166bdb 100644 --- a/src/plugins/imagery/lib/eventHelpers.js +++ b/src/plugins/imagery/lib/eventHelpers.js @@ -20,6 +20,9 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ +/** + * @type {EventHelpers} + */ const helperFunctions = { listenTo: function (object, event, callback, context) { if (!this._listeningTo) { @@ -95,3 +98,8 @@ const helperFunctions = { }; export default helperFunctions; +/** + * @typedef {Object} EventHelpers + * @property {(object: any, event: string, callback: Function, context?: any) => void} listenTo + * @property {(object: any, event?: string, callback?: Function, context?: any) => void} stopListening + */ diff --git a/src/plugins/inspectorViews/styles/StylesManager.js b/src/plugins/inspectorViews/styles/StylesManager.js index b23ea95cf4..f27f87bcf5 100644 --- a/src/plugins/inspectorViews/styles/StylesManager.js +++ b/src/plugins/inspectorViews/styles/StylesManager.js @@ -1,4 +1,4 @@ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; const LOCAL_STORAGE_KEY = 'mct-saved-styles'; const LIMIT = 20; diff --git a/src/plugins/localTimeSystem/LocalTimeFormat.js b/src/plugins/localTimeSystem/LocalTimeFormat.js index 50d2ad4932..c46733d797 100644 --- a/src/plugins/localTimeSystem/LocalTimeFormat.js +++ b/src/plugins/localTimeSystem/LocalTimeFormat.js @@ -38,7 +38,6 @@ const DATE_FORMATS = [DATE_FORMAT, 'YYYY-MM-DD h:mm:ss a', 'YYYY-MM-DD h:mm a', * * @implements {Format} * @constructor - * @memberof platform/commonUI/formats */ export default function LocalTimeFormat() { this.key = 'local-format'; diff --git a/src/plugins/localTimeSystem/pluginSpec.js b/src/plugins/localTimeSystem/pluginSpec.js index af27504d55..33ccbff6cf 100644 --- a/src/plugins/localTimeSystem/pluginSpec.js +++ b/src/plugins/localTimeSystem/pluginSpec.js @@ -61,7 +61,7 @@ describe('The local time', () => { }); it('can be set to be the main time system', () => { - expect(openmct.time.timeSystem().key).toBe(LOCAL_SYSTEM_KEY); + expect(openmct.time.getTimeSystem().key).toBe(LOCAL_SYSTEM_KEY); }); it('uses the local-format time format', () => { diff --git a/src/plugins/notebook/snapshot-container.js b/src/plugins/notebook/snapshot-container.js index e2ca7e531f..62a66b4bc6 100644 --- a/src/plugins/notebook/snapshot-container.js +++ b/src/plugins/notebook/snapshot-container.js @@ -1,4 +1,4 @@ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { EVENT_SNAPSHOTS_UPDATED } from './notebook-constants.js'; const NOTEBOOK_SNAPSHOT_STORAGE = 'notebook-snapshot-storage'; diff --git a/src/plugins/persistence/couch/CouchDocument.js b/src/plugins/persistence/couch/CouchDocument.js index edcdc8567b..b7c51f2ced 100644 --- a/src/plugins/persistence/couch/CouchDocument.js +++ b/src/plugins/persistence/couch/CouchDocument.js @@ -27,7 +27,6 @@ * metadata field which contains a subset of information found * in the model itself (to support search optimization with * CouchDB views.) - * @memberof platform/persistence/couch * @constructor * @param {string} id the id under which to store this mode * @param {Object} model the model to store diff --git a/src/plugins/plan/PlanViewConfiguration.js b/src/plugins/plan/PlanViewConfiguration.js index 5c51efebb8..9ccedb1182 100644 --- a/src/plugins/plan/PlanViewConfiguration.js +++ b/src/plugins/plan/PlanViewConfiguration.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; export const DEFAULT_CONFIGURATION = { clipActivityNames: false, diff --git a/src/plugins/plan/inspector/components/PlanViewConfiguration.vue b/src/plugins/plan/inspector/components/PlanViewConfiguration.vue index ff674027ff..d363caf6c3 100644 --- a/src/plugins/plan/inspector/components/PlanViewConfiguration.vue +++ b/src/plugins/plan/inspector/components/PlanViewConfiguration.vue @@ -77,7 +77,7 @@ export default { inject: ['openmct'], data() { const selection = this.openmct.selection.get(); - /** @type {import('../../../api/objects/ObjectAPI').DomainObject} */ + /** @type {import('openmct').DomainObject} */ const domainObject = selection[0][0].context.item; const planViewConfiguration = markRaw(new PlanViewConfiguration(domainObject, this.openmct)); diff --git a/src/plugins/plot/configuration/Model.js b/src/plugins/plot/configuration/Model.js index fd6476549a..0c8ca27776 100644 --- a/src/plugins/plot/configuration/Model.js +++ b/src/plugins/plot/configuration/Model.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'eventemitter3'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; import eventHelpers from '../lib/eventHelpers.js'; diff --git a/src/plugins/plot/configuration/PlotConfigurationModel.js b/src/plugins/plot/configuration/PlotConfigurationModel.js index ba9ed96a47..73c30e3345 100644 --- a/src/plugins/plot/configuration/PlotConfigurationModel.js +++ b/src/plugins/plot/configuration/PlotConfigurationModel.js @@ -126,7 +126,7 @@ export default class PlotConfigurationModel extends Model { } /** * Retrieve the persisted series config for a given identifier. - * @param {import('./PlotSeries').Identifier} identifier + * @param {import('openmct').Identifier} identifier * @returns {import('./PlotSeries').PlotSeriesModelType=} */ getPersistedSeriesConfig(identifier) { diff --git a/src/plugins/plot/draw/Draw2D.js b/src/plugins/plot/draw/Draw2D.js index 372bdcbefb..a18e6b798a 100644 --- a/src/plugins/plot/draw/Draw2D.js +++ b/src/plugins/plot/draw/Draw2D.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import eventHelpers from '../lib/eventHelpers.js'; import { MARKER_SHAPES } from './MarkerShapes.js'; @@ -138,5 +138,4 @@ class Draw2D extends EventEmitter { } } } - export default Draw2D; diff --git a/src/plugins/plot/draw/DrawWebGL.js b/src/plugins/plot/draw/DrawWebGL.js index 5fd6456a84..17784198f4 100644 --- a/src/plugins/plot/draw/DrawWebGL.js +++ b/src/plugins/plot/draw/DrawWebGL.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import eventHelpers from '../lib/eventHelpers.js'; import { MARKER_SHAPES } from './MarkerShapes.js'; @@ -297,5 +297,4 @@ class DrawWebGL extends EventEmitter { } } } - export default DrawWebGL; diff --git a/src/plugins/plot/lib/eventHelpers.js b/src/plugins/plot/lib/eventHelpers.js index 20bb3838ea..41865d78ca 100644 --- a/src/plugins/plot/lib/eventHelpers.js +++ b/src/plugins/plot/lib/eventHelpers.js @@ -19,8 +19,10 @@ * this source code distribution or the Licensing information page available * at runtime from the About dialog for additional information. *****************************************************************************/ -/*jscs:disable disallowDanglingUnderscores */ +/** + * @type {EventHelpers} + */ const helperFunctions = { listenTo: function (object, event, callback, context) { if (!this._listeningTo) { @@ -94,7 +96,7 @@ export default helperFunctions; /** @typedef {{ - listenTo: (object: any, event: any, callback: any, context: any) => void + listenTo: (object: any, event: any, callback: any, context: any) => void, stopListening: (object: any, event: any, callback: any, context: any) => void }} EventHelpers */ diff --git a/src/plugins/plot/overlayPlot/pluginSpec.js b/src/plugins/plot/overlayPlot/pluginSpec.js index 4c3d46f63f..10f2630433 100644 --- a/src/plugins/plot/overlayPlot/pluginSpec.js +++ b/src/plugins/plot/overlayPlot/pluginSpec.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import mount from 'utils/mount'; import { createMouseEvent, diff --git a/src/plugins/plot/pluginSpec.js b/src/plugins/plot/pluginSpec.js index fedfa8053e..a7e33ce2d6 100644 --- a/src/plugins/plot/pluginSpec.js +++ b/src/plugins/plot/pluginSpec.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import mount from 'utils/mount'; import { createMouseEvent, diff --git a/src/plugins/plot/stackedPlot/pluginSpec.js b/src/plugins/plot/stackedPlot/pluginSpec.js index 7f6d2da2e0..e845588001 100644 --- a/src/plugins/plot/stackedPlot/pluginSpec.js +++ b/src/plugins/plot/stackedPlot/pluginSpec.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import mount from 'utils/mount'; import { createMouseEvent, diff --git a/src/plugins/plugins.js b/src/plugins/plugins.js index 2c2b6809ca..3bb2af20aa 100644 --- a/src/plugins/plugins.js +++ b/src/plugins/plugins.js @@ -87,6 +87,9 @@ import ViewDatumAction from './viewDatumAction/plugin.js'; import ViewLargeAction from './viewLargeAction/plugin.js'; import WebPagePlugin from './webPage/plugin.js'; +/** + * @type {Object} + */ const plugins = {}; plugins.example = {}; diff --git a/src/plugins/remoteClock/RemoteClock.js b/src/plugins/remoteClock/RemoteClock.js index 3421bae789..b97ae8d027 100644 --- a/src/plugins/remoteClock/RemoteClock.js +++ b/src/plugins/remoteClock/RemoteClock.js @@ -149,8 +149,6 @@ export default class RemoteClock extends DefaultClock { /** * Waits for the clock to have a non-default tick value. - * - * @private */ #waitForReady() { const waitForInitialTick = (resolve) => { diff --git a/src/plugins/summaryWidget/src/Condition.js b/src/plugins/summaryWidget/src/Condition.js index cdfd614bb0..5d356ea250 100644 --- a/src/plugins/summaryWidget/src/Condition.js +++ b/src/plugins/summaryWidget/src/Condition.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import * as templateHelpers from '../../../utils/template/templateHelpers.js'; import conditionTemplate from '../res/conditionTemplate.html'; diff --git a/src/plugins/summaryWidget/src/ConditionManager.js b/src/plugins/summaryWidget/src/ConditionManager.js index 493d391a23..3db2907981 100644 --- a/src/plugins/summaryWidget/src/ConditionManager.js +++ b/src/plugins/summaryWidget/src/ConditionManager.js @@ -1,4 +1,4 @@ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; import { makeKeyString } from 'objectUtils'; diff --git a/src/plugins/summaryWidget/src/Rule.js b/src/plugins/summaryWidget/src/Rule.js index 2e3a1c5e0a..cbc6fec546 100644 --- a/src/plugins/summaryWidget/src/Rule.js +++ b/src/plugins/summaryWidget/src/Rule.js @@ -1,4 +1,4 @@ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; import * as templateHelpers from '../../../utils/template/templateHelpers.js'; diff --git a/src/plugins/summaryWidget/src/TestDataItem.js b/src/plugins/summaryWidget/src/TestDataItem.js index 3d2225d310..056b3d28f4 100644 --- a/src/plugins/summaryWidget/src/TestDataItem.js +++ b/src/plugins/summaryWidget/src/TestDataItem.js @@ -1,4 +1,4 @@ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import * as templateHelpers from '../../../utils/template/templateHelpers.js'; import itemTemplate from '../res/testDataItemTemplate.html'; diff --git a/src/plugins/summaryWidget/src/WidgetDnD.js b/src/plugins/summaryWidget/src/WidgetDnD.js index d83c2c60e1..c28d4647c1 100644 --- a/src/plugins/summaryWidget/src/WidgetDnD.js +++ b/src/plugins/summaryWidget/src/WidgetDnD.js @@ -1,4 +1,4 @@ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import * as templateHelpers from '../../../utils/template/templateHelpers.js'; import ruleImageTemplate from '../res/ruleImageTemplate.html'; diff --git a/src/plugins/summaryWidget/src/input/Palette.js b/src/plugins/summaryWidget/src/input/Palette.js index d8c3723097..6cfe58abb4 100644 --- a/src/plugins/summaryWidget/src/input/Palette.js +++ b/src/plugins/summaryWidget/src/input/Palette.js @@ -1,4 +1,4 @@ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import * as templateHelpers from '../../../../utils/template/templateHelpers.js'; import paletteTemplate from '../../res/input/paletteTemplate.html'; diff --git a/src/plugins/summaryWidget/src/input/Select.js b/src/plugins/summaryWidget/src/input/Select.js index 4d1d790014..c46ab7f97c 100644 --- a/src/plugins/summaryWidget/src/input/Select.js +++ b/src/plugins/summaryWidget/src/input/Select.js @@ -1,4 +1,4 @@ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import * as templateHelpers from '../../../../utils/template/templateHelpers.js'; import selectTemplate from '../../res/input/selectTemplate.html'; diff --git a/src/plugins/tabs/pluginSpec.js b/src/plugins/tabs/pluginSpec.js index e7c6a100fb..cc95bfebea 100644 --- a/src/plugins/tabs/pluginSpec.js +++ b/src/plugins/tabs/pluginSpec.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { createOpenMct, resetApplicationState } from 'utils/testing'; import { nextTick } from 'vue'; diff --git a/src/plugins/telemetryMean/src/MeanTelemetryProviderSpec.js b/src/plugins/telemetryMean/src/MeanTelemetryProviderSpec.js index a69f6b2ca2..6e3936b049 100644 --- a/src/plugins/telemetryMean/src/MeanTelemetryProviderSpec.js +++ b/src/plugins/telemetryMean/src/MeanTelemetryProviderSpec.js @@ -611,7 +611,7 @@ describe('The Mean Telemetry Provider', function () { } function createMockTimeApi() { - return jasmine.createSpyObj('timeApi', ['getTimeSystem']); + return jasmine.createSpyObj('timeApi', ['getTimeSystem', 'setTimeSystem']); } function setTimeSystemTo(timeSystemKey) { diff --git a/src/plugins/telemetryTable/TelemetryTable.js b/src/plugins/telemetryTable/TelemetryTable.js index 766b4e9e57..dc6ad39dbe 100644 --- a/src/plugins/telemetryTable/TelemetryTable.js +++ b/src/plugins/telemetryTable/TelemetryTable.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; import StalenessUtils from '../../utils/staleness.js'; diff --git a/src/plugins/telemetryTable/TelemetryTableConfiguration.js b/src/plugins/telemetryTable/TelemetryTableConfiguration.js index 96aa612f16..6b9af55f14 100644 --- a/src/plugins/telemetryTable/TelemetryTableConfiguration.js +++ b/src/plugins/telemetryTable/TelemetryTableConfiguration.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; export default class TelemetryTableConfiguration extends EventEmitter { diff --git a/src/plugins/telemetryTable/collections/TableRowCollection.js b/src/plugins/telemetryTable/collections/TableRowCollection.js index c78d1014fd..98f879c676 100644 --- a/src/plugins/telemetryTable/collections/TableRowCollection.js +++ b/src/plugins/telemetryTable/collections/TableRowCollection.js @@ -19,7 +19,7 @@ * this source code distribution or the Licensing information page available * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; /** diff --git a/src/plugins/timeline/pluginSpec.js b/src/plugins/timeline/pluginSpec.js index ee717fa507..c2479aa58c 100644 --- a/src/plugins/timeline/pluginSpec.js +++ b/src/plugins/timeline/pluginSpec.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { nextTick } from 'vue'; import { createOpenMct, resetApplicationState } from '@/utils/testing'; diff --git a/src/plugins/timelist/pluginSpec.js b/src/plugins/timelist/pluginSpec.js index 63f082c623..d8f38add76 100644 --- a/src/plugins/timelist/pluginSpec.js +++ b/src/plugins/timelist/pluginSpec.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import { createOpenMct, resetApplicationState } from 'utils/testing'; import { nextTick } from 'vue'; diff --git a/src/plugins/timer/plugin.js b/src/plugins/timer/plugin.js index 0c5a4d1df4..fac7952b66 100644 --- a/src/plugins/timer/plugin.js +++ b/src/plugins/timer/plugin.js @@ -26,6 +26,7 @@ import StartTimerAction from './actions/StartTimerAction.js'; import StopTimerAction from './actions/StopTimerAction.js'; import TimerViewProvider from './TimerViewProvider.js'; +/** @type {OpenMCTPlugin} */ export default function TimerPlugin() { return function install(openmct) { openmct.types.addType('timer', { diff --git a/src/plugins/utcTimeSystem/DurationFormat.js b/src/plugins/utcTimeSystem/DurationFormat.js index 657e8c2f70..f0bb1d5f2a 100644 --- a/src/plugins/utcTimeSystem/DurationFormat.js +++ b/src/plugins/utcTimeSystem/DurationFormat.js @@ -12,7 +12,6 @@ const DATE_FORMATS = [DATE_FORMAT]; * * @implements {Format} * @constructor - * @memberof platform/commonUI/formats */ class DurationFormat { constructor() { diff --git a/src/plugins/utcTimeSystem/UTCTimeFormat.js b/src/plugins/utcTimeSystem/UTCTimeFormat.js index 331d833750..916c9f5784 100644 --- a/src/plugins/utcTimeSystem/UTCTimeFormat.js +++ b/src/plugins/utcTimeSystem/UTCTimeFormat.js @@ -28,7 +28,6 @@ import moment from 'moment'; * * @implements {Format} * @constructor - * @memberof platform/commonUI/formats */ export default class UTCTimeFormat { constructor() { diff --git a/src/selection/Selection.js b/src/selection/Selection.js index 3254a88d41..754f6c303a 100644 --- a/src/selection/Selection.js +++ b/src/selection/Selection.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import _ from 'lodash'; /** @@ -30,12 +30,13 @@ import _ from 'lodash'; */ /** - * @typedef {import('../../openmct').OpenMCT} OpenMCT + * @typedef {import('../../src/MCT').MCT} OpenMCT */ /** * Manages selection state for Open MCT - * @private + * @constructor + * @extends EventEmitter */ export default class Selection extends EventEmitter { /** diff --git a/src/ui/composables/edit.js b/src/ui/composables/edit.js index 90dd3a4d4e..571abbb2ed 100644 --- a/src/ui/composables/edit.js +++ b/src/ui/composables/edit.js @@ -27,7 +27,7 @@ import { useEventEmitter } from './event.js'; /** * Provides a reactive `isEditing` property that reflects the current editing state of the * application. - * @param {OpenMCT} openmct the Open MCT API + * @param {import('openmct').OpenMCT} openmct the Open MCT API * @returns {{ * isEditing: import('vue').Ref * }} diff --git a/src/ui/composables/resize.js b/src/ui/composables/resize.js index 7a5e3601c3..72ed51eeff 100644 --- a/src/ui/composables/resize.js +++ b/src/ui/composables/resize.js @@ -64,7 +64,7 @@ export function useResizeObserver() { * A composable function which can be used to listen to and handle window resize events. * Throttles the resize event to prevent performance issues. * @param {number} [throttleMs=100] The number of milliseconds to throttle the resize event. - * @returns {Ref<{ width: number, height: number }>} windowSize + * @returns {{ windowSize: { width: number, height: number } }} */ export function useWindowResize(throttleMs = 100) { const windowSize = reactive({ width: window.innerWidth, height: window.innerHeight }); diff --git a/src/ui/layout/RecentObjectsList.vue b/src/ui/layout/RecentObjectsList.vue index 67f665af55..3a0b78289d 100644 --- a/src/ui/layout/RecentObjectsList.vue +++ b/src/ui/layout/RecentObjectsList.vue @@ -251,7 +251,7 @@ export default { /** * Returns true if the `domainObject` supports composition and we are not already * tracking its composition. - * @param {import('../../api/objects/ObjectAPI').DomainObject} domainObject + * @param {import('openmct').DomainObject} domainObject * @param {string} navigationPath */ shouldTrackCompositionFor(domainObject, navigationPath) { diff --git a/src/ui/preview/PreviewAction.js b/src/ui/preview/PreviewAction.js index 137eacf197..2549c2e7f2 100644 --- a/src/ui/preview/PreviewAction.js +++ b/src/ui/preview/PreviewAction.js @@ -19,7 +19,7 @@ * this source code distribution or the Licensing information page available * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import mount from 'utils/mount'; import PreviewContainer from './PreviewContainer.vue'; diff --git a/src/ui/registries/InspectorViewRegistry.js b/src/ui/registries/InspectorViewRegistry.js index 183080147f..5647b5f229 100644 --- a/src/ui/registries/InspectorViewRegistry.js +++ b/src/ui/registries/InspectorViewRegistry.js @@ -25,19 +25,17 @@ const DEFAULT_VIEW_PRIORITY = 0; /** * A InspectorViewRegistry maintains the definitions for views * that may occur in the inspector. - * - * @interface InspectorViewRegistry - * @memberof module:openmct */ export default class InspectorViewRegistry { constructor() { + /** @type {Record} */ this.providers = {}; } /** * - * @param {Object} selection the object to be viewed - * @returns {module:openmct.InspectorViewRegistry[]} any providers + * @param {DomainObject} selection the object to be viewed + * @returns {ViewProvider[]} any providers * which can provide views of this object * @private for platform-internal use */ @@ -63,11 +61,9 @@ export default class InspectorViewRegistry { } /** - * Registers a new type of view. + * Registers a new inspector view provider. * - * @param {module:openmct.InspectorViewRegistry} provider the provider for this view - * @method addProvider - * @memberof module:openmct.InspectorViewRegistry# + * @param {ViewProvider} provider the provider for this view */ addProvider(provider) { const key = provider.key; @@ -88,75 +84,25 @@ export default class InspectorViewRegistry { this.providers[key] = provider; } + /** + * Retrieves a view provider by its key. + * @param {string} key the key of the view provider + * @returns {ViewProvider} the view provider + */ getByProviderKey(key) { return this.providers[key]; } + /** + * @returns {ViewProvider[]} all providers + */ #getAllProviders() { return Object.values(this.providers); } } /** - * A View is used to provide displayable content, and to react to - * associated life cycle events. - * - * @name View - * @interface - * @memberof module:openmct - */ - -/** - * Populate the supplied DOM element with the contents of this view. - * - * View implementations should use this method to attach any - * listeners or acquire other resources that are necessary to keep - * the contents of this view up-to-date. - * - * @param {HTMLElement} container the DOM element to populate - * @method show - * @memberof module:openmct.View# - */ - -/** - * Release any resources associated with this view. - * - * View implementations should use this method to detach any - * listeners or release other resources that are no longer necessary - * once a view is no longer used. - * - * @method destroy - * @memberof module:openmct.View# - */ - -/** - * Exposes types of views in inspector. - * - * @interface InspectorViewProvider - * @property {string} key a unique identifier for this view - * @property {string} name the human-readable name of this view - * @property {string} [description] a longer-form description (typically - * a single sentence or short paragraph) of this kind of view - * @property {string} [cssClass] the CSS class to apply to labels for this - * view (to add icons, for instance) - * @memberof module:openmct - */ - -/** - * Checks if this provider can supply views for a selection. - * - * @method canView - * @memberof module:openmct.InspectorViewProvider# - * @param {module:openmct.selection} selection - * @returns {boolean} 'true' if the view applies to the provided selection, - * otherwise 'false'. - */ - -/** - * Provides a view of the selection object in the inspector. - * - * @method view - * @memberof module:openmct.InspectorViewProvider# - * @param {module:openmct.selection} selection the selection object - * @returns {module:openmct.View} a view of this selection + * @typedef {import("openmct").View} View + * @typedef {import("openmct").ViewProvider} ViewProvider + * @typedef {import('openmct').DomainObject} DomainObject */ diff --git a/src/ui/registries/ToolbarRegistry.js b/src/ui/registries/ToolbarRegistry.js index 44ee72d349..2bfa3af46e 100644 --- a/src/ui/registries/ToolbarRegistry.js +++ b/src/ui/registries/ToolbarRegistry.js @@ -24,7 +24,6 @@ * A ToolbarRegistry maintains the definitions for toolbars. * * @interface ToolbarRegistry - * @memberof module:openmct */ export default function ToolbarRegistry() { this.providers = {}; @@ -70,7 +69,6 @@ ToolbarRegistry.prototype.getByProviderKey = function (key) { * * @param {module:openmct.ToolbarRegistry} provider the provider for this toolbar * @method addProvider - * @memberof module:openmct.ToolbarRegistry# */ ToolbarRegistry.prototype.addProvider = function (provider) { const key = provider.key; @@ -94,14 +92,12 @@ ToolbarRegistry.prototype.addProvider = function (provider) { * @property {string} name the human-readable name of this toolbar * @property {string} [description] a longer-form description (typically * a single sentence or short paragraph) of this kind of toolbar - * @memberof module:openmct */ /** * Checks if this provider can supply toolbar for a selection. * * @method forSelection - * @memberof module:openmct.ToolbarProvider# * @param {module:openmct.selection} selection * @returns {boolean} 'true' if the toolbar applies to the provided selection, * otherwise 'false'. @@ -111,7 +107,6 @@ ToolbarRegistry.prototype.addProvider = function (provider) { * Provides controls that comprise a toolbar. * * @method toolbar - * @memberof module:openmct.ToolbarProvider# * @param {Object} selection the selection object * @returns {Object[]} an array of objects defining controls for the toolbar. */ diff --git a/src/ui/registries/ViewRegistry.js b/src/ui/registries/ViewRegistry.js index 237ce65427..8f2bdfc4c1 100644 --- a/src/ui/registries/ViewRegistry.js +++ b/src/ui/registries/ViewRegistry.js @@ -20,29 +20,28 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; const DEFAULT_VIEW_PRIORITY = 100; /** * A ViewRegistry maintains the definitions for different kinds of views * that may occur in different places in the user interface. - * @interface ViewRegistry - * @memberof module:openmct */ export default class ViewRegistry extends EventEmitter { constructor() { super(); EventEmitter.apply(this); + /** @type {Record} */ this.providers = {}; } + /** - * @private for platform-internal use - * @param {*} item the object to be viewed - * @param {array} objectPath - The current contextual object path of the view object - * eg current domainObject is located under MyItems which is under Root - * @returns {module:openmct.ViewProvider[]} any providers - * which can provide views of this object + * for platform-internal use + * @param {import('openmct').DomainObject} item the object to be viewed + * @param {import('openmct').ObjectPath} objectPath - The current contextual object path of the view object + * @returns {ViewProvider[]} a list of providers that can provide views for this object, sorted by + * descending priority */ get(item, objectPath) { if (objectPath === undefined) { @@ -62,18 +61,18 @@ export default class ViewRegistry extends EventEmitter { }) .sort(byPriority); } + /** * @private */ getAllProviders() { return Object.values(this.providers); } + /** * Register a new type of view. * - * @param {module:openmct.ViewProvider} provider the provider for this view - * @method addProvider - * @memberof module:openmct.ViewRegistry# + * @param {ViewProvider} provider the provider for this view */ addProvider(provider) { const key = provider.key; @@ -100,12 +99,16 @@ export default class ViewRegistry extends EventEmitter { this.providers[key] = provider; } + /** - * @private + * Returns the view provider by key + * @param {string} key + * @returns {ViewProvider} */ getByProviderKey(key) { return this.providers[key]; } + /** * Used internally to support seamless usage of new views with old * views. @@ -119,150 +122,69 @@ export default class ViewRegistry extends EventEmitter { } /** - * A View is used to provide displayable content, and to react to - * associated life cycle events. - * - * @name View - * @interface - * @memberof module:openmct + * @typedef {import('openmct').DomainObject} DomainObject + * @typedef {import('openmct').ObjectPath} ObjectPath */ /** - * Populate the supplied DOM element with the contents of this view. + * @typedef {Object} ViewOptions + * @property {() => void} [renderWhenVisible] + * This function can be used for all rendering logic that would otherwise be executed within a + * `requestAnimationFrame` call. When called, `renderWhenVisible` will either execute the provided + * function immediately (via `requestAnimationFrame`) if the view is currently visible, or defer its + * execution until the view becomes visible. * + * Additionally, `renderWhenVisible` returns a boolean value indicating whether the provided + * function was executed immediately (`true`) or deferred (`false`). + * Monitoring of visibility begins after the first call to `renderWhenVisible` is made. + */ + +/** + * @typedef {Object} View + * A View is used to provide displayable content, and to react to + * associated life cycle events. + * @property {(container: HTMLElement, isEditing: boolean | undefined, viewOptions: ViewOptions | undefined) => void} show + * Populate the supplied DOM element with the contents of this view. * View implementations should use this method to attach any * listeners or acquire other resources that are necessary to keep * the contents of this view up-to-date. * - * @param {HTMLElement} container the DOM element to populate - * @method show - * @memberof module:openmct.View# - */ + * - `container`: The DOM element where the view should be rendered. + * - `isEditing`: Indicates whether the view is in editing mode. + * - `viewOptions`: An object with configuration options for the view. + * @property {() => void} destroy - Release any resources associated with this view. + * View implementations should use this method to detach any listeners or release other resources + * that are no longer necessary once a view is no longer used. + * @property {() => { item: DomainObject, isMultiSelectEvent: boolean }} [getSelectionContext] + * A function that returns the selection context of the view. -/** - * Indicates whether or not the application is in edit mode. This supports - * views that have distinct visual and behavioral elements when the - * navigated object is being edited. - * - * For cases where a completely separate view is desired for editing purposes, - * see {@link openmct.ViewProvider#edit} - * - * @param {boolean} isEditing - * @method show - * @memberof module:openmct.View# - */ - -/** - * Release any resources associated with this view. - * - * View implementations should use this method to detach any - * listeners or release other resources that are no longer necessary - * once a view is no longer used. - * - * @method destroy - * @memberof module:openmct.View# - */ - -/** - * Returns the selection context. - * - * View implementations should use this method to customize - * the selection context. - * - * @method getSelectionContext - * @memberof module:openmct.View# + * View implementations may use this method to customize the selection context. */ /** * Exposes types of views in Open MCT. * - * @interface ViewProvider - * @property {string} key a unique identifier for this view - * @property {string} name the human-readable name of this view - * @property {string} [description] a longer-form description (typically - * a single sentence or short paragraph) of this kind of view - * @property {string} [cssClass] the CSS class to apply to labels for this - * view (to add icons, for instance) - * @memberof module:openmct - */ - -/** - * Check if this provider can supply views for a domain object. + * @typedef {Object} ViewProvider + * @property {string} key - The unique key that identifies this view + * @property {string} name - The name of the view + * @property {string} [cssClass] - The CSS class to apply to labels for this view (to add icons, + * for instance) + * @property {(domainObject: DomainObject, objectPath: ObjectPath) => boolean} canView + * Returns true if this provider is able to supply views for the given {@link DomainObject}. * * When called by Open MCT, this may include additional arguments * which are on the path to the object to be viewed; for instance, * when viewing "A Folder" within "My Items", this method will be - * invoked with "A Folder" (as a domain object) as the first argument - * - * @method canView - * @memberof module:openmct.ViewProvider# - * @param {module:openmct.DomainObject} domainObject the domain object - * to be viewed - * @param {array} objectPath - The current contextual object path of the view object - * eg current domainObject is located under MyItems which is under Root - * @returns {boolean} 'true' if the view applies to the provided object, - * otherwise 'false'. - */ - -/** + * invoked with "A Folder" (as a {@link DomainObject}) as the first argument. + * @property {(domainObject: DomainObject, objectPath: ObjectPath) => boolean} [canEdit] * An optional function that defines whether or not this view can be used to edit a given object. - * If not provided, will default to `false` and the view will not support editing. To support editing, - * return true from this function and then - - * * Return a {@link openmct.View} from the `view` function, using the `onEditModeChange` callback to - * add and remove editing elements from the view - * OR - * * Return a {@link openmct.View} from the `view` function defining a read-only view. - * AND - * * Define an {@link openmct.ViewProvider#Edit} function on the view provider that returns an - * editing-specific view. + * If not provided, will default to `false` and the view will not support editing. + * @property {(domainObject: DomainObject, objectPath: ObjectPath) => View} view A function that + * provides a view for the provided domain object. + * @property {(domainObject: DomainObject) => number} [priority] + * A function that returns the priority of the view. The more positive the value, the higher the + * priority. Similarly, the more negative the value, the lower the priority. * - * @method canEdit - * @memberof module:openmct.ViewProvider# - * @param {module:openmct.DomainObject} domainObject the domain object - * to be edited - * @param {array} objectPath - The current contextual object path of the view object - * eg current domainObject is located under MyItems which is under Root - * @returns {boolean} 'true' if the view can be used to edit the provided object, - * otherwise 'false'. - */ - -/** - * Optional method determining the priority of a given view. If this - * function is not defined on a view provider, then a default priority - * of 100 will be applicable for all objects supported by this view. - * - * @method priority - * @memberof module:openmct.ViewProvider# - * @param {module:openmct.DomainObject} domainObject the domain object - * to be viewed - * @returns {number} The priority of the view. If multiple views could apply - * to an object, the view that returns the lowest number will be - * the default view. - */ - -/** - * Provide a view of this object. - * - * When called by Open MCT, the following arguments will be passed to it: - * @param {Object} domainObject - the domainObject that the view is provided for - * @param {array} objectPath - The current contextual object path of the view object - * eg current domainObject is located under MyItems which is under Root - * - * @method view - * @memberof module:openmct.ViewProvider# - * @param {*} object the object to be viewed - * @returns {module:openmct.View} a view of this domain object - */ - -/** - * Provide an edit-mode specific view of this object. - * - * If optionally specified, this function will be called when the application - * enters edit mode. This will cause the active non-edit mode view and its - * dom element to be destroyed. - * - * @method edit - * @memberof module:openmct.ViewProvider# - * @param {*} object the object to be edit - * @returns {module:openmct.View} an editable view of this domain object + * If not provided, the default priority of 100 will be used. This value is used to sort the views + * by descending priority if there are multiple views that can be shown for a given object. */ diff --git a/src/ui/router/ApplicationRouter.js b/src/ui/router/ApplicationRouter.js index a1f5996427..1924123f5a 100644 --- a/src/ui/router/ApplicationRouter.js +++ b/src/ui/router/ApplicationRouter.js @@ -20,26 +20,28 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; import LocationBar from 'location-bar'; import _ from 'lodash'; class ApplicationRouter extends EventEmitter { /** - * events - * change:params -> notify listeners w/ new, old, and changed. - * change:path -> notify listeners w/ new, old paths. - * - * methods: - * update(path, params) -> updates path and params at the same time. Only - * updates specified params, other params are not modified. - * updateParams(newParams) -> update only specified params, leaving rest - * intact. Does not modify path. - * updatePath(path); + * events + * change:params -> notify listeners w/ new, old, and changed. + * change:path -> notify listeners w/ new, old paths. + * + * methods: + * update(path, params) -> updates path and params at the same time. Only + * updates specified params, other params are not modified. + * updateParams(newParams) -> update only specified params, leaving rest + * intact. Does not modify path. + * updatePath(path); + * + * route(path, handler); + * start(); Start routing. + * @param {import('../../../openmct').OpenMCT} openmct + */ - * route(path, handler); - * start(); Start routing. - */ constructor(openmct) { super(); @@ -47,6 +49,7 @@ class ApplicationRouter extends EventEmitter { this.openmct = openmct; this.routes = []; this.started = false; + this.path = null; this.setHash = _.debounce(this.setHash.bind(this), 300); @@ -159,8 +162,8 @@ class ApplicationRouter extends EventEmitter { /** * Add routes listeners * - * @param {string} matcher Regex to match value in url - * @param {@function} callback function called when found match in url + * @param {RegExp} matcher Regex to match value in url + * @param {Function} callback function called when found match in url */ route(matcher, callback) { this.routes.push({ diff --git a/src/ui/router/Browse.js b/src/ui/router/Browse.js index f64eb7c574..8a996ae163 100644 --- a/src/ui/router/Browse.js +++ b/src/ui/router/Browse.js @@ -19,15 +19,36 @@ * this source code distribution or the Licensing information page available * at runtime from the About dialog for additional information. *****************************************************************************/ - -export default class Browse { +class Browse { + /** + * @type {number} + */ #navigateCall = 0; + /** + * @type {Object?} + */ #browseObject = null; + /** + * @type {Function | undefined} + */ #unobserve = undefined; + /** + * @type {string | undefined} + */ #currentObjectPath = undefined; + /** + * @type {boolean} + */ #isRoutingInProgress = false; + /** + * @type {import('../../../openmct').OpenMCT} + */ #openmct; + /** + * + * @param {import('../../../openmct').OpenMCT} openmct + */ constructor(openmct) { this.#openmct = openmct; this.#openmct.router.route(/^\/browse\/?$/, this.#navigateToFirstChildOfRoot.bind(this)); @@ -165,3 +186,5 @@ export default class Browse { this.#navigateToPath(navigatePath, params.view); } } + +export default Browse; diff --git a/src/utils/agent/Agent.js b/src/utils/agent/Agent.js index 9733554fa9..245bfef76b 100644 --- a/src/utils/agent/Agent.js +++ b/src/utils/agent/Agent.js @@ -25,7 +25,6 @@ * device names * @constructor * @param window the browser object model - * @memberof /utils/agent */ export default class Agent { constructor(window) { @@ -63,7 +62,7 @@ export default class Agent { } /** * Check if the user is on a tablet sized android device - * @returns {boolean} true on an android tablet + * @returns {boolean | undefined} true on an android tablet */ isAndroidTablet() { if (this.mobileName === 'Android') { @@ -78,7 +77,7 @@ export default class Agent { } /** * Check if the user is on a tablet-sized mobile device. - * @returns {boolean} true on a tablet + * @returns {boolean | undefined} true on a tablet */ isTablet() { return ( diff --git a/src/utils/clock/DefaultClock.js b/src/utils/clock/DefaultClock.js index a509718880..6b0d0c4e2b 100644 --- a/src/utils/clock/DefaultClock.js +++ b/src/utils/clock/DefaultClock.js @@ -19,22 +19,18 @@ * this source code distribution or the Licensing information page available * at runtime from the About dialog for additional information. *****************************************************************************/ -import EventEmitter from 'EventEmitter'; +import { EventEmitter } from 'eventemitter3'; /** * A {@link openmct.TimeAPI.Clock} that updates the temporal bounds of the - * application based on values provided by a ticking clock, - * with the periodicity specified (optionally). - * @param {number} period The periodicity with which the clock should tick + * application based on values provided by a ticking clock. * @constructor */ - export default class DefaultClock extends EventEmitter { constructor() { super(); this.key = 'clock'; - this.cssClass = 'icon-clock'; this.name = 'Clock'; this.description = 'A default clock for openmct.'; @@ -49,38 +45,53 @@ export default class DefaultClock extends EventEmitter { * Register a listener for the clock. When it ticks, the * clock will provide the time from the configured endpoint * - * @param listener - * @returns {function} a function for deregistering the provided listener + * @override + * @param {string | symbol} event the event to listen for + * @param {Function} fn the function to call when the event is emitted + * @param {*} [context] the context to use for the function call + * @returns {this} a function for deregistering the provided listener */ - on(event) { - let result = super.on.apply(this, arguments); + on(event, fn, context) { + super.on(event, fn, context); if (this.listeners(event).length === 1) { this.start(); } - return result; + return this; } /** * Register a listener for the clock. When it ticks, the * clock will provide the current local system time * - * @param listener - * @returns {function} a function for deregistering the provided listener + * @override + * @param {string | symbol} event the event to listen for + * @param {Function} [fn] the function to call when the event is emitted + * @param {*} [context] the context to use for the function call + * @param {boolean} [once] + * @returns {this} */ - off(event) { - let result = super.off.apply(this, arguments); + off(event, fn, context, once) { + super.off(event, fn, context, once); if (this.listeners(event).length === 0) { this.stop(); } - return result; + return this; + } + + stop() { + throw new Error("Method 'stop()' must be implemented."); + } + + start() { + throw new Error("Method 'start()' must be implemented."); } /** - * @returns {number} The last value provided for a clock tick + * @returns {number} The most recent value provided for a clock tick */ currentValue() { return this.lastTick; diff --git a/src/utils/duration.js b/src/utils/duration.js index e9fcf114a1..517d80c8d1 100644 --- a/src/utils/duration.js +++ b/src/utils/duration.js @@ -63,7 +63,21 @@ export function millisecondsToDHMS(numericDuration) { return `${dhms ? '+' : ''} ${dhms}`; } -export function getPreciseDuration(value, { excludeMilliSeconds, useDayFormat } = {}) { +/** + * + * @param {number} value + * @param {Object} options + * @param {boolean | undefined} options.excludeMilliSeconds + * @param {boolean | undefined} options.useDayFormat + * @returns {string} + */ +export function getPreciseDuration( + value, + { excludeMilliSeconds, useDayFormat } = { + excludeMilliSeconds: null, + useDayFormat: null + } +) { let preciseDuration; const ms = value || 0; diff --git a/src/utils/mount.js b/src/utils/mount.js index 565af579cc..bd24e9b655 100644 --- a/src/utils/mount.js +++ b/src/utils/mount.js @@ -1,10 +1,24 @@ import { createApp, defineComponent } from 'vue'; +/** + * @typedef {import('vue').Component} Component + */ + +/** + * Mounts a Vue component to a DOM element. + * @param {Component | any} component + * @param {Object} [options={}] the options object + * @param {Object} [options.props] the props for the component + * @param {Object} [options.children] the children for the component + * @param {HTMLElement} [options.element] the element to mount the component to + * @returns {Object} + */ export default function mount(component, { props, children, element } = {}) { let el = element; if (!el) { el = document.createElement('div'); } + /** @type {Component | any} */ let vueComponent = defineComponent(component); let app = createApp(vueComponent); let mountedComponentInstance = app.mount(el); diff --git a/src/utils/staleness.js b/src/utils/staleness.js index 06c3b17734..533dc6059b 100644 --- a/src/utils/staleness.js +++ b/src/utils/staleness.js @@ -20,7 +20,7 @@ * at runtime from the About dialog for additional information. *****************************************************************************/ -export default class StalenessUtils { +class StalenessUtils { constructor(openmct, domainObject) { this.openmct = openmct; this.domainObject = domainObject; @@ -29,6 +29,7 @@ export default class StalenessUtils { this.setTimeSystem(this.openmct.time.getTimeSystem()); this.watchTimeSystem(); + this.timeSystemKey = null; } shouldUpdateStaleness(stalenessResponse, id) { @@ -54,25 +55,26 @@ export default class StalenessUtils { } setTimeSystem(timeSystem) { - let metadataValue = { format: timeSystem.key }; - - if (this.metadata) { - metadataValue = this.metadata.value(timeSystem.key) ?? metadataValue; - } + this.timeSystem = timeSystem; + } + parseTime(stalenessResponse) { + const metadataValue = this.metadata.value(this.timeSystem.key) ?? { + format: this.timeSystem.key + }; const valueFormatter = this.openmct.telemetry.getValueFormatter(metadataValue); - this.parseTime = (stalenessResponse) => { - const stalenessDatum = { - ...stalenessResponse, - source: stalenessResponse[timeSystem.key] - }; - - return valueFormatter.parse(stalenessDatum); + const stalenessDatum = { + ...stalenessResponse, + source: stalenessResponse[this.timeSystem.key] }; + + return valueFormatter.parse(stalenessDatum); } destroy() { this.unwatchTimeSystem(); } } + +export default StalenessUtils; diff --git a/src/utils/visibility/VisibilityObserver.js b/src/utils/visibility/VisibilityObserver.js index 43bf1f5a70..a6c944ec77 100644 --- a/src/utils/visibility/VisibilityObserver.js +++ b/src/utils/visibility/VisibilityObserver.js @@ -23,10 +23,27 @@ /** * Optimizes `requestAnimationFrame` calls to only execute when the element is visible in the viewport. */ -export default class VisibilityObserver { +class VisibilityObserver { + /** + * @type {HTMLElement | null} + */ #element; + /** + * @type {IntersectionObserver | null} + */ #observer; + /** + * @type {(() => void) | null} + */ lastUnfiredFunc; + /** + * @type {boolean | null} + */ + isIntersecting; + /** + * @type {boolean} + */ + calledOnce; /** * Constructs a VisibilityObserver instance to manage visibility-based requestAnimationFrame calls. @@ -50,12 +67,16 @@ export default class VisibilityObserver { this.renderWhenVisible = this.renderWhenVisible.bind(this); } + /** + * @returns {boolean} + */ #inOverlay() { - return this.#element.closest('.js-overlay') !== null; + return this.#element?.closest('.js-overlay'); } - #observerCallback = ([entry]) => { - if (entry.target === this.#element) { + #observerCallback = (entries) => { + const entry = entries[0]; + if (entry && entry.target === this.#element) { if (this.#inOverlay() && !entry.isIntersecting) { this.isIntersecting = true; } else { @@ -73,32 +94,38 @@ export default class VisibilityObserver { * If the element is not visible, the function is stored and called when the element becomes visible. * Note that if called multiple times while not visible, only the last execution is stored and executed. * - * @param {Function} func - The function to execute. + * @param {() => void} func - The function to execute. * @returns {boolean} True if the function was executed immediately, false otherwise. */ renderWhenVisible(func) { if (!this.calledOnce) { this.calledOnce = true; + if (!this.#observer || !this.#element) { + this.lastUnfiredFunc = func; + return false; + } this.#observer.observe(this.#element); - window.requestAnimationFrame(func); - return true; - } else if (this.isIntersecting) { - window.requestAnimationFrame(func); - return true; - } else { + } else if (!this.isIntersecting) { this.lastUnfiredFunc = func; return false; } + window.requestAnimationFrame(func); + return true; } /** * Stops observing the element for visibility changes and cleans up resources to prevent memory leaks. */ destroy() { - this.#observer.unobserve(this.#element); + if (this.#observer && this.#element) { + this.#observer.unobserve(this.#element); + } + this.#element = null; this.isIntersecting = null; this.#observer = null; this.lastUnfiredFunc = null; } } + +export default VisibilityObserver; diff --git a/src/utils/vueWrapHtmlElement.js b/src/utils/vueWrapHtmlElement.js index 332d76cc19..1762f80ef5 100644 --- a/src/utils/vueWrapHtmlElement.js +++ b/src/utils/vueWrapHtmlElement.js @@ -9,6 +9,7 @@ import { defineComponent, h, onMounted, ref } from 'vue'; export default function vueWrapHtmlElement(element) { return defineComponent({ setup() { + /** @type {import('vue').Ref} */ const wrapper = ref(null); onMounted(() => { diff --git a/tsconfig.json b/tsconfig.json index cb6a862477..4ebb82543c 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -14,17 +14,7 @@ "esModuleInterop": true, "noImplicitOverride": true, "noImplicitAny": false, - "module": "esnext", - "moduleResolution": "node", - "outFile": "dist/types/index.d.ts", - "skipLibCheck": true, - "target": "ES2015", - "paths": { - // matches the alias in webpack config, so that types for those imports are visible. - "@/*": [ - "src/*" - ] - } + "outFile": "dist/types/index.d.ts" }, "include": [ "src/api/**/*.js"