mirror of
https://github.com/nasa/openmct.git
synced 2025-06-27 11:32:13 +00:00
Compare commits
62 Commits
tests-acti
...
change-met
Author | SHA1 | Date | |
---|---|---|---|
32e6ae9885 | |||
b8ded0a16e | |||
b68f79f427 | |||
221d10d3e6 | |||
22d32eed1d | |||
5d656f0963 | |||
201d622b85 | |||
3571004f5c | |||
16249c3790 | |||
5377f0d0b3 | |||
15778b00a0 | |||
169eec0a51 | |||
f789775b1c | |||
fc59a4dce4 | |||
29128a891d | |||
dd3d4c8c3a | |||
4047c888be | |||
1499286bee | |||
6226763c37 | |||
7623a0648f | |||
b7085f7f62 | |||
55c851873c | |||
2b143dfc0f | |||
9405272f3b | |||
a9be9f1827 | |||
abb1a5c75b | |||
5e2fe7dc42 | |||
e4d6e90c35 | |||
84d9a525a9 | |||
0aca0ce6a6 | |||
c0742d521c | |||
92737b43af | |||
8b0f6885ee | |||
9af2d15cef | |||
e60d8d08a4 | |||
3e9b567fce | |||
6f51de85db | |||
f202ae19cb | |||
668bd75025 | |||
e6e07cf959 | |||
2f8431905f | |||
23aba14dfe | |||
b0fa955914 | |||
98207a3e0d | |||
26b81345f2 | |||
ac2034b243 | |||
351848ad56 | |||
cbac495f93 | |||
15ef5b7623 | |||
46c7ac944f | |||
aa4bfab462 | |||
f5cbb37e5a | |||
8d9079984a | |||
41783d8939 | |||
441ad58fe7 | |||
06a6a3f773 | |||
52fab78625 | |||
5eb6c15959 | |||
ce8c31cfa4 | |||
d80c0eef8e | |||
55829dcf05 | |||
d78956327c |
20
.eslintrc.js
20
.eslintrc.js
@ -54,7 +54,7 @@ module.exports = {
|
|||||||
{
|
{
|
||||||
"anonymous": "always",
|
"anonymous": "always",
|
||||||
"asyncArrow": "always",
|
"asyncArrow": "always",
|
||||||
"named": "never",
|
"named": "never"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"array-bracket-spacing": "error",
|
"array-bracket-spacing": "error",
|
||||||
@ -178,7 +178,10 @@ module.exports = {
|
|||||||
//https://eslint.org/docs/rules/no-whitespace-before-property
|
//https://eslint.org/docs/rules/no-whitespace-before-property
|
||||||
"no-whitespace-before-property": "error",
|
"no-whitespace-before-property": "error",
|
||||||
// https://eslint.org/docs/rules/object-curly-newline
|
// https://eslint.org/docs/rules/object-curly-newline
|
||||||
"object-curly-newline": ["error", {"consistent": true, "multiline": true}],
|
"object-curly-newline": ["error", {
|
||||||
|
"consistent": true,
|
||||||
|
"multiline": true
|
||||||
|
}],
|
||||||
// https://eslint.org/docs/rules/object-property-newline
|
// https://eslint.org/docs/rules/object-property-newline
|
||||||
"object-property-newline": "error",
|
"object-property-newline": "error",
|
||||||
// https://eslint.org/docs/rules/brace-style
|
// https://eslint.org/docs/rules/brace-style
|
||||||
@ -188,7 +191,7 @@ module.exports = {
|
|||||||
// https://eslint.org/docs/rules/operator-linebreak
|
// https://eslint.org/docs/rules/operator-linebreak
|
||||||
"operator-linebreak": ["error", "before", {"overrides": {"=": "after"}}],
|
"operator-linebreak": ["error", "before", {"overrides": {"=": "after"}}],
|
||||||
// https://eslint.org/docs/rules/padding-line-between-statements
|
// https://eslint.org/docs/rules/padding-line-between-statements
|
||||||
"padding-line-between-statements":["error", {
|
"padding-line-between-statements": ["error", {
|
||||||
"blankLine": "always",
|
"blankLine": "always",
|
||||||
"prev": "multiline-block-like",
|
"prev": "multiline-block-like",
|
||||||
"next": "*"
|
"next": "*"
|
||||||
@ -200,11 +203,17 @@ module.exports = {
|
|||||||
// https://eslint.org/docs/rules/space-infix-ops
|
// https://eslint.org/docs/rules/space-infix-ops
|
||||||
"space-infix-ops": "error",
|
"space-infix-ops": "error",
|
||||||
// https://eslint.org/docs/rules/space-unary-ops
|
// https://eslint.org/docs/rules/space-unary-ops
|
||||||
"space-unary-ops": ["error", {"words": true, "nonwords": false}],
|
"space-unary-ops": ["error", {
|
||||||
|
"words": true,
|
||||||
|
"nonwords": false
|
||||||
|
}],
|
||||||
// https://eslint.org/docs/rules/arrow-spacing
|
// https://eslint.org/docs/rules/arrow-spacing
|
||||||
"arrow-spacing": "error",
|
"arrow-spacing": "error",
|
||||||
// https://eslint.org/docs/rules/semi-spacing
|
// https://eslint.org/docs/rules/semi-spacing
|
||||||
"semi-spacing": ["error", {"before": false, "after": true}],
|
"semi-spacing": ["error", {
|
||||||
|
"before": false,
|
||||||
|
"after": true
|
||||||
|
}],
|
||||||
|
|
||||||
"vue/html-indent": [
|
"vue/html-indent": [
|
||||||
"error",
|
"error",
|
||||||
@ -237,6 +246,7 @@ module.exports = {
|
|||||||
}],
|
}],
|
||||||
"vue/multiline-html-element-content-newline": "off",
|
"vue/multiline-html-element-content-newline": "off",
|
||||||
"vue/singleline-html-element-content-newline": "off",
|
"vue/singleline-html-element-content-newline": "off",
|
||||||
|
"vue/no-mutating-props": "off"
|
||||||
|
|
||||||
},
|
},
|
||||||
"overrides": [
|
"overrides": [
|
||||||
|
@ -182,7 +182,7 @@ The following guidelines are provided for anyone contributing source code to the
|
|||||||
1. Avoid the use of "magic" values.
|
1. Avoid the use of "magic" values.
|
||||||
eg.
|
eg.
|
||||||
```JavaScript
|
```JavaScript
|
||||||
Const UNAUTHORIZED = 401
|
const UNAUTHORIZED = 401;
|
||||||
if (responseCode === UNAUTHORIZED)
|
if (responseCode === UNAUTHORIZED)
|
||||||
```
|
```
|
||||||
is preferable to
|
is preferable to
|
||||||
|
@ -138,7 +138,7 @@ define([
|
|||||||
"id": "styleguide:home",
|
"id": "styleguide:home",
|
||||||
"priority": "preferred",
|
"priority": "preferred",
|
||||||
"model": {
|
"model": {
|
||||||
"type": "folder",
|
"type": "noneditable.folder",
|
||||||
"name": "Style Guide Home",
|
"name": "Style Guide Home",
|
||||||
"location": "ROOT",
|
"location": "ROOT",
|
||||||
"composition": [
|
"composition": [
|
||||||
@ -155,7 +155,7 @@ define([
|
|||||||
"id": "styleguide:ui-elements",
|
"id": "styleguide:ui-elements",
|
||||||
"priority": "preferred",
|
"priority": "preferred",
|
||||||
"model": {
|
"model": {
|
||||||
"type": "folder",
|
"type": "noneditable.folder",
|
||||||
"name": "UI Elements",
|
"name": "UI Elements",
|
||||||
"location": "styleguide:home",
|
"location": "styleguide:home",
|
||||||
"composition": [
|
"composition": [
|
||||||
|
@ -86,7 +86,9 @@
|
|||||||
openmct.install(openmct.plugins.MyItems());
|
openmct.install(openmct.plugins.MyItems());
|
||||||
openmct.install(openmct.plugins.Generator());
|
openmct.install(openmct.plugins.Generator());
|
||||||
openmct.install(openmct.plugins.ExampleImagery());
|
openmct.install(openmct.plugins.ExampleImagery());
|
||||||
|
openmct.install(openmct.plugins.PlanLayout());
|
||||||
openmct.install(openmct.plugins.Timeline());
|
openmct.install(openmct.plugins.Timeline());
|
||||||
|
openmct.install(openmct.plugins.PlotVue());
|
||||||
openmct.install(openmct.plugins.UTCTimeSystem());
|
openmct.install(openmct.plugins.UTCTimeSystem());
|
||||||
openmct.install(openmct.plugins.AutoflowView({
|
openmct.install(openmct.plugins.AutoflowView({
|
||||||
type: "telemetry.panel"
|
type: "telemetry.panel"
|
||||||
@ -131,10 +133,10 @@
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
// maximum recent bounds to retain in conductor history
|
// maximum recent bounds to retain in conductor history
|
||||||
records: 10,
|
records: 10
|
||||||
// maximum duration between start and end bounds
|
// maximum duration between start and end bounds
|
||||||
// for utc-based time systems this is in milliseconds
|
// for utc-based time systems this is in milliseconds
|
||||||
limit: ONE_DAY
|
// limit: ONE_DAY
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Realtime",
|
name: "Realtime",
|
||||||
|
@ -86,7 +86,7 @@ module.exports = (config) => {
|
|||||||
reports: ['html', 'lcovonly', 'text-summary'],
|
reports: ['html', 'lcovonly', 'text-summary'],
|
||||||
thresholds: {
|
thresholds: {
|
||||||
global: {
|
global: {
|
||||||
lines: 65
|
lines: 66
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "openmct",
|
"name": "openmct",
|
||||||
"version": "1.4.1-SNAPSHOT",
|
"version": "1.6.2-SNAPSHOT",
|
||||||
"description": "The Open MCT core platform",
|
"description": "The Open MCT core platform",
|
||||||
"dependencies": {},
|
"dependencies": {},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
@ -23,7 +23,7 @@
|
|||||||
"d3-time": "1.0.x",
|
"d3-time": "1.0.x",
|
||||||
"d3-time-format": "2.1.x",
|
"d3-time-format": "2.1.x",
|
||||||
"eslint": "7.0.0",
|
"eslint": "7.0.0",
|
||||||
"eslint-plugin-vue": "^6.0.0",
|
"eslint-plugin-vue": "^7.5.0",
|
||||||
"eslint-plugin-you-dont-need-lodash-underscore": "^6.10.0",
|
"eslint-plugin-you-dont-need-lodash-underscore": "^6.10.0",
|
||||||
"eventemitter3": "^1.2.0",
|
"eventemitter3": "^1.2.0",
|
||||||
"exports-loader": "^0.7.0",
|
"exports-loader": "^0.7.0",
|
||||||
|
@ -71,10 +71,10 @@ define(
|
|||||||
openmct.editor.cancel();
|
openmct.editor.cancel();
|
||||||
}
|
}
|
||||||
|
|
||||||
function isFirstViewEditable(domainObject) {
|
function isFirstViewEditable(domainObject, objectPath) {
|
||||||
let firstView = openmct.objectViews.get(domainObject)[0];
|
let firstView = openmct.objectViews.get(domainObject, objectPath)[0];
|
||||||
|
|
||||||
return firstView && firstView.canEdit && firstView.canEdit(domainObject);
|
return firstView && firstView.canEdit && firstView.canEdit(domainObject, objectPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
function navigateAndEdit(object) {
|
function navigateAndEdit(object) {
|
||||||
@ -88,7 +88,7 @@ define(
|
|||||||
|
|
||||||
window.location.href = url;
|
window.location.href = url;
|
||||||
|
|
||||||
if (isFirstViewEditable(object.useCapability('adapter'))) {
|
if (isFirstViewEditable(object.useCapability('adapter'), objectPath)) {
|
||||||
openmct.editor.edit();
|
openmct.editor.edit();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -44,9 +44,9 @@ define(
|
|||||||
// is also invoked during the create process which should be allowed,
|
// is also invoked during the create process which should be allowed,
|
||||||
// because it may be saved elsewhere
|
// because it may be saved elsewhere
|
||||||
if ((key === 'edit' && category === 'view-control') || key === 'properties') {
|
if ((key === 'edit' && category === 'view-control') || key === 'properties') {
|
||||||
let newStyleObject = objectUtils.toNewFormat(domainObject, domainObject.getId());
|
let identifier = this.openmct.objects.parseKeyString(domainObject.getId());
|
||||||
|
|
||||||
return this.openmct.objects.isPersistable(newStyleObject);
|
return this.openmct.objects.isPersistable(identifier);
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@ -43,7 +43,8 @@ define(
|
|||||||
);
|
);
|
||||||
|
|
||||||
mockObjectAPI = jasmine.createSpyObj('objectAPI', [
|
mockObjectAPI = jasmine.createSpyObj('objectAPI', [
|
||||||
'isPersistable'
|
'isPersistable',
|
||||||
|
'parseKeyString'
|
||||||
]);
|
]);
|
||||||
|
|
||||||
mockAPI = {
|
mockAPI = {
|
||||||
|
@ -48,9 +48,9 @@ define(
|
|||||||
// prevents editing of objects that cannot be persisted, so we can assume that this
|
// prevents editing of objects that cannot be persisted, so we can assume that this
|
||||||
// is a new object.
|
// is a new object.
|
||||||
if (!(parent.hasCapability('editor') && parent.getCapability('editor').isEditContextRoot())) {
|
if (!(parent.hasCapability('editor') && parent.getCapability('editor').isEditContextRoot())) {
|
||||||
let newStyleObject = objectUtils.toNewFormat(parent, parent.getId());
|
let identifier = this.openmct.objects.parseKeyString(parent.getId());
|
||||||
|
|
||||||
return this.openmct.objects.isPersistable(newStyleObject);
|
return this.openmct.objects.isPersistable(identifier);
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@ -33,7 +33,8 @@ define(
|
|||||||
|
|
||||||
beforeEach(function () {
|
beforeEach(function () {
|
||||||
objectAPI = jasmine.createSpyObj('objectsAPI', [
|
objectAPI = jasmine.createSpyObj('objectsAPI', [
|
||||||
'isPersistable'
|
'isPersistable',
|
||||||
|
'parseKeyString'
|
||||||
]);
|
]);
|
||||||
|
|
||||||
mockOpenMCT = {
|
mockOpenMCT = {
|
||||||
|
@ -37,7 +37,7 @@ define(
|
|||||||
this.$q = $q;
|
this.$q = $q;
|
||||||
}
|
}
|
||||||
|
|
||||||
LocatingObjectDecorator.prototype.getObjects = function (ids) {
|
LocatingObjectDecorator.prototype.getObjects = function (ids, abortSignal) {
|
||||||
var $q = this.$q,
|
var $q = this.$q,
|
||||||
$log = this.$log,
|
$log = this.$log,
|
||||||
objectService = this.objectService,
|
objectService = this.objectService,
|
||||||
@ -79,7 +79,7 @@ define(
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return objectService.getObjects([id]).then(attachContext);
|
return objectService.getObjects([id], abortSignal).then(attachContext);
|
||||||
}
|
}
|
||||||
|
|
||||||
ids.forEach(function (id) {
|
ids.forEach(function (id) {
|
||||||
|
@ -32,7 +32,8 @@
|
|||||||
function indexItem(id, model) {
|
function indexItem(id, model) {
|
||||||
indexedItems.push({
|
indexedItems.push({
|
||||||
id: id,
|
id: id,
|
||||||
name: model.name.toLowerCase()
|
name: model.name.toLowerCase(),
|
||||||
|
type: model.type
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -125,13 +125,12 @@ define([
|
|||||||
* @param topic the topicService.
|
* @param topic the topicService.
|
||||||
*/
|
*/
|
||||||
GenericSearchProvider.prototype.indexOnMutation = function (topic) {
|
GenericSearchProvider.prototype.indexOnMutation = function (topic) {
|
||||||
var mutationTopic = topic('mutation'),
|
let mutationTopic = topic('mutation');
|
||||||
provider = this;
|
|
||||||
|
|
||||||
mutationTopic.listen(function (mutatedObject) {
|
mutationTopic.listen(mutatedObject => {
|
||||||
var editor = mutatedObject.getCapability('editor');
|
let editor = mutatedObject.getCapability('editor');
|
||||||
if (!editor || !editor.inEditContext()) {
|
if (!editor || !editor.inEditContext()) {
|
||||||
provider.index(
|
this.index(
|
||||||
mutatedObject.getId(),
|
mutatedObject.getId(),
|
||||||
mutatedObject.getModel()
|
mutatedObject.getModel()
|
||||||
);
|
);
|
||||||
@ -147,11 +146,16 @@ define([
|
|||||||
* @param {String} id to be indexed.
|
* @param {String} id to be indexed.
|
||||||
*/
|
*/
|
||||||
GenericSearchProvider.prototype.scheduleForIndexing = function (id) {
|
GenericSearchProvider.prototype.scheduleForIndexing = function (id) {
|
||||||
|
const identifier = objectUtils.parseKeyString(id);
|
||||||
|
const objectProvider = this.openmct.objects.getProvider(identifier);
|
||||||
|
|
||||||
|
if (objectProvider === undefined || objectProvider.search === undefined) {
|
||||||
if (!this.indexedIds[id] && !this.pendingIndex[id]) {
|
if (!this.indexedIds[id] && !this.pendingIndex[id]) {
|
||||||
this.indexedIds[id] = true;
|
this.indexedIds[id] = true;
|
||||||
this.pendingIndex[id] = true;
|
this.pendingIndex[id] = true;
|
||||||
this.idsToIndex.push(id);
|
this.idsToIndex.push(id);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
this.keepIndexing();
|
this.keepIndexing();
|
||||||
};
|
};
|
||||||
@ -262,6 +266,7 @@ define([
|
|||||||
return {
|
return {
|
||||||
id: hit.item.id,
|
id: hit.item.id,
|
||||||
model: hit.item.model,
|
model: hit.item.model,
|
||||||
|
type: hit.item.type,
|
||||||
score: hit.matchCount
|
score: hit.matchCount
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
@ -41,7 +41,8 @@
|
|||||||
indexedItems.push({
|
indexedItems.push({
|
||||||
id: id,
|
id: id,
|
||||||
vector: vector,
|
vector: vector,
|
||||||
model: model
|
model: model,
|
||||||
|
type: model.type
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -80,12 +80,15 @@ define([
|
|||||||
* @param {Function} [filter] if provided, will be called for every
|
* @param {Function} [filter] if provided, will be called for every
|
||||||
* potential modelResult. If it returns false, the model result will be
|
* potential modelResult. If it returns false, the model result will be
|
||||||
* excluded from the search results.
|
* excluded from the search results.
|
||||||
|
* @param {AbortController.signal} abortSignal (optional) can pass in an abortSignal to cancel any
|
||||||
|
* downstream fetch requests.
|
||||||
* @returns {Promise} A Promise for a search result object.
|
* @returns {Promise} A Promise for a search result object.
|
||||||
*/
|
*/
|
||||||
SearchAggregator.prototype.query = function (
|
SearchAggregator.prototype.query = function (
|
||||||
inputText,
|
inputText,
|
||||||
maxResults,
|
maxResults,
|
||||||
filter
|
filter,
|
||||||
|
abortSignal
|
||||||
) {
|
) {
|
||||||
|
|
||||||
var aggregator = this,
|
var aggregator = this,
|
||||||
@ -120,7 +123,7 @@ define([
|
|||||||
modelResults = aggregator.applyFilter(modelResults, filter);
|
modelResults = aggregator.applyFilter(modelResults, filter);
|
||||||
modelResults = aggregator.removeDuplicates(modelResults);
|
modelResults = aggregator.removeDuplicates(modelResults);
|
||||||
|
|
||||||
return aggregator.asObjectResults(modelResults);
|
return aggregator.asObjectResults(modelResults, abortSignal);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -193,16 +196,19 @@ define([
|
|||||||
* Convert modelResults to objectResults by fetching them from the object
|
* Convert modelResults to objectResults by fetching them from the object
|
||||||
* service.
|
* service.
|
||||||
*
|
*
|
||||||
|
* @param {Object} modelResults an object containing the results from the search
|
||||||
|
* @param {AbortController.signal} abortSignal (optional) abort signal to cancel any
|
||||||
|
* downstream fetch requests
|
||||||
* @returns {Promise} for an objectResults object.
|
* @returns {Promise} for an objectResults object.
|
||||||
*/
|
*/
|
||||||
SearchAggregator.prototype.asObjectResults = function (modelResults) {
|
SearchAggregator.prototype.asObjectResults = function (modelResults, abortSignal) {
|
||||||
var objectIds = modelResults.hits.map(function (modelResult) {
|
var objectIds = modelResults.hits.map(function (modelResult) {
|
||||||
return modelResult.id;
|
return modelResult.id;
|
||||||
});
|
});
|
||||||
|
|
||||||
return this
|
return this
|
||||||
.objectService
|
.objectService
|
||||||
.getObjects(objectIds)
|
.getObjects(objectIds, abortSignal)
|
||||||
.then(function (objects) {
|
.then(function (objects) {
|
||||||
|
|
||||||
var objectResults = {
|
var objectResults = {
|
||||||
|
@ -219,7 +219,7 @@ define([
|
|||||||
* @memberof module:openmct.MCT#
|
* @memberof module:openmct.MCT#
|
||||||
* @name objects
|
* @name objects
|
||||||
*/
|
*/
|
||||||
this.objects = new api.ObjectAPI();
|
this.objects = new api.ObjectAPI.default(this.types, this);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An interface for retrieving and interpreting telemetry data associated
|
* An interface for retrieving and interpreting telemetry data associated
|
||||||
@ -283,6 +283,7 @@ define([
|
|||||||
this.install(this.plugins.NewFolderAction());
|
this.install(this.plugins.NewFolderAction());
|
||||||
this.install(this.plugins.ViewDatumAction());
|
this.install(this.plugins.ViewDatumAction());
|
||||||
this.install(this.plugins.ObjectInterceptors());
|
this.install(this.plugins.ObjectInterceptors());
|
||||||
|
this.install(this.plugins.NonEditableFolder());
|
||||||
}
|
}
|
||||||
|
|
||||||
MCT.prototype = Object.create(EventEmitter.prototype);
|
MCT.prototype = Object.create(EventEmitter.prototype);
|
||||||
@ -371,7 +372,7 @@ define([
|
|||||||
* MCT; if undefined, MCT will be run in the body of the document
|
* MCT; if undefined, MCT will be run in the body of the document
|
||||||
*/
|
*/
|
||||||
MCT.prototype.start = function (domElement = document.body, isHeadlessMode = false) {
|
MCT.prototype.start = function (domElement = document.body, isHeadlessMode = false) {
|
||||||
if (!this.plugins.DisplayLayout._installed) {
|
if (this.types.get('layout') === undefined) {
|
||||||
this.install(this.plugins.DisplayLayout({
|
this.install(this.plugins.DisplayLayout({
|
||||||
showAsView: ['summary-widget']
|
showAsView: ['summary-widget']
|
||||||
}));
|
}));
|
||||||
|
@ -32,7 +32,7 @@ define([], function () {
|
|||||||
if (Object.prototype.hasOwnProperty.call(view, 'provider')) {
|
if (Object.prototype.hasOwnProperty.call(view, 'provider')) {
|
||||||
const domainObject = legacyObject.useCapability('adapter');
|
const domainObject = legacyObject.useCapability('adapter');
|
||||||
|
|
||||||
return view.provider.canView(domainObject);
|
return view.provider.canView(domainObject, this.openmct.router.path);
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@ -61,6 +61,7 @@ define([
|
|||||||
const newStyleObject = utils.toNewFormat(legacyObject.getModel(), legacyObject.getId());
|
const newStyleObject = utils.toNewFormat(legacyObject.getModel(), legacyObject.getId());
|
||||||
const keystring = utils.makeKeyString(newStyleObject.identifier);
|
const keystring = utils.makeKeyString(newStyleObject.identifier);
|
||||||
|
|
||||||
|
this.eventEmitter.emit(keystring + ':$_synchronize_model', newStyleObject);
|
||||||
this.eventEmitter.emit(keystring + ":*", newStyleObject);
|
this.eventEmitter.emit(keystring + ":*", newStyleObject);
|
||||||
this.eventEmitter.emit('mutation', newStyleObject);
|
this.eventEmitter.emit('mutation', newStyleObject);
|
||||||
}.bind(this);
|
}.bind(this);
|
||||||
@ -138,18 +139,26 @@ define([
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
ObjectServiceProvider.prototype.superSecretFallbackSearch = function (query, abortSignal) {
|
||||||
|
const searchService = this.$injector.get('searchService');
|
||||||
|
|
||||||
|
// need to pass the abortSignal down, so need to
|
||||||
|
// pass in undefined for maxResults and filter on query
|
||||||
|
return searchService.query(query, undefined, undefined, abortSignal);
|
||||||
|
};
|
||||||
|
|
||||||
// Injects new object API as a decorator so that it hijacks all requests.
|
// Injects new object API as a decorator so that it hijacks all requests.
|
||||||
// Object providers implemented on new API should just work, old API should just work, many things may break.
|
// Object providers implemented on new API should just work, old API should just work, many things may break.
|
||||||
function LegacyObjectAPIInterceptor(openmct, ROOTS, instantiate, topic, objectService) {
|
function LegacyObjectAPIInterceptor(openmct, ROOTS, instantiate, topic, objectService) {
|
||||||
const eventEmitter = openmct.objects.eventEmitter;
|
const eventEmitter = openmct.objects.eventEmitter;
|
||||||
|
|
||||||
this.getObjects = function (keys) {
|
this.getObjects = function (keys, abortSignal) {
|
||||||
const results = {};
|
const results = {};
|
||||||
|
|
||||||
const promises = keys.map(function (keyString) {
|
const promises = keys.map(function (keyString) {
|
||||||
const key = utils.parseKeyString(keyString);
|
const key = utils.parseKeyString(keyString);
|
||||||
|
|
||||||
return openmct.objects.get(key)
|
return openmct.objects.get(key, abortSignal)
|
||||||
.then(function (object) {
|
.then(function (object) {
|
||||||
object = utils.toOldFormat(object);
|
object = utils.toOldFormat(object);
|
||||||
results[keyString] = instantiate(object, keyString);
|
results[keyString] = instantiate(object, keyString);
|
||||||
|
242
src/api/actions/ActionCollectionSpec.js
Normal file
242
src/api/actions/ActionCollectionSpec.js
Normal file
@ -0,0 +1,242 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2020, 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 ActionCollection from './ActionCollection';
|
||||||
|
import { createOpenMct, resetApplicationState } from '../../utils/testing';
|
||||||
|
|
||||||
|
describe('The ActionCollection', () => {
|
||||||
|
let openmct;
|
||||||
|
let actionCollection;
|
||||||
|
let mockApplicableActions;
|
||||||
|
let mockObjectPath;
|
||||||
|
let mockView;
|
||||||
|
let mockIdentifierService;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
openmct = createOpenMct();
|
||||||
|
openmct.$injector = jasmine.createSpyObj('$injector', ['get']);
|
||||||
|
mockIdentifierService = jasmine.createSpyObj(
|
||||||
|
'identifierService',
|
||||||
|
['parse']
|
||||||
|
);
|
||||||
|
mockIdentifierService.parse.and.returnValue({
|
||||||
|
getSpace: () => {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
openmct.$injector.get.and.returnValue(mockIdentifierService);
|
||||||
|
mockObjectPath = [
|
||||||
|
{
|
||||||
|
name: 'mock folder',
|
||||||
|
type: 'fake-folder',
|
||||||
|
identifier: {
|
||||||
|
key: 'mock-folder',
|
||||||
|
namespace: ''
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'mock parent folder',
|
||||||
|
type: 'fake-folder',
|
||||||
|
identifier: {
|
||||||
|
key: 'mock-parent-folder',
|
||||||
|
namespace: ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
];
|
||||||
|
openmct.objects.addProvider('', jasmine.createSpyObj('mockMutableObjectProvider', [
|
||||||
|
'create',
|
||||||
|
'update'
|
||||||
|
]));
|
||||||
|
mockView = {
|
||||||
|
getViewContext: () => {
|
||||||
|
return {
|
||||||
|
onlyAppliesToTestCase: true
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
mockApplicableActions = {
|
||||||
|
'test-action-object-path': {
|
||||||
|
name: 'Test Action Object Path',
|
||||||
|
key: 'test-action-object-path',
|
||||||
|
cssClass: 'test-action-object-path',
|
||||||
|
description: 'This is a test action for object path',
|
||||||
|
group: 'action',
|
||||||
|
priority: 9,
|
||||||
|
appliesTo: (objectPath) => {
|
||||||
|
if (objectPath.length) {
|
||||||
|
return objectPath[0].type === 'fake-folder';
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
invoke: () => {
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'test-action-view': {
|
||||||
|
name: 'Test Action View',
|
||||||
|
key: 'test-action-view',
|
||||||
|
cssClass: 'test-action-view',
|
||||||
|
description: 'This is a test action for view',
|
||||||
|
group: 'action',
|
||||||
|
priority: 9,
|
||||||
|
showInStatusBar: true,
|
||||||
|
appliesTo: (objectPath, view = {}) => {
|
||||||
|
if (view.getViewContext) {
|
||||||
|
let viewContext = view.getViewContext();
|
||||||
|
|
||||||
|
return viewContext.onlyAppliesToTestCase;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
invoke: () => {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
actionCollection = new ActionCollection(mockApplicableActions, mockObjectPath, mockView, openmct);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
actionCollection.destroy();
|
||||||
|
resetApplicationState(openmct);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("disable method invoked with action keys", () => {
|
||||||
|
it("marks those actions as isDisabled", () => {
|
||||||
|
let actionKey = 'test-action-object-path';
|
||||||
|
let actionsObject = actionCollection.getActionsObject();
|
||||||
|
let action = actionsObject[actionKey];
|
||||||
|
|
||||||
|
expect(action.isDisabled).toBeFalsy();
|
||||||
|
|
||||||
|
actionCollection.disable([actionKey]);
|
||||||
|
actionsObject = actionCollection.getActionsObject();
|
||||||
|
action = actionsObject[actionKey];
|
||||||
|
|
||||||
|
expect(action.isDisabled).toBeTrue();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("enable method invoked with action keys", () => {
|
||||||
|
it("marks the isDisabled property as false", () => {
|
||||||
|
let actionKey = 'test-action-object-path';
|
||||||
|
|
||||||
|
actionCollection.disable([actionKey]);
|
||||||
|
|
||||||
|
let actionsObject = actionCollection.getActionsObject();
|
||||||
|
let action = actionsObject[actionKey];
|
||||||
|
|
||||||
|
expect(action.isDisabled).toBeTrue();
|
||||||
|
|
||||||
|
actionCollection.enable([actionKey]);
|
||||||
|
actionsObject = actionCollection.getActionsObject();
|
||||||
|
action = actionsObject[actionKey];
|
||||||
|
|
||||||
|
expect(action.isDisabled).toBeFalse();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("hide method invoked with action keys", () => {
|
||||||
|
it("marks those actions as isHidden", () => {
|
||||||
|
let actionKey = 'test-action-object-path';
|
||||||
|
let actionsObject = actionCollection.getActionsObject();
|
||||||
|
let action = actionsObject[actionKey];
|
||||||
|
|
||||||
|
expect(action.isHidden).toBeFalsy();
|
||||||
|
|
||||||
|
actionCollection.hide([actionKey]);
|
||||||
|
actionsObject = actionCollection.getActionsObject();
|
||||||
|
action = actionsObject[actionKey];
|
||||||
|
|
||||||
|
expect(action.isHidden).toBeTrue();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("show method invoked with action keys", () => {
|
||||||
|
it("marks the isHidden property as false", () => {
|
||||||
|
let actionKey = 'test-action-object-path';
|
||||||
|
|
||||||
|
actionCollection.hide([actionKey]);
|
||||||
|
|
||||||
|
let actionsObject = actionCollection.getActionsObject();
|
||||||
|
let action = actionsObject[actionKey];
|
||||||
|
|
||||||
|
expect(action.isHidden).toBeTrue();
|
||||||
|
|
||||||
|
actionCollection.show([actionKey]);
|
||||||
|
actionsObject = actionCollection.getActionsObject();
|
||||||
|
action = actionsObject[actionKey];
|
||||||
|
|
||||||
|
expect(action.isHidden).toBeFalse();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getVisibleActions method", () => {
|
||||||
|
it("returns an array of non hidden actions", () => {
|
||||||
|
let action1Key = 'test-action-object-path';
|
||||||
|
let action2Key = 'test-action-view';
|
||||||
|
|
||||||
|
actionCollection.hide([action1Key]);
|
||||||
|
|
||||||
|
let visibleActions = actionCollection.getVisibleActions();
|
||||||
|
|
||||||
|
expect(Array.isArray(visibleActions)).toBeTrue();
|
||||||
|
expect(visibleActions.length).toEqual(1);
|
||||||
|
expect(visibleActions[0].key).toEqual(action2Key);
|
||||||
|
|
||||||
|
actionCollection.show([action1Key]);
|
||||||
|
visibleActions = actionCollection.getVisibleActions();
|
||||||
|
|
||||||
|
expect(visibleActions.length).toEqual(2);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getStatusBarActions method", () => {
|
||||||
|
it("returns an array of non disabled, non hidden statusBar actions", () => {
|
||||||
|
let action2Key = 'test-action-view';
|
||||||
|
|
||||||
|
let statusBarActions = actionCollection.getStatusBarActions();
|
||||||
|
|
||||||
|
expect(Array.isArray(statusBarActions)).toBeTrue();
|
||||||
|
expect(statusBarActions.length).toEqual(1);
|
||||||
|
expect(statusBarActions[0].key).toEqual(action2Key);
|
||||||
|
|
||||||
|
actionCollection.disable([action2Key]);
|
||||||
|
statusBarActions = actionCollection.getStatusBarActions();
|
||||||
|
|
||||||
|
expect(statusBarActions.length).toEqual(0);
|
||||||
|
|
||||||
|
actionCollection.enable([action2Key]);
|
||||||
|
statusBarActions = actionCollection.getStatusBarActions();
|
||||||
|
|
||||||
|
expect(statusBarActions.length).toEqual(1);
|
||||||
|
expect(statusBarActions[0].key).toEqual(action2Key);
|
||||||
|
|
||||||
|
actionCollection.hide([action2Key]);
|
||||||
|
statusBarActions = actionCollection.getStatusBarActions();
|
||||||
|
|
||||||
|
expect(statusBarActions.length).toEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
@ -22,22 +22,41 @@
|
|||||||
|
|
||||||
import ActionsAPI from './ActionsAPI';
|
import ActionsAPI from './ActionsAPI';
|
||||||
import { createOpenMct, resetApplicationState } from '../../utils/testing';
|
import { createOpenMct, resetApplicationState } from '../../utils/testing';
|
||||||
|
import ActionCollection from './ActionCollection';
|
||||||
|
|
||||||
describe('The Actions API', () => {
|
describe('The Actions API', () => {
|
||||||
let openmct;
|
let openmct;
|
||||||
let actionsAPI;
|
let actionsAPI;
|
||||||
let mockAction;
|
let mockAction;
|
||||||
let mockObjectPath;
|
let mockObjectPath;
|
||||||
|
let mockObjectPathAction;
|
||||||
let mockViewContext1;
|
let mockViewContext1;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
openmct = createOpenMct();
|
openmct = createOpenMct();
|
||||||
actionsAPI = new ActionsAPI(openmct);
|
actionsAPI = new ActionsAPI(openmct);
|
||||||
|
mockObjectPathAction = {
|
||||||
|
name: 'Test Action Object Path',
|
||||||
|
key: 'test-action-object-path',
|
||||||
|
cssClass: 'test-action-object-path',
|
||||||
|
description: 'This is a test action for object path',
|
||||||
|
group: 'action',
|
||||||
|
priority: 9,
|
||||||
|
appliesTo: (objectPath) => {
|
||||||
|
if (objectPath.length) {
|
||||||
|
return objectPath[0].type === 'fake-folder';
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
invoke: () => {
|
||||||
|
}
|
||||||
|
};
|
||||||
mockAction = {
|
mockAction = {
|
||||||
name: 'Test Action',
|
name: 'Test Action View',
|
||||||
key: 'test-action',
|
key: 'test-action-view',
|
||||||
cssClass: 'test-action',
|
cssClass: 'test-action-view',
|
||||||
description: 'This is a test action',
|
description: 'This is a test action for view',
|
||||||
group: 'action',
|
group: 'action',
|
||||||
priority: 9,
|
priority: 9,
|
||||||
appliesTo: (objectPath, view = {}) => {
|
appliesTo: (objectPath, view = {}) => {
|
||||||
@ -45,8 +64,6 @@ describe('The Actions API', () => {
|
|||||||
let viewContext = view.getViewContext();
|
let viewContext = view.getViewContext();
|
||||||
|
|
||||||
return viewContext.onlyAppliesToTestCase;
|
return viewContext.onlyAppliesToTestCase;
|
||||||
} else if (objectPath.length) {
|
|
||||||
return objectPath[0].type === 'fake-folder';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
@ -100,9 +117,32 @@ describe('The Actions API', () => {
|
|||||||
describe("get method", () => {
|
describe("get method", () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
actionsAPI.register(mockAction);
|
actionsAPI.register(mockAction);
|
||||||
|
actionsAPI.register(mockObjectPathAction);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns an object with relevant actions when invoked with objectPath only", () => {
|
it("returns an ActionCollection when invoked with an objectPath only", () => {
|
||||||
|
let actionCollection = actionsAPI.get(mockObjectPath);
|
||||||
|
let instanceOfActionCollection = actionCollection instanceof ActionCollection;
|
||||||
|
|
||||||
|
expect(instanceOfActionCollection).toBeTrue();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns an ActionCollection when invoked with an objectPath and view", () => {
|
||||||
|
let actionCollection = actionsAPI.get(mockObjectPath, mockViewContext1);
|
||||||
|
let instanceOfActionCollection = actionCollection instanceof ActionCollection;
|
||||||
|
|
||||||
|
expect(instanceOfActionCollection).toBeTrue();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns relevant actions when invoked with objectPath only", () => {
|
||||||
|
let actionCollection = actionsAPI.get(mockObjectPath);
|
||||||
|
let action = actionCollection.getActionsObject()[mockObjectPathAction.key];
|
||||||
|
|
||||||
|
expect(action.key).toEqual(mockObjectPathAction.key);
|
||||||
|
expect(action.name).toEqual(mockObjectPathAction.name);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns relevant actions when invoked with objectPath and view", () => {
|
||||||
let actionCollection = actionsAPI.get(mockObjectPath, mockViewContext1);
|
let actionCollection = actionsAPI.get(mockObjectPath, mockViewContext1);
|
||||||
let action = actionCollection.getActionsObject()[mockAction.key];
|
let action = actionCollection.getActionsObject()[mockAction.key];
|
||||||
|
|
||||||
|
@ -60,6 +60,17 @@ define([
|
|||||||
};
|
};
|
||||||
this.onProviderAdd = this.onProviderAdd.bind(this);
|
this.onProviderAdd = this.onProviderAdd.bind(this);
|
||||||
this.onProviderRemove = this.onProviderRemove.bind(this);
|
this.onProviderRemove = this.onProviderRemove.bind(this);
|
||||||
|
this.mutables = {};
|
||||||
|
|
||||||
|
if (this.domainObject.isMutable) {
|
||||||
|
this.returnMutables = true;
|
||||||
|
let unobserve = this.domainObject.$on('$_destroy', () => {
|
||||||
|
Object.values(this.mutables).forEach(mutable => {
|
||||||
|
this.publicAPI.objects.destroyMutable(mutable);
|
||||||
|
});
|
||||||
|
unobserve();
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -75,10 +86,6 @@ define([
|
|||||||
throw new Error('Event not supported by composition: ' + event);
|
throw new Error('Event not supported by composition: ' + event);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!this.mutationListener) {
|
|
||||||
this._synchronize();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.provider.on && this.provider.off) {
|
if (this.provider.on && this.provider.off) {
|
||||||
if (event === 'add') {
|
if (event === 'add') {
|
||||||
this.provider.on(
|
this.provider.on(
|
||||||
@ -189,6 +196,13 @@ define([
|
|||||||
|
|
||||||
this.provider.add(this.domainObject, child.identifier);
|
this.provider.add(this.domainObject, child.identifier);
|
||||||
} else {
|
} else {
|
||||||
|
if (this.returnMutables && this.publicAPI.objects.supportsMutation(child.identifier)) {
|
||||||
|
let keyString = this.publicAPI.objects.makeKeyString(child.identifier);
|
||||||
|
|
||||||
|
child = this.publicAPI.objects._toMutable(child);
|
||||||
|
this.mutables[keyString] = child;
|
||||||
|
}
|
||||||
|
|
||||||
this.emit('add', child);
|
this.emit('add', child);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -202,6 +216,8 @@ define([
|
|||||||
* @name load
|
* @name load
|
||||||
*/
|
*/
|
||||||
CompositionCollection.prototype.load = function () {
|
CompositionCollection.prototype.load = function () {
|
||||||
|
this.cleanUpMutables();
|
||||||
|
|
||||||
return this.provider.load(this.domainObject)
|
return this.provider.load(this.domainObject)
|
||||||
.then(function (children) {
|
.then(function (children) {
|
||||||
return Promise.all(children.map((c) => this.publicAPI.objects.get(c)));
|
return Promise.all(children.map((c) => this.publicAPI.objects.get(c)));
|
||||||
@ -234,6 +250,14 @@ define([
|
|||||||
if (!skipMutate) {
|
if (!skipMutate) {
|
||||||
this.provider.remove(this.domainObject, child.identifier);
|
this.provider.remove(this.domainObject, child.identifier);
|
||||||
} else {
|
} else {
|
||||||
|
if (this.returnMutables) {
|
||||||
|
let keyString = this.publicAPI.objects.makeKeyString(child);
|
||||||
|
if (this.mutables[keyString] !== undefined && this.mutables[keyString].isMutable) {
|
||||||
|
this.publicAPI.objects.destroyMutable(this.mutables[keyString]);
|
||||||
|
delete this.mutables[keyString];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
this.emit('remove', child);
|
this.emit('remove', child);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -281,12 +305,6 @@ define([
|
|||||||
this.remove(child, true);
|
this.remove(child, true);
|
||||||
};
|
};
|
||||||
|
|
||||||
CompositionCollection.prototype._synchronize = function () {
|
|
||||||
this.mutationListener = this.publicAPI.objects.observe(this.domainObject, '*', (newDomainObject) => {
|
|
||||||
this.domainObject = JSON.parse(JSON.stringify(newDomainObject));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
CompositionCollection.prototype._destroy = function () {
|
CompositionCollection.prototype._destroy = function () {
|
||||||
if (this.mutationListener) {
|
if (this.mutationListener) {
|
||||||
this.mutationListener();
|
this.mutationListener();
|
||||||
@ -308,5 +326,11 @@ define([
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
CompositionCollection.prototype.cleanUpMutables = function () {
|
||||||
|
Object.values(this.mutables).forEach(mutable => {
|
||||||
|
this.publicAPI.objects.destroyMutable(mutable);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
return CompositionCollection;
|
return CompositionCollection;
|
||||||
});
|
});
|
||||||
|
@ -38,7 +38,7 @@ class MenuAPI {
|
|||||||
this._showObjectMenu = this._showObjectMenu.bind(this);
|
this._showObjectMenu = this._showObjectMenu.bind(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
showMenu(x, y, actions) {
|
showMenu(x, y, actions, onDestroy) {
|
||||||
if (this.menuComponent) {
|
if (this.menuComponent) {
|
||||||
this.menuComponent.dismiss();
|
this.menuComponent.dismiss();
|
||||||
}
|
}
|
||||||
@ -46,7 +46,8 @@ class MenuAPI {
|
|||||||
let options = {
|
let options = {
|
||||||
x,
|
x,
|
||||||
y,
|
y,
|
||||||
actions
|
actions,
|
||||||
|
onDestroy
|
||||||
};
|
};
|
||||||
|
|
||||||
this.menuComponent = new Menu(options);
|
this.menuComponent = new Menu(options);
|
||||||
|
@ -31,6 +31,7 @@ describe ('The Menu API', () => {
|
|||||||
let x;
|
let x;
|
||||||
let y;
|
let y;
|
||||||
let result;
|
let result;
|
||||||
|
let onDestroy;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
openmct = createOpenMct();
|
openmct = createOpenMct();
|
||||||
@ -73,7 +74,9 @@ describe ('The Menu API', () => {
|
|||||||
let vueComponent;
|
let vueComponent;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
menuAPI.showMenu(x, y, actionsArray);
|
onDestroy = jasmine.createSpy('onDestroy');
|
||||||
|
|
||||||
|
menuAPI.showMenu(x, y, actionsArray, onDestroy);
|
||||||
vueComponent = menuAPI.menuComponent.component;
|
vueComponent = menuAPI.menuComponent.component;
|
||||||
menuComponent = document.querySelector(".c-menu");
|
menuComponent = document.querySelector(".c-menu");
|
||||||
|
|
||||||
@ -120,6 +123,12 @@ describe ('The Menu API', () => {
|
|||||||
|
|
||||||
expect(vueComponent.$destroy).toHaveBeenCalled();
|
expect(vueComponent.$destroy).toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("invokes the onDestroy callback if passed in", () => {
|
||||||
|
document.body.click();
|
||||||
|
|
||||||
|
expect(onDestroy).toHaveBeenCalled();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -30,12 +30,12 @@ class Menu extends EventEmitter {
|
|||||||
this.options = options;
|
this.options = options;
|
||||||
|
|
||||||
this.component = new Vue({
|
this.component = new Vue({
|
||||||
provide: {
|
|
||||||
actions: options.actions
|
|
||||||
},
|
|
||||||
components: {
|
components: {
|
||||||
MenuComponent
|
MenuComponent
|
||||||
},
|
},
|
||||||
|
provide: {
|
||||||
|
actions: options.actions
|
||||||
|
},
|
||||||
template: '<menu-component />'
|
template: '<menu-component />'
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -75,13 +75,20 @@ export default class NotificationAPI extends EventEmitter {
|
|||||||
* Info notifications are low priority informational messages for the user. They will be auto-destroy after a brief
|
* Info notifications are low priority informational messages for the user. They will be auto-destroy after a brief
|
||||||
* period of time.
|
* period of time.
|
||||||
* @param {string} message The message to display to the user
|
* @param {string} message The message to display to the user
|
||||||
|
* @param {Object} [options] object with following properties
|
||||||
|
* autoDismissTimeout: {number} in miliseconds 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
|
||||||
* @returns {InfoNotification}
|
* @returns {InfoNotification}
|
||||||
*/
|
*/
|
||||||
info(message) {
|
info(message, options = {}) {
|
||||||
let notificationModel = {
|
let notificationModel = {
|
||||||
message: message,
|
message: message,
|
||||||
autoDismiss: true,
|
autoDismiss: true,
|
||||||
severity: "info"
|
severity: "info",
|
||||||
|
options
|
||||||
};
|
};
|
||||||
|
|
||||||
return this._notify(notificationModel);
|
return this._notify(notificationModel);
|
||||||
@ -90,12 +97,19 @@ export default class NotificationAPI extends EventEmitter {
|
|||||||
/**
|
/**
|
||||||
* Present an alert to the user.
|
* Present an alert to the user.
|
||||||
* @param {string} message The message to display to the user.
|
* @param {string} message The message to display to the user.
|
||||||
|
* @param {Object} [options] object with following properties
|
||||||
|
* autoDismissTimeout: {number} in miliseconds 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
|
||||||
* @returns {Notification}
|
* @returns {Notification}
|
||||||
*/
|
*/
|
||||||
alert(message) {
|
alert(message, options = {}) {
|
||||||
let notificationModel = {
|
let notificationModel = {
|
||||||
message: message,
|
message: message,
|
||||||
severity: "alert"
|
severity: "alert",
|
||||||
|
options
|
||||||
};
|
};
|
||||||
|
|
||||||
return this._notify(notificationModel);
|
return this._notify(notificationModel);
|
||||||
@ -104,12 +118,19 @@ export default class NotificationAPI extends EventEmitter {
|
|||||||
/**
|
/**
|
||||||
* Present an error message to the user
|
* Present an error message to the user
|
||||||
* @param {string} message
|
* @param {string} message
|
||||||
|
* @param {Object} [options] object with following properties
|
||||||
|
* autoDismissTimeout: {number} in miliseconds 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
|
||||||
* @returns {Notification}
|
* @returns {Notification}
|
||||||
*/
|
*/
|
||||||
error(message) {
|
error(message, options = {}) {
|
||||||
let notificationModel = {
|
let notificationModel = {
|
||||||
message: message,
|
message: message,
|
||||||
severity: "error"
|
severity: "error",
|
||||||
|
options
|
||||||
};
|
};
|
||||||
|
|
||||||
return this._notify(notificationModel);
|
return this._notify(notificationModel);
|
||||||
@ -325,9 +346,11 @@ export default class NotificationAPI extends EventEmitter {
|
|||||||
this.emit('notification', notification);
|
this.emit('notification', notification);
|
||||||
|
|
||||||
if (notification.model.autoDismiss || this._selectNextNotification()) {
|
if (notification.model.autoDismiss || this._selectNextNotification()) {
|
||||||
|
const autoDismissTimeout = notification.model.options.autoDismissTimeout
|
||||||
|
|| DEFAULT_AUTO_DISMISS_TIMEOUT;
|
||||||
this.activeTimeout = setTimeout(() => {
|
this.activeTimeout = setTimeout(() => {
|
||||||
this._dismissOrMinimize(notification);
|
this._dismissOrMinimize(notification);
|
||||||
}, DEFAULT_AUTO_DISMISS_TIMEOUT);
|
}, autoDismissTimeout);
|
||||||
} else {
|
} else {
|
||||||
delete this.activeTimeout;
|
delete this.activeTimeout;
|
||||||
}
|
}
|
||||||
|
154
src/api/notifications/NotificationAPISpec.js
Normal file
154
src/api/notifications/NotificationAPISpec.js
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2020, 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 NotificationAPI from './NotificationAPI';
|
||||||
|
|
||||||
|
describe('The Notifiation API', () => {
|
||||||
|
let notificationAPIInstance;
|
||||||
|
let defaultTimeout = 4000;
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
notificationAPIInstance = new NotificationAPI();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('the info method', () => {
|
||||||
|
let message = 'Example Notification Message';
|
||||||
|
let severity = 'info';
|
||||||
|
let notificationModel;
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
notificationModel = notificationAPIInstance.info(message).model;
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
notificationAPIInstance.dismissAllNotifications();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('shows a string message with info severity', () => {
|
||||||
|
expect(notificationModel.message).toEqual(message);
|
||||||
|
expect(notificationModel.severity).toEqual(severity);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('auto dismisses the notification after a brief timeout', (done) => {
|
||||||
|
window.setTimeout(() => {
|
||||||
|
expect(notificationAPIInstance.notifications.length).toEqual(0);
|
||||||
|
done();
|
||||||
|
}, defaultTimeout);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('the alert method', () => {
|
||||||
|
let message = 'Example alert message';
|
||||||
|
let severity = 'alert';
|
||||||
|
let notificationModel;
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
notificationModel = notificationAPIInstance.alert(message).model;
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
notificationAPIInstance.dismissAllNotifications();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('shows a string message, with alert severity', () => {
|
||||||
|
expect(notificationModel.message).toEqual(message);
|
||||||
|
expect(notificationModel.severity).toEqual(severity);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not auto dismiss the notification', (done) => {
|
||||||
|
window.setTimeout(() => {
|
||||||
|
expect(notificationAPIInstance.notifications.length).toEqual(1);
|
||||||
|
done();
|
||||||
|
}, defaultTimeout);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('the error method', () => {
|
||||||
|
let message = 'Example error message';
|
||||||
|
let severity = 'error';
|
||||||
|
let notificationModel;
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
notificationModel = notificationAPIInstance.error(message).model;
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
notificationAPIInstance.dismissAllNotifications();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('shows a string message, with severity error', () => {
|
||||||
|
expect(notificationModel.message).toEqual(message);
|
||||||
|
expect(notificationModel.severity).toEqual(severity);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not auto dismiss the notification', (done) => {
|
||||||
|
window.setTimeout(() => {
|
||||||
|
expect(notificationAPIInstance.notifications.length).toEqual(1);
|
||||||
|
done();
|
||||||
|
}, defaultTimeout);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('the progress method', () => {
|
||||||
|
let title = 'This is a progress notification';
|
||||||
|
let message1 = 'Example progress message 1';
|
||||||
|
let message2 = 'Example progress message 2';
|
||||||
|
let percentage1 = 50;
|
||||||
|
let percentage2 = 99.9;
|
||||||
|
let severity = 'info';
|
||||||
|
let notification;
|
||||||
|
let updatedPercentage;
|
||||||
|
let updatedMessage;
|
||||||
|
|
||||||
|
beforeAll(() => {
|
||||||
|
notification = notificationAPIInstance.progress(title, percentage1, message1);
|
||||||
|
notification.on('progress', (percentage, text) => {
|
||||||
|
updatedPercentage = percentage;
|
||||||
|
updatedMessage = text;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
notificationAPIInstance.dismissAllNotifications();
|
||||||
|
});
|
||||||
|
|
||||||
|
it ('shows a notification with a message, progress message, percentage and info severity', () => {
|
||||||
|
expect(notification.model.message).toEqual(title);
|
||||||
|
expect(notification.model.severity).toEqual(severity);
|
||||||
|
expect(notification.model.progressText).toEqual(message1);
|
||||||
|
expect(notification.model.progressPerc).toEqual(percentage1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it ('allows dynamically updating the progress attributes', () => {
|
||||||
|
notification.progress(percentage2, message2);
|
||||||
|
|
||||||
|
expect(updatedPercentage).toEqual(percentage2);
|
||||||
|
expect(updatedMessage).toEqual(message2);
|
||||||
|
});
|
||||||
|
|
||||||
|
it ('allows dynamically dismissing of progress notification', () => {
|
||||||
|
notification.dismiss();
|
||||||
|
|
||||||
|
expect(notificationAPIInstance.notifications.length).toEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
147
src/api/objects/MutableDomainObject.js
Normal file
147
src/api/objects/MutableDomainObject.js
Normal file
@ -0,0 +1,147 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2020, 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 _ from 'lodash';
|
||||||
|
import utils from './object-utils.js';
|
||||||
|
import EventEmitter from 'EventEmitter';
|
||||||
|
|
||||||
|
const ANY_OBJECT_EVENT = 'mutation';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wraps a domain object to keep its model synchronized with other instances of the same object.
|
||||||
|
*
|
||||||
|
* Creating a MutableDomainObject will automatically register listeners to keep its model in sync. As such, developers
|
||||||
|
* should be careful to destroy MutableDomainObject in order to avoid memory leaks.
|
||||||
|
*
|
||||||
|
* All Open MCT API functions that provide objects will provide MutableDomainObjects where possible, except
|
||||||
|
* `openmct.objects.get()`, and will manage that object's lifecycle for you. Calling `openmct.objects.getMutable()`
|
||||||
|
* will result in the creation of a new MutableDomainObject and you will be responsible for destroying it
|
||||||
|
* (via openmct.objects.destroy) when you're done with it.
|
||||||
|
*
|
||||||
|
* @typedef MutableDomainObject
|
||||||
|
* @memberof module:openmct
|
||||||
|
*/
|
||||||
|
class MutableDomainObject {
|
||||||
|
constructor(eventEmitter) {
|
||||||
|
Object.defineProperties(this, {
|
||||||
|
_globalEventEmitter: {
|
||||||
|
value: eventEmitter,
|
||||||
|
// Property should not be serialized
|
||||||
|
enumerable: false
|
||||||
|
},
|
||||||
|
_instanceEventEmitter: {
|
||||||
|
value: new EventEmitter(),
|
||||||
|
// Property should not be serialized
|
||||||
|
enumerable: false
|
||||||
|
},
|
||||||
|
_observers: {
|
||||||
|
value: [],
|
||||||
|
// Property should not be serialized
|
||||||
|
enumerable: false
|
||||||
|
},
|
||||||
|
isMutable: {
|
||||||
|
value: true,
|
||||||
|
// Property should not be serialized
|
||||||
|
enumerable: false
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
$observe(path, callback) {
|
||||||
|
let fullPath = qualifiedEventName(this, path);
|
||||||
|
let eventOff =
|
||||||
|
this._globalEventEmitter.off.bind(this._globalEventEmitter, fullPath, callback);
|
||||||
|
|
||||||
|
this._globalEventEmitter.on(fullPath, callback);
|
||||||
|
this._observers.push(eventOff);
|
||||||
|
|
||||||
|
return eventOff;
|
||||||
|
}
|
||||||
|
$set(path, value) {
|
||||||
|
_.set(this, path, value);
|
||||||
|
_.set(this, 'modified', Date.now());
|
||||||
|
|
||||||
|
//Emit secret synchronization event first, so that all objects are in sync before subsequent events fired.
|
||||||
|
this._globalEventEmitter.emit(qualifiedEventName(this, '$_synchronize_model'), this);
|
||||||
|
|
||||||
|
//Emit a general "any object" event
|
||||||
|
this._globalEventEmitter.emit(ANY_OBJECT_EVENT, this);
|
||||||
|
//Emit wildcard event, with path so that callback knows what changed
|
||||||
|
this._globalEventEmitter.emit(qualifiedEventName(this, '*'), this, path, value);
|
||||||
|
|
||||||
|
//Emit events specific to properties affected
|
||||||
|
let parentPropertiesList = path.split('.');
|
||||||
|
for (let index = parentPropertiesList.length; index > 0; index--) {
|
||||||
|
let parentPropertyPath = parentPropertiesList.slice(0, index).join('.');
|
||||||
|
this._globalEventEmitter.emit(qualifiedEventName(this, parentPropertyPath), _.get(this, parentPropertyPath));
|
||||||
|
}
|
||||||
|
|
||||||
|
//TODO: Emit events for listeners of child properties when parent changes.
|
||||||
|
// Do it at observer time - also register observers for parent attribute path.
|
||||||
|
}
|
||||||
|
|
||||||
|
$refresh(model) {
|
||||||
|
//TODO: Currently we are updating the entire object.
|
||||||
|
// In the future we could update a specific property of the object using the 'path' parameter.
|
||||||
|
this._globalEventEmitter.emit(qualifiedEventName(this, '$_synchronize_model'), model);
|
||||||
|
|
||||||
|
//Emit wildcard event, with path so that callback knows what changed
|
||||||
|
this._globalEventEmitter.emit(qualifiedEventName(this, '*'), this);
|
||||||
|
}
|
||||||
|
|
||||||
|
$on(event, callback) {
|
||||||
|
this._instanceEventEmitter.on(event, callback);
|
||||||
|
|
||||||
|
return () => this._instanceEventEmitter.off(event, callback);
|
||||||
|
}
|
||||||
|
$destroy() {
|
||||||
|
this._observers.forEach(observer => observer());
|
||||||
|
delete this._globalEventEmitter;
|
||||||
|
delete this._observers;
|
||||||
|
this._instanceEventEmitter.emit('$_destroy');
|
||||||
|
}
|
||||||
|
|
||||||
|
static createMutable(object, mutationTopic) {
|
||||||
|
let mutable = Object.create(new MutableDomainObject(mutationTopic));
|
||||||
|
Object.assign(mutable, object);
|
||||||
|
|
||||||
|
mutable.$observe('$_synchronize_model', (updatedObject) => {
|
||||||
|
let clone = JSON.parse(JSON.stringify(updatedObject));
|
||||||
|
let deleted = _.difference(Object.keys(mutable), Object.keys(updatedObject));
|
||||||
|
deleted.forEach((propertyName) => delete mutable[propertyName]);
|
||||||
|
Object.assign(mutable, clone);
|
||||||
|
});
|
||||||
|
|
||||||
|
return mutable;
|
||||||
|
}
|
||||||
|
|
||||||
|
static mutateObject(object, path, value) {
|
||||||
|
_.set(object, path, value);
|
||||||
|
_.set(object, 'modified', Date.now());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function qualifiedEventName(object, eventName) {
|
||||||
|
let keystring = utils.makeKeyString(object.identifier);
|
||||||
|
|
||||||
|
return [keystring, eventName].join(':');
|
||||||
|
}
|
||||||
|
|
||||||
|
export default MutableDomainObject;
|
@ -1,102 +0,0 @@
|
|||||||
/*****************************************************************************
|
|
||||||
* Open MCT, Copyright (c) 2014-2020, 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.
|
|
||||||
*****************************************************************************/
|
|
||||||
|
|
||||||
define([
|
|
||||||
'objectUtils',
|
|
||||||
'lodash'
|
|
||||||
], function (
|
|
||||||
utils,
|
|
||||||
_
|
|
||||||
) {
|
|
||||||
const ANY_OBJECT_EVENT = "mutation";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The MutableObject wraps a DomainObject and provides getters and
|
|
||||||
* setters for
|
|
||||||
* @param eventEmitter
|
|
||||||
* @param object
|
|
||||||
* @interface MutableObject
|
|
||||||
*/
|
|
||||||
function MutableObject(eventEmitter, object) {
|
|
||||||
this.eventEmitter = eventEmitter;
|
|
||||||
this.object = object;
|
|
||||||
this.unlisteners = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
function qualifiedEventName(object, eventName) {
|
|
||||||
const keystring = utils.makeKeyString(object.identifier);
|
|
||||||
|
|
||||||
return [keystring, eventName].join(':');
|
|
||||||
}
|
|
||||||
|
|
||||||
MutableObject.prototype.stopListening = function () {
|
|
||||||
this.unlisteners.forEach(function (unlisten) {
|
|
||||||
unlisten();
|
|
||||||
});
|
|
||||||
this.unlisteners = [];
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Observe changes to this domain object.
|
|
||||||
* @param {string} path the property to observe
|
|
||||||
* @param {Function} callback a callback to invoke when new values for
|
|
||||||
* this property are observed
|
|
||||||
* @method on
|
|
||||||
* @memberof module:openmct.MutableObject#
|
|
||||||
*/
|
|
||||||
MutableObject.prototype.on = function (path, callback) {
|
|
||||||
const fullPath = qualifiedEventName(this.object, path);
|
|
||||||
const eventOff =
|
|
||||||
this.eventEmitter.off.bind(this.eventEmitter, fullPath, callback);
|
|
||||||
|
|
||||||
this.eventEmitter.on(fullPath, callback);
|
|
||||||
this.unlisteners.push(eventOff);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Modify this domain object.
|
|
||||||
* @param {string} path the property to modify
|
|
||||||
* @param {*} value the new value for this property
|
|
||||||
* @method set
|
|
||||||
* @memberof module:openmct.MutableObject#
|
|
||||||
*/
|
|
||||||
MutableObject.prototype.set = function (path, value) {
|
|
||||||
_.set(this.object, path, value);
|
|
||||||
_.set(this.object, 'modified', Date.now());
|
|
||||||
|
|
||||||
const handleRecursiveMutation = function (newObject) {
|
|
||||||
this.object = newObject;
|
|
||||||
}.bind(this);
|
|
||||||
|
|
||||||
//Emit wildcard event
|
|
||||||
this.eventEmitter.emit(qualifiedEventName(this.object, '*'), this.object);
|
|
||||||
//Emit a general "any object" event
|
|
||||||
this.eventEmitter.emit(ANY_OBJECT_EVENT, this.object);
|
|
||||||
|
|
||||||
this.eventEmitter.on(qualifiedEventName(this.object, '*'), handleRecursiveMutation);
|
|
||||||
//Emit event specific to property
|
|
||||||
this.eventEmitter.emit(qualifiedEventName(this.object, path), value);
|
|
||||||
this.eventEmitter.off(qualifiedEventName(this.object, '*'), handleRecursiveMutation);
|
|
||||||
};
|
|
||||||
|
|
||||||
return MutableObject;
|
|
||||||
});
|
|
@ -20,68 +20,79 @@
|
|||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
define([
|
import utils from 'objectUtils';
|
||||||
'lodash',
|
import MutableDomainObject from './MutableDomainObject';
|
||||||
'objectUtils',
|
import RootRegistry from './RootRegistry';
|
||||||
'./MutableObject',
|
import RootObjectProvider from './RootObjectProvider';
|
||||||
'./RootRegistry',
|
import EventEmitter from 'EventEmitter';
|
||||||
'./RootObjectProvider',
|
import InterceptorRegistry from './InterceptorRegistry';
|
||||||
'./InterceptorRegistry',
|
|
||||||
'EventEmitter'
|
|
||||||
], function (
|
|
||||||
_,
|
|
||||||
utils,
|
|
||||||
MutableObject,
|
|
||||||
RootRegistry,
|
|
||||||
RootObjectProvider,
|
|
||||||
InterceptorRegistry,
|
|
||||||
EventEmitter
|
|
||||||
) {
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utilities for loading, saving, and manipulating domain objects.
|
* Utilities for loading, saving, and manipulating domain objects.
|
||||||
* @interface ObjectAPI
|
* @interface ObjectAPI
|
||||||
* @memberof module:openmct
|
* @memberof module:openmct
|
||||||
*/
|
*/
|
||||||
|
|
||||||
function ObjectAPI() {
|
function ObjectAPI(typeRegistry, openmct) {
|
||||||
|
this.typeRegistry = typeRegistry;
|
||||||
this.eventEmitter = new EventEmitter();
|
this.eventEmitter = new EventEmitter();
|
||||||
this.providers = {};
|
this.providers = {};
|
||||||
this.rootRegistry = new RootRegistry();
|
this.rootRegistry = new RootRegistry();
|
||||||
this.rootProvider = new RootObjectProvider.default(this.rootRegistry);
|
this.injectIdentifierService = function () {
|
||||||
this.cache = {};
|
this.identifierService = openmct.$injector.get("identifierService");
|
||||||
this.interceptorRegistry = new InterceptorRegistry.default();
|
};
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
this.rootProvider = new RootObjectProvider(this.rootRegistry);
|
||||||
|
this.cache = {};
|
||||||
|
this.interceptorRegistry = new InterceptorRegistry();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
* Set fallback provider, this is an internal API for legacy reasons.
|
* Set fallback provider, this is an internal API for legacy reasons.
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
ObjectAPI.prototype.supersecretSetFallbackProvider = function (p) {
|
ObjectAPI.prototype.supersecretSetFallbackProvider = function (p) {
|
||||||
this.fallbackProvider = p;
|
this.fallbackProvider = p;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
ObjectAPI.prototype.getIdentifierService = function () {
|
||||||
|
// Lazily acquire identifier service
|
||||||
|
if (!this.identifierService) {
|
||||||
|
this.injectIdentifierService();
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.identifierService;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
* Retrieve the provider for a given identifier.
|
* Retrieve the provider for a given identifier.
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
ObjectAPI.prototype.getProvider = function (identifier) {
|
ObjectAPI.prototype.getProvider = function (identifier) {
|
||||||
|
//handles the '' vs 'mct' namespace issue
|
||||||
|
const keyString = utils.makeKeyString(identifier);
|
||||||
|
const identifierService = this.getIdentifierService();
|
||||||
|
const namespace = identifierService.parse(keyString).getSpace();
|
||||||
|
|
||||||
if (identifier.key === 'ROOT') {
|
if (identifier.key === 'ROOT') {
|
||||||
return this.rootProvider;
|
return this.rootProvider;
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.providers[identifier.namespace] || this.fallbackProvider;
|
return this.providers[namespace] || this.fallbackProvider;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the root-level object.
|
* Get the root-level object.
|
||||||
* @returns {Promise.<DomainObject>} a promise for the root object
|
* @returns {Promise.<DomainObject>} a promise for the root object
|
||||||
*/
|
*/
|
||||||
ObjectAPI.prototype.getRoot = function () {
|
ObjectAPI.prototype.getRoot = function () {
|
||||||
return this.rootProvider.get();
|
return this.rootProvider.get();
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Register a new object provider for a particular namespace.
|
* Register a new object provider for a particular namespace.
|
||||||
*
|
*
|
||||||
* @param {string} namespace the namespace for which to provide objects
|
* @param {string} namespace the namespace for which to provide objects
|
||||||
@ -90,11 +101,11 @@ define([
|
|||||||
* @memberof {module:openmct.ObjectAPI#}
|
* @memberof {module:openmct.ObjectAPI#}
|
||||||
* @name addProvider
|
* @name addProvider
|
||||||
*/
|
*/
|
||||||
ObjectAPI.prototype.addProvider = function (namespace, provider) {
|
ObjectAPI.prototype.addProvider = function (namespace, provider) {
|
||||||
this.providers[namespace] = provider;
|
this.providers[namespace] = provider;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Provides the ability to read, write, and delete domain objects.
|
* Provides the ability to read, write, and delete domain objects.
|
||||||
*
|
*
|
||||||
* When registering a new object provider, all methods on this interface
|
* When registering a new object provider, all methods on this interface
|
||||||
@ -104,7 +115,7 @@ define([
|
|||||||
* @memberof module:openmct
|
* @memberof module:openmct
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create the given domain object in the corresponding persistence store
|
* Create the given domain object in the corresponding persistence store
|
||||||
*
|
*
|
||||||
* @method create
|
* @method create
|
||||||
@ -115,7 +126,7 @@ define([
|
|||||||
* has been created, or be rejected if it cannot be saved
|
* has been created, or be rejected if it cannot be saved
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update this domain object in its persistence store
|
* Update this domain object in its persistence store
|
||||||
*
|
*
|
||||||
* @method update
|
* @method update
|
||||||
@ -126,7 +137,7 @@ define([
|
|||||||
* has been updated, or be rejected if it cannot be saved
|
* has been updated, or be rejected if it cannot be saved
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Delete this domain object.
|
* Delete this domain object.
|
||||||
*
|
*
|
||||||
* @method delete
|
* @method delete
|
||||||
@ -137,27 +148,18 @@ define([
|
|||||||
* has been deleted, or be rejected if it cannot be deleted
|
* has been deleted, or be rejected if it cannot be deleted
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get a domain object.
|
* Get a domain object.
|
||||||
*
|
*
|
||||||
* @method get
|
* @method get
|
||||||
* @memberof module:openmct.ObjectProvider#
|
* @memberof module:openmct.ObjectProvider#
|
||||||
* @param {string} key the key for the domain object to load
|
* @param {string} key the key for the domain object to load
|
||||||
|
* @param {AbortController.signal} abortSignal (optional) signal to abort fetch requests
|
||||||
* @returns {Promise} a promise which will resolve when the domain object
|
* @returns {Promise} a promise which will resolve when the domain object
|
||||||
* has been saved, or be rejected if it cannot be saved
|
* has been saved, or be rejected if it cannot be saved
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
ObjectAPI.prototype.get = function (identifier, abortSignal) {
|
||||||
* Get a domain object.
|
|
||||||
*
|
|
||||||
* @method get
|
|
||||||
* @memberof module:openmct.ObjectAPI#
|
|
||||||
* @param {module:openmct.ObjectAPI~Identifier} identifier
|
|
||||||
* the identifier for the domain object to load
|
|
||||||
* @returns {Promise} a promise which will resolve when the domain object
|
|
||||||
* has been saved, or be rejected if it cannot be saved
|
|
||||||
*/
|
|
||||||
ObjectAPI.prototype.get = function (identifier) {
|
|
||||||
let keystring = this.makeKeyString(identifier);
|
let keystring = this.makeKeyString(identifier);
|
||||||
if (this.cache[keystring] !== undefined) {
|
if (this.cache[keystring] !== undefined) {
|
||||||
return this.cache[keystring];
|
return this.cache[keystring];
|
||||||
@ -174,34 +176,99 @@ define([
|
|||||||
throw new Error('Provider does not support get!');
|
throw new Error('Provider does not support get!');
|
||||||
}
|
}
|
||||||
|
|
||||||
let objectPromise = provider.get(identifier);
|
let objectPromise = provider.get(identifier, abortSignal);
|
||||||
|
|
||||||
this.cache[keystring] = objectPromise;
|
this.cache[keystring] = objectPromise;
|
||||||
|
|
||||||
return objectPromise.then(result => {
|
return objectPromise.then(result => {
|
||||||
delete this.cache[keystring];
|
delete this.cache[keystring];
|
||||||
const interceptors = this.listGetInterceptors(identifier, result);
|
result = this.applyGetInterceptors(identifier, result);
|
||||||
interceptors.forEach(interceptor => {
|
|
||||||
result = interceptor.invoke(identifier, result);
|
|
||||||
});
|
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
ObjectAPI.prototype.delete = function () {
|
/**
|
||||||
|
* Search for domain objects.
|
||||||
|
*
|
||||||
|
* Object providersSearches and combines results of each object provider search.
|
||||||
|
* Objects without search provided will have been indexed
|
||||||
|
* and will be searched using the fallback indexed 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
|
||||||
|
* @returns {Array.<Promise.<module:openmct.DomainObject>>}
|
||||||
|
* an array of promises returned from each object provider's search function
|
||||||
|
* each resolving to domain objects matching provided search query and options.
|
||||||
|
*/
|
||||||
|
ObjectAPI.prototype.search = function (query, abortSignal) {
|
||||||
|
const searchPromises = Object.values(this.providers)
|
||||||
|
.filter(provider => provider.search !== undefined)
|
||||||
|
.map(provider => provider.search(query, abortSignal));
|
||||||
|
|
||||||
|
searchPromises.push(this.fallbackProvider.superSecretFallbackSearch(query, abortSignal)
|
||||||
|
.then(results => results.hits
|
||||||
|
.map(hit => {
|
||||||
|
let domainObject = utils.toNewFormat(hit.object.getModel(), hit.object.getId());
|
||||||
|
domainObject = this.applyGetInterceptors(domainObject.identifier, domainObject);
|
||||||
|
|
||||||
|
return domainObject;
|
||||||
|
})));
|
||||||
|
|
||||||
|
return searchPromises;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Will fetch object for the given identifier, returning a version of the object that will automatically keep
|
||||||
|
* itself updated as it is mutated. Before using this function, you should ask yourself whether you really need it.
|
||||||
|
* The platform will provide mutable objects to views automatically if the underlying object can be mutated. The
|
||||||
|
* 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.<MutableDomainObject>} a promise that will resolve with a MutableDomainObject if
|
||||||
|
* the object can be mutated.
|
||||||
|
*/
|
||||||
|
ObjectAPI.prototype.getMutable = function (identifier) {
|
||||||
|
if (!this.supportsMutation(identifier)) {
|
||||||
|
throw new Error(`Object "${this.makeKeyString(identifier)}" does not support mutation.`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.get(identifier).then((object) => {
|
||||||
|
return this._toMutable(object);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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
|
||||||
|
*/
|
||||||
|
ObjectAPI.prototype.destroyMutable = function (domainObject) {
|
||||||
|
if (domainObject.isMutable) {
|
||||||
|
return domainObject.$destroy();
|
||||||
|
} else {
|
||||||
|
throw new Error("Attempted to destroy non-mutable domain object");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
ObjectAPI.prototype.delete = function () {
|
||||||
throw new Error('Delete not implemented');
|
throw new Error('Delete not implemented');
|
||||||
};
|
};
|
||||||
|
|
||||||
ObjectAPI.prototype.isPersistable = function (domainObject) {
|
ObjectAPI.prototype.isPersistable = function (idOrKeyString) {
|
||||||
let provider = this.getProvider(domainObject.identifier);
|
let identifier = utils.parseKeyString(idOrKeyString);
|
||||||
|
let provider = this.getProvider(identifier);
|
||||||
|
|
||||||
return provider !== undefined
|
return provider !== undefined
|
||||||
&& provider.create !== undefined
|
&& provider.create !== undefined
|
||||||
&& provider.update !== undefined;
|
&& provider.update !== undefined;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Save this domain object in its current state. EXPERIMENTAL
|
* Save this domain object in its current state. EXPERIMENTAL
|
||||||
*
|
*
|
||||||
* @private
|
* @private
|
||||||
@ -211,12 +278,12 @@ define([
|
|||||||
* @returns {Promise} a promise which will resolve when the domain object
|
* @returns {Promise} a promise which will resolve when the domain object
|
||||||
* has been saved, or be rejected if it cannot be saved
|
* has been saved, or be rejected if it cannot be saved
|
||||||
*/
|
*/
|
||||||
ObjectAPI.prototype.save = function (domainObject) {
|
ObjectAPI.prototype.save = function (domainObject) {
|
||||||
let provider = this.getProvider(domainObject.identifier);
|
let provider = this.getProvider(domainObject.identifier);
|
||||||
let savedResolve;
|
let savedResolve;
|
||||||
let result;
|
let result;
|
||||||
|
|
||||||
if (!this.isPersistable(domainObject)) {
|
if (!this.isPersistable(domainObject.identifier)) {
|
||||||
result = Promise.reject('Object provider does not support saving');
|
result = Promise.reject('Object provider does not support saving');
|
||||||
} else if (hasAlreadyBeenPersisted(domainObject)) {
|
} else if (hasAlreadyBeenPersisted(domainObject)) {
|
||||||
result = Promise.resolve(true);
|
result = Promise.resolve(true);
|
||||||
@ -239,9 +306,9 @@ define([
|
|||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Add a root-level object.
|
* Add a root-level object.
|
||||||
* @param {module:openmct.ObjectAPI~Identifier|function} an array of
|
* @param {module:openmct.ObjectAPI~Identifier|function} an array of
|
||||||
* identifiers for root level objects, or a function that returns a
|
* identifiers for root level objects, or a function that returns a
|
||||||
@ -249,11 +316,45 @@ define([
|
|||||||
* @method addRoot
|
* @method addRoot
|
||||||
* @memberof module:openmct.ObjectAPI#
|
* @memberof module:openmct.ObjectAPI#
|
||||||
*/
|
*/
|
||||||
ObjectAPI.prototype.addRoot = function (key) {
|
ObjectAPI.prototype.addRoot = function (key) {
|
||||||
this.rootRegistry.addRoot(key);
|
this.rootRegistry.addRoot(key);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* Register an object interceptor that transforms a domain object requested via module:openmct.ObjectAPI.get
|
||||||
|
* The domain object will be transformed after it is retrieved from the persistence store
|
||||||
|
* The domain object will be transformed only if the interceptor is applicable to that domain object as defined by the InterceptorDef
|
||||||
|
*
|
||||||
|
* @param {module:openmct.InterceptorDef} interceptorDef the interceptor definition to add
|
||||||
|
* @method addGetInterceptor
|
||||||
|
* @memberof module:openmct.InterceptorRegistry#
|
||||||
|
*/
|
||||||
|
ObjectAPI.prototype.addGetInterceptor = function (interceptorDef) {
|
||||||
|
this.interceptorRegistry.addInterceptor(interceptorDef);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve the interceptors for a given domain object.
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
ObjectAPI.prototype.listGetInterceptors = function (identifier, object) {
|
||||||
|
return this.interceptorRegistry.getInterceptors(identifier, object);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Inovke interceptors if applicable for a given domain object.
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
ObjectAPI.prototype.applyGetInterceptors = function (identifier, domainObject) {
|
||||||
|
const interceptors = this.listGetInterceptors(identifier, domainObject);
|
||||||
|
interceptors.forEach(interceptor => {
|
||||||
|
domainObject = interceptor.invoke(identifier, domainObject);
|
||||||
|
});
|
||||||
|
|
||||||
|
return domainObject;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
* Modify a domain object.
|
* Modify a domain object.
|
||||||
* @param {module:openmct.DomainObject} object the object to mutate
|
* @param {module:openmct.DomainObject} object the object to mutate
|
||||||
* @param {string} path the property to modify
|
* @param {string} path the property to modify
|
||||||
@ -261,14 +362,67 @@ define([
|
|||||||
* @method mutate
|
* @method mutate
|
||||||
* @memberof module:openmct.ObjectAPI#
|
* @memberof module:openmct.ObjectAPI#
|
||||||
*/
|
*/
|
||||||
ObjectAPI.prototype.mutate = function (domainObject, path, value) {
|
ObjectAPI.prototype.mutate = function (domainObject, path, value) {
|
||||||
const mutableObject =
|
if (!this.supportsMutation(domainObject.identifier)) {
|
||||||
new MutableObject(this.eventEmitter, domainObject);
|
throw `Error: Attempted to mutate immutable object ${domainObject.name}`;
|
||||||
|
}
|
||||||
|
|
||||||
return mutableObject.set(path, value);
|
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.
|
||||||
|
let mutableDomainObject = this._toMutable(domainObject);
|
||||||
|
|
||||||
/**
|
//Mutate original object
|
||||||
|
MutableDomainObject.mutateObject(domainObject, path, value);
|
||||||
|
|
||||||
|
//Mutate temporary mutable object, in the process informing any other mutable instances
|
||||||
|
mutableDomainObject.$set(path, value);
|
||||||
|
|
||||||
|
//Destroy temporary mutable object
|
||||||
|
this.destroyMutable(mutableDomainObject);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
ObjectAPI.prototype._toMutable = function (object) {
|
||||||
|
let mutableObject;
|
||||||
|
|
||||||
|
if (object.isMutable) {
|
||||||
|
mutableObject = object;
|
||||||
|
} else {
|
||||||
|
mutableObject = MutableDomainObject.createMutable(object, this.eventEmitter);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if provider supports realtime updates
|
||||||
|
let identifier = utils.parseKeyString(mutableObject.identifier);
|
||||||
|
let provider = this.getProvider(identifier);
|
||||||
|
|
||||||
|
if (provider !== undefined
|
||||||
|
&& provider.observe !== undefined) {
|
||||||
|
let unobserve = provider.observe(identifier, (updatedModel) => {
|
||||||
|
mutableObject.$refresh(updatedModel);
|
||||||
|
});
|
||||||
|
mutableObject.$on('$destroy', () => {
|
||||||
|
unobserve();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return mutableObject;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param module:openmct.ObjectAPI~Identifier identifier An object identifier
|
||||||
|
* @returns {boolean} true if the object can be mutated, otherwise returns false
|
||||||
|
*/
|
||||||
|
ObjectAPI.prototype.supportsMutation = function (identifier) {
|
||||||
|
return this.isPersistable(identifier);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
* Observe changes to a domain object.
|
* Observe changes to a domain object.
|
||||||
* @param {module:openmct.DomainObject} object the object to observe
|
* @param {module:openmct.DomainObject} object the object to observe
|
||||||
* @param {string} path the property to observe
|
* @param {string} path the property to observe
|
||||||
@ -277,36 +431,47 @@ define([
|
|||||||
* @method observe
|
* @method observe
|
||||||
* @memberof module:openmct.ObjectAPI#
|
* @memberof module:openmct.ObjectAPI#
|
||||||
*/
|
*/
|
||||||
ObjectAPI.prototype.observe = function (domainObject, path, callback) {
|
ObjectAPI.prototype.observe = function (domainObject, path, callback) {
|
||||||
const mutableObject =
|
if (domainObject.isMutable) {
|
||||||
new MutableObject(this.eventEmitter, domainObject);
|
return domainObject.$observe(path, callback);
|
||||||
mutableObject.on(path, callback);
|
} else {
|
||||||
|
let mutable = this._toMutable(domainObject);
|
||||||
|
mutable.$observe(path, callback);
|
||||||
|
|
||||||
return mutableObject.stopListening.bind(mutableObject);
|
return () => mutable.$destroy();
|
||||||
};
|
}
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {module:openmct.ObjectAPI~Identifier} identifier
|
* @param {module:openmct.ObjectAPI~Identifier} identifier
|
||||||
* @returns {string} A string representation of the given identifier, including namespace and key
|
* @returns {string} A string representation of the given identifier, including namespace and key
|
||||||
*/
|
*/
|
||||||
ObjectAPI.prototype.makeKeyString = function (identifier) {
|
ObjectAPI.prototype.makeKeyString = function (identifier) {
|
||||||
return utils.makeKeyString(identifier);
|
return utils.makeKeyString(identifier);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* @param {string} keyString A string representation of the given identifier, that is, a namespace and key separated by a colon.
|
||||||
|
* @returns {module:openmct.ObjectAPI~Identifier} An identifier object
|
||||||
|
*/
|
||||||
|
ObjectAPI.prototype.parseKeyString = function (keyString) {
|
||||||
|
return utils.parseKeyString(keyString);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
* Given any number of identifiers, will return true if they are all equal, otherwise false.
|
* Given any number of identifiers, will return true if they are all equal, otherwise false.
|
||||||
* @param {module:openmct.ObjectAPI~Identifier[]} identifiers
|
* @param {module:openmct.ObjectAPI~Identifier[]} identifiers
|
||||||
*/
|
*/
|
||||||
ObjectAPI.prototype.areIdsEqual = function (...identifiers) {
|
ObjectAPI.prototype.areIdsEqual = function (...identifiers) {
|
||||||
return identifiers.map(utils.parseKeyString)
|
return identifiers.map(utils.parseKeyString)
|
||||||
.every(identifier => {
|
.every(identifier => {
|
||||||
return identifier === identifiers[0]
|
return identifier === identifiers[0]
|
||||||
|| (identifier.namespace === identifiers[0].namespace
|
|| (identifier.namespace === identifiers[0].namespace
|
||||||
&& identifier.key === identifiers[0].key);
|
&& identifier.key === identifiers[0].key);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
ObjectAPI.prototype.getOriginalPath = function (identifier, path = []) {
|
ObjectAPI.prototype.getOriginalPath = function (identifier, path = []) {
|
||||||
return this.get(identifier).then((domainObject) => {
|
return this.get(identifier).then((domainObject) => {
|
||||||
path.push(domainObject);
|
path.push(domainObject);
|
||||||
let location = domainObject.location;
|
let location = domainObject.location;
|
||||||
@ -317,30 +482,9 @@ define([
|
|||||||
return path;
|
return path;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Register an object interceptor that transforms a domain object requested via module:openmct.ObjectAPI.get
|
|
||||||
* The domain object will be transformed after it is retrieved from the persistence store
|
|
||||||
* The domain object will be transformed only if the interceptor is applicable to that domain object as defined by the InterceptorDef
|
|
||||||
*
|
|
||||||
* @param {module:openmct.InterceptorDef} interceptorDef the interceptor definition to add
|
|
||||||
* @method addGetInterceptor
|
|
||||||
* @memberof module:openmct.InterceptorRegistry#
|
|
||||||
*/
|
|
||||||
ObjectAPI.prototype.addGetInterceptor = function (interceptorDef) {
|
|
||||||
this.interceptorRegistry.addInterceptor(interceptorDef);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieve the interceptors for a given domain object.
|
|
||||||
* @private
|
|
||||||
*/
|
|
||||||
ObjectAPI.prototype.listGetInterceptors = function (identifier, object) {
|
|
||||||
return this.interceptorRegistry.getInterceptors(identifier, object);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Uniquely identifies a domain object.
|
* Uniquely identifies a domain object.
|
||||||
*
|
*
|
||||||
* @typedef Identifier
|
* @typedef Identifier
|
||||||
@ -351,7 +495,7 @@ define([
|
|||||||
* within that namespace
|
* within that namespace
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A domain object is an entity of relevance to a user's workflow, that
|
* 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
|
* should appear as a distinct and meaningful object within the user
|
||||||
* interface. Examples of domain objects are folders, telemetry sensors,
|
* interface. Examples of domain objects are folders, telemetry sensors,
|
||||||
@ -375,10 +519,9 @@ define([
|
|||||||
* @memberof module:openmct
|
* @memberof module:openmct
|
||||||
*/
|
*/
|
||||||
|
|
||||||
function hasAlreadyBeenPersisted(domainObject) {
|
function hasAlreadyBeenPersisted(domainObject) {
|
||||||
return domainObject.persisted !== undefined
|
return domainObject.persisted !== undefined
|
||||||
&& domainObject.persisted === domainObject.modified;
|
&& domainObject.persisted === domainObject.modified;
|
||||||
}
|
}
|
||||||
|
|
||||||
return ObjectAPI;
|
export default ObjectAPI;
|
||||||
});
|
|
||||||
|
119
src/api/objects/ObjectAPISearchSpec.js
Normal file
119
src/api/objects/ObjectAPISearchSpec.js
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
import ObjectAPI from './ObjectAPI.js';
|
||||||
|
|
||||||
|
describe("The Object API Search Function", () => {
|
||||||
|
const MOCK_PROVIDER_KEY = 'mockProvider';
|
||||||
|
const ANOTHER_MOCK_PROVIDER_KEY = 'anotherMockProvider';
|
||||||
|
const MOCK_PROVIDER_SEARCH_DELAY = 15000;
|
||||||
|
const ANOTHER_MOCK_PROVIDER_SEARCH_DELAY = 20000;
|
||||||
|
const TOTAL_TIME_ELAPSED = 21000;
|
||||||
|
const BASE_TIME = new Date(2021, 0, 1);
|
||||||
|
|
||||||
|
let objectAPI;
|
||||||
|
let mockObjectProvider;
|
||||||
|
let anotherMockObjectProvider;
|
||||||
|
let mockFallbackProvider;
|
||||||
|
let fallbackProviderSearchResults;
|
||||||
|
let resultsPromises;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
jasmine.clock().install();
|
||||||
|
jasmine.clock().mockDate(BASE_TIME);
|
||||||
|
|
||||||
|
resultsPromises = [];
|
||||||
|
fallbackProviderSearchResults = {
|
||||||
|
hits: []
|
||||||
|
};
|
||||||
|
|
||||||
|
objectAPI = new ObjectAPI();
|
||||||
|
|
||||||
|
mockObjectProvider = jasmine.createSpyObj("mock object provider", [
|
||||||
|
"search"
|
||||||
|
]);
|
||||||
|
anotherMockObjectProvider = jasmine.createSpyObj("another mock object provider", [
|
||||||
|
"search"
|
||||||
|
]);
|
||||||
|
mockFallbackProvider = jasmine.createSpyObj("super secret fallback provider", [
|
||||||
|
"superSecretFallbackSearch"
|
||||||
|
]);
|
||||||
|
objectAPI.addProvider('objects', mockObjectProvider);
|
||||||
|
objectAPI.addProvider('other-objects', anotherMockObjectProvider);
|
||||||
|
objectAPI.supersecretSetFallbackProvider(mockFallbackProvider);
|
||||||
|
|
||||||
|
mockObjectProvider.search.and.callFake(() => {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
const mockProviderSearch = {
|
||||||
|
name: MOCK_PROVIDER_KEY,
|
||||||
|
start: new Date()
|
||||||
|
};
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
mockProviderSearch.end = new Date();
|
||||||
|
|
||||||
|
return resolve(mockProviderSearch);
|
||||||
|
}, MOCK_PROVIDER_SEARCH_DELAY);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
anotherMockObjectProvider.search.and.callFake(() => {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
const anotherMockProviderSearch = {
|
||||||
|
name: ANOTHER_MOCK_PROVIDER_KEY,
|
||||||
|
start: new Date()
|
||||||
|
};
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
anotherMockProviderSearch.end = new Date();
|
||||||
|
|
||||||
|
return resolve(anotherMockProviderSearch);
|
||||||
|
}, ANOTHER_MOCK_PROVIDER_SEARCH_DELAY);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
mockFallbackProvider.superSecretFallbackSearch.and.callFake(
|
||||||
|
() => new Promise(
|
||||||
|
resolve => setTimeout(
|
||||||
|
() => resolve(fallbackProviderSearchResults),
|
||||||
|
50
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
resultsPromises = objectAPI.search('foo');
|
||||||
|
|
||||||
|
jasmine.clock().tick(TOTAL_TIME_ELAPSED);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
jasmine.clock().uninstall();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("uses each objects given provider's search function", () => {
|
||||||
|
expect(mockObjectProvider.search).toHaveBeenCalled();
|
||||||
|
expect(anotherMockObjectProvider.search).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("uses the fallback indexed search for objects without a search function provided", () => {
|
||||||
|
expect(mockFallbackProvider.superSecretFallbackSearch).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("provides each providers results as promises that resolve in parallel", async () => {
|
||||||
|
const results = await Promise.all(resultsPromises);
|
||||||
|
const mockProviderResults = results.find(
|
||||||
|
result => result.name === MOCK_PROVIDER_KEY
|
||||||
|
);
|
||||||
|
const anotherMockProviderResults = results.find(
|
||||||
|
result => result.name === ANOTHER_MOCK_PROVIDER_KEY
|
||||||
|
);
|
||||||
|
const mockProviderStart = mockProviderResults.start.getTime();
|
||||||
|
const mockProviderEnd = mockProviderResults.end.getTime();
|
||||||
|
const anotherMockProviderStart = anotherMockProviderResults.start.getTime();
|
||||||
|
const anotherMockProviderEnd = anotherMockProviderResults.end.getTime();
|
||||||
|
const searchElapsedTime = Math.max(mockProviderEnd, anotherMockProviderEnd)
|
||||||
|
- Math.min(mockProviderEnd, anotherMockProviderEnd);
|
||||||
|
|
||||||
|
expect(mockProviderStart).toBeLessThan(anotherMockProviderEnd);
|
||||||
|
expect(anotherMockProviderStart).toBeLessThan(mockProviderEnd);
|
||||||
|
expect(searchElapsedTime).toBeLessThan(
|
||||||
|
MOCK_PROVIDER_SEARCH_DELAY
|
||||||
|
+ ANOTHER_MOCK_PROVIDER_SEARCH_DELAY
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
@ -2,12 +2,30 @@ import ObjectAPI from './ObjectAPI.js';
|
|||||||
|
|
||||||
describe("The Object API", () => {
|
describe("The Object API", () => {
|
||||||
let objectAPI;
|
let objectAPI;
|
||||||
|
let typeRegistry;
|
||||||
|
let openmct = {};
|
||||||
|
let mockIdentifierService;
|
||||||
let mockDomainObject;
|
let mockDomainObject;
|
||||||
const TEST_NAMESPACE = "test-namespace";
|
const TEST_NAMESPACE = "test-namespace";
|
||||||
const FIFTEEN_MINUTES = 15 * 60 * 1000;
|
const FIFTEEN_MINUTES = 15 * 60 * 1000;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
objectAPI = new ObjectAPI();
|
typeRegistry = jasmine.createSpyObj('typeRegistry', [
|
||||||
|
'get'
|
||||||
|
]);
|
||||||
|
openmct.$injector = jasmine.createSpyObj('$injector', ['get']);
|
||||||
|
mockIdentifierService = jasmine.createSpyObj(
|
||||||
|
'identifierService',
|
||||||
|
['parse']
|
||||||
|
);
|
||||||
|
mockIdentifierService.parse.and.returnValue({
|
||||||
|
getSpace: () => {
|
||||||
|
return TEST_NAMESPACE;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
openmct.$injector.get.and.returnValue(mockIdentifierService);
|
||||||
|
objectAPI = new ObjectAPI(typeRegistry, openmct);
|
||||||
mockDomainObject = {
|
mockDomainObject = {
|
||||||
identifier: {
|
identifier: {
|
||||||
namespace: TEST_NAMESPACE,
|
namespace: TEST_NAMESPACE,
|
||||||
@ -33,6 +51,7 @@ describe("The Object API", () => {
|
|||||||
"update"
|
"update"
|
||||||
]);
|
]);
|
||||||
mockProvider.create.and.returnValue(Promise.resolve(true));
|
mockProvider.create.and.returnValue(Promise.resolve(true));
|
||||||
|
mockProvider.update.and.returnValue(Promise.resolve(true));
|
||||||
objectAPI.addProvider(TEST_NAMESPACE, mockProvider);
|
objectAPI.addProvider(TEST_NAMESPACE, mockProvider);
|
||||||
});
|
});
|
||||||
it("Calls 'create' on provider if object is new", () => {
|
it("Calls 'create' on provider if object is new", () => {
|
||||||
@ -128,4 +147,155 @@ describe("The Object API", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("the mutation API", () => {
|
||||||
|
let testObject;
|
||||||
|
let updatedTestObject;
|
||||||
|
let mutable;
|
||||||
|
let mockProvider;
|
||||||
|
let callbacks = [];
|
||||||
|
|
||||||
|
beforeEach(function () {
|
||||||
|
objectAPI = new ObjectAPI(typeRegistry, openmct);
|
||||||
|
testObject = {
|
||||||
|
identifier: {
|
||||||
|
namespace: TEST_NAMESPACE,
|
||||||
|
key: 'test-key'
|
||||||
|
},
|
||||||
|
name: 'test object',
|
||||||
|
otherAttribute: 'other-attribute-value',
|
||||||
|
objectAttribute: {
|
||||||
|
embeddedObject: {
|
||||||
|
embeddedKey: 'embedded-value'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
updatedTestObject = Object.assign({otherAttribute: 'changed-attribute-value'}, testObject);
|
||||||
|
mockProvider = jasmine.createSpyObj("mock provider", [
|
||||||
|
"get",
|
||||||
|
"create",
|
||||||
|
"update",
|
||||||
|
"observe",
|
||||||
|
"observeObjectChanges"
|
||||||
|
]);
|
||||||
|
mockProvider.get.and.returnValue(Promise.resolve(testObject));
|
||||||
|
mockProvider.observeObjectChanges.and.callFake(() => {
|
||||||
|
callbacks[0](updatedTestObject);
|
||||||
|
callbacks.splice(0, 1);
|
||||||
|
});
|
||||||
|
mockProvider.observe.and.callFake((id, callback) => {
|
||||||
|
if (callbacks.length === 0) {
|
||||||
|
callbacks.push(callback);
|
||||||
|
} else {
|
||||||
|
callbacks[0] = callback;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
objectAPI.addProvider(TEST_NAMESPACE, mockProvider);
|
||||||
|
|
||||||
|
return objectAPI.getMutable(testObject.identifier)
|
||||||
|
.then(object => {
|
||||||
|
mutable = object;
|
||||||
|
|
||||||
|
return mutable;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
mutable.$destroy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('mutates the original object', () => {
|
||||||
|
const MUTATED_NAME = 'mutated name';
|
||||||
|
objectAPI.mutate(testObject, 'name', MUTATED_NAME);
|
||||||
|
expect(testObject.name).toBe(MUTATED_NAME);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe ('uses a MutableDomainObject', () => {
|
||||||
|
it('and retains properties of original object ', function () {
|
||||||
|
expect(hasOwnProperty(mutable, 'identifier')).toBe(true);
|
||||||
|
expect(hasOwnProperty(mutable, 'otherAttribute')).toBe(true);
|
||||||
|
expect(mutable.identifier).toEqual(testObject.identifier);
|
||||||
|
expect(mutable.otherAttribute).toEqual(testObject.otherAttribute);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('that is identical to original object when serialized', function () {
|
||||||
|
expect(JSON.stringify(mutable)).toEqual(JSON.stringify(testObject));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('that observes for object changes', function () {
|
||||||
|
let mockListener = jasmine.createSpy('mockListener');
|
||||||
|
objectAPI.observe(testObject, '*', mockListener);
|
||||||
|
mockProvider.observeObjectChanges();
|
||||||
|
expect(mockListener).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('uses events', function () {
|
||||||
|
let testObjectDuplicate;
|
||||||
|
let mutableSecondInstance;
|
||||||
|
|
||||||
|
beforeEach(function () {
|
||||||
|
// Duplicate object to guarantee we are not sharing object instance, which would invalidate test
|
||||||
|
testObjectDuplicate = JSON.parse(JSON.stringify(testObject));
|
||||||
|
mutableSecondInstance = objectAPI._toMutable(testObjectDuplicate);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
mutableSecondInstance.$destroy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('to stay synchronized when mutated', function () {
|
||||||
|
objectAPI.mutate(mutable, 'otherAttribute', 'new-attribute-value');
|
||||||
|
expect(mutableSecondInstance.otherAttribute).toBe('new-attribute-value');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('to indicate when a property changes', function () {
|
||||||
|
let mutationCallback = jasmine.createSpy('mutation-callback');
|
||||||
|
let unlisten;
|
||||||
|
|
||||||
|
return new Promise(function (resolve) {
|
||||||
|
mutationCallback.and.callFake(resolve);
|
||||||
|
unlisten = objectAPI.observe(mutableSecondInstance, 'otherAttribute', mutationCallback);
|
||||||
|
objectAPI.mutate(mutable, 'otherAttribute', 'some-new-value');
|
||||||
|
}).then(function () {
|
||||||
|
expect(mutationCallback).toHaveBeenCalledWith('some-new-value');
|
||||||
|
unlisten();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('to indicate when a child property has changed', function () {
|
||||||
|
let embeddedKeyCallback = jasmine.createSpy('embeddedKeyCallback');
|
||||||
|
let embeddedObjectCallback = jasmine.createSpy('embeddedObjectCallback');
|
||||||
|
let objectAttributeCallback = jasmine.createSpy('objectAttribute');
|
||||||
|
let listeners = [];
|
||||||
|
|
||||||
|
return new Promise(function (resolve) {
|
||||||
|
objectAttributeCallback.and.callFake(resolve);
|
||||||
|
|
||||||
|
listeners.push(objectAPI.observe(mutableSecondInstance, 'objectAttribute.embeddedObject.embeddedKey', embeddedKeyCallback));
|
||||||
|
listeners.push(objectAPI.observe(mutableSecondInstance, 'objectAttribute.embeddedObject', embeddedObjectCallback));
|
||||||
|
listeners.push(objectAPI.observe(mutableSecondInstance, 'objectAttribute', objectAttributeCallback));
|
||||||
|
|
||||||
|
objectAPI.mutate(mutable, 'objectAttribute.embeddedObject.embeddedKey', 'updated-embedded-value');
|
||||||
|
}).then(function () {
|
||||||
|
expect(embeddedKeyCallback).toHaveBeenCalledWith('updated-embedded-value');
|
||||||
|
expect(embeddedObjectCallback).toHaveBeenCalledWith({
|
||||||
|
embeddedKey: 'updated-embedded-value'
|
||||||
|
});
|
||||||
|
expect(objectAttributeCallback).toHaveBeenCalledWith({
|
||||||
|
embeddedObject: {
|
||||||
|
embeddedKey: 'updated-embedded-value'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
listeners.forEach(listener => listener());
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
function hasOwnProperty(object, property) {
|
||||||
|
return Object.prototype.hasOwnProperty.call(object, property);
|
||||||
|
}
|
||||||
|
@ -48,7 +48,7 @@ define([
|
|||||||
this.providers.push(function () {
|
this.providers.push(function () {
|
||||||
return key;
|
return key;
|
||||||
});
|
});
|
||||||
} else if (_.isFunction(key)) {
|
} else if (typeof key === "function") {
|
||||||
this.providers.push(key);
|
this.providers.push(key);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -6,6 +6,9 @@ class Dialog extends Overlay {
|
|||||||
constructor({iconClass, message, title, hint, timestamp, ...options}) {
|
constructor({iconClass, message, title, hint, timestamp, ...options}) {
|
||||||
|
|
||||||
let component = new Vue({
|
let component = new Vue({
|
||||||
|
components: {
|
||||||
|
DialogComponent: DialogComponent
|
||||||
|
},
|
||||||
provide: {
|
provide: {
|
||||||
iconClass,
|
iconClass,
|
||||||
message,
|
message,
|
||||||
@ -13,9 +16,6 @@ class Dialog extends Overlay {
|
|||||||
hint,
|
hint,
|
||||||
timestamp
|
timestamp
|
||||||
},
|
},
|
||||||
components: {
|
|
||||||
DialogComponent: DialogComponent
|
|
||||||
},
|
|
||||||
template: '<dialog-component></dialog-component>'
|
template: '<dialog-component></dialog-component>'
|
||||||
}).$mount();
|
}).$mount();
|
||||||
|
|
||||||
|
@ -7,6 +7,9 @@ let component;
|
|||||||
class ProgressDialog extends Overlay {
|
class ProgressDialog extends Overlay {
|
||||||
constructor({progressPerc, progressText, iconClass, message, title, hint, timestamp, ...options}) {
|
constructor({progressPerc, progressText, iconClass, message, title, hint, timestamp, ...options}) {
|
||||||
component = new Vue({
|
component = new Vue({
|
||||||
|
components: {
|
||||||
|
ProgressDialogComponent: ProgressDialogComponent
|
||||||
|
},
|
||||||
provide: {
|
provide: {
|
||||||
iconClass,
|
iconClass,
|
||||||
message,
|
message,
|
||||||
@ -14,9 +17,6 @@ class ProgressDialog extends Overlay {
|
|||||||
hint,
|
hint,
|
||||||
timestamp
|
timestamp
|
||||||
},
|
},
|
||||||
components: {
|
|
||||||
ProgressDialogComponent: ProgressDialogComponent
|
|
||||||
},
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
model: {
|
model: {
|
||||||
|
@ -38,12 +38,12 @@
|
|||||||
|
|
||||||
<script>
|
<script>
|
||||||
export default {
|
export default {
|
||||||
|
inject: ['dismiss', 'element', 'buttons', 'dismissable'],
|
||||||
data: function () {
|
data: function () {
|
||||||
return {
|
return {
|
||||||
focusIndex: -1
|
focusIndex: -1
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
inject: ['dismiss', 'element', 'buttons', 'dismissable'],
|
|
||||||
mounted() {
|
mounted() {
|
||||||
const element = this.$refs.element;
|
const element = this.$refs.element;
|
||||||
element.appendChild(this.element);
|
element.appendChild(this.element);
|
||||||
|
37
src/plugins/CouchDBSearchFolder/plugin.js
Normal file
37
src/plugins/CouchDBSearchFolder/plugin.js
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
export default function (folderName, couchPlugin, searchFilter) {
|
||||||
|
return function install(openmct) {
|
||||||
|
const couchProvider = couchPlugin.couchProvider;
|
||||||
|
|
||||||
|
openmct.objects.addRoot({
|
||||||
|
namespace: 'couch-search',
|
||||||
|
key: 'couch-search'
|
||||||
|
});
|
||||||
|
|
||||||
|
openmct.objects.addProvider('couch-search', {
|
||||||
|
get(identifier) {
|
||||||
|
if (identifier.key !== 'couch-search') {
|
||||||
|
return undefined;
|
||||||
|
} else {
|
||||||
|
return Promise.resolve({
|
||||||
|
identifier,
|
||||||
|
type: 'folder',
|
||||||
|
name: folderName || "CouchDB Documents"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
openmct.composition.addProvider({
|
||||||
|
appliesTo(domainObject) {
|
||||||
|
return domainObject.identifier.namespace === 'couch-search'
|
||||||
|
&& domainObject.identifier.key === 'couch-search';
|
||||||
|
},
|
||||||
|
load() {
|
||||||
|
return couchProvider.getObjectsByFilter(searchFilter).then(objects => {
|
||||||
|
return objects.map(object => object.identifier);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
91
src/plugins/CouchDBSearchFolder/pluginSpec.js
Normal file
91
src/plugins/CouchDBSearchFolder/pluginSpec.js
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2020, 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 {createOpenMct, resetApplicationState} from "utils/testing";
|
||||||
|
import CouchDBSearchFolderPlugin from './plugin';
|
||||||
|
|
||||||
|
describe('the plugin', function () {
|
||||||
|
let identifier = {
|
||||||
|
namespace: 'couch-search',
|
||||||
|
key: "couch-search"
|
||||||
|
};
|
||||||
|
let testPath = '/test/db';
|
||||||
|
let openmct;
|
||||||
|
let composition;
|
||||||
|
|
||||||
|
beforeEach((done) => {
|
||||||
|
|
||||||
|
openmct = createOpenMct();
|
||||||
|
|
||||||
|
let couchPlugin = openmct.plugins.CouchDB(testPath);
|
||||||
|
openmct.install(couchPlugin);
|
||||||
|
|
||||||
|
openmct.install(new CouchDBSearchFolderPlugin('CouchDB Documents', couchPlugin, {
|
||||||
|
"selector": {
|
||||||
|
"model": {
|
||||||
|
"type": "plan"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
|
||||||
|
openmct.on('start', done);
|
||||||
|
openmct.startHeadless();
|
||||||
|
|
||||||
|
composition = openmct.composition.get({identifier});
|
||||||
|
|
||||||
|
spyOn(couchPlugin.couchProvider, 'getObjectsByFilter').and.returnValue(Promise.resolve([
|
||||||
|
{
|
||||||
|
identifier: {
|
||||||
|
key: "1",
|
||||||
|
namespace: "mct"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
identifier: {
|
||||||
|
key: "2",
|
||||||
|
namespace: "mct"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]));
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
return resetApplicationState(openmct);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('provides a folder to hold plans', () => {
|
||||||
|
openmct.objects.get(identifier).then((object) => {
|
||||||
|
expect(object).toEqual({
|
||||||
|
identifier,
|
||||||
|
type: 'folder',
|
||||||
|
name: "CouchDB Documents"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('provides composition for couch search folders', () => {
|
||||||
|
composition.load().then((objects) => {
|
||||||
|
expect(objects.length).toEqual(2);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
@ -44,11 +44,15 @@ export default function LADTableViewProvider(openmct) {
|
|||||||
LadTableComponent: LadTable
|
LadTableComponent: LadTable
|
||||||
},
|
},
|
||||||
provide: {
|
provide: {
|
||||||
openmct,
|
openmct
|
||||||
|
},
|
||||||
|
data: () => {
|
||||||
|
return {
|
||||||
domainObject,
|
domainObject,
|
||||||
objectPath
|
objectPath
|
||||||
|
};
|
||||||
},
|
},
|
||||||
template: '<lad-table-component></lad-table-component>'
|
template: '<lad-table-component :domain-object="domainObject" :object-path="objectPath"></lad-table-component>'
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
destroy: function (element) {
|
destroy: function (element) {
|
||||||
|
@ -26,7 +26,7 @@
|
|||||||
class="js-lad-table__body__row"
|
class="js-lad-table__body__row"
|
||||||
@contextmenu.prevent="showContextMenu"
|
@contextmenu.prevent="showContextMenu"
|
||||||
>
|
>
|
||||||
<td class="js-first-data">{{ name }}</td>
|
<td class="js-first-data">{{ domainObject.name }}</td>
|
||||||
<td class="js-second-data">{{ formattedTimestamp }}</td>
|
<td class="js-second-data">{{ formattedTimestamp }}</td>
|
||||||
<td
|
<td
|
||||||
class="js-third-data"
|
class="js-third-data"
|
||||||
@ -50,12 +50,16 @@ const CONTEXT_MENU_ACTIONS = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct', 'objectPath'],
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
domainObject: {
|
domainObject: {
|
||||||
type: Object,
|
type: Object,
|
||||||
required: true
|
required: true
|
||||||
},
|
},
|
||||||
|
objectPath: {
|
||||||
|
type: Array,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
hasUnits: {
|
hasUnits: {
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
requred: true
|
requred: true
|
||||||
@ -66,7 +70,6 @@ export default {
|
|||||||
currentObjectPath.unshift(this.domainObject);
|
currentObjectPath.unshift(this.domainObject);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
name: this.domainObject.name,
|
|
||||||
timestamp: undefined,
|
timestamp: undefined,
|
||||||
value: '---',
|
value: '---',
|
||||||
valueClass: '',
|
valueClass: '',
|
||||||
@ -89,14 +92,6 @@ export default {
|
|||||||
.telemetry
|
.telemetry
|
||||||
.limitEvaluator(this.domainObject);
|
.limitEvaluator(this.domainObject);
|
||||||
|
|
||||||
this.stopWatchingMutation = this.openmct
|
|
||||||
.objects
|
|
||||||
.observe(
|
|
||||||
this.domainObject,
|
|
||||||
'*',
|
|
||||||
this.updateName
|
|
||||||
);
|
|
||||||
|
|
||||||
this.openmct.time.on('timeSystem', this.updateTimeSystem);
|
this.openmct.time.on('timeSystem', this.updateTimeSystem);
|
||||||
this.openmct.time.on('bounds', this.updateBounds);
|
this.openmct.time.on('bounds', this.updateBounds);
|
||||||
|
|
||||||
@ -119,7 +114,6 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
destroyed() {
|
destroyed() {
|
||||||
this.stopWatchingMutation();
|
|
||||||
this.unsubscribe();
|
this.unsubscribe();
|
||||||
this.openmct.time.off('timeSystem', this.updateTimeSystem);
|
this.openmct.time.off('timeSystem', this.updateTimeSystem);
|
||||||
this.openmct.time.off('bounds', this.updateBounds);
|
this.openmct.time.off('bounds', this.updateBounds);
|
||||||
@ -160,9 +154,6 @@ export default {
|
|||||||
})
|
})
|
||||||
.then((array) => this.updateValues(array[array.length - 1]));
|
.then((array) => this.updateValues(array[array.length - 1]));
|
||||||
},
|
},
|
||||||
updateName(name) {
|
|
||||||
this.name = name;
|
|
||||||
},
|
|
||||||
updateBounds(bounds, isTick) {
|
updateBounds(bounds, isTick) {
|
||||||
this.bounds = bounds;
|
this.bounds = bounds;
|
||||||
if (!isTick) {
|
if (!isTick) {
|
||||||
|
@ -36,6 +36,7 @@
|
|||||||
v-for="item in items"
|
v-for="item in items"
|
||||||
:key="item.key"
|
:key="item.key"
|
||||||
:domain-object="item.domainObject"
|
:domain-object="item.domainObject"
|
||||||
|
:object-path="objectPath"
|
||||||
:has-units="hasUnits"
|
:has-units="hasUnits"
|
||||||
/>
|
/>
|
||||||
</tbody>
|
</tbody>
|
||||||
@ -47,10 +48,20 @@
|
|||||||
import LadRow from './LADRow.vue';
|
import LadRow from './LADRow.vue';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct', 'domainObject', 'objectPath'],
|
|
||||||
components: {
|
components: {
|
||||||
LadRow
|
LadRow
|
||||||
},
|
},
|
||||||
|
inject: ['openmct'],
|
||||||
|
props: {
|
||||||
|
domainObject: {
|
||||||
|
type: Object,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
objectPath: {
|
||||||
|
type: Array,
|
||||||
|
required: true
|
||||||
|
}
|
||||||
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
items: []
|
items: []
|
||||||
|
@ -57,10 +57,10 @@
|
|||||||
import LadRow from './LADRow.vue';
|
import LadRow from './LADRow.vue';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct', 'domainObject'],
|
|
||||||
components: {
|
components: {
|
||||||
LadRow
|
LadRow
|
||||||
},
|
},
|
||||||
|
inject: ['openmct', 'domainObject'],
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
ladTableObjects: [],
|
ladTableObjects: [],
|
||||||
|
@ -98,7 +98,7 @@ describe("The LAD Table", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it("should provide a table view only for lad table objects", () => {
|
it("should provide a table view only for lad table objects", () => {
|
||||||
let applicableViews = openmct.objectViews.get(mockObj.ladTable);
|
let applicableViews = openmct.objectViews.get(mockObj.ladTable, []);
|
||||||
|
|
||||||
let ladTableView = applicableViews.find(
|
let ladTableView = applicableViews.find(
|
||||||
(viewProvider) => viewProvider.key === ladTableKey
|
(viewProvider) => viewProvider.key === ladTableKey
|
||||||
@ -185,7 +185,7 @@ describe("The LAD Table", () => {
|
|||||||
end: bounds.end
|
end: bounds.end
|
||||||
});
|
});
|
||||||
|
|
||||||
applicableViews = openmct.objectViews.get(mockObj.ladTable);
|
applicableViews = openmct.objectViews.get(mockObj.ladTable, []);
|
||||||
ladTableViewProvider = applicableViews.find((viewProvider) => viewProvider.key === ladTableKey);
|
ladTableViewProvider = applicableViews.find((viewProvider) => viewProvider.key === ladTableKey);
|
||||||
ladTableView = ladTableViewProvider.view(mockObj.ladTable, [mockObj.ladTable]);
|
ladTableView = ladTableViewProvider.view(mockObj.ladTable, [mockObj.ladTable]);
|
||||||
ladTableView.show(child, true);
|
ladTableView.show(child, true);
|
||||||
@ -296,7 +296,7 @@ describe("The LAD Table Set", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it("should provide a lad table set view only for lad table set objects", () => {
|
it("should provide a lad table set view only for lad table set objects", () => {
|
||||||
let applicableViews = openmct.objectViews.get(mockObj.ladTableSet);
|
let applicableViews = openmct.objectViews.get(mockObj.ladTableSet, []);
|
||||||
|
|
||||||
let ladTableSetView = applicableViews.find(
|
let ladTableSetView = applicableViews.find(
|
||||||
(viewProvider) => viewProvider.key === ladTableSetKey
|
(viewProvider) => viewProvider.key === ladTableSetKey
|
||||||
@ -391,7 +391,7 @@ describe("The LAD Table Set", () => {
|
|||||||
end: bounds.end
|
end: bounds.end
|
||||||
});
|
});
|
||||||
|
|
||||||
applicableViews = openmct.objectViews.get(mockObj.ladTableSet);
|
applicableViews = openmct.objectViews.get(mockObj.ladTableSet, []);
|
||||||
ladTableSetViewProvider = applicableViews.find((viewProvider) => viewProvider.key === ladTableSetKey);
|
ladTableSetViewProvider = applicableViews.find((viewProvider) => viewProvider.key === ladTableSetKey);
|
||||||
ladTableSetView = ladTableSetViewProvider.view(mockObj.ladTableSet, [mockObj.ladTableSet]);
|
ladTableSetView = ladTableSetViewProvider.view(mockObj.ladTableSet, [mockObj.ladTableSet]);
|
||||||
ladTableSetView.show(child, true);
|
ladTableSetView.show(child, true);
|
||||||
|
@ -67,11 +67,11 @@ describe("AutoflowTabularPlugin", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it("applies its view to the type from options", () => {
|
it("applies its view to the type from options", () => {
|
||||||
expect(provider.canView(testObject)).toBe(true);
|
expect(provider.canView(testObject, [])).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("does not apply to other types", () => {
|
it("does not apply to other types", () => {
|
||||||
expect(provider.canView({ type: 'foo' })).toBe(false);
|
expect(provider.canView({ type: 'foo' }, [])).toBe(false);
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("provides a view which", () => {
|
describe("provides a view which", () => {
|
||||||
|
@ -37,12 +37,12 @@ define([
|
|||||||
return function install(openmct) {
|
return function install(openmct) {
|
||||||
if (installIndicator) {
|
if (installIndicator) {
|
||||||
let component = new Vue ({
|
let component = new Vue ({
|
||||||
provide: {
|
|
||||||
openmct
|
|
||||||
},
|
|
||||||
components: {
|
components: {
|
||||||
GlobalClearIndicator: GlobaClearIndicator.default
|
GlobalClearIndicator: GlobaClearIndicator.default
|
||||||
},
|
},
|
||||||
|
provide: {
|
||||||
|
openmct
|
||||||
|
},
|
||||||
template: '<GlobalClearIndicator></GlobalClearIndicator>'
|
template: '<GlobalClearIndicator></GlobalClearIndicator>'
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -75,7 +75,8 @@ export default class Condition extends EventEmitter {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.isTelemetryUsed(datum.id)) {
|
// if all the criteria in this condition have no telemetry, we want to force the condition result to evaluate
|
||||||
|
if (this.hasNoTelemetry() || this.isTelemetryUsed(datum.id)) {
|
||||||
|
|
||||||
this.criteria.forEach(criterion => {
|
this.criteria.forEach(criterion => {
|
||||||
if (this.isAnyOrAllTelemetry(criterion)) {
|
if (this.isAnyOrAllTelemetry(criterion)) {
|
||||||
@ -93,6 +94,12 @@ export default class Condition extends EventEmitter {
|
|||||||
return (criterion.telemetry && (criterion.telemetry === 'all' || criterion.telemetry === 'any'));
|
return (criterion.telemetry && (criterion.telemetry === 'all' || criterion.telemetry === 'any'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
hasNoTelemetry() {
|
||||||
|
return this.criteria.every((criterion) => {
|
||||||
|
return !this.isAnyOrAllTelemetry(criterion) && criterion.telemetry === '';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
isTelemetryUsed(id) {
|
isTelemetryUsed(id) {
|
||||||
return this.criteria.some(criterion => {
|
return this.criteria.some(criterion => {
|
||||||
return this.isAnyOrAllTelemetry(criterion) || criterion.telemetryObjectIdAsString === id;
|
return this.isAnyOrAllTelemetry(criterion) || criterion.telemetryObjectIdAsString === id;
|
||||||
@ -250,10 +257,17 @@ export default class Condition extends EventEmitter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
getTriggerDescription() {
|
getTriggerDescription() {
|
||||||
|
if (this.trigger) {
|
||||||
return {
|
return {
|
||||||
conjunction: TRIGGER_CONJUNCTION[this.trigger],
|
conjunction: TRIGGER_CONJUNCTION[this.trigger],
|
||||||
prefix: `${TRIGGER_LABEL[this.trigger]}: `
|
prefix: `${TRIGGER_LABEL[this.trigger]}: `
|
||||||
};
|
};
|
||||||
|
} else {
|
||||||
|
return {
|
||||||
|
conjunction: '',
|
||||||
|
prefix: ''
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
requestLADConditionResult() {
|
requestLADConditionResult() {
|
||||||
|
@ -34,6 +34,9 @@ export default class ConditionManager extends EventEmitter {
|
|||||||
this.composition = this.openmct.composition.get(conditionSetDomainObject);
|
this.composition = this.openmct.composition.get(conditionSetDomainObject);
|
||||||
this.composition.on('add', this.subscribeToTelemetry, this);
|
this.composition.on('add', this.subscribeToTelemetry, this);
|
||||||
this.composition.on('remove', this.unsubscribeFromTelemetry, this);
|
this.composition.on('remove', this.unsubscribeFromTelemetry, this);
|
||||||
|
|
||||||
|
this.shouldEvaluateNewTelemetry = this.shouldEvaluateNewTelemetry.bind(this);
|
||||||
|
|
||||||
this.compositionLoad = this.composition.load();
|
this.compositionLoad = this.composition.load();
|
||||||
this.subscriptions = {};
|
this.subscriptions = {};
|
||||||
this.telemetryObjects = {};
|
this.telemetryObjects = {};
|
||||||
@ -79,6 +82,17 @@ export default class ConditionManager extends EventEmitter {
|
|||||||
delete this.subscriptions[id];
|
delete this.subscriptions[id];
|
||||||
delete this.telemetryObjects[id];
|
delete this.telemetryObjects[id];
|
||||||
this.removeConditionTelemetryObjects();
|
this.removeConditionTelemetryObjects();
|
||||||
|
|
||||||
|
//force re-computation of condition set result as we might be in a state where
|
||||||
|
// there is no telemetry datum coming in for a while or at all.
|
||||||
|
let latestTimestamp = getLatestTimestamp(
|
||||||
|
{},
|
||||||
|
{},
|
||||||
|
this.timeSystems,
|
||||||
|
this.openmct.time.timeSystem()
|
||||||
|
);
|
||||||
|
this.updateConditionResults({id: id});
|
||||||
|
this.updateCurrentCondition(latestTimestamp);
|
||||||
}
|
}
|
||||||
|
|
||||||
initialize() {
|
initialize() {
|
||||||
@ -326,6 +340,10 @@ export default class ConditionManager extends EventEmitter {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
shouldEvaluateNewTelemetry(currentTimestamp) {
|
||||||
|
return this.openmct.time.bounds().end >= currentTimestamp;
|
||||||
|
}
|
||||||
|
|
||||||
telemetryReceived(endpoint, datum) {
|
telemetryReceived(endpoint, datum) {
|
||||||
if (!this.isTelemetryUsed(endpoint)) {
|
if (!this.isTelemetryUsed(endpoint)) {
|
||||||
return;
|
return;
|
||||||
@ -334,16 +352,21 @@ export default class ConditionManager extends EventEmitter {
|
|||||||
const normalizedDatum = this.createNormalizedDatum(datum, endpoint);
|
const normalizedDatum = this.createNormalizedDatum(datum, endpoint);
|
||||||
const timeSystemKey = this.openmct.time.timeSystem().key;
|
const timeSystemKey = this.openmct.time.timeSystem().key;
|
||||||
let timestamp = {};
|
let timestamp = {};
|
||||||
timestamp[timeSystemKey] = normalizedDatum[timeSystemKey];
|
const currentTimestamp = normalizedDatum[timeSystemKey];
|
||||||
|
timestamp[timeSystemKey] = currentTimestamp;
|
||||||
|
if (this.shouldEvaluateNewTelemetry(currentTimestamp)) {
|
||||||
|
this.updateConditionResults(normalizedDatum);
|
||||||
|
this.updateCurrentCondition(timestamp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
updateConditionResults(normalizedDatum) {
|
||||||
//We want to stop when the first condition evaluates to true.
|
//We want to stop when the first condition evaluates to true.
|
||||||
this.conditions.some((condition) => {
|
this.conditions.some((condition) => {
|
||||||
condition.updateResult(normalizedDatum);
|
condition.updateResult(normalizedDatum);
|
||||||
|
|
||||||
return condition.result === true;
|
return condition.result === true;
|
||||||
});
|
});
|
||||||
|
|
||||||
this.updateCurrentCondition(timestamp);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
updateCurrentCondition(timestamp) {
|
updateCurrentCondition(timestamp) {
|
||||||
|
@ -86,6 +86,7 @@ export default class StyleRuleManager extends EventEmitter {
|
|||||||
updateObjectStyleConfig(styleConfiguration) {
|
updateObjectStyleConfig(styleConfiguration) {
|
||||||
if (!styleConfiguration || !styleConfiguration.conditionSetIdentifier) {
|
if (!styleConfiguration || !styleConfiguration.conditionSetIdentifier) {
|
||||||
this.initialize(styleConfiguration || {});
|
this.initialize(styleConfiguration || {});
|
||||||
|
this.applyStaticStyle();
|
||||||
this.destroy();
|
this.destroy();
|
||||||
} else {
|
} else {
|
||||||
let isNewConditionSet = !this.conditionSetIdentifier
|
let isNewConditionSet = !this.conditionSetIdentifier
|
||||||
@ -158,7 +159,6 @@ export default class StyleRuleManager extends EventEmitter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
destroy() {
|
destroy() {
|
||||||
this.applyStaticStyle();
|
|
||||||
if (this.stopProvidingTelemetry) {
|
if (this.stopProvidingTelemetry) {
|
||||||
this.stopProvidingTelemetry();
|
this.stopProvidingTelemetry();
|
||||||
delete this.stopProvidingTelemetry;
|
delete this.stopProvidingTelemetry;
|
||||||
|
@ -195,11 +195,11 @@ import { TRIGGER, TRIGGER_LABEL } from "@/plugins/condition/utils/constants";
|
|||||||
import uuid from 'uuid';
|
import uuid from 'uuid';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct'],
|
|
||||||
components: {
|
components: {
|
||||||
Criterion,
|
Criterion,
|
||||||
ConditionDescription
|
ConditionDescription
|
||||||
},
|
},
|
||||||
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
currentConditionId: {
|
currentConditionId: {
|
||||||
type: String,
|
type: String,
|
||||||
|
@ -81,10 +81,10 @@ import Condition from './Condition.vue';
|
|||||||
import ConditionManager from '../ConditionManager';
|
import ConditionManager from '../ConditionManager';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct', 'domainObject'],
|
|
||||||
components: {
|
components: {
|
||||||
Condition
|
Condition
|
||||||
},
|
},
|
||||||
|
inject: ['openmct', 'domainObject'],
|
||||||
props: {
|
props: {
|
||||||
isEditing: Boolean,
|
isEditing: Boolean,
|
||||||
testData: {
|
testData: {
|
||||||
|
@ -58,11 +58,11 @@ import TestData from './TestData.vue';
|
|||||||
import ConditionCollection from './ConditionCollection.vue';
|
import ConditionCollection from './ConditionCollection.vue';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ["openmct", "domainObject"],
|
|
||||||
components: {
|
components: {
|
||||||
TestData,
|
TestData,
|
||||||
ConditionCollection
|
ConditionCollection
|
||||||
},
|
},
|
||||||
|
inject: ["openmct", "domainObject"],
|
||||||
props: {
|
props: {
|
||||||
isEditing: Boolean
|
isEditing: Boolean
|
||||||
},
|
},
|
||||||
|
@ -31,7 +31,6 @@
|
|||||||
v-model="expanded"
|
v-model="expanded"
|
||||||
class="c-tree__item__view-control"
|
class="c-tree__item__view-control"
|
||||||
:enabled="hasChildren"
|
:enabled="hasChildren"
|
||||||
:propagate="false"
|
|
||||||
/>
|
/>
|
||||||
<div class="c-tree__item__label c-object-label">
|
<div class="c-tree__item__label c-object-label">
|
||||||
<div
|
<div
|
||||||
@ -42,7 +41,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<ul
|
<ul
|
||||||
v-if="expanded"
|
v-if="expanded && !isLoading"
|
||||||
class="c-tree"
|
class="c-tree"
|
||||||
>
|
>
|
||||||
<li
|
<li
|
||||||
@ -69,10 +68,10 @@ import viewControl from '@/ui/components/viewControl.vue';
|
|||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'ConditionSetDialogTreeItem',
|
name: 'ConditionSetDialogTreeItem',
|
||||||
inject: ['openmct'],
|
|
||||||
components: {
|
components: {
|
||||||
viewControl
|
viewControl
|
||||||
},
|
},
|
||||||
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
node: {
|
node: {
|
||||||
type: Object,
|
type: Object,
|
||||||
|
@ -41,7 +41,7 @@
|
|||||||
></div>
|
></div>
|
||||||
<!-- end loading -->
|
<!-- end loading -->
|
||||||
|
|
||||||
<div v-if="(allTreeItems.length === 0) || (searchValue && filteredTreeItems.length === 0)"
|
<div v-if="shouldDisplayNoResultsText"
|
||||||
class="c-tree-and-search__no-results"
|
class="c-tree-and-search__no-results"
|
||||||
>
|
>
|
||||||
No results found
|
No results found
|
||||||
@ -63,7 +63,7 @@
|
|||||||
<!-- end main tree -->
|
<!-- end main tree -->
|
||||||
|
|
||||||
<!-- search tree -->
|
<!-- search tree -->
|
||||||
<ul v-if="searchValue"
|
<ul v-if="searchValue && !isLoading"
|
||||||
class="c-tree-and-search__tree c-tree"
|
class="c-tree-and-search__tree c-tree"
|
||||||
>
|
>
|
||||||
<condition-set-dialog-tree-item
|
<condition-set-dialog-tree-item
|
||||||
@ -80,16 +80,17 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
import debounce from 'lodash/debounce';
|
||||||
import search from '@/ui/components/search.vue';
|
import search from '@/ui/components/search.vue';
|
||||||
import ConditionSetDialogTreeItem from './ConditionSetDialogTreeItem.vue';
|
import ConditionSetDialogTreeItem from './ConditionSetDialogTreeItem.vue';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct'],
|
|
||||||
name: 'ConditionSetSelectorDialog',
|
name: 'ConditionSetSelectorDialog',
|
||||||
components: {
|
components: {
|
||||||
search,
|
search,
|
||||||
ConditionSetDialogTreeItem
|
ConditionSetDialogTreeItem
|
||||||
},
|
},
|
||||||
|
inject: ['openmct'],
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
expanded: false,
|
expanded: false,
|
||||||
@ -100,8 +101,20 @@ export default {
|
|||||||
selectedItem: undefined
|
selectedItem: undefined
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
computed: {
|
||||||
|
shouldDisplayNoResultsText() {
|
||||||
|
if (this.isLoading) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.allTreeItems.length === 0
|
||||||
|
|| (this.searchValue && this.filteredTreeItems.length === 0);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
created() {
|
||||||
|
this.getDebouncedFilteredChildren = debounce(this.getFilteredChildren, 400);
|
||||||
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
this.searchService = this.openmct.$injector.get('searchService');
|
|
||||||
this.getAllChildren();
|
this.getAllChildren();
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
@ -124,37 +137,44 @@ export default {
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
getFilteredChildren() {
|
getFilteredChildren() {
|
||||||
this.searchService.query(this.searchValue).then(children => {
|
// clear any previous search results
|
||||||
this.filteredTreeItems = children.hits.map(child => {
|
this.filteredTreeItems = [];
|
||||||
|
|
||||||
let context = child.object.getCapability('context');
|
const promises = this.openmct.objects.search(this.searchValue)
|
||||||
let object = child.object.useCapability('adapter');
|
.map(promise => promise
|
||||||
let objectPath = [];
|
.then(results => this.aggregateFilteredChildren(results)));
|
||||||
let navigateToParent;
|
|
||||||
|
|
||||||
if (context) {
|
Promise.all(promises).then(() => {
|
||||||
objectPath = context.getPath().slice(1)
|
this.isLoading = false;
|
||||||
.map(oldObject => oldObject.useCapability('adapter'))
|
});
|
||||||
.reverse();
|
},
|
||||||
navigateToParent = '/browse/' + objectPath.slice(1)
|
async aggregateFilteredChildren(results) {
|
||||||
.map((parent) => this.openmct.objects.makeKeyString(parent.identifier))
|
for (const object of results) {
|
||||||
|
const objectPath = await this.openmct.objects.getOriginalPath(object.identifier);
|
||||||
|
|
||||||
|
const navigateToParent = '/browse/'
|
||||||
|
+ objectPath.slice(1)
|
||||||
|
.map(parent => this.openmct.objects.makeKeyString(parent.identifier))
|
||||||
.join('/');
|
.join('/');
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
const filteredChild = {
|
||||||
id: this.openmct.objects.makeKeyString(object.identifier),
|
id: this.openmct.objects.makeKeyString(object.identifier),
|
||||||
object,
|
object,
|
||||||
objectPath,
|
objectPath,
|
||||||
navigateToParent
|
navigateToParent
|
||||||
};
|
};
|
||||||
});
|
|
||||||
});
|
this.filteredTreeItems.push(filteredChild);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
searchTree(value) {
|
searchTree(value) {
|
||||||
this.searchValue = value;
|
this.searchValue = value;
|
||||||
|
this.isLoading = true;
|
||||||
|
|
||||||
if (this.searchValue !== '') {
|
if (this.searchValue !== '') {
|
||||||
this.getFilteredChildren();
|
this.getDebouncedFilteredChildren();
|
||||||
|
} else {
|
||||||
|
this.isLoading = false;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
handleItemSelection(item, node) {
|
handleItemSelection(item, node) {
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
|
|
||||||
import { createOpenMct, resetApplicationState } from "utils/testing";
|
import { createOpenMct, resetApplicationState } from "utils/testing";
|
||||||
import ConditionPlugin from "./plugin";
|
import ConditionPlugin from "./plugin";
|
||||||
|
import stylesManager from '@/ui/inspector/styles/StylesManager';
|
||||||
import StylesView from "./components/inspector/StylesView.vue";
|
import StylesView from "./components/inspector/StylesView.vue";
|
||||||
import Vue from 'vue';
|
import Vue from 'vue';
|
||||||
import {getApplicableStylesForItem} from "./utils/styleUtils";
|
import {getApplicableStylesForItem} from "./utils/styleUtils";
|
||||||
@ -135,7 +136,7 @@ describe('the plugin', function () {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const applicableViews = openmct.objectViews.get(testViewObject);
|
const applicableViews = openmct.objectViews.get(testViewObject, []);
|
||||||
let conditionSetView = applicableViews.find((viewProvider) => viewProvider.key === 'conditionSet.view');
|
let conditionSetView = applicableViews.find((viewProvider) => viewProvider.key === 'conditionSet.view');
|
||||||
expect(conditionSetView).toBeDefined();
|
expect(conditionSetView).toBeDefined();
|
||||||
});
|
});
|
||||||
@ -400,14 +401,15 @@ describe('the plugin', function () {
|
|||||||
let viewContainer = document.createElement('div');
|
let viewContainer = document.createElement('div');
|
||||||
child.append(viewContainer);
|
child.append(viewContainer);
|
||||||
component = new Vue({
|
component = new Vue({
|
||||||
provide: {
|
|
||||||
openmct: openmct,
|
|
||||||
selection: selection
|
|
||||||
},
|
|
||||||
el: viewContainer,
|
el: viewContainer,
|
||||||
components: {
|
components: {
|
||||||
StylesView
|
StylesView
|
||||||
},
|
},
|
||||||
|
provide: {
|
||||||
|
openmct: openmct,
|
||||||
|
selection: selection,
|
||||||
|
stylesManager
|
||||||
|
},
|
||||||
template: '<styles-view/>'
|
template: '<styles-view/>'
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -541,7 +543,6 @@ describe('the plugin', function () {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should evaluate as stale when telemetry is not received in the allotted time', (done) => {
|
it('should evaluate as stale when telemetry is not received in the allotted time', (done) => {
|
||||||
|
|
||||||
let conditionMgr = new ConditionManager(conditionSetDomainObject, openmct);
|
let conditionMgr = new ConditionManager(conditionSetDomainObject, openmct);
|
||||||
conditionMgr.on('conditionSetResultUpdated', mockListener);
|
conditionMgr.on('conditionSetResultUpdated', mockListener);
|
||||||
conditionMgr.telemetryObjects = {
|
conditionMgr.telemetryObjects = {
|
||||||
@ -563,7 +564,7 @@ describe('the plugin', function () {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should not evaluate as stale when telemetry is received in the allotted time', (done) => {
|
it('should not evaluate as stale when telemetry is received in the allotted time', (done) => {
|
||||||
const date = Date.now();
|
const date = 1;
|
||||||
conditionSetDomainObject.configuration.conditionCollection[0].configuration.criteria[0].input = ["0.4"];
|
conditionSetDomainObject.configuration.conditionCollection[0].configuration.criteria[0].input = ["0.4"];
|
||||||
let conditionMgr = new ConditionManager(conditionSetDomainObject, openmct);
|
let conditionMgr = new ConditionManager(conditionSetDomainObject, openmct);
|
||||||
conditionMgr.on('conditionSetResultUpdated', mockListener);
|
conditionMgr.on('conditionSetResultUpdated', mockListener);
|
||||||
|
@ -56,14 +56,14 @@ define([
|
|||||||
return {
|
return {
|
||||||
show: function (element) {
|
show: function (element) {
|
||||||
component = new Vue({
|
component = new Vue({
|
||||||
provide: {
|
|
||||||
openmct,
|
|
||||||
objectPath
|
|
||||||
},
|
|
||||||
el: element,
|
el: element,
|
||||||
components: {
|
components: {
|
||||||
AlphanumericFormatView: AlphanumericFormatView.default
|
AlphanumericFormatView: AlphanumericFormatView.default
|
||||||
},
|
},
|
||||||
|
provide: {
|
||||||
|
openmct,
|
||||||
|
objectPath
|
||||||
|
},
|
||||||
template: '<alphanumeric-format-view ref="alphanumericFormatView"></alphanumeric-format-view>'
|
template: '<alphanumeric-format-view ref="alphanumericFormatView"></alphanumeric-format-view>'
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
@ -51,11 +51,11 @@ export default {
|
|||||||
height: 5
|
height: 5
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
inject: ['openmct'],
|
|
||||||
components: {
|
components: {
|
||||||
LayoutFrame
|
LayoutFrame
|
||||||
},
|
},
|
||||||
mixins: [conditionalStylesMixin],
|
mixins: [conditionalStylesMixin],
|
||||||
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
item: {
|
item: {
|
||||||
type: Object,
|
type: Object,
|
||||||
|
@ -152,10 +152,7 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
let domainObject = JSON.parse(JSON.stringify(this.domainObject));
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
internalDomainObject: domainObject,
|
|
||||||
initSelectIndex: undefined,
|
initSelectIndex: undefined,
|
||||||
selection: [],
|
selection: [],
|
||||||
showGrid: true
|
showGrid: true
|
||||||
@ -163,10 +160,10 @@ export default {
|
|||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
gridSize() {
|
gridSize() {
|
||||||
return this.internalDomainObject.configuration.layoutGrid;
|
return this.domainObject.configuration.layoutGrid;
|
||||||
},
|
},
|
||||||
layoutItems() {
|
layoutItems() {
|
||||||
return this.internalDomainObject.configuration.items;
|
return this.domainObject.configuration.items;
|
||||||
},
|
},
|
||||||
selectedLayoutItems() {
|
selectedLayoutItems() {
|
||||||
return this.layoutItems.filter(item => {
|
return this.layoutItems.filter(item => {
|
||||||
@ -174,7 +171,7 @@ export default {
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
layoutDimensions() {
|
layoutDimensions() {
|
||||||
return this.internalDomainObject.configuration.layoutDimensions;
|
return this.domainObject.configuration.layoutDimensions;
|
||||||
},
|
},
|
||||||
shouldDisplayLayoutDimensions() {
|
shouldDisplayLayoutDimensions() {
|
||||||
return this.layoutDimensions
|
return this.layoutDimensions
|
||||||
@ -206,12 +203,9 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
this.unlisten = this.openmct.objects.observe(this.internalDomainObject, '*', function (obj) {
|
|
||||||
this.internalDomainObject = JSON.parse(JSON.stringify(obj));
|
|
||||||
}.bind(this));
|
|
||||||
this.openmct.selection.on('change', this.setSelection);
|
this.openmct.selection.on('change', this.setSelection);
|
||||||
this.initializeItems();
|
this.initializeItems();
|
||||||
this.composition = this.openmct.composition.get(this.internalDomainObject);
|
this.composition = this.openmct.composition.get(this.domainObject);
|
||||||
this.composition.on('add', this.addChild);
|
this.composition.on('add', this.addChild);
|
||||||
this.composition.on('remove', this.removeChild);
|
this.composition.on('remove', this.removeChild);
|
||||||
this.composition.load();
|
this.composition.load();
|
||||||
@ -220,7 +214,6 @@ export default {
|
|||||||
this.openmct.selection.off('change', this.setSelection);
|
this.openmct.selection.off('change', this.setSelection);
|
||||||
this.composition.off('add', this.addChild);
|
this.composition.off('add', this.addChild);
|
||||||
this.composition.off('remove', this.removeChild);
|
this.composition.off('remove', this.removeChild);
|
||||||
this.unlisten();
|
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
addElement(itemType, element) {
|
addElement(itemType, element) {
|
||||||
@ -347,7 +340,7 @@ export default {
|
|||||||
this.startingMinY2 = undefined;
|
this.startingMinY2 = undefined;
|
||||||
},
|
},
|
||||||
mutate(path, value) {
|
mutate(path, value) {
|
||||||
this.openmct.objects.mutate(this.internalDomainObject, path, value);
|
this.openmct.objects.mutate(this.domainObject, path, value);
|
||||||
},
|
},
|
||||||
handleDrop($event) {
|
handleDrop($event) {
|
||||||
if (!$event.dataTransfer.types.includes('openmct/domain-object-path')) {
|
if (!$event.dataTransfer.types.includes('openmct/domain-object-path')) {
|
||||||
@ -387,11 +380,11 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
containsObject(identifier) {
|
containsObject(identifier) {
|
||||||
return _.get(this.internalDomainObject, 'composition')
|
return _.get(this.domainObject, 'composition')
|
||||||
.some(childId => this.openmct.objects.areIdsEqual(childId, identifier));
|
.some(childId => this.openmct.objects.areIdsEqual(childId, identifier));
|
||||||
},
|
},
|
||||||
handleDragOver($event) {
|
handleDragOver($event) {
|
||||||
if (this.internalDomainObject.locked) {
|
if (this.domainObject.locked) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -420,7 +413,7 @@ export default {
|
|||||||
item.id = uuid();
|
item.id = uuid();
|
||||||
this.trackItem(item);
|
this.trackItem(item);
|
||||||
this.layoutItems.push(item);
|
this.layoutItems.push(item);
|
||||||
this.openmct.objects.mutate(this.internalDomainObject, "configuration.items", this.layoutItems);
|
this.openmct.objects.mutate(this.domainObject, "configuration.items", this.layoutItems);
|
||||||
this.initSelectIndex = this.layoutItems.length - 1;
|
this.initSelectIndex = this.layoutItems.length - 1;
|
||||||
},
|
},
|
||||||
trackItem(item) {
|
trackItem(item) {
|
||||||
@ -477,7 +470,7 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
removeFromComposition(keyString) {
|
removeFromComposition(keyString) {
|
||||||
let composition = _.get(this.internalDomainObject, 'composition');
|
let composition = _.get(this.domainObject, 'composition');
|
||||||
composition = composition.filter(identifier => {
|
composition = composition.filter(identifier => {
|
||||||
return this.openmct.objects.makeKeyString(identifier) !== keyString;
|
return this.openmct.objects.makeKeyString(identifier) !== keyString;
|
||||||
});
|
});
|
||||||
@ -629,10 +622,10 @@ export default {
|
|||||||
createNewDomainObject(domainObject, composition, viewType, nameExtension, model) {
|
createNewDomainObject(domainObject, composition, viewType, nameExtension, model) {
|
||||||
let identifier = {
|
let identifier = {
|
||||||
key: uuid(),
|
key: uuid(),
|
||||||
namespace: this.internalDomainObject.identifier.namespace
|
namespace: this.domainObject.identifier.namespace
|
||||||
};
|
};
|
||||||
let type = this.openmct.types.get(viewType);
|
let type = this.openmct.types.get(viewType);
|
||||||
let parentKeyString = this.openmct.objects.makeKeyString(this.internalDomainObject.identifier);
|
let parentKeyString = this.openmct.objects.makeKeyString(this.domainObject.identifier);
|
||||||
let objectName = nameExtension ? `${domainObject.name}-${nameExtension}` : domainObject.name;
|
let objectName = nameExtension ? `${domainObject.name}-${nameExtension}` : domainObject.name;
|
||||||
let object = {};
|
let object = {};
|
||||||
|
|
||||||
@ -689,7 +682,7 @@ export default {
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
duplicateItem(selectedItems) {
|
duplicateItem(selectedItems) {
|
||||||
let objectStyles = this.internalDomainObject.configuration.objectStyles || {};
|
let objectStyles = this.domainObject.configuration.objectStyles || {};
|
||||||
let selectItemsArray = [];
|
let selectItemsArray = [];
|
||||||
let newDomainObjectsArray = [];
|
let newDomainObjectsArray = [];
|
||||||
|
|
||||||
@ -728,8 +721,8 @@ export default {
|
|||||||
});
|
});
|
||||||
|
|
||||||
this.$nextTick(() => {
|
this.$nextTick(() => {
|
||||||
this.openmct.objects.mutate(this.internalDomainObject, "configuration.items", this.layoutItems);
|
this.openmct.objects.mutate(this.domainObject, "configuration.items", this.layoutItems);
|
||||||
this.openmct.objects.mutate(this.internalDomainObject, "configuration.objectStyles", objectStyles);
|
this.openmct.objects.mutate(this.domainObject, "configuration.objectStyles", objectStyles);
|
||||||
this.$el.click(); //clear selection;
|
this.$el.click(); //clear selection;
|
||||||
|
|
||||||
newDomainObjectsArray.forEach(domainObject => {
|
newDomainObjectsArray.forEach(domainObject => {
|
||||||
@ -768,13 +761,13 @@ export default {
|
|||||||
};
|
};
|
||||||
this.createNewDomainObject(mockDomainObject, overlayPlotIdentifiers, viewType).then((newDomainObject) => {
|
this.createNewDomainObject(mockDomainObject, overlayPlotIdentifiers, viewType).then((newDomainObject) => {
|
||||||
let newDomainObjectKeyString = this.openmct.objects.makeKeyString(newDomainObject.identifier);
|
let newDomainObjectKeyString = this.openmct.objects.makeKeyString(newDomainObject.identifier);
|
||||||
let internalDomainObjectKeyString = this.openmct.objects.makeKeyString(this.internalDomainObject.identifier);
|
let domainObjectKeyString = this.openmct.objects.makeKeyString(this.domainObject.identifier);
|
||||||
|
|
||||||
this.composition.add(newDomainObject);
|
this.composition.add(newDomainObject);
|
||||||
this.addItem('subobject-view', newDomainObject, position);
|
this.addItem('subobject-view', newDomainObject, position);
|
||||||
|
|
||||||
overlayPlots.forEach(overlayPlot => {
|
overlayPlots.forEach(overlayPlot => {
|
||||||
if (overlayPlot.location === internalDomainObjectKeyString) {
|
if (overlayPlot.location === domainObjectKeyString) {
|
||||||
this.openmct.objects.mutate(overlayPlot, 'location', newDomainObjectKeyString);
|
this.openmct.objects.mutate(overlayPlot, 'location', newDomainObjectKeyString);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -51,11 +51,11 @@ export default {
|
|||||||
url: element.url
|
url: element.url
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
inject: ['openmct'],
|
|
||||||
components: {
|
components: {
|
||||||
LayoutFrame
|
LayoutFrame
|
||||||
},
|
},
|
||||||
mixins: [conditionalStylesMixin],
|
mixins: [conditionalStylesMixin],
|
||||||
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
item: {
|
item: {
|
||||||
type: Object,
|
type: Object,
|
||||||
|
@ -99,8 +99,8 @@ export default {
|
|||||||
stroke: '#717171'
|
stroke: '#717171'
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
inject: ['openmct'],
|
|
||||||
mixins: [conditionalStylesMixin],
|
mixins: [conditionalStylesMixin],
|
||||||
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
item: {
|
item: {
|
||||||
type: Object,
|
type: Object,
|
||||||
|
@ -80,11 +80,11 @@ export default {
|
|||||||
viewKey
|
viewKey
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
inject: ['openmct', 'objectPath'],
|
|
||||||
components: {
|
components: {
|
||||||
ObjectFrame,
|
ObjectFrame,
|
||||||
LayoutFrame
|
LayoutFrame
|
||||||
},
|
},
|
||||||
|
inject: ['openmct', 'objectPath'],
|
||||||
props: {
|
props: {
|
||||||
item: {
|
item: {
|
||||||
type: Object,
|
type: Object,
|
||||||
@ -109,7 +109,8 @@ export default {
|
|||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
domainObject: undefined,
|
domainObject: undefined,
|
||||||
currentObjectPath: []
|
currentObjectPath: [],
|
||||||
|
mutablePromise: undefined
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
@ -129,17 +130,31 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
|
if (this.openmct.objects.supportsMutation(this.item.identifier)) {
|
||||||
|
this.mutablePromise = this.openmct.objects.getMutable(this.item.identifier)
|
||||||
|
.then(this.setObject);
|
||||||
|
} else {
|
||||||
this.openmct.objects.get(this.item.identifier)
|
this.openmct.objects.get(this.item.identifier)
|
||||||
.then(this.setObject);
|
.then(this.setObject);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
destroyed() {
|
beforeDestroy() {
|
||||||
if (this.removeSelectable) {
|
if (this.removeSelectable) {
|
||||||
this.removeSelectable();
|
this.removeSelectable();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this.mutablePromise) {
|
||||||
|
this.mutablePromise.then(() => {
|
||||||
|
this.openmct.objects.destroyMutable(this.domainObject);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
this.openmct.objects.destroyMutable(this.domainObject);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
setObject(domainObject) {
|
setObject(domainObject) {
|
||||||
this.domainObject = domainObject;
|
this.domainObject = domainObject;
|
||||||
|
this.mutablePromise = undefined;
|
||||||
this.currentObjectPath = [this.domainObject].concat(this.objectPath.slice());
|
this.currentObjectPath = [this.domainObject].concat(this.objectPath.slice());
|
||||||
this.$nextTick(() => {
|
this.$nextTick(() => {
|
||||||
let reference = this.$refs.objectFrame;
|
let reference = this.$refs.objectFrame;
|
||||||
|
@ -98,11 +98,11 @@ export default {
|
|||||||
font: 'default'
|
font: 'default'
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
inject: ['openmct', 'objectPath'],
|
|
||||||
components: {
|
components: {
|
||||||
LayoutFrame
|
LayoutFrame
|
||||||
},
|
},
|
||||||
mixins: [conditionalStylesMixin],
|
mixins: [conditionalStylesMixin],
|
||||||
|
inject: ['openmct', 'objectPath'],
|
||||||
props: {
|
props: {
|
||||||
item: {
|
item: {
|
||||||
type: Object,
|
type: Object,
|
||||||
@ -131,7 +131,8 @@ export default {
|
|||||||
domainObject: undefined,
|
domainObject: undefined,
|
||||||
formats: undefined,
|
formats: undefined,
|
||||||
viewKey: `alphanumeric-format-${Math.random()}`,
|
viewKey: `alphanumeric-format-${Math.random()}`,
|
||||||
status: ''
|
status: '',
|
||||||
|
mutablePromise: undefined
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
@ -212,14 +213,20 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
|
if (this.openmct.objects.supportsMutation(this.item.identifier)) {
|
||||||
|
this.mutablePromise = this.openmct.objects.getMutable(this.item.identifier)
|
||||||
|
.then(this.setObject);
|
||||||
|
} else {
|
||||||
this.openmct.objects.get(this.item.identifier)
|
this.openmct.objects.get(this.item.identifier)
|
||||||
.then(this.setObject);
|
.then(this.setObject);
|
||||||
|
}
|
||||||
|
|
||||||
this.openmct.time.on("bounds", this.refreshData);
|
this.openmct.time.on("bounds", this.refreshData);
|
||||||
|
|
||||||
this.status = this.openmct.status.get(this.item.identifier);
|
this.status = this.openmct.status.get(this.item.identifier);
|
||||||
this.removeStatusListener = this.openmct.status.observe(this.item.identifier, this.setStatus);
|
this.removeStatusListener = this.openmct.status.observe(this.item.identifier, this.setStatus);
|
||||||
},
|
},
|
||||||
destroyed() {
|
beforeDestroy() {
|
||||||
this.removeSubscription();
|
this.removeSubscription();
|
||||||
this.removeStatusListener();
|
this.removeStatusListener();
|
||||||
|
|
||||||
@ -228,13 +235,22 @@ export default {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this.openmct.time.off("bounds", this.refreshData);
|
this.openmct.time.off("bounds", this.refreshData);
|
||||||
|
|
||||||
|
if (this.mutablePromise) {
|
||||||
|
this.mutablePromise.then(() => {
|
||||||
|
this.openmct.objects.destroyMutable(this.domainObject);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
this.openmct.objects.destroyMutable(this.domainObject);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
formattedValueForCopy() {
|
formattedValueForCopy() {
|
||||||
const timeFormatterKey = this.openmct.time.timeSystem().key;
|
const timeFormatterKey = this.openmct.time.timeSystem().key;
|
||||||
const timeFormatter = this.formats[timeFormatterKey];
|
const timeFormatter = this.formats[timeFormatterKey];
|
||||||
|
const unit = this.unit ? ` ${this.unit}` : '';
|
||||||
|
|
||||||
return `At ${timeFormatter.format(this.datum)} ${this.domainObject.name} had a value of ${this.telemetryValue} ${this.unit}`;
|
return `At ${timeFormatter.format(this.datum)} ${this.domainObject.name} had a value of ${this.telemetryValue}${unit}`;
|
||||||
},
|
},
|
||||||
requestHistoricalData() {
|
requestHistoricalData() {
|
||||||
let bounds = this.openmct.time.bounds();
|
let bounds = this.openmct.time.bounds();
|
||||||
@ -285,6 +301,7 @@ export default {
|
|||||||
},
|
},
|
||||||
setObject(domainObject) {
|
setObject(domainObject) {
|
||||||
this.domainObject = domainObject;
|
this.domainObject = domainObject;
|
||||||
|
this.mutablePromise = undefined;
|
||||||
this.keyString = this.openmct.objects.makeKeyString(domainObject.identifier);
|
this.keyString = this.openmct.objects.makeKeyString(domainObject.identifier);
|
||||||
this.metadata = this.openmct.telemetry.getMetadata(this.domainObject);
|
this.metadata = this.openmct.telemetry.getMetadata(this.domainObject);
|
||||||
this.limitEvaluator = this.openmct.telemetry.limitEvaluator(this.domainObject);
|
this.limitEvaluator = this.openmct.telemetry.limitEvaluator(this.domainObject);
|
||||||
|
@ -59,11 +59,11 @@ export default {
|
|||||||
font: 'default'
|
font: 'default'
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
inject: ['openmct'],
|
|
||||||
components: {
|
components: {
|
||||||
LayoutFrame
|
LayoutFrame
|
||||||
},
|
},
|
||||||
mixins: [conditionalStylesMixin],
|
mixins: [conditionalStylesMixin],
|
||||||
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
item: {
|
item: {
|
||||||
type: Object,
|
type: Object,
|
||||||
|
@ -83,7 +83,7 @@ describe('the plugin', function () {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const applicableViews = openmct.objectViews.get(testViewObject);
|
const applicableViews = openmct.objectViews.get(testViewObject, []);
|
||||||
let displayLayoutViewProvider = applicableViews.find((viewProvider) => viewProvider.key === 'layout.view');
|
let displayLayoutViewProvider = applicableViews.find((viewProvider) => viewProvider.key === 'layout.view');
|
||||||
expect(displayLayoutViewProvider).toBeDefined();
|
expect(displayLayoutViewProvider).toBeDefined();
|
||||||
});
|
});
|
||||||
|
@ -89,7 +89,7 @@ export default class DuplicateAction {
|
|||||||
{
|
{
|
||||||
key: "name",
|
key: "name",
|
||||||
control: "textfield",
|
control: "textfield",
|
||||||
name: "Folder Name",
|
name: "Name",
|
||||||
pattern: "\\S+",
|
pattern: "\\S+",
|
||||||
required: true,
|
required: true,
|
||||||
cssClass: "l-input-lg"
|
cssClass: "l-input-lg"
|
||||||
|
@ -48,13 +48,14 @@ export default class DuplicateTask {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Execute the duplicate/copy task with the objects provided in the constructor.
|
* Execute the duplicate/copy task with the objects provided.
|
||||||
* @returns {promise} Which will resolve with a clone of the object
|
* @returns {promise} Which will resolve with a clone of the object
|
||||||
* once complete.
|
* once complete.
|
||||||
*/
|
*/
|
||||||
async duplicate(domainObject, parent, filter) {
|
async duplicate(domainObject, parent, filter) {
|
||||||
this.domainObject = domainObject;
|
this.domainObject = domainObject;
|
||||||
this.parent = parent;
|
this.parent = parent;
|
||||||
|
this.namespace = parent.identifier.namespace;
|
||||||
this.filter = filter || this.isCreatable;
|
this.filter = filter || this.isCreatable;
|
||||||
|
|
||||||
await this.buildDuplicationPlan();
|
await this.buildDuplicationPlan();
|
||||||
@ -78,8 +79,9 @@ export default class DuplicateTask {
|
|||||||
*/
|
*/
|
||||||
async buildDuplicationPlan() {
|
async buildDuplicationPlan() {
|
||||||
let domainObjectClone = await this.duplicateObject(this.domainObject);
|
let domainObjectClone = await this.duplicateObject(this.domainObject);
|
||||||
|
|
||||||
if (domainObjectClone !== this.domainObject) {
|
if (domainObjectClone !== this.domainObject) {
|
||||||
domainObjectClone.location = this.getId(this.parent);
|
domainObjectClone.location = this.getKeyString(this.parent);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.firstClone = domainObjectClone;
|
this.firstClone = domainObjectClone;
|
||||||
@ -96,13 +98,14 @@ export default class DuplicateTask {
|
|||||||
let initialCount = this.clones.length;
|
let initialCount = this.clones.length;
|
||||||
let dialog = this.openmct.overlays.progressDialog({
|
let dialog = this.openmct.overlays.progressDialog({
|
||||||
progressPerc: 0,
|
progressPerc: 0,
|
||||||
message: `Duplicating ${initialCount} files.`,
|
message: `Duplicating ${initialCount} objects.`,
|
||||||
iconClass: 'info',
|
iconClass: 'info',
|
||||||
title: 'Duplicating'
|
title: 'Duplicating'
|
||||||
});
|
});
|
||||||
let clonesDone = Promise.all(this.clones.map(clone => {
|
|
||||||
|
let clonesDone = Promise.all(this.clones.map((clone) => {
|
||||||
let percentPersisted = Math.ceil(100 * (++this.persisted / initialCount));
|
let percentPersisted = Math.ceil(100 * (++this.persisted / initialCount));
|
||||||
let message = `Duplicating ${initialCount - this.persisted} files.`;
|
let message = `Duplicating ${initialCount - this.persisted} objects.`;
|
||||||
|
|
||||||
dialog.updateProgress(percentPersisted, message);
|
dialog.updateProgress(percentPersisted, message);
|
||||||
|
|
||||||
@ -110,6 +113,7 @@ export default class DuplicateTask {
|
|||||||
}));
|
}));
|
||||||
|
|
||||||
await clonesDone;
|
await clonesDone;
|
||||||
|
|
||||||
dialog.dismiss();
|
dialog.dismiss();
|
||||||
this.openmct.notifications.info(`Duplicated ${this.persisted} objects.`);
|
this.openmct.notifications.info(`Duplicated ${this.persisted} objects.`);
|
||||||
|
|
||||||
@ -141,10 +145,7 @@ export default class DuplicateTask {
|
|||||||
async duplicateObject(originalObject) {
|
async duplicateObject(originalObject) {
|
||||||
// Check if the creatable (or other passed in filter).
|
// Check if the creatable (or other passed in filter).
|
||||||
if (this.filter(originalObject)) {
|
if (this.filter(originalObject)) {
|
||||||
// Clone original object
|
|
||||||
let clone = this.cloneObjectModel(originalObject);
|
let clone = this.cloneObjectModel(originalObject);
|
||||||
|
|
||||||
// Get children, if any
|
|
||||||
let composeesCollection = this.openmct.composition.get(originalObject);
|
let composeesCollection = this.openmct.composition.get(originalObject);
|
||||||
let composees;
|
let composees;
|
||||||
|
|
||||||
@ -152,7 +153,6 @@ export default class DuplicateTask {
|
|||||||
composees = await composeesCollection.load();
|
composees = await composeesCollection.load();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Recursively duplicate children
|
|
||||||
return this.duplicateComposees(clone, composees);
|
return this.duplicateComposees(clone, composees);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -160,36 +160,6 @@ export default class DuplicateTask {
|
|||||||
return originalObject;
|
return originalObject;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Update identifiers in a cloned object model (or part of
|
|
||||||
* a cloned object model) to reflect new identifiers after
|
|
||||||
* duplicating.
|
|
||||||
* @private
|
|
||||||
*/
|
|
||||||
rewriteIdentifiers(obj, idMap) {
|
|
||||||
function lookupValue(value) {
|
|
||||||
return (typeof value === 'string' && idMap[value]) || value;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Array.isArray(obj)) {
|
|
||||||
obj.forEach((value, index) => {
|
|
||||||
obj[index] = lookupValue(value);
|
|
||||||
this.rewriteIdentifiers(obj[index], idMap);
|
|
||||||
});
|
|
||||||
} else if (obj && typeof obj === 'object') {
|
|
||||||
Object.keys(obj).forEach((key) => {
|
|
||||||
let value = obj[key];
|
|
||||||
obj[key] = lookupValue(value);
|
|
||||||
if (idMap[key]) {
|
|
||||||
delete obj[key];
|
|
||||||
obj[idMap[key]] = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.rewriteIdentifiers(value, idMap);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Given an array of objects composed by a parent, clone them, then
|
* Given an array of objects composed by a parent, clone them, then
|
||||||
* add them to the parent.
|
* add them to the parent.
|
||||||
@ -197,34 +167,67 @@ export default class DuplicateTask {
|
|||||||
* @returns {*}
|
* @returns {*}
|
||||||
*/
|
*/
|
||||||
async duplicateComposees(clonedParent, composees = []) {
|
async duplicateComposees(clonedParent, composees = []) {
|
||||||
let idMap = {};
|
let idMappings = [];
|
||||||
|
|
||||||
let allComposeesDuplicated = composees.reduce(async (previousPromise, nextComposee) => {
|
let allComposeesDuplicated = composees.reduce(async (previousPromise, nextComposee) => {
|
||||||
await previousPromise;
|
await previousPromise;
|
||||||
|
|
||||||
let clonedComposee = await this.duplicateObject(nextComposee);
|
let clonedComposee = await this.duplicateObject(nextComposee);
|
||||||
idMap[this.getId(nextComposee)] = this.getId(clonedComposee);
|
|
||||||
await this.composeChild(clonedComposee, clonedParent, clonedComposee !== nextComposee);
|
if (clonedComposee) {
|
||||||
|
idMappings.push({
|
||||||
|
newId: clonedComposee.identifier,
|
||||||
|
oldId: nextComposee.identifier
|
||||||
|
});
|
||||||
|
this.composeChild(clonedComposee, clonedParent, clonedComposee !== nextComposee);
|
||||||
|
}
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}, Promise.resolve());
|
}, Promise.resolve());
|
||||||
|
|
||||||
await allComposeesDuplicated;
|
await allComposeesDuplicated;
|
||||||
|
|
||||||
this.rewriteIdentifiers(clonedParent, idMap);
|
clonedParent = this.rewriteIdentifiers(clonedParent, idMappings);
|
||||||
this.clones.push(clonedParent);
|
this.clones.push(clonedParent);
|
||||||
|
|
||||||
return clonedParent;
|
return clonedParent;
|
||||||
}
|
}
|
||||||
|
|
||||||
async composeChild(child, parent, setLocation) {
|
/**
|
||||||
const PERSIST_BOOL = false;
|
* Update identifiers in a cloned object model (or part of
|
||||||
let parentComposition = this.openmct.composition.get(parent);
|
* a cloned object model) to reflect new identifiers after
|
||||||
await parentComposition.load();
|
* duplicating.
|
||||||
parentComposition.add(child, PERSIST_BOOL);
|
* @private
|
||||||
|
*/
|
||||||
|
rewriteIdentifiers(clonedParent, childIdMappings) {
|
||||||
|
for (let { newId, oldId } of childIdMappings) {
|
||||||
|
let newIdKeyString = this.openmct.objects.makeKeyString(newId);
|
||||||
|
let oldIdKeyString = this.openmct.objects.makeKeyString(oldId);
|
||||||
|
|
||||||
|
// regex replace keystrings
|
||||||
|
clonedParent = JSON.stringify(clonedParent).replace(new RegExp(oldIdKeyString, 'g'), newIdKeyString);
|
||||||
|
|
||||||
|
// parse reviver to replace identifiers
|
||||||
|
clonedParent = JSON.parse(clonedParent, (key, value) => {
|
||||||
|
if (Object.prototype.hasOwnProperty.call(value, 'key')
|
||||||
|
&& Object.prototype.hasOwnProperty.call(value, 'namespace')
|
||||||
|
&& value.key === oldId.key
|
||||||
|
&& value.namespace === oldId.namespace) {
|
||||||
|
return newId;
|
||||||
|
} else {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return clonedParent;
|
||||||
|
}
|
||||||
|
|
||||||
|
composeChild(child, parent, setLocation) {
|
||||||
|
parent.composition.push(child.identifier);
|
||||||
|
|
||||||
//If a location is not specified, set it.
|
//If a location is not specified, set it.
|
||||||
if (setLocation && child.location === undefined) {
|
if (setLocation && child.location === undefined) {
|
||||||
let parentKeyString = this.getId(parent);
|
let parentKeyString = this.getKeyString(parent);
|
||||||
child.location = parentKeyString;
|
child.location = parentKeyString;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -239,7 +242,7 @@ export default class DuplicateTask {
|
|||||||
let clone = JSON.parse(JSON.stringify(domainObject));
|
let clone = JSON.parse(JSON.stringify(domainObject));
|
||||||
let identifier = {
|
let identifier = {
|
||||||
key: uuid(),
|
key: uuid(),
|
||||||
namespace: domainObject.identifier.namespace
|
namespace: this.namespace // set to NEW parent's namespace
|
||||||
};
|
};
|
||||||
|
|
||||||
if (clone.modified || clone.persisted || clone.location) {
|
if (clone.modified || clone.persisted || clone.location) {
|
||||||
@ -260,7 +263,7 @@ export default class DuplicateTask {
|
|||||||
return clone;
|
return clone;
|
||||||
}
|
}
|
||||||
|
|
||||||
getId(domainObject) {
|
getKeyString(domainObject) {
|
||||||
return this.openmct.objects.makeKeyString(domainObject.identifier);
|
return this.openmct.objects.makeKeyString(domainObject.identifier);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -57,6 +57,7 @@ describe("The Duplicate Action plugin", () => {
|
|||||||
overwrite: {
|
overwrite: {
|
||||||
folder: {
|
folder: {
|
||||||
name: "Parent Folder",
|
name: "Parent Folder",
|
||||||
|
type: "folder",
|
||||||
composition: [childObject.identifier]
|
composition: [childObject.identifier]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -104,6 +105,7 @@ describe("The Duplicate Action plugin", () => {
|
|||||||
|
|
||||||
// already installed by default, but never hurts, just adds to context menu
|
// already installed by default, but never hurts, just adds to context menu
|
||||||
openmct.install(DuplicateActionPlugin());
|
openmct.install(DuplicateActionPlugin());
|
||||||
|
openmct.types.addType('folder', {creatable: true});
|
||||||
|
|
||||||
openmct.on('start', done);
|
openmct.on('start', done);
|
||||||
openmct.startHeadless();
|
openmct.startHeadless();
|
||||||
|
@ -47,13 +47,13 @@ define([
|
|||||||
return {
|
return {
|
||||||
show: function (element) {
|
show: function (element) {
|
||||||
component = new Vue({
|
component = new Vue({
|
||||||
provide: {
|
|
||||||
openmct
|
|
||||||
},
|
|
||||||
el: element,
|
el: element,
|
||||||
components: {
|
components: {
|
||||||
FiltersView: FiltersView.default
|
FiltersView: FiltersView.default
|
||||||
},
|
},
|
||||||
|
provide: {
|
||||||
|
openmct
|
||||||
|
},
|
||||||
template: '<filters-view></filters-view>'
|
template: '<filters-view></filters-view>'
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
@ -65,11 +65,11 @@ import ToggleSwitch from '../../../ui/components/ToggleSwitch.vue';
|
|||||||
import isEmpty from 'lodash/isEmpty';
|
import isEmpty from 'lodash/isEmpty';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct'],
|
|
||||||
components: {
|
components: {
|
||||||
FilterField,
|
FilterField,
|
||||||
ToggleSwitch
|
ToggleSwitch
|
||||||
},
|
},
|
||||||
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
filterObject: {
|
filterObject: {
|
||||||
type: Object,
|
type: Object,
|
||||||
|
@ -41,10 +41,10 @@
|
|||||||
import FilterField from './FilterField.vue';
|
import FilterField from './FilterField.vue';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct'],
|
|
||||||
components: {
|
components: {
|
||||||
FilterField
|
FilterField
|
||||||
},
|
},
|
||||||
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
globalMetadata: {
|
globalMetadata: {
|
||||||
type: Object,
|
type: Object,
|
||||||
|
@ -87,12 +87,12 @@ import DropHint from './dropHint.vue';
|
|||||||
const MIN_FRAME_SIZE = 5;
|
const MIN_FRAME_SIZE = 5;
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct'],
|
|
||||||
components: {
|
components: {
|
||||||
FrameComponent,
|
FrameComponent,
|
||||||
ResizeHandle,
|
ResizeHandle,
|
||||||
DropHint
|
DropHint
|
||||||
},
|
},
|
||||||
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
container: {
|
container: {
|
||||||
type: Object,
|
type: Object,
|
||||||
|
@ -28,7 +28,7 @@
|
|||||||
></div>
|
></div>
|
||||||
|
|
||||||
<div
|
<div
|
||||||
v-if="areAllContainersEmpty()"
|
v-if="allContainersAreEmpty"
|
||||||
class="c-fl__empty"
|
class="c-fl__empty"
|
||||||
>
|
>
|
||||||
<span class="c-fl__empty-message">This Flexible Layout is currently empty</span>
|
<span class="c-fl__empty-message">This Flexible Layout is currently empty</span>
|
||||||
@ -94,7 +94,6 @@ import Container from '../utils/container';
|
|||||||
import Frame from '../utils/frame';
|
import Frame from '../utils/frame';
|
||||||
import ResizeHandle from './resizeHandle.vue';
|
import ResizeHandle from './resizeHandle.vue';
|
||||||
import DropHint from './dropHint.vue';
|
import DropHint from './dropHint.vue';
|
||||||
import RemoveAction from '../../remove/RemoveAction.js';
|
|
||||||
|
|
||||||
const MIN_CONTAINER_SIZE = 5;
|
const MIN_CONTAINER_SIZE = 5;
|
||||||
|
|
||||||
@ -140,19 +139,20 @@ function sizeToFill(items) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct', 'objectPath', 'layoutObject'],
|
|
||||||
components: {
|
components: {
|
||||||
ContainerComponent,
|
ContainerComponent,
|
||||||
ResizeHandle,
|
ResizeHandle,
|
||||||
DropHint
|
DropHint
|
||||||
},
|
},
|
||||||
|
inject: ['openmct', 'objectPath', 'layoutObject'],
|
||||||
props: {
|
props: {
|
||||||
isEditing: Boolean
|
isEditing: Boolean
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
domainObject: this.layoutObject,
|
domainObject: this.layoutObject,
|
||||||
newFrameLocation: []
|
newFrameLocation: [],
|
||||||
|
identifierMap: {}
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
@ -168,26 +168,30 @@ export default {
|
|||||||
},
|
},
|
||||||
rowsLayout() {
|
rowsLayout() {
|
||||||
return this.domainObject.configuration.rowsLayout;
|
return this.domainObject.configuration.rowsLayout;
|
||||||
|
},
|
||||||
|
allContainersAreEmpty() {
|
||||||
|
return this.containers.every(container => container.frames.length === 0);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
|
this.buildIdentifierMap();
|
||||||
this.composition = this.openmct.composition.get(this.domainObject);
|
this.composition = this.openmct.composition.get(this.domainObject);
|
||||||
this.composition.on('remove', this.removeChildObject);
|
this.composition.on('remove', this.removeChildObject);
|
||||||
this.composition.on('add', this.addFrame);
|
this.composition.on('add', this.addFrame);
|
||||||
|
this.composition.load();
|
||||||
this.RemoveAction = new RemoveAction(this.openmct);
|
|
||||||
|
|
||||||
this.unobserve = this.openmct.objects.observe(this.domainObject, '*', this.updateDomainObject);
|
|
||||||
},
|
},
|
||||||
beforeDestroy() {
|
beforeDestroy() {
|
||||||
this.composition.off('remove', this.removeChildObject);
|
this.composition.off('remove', this.removeChildObject);
|
||||||
this.composition.off('add', this.addFrame);
|
this.composition.off('add', this.addFrame);
|
||||||
|
|
||||||
this.unobserve();
|
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
areAllContainersEmpty() {
|
buildIdentifierMap() {
|
||||||
return !this.containers.filter(container => container.frames.length).length;
|
this.containers.forEach(container => {
|
||||||
|
container.frames.forEach(frame => {
|
||||||
|
let keystring = this.openmct.objects.makeKeyString(frame.domainObjectIdentifier);
|
||||||
|
this.identifierMap[keystring] = true;
|
||||||
|
});
|
||||||
|
});
|
||||||
},
|
},
|
||||||
addContainer() {
|
addContainer() {
|
||||||
let container = new Container();
|
let container = new Container();
|
||||||
@ -236,6 +240,9 @@ export default {
|
|||||||
this.newFrameLocation = [containerIndex, insertFrameIndex];
|
this.newFrameLocation = [containerIndex, insertFrameIndex];
|
||||||
},
|
},
|
||||||
addFrame(domainObject) {
|
addFrame(domainObject) {
|
||||||
|
let keystring = this.openmct.objects.makeKeyString(domainObject.identifier);
|
||||||
|
|
||||||
|
if (!this.identifierMap[keystring]) {
|
||||||
let containerIndex = this.newFrameLocation.length ? this.newFrameLocation[0] : 0;
|
let containerIndex = this.newFrameLocation.length ? this.newFrameLocation[0] : 0;
|
||||||
let container = this.containers[containerIndex];
|
let container = this.containers[containerIndex];
|
||||||
let frameIndex = this.newFrameLocation.length ? this.newFrameLocation[1] : container.frames.length;
|
let frameIndex = this.newFrameLocation.length ? this.newFrameLocation[1] : container.frames.length;
|
||||||
@ -246,6 +253,8 @@ export default {
|
|||||||
|
|
||||||
this.newFrameLocation = [];
|
this.newFrameLocation = [];
|
||||||
this.persist(containerIndex);
|
this.persist(containerIndex);
|
||||||
|
this.identifierMap[keystring] = true;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
deleteFrame(frameId) {
|
deleteFrame(frameId) {
|
||||||
let container = this.containers
|
let container = this.containers
|
||||||
@ -254,16 +263,15 @@ export default {
|
|||||||
.frames
|
.frames
|
||||||
.filter((f => f.id === frameId))[0];
|
.filter((f => f.id === frameId))[0];
|
||||||
|
|
||||||
this.removeFromComposition(frame.domainObjectIdentifier)
|
this.removeFromComposition(frame.domainObjectIdentifier);
|
||||||
.then(() => {
|
|
||||||
|
this.$nextTick().then(() => {
|
||||||
sizeToFill(container.frames);
|
sizeToFill(container.frames);
|
||||||
this.setSelectionToParent();
|
this.setSelectionToParent();
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
removeFromComposition(identifier) {
|
removeFromComposition(identifier) {
|
||||||
return this.openmct.objects.get(identifier).then((childDomainObject) => {
|
this.composition.remove({identifier});
|
||||||
this.RemoveAction.removeFromComposition(this.domainObject, childDomainObject);
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
setSelectionToParent() {
|
setSelectionToParent() {
|
||||||
this.$el.click();
|
this.$el.click();
|
||||||
@ -342,6 +350,9 @@ export default {
|
|||||||
removeChildObject(identifier) {
|
removeChildObject(identifier) {
|
||||||
let removeIdentifier = this.openmct.objects.makeKeyString(identifier);
|
let removeIdentifier = this.openmct.objects.makeKeyString(identifier);
|
||||||
|
|
||||||
|
this.identifierMap[removeIdentifier] = undefined;
|
||||||
|
delete this.identifierMap[removeIdentifier];
|
||||||
|
|
||||||
this.containers.forEach(container => {
|
this.containers.forEach(container => {
|
||||||
container.frames = container.frames.filter(frame => {
|
container.frames = container.frames.filter(frame => {
|
||||||
let frameIdentifier = this.openmct.objects.makeKeyString(frame.domainObjectIdentifier);
|
let frameIdentifier = this.openmct.objects.makeKeyString(frame.domainObjectIdentifier);
|
||||||
|
@ -58,10 +58,10 @@
|
|||||||
import ObjectFrame from '../../../ui/components/ObjectFrame.vue';
|
import ObjectFrame from '../../../ui/components/ObjectFrame.vue';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct'],
|
|
||||||
components: {
|
components: {
|
||||||
ObjectFrame
|
ObjectFrame
|
||||||
},
|
},
|
||||||
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
frame: {
|
frame: {
|
||||||
type: Object,
|
type: Object,
|
||||||
|
@ -44,15 +44,15 @@ define([
|
|||||||
return {
|
return {
|
||||||
show: function (element, isEditing) {
|
show: function (element, isEditing) {
|
||||||
component = new Vue({
|
component = new Vue({
|
||||||
|
el: element,
|
||||||
|
components: {
|
||||||
|
FlexibleLayoutComponent: FlexibleLayoutComponent.default
|
||||||
|
},
|
||||||
provide: {
|
provide: {
|
||||||
openmct,
|
openmct,
|
||||||
objectPath,
|
objectPath,
|
||||||
layoutObject: domainObject
|
layoutObject: domainObject
|
||||||
},
|
},
|
||||||
el: element,
|
|
||||||
components: {
|
|
||||||
FlexibleLayoutComponent: FlexibleLayoutComponent.default
|
|
||||||
},
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
isEditing: isEditing
|
isEditing: isEditing
|
||||||
|
@ -22,18 +22,22 @@
|
|||||||
|
|
||||||
define([
|
define([
|
||||||
'./components/GridView.vue',
|
'./components/GridView.vue',
|
||||||
|
'./constants.js',
|
||||||
'vue'
|
'vue'
|
||||||
], function (
|
], function (
|
||||||
GridViewComponent,
|
GridViewComponent,
|
||||||
|
constants,
|
||||||
Vue
|
Vue
|
||||||
) {
|
) {
|
||||||
function FolderGridView(openmct) {
|
function FolderGridView(openmct) {
|
||||||
|
const ALLOWED_FOLDER_TYPES = constants.ALLOWED_FOLDER_TYPES;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
key: 'grid',
|
key: 'grid',
|
||||||
name: 'Grid View',
|
name: 'Grid View',
|
||||||
cssClass: 'icon-thumbs-strip',
|
cssClass: 'icon-thumbs-strip',
|
||||||
canView: function (domainObject) {
|
canView: function (domainObject) {
|
||||||
return domainObject.type === 'folder';
|
return ALLOWED_FOLDER_TYPES.includes(domainObject.type);
|
||||||
},
|
},
|
||||||
view: function (domainObject) {
|
view: function (domainObject) {
|
||||||
let component;
|
let component;
|
||||||
|
@ -22,20 +22,24 @@
|
|||||||
|
|
||||||
define([
|
define([
|
||||||
'./components/ListView.vue',
|
'./components/ListView.vue',
|
||||||
|
'./constants.js',
|
||||||
'vue',
|
'vue',
|
||||||
'moment'
|
'moment'
|
||||||
], function (
|
], function (
|
||||||
ListViewComponent,
|
ListViewComponent,
|
||||||
|
constants,
|
||||||
Vue,
|
Vue,
|
||||||
Moment
|
Moment
|
||||||
) {
|
) {
|
||||||
function FolderListView(openmct) {
|
function FolderListView(openmct) {
|
||||||
|
const ALLOWED_FOLDER_TYPES = constants.ALLOWED_FOLDER_TYPES;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
key: 'list-view',
|
key: 'list-view',
|
||||||
name: 'List View',
|
name: 'List View',
|
||||||
cssClass: 'icon-list-view',
|
cssClass: 'icon-list-view',
|
||||||
canView: function (domainObject) {
|
canView: function (domainObject) {
|
||||||
return domainObject.type === 'folder';
|
return ALLOWED_FOLDER_TYPES.includes(domainObject.type);
|
||||||
},
|
},
|
||||||
view: function (domainObject) {
|
view: function (domainObject) {
|
||||||
let component;
|
let component;
|
||||||
|
@ -1,7 +1,9 @@
|
|||||||
<template>
|
<template>
|
||||||
<tr
|
<tr
|
||||||
class="c-list-item"
|
class="c-list-item"
|
||||||
:class="{ 'is-alias': item.isAlias === true }"
|
:class="{
|
||||||
|
'is-alias': item.isAlias === true
|
||||||
|
}"
|
||||||
@click="navigate"
|
@click="navigate"
|
||||||
>
|
>
|
||||||
<td class="c-list-item__name">
|
<td class="c-list-item__name">
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2020, United States Government
|
* Open MCT, Copyright (c) 2014-2021, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -20,13 +20,4 @@
|
|||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
define([
|
export const ALLOWED_FOLDER_TYPES = ['folder', 'noneditable.folder'];
|
||||||
"EventEmitter"
|
|
||||||
], function (
|
|
||||||
EventEmitter
|
|
||||||
) {
|
|
||||||
/**
|
|
||||||
* Provides a singleton event bus for sharing between objects.
|
|
||||||
*/
|
|
||||||
return new EventEmitter();
|
|
||||||
});
|
|
73
src/plugins/goToOriginalAction/pluginSpec.js
Normal file
73
src/plugins/goToOriginalAction/pluginSpec.js
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2020, 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 {
|
||||||
|
createOpenMct,
|
||||||
|
resetApplicationState
|
||||||
|
} from 'utils/testing';
|
||||||
|
|
||||||
|
describe("the plugin", () => {
|
||||||
|
let openmct;
|
||||||
|
let goToFolderAction;
|
||||||
|
let mockObjectPath;
|
||||||
|
|
||||||
|
beforeEach((done) => {
|
||||||
|
openmct = createOpenMct();
|
||||||
|
|
||||||
|
openmct.on('start', done);
|
||||||
|
openmct.startHeadless();
|
||||||
|
|
||||||
|
goToFolderAction = openmct.actions._allActions.goToOriginal;
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
return resetApplicationState(openmct);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('installs the go to folder action', () => {
|
||||||
|
expect(goToFolderAction).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when invoked', () => {
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
mockObjectPath = [{
|
||||||
|
name: 'mock folder',
|
||||||
|
type: 'folder',
|
||||||
|
identifier: {
|
||||||
|
key: 'mock-folder',
|
||||||
|
namespace: ''
|
||||||
|
}
|
||||||
|
}];
|
||||||
|
spyOn(openmct.objects, 'get').and.returnValue(Promise.resolve({
|
||||||
|
identifier: {
|
||||||
|
namespace: '',
|
||||||
|
key: 'test'
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
goToFolderAction.invoke(mockObjectPath);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('goes to the original location', () => {
|
||||||
|
expect(window.location.href).toContain('context.html#/browse/?tc.mode=fixed&tc.startBound=0&tc.endBound=1&tc.timeSystem=utc');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
@ -1,3 +1,25 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2021, 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 ImageryViewLayout from './components/ImageryViewLayout.vue';
|
import ImageryViewLayout from './components/ImageryViewLayout.vue';
|
||||||
import Vue from 'vue';
|
import Vue from 'vue';
|
||||||
|
|
||||||
|
131
src/plugins/imagery/components/Compass/Compass.vue
Normal file
131
src/plugins/imagery/components/Compass/Compass.vue
Normal file
@ -0,0 +1,131 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2021, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div
|
||||||
|
class="c-compass"
|
||||||
|
:style="compassDimensionsStyle"
|
||||||
|
>
|
||||||
|
<CompassHUD
|
||||||
|
v-if="hasCameraFieldOfView"
|
||||||
|
:heading="heading"
|
||||||
|
:sun-heading="sunHeading"
|
||||||
|
:camera-angle-of-view="cameraAngleOfView"
|
||||||
|
:camera-pan="cameraPan"
|
||||||
|
/>
|
||||||
|
<CompassRose
|
||||||
|
v-if="hasCameraFieldOfView"
|
||||||
|
:heading="heading"
|
||||||
|
:sun-heading="sunHeading"
|
||||||
|
:camera-angle-of-view="cameraAngleOfView"
|
||||||
|
:camera-pan="cameraPan"
|
||||||
|
:lock-compass="lockCompass"
|
||||||
|
@toggle-lock-compass="toggleLockCompass"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import CompassHUD from './CompassHUD.vue';
|
||||||
|
import CompassRose from './CompassRose.vue';
|
||||||
|
|
||||||
|
const CAMERA_ANGLE_OF_VIEW = 70;
|
||||||
|
|
||||||
|
export default {
|
||||||
|
components: {
|
||||||
|
CompassHUD,
|
||||||
|
CompassRose
|
||||||
|
},
|
||||||
|
props: {
|
||||||
|
containerWidth: {
|
||||||
|
type: Number,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
containerHeight: {
|
||||||
|
type: Number,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
naturalAspectRatio: {
|
||||||
|
type: Number,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
image: {
|
||||||
|
type: Object,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
lockCompass: {
|
||||||
|
type: Boolean,
|
||||||
|
required: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
hasCameraFieldOfView() {
|
||||||
|
return this.heading !== undefined && this.cameraPan !== undefined;
|
||||||
|
},
|
||||||
|
// compass direction from north in degrees
|
||||||
|
heading() {
|
||||||
|
return this.image.heading;
|
||||||
|
},
|
||||||
|
pitch() {
|
||||||
|
return this.image.pitch;
|
||||||
|
},
|
||||||
|
// compass direction from north in degrees
|
||||||
|
sunHeading() {
|
||||||
|
return this.image.sunOrientation;
|
||||||
|
},
|
||||||
|
// relative direction from heading in degrees
|
||||||
|
cameraPan() {
|
||||||
|
return this.image.cameraPan;
|
||||||
|
},
|
||||||
|
cameraTilt() {
|
||||||
|
return this.image.cameraTilt;
|
||||||
|
},
|
||||||
|
cameraAngleOfView() {
|
||||||
|
return CAMERA_ANGLE_OF_VIEW;
|
||||||
|
},
|
||||||
|
compassDimensionsStyle() {
|
||||||
|
const containerAspectRatio = this.containerWidth / this.containerHeight;
|
||||||
|
|
||||||
|
let width;
|
||||||
|
let height;
|
||||||
|
|
||||||
|
if (containerAspectRatio < this.naturalAspectRatio) {
|
||||||
|
width = '100%';
|
||||||
|
height = `${ this.containerWidth / this.naturalAspectRatio }px`;
|
||||||
|
} else {
|
||||||
|
width = `${ this.containerHeight * this.naturalAspectRatio }px`;
|
||||||
|
height = '100%';
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
width: width,
|
||||||
|
height: height
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
toggleLockCompass() {
|
||||||
|
this.$emit('toggle-lock-compass');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
145
src/plugins/imagery/components/Compass/CompassHUD.vue
Normal file
145
src/plugins/imagery/components/Compass/CompassHUD.vue
Normal file
@ -0,0 +1,145 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2021, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div
|
||||||
|
class="c-compass__hud c-hud"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
v-for="point in visibleCompassPoints"
|
||||||
|
:key="point.direction"
|
||||||
|
:class="point.class"
|
||||||
|
:style="point.style"
|
||||||
|
>
|
||||||
|
{{ point.direction }}
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-if="isSunInRange"
|
||||||
|
ref="sun"
|
||||||
|
class="c-hud__sun"
|
||||||
|
:style="sunPositionStyle"
|
||||||
|
></div>
|
||||||
|
<div class="c-hud__range"></div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {
|
||||||
|
rotate,
|
||||||
|
inRange,
|
||||||
|
percentOfRange
|
||||||
|
} from './utils';
|
||||||
|
|
||||||
|
const COMPASS_POINTS = [
|
||||||
|
{
|
||||||
|
direction: 'N',
|
||||||
|
class: 'c-hud__dir',
|
||||||
|
degrees: 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
direction: 'NE',
|
||||||
|
class: 'c-hud__dir--sub',
|
||||||
|
degrees: 45
|
||||||
|
},
|
||||||
|
{
|
||||||
|
direction: 'E',
|
||||||
|
class: 'c-hud__dir',
|
||||||
|
degrees: 90
|
||||||
|
},
|
||||||
|
{
|
||||||
|
direction: 'SE',
|
||||||
|
class: 'c-hud__dir--sub',
|
||||||
|
degrees: 135
|
||||||
|
},
|
||||||
|
{
|
||||||
|
direction: 'S',
|
||||||
|
class: 'c-hud__dir',
|
||||||
|
degrees: 180
|
||||||
|
},
|
||||||
|
{
|
||||||
|
direction: 'SW',
|
||||||
|
class: 'c-hud__dir--sub',
|
||||||
|
degrees: 225
|
||||||
|
},
|
||||||
|
{
|
||||||
|
direction: 'W',
|
||||||
|
class: 'c-hud__dir',
|
||||||
|
degrees: 270
|
||||||
|
},
|
||||||
|
{
|
||||||
|
direction: 'NW',
|
||||||
|
class: 'c-hud__dir--sub',
|
||||||
|
degrees: 315
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
heading: {
|
||||||
|
type: Number,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
sunHeading: {
|
||||||
|
type: Number,
|
||||||
|
default: undefined
|
||||||
|
},
|
||||||
|
cameraAngleOfView: {
|
||||||
|
type: Number,
|
||||||
|
default: undefined
|
||||||
|
},
|
||||||
|
cameraPan: {
|
||||||
|
type: Number,
|
||||||
|
required: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
visibleCompassPoints() {
|
||||||
|
return COMPASS_POINTS
|
||||||
|
.filter(point => inRange(point.degrees, this.visibleRange))
|
||||||
|
.map(point => {
|
||||||
|
const percentage = percentOfRange(point.degrees, this.visibleRange);
|
||||||
|
point.style = Object.assign(
|
||||||
|
{ left: `${ percentage * 100 }%` }
|
||||||
|
);
|
||||||
|
|
||||||
|
return point;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
isSunInRange() {
|
||||||
|
return inRange(this.sunHeading, this.visibleRange);
|
||||||
|
},
|
||||||
|
sunPositionStyle() {
|
||||||
|
const percentage = percentOfRange(this.sunHeading, this.visibleRange);
|
||||||
|
|
||||||
|
return {
|
||||||
|
left: `${ percentage * 100 }%`
|
||||||
|
};
|
||||||
|
},
|
||||||
|
visibleRange() {
|
||||||
|
return [
|
||||||
|
rotate(this.heading, this.cameraPan, -this.cameraAngleOfView / 2),
|
||||||
|
rotate(this.heading, this.cameraPan, this.cameraAngleOfView / 2)
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
267
src/plugins/imagery/components/Compass/CompassRose.vue
Normal file
267
src/plugins/imagery/components/Compass/CompassRose.vue
Normal file
@ -0,0 +1,267 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2021, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div
|
||||||
|
class="c-direction-rose"
|
||||||
|
@click="toggleLockCompass"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="c-nsew"
|
||||||
|
:style="rotateFrameStyle"
|
||||||
|
>
|
||||||
|
<svg
|
||||||
|
class="c-nsew__minor-ticks"
|
||||||
|
viewBox="0 0 100 100"
|
||||||
|
>
|
||||||
|
<rect
|
||||||
|
class="c-nsew__tick c-tick-ne"
|
||||||
|
x="49"
|
||||||
|
y="0"
|
||||||
|
width="2"
|
||||||
|
height="5"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
class="c-nsew__tick c-tick-se"
|
||||||
|
x="95"
|
||||||
|
y="49"
|
||||||
|
width="5"
|
||||||
|
height="2"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
class="c-nsew__tick c-tick-sw"
|
||||||
|
x="49"
|
||||||
|
y="95"
|
||||||
|
width="2"
|
||||||
|
height="5"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
class="c-nsew__tick c-tick-nw"
|
||||||
|
x="0"
|
||||||
|
y="49"
|
||||||
|
width="5"
|
||||||
|
height="2"
|
||||||
|
/>
|
||||||
|
|
||||||
|
</svg>
|
||||||
|
|
||||||
|
<svg
|
||||||
|
class="c-nsew__ticks"
|
||||||
|
viewBox="0 0 100 100"
|
||||||
|
>
|
||||||
|
<polygon
|
||||||
|
class="c-nsew__tick c-tick-n"
|
||||||
|
points="50,0 57,5 43,5"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
class="c-nsew__tick c-tick-e"
|
||||||
|
x="95"
|
||||||
|
y="49"
|
||||||
|
width="5"
|
||||||
|
height="2"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
class="c-nsew__tick c-tick-w"
|
||||||
|
x="0"
|
||||||
|
y="49"
|
||||||
|
width="5"
|
||||||
|
height="2"
|
||||||
|
/>
|
||||||
|
<rect
|
||||||
|
class="c-nsew__tick c-tick-s"
|
||||||
|
x="49"
|
||||||
|
y="95"
|
||||||
|
width="2"
|
||||||
|
height="5"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<text
|
||||||
|
class="c-nsew__label c-label-n"
|
||||||
|
text-anchor="middle"
|
||||||
|
:transform="northTextTransform"
|
||||||
|
>N</text>
|
||||||
|
<text
|
||||||
|
class="c-nsew__label c-label-e"
|
||||||
|
text-anchor="middle"
|
||||||
|
:transform="eastTextTransform"
|
||||||
|
>E</text>
|
||||||
|
<text
|
||||||
|
class="c-nsew__label c-label-w"
|
||||||
|
text-anchor="middle"
|
||||||
|
:transform="southTextTransform"
|
||||||
|
>W</text>
|
||||||
|
<text
|
||||||
|
class="c-nsew__label c-label-s"
|
||||||
|
text-anchor="middle"
|
||||||
|
:transform="westTextTransform"
|
||||||
|
>S</text>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
class="c-spacecraft-body"
|
||||||
|
:style="headingStyle"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-if="showSunHeading"
|
||||||
|
class="c-sun"
|
||||||
|
:style="sunHeadingStyle"
|
||||||
|
></div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
v-if="showCameraFOV"
|
||||||
|
class="c-cam-field"
|
||||||
|
:style="cameraHeadingStyle"
|
||||||
|
>
|
||||||
|
<div class="cam-field-half cam-field-half-l">
|
||||||
|
<div
|
||||||
|
class="cam-field-area"
|
||||||
|
:style="cameraFOVStyleLeftHalf"
|
||||||
|
></div>
|
||||||
|
</div>
|
||||||
|
<div class="cam-field-half cam-field-half-r">
|
||||||
|
<div
|
||||||
|
class="cam-field-area"
|
||||||
|
:style="cameraFOVStyleRightHalf"
|
||||||
|
></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import { rotate } from './utils';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
heading: {
|
||||||
|
type: Number,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
sunHeading: {
|
||||||
|
type: Number,
|
||||||
|
default: undefined
|
||||||
|
},
|
||||||
|
cameraAngleOfView: {
|
||||||
|
type: Number,
|
||||||
|
default: undefined
|
||||||
|
},
|
||||||
|
cameraPan: {
|
||||||
|
type: Number,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
lockCompass: {
|
||||||
|
type: Boolean,
|
||||||
|
required: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
cameraHeading() {
|
||||||
|
return rotate(this.heading, this.cameraPan);
|
||||||
|
},
|
||||||
|
compassHeading() {
|
||||||
|
return this.lockCompass ? this.cameraHeading : 0;
|
||||||
|
},
|
||||||
|
north() {
|
||||||
|
return rotate(this.compassHeading, -this.cameraHeading);
|
||||||
|
},
|
||||||
|
rotateFrameStyle() {
|
||||||
|
return { transform: `rotate(${ this.north }deg)` };
|
||||||
|
},
|
||||||
|
northTextTransform() {
|
||||||
|
return this.cardinalPointsTextTransform.north;
|
||||||
|
},
|
||||||
|
eastTextTransform() {
|
||||||
|
return this.cardinalPointsTextTransform.east;
|
||||||
|
},
|
||||||
|
southTextTransform() {
|
||||||
|
return this.cardinalPointsTextTransform.south;
|
||||||
|
},
|
||||||
|
westTextTransform() {
|
||||||
|
return this.cardinalPointsTextTransform.west;
|
||||||
|
},
|
||||||
|
cardinalPointsTextTransform() {
|
||||||
|
/**
|
||||||
|
* cardinal points text must be rotated
|
||||||
|
* in the opposite direction that north is rotated
|
||||||
|
* to keep text vertically oriented
|
||||||
|
*/
|
||||||
|
const rotation = `rotate(${ -this.north })`;
|
||||||
|
|
||||||
|
return {
|
||||||
|
north: `translate(50,15) ${ rotation }`,
|
||||||
|
east: `translate(87,50) ${ rotation }`,
|
||||||
|
south: `translate(13,50) ${ rotation }`,
|
||||||
|
west: `translate(50,87) ${ rotation }`
|
||||||
|
};
|
||||||
|
},
|
||||||
|
headingStyle() {
|
||||||
|
const rotation = rotate(this.north, this.heading);
|
||||||
|
|
||||||
|
return {
|
||||||
|
transform: `translateX(-50%) rotate(${ rotation }deg)`
|
||||||
|
};
|
||||||
|
},
|
||||||
|
cameraHeadingStyle() {
|
||||||
|
const rotation = rotate(this.north, this.cameraHeading);
|
||||||
|
|
||||||
|
return {
|
||||||
|
transform: `rotate(${ rotation }deg)`
|
||||||
|
};
|
||||||
|
},
|
||||||
|
showSunHeading() {
|
||||||
|
return this.sunHeading !== undefined;
|
||||||
|
},
|
||||||
|
sunHeadingStyle() {
|
||||||
|
const rotation = rotate(this.north, this.sunHeading);
|
||||||
|
|
||||||
|
return {
|
||||||
|
transform: `rotate(${ rotation }deg)`
|
||||||
|
};
|
||||||
|
},
|
||||||
|
showCameraFOV() {
|
||||||
|
return this.cameraPan !== undefined && this.cameraAngleOfView > 0;
|
||||||
|
},
|
||||||
|
// left half of camera field of view
|
||||||
|
// rotated counter-clockwise from camera field of view heading
|
||||||
|
cameraFOVStyleLeftHalf() {
|
||||||
|
return {
|
||||||
|
transform: `translateX(50%) rotate(${ -this.cameraAngleOfView / 2 }deg)`
|
||||||
|
};
|
||||||
|
},
|
||||||
|
// right half of camera field of view
|
||||||
|
// rotated clockwise from camera field of view heading
|
||||||
|
cameraFOVStyleRightHalf() {
|
||||||
|
return {
|
||||||
|
transform: `translateX(-50%) rotate(${ this.cameraAngleOfView / 2 }deg)`
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
toggleLockCompass() {
|
||||||
|
this.$emit('toggle-lock-compass');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
214
src/plugins/imagery/components/Compass/compass.scss
Normal file
214
src/plugins/imagery/components/Compass/compass.scss
Normal file
@ -0,0 +1,214 @@
|
|||||||
|
/***************************** THEME/UI CONSTANTS AND MIXINS */
|
||||||
|
$interfaceKeyColor: #00B9C5;
|
||||||
|
$elemBg: rgba(black, 0.7);
|
||||||
|
|
||||||
|
@mixin sun($position: 'circle closest-side') {
|
||||||
|
$color: #ff9900;
|
||||||
|
$gradEdgePerc: 60%;
|
||||||
|
background: radial-gradient(#{$position}, $color, $color $gradEdgePerc, rgba($color, 0.4) $gradEdgePerc + 5%, transparent);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.c-compass {
|
||||||
|
position: absolute;
|
||||||
|
left: 50%;
|
||||||
|
top: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
z-index: 1;
|
||||||
|
@include userSelectNone;
|
||||||
|
}
|
||||||
|
|
||||||
|
/***************************** COMPASS HUD */
|
||||||
|
.c-hud {
|
||||||
|
// To be placed within a imagery view, in the bounding box of the image
|
||||||
|
$m: 1px;
|
||||||
|
$padTB: 2px;
|
||||||
|
$padLR: $padTB;
|
||||||
|
color: $interfaceKeyColor;
|
||||||
|
font-size: 0.8em;
|
||||||
|
position: absolute;
|
||||||
|
top: $m; right: $m; left: $m;
|
||||||
|
height: 18px;
|
||||||
|
|
||||||
|
svg, div {
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__display {
|
||||||
|
height: 30px;
|
||||||
|
pointer-events: all;
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
left: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__range {
|
||||||
|
border: 1px solid $interfaceKeyColor;
|
||||||
|
border-top-color: transparent;
|
||||||
|
position: absolute;
|
||||||
|
top: 50%; right: $padLR; bottom: $padTB; left: $padLR;
|
||||||
|
}
|
||||||
|
|
||||||
|
[class*="__dir"] {
|
||||||
|
// NSEW
|
||||||
|
display: inline-block;
|
||||||
|
font-weight: bold;
|
||||||
|
text-shadow: 0 1px 2px black;
|
||||||
|
top: 50%;
|
||||||
|
transform: translate(-50%,-50%);
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
[class*="__dir--sub"] {
|
||||||
|
font-weight: normal;
|
||||||
|
opacity: 0.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__sun {
|
||||||
|
$s: 10px;
|
||||||
|
@include sun('circle farthest-side at bottom');
|
||||||
|
bottom: $padTB + 2px;
|
||||||
|
height: $s; width: $s*2;
|
||||||
|
opacity: 0.8;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/***************************** COMPASS DIRECTIONS */
|
||||||
|
.c-nsew {
|
||||||
|
$color: $interfaceKeyColor;
|
||||||
|
$inset: 7%;
|
||||||
|
$tickHeightPerc: 15%;
|
||||||
|
text-shadow: black 0 0 10px;
|
||||||
|
top: $inset; right: $inset; bottom: $inset; left: $inset;
|
||||||
|
z-index: 3;
|
||||||
|
|
||||||
|
&__tick,
|
||||||
|
&__label {
|
||||||
|
fill: $color;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__minor-ticks {
|
||||||
|
opacity: 0.5;
|
||||||
|
transform-origin: center;
|
||||||
|
transform: rotate(45deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
&__label {
|
||||||
|
dominant-baseline: central;
|
||||||
|
font-size: 0.8em;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.c-label-n {
|
||||||
|
font-size: 1.1em;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/***************************** CAMERA FIELD ANGLE */
|
||||||
|
.c-cam-field {
|
||||||
|
$color: white;
|
||||||
|
opacity: 0.2;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
z-index: 2;
|
||||||
|
|
||||||
|
.cam-field-half {
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
|
||||||
|
.cam-field-area {
|
||||||
|
background: $color;
|
||||||
|
top: -30%;
|
||||||
|
right: 0;
|
||||||
|
bottom: -30%;
|
||||||
|
left: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// clip-paths overlap a bit to avoid a gap between halves
|
||||||
|
&-l {
|
||||||
|
clip-path: polygon(0 0, 50.5% 0, 50.5% 100%, 0 100%);
|
||||||
|
.cam-field-area {
|
||||||
|
transform-origin: left center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&-r {
|
||||||
|
clip-path: polygon(49.5% 0, 100% 0, 100% 100%, 49.5% 100%);
|
||||||
|
.cam-field-area {
|
||||||
|
transform-origin: right center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/***************************** SPACECRAFT BODY */
|
||||||
|
.c-spacecraft-body {
|
||||||
|
$color: $interfaceKeyColor;
|
||||||
|
$s: 30%;
|
||||||
|
background: $color;
|
||||||
|
border-radius: 3px;
|
||||||
|
height: $s; width: $s;
|
||||||
|
left: 50%; top: 50%;
|
||||||
|
opacity: 0.4;
|
||||||
|
transform-origin: center top;
|
||||||
|
|
||||||
|
&:before {
|
||||||
|
// Direction arrow
|
||||||
|
$color: rgba(black, 0.5);
|
||||||
|
$arwPointerY: 60%;
|
||||||
|
$arwBodyOffset: 25%;
|
||||||
|
background: $color;
|
||||||
|
content: '';
|
||||||
|
display: block;
|
||||||
|
position: absolute;
|
||||||
|
top: 10%; right: 20%; bottom: 50%; left: 20%;
|
||||||
|
clip-path: polygon(50% 0, 100% $arwPointerY, 100%-$arwBodyOffset $arwPointerY, 100%-$arwBodyOffset 100%, $arwBodyOffset 100%, $arwBodyOffset $arwPointerY, 0 $arwPointerY);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/***************************** DIRECTION ROSE */
|
||||||
|
.c-direction-rose {
|
||||||
|
$d: 100px;
|
||||||
|
$c2: rgba(white, 0.1);
|
||||||
|
background: $elemBg;
|
||||||
|
background-image: radial-gradient(circle closest-side, transparent, transparent 80%, $c2);
|
||||||
|
width: $d;
|
||||||
|
height: $d;
|
||||||
|
transform-origin: 0 0;
|
||||||
|
position: absolute;
|
||||||
|
bottom: 10px; left: 10px;
|
||||||
|
clip-path: circle(50% at 50% 50%);
|
||||||
|
border-radius: 100%;
|
||||||
|
|
||||||
|
svg, div {
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sun
|
||||||
|
.c-sun {
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
|
||||||
|
&:before {
|
||||||
|
$s: 35%;
|
||||||
|
@include sun();
|
||||||
|
content: '';
|
||||||
|
display: block;
|
||||||
|
position: absolute;
|
||||||
|
opacity: 0.7;
|
||||||
|
top: 0; left: 50%;
|
||||||
|
height:$s; width: $s;
|
||||||
|
transform: translate(-50%, -60%);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
84
src/plugins/imagery/components/Compass/pluginSpec.js
Normal file
84
src/plugins/imagery/components/Compass/pluginSpec.js
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2020, 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 Compass from './Compass.vue';
|
||||||
|
import Vue from 'vue';
|
||||||
|
|
||||||
|
const COMPASS_ROSE_CLASS = '.c-direction-rose';
|
||||||
|
const COMPASS_HUD_CLASS = '.c-compass__hud';
|
||||||
|
|
||||||
|
describe("The Compass component", () => {
|
||||||
|
let app;
|
||||||
|
let instance;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
let imageDatum = {
|
||||||
|
heading: 100,
|
||||||
|
roll: 90,
|
||||||
|
pitch: 90,
|
||||||
|
cameraTilt: 100,
|
||||||
|
cameraPan: 90,
|
||||||
|
sunAngle: 30
|
||||||
|
};
|
||||||
|
let propsData = {
|
||||||
|
containerWidth: 600,
|
||||||
|
containerHeight: 600,
|
||||||
|
naturalAspectRatio: 0.9,
|
||||||
|
image: imageDatum
|
||||||
|
};
|
||||||
|
|
||||||
|
app = new Vue({
|
||||||
|
components: { Compass },
|
||||||
|
data() {
|
||||||
|
return propsData;
|
||||||
|
},
|
||||||
|
template: `<Compass
|
||||||
|
:container-width="containerWidth"
|
||||||
|
:container-height="containerHeight"
|
||||||
|
:natural-aspect-ratio="naturalAspectRatio"
|
||||||
|
:image="image" />`
|
||||||
|
});
|
||||||
|
instance = app.$mount();
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
app.$destroy();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when a heading value exists on the image", () => {
|
||||||
|
|
||||||
|
it("should display a compass rose", () => {
|
||||||
|
let compassRoseElement = instance.$el.querySelector(COMPASS_ROSE_CLASS
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(compassRoseElement).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should display a compass HUD", () => {
|
||||||
|
let compassHUDElement = instance.$el.querySelector(COMPASS_HUD_CLASS);
|
||||||
|
|
||||||
|
expect(compassHUDElement).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
33
src/plugins/imagery/components/Compass/utils.js
Normal file
33
src/plugins/imagery/components/Compass/utils.js
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
export function rotate(direction, ...rotations) {
|
||||||
|
const rotation = rotations.reduce((a, b) => a + b, 0);
|
||||||
|
|
||||||
|
return normalizeCompassDirection(direction + rotation);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function normalizeCompassDirection(degrees) {
|
||||||
|
const base = degrees % 360;
|
||||||
|
|
||||||
|
return base >= 0 ? base : 360 + base;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function inRange(degrees, [min, max]) {
|
||||||
|
return min > max
|
||||||
|
? (degrees >= min && degrees < 360) || (degrees <= max && degrees >= 0)
|
||||||
|
: degrees >= min && degrees <= max;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function percentOfRange(degrees, [min, max]) {
|
||||||
|
let distance = degrees;
|
||||||
|
let minRange = min;
|
||||||
|
let maxRange = max;
|
||||||
|
|
||||||
|
if (min > max) {
|
||||||
|
if (distance < max) {
|
||||||
|
distance += 360;
|
||||||
|
}
|
||||||
|
|
||||||
|
maxRange += 360;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (distance - minRange) / (maxRange - minRange);
|
||||||
|
}
|
@ -1,3 +1,25 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2021, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div
|
<div
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
@ -36,14 +58,25 @@
|
|||||||
<div class="c-imagery__main-image__bg"
|
<div class="c-imagery__main-image__bg"
|
||||||
:class="{'paused unnsynced': isPaused,'stale':false }"
|
:class="{'paused unnsynced': isPaused,'stale':false }"
|
||||||
>
|
>
|
||||||
<div class="c-imagery__main-image__image"
|
<img
|
||||||
|
ref="focusedImage"
|
||||||
|
class="c-imagery__main-image__image js-imageryView-image"
|
||||||
|
:src="imageUrl"
|
||||||
:style="{
|
:style="{
|
||||||
'background-image': imageUrl ? `url(${imageUrl})` : 'none',
|
|
||||||
'filter': `brightness(${filters.brightness}%) contrast(${filters.contrast}%)`
|
'filter': `brightness(${filters.brightness}%) contrast(${filters.contrast}%)`
|
||||||
}"
|
}"
|
||||||
:data-openmct-image-timestamp="time"
|
:data-openmct-image-timestamp="time"
|
||||||
:data-openmct-object-keystring="keyString"
|
:data-openmct-object-keystring="keyString"
|
||||||
></div>
|
>
|
||||||
|
<Compass
|
||||||
|
v-if="shouldDisplayCompass"
|
||||||
|
:container-width="imageContainerWidth"
|
||||||
|
:container-height="imageContainerHeight"
|
||||||
|
:natural-aspect-ratio="focusedImageNaturalAspectRatio"
|
||||||
|
:image="focusedImage"
|
||||||
|
:lock-compass="lockCompass"
|
||||||
|
@toggle-lock-compass="toggleLockCompass"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div class="c-local-controls c-local-controls--show-on-hover c-imagery__prev-next-buttons">
|
<div class="c-local-controls c-local-controls--show-on-hover c-imagery__prev-next-buttons">
|
||||||
<button class="c-nav c-nav--prev"
|
<button class="c-nav c-nav--prev"
|
||||||
@ -61,11 +94,25 @@
|
|||||||
<div class="c-imagery__control-bar">
|
<div class="c-imagery__control-bar">
|
||||||
<div class="c-imagery__time">
|
<div class="c-imagery__time">
|
||||||
<div class="c-imagery__timestamp u-style-receiver js-style-receiver">{{ time }}</div>
|
<div class="c-imagery__timestamp u-style-receiver js-style-receiver">{{ time }}</div>
|
||||||
|
|
||||||
|
<!-- image fresh -->
|
||||||
<div
|
<div
|
||||||
v-if="canTrackDuration"
|
v-if="canTrackDuration"
|
||||||
:class="{'c-imagery--new': isImageNew && !refreshCSS}"
|
:class="{'c-imagery--new': isImageNew && !refreshCSS}"
|
||||||
class="c-imagery__age icon-timer"
|
class="c-imagery__age icon-timer"
|
||||||
>{{ formattedDuration }}</div>
|
>{{ formattedDuration }}</div>
|
||||||
|
|
||||||
|
<!-- spacecraft position fresh -->
|
||||||
|
<div
|
||||||
|
v-if="relatedTelemetry.hasRelatedTelemetry && isSpacecraftPositionFresh"
|
||||||
|
class="c-imagery__age icon-check c-imagery--new"
|
||||||
|
>POS</div>
|
||||||
|
|
||||||
|
<!-- camera position fresh -->
|
||||||
|
<div
|
||||||
|
v-if="relatedTelemetry.hasRelatedTelemetry && isCameraPositionFresh"
|
||||||
|
class="c-imagery__age icon-check c-imagery--new"
|
||||||
|
>CAM</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="h-local-controls">
|
<div class="h-local-controls">
|
||||||
<button
|
<button
|
||||||
@ -76,28 +123,32 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div ref="thumbsWrapper"
|
<div
|
||||||
|
ref="thumbsWrapper"
|
||||||
class="c-imagery__thumbs-wrapper"
|
class="c-imagery__thumbs-wrapper"
|
||||||
:class="{'is-paused': isPaused}"
|
:class="{'is-paused': isPaused}"
|
||||||
@scroll="handleScroll"
|
@scroll="handleScroll"
|
||||||
>
|
>
|
||||||
<div v-for="(datum, index) in imageHistory"
|
<div v-for="(image, index) in imageHistory"
|
||||||
:key="datum.url"
|
:key="image.url + image.time"
|
||||||
class="c-imagery__thumb c-thumb"
|
class="c-imagery__thumb c-thumb"
|
||||||
:class="{ selected: focusedImageIndex === index && isPaused }"
|
:class="{ selected: focusedImageIndex === index && isPaused }"
|
||||||
@click="setFocusedImage(index, thumbnailClick)"
|
@click="setFocusedImage(index, thumbnailClick)"
|
||||||
>
|
>
|
||||||
<img class="c-thumb__image"
|
<img class="c-thumb__image"
|
||||||
:src="formatImageUrl(datum)"
|
:src="image.url"
|
||||||
>
|
>
|
||||||
<div class="c-thumb__timestamp">{{ formatTime(datum) }}</div>
|
<div class="c-thumb__timestamp">{{ image.formattedTime }}</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
import _ from 'lodash';
|
||||||
import moment from 'moment';
|
import moment from 'moment';
|
||||||
|
import Compass from './Compass/Compass.vue';
|
||||||
|
import RelatedTelemetry from './RelatedTelemetry/RelatedTelemetry';
|
||||||
|
|
||||||
const DEFAULT_DURATION_FORMATTER = 'duration';
|
const DEFAULT_DURATION_FORMATTER = 'duration';
|
||||||
const REFRESH_CSS_MS = 500;
|
const REFRESH_CSS_MS = 500;
|
||||||
@ -116,6 +167,9 @@ const ARROW_RIGHT = 39;
|
|||||||
const ARROW_LEFT = 37;
|
const ARROW_LEFT = 37;
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
components: {
|
||||||
|
Compass
|
||||||
|
},
|
||||||
inject: ['openmct', 'domainObject'],
|
inject: ['openmct', 'domainObject'],
|
||||||
data() {
|
data() {
|
||||||
let timeSystem = this.openmct.time.timeSystem();
|
let timeSystem = this.openmct.time.timeSystem();
|
||||||
@ -137,7 +191,15 @@ export default {
|
|||||||
refreshCSS: false,
|
refreshCSS: false,
|
||||||
keyString: undefined,
|
keyString: undefined,
|
||||||
focusedImageIndex: undefined,
|
focusedImageIndex: undefined,
|
||||||
numericDuration: undefined
|
focusedImageRelatedTelemetry: {},
|
||||||
|
numericDuration: undefined,
|
||||||
|
metadataEndpoints: {},
|
||||||
|
relatedTelemetry: {},
|
||||||
|
latestRelatedTelemetry: {},
|
||||||
|
focusedImageNaturalAspectRatio: undefined,
|
||||||
|
imageContainerWidth: undefined,
|
||||||
|
imageContainerHeight: undefined,
|
||||||
|
lockCompass: true
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
@ -195,15 +257,83 @@ export default {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
|
},
|
||||||
|
shouldDisplayCompass() {
|
||||||
|
return this.focusedImage !== undefined
|
||||||
|
&& this.focusedImageNaturalAspectRatio !== undefined
|
||||||
|
&& this.imageContainerWidth !== undefined
|
||||||
|
&& this.imageContainerHeight !== undefined;
|
||||||
|
},
|
||||||
|
isSpacecraftPositionFresh() {
|
||||||
|
let isFresh = undefined;
|
||||||
|
let latest = this.latestRelatedTelemetry;
|
||||||
|
let focused = this.focusedImageRelatedTelemetry;
|
||||||
|
|
||||||
|
if (this.relatedTelemetry.hasRelatedTelemetry) {
|
||||||
|
isFresh = true;
|
||||||
|
for (let key of this.spacecraftPositionKeys) {
|
||||||
|
if (this.relatedTelemetry[key] && latest[key] && focused[key]) {
|
||||||
|
isFresh = Boolean(this.relatedTelemetry[key].comparisonFunction(latest[key], focused[key]));
|
||||||
|
} else {
|
||||||
|
isFresh = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return isFresh;
|
||||||
|
},
|
||||||
|
isSpacecraftOrientationFresh() {
|
||||||
|
let isFresh = undefined;
|
||||||
|
let latest = this.latestRelatedTelemetry;
|
||||||
|
let focused = this.focusedImageRelatedTelemetry;
|
||||||
|
|
||||||
|
if (this.relatedTelemetry.hasRelatedTelemetry) {
|
||||||
|
isFresh = true;
|
||||||
|
for (let key of this.spacecraftOrientationKeys) {
|
||||||
|
if (this.relatedTelemetry[key] && latest[key] && focused[key]) {
|
||||||
|
isFresh = Boolean(this.relatedTelemetry[key].comparisonFunction(latest[key], focused[key]));
|
||||||
|
} else {
|
||||||
|
isFresh = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return isFresh;
|
||||||
|
},
|
||||||
|
isCameraPositionFresh() {
|
||||||
|
let isFresh = undefined;
|
||||||
|
let latest = this.latestRelatedTelemetry;
|
||||||
|
let focused = this.focusedImageRelatedTelemetry;
|
||||||
|
|
||||||
|
if (this.relatedTelemetry.hasRelatedTelemetry) {
|
||||||
|
isFresh = true;
|
||||||
|
|
||||||
|
// camera freshness relies on spacecraft position freshness
|
||||||
|
if (this.isSpacecraftPositionFresh && this.isSpacecraftOrientationFresh) {
|
||||||
|
for (let key of this.cameraKeys) {
|
||||||
|
if (this.relatedTelemetry[key] && latest[key] && focused[key]) {
|
||||||
|
isFresh = Boolean(this.relatedTelemetry[key].comparisonFunction(latest[key], focused[key]));
|
||||||
|
} else {
|
||||||
|
isFresh = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
isFresh = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return isFresh;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
focusedImageIndex() {
|
focusedImageIndex() {
|
||||||
this.trackDuration();
|
this.trackDuration();
|
||||||
this.resetAgeCSS();
|
this.resetAgeCSS();
|
||||||
|
this.updateRelatedTelemetryForFocusedImage();
|
||||||
|
this.getImageNaturalDimensions();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {
|
async mounted() {
|
||||||
// listen
|
// listen
|
||||||
this.openmct.time.on('bounds', this.boundsChange);
|
this.openmct.time.on('bounds', this.boundsChange);
|
||||||
this.openmct.time.on('timeSystem', this.timeSystemChange);
|
this.openmct.time.on('timeSystem', this.timeSystemChange);
|
||||||
@ -212,8 +342,15 @@ export default {
|
|||||||
// set
|
// set
|
||||||
this.keyString = this.openmct.objects.makeKeyString(this.domainObject.identifier);
|
this.keyString = this.openmct.objects.makeKeyString(this.domainObject.identifier);
|
||||||
this.metadata = this.openmct.telemetry.getMetadata(this.domainObject);
|
this.metadata = this.openmct.telemetry.getMetadata(this.domainObject);
|
||||||
|
this.imageHints = { ...this.metadata.valuesForHints(['image'])[0] };
|
||||||
this.durationFormatter = this.getFormatter(this.timeSystem.durationFormat || DEFAULT_DURATION_FORMATTER);
|
this.durationFormatter = this.getFormatter(this.timeSystem.durationFormat || DEFAULT_DURATION_FORMATTER);
|
||||||
this.imageFormatter = this.openmct.telemetry.getValueFormatter(this.metadata.valuesForHints(['image'])[0]);
|
this.imageFormatter = this.openmct.telemetry.getValueFormatter(this.imageHints);
|
||||||
|
|
||||||
|
// related telemetry keys
|
||||||
|
this.spacecraftPositionKeys = ['positionX', 'positionY', 'positionZ'];
|
||||||
|
this.spacecraftOrientationKeys = ['heading', 'roll', 'pitch'];
|
||||||
|
this.cameraKeys = ['cameraPan', 'cameraTilt'];
|
||||||
|
this.sunKeys = ['sunOrientation'];
|
||||||
|
|
||||||
// initialize
|
// initialize
|
||||||
this.timeKey = this.timeSystem.key;
|
this.timeKey = this.timeSystem.key;
|
||||||
@ -222,6 +359,18 @@ export default {
|
|||||||
// kickoff
|
// kickoff
|
||||||
this.subscribe();
|
this.subscribe();
|
||||||
this.requestHistory();
|
this.requestHistory();
|
||||||
|
|
||||||
|
// related telemetry
|
||||||
|
await this.initializeRelatedTelemetry();
|
||||||
|
this.updateRelatedTelemetryForFocusedImage();
|
||||||
|
this.trackLatestRelatedTelemetry();
|
||||||
|
|
||||||
|
// for scrolling through images quickly and resizing the object view
|
||||||
|
_.debounce(this.updateRelatedTelemetryForFocusedImage, 400);
|
||||||
|
_.debounce(this.resizeImageContainer, 400);
|
||||||
|
|
||||||
|
this.imageContainerResizeObserver = new ResizeObserver(this.resizeImageContainer);
|
||||||
|
this.imageContainerResizeObserver.observe(this.$refs.focusedImage);
|
||||||
},
|
},
|
||||||
updated() {
|
updated() {
|
||||||
this.scrollToRight();
|
this.scrollToRight();
|
||||||
@ -232,12 +381,115 @@ export default {
|
|||||||
delete this.unsubscribe;
|
delete this.unsubscribe;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.imageContainerResizeObserver.disconnect();
|
||||||
|
|
||||||
|
if (this.relatedTelemetry.hasRelatedTelemetry) {
|
||||||
|
this.relatedTelemetry.destroy();
|
||||||
|
}
|
||||||
|
|
||||||
this.stopDurationTracking();
|
this.stopDurationTracking();
|
||||||
this.openmct.time.off('bounds', this.boundsChange);
|
this.openmct.time.off('bounds', this.boundsChange);
|
||||||
this.openmct.time.off('timeSystem', this.timeSystemChange);
|
this.openmct.time.off('timeSystem', this.timeSystemChange);
|
||||||
this.openmct.time.off('clock', this.clockChange);
|
this.openmct.time.off('clock', this.clockChange);
|
||||||
|
|
||||||
|
// unsubscribe from related telemetry
|
||||||
|
if (this.relatedTelemetry.hasRelatedTelemetry) {
|
||||||
|
for (let key of this.relatedTelemetry.keys) {
|
||||||
|
if (this.relatedTelemetry[key] && this.relatedTelemetry[key].unsubscribe) {
|
||||||
|
this.relatedTelemetry[key].unsubscribe();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
async initializeRelatedTelemetry() {
|
||||||
|
this.relatedTelemetry = new RelatedTelemetry(
|
||||||
|
this.openmct,
|
||||||
|
this.domainObject,
|
||||||
|
[...this.spacecraftPositionKeys, ...this.spacecraftOrientationKeys, ...this.cameraKeys, ...this.sunKeys]
|
||||||
|
);
|
||||||
|
|
||||||
|
if (this.relatedTelemetry.hasRelatedTelemetry) {
|
||||||
|
await this.relatedTelemetry.load();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
async getMostRecentRelatedTelemetry(key, targetDatum) {
|
||||||
|
if (!this.relatedTelemetry.hasRelatedTelemetry) {
|
||||||
|
throw new Error(`${this.domainObject.name} does not have any related telemetry`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.relatedTelemetry[key]) {
|
||||||
|
throw new Error(`${key} does not exist on related telemetry`);
|
||||||
|
}
|
||||||
|
|
||||||
|
let mostRecent;
|
||||||
|
let valueKey = this.relatedTelemetry[key].historical.valueKey;
|
||||||
|
let valuesOnTelemetry = this.relatedTelemetry[key].hasTelemetryOnDatum;
|
||||||
|
|
||||||
|
if (valuesOnTelemetry) {
|
||||||
|
mostRecent = targetDatum[valueKey];
|
||||||
|
|
||||||
|
if (mostRecent) {
|
||||||
|
return mostRecent;
|
||||||
|
} else {
|
||||||
|
console.warn(`Related Telemetry for ${key} does NOT exist on this telemetry datum as configuration implied.`);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mostRecent = await this.relatedTelemetry[key].requestLatestFor(targetDatum);
|
||||||
|
|
||||||
|
return mostRecent[valueKey];
|
||||||
|
},
|
||||||
|
// will subscribe to data for this key if not already done
|
||||||
|
subscribeToDataForKey(key) {
|
||||||
|
if (this.relatedTelemetry[key].isSubscribed) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.relatedTelemetry[key].realtimeDomainObject) {
|
||||||
|
this.relatedTelemetry[key].unsubscribe = this.openmct.telemetry.subscribe(
|
||||||
|
this.relatedTelemetry[key].realtimeDomainObject, datum => {
|
||||||
|
this.relatedTelemetry[key].listeners.forEach(callback => {
|
||||||
|
callback(datum);
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
this.relatedTelemetry[key].isSubscribed = true;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
async updateRelatedTelemetryForFocusedImage() {
|
||||||
|
if (!this.relatedTelemetry.hasRelatedTelemetry || !this.focusedImage) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// set data ON image telemetry as well as in focusedImageRelatedTelemetry
|
||||||
|
for (let key of this.relatedTelemetry.keys) {
|
||||||
|
if (this.relatedTelemetry[key] && this.relatedTelemetry[key].historical) {
|
||||||
|
let valuesOnTelemetry = this.relatedTelemetry[key].hasTelemetryOnDatum;
|
||||||
|
let value = await this.getMostRecentRelatedTelemetry(key, this.focusedImage);
|
||||||
|
|
||||||
|
if (!valuesOnTelemetry) {
|
||||||
|
this.$set(this.imageHistory[this.focusedImageIndex], key, value); // manually add to telemetry
|
||||||
|
}
|
||||||
|
|
||||||
|
this.$set(this.focusedImageRelatedTelemetry, key, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
trackLatestRelatedTelemetry() {
|
||||||
|
[...this.spacecraftPositionKeys, ...this.spacecraftOrientationKeys, ...this.cameraKeys, ...this.sunKeys].forEach(key => {
|
||||||
|
if (this.relatedTelemetry[key] && this.relatedTelemetry[key].subscribe) {
|
||||||
|
this.relatedTelemetry[key].subscribe((datum) => {
|
||||||
|
let valueKey = this.relatedTelemetry[key].realtime.valueKey;
|
||||||
|
this.$set(this.latestRelatedTelemetry, key, datum[valueKey]);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
focusElement() {
|
focusElement() {
|
||||||
this.$el.focus();
|
this.$el.focus();
|
||||||
},
|
},
|
||||||
@ -358,6 +610,7 @@ export default {
|
|||||||
this.requestCount++;
|
this.requestCount++;
|
||||||
const requestId = this.requestCount;
|
const requestId = this.requestCount;
|
||||||
this.imageHistory = [];
|
this.imageHistory = [];
|
||||||
|
|
||||||
let data = await this.openmct.telemetry
|
let data = await this.openmct.telemetry
|
||||||
.request(this.domainObject, bounds) || [];
|
.request(this.domainObject, bounds) || [];
|
||||||
|
|
||||||
@ -393,7 +646,12 @@ export default {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.imageHistory.push(datum);
|
let image = { ...datum };
|
||||||
|
image.formattedTime = this.formatTime(datum);
|
||||||
|
image.url = this.formatImageUrl(datum);
|
||||||
|
image.time = datum[this.timeKey];
|
||||||
|
|
||||||
|
this.imageHistory.push(image);
|
||||||
|
|
||||||
if (setFocused) {
|
if (setFocused) {
|
||||||
this.setFocusedImage(this.imageHistory.length - 1);
|
this.setFocusedImage(this.imageHistory.length - 1);
|
||||||
@ -509,6 +767,28 @@ export default {
|
|||||||
},
|
},
|
||||||
isLeftOrRightArrowKey(keyCode) {
|
isLeftOrRightArrowKey(keyCode) {
|
||||||
return [ARROW_RIGHT, ARROW_LEFT].includes(keyCode);
|
return [ARROW_RIGHT, ARROW_LEFT].includes(keyCode);
|
||||||
|
},
|
||||||
|
getImageNaturalDimensions() {
|
||||||
|
this.focusedImageNaturalAspectRatio = undefined;
|
||||||
|
|
||||||
|
const img = this.$refs.focusedImage;
|
||||||
|
|
||||||
|
// TODO - should probably cache this
|
||||||
|
img.addEventListener('load', () => {
|
||||||
|
this.focusedImageNaturalAspectRatio = img.naturalWidth / img.naturalHeight;
|
||||||
|
}, { once: true });
|
||||||
|
},
|
||||||
|
resizeImageContainer() {
|
||||||
|
if (this.$refs.focusedImage.clientWidth !== this.imageContainerWidth) {
|
||||||
|
this.imageContainerWidth = this.$refs.focusedImage.clientWidth;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.$refs.focusedImage.clientHeight !== this.imageContainerHeight) {
|
||||||
|
this.imageContainerHeight = this.$refs.focusedImage.clientHeight;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
toggleLockCompass() {
|
||||||
|
this.lockCompass = !this.lockCompass;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -0,0 +1,164 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2021, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
function copyRelatedMetadata(metadata) {
|
||||||
|
let compare = metadata.comparisonFunction;
|
||||||
|
let copiedMetadata = JSON.parse(JSON.stringify(metadata));
|
||||||
|
copiedMetadata.comparisonFunction = compare;
|
||||||
|
|
||||||
|
return copiedMetadata;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default class RelatedTelemetry {
|
||||||
|
|
||||||
|
constructor(openmct, domainObject, telemetryKeys) {
|
||||||
|
this._openmct = openmct;
|
||||||
|
this._domainObject = domainObject;
|
||||||
|
|
||||||
|
let metadata = this._openmct.telemetry.getMetadata(this._domainObject);
|
||||||
|
let imageHints = metadata.valuesForHints(['image'])[0];
|
||||||
|
|
||||||
|
this.hasRelatedTelemetry = imageHints.relatedTelemetry !== undefined;
|
||||||
|
|
||||||
|
if (this.hasRelatedTelemetry) {
|
||||||
|
this.keys = telemetryKeys;
|
||||||
|
|
||||||
|
this._timeFormatter = undefined;
|
||||||
|
this._timeSystemChange(this._openmct.time.timeSystem());
|
||||||
|
|
||||||
|
// grab related telemetry metadata
|
||||||
|
for (let key of this.keys) {
|
||||||
|
if (imageHints.relatedTelemetry[key]) {
|
||||||
|
this[key] = copyRelatedMetadata(imageHints.relatedTelemetry[key]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.load = this.load.bind(this);
|
||||||
|
this._parseTime = this._parseTime.bind(this);
|
||||||
|
this._timeSystemChange = this._timeSystemChange.bind(this);
|
||||||
|
this.destroy = this.destroy.bind(this);
|
||||||
|
|
||||||
|
this._openmct.time.on('timeSystem', this._timeSystemChange);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async load() {
|
||||||
|
if (!this.hasRelatedTelemetry) {
|
||||||
|
throw new Error('This domain object does not have related telemetry, use "hasRelatedTelemetry" to check before loading.');
|
||||||
|
}
|
||||||
|
|
||||||
|
await Promise.all(
|
||||||
|
this.keys.map(async (key) => {
|
||||||
|
if (this[key]) {
|
||||||
|
if (this[key].historical) {
|
||||||
|
await this._initializeHistorical(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this[key].realtime && this[key].realtime.telemetryObjectId) {
|
||||||
|
await this._intializeRealtime(key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
async _initializeHistorical(key) {
|
||||||
|
if (this[key].historical.telemetryObjectId) {
|
||||||
|
this[key].historicalDomainObject = await this._openmct.objects.get(this[key].historical.telemetryObjectId);
|
||||||
|
|
||||||
|
this[key].requestLatestFor = async (datum) => {
|
||||||
|
const options = {
|
||||||
|
start: this._openmct.time.bounds().start,
|
||||||
|
end: this._parseTime(datum),
|
||||||
|
strategy: 'latest'
|
||||||
|
};
|
||||||
|
let results = await this._openmct.telemetry
|
||||||
|
.request(this[key].historicalDomainObject, options);
|
||||||
|
|
||||||
|
return results[results.length - 1];
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
this[key].historical.hasTelemetryOnDatum = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async _intializeRealtime(key) {
|
||||||
|
this[key].realtimeDomainObject = await this._openmct.objects.get(this[key].realtime.telemetryObjectId);
|
||||||
|
this[key].listeners = [];
|
||||||
|
this[key].subscribe = (callback) => {
|
||||||
|
|
||||||
|
if (!this[key].isSubscribed) {
|
||||||
|
this._subscribeToDataForKey(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this[key].listeners.includes(callback)) {
|
||||||
|
this[key].listeners.push(callback);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
this[key].listeners.remove(callback);
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
return () => {};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
_subscribeToDataForKey(key) {
|
||||||
|
if (this[key].isSubscribed) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this[key].realtimeDomainObject) {
|
||||||
|
this[key].unsubscribe = this._openmct.telemetry.subscribe(
|
||||||
|
this[key].realtimeDomainObject, datum => {
|
||||||
|
this[key].listeners.forEach(callback => {
|
||||||
|
callback(datum);
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
this[key].isSubscribed = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_parseTime(datum) {
|
||||||
|
return this._timeFormatter.parse(datum);
|
||||||
|
}
|
||||||
|
|
||||||
|
_timeSystemChange(system) {
|
||||||
|
let key = system.key;
|
||||||
|
let metadata = this._openmct.telemetry.getMetadata(this._domainObject);
|
||||||
|
let metadataValue = metadata.value(key) || { format: key };
|
||||||
|
this._timeFormatter = this._openmct.telemetry.getValueFormatter(metadataValue);
|
||||||
|
}
|
||||||
|
|
||||||
|
destroy() {
|
||||||
|
this._openmct.time.off('timeSystem', this._timeSystemChange);
|
||||||
|
for (let key of this.keys) {
|
||||||
|
if (this[key] && this[key].unsubscribe) {
|
||||||
|
this[key].unsubscribe();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -23,6 +23,7 @@
|
|||||||
background-color: $colorPlotBg;
|
background-color: $colorPlotBg;
|
||||||
border: 1px solid transparent;
|
border: 1px solid transparent;
|
||||||
flex: 1 1 auto;
|
flex: 1 1 auto;
|
||||||
|
height: 0;
|
||||||
|
|
||||||
&.unnsynced{
|
&.unnsynced{
|
||||||
@include sUnsynced();
|
@include sUnsynced();
|
||||||
@ -30,10 +31,9 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
&__image {
|
&__image {
|
||||||
@include abs(); // Safari fix
|
height: 100%;
|
||||||
background-position: center;
|
width: 100%;
|
||||||
background-repeat: no-repeat;
|
object-fit: contain;
|
||||||
background-size: contain;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -71,13 +71,14 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
&__age {
|
&__age {
|
||||||
border-radius: $controlCr;
|
border-radius: $smallCr;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
align-items: baseline;
|
align-items: center;
|
||||||
padding: 1px $interiorMarginSm;
|
padding: 2px $interiorMarginSm;
|
||||||
|
|
||||||
&:before {
|
&:before {
|
||||||
|
font-size: 0.9em;
|
||||||
opacity: 0.5;
|
opacity: 0.5;
|
||||||
margin-right: $interiorMarginSm;
|
margin-right: $interiorMarginSm;
|
||||||
}
|
}
|
||||||
@ -86,8 +87,9 @@
|
|||||||
&--new {
|
&--new {
|
||||||
// New imagery
|
// New imagery
|
||||||
$bgColor: $colorOk;
|
$bgColor: $colorOk;
|
||||||
|
color: $colorOkFg;
|
||||||
background: rgba($bgColor, 0.5);
|
background: rgba($bgColor, 0.5);
|
||||||
@include flash($animName: flashImageAge, $dur: 250ms, $valStart: rgba($colorOk, 0.7), $valEnd: rgba($colorOk, 0));
|
@include flash($animName: flashImageAge, $iter: 2, $dur: 250ms, $valStart: rgba($colorOk, 0.7), $valEnd: rgba($colorOk, 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
&__thumbs-wrapper {
|
&__thumbs-wrapper {
|
||||||
|
@ -1,3 +1,25 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2021, 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 ImageryViewProvider from './ImageryViewProvider';
|
import ImageryViewProvider from './ImageryViewProvider';
|
||||||
|
|
||||||
export default function () {
|
export default function () {
|
||||||
|
373
src/plugins/imagery/pluginSpec.js
Normal file
373
src/plugins/imagery/pluginSpec.js
Normal file
@ -0,0 +1,373 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2020, 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 ImageryPlugin from './plugin.js';
|
||||||
|
import Vue from 'vue';
|
||||||
|
import {
|
||||||
|
createOpenMct,
|
||||||
|
resetApplicationState,
|
||||||
|
simulateKeyEvent
|
||||||
|
} from 'utils/testing';
|
||||||
|
|
||||||
|
const ONE_MINUTE = 1000 * 60;
|
||||||
|
const TEN_MINUTES = ONE_MINUTE * 10;
|
||||||
|
const MAIN_IMAGE_CLASS = '.js-imageryView-image';
|
||||||
|
const NEW_IMAGE_CLASS = '.c-imagery__age.c-imagery--new';
|
||||||
|
const REFRESH_CSS_MS = 500;
|
||||||
|
const TOLERANCE = 0.50;
|
||||||
|
|
||||||
|
function comparisonFunction(valueOne, valueTwo) {
|
||||||
|
let larger = valueOne;
|
||||||
|
let smaller = valueTwo;
|
||||||
|
|
||||||
|
if (larger < smaller) {
|
||||||
|
larger = valueTwo;
|
||||||
|
smaller = valueOne;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (larger - smaller) < TOLERANCE;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getImageInfo(doc) {
|
||||||
|
let imageElement = doc.querySelectorAll(MAIN_IMAGE_CLASS)[0];
|
||||||
|
let timestamp = imageElement.dataset.openmctImageTimestamp;
|
||||||
|
let identifier = imageElement.dataset.openmctObjectKeystring;
|
||||||
|
let url = imageElement.src;
|
||||||
|
|
||||||
|
return {
|
||||||
|
timestamp,
|
||||||
|
identifier,
|
||||||
|
url
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function isNew(doc) {
|
||||||
|
let newIcon = doc.querySelectorAll(NEW_IMAGE_CLASS);
|
||||||
|
|
||||||
|
return newIcon.length !== 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
function generateTelemetry(start, count) {
|
||||||
|
let telemetry = [];
|
||||||
|
|
||||||
|
for (let i = 1, l = count + 1; i < l; i++) {
|
||||||
|
let stringRep = i + 'minute';
|
||||||
|
let logo = 'images/logo-openmct.svg';
|
||||||
|
|
||||||
|
telemetry.push({
|
||||||
|
"name": stringRep + " Imagery",
|
||||||
|
"utc": start + (i * ONE_MINUTE),
|
||||||
|
"url": location.host + '/' + logo + '?time=' + stringRep,
|
||||||
|
"timeId": stringRep,
|
||||||
|
"value": 100
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return telemetry;
|
||||||
|
}
|
||||||
|
|
||||||
|
describe("The Imagery View Layout", () => {
|
||||||
|
const imageryKey = 'example.imagery';
|
||||||
|
const START = Date.now();
|
||||||
|
const COUNT = 10;
|
||||||
|
|
||||||
|
let openmct;
|
||||||
|
let imageryPlugin;
|
||||||
|
let parent;
|
||||||
|
let child;
|
||||||
|
let timeFormat = 'utc';
|
||||||
|
let bounds = {
|
||||||
|
start: START - TEN_MINUTES,
|
||||||
|
end: START
|
||||||
|
};
|
||||||
|
let imageTelemetry = generateTelemetry(START - TEN_MINUTES, COUNT);
|
||||||
|
let imageryObject = {
|
||||||
|
identifier: {
|
||||||
|
namespace: "",
|
||||||
|
key: "imageryId"
|
||||||
|
},
|
||||||
|
name: "Example Imagery",
|
||||||
|
type: "example.imagery",
|
||||||
|
location: "parentId",
|
||||||
|
modified: 0,
|
||||||
|
persisted: 0,
|
||||||
|
telemetry: {
|
||||||
|
values: [
|
||||||
|
{
|
||||||
|
"name": "Image",
|
||||||
|
"key": "url",
|
||||||
|
"format": "image",
|
||||||
|
"hints": {
|
||||||
|
"image": 1,
|
||||||
|
"priority": 3
|
||||||
|
},
|
||||||
|
"source": "url",
|
||||||
|
"relatedTelemetry": {
|
||||||
|
"heading": {
|
||||||
|
"comparisonFunction": comparisonFunction,
|
||||||
|
"historical": {
|
||||||
|
"telemetryObjectId": "heading",
|
||||||
|
"valueKey": "value"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"roll": {
|
||||||
|
"comparisonFunction": comparisonFunction,
|
||||||
|
"historical": {
|
||||||
|
"telemetryObjectId": "roll",
|
||||||
|
"valueKey": "value"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"pitch": {
|
||||||
|
"comparisonFunction": comparisonFunction,
|
||||||
|
"historical": {
|
||||||
|
"telemetryObjectId": "pitch",
|
||||||
|
"valueKey": "value"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"cameraPan": {
|
||||||
|
"comparisonFunction": comparisonFunction,
|
||||||
|
"historical": {
|
||||||
|
"telemetryObjectId": "cameraPan",
|
||||||
|
"valueKey": "value"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"cameraTilt": {
|
||||||
|
"comparisonFunction": comparisonFunction,
|
||||||
|
"historical": {
|
||||||
|
"telemetryObjectId": "cameraTilt",
|
||||||
|
"valueKey": "value"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"sunOrientation": {
|
||||||
|
"comparisonFunction": comparisonFunction,
|
||||||
|
"historical": {
|
||||||
|
"telemetryObjectId": "sunOrientation",
|
||||||
|
"valueKey": "value"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Name",
|
||||||
|
"key": "name",
|
||||||
|
"source": "name",
|
||||||
|
"hints": {
|
||||||
|
"priority": 0
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Time",
|
||||||
|
"key": "utc",
|
||||||
|
"format": "utc",
|
||||||
|
"hints": {
|
||||||
|
"domain": 2,
|
||||||
|
"priority": 1
|
||||||
|
},
|
||||||
|
"source": "utc"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Local Time",
|
||||||
|
"key": "local",
|
||||||
|
"format": "local-format",
|
||||||
|
"hints": {
|
||||||
|
"domain": 1,
|
||||||
|
"priority": 2
|
||||||
|
},
|
||||||
|
"source": "local"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// this setups up the app
|
||||||
|
beforeEach((done) => {
|
||||||
|
const appHolder = document.createElement('div');
|
||||||
|
appHolder.style.width = '640px';
|
||||||
|
appHolder.style.height = '480px';
|
||||||
|
|
||||||
|
openmct = createOpenMct();
|
||||||
|
|
||||||
|
parent = document.createElement('div');
|
||||||
|
child = document.createElement('div');
|
||||||
|
parent.appendChild(child);
|
||||||
|
|
||||||
|
spyOn(window, 'ResizeObserver').and.returnValue({
|
||||||
|
observe() {},
|
||||||
|
disconnect() {}
|
||||||
|
});
|
||||||
|
|
||||||
|
spyOn(openmct.telemetry, 'request').and.returnValue(Promise.resolve([]));
|
||||||
|
|
||||||
|
imageryPlugin = new ImageryPlugin();
|
||||||
|
openmct.install(imageryPlugin);
|
||||||
|
|
||||||
|
spyOn(openmct.objects, 'get').and.returnValue(Promise.resolve({}));
|
||||||
|
|
||||||
|
openmct.time.timeSystem(timeFormat, {
|
||||||
|
start: 0,
|
||||||
|
end: 4
|
||||||
|
});
|
||||||
|
|
||||||
|
openmct.on('start', done);
|
||||||
|
openmct.startHeadless(appHolder);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
return resetApplicationState(openmct);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should provide an imagery view only for imagery producing objects", () => {
|
||||||
|
let applicableViews = openmct.objectViews.get(imageryObject, []);
|
||||||
|
let imageryView = applicableViews.find(
|
||||||
|
viewProvider => viewProvider.key === imageryKey
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(imageryView).toBeDefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("imagery view", () => {
|
||||||
|
let applicableViews;
|
||||||
|
let imageryViewProvider;
|
||||||
|
let imageryView;
|
||||||
|
|
||||||
|
beforeEach(async (done) => {
|
||||||
|
let telemetryRequestResolve;
|
||||||
|
let telemetryRequestPromise = new Promise((resolve) => {
|
||||||
|
telemetryRequestResolve = resolve;
|
||||||
|
});
|
||||||
|
|
||||||
|
openmct.telemetry.request.and.callFake(() => {
|
||||||
|
telemetryRequestResolve(imageTelemetry);
|
||||||
|
|
||||||
|
return telemetryRequestPromise;
|
||||||
|
});
|
||||||
|
|
||||||
|
openmct.time.clock('local', {
|
||||||
|
start: bounds.start,
|
||||||
|
end: bounds.end + 100
|
||||||
|
});
|
||||||
|
|
||||||
|
applicableViews = openmct.objectViews.get(imageryObject, []);
|
||||||
|
imageryViewProvider = applicableViews.find(viewProvider => viewProvider.key === imageryKey);
|
||||||
|
imageryView = imageryViewProvider.view(imageryObject);
|
||||||
|
imageryView.show(child);
|
||||||
|
|
||||||
|
await telemetryRequestPromise;
|
||||||
|
await Vue.nextTick();
|
||||||
|
|
||||||
|
return done();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
imageryView.destroy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("on mount should show the the most recent image", () => {
|
||||||
|
const imageInfo = getImageInfo(parent);
|
||||||
|
|
||||||
|
expect(imageInfo.url.indexOf(imageTelemetry[COUNT - 1].timeId)).not.toEqual(-1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should show the clicked thumbnail as the main image", async () => {
|
||||||
|
const target = imageTelemetry[5].url;
|
||||||
|
parent.querySelectorAll(`img[src='${target}']`)[0].click();
|
||||||
|
await Vue.nextTick();
|
||||||
|
const imageInfo = getImageInfo(parent);
|
||||||
|
|
||||||
|
expect(imageInfo.url.indexOf(imageTelemetry[5].timeId)).not.toEqual(-1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should show that an image is new", async (done) => {
|
||||||
|
await Vue.nextTick();
|
||||||
|
|
||||||
|
// used in code, need to wait to the 500ms here too
|
||||||
|
setTimeout(() => {
|
||||||
|
const imageIsNew = isNew(parent);
|
||||||
|
|
||||||
|
expect(imageIsNew).toBeTrue();
|
||||||
|
done();
|
||||||
|
}, REFRESH_CSS_MS);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should show that an image is not new", async (done) => {
|
||||||
|
const target = imageTelemetry[2].url;
|
||||||
|
parent.querySelectorAll(`img[src='${target}']`)[0].click();
|
||||||
|
|
||||||
|
await Vue.nextTick();
|
||||||
|
|
||||||
|
// used in code, need to wait to the 500ms here too
|
||||||
|
setTimeout(() => {
|
||||||
|
const imageIsNew = isNew(parent);
|
||||||
|
|
||||||
|
expect(imageIsNew).toBeFalse();
|
||||||
|
done();
|
||||||
|
}, REFRESH_CSS_MS);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should navigate via arrow keys", async () => {
|
||||||
|
let keyOpts = {
|
||||||
|
element: parent.querySelector('.c-imagery'),
|
||||||
|
key: 'ArrowLeft',
|
||||||
|
keyCode: 37,
|
||||||
|
type: 'keyup'
|
||||||
|
};
|
||||||
|
|
||||||
|
simulateKeyEvent(keyOpts);
|
||||||
|
|
||||||
|
await Vue.nextTick();
|
||||||
|
|
||||||
|
const imageInfo = getImageInfo(parent);
|
||||||
|
|
||||||
|
expect(imageInfo.url.indexOf(imageTelemetry[COUNT - 2].timeId)).not.toEqual(-1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should navigate via numerous arrow keys", async () => {
|
||||||
|
let element = parent.querySelector('.c-imagery');
|
||||||
|
let type = 'keyup';
|
||||||
|
let leftKeyOpts = {
|
||||||
|
element,
|
||||||
|
type,
|
||||||
|
key: 'ArrowLeft',
|
||||||
|
keyCode: 37
|
||||||
|
};
|
||||||
|
let rightKeyOpts = {
|
||||||
|
element,
|
||||||
|
type,
|
||||||
|
key: 'ArrowRight',
|
||||||
|
keyCode: 39
|
||||||
|
};
|
||||||
|
|
||||||
|
// left thrice
|
||||||
|
simulateKeyEvent(leftKeyOpts);
|
||||||
|
simulateKeyEvent(leftKeyOpts);
|
||||||
|
simulateKeyEvent(leftKeyOpts);
|
||||||
|
// right once
|
||||||
|
simulateKeyEvent(rightKeyOpts);
|
||||||
|
|
||||||
|
await Vue.nextTick();
|
||||||
|
|
||||||
|
const imageInfo = getImageInfo(parent);
|
||||||
|
|
||||||
|
expect(imageInfo.url.indexOf(imageTelemetry[COUNT - 3].timeId)).not.toEqual(-1);
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
99
src/plugins/interceptors/pluginSpec.js
Normal file
99
src/plugins/interceptors/pluginSpec.js
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2020, 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 { createOpenMct, resetApplicationState } from "utils/testing";
|
||||||
|
import InterceptorPlugin from "./plugin";
|
||||||
|
|
||||||
|
describe('the plugin', function () {
|
||||||
|
let element;
|
||||||
|
let child;
|
||||||
|
let openmct;
|
||||||
|
const TEST_NAMESPACE = 'test';
|
||||||
|
|
||||||
|
beforeEach((done) => {
|
||||||
|
const appHolder = document.createElement('div');
|
||||||
|
appHolder.style.width = '640px';
|
||||||
|
appHolder.style.height = '480px';
|
||||||
|
|
||||||
|
openmct = createOpenMct();
|
||||||
|
openmct.install(new InterceptorPlugin(openmct));
|
||||||
|
|
||||||
|
element = document.createElement('div');
|
||||||
|
element.style.width = '640px';
|
||||||
|
element.style.height = '480px';
|
||||||
|
child = document.createElement('div');
|
||||||
|
child.style.width = '640px';
|
||||||
|
child.style.height = '480px';
|
||||||
|
element.appendChild(child);
|
||||||
|
|
||||||
|
openmct.on('start', done);
|
||||||
|
openmct.startHeadless(appHolder);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
return resetApplicationState(openmct);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('the missingObjectInterceptor', () => {
|
||||||
|
let mockProvider;
|
||||||
|
beforeEach(() => {
|
||||||
|
mockProvider = jasmine.createSpyObj("mock provider", [
|
||||||
|
"get"
|
||||||
|
]);
|
||||||
|
mockProvider.get.and.returnValue(Promise.resolve(undefined));
|
||||||
|
openmct.objects.addProvider(TEST_NAMESPACE, mockProvider);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns missing objects', (done) => {
|
||||||
|
const identifier = {
|
||||||
|
namespace: TEST_NAMESPACE,
|
||||||
|
key: 'hello'
|
||||||
|
};
|
||||||
|
openmct.objects.get(identifier).then((testObject) => {
|
||||||
|
expect(testObject).toEqual({
|
||||||
|
identifier,
|
||||||
|
type: 'unknown',
|
||||||
|
name: 'Missing: test:hello'
|
||||||
|
});
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns the My items object if not found', (done) => {
|
||||||
|
const identifier = {
|
||||||
|
namespace: TEST_NAMESPACE,
|
||||||
|
key: 'mine'
|
||||||
|
};
|
||||||
|
openmct.objects.get(identifier).then((testObject) => {
|
||||||
|
expect(testObject).toEqual({
|
||||||
|
identifier,
|
||||||
|
"name": "My Items",
|
||||||
|
"type": "folder",
|
||||||
|
"composition": [],
|
||||||
|
"location": "ROOT"
|
||||||
|
});
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
114
src/plugins/localTimeSystem/pluginSpec.js
Normal file
114
src/plugins/localTimeSystem/pluginSpec.js
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2020, 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 {
|
||||||
|
createOpenMct,
|
||||||
|
resetApplicationState
|
||||||
|
} from 'utils/testing';
|
||||||
|
|
||||||
|
describe("The local time", () => {
|
||||||
|
const LOCAL_FORMAT_KEY = 'local-format';
|
||||||
|
const LOCAL_SYSTEM_KEY = 'local';
|
||||||
|
const JUNK = "junk";
|
||||||
|
const TIMESTAMP = -14256000000;
|
||||||
|
const DATESTRING = '1969-07-20 12:00:00.000 am';
|
||||||
|
let openmct;
|
||||||
|
|
||||||
|
beforeEach((done) => {
|
||||||
|
|
||||||
|
openmct = createOpenMct();
|
||||||
|
|
||||||
|
openmct.install(openmct.plugins.LocalTimeSystem());
|
||||||
|
|
||||||
|
openmct.on('start', done);
|
||||||
|
openmct.startHeadless();
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
return resetApplicationState(openmct);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("system", function () {
|
||||||
|
|
||||||
|
let localTimeSystem;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
localTimeSystem = openmct.time.timeSystem(LOCAL_SYSTEM_KEY, {
|
||||||
|
start: 0,
|
||||||
|
end: 4
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("is installed", () => {
|
||||||
|
let timeSystems = openmct.time.getAllTimeSystems();
|
||||||
|
let local = timeSystems.find(ts => ts.key === LOCAL_SYSTEM_KEY);
|
||||||
|
|
||||||
|
expect(local).not.toEqual(-1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("can be set to be the main time system", () => {
|
||||||
|
expect(openmct.time.timeSystem().key).toBe(LOCAL_SYSTEM_KEY);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("uses the local-format time format", () => {
|
||||||
|
expect(localTimeSystem.timeFormat).toBe(LOCAL_FORMAT_KEY);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("is UTC based", () => {
|
||||||
|
expect(localTimeSystem.isUTCBased).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("defines expected metadata", () => {
|
||||||
|
expect(localTimeSystem.key).toBe(LOCAL_SYSTEM_KEY);
|
||||||
|
expect(localTimeSystem.name).toBeDefined();
|
||||||
|
expect(localTimeSystem.cssClass).toBeDefined();
|
||||||
|
expect(localTimeSystem.durationFormat).toBeDefined();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("formatter can be obtained from the telemetry API and", () => {
|
||||||
|
|
||||||
|
let localTimeFormatter;
|
||||||
|
let dateString;
|
||||||
|
let timeStamp;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
localTimeFormatter = openmct.telemetry.getFormatter(LOCAL_FORMAT_KEY);
|
||||||
|
dateString = localTimeFormatter.format(TIMESTAMP);
|
||||||
|
timeStamp = localTimeFormatter.parse(DATESTRING);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("will format a timestamp in local time format", () => {
|
||||||
|
expect(localTimeFormatter.format(TIMESTAMP)).toBe(dateString);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("will parse an local time Date String into milliseconds", () => {
|
||||||
|
expect(localTimeFormatter.parse(DATESTRING)).toBe(timeStamp);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("will validate correctly", () => {
|
||||||
|
expect(localTimeFormatter.validate(DATESTRING)).toBe(true);
|
||||||
|
expect(localTimeFormatter.validate(JUNK)).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
33
src/plugins/nonEditableFolder/plugin.js
Normal file
33
src/plugins/nonEditableFolder/plugin.js
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2020, 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 function () {
|
||||||
|
return function (openmct) {
|
||||||
|
openmct.types.addType("noneditable.folder", {
|
||||||
|
name: "Non-Editable Folder",
|
||||||
|
key: "noneditable.folder",
|
||||||
|
description: "Create folders to organize other objects or links to objects without the ability to edit it's properties.",
|
||||||
|
cssClass: "icon-folder",
|
||||||
|
creatable: false
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}
|
50
src/plugins/nonEditableFolder/pluginSpec.js
Normal file
50
src/plugins/nonEditableFolder/pluginSpec.js
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2020, 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 {
|
||||||
|
createOpenMct,
|
||||||
|
resetApplicationState
|
||||||
|
} from 'utils/testing';
|
||||||
|
|
||||||
|
describe("the plugin", () => {
|
||||||
|
const NON_EDITABLE_FOLDER_KEY = 'noneditable.folder';
|
||||||
|
let openmct;
|
||||||
|
|
||||||
|
beforeEach((done) => {
|
||||||
|
openmct = createOpenMct();
|
||||||
|
openmct.install(openmct.plugins.NonEditableFolder());
|
||||||
|
|
||||||
|
openmct.on('start', done);
|
||||||
|
openmct.startHeadless();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
return resetApplicationState(openmct);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('adds the new non-editable folder type', () => {
|
||||||
|
const type = openmct.types.get(NON_EDITABLE_FOLDER_KEY);
|
||||||
|
|
||||||
|
expect(type).toBeDefined();
|
||||||
|
expect(type.definition.creatable).toBeFalse();
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
@ -97,7 +97,8 @@
|
|||||||
:selected-page="getSelectedPage()"
|
:selected-page="getSelectedPage()"
|
||||||
:selected-section="getSelectedSection()"
|
:selected-section="getSelectedSection()"
|
||||||
:read-only="false"
|
:read-only="false"
|
||||||
@updateEntries="updateEntries"
|
@deleteEntry="deleteEntry"
|
||||||
|
@updateEntry="updateEntry"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -111,19 +112,20 @@ import Search from '@/ui/components/search.vue';
|
|||||||
import SearchResults from './SearchResults.vue';
|
import SearchResults from './SearchResults.vue';
|
||||||
import Sidebar from './Sidebar.vue';
|
import Sidebar from './Sidebar.vue';
|
||||||
import { clearDefaultNotebook, getDefaultNotebook, setDefaultNotebook, setDefaultNotebookSection, setDefaultNotebookPage } from '../utils/notebook-storage';
|
import { clearDefaultNotebook, getDefaultNotebook, setDefaultNotebook, setDefaultNotebookSection, setDefaultNotebookPage } from '../utils/notebook-storage';
|
||||||
import { addNotebookEntry, createNewEmbed, getNotebookEntries, mutateObject } from '../utils/notebook-entries';
|
import { addNotebookEntry, createNewEmbed, getEntryPosById, getNotebookEntries, mutateObject } from '../utils/notebook-entries';
|
||||||
import objectUtils from 'objectUtils';
|
import objectUtils from 'objectUtils';
|
||||||
|
|
||||||
import { throttle } from 'lodash';
|
import { throttle } from 'lodash';
|
||||||
|
import objectLink from '../../../ui/mixins/object-link';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct', 'domainObject', 'snapshotContainer'],
|
|
||||||
components: {
|
components: {
|
||||||
NotebookEntry,
|
NotebookEntry,
|
||||||
Search,
|
Search,
|
||||||
SearchResults,
|
SearchResults,
|
||||||
Sidebar
|
Sidebar
|
||||||
},
|
},
|
||||||
|
inject: ['openmct', 'domainObject', 'snapshotContainer'],
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
defaultPageId: getDefaultNotebook() ? getDefaultNotebook().page.id : '',
|
defaultPageId: getDefaultNotebook() ? getDefaultNotebook().page.id : '',
|
||||||
@ -182,7 +184,9 @@ export default {
|
|||||||
mounted() {
|
mounted() {
|
||||||
this.unlisten = this.openmct.objects.observe(this.internalDomainObject, '*', this.updateInternalDomainObject);
|
this.unlisten = this.openmct.objects.observe(this.internalDomainObject, '*', this.updateInternalDomainObject);
|
||||||
this.formatSidebar();
|
this.formatSidebar();
|
||||||
|
|
||||||
window.addEventListener('orientationchange', this.formatSidebar);
|
window.addEventListener('orientationchange', this.formatSidebar);
|
||||||
|
window.addEventListener("hashchange", this.navigateToSectionPage, false);
|
||||||
|
|
||||||
this.navigateToSectionPage();
|
this.navigateToSectionPage();
|
||||||
},
|
},
|
||||||
@ -190,6 +194,9 @@ export default {
|
|||||||
if (this.unlisten) {
|
if (this.unlisten) {
|
||||||
this.unlisten();
|
this.unlisten();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
window.removeEventListener('orientationchange', this.formatSidebar);
|
||||||
|
window.removeEventListener("hashchange", this.navigateToSectionPage);
|
||||||
},
|
},
|
||||||
updated: function () {
|
updated: function () {
|
||||||
this.$nextTick(() => {
|
this.$nextTick(() => {
|
||||||
@ -226,18 +233,49 @@ export default {
|
|||||||
createNotebookStorageObject() {
|
createNotebookStorageObject() {
|
||||||
const notebookMeta = {
|
const notebookMeta = {
|
||||||
name: this.internalDomainObject.name,
|
name: this.internalDomainObject.name,
|
||||||
identifier: this.internalDomainObject.identifier
|
identifier: this.internalDomainObject.identifier,
|
||||||
|
link: this.getLinktoNotebook()
|
||||||
};
|
};
|
||||||
const page = this.getSelectedPage();
|
const page = this.getSelectedPage();
|
||||||
const section = this.getSelectedSection();
|
const section = this.getSelectedSection();
|
||||||
|
|
||||||
return {
|
return {
|
||||||
domainObject: this.internalDomainObject,
|
|
||||||
notebookMeta,
|
notebookMeta,
|
||||||
section,
|
page,
|
||||||
page
|
section
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
deleteEntry(entryId) {
|
||||||
|
const self = this;
|
||||||
|
const entryPos = getEntryPosById(entryId, this.internalDomainObject, this.selectedSection, this.selectedPage);
|
||||||
|
if (entryPos === -1) {
|
||||||
|
this.openmct.notifications.alert('Warning: unable to delete entry');
|
||||||
|
console.error(`unable to delete entry ${entryId} from section ${this.selectedSection}, page ${this.selectedPage}`);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const dialog = this.openmct.overlays.dialog({
|
||||||
|
iconClass: 'alert',
|
||||||
|
message: 'This action will permanently delete this entry. Do you wish to continue?',
|
||||||
|
buttons: [
|
||||||
|
{
|
||||||
|
label: "Ok",
|
||||||
|
emphasis: true,
|
||||||
|
callback: () => {
|
||||||
|
const entries = getNotebookEntries(self.internalDomainObject, self.selectedSection, self.selectedPage);
|
||||||
|
entries.splice(entryPos, 1);
|
||||||
|
self.updateEntries(entries);
|
||||||
|
dialog.dismiss();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Cancel",
|
||||||
|
callback: () => dialog.dismiss()
|
||||||
|
}
|
||||||
|
]
|
||||||
|
});
|
||||||
|
},
|
||||||
dragOver(event) {
|
dragOver(event) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
event.dataTransfer.dropEffect = "copy";
|
event.dataTransfer.dropEffect = "copy";
|
||||||
@ -311,6 +349,20 @@ export default {
|
|||||||
|
|
||||||
return this.openmct.objects.get(oldNotebookStorage.notebookMeta.identifier);
|
return this.openmct.objects.get(oldNotebookStorage.notebookMeta.identifier);
|
||||||
},
|
},
|
||||||
|
getLinktoNotebook() {
|
||||||
|
const objectPath = this.openmct.router.path;
|
||||||
|
const link = objectLink.computed.objectLink.call({
|
||||||
|
objectPath,
|
||||||
|
openmct: this.openmct
|
||||||
|
});
|
||||||
|
|
||||||
|
const selectedSection = this.selectedSection;
|
||||||
|
const selectedPage = this.selectedPage;
|
||||||
|
const sectionId = selectedSection ? selectedSection.id : '';
|
||||||
|
const pageId = selectedPage ? selectedPage.id : '';
|
||||||
|
|
||||||
|
return `${link}?sectionId=${sectionId}&pageId=${pageId}`;
|
||||||
|
},
|
||||||
getPage(section, id) {
|
getPage(section, id) {
|
||||||
return section.pages.find(p => p.id === id);
|
return section.pages.find(p => p.id === id);
|
||||||
},
|
},
|
||||||
@ -395,6 +447,12 @@ export default {
|
|||||||
return s;
|
return s;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const selectedSectionId = this.selectedSection && this.selectedSection.id;
|
||||||
|
const selectedPageId = this.selectedPage && this.selectedPage.id;
|
||||||
|
if (selectedPageId === pageId && selectedSectionId === sectionId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
this.sectionsChanged({ sections });
|
this.sectionsChanged({ sections });
|
||||||
},
|
},
|
||||||
newEntry(embed = null) {
|
newEntry(embed = null) {
|
||||||
@ -442,10 +500,10 @@ export default {
|
|||||||
async updateDefaultNotebook(notebookStorage) {
|
async updateDefaultNotebook(notebookStorage) {
|
||||||
const defaultNotebookObject = await this.getDefaultNotebookObject();
|
const defaultNotebookObject = await this.getDefaultNotebookObject();
|
||||||
if (!defaultNotebookObject) {
|
if (!defaultNotebookObject) {
|
||||||
setDefaultNotebook(this.openmct, notebookStorage);
|
setDefaultNotebook(this.openmct, notebookStorage, this.internalDomainObject);
|
||||||
} else if (objectUtils.makeKeyString(defaultNotebookObject.identifier) !== objectUtils.makeKeyString(notebookStorage.notebookMeta.identifier)) {
|
} else if (objectUtils.makeKeyString(defaultNotebookObject.identifier) !== objectUtils.makeKeyString(notebookStorage.notebookMeta.identifier)) {
|
||||||
this.removeDefaultClass(defaultNotebookObject);
|
this.removeDefaultClass(defaultNotebookObject);
|
||||||
setDefaultNotebook(this.openmct, notebookStorage);
|
setDefaultNotebook(this.openmct, notebookStorage, this.internalDomainObject);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.defaultSectionId && this.defaultSectionId.length === 0 || this.defaultSectionId !== notebookStorage.section.id) {
|
if (this.defaultSectionId && this.defaultSectionId.length === 0 || this.defaultSectionId !== notebookStorage.section.id) {
|
||||||
@ -514,6 +572,13 @@ export default {
|
|||||||
|
|
||||||
setDefaultNotebookSection(section);
|
setDefaultNotebookSection(section);
|
||||||
},
|
},
|
||||||
|
updateEntry(entry) {
|
||||||
|
const entries = getNotebookEntries(this.internalDomainObject, this.selectedSection, this.selectedPage);
|
||||||
|
const entryPos = getEntryPosById(entry.id, this.internalDomainObject, this.selectedSection, this.selectedPage);
|
||||||
|
entries[entryPos] = entry;
|
||||||
|
|
||||||
|
this.updateEntries(entries);
|
||||||
|
},
|
||||||
updateEntries(entries) {
|
updateEntries(entries) {
|
||||||
const configuration = this.internalDomainObject.configuration;
|
const configuration = this.internalDomainObject.configuration;
|
||||||
const notebookEntries = configuration.entries || {};
|
const notebookEntries = configuration.entries || {};
|
||||||
|
@ -33,10 +33,10 @@ import SnapshotTemplate from './snapshot-template.html';
|
|||||||
import Vue from 'vue';
|
import Vue from 'vue';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct'],
|
|
||||||
components: {
|
components: {
|
||||||
PopupMenu
|
PopupMenu
|
||||||
},
|
},
|
||||||
|
inject: ['openmct'],
|
||||||
props: {
|
props: {
|
||||||
embed: {
|
embed: {
|
||||||
type: Object,
|
type: Object,
|
||||||
@ -143,7 +143,9 @@ export default {
|
|||||||
this.openmct.notifications.alert(message);
|
this.openmct.notifications.alert(message);
|
||||||
}
|
}
|
||||||
|
|
||||||
window.location.hash = hash;
|
const relativeHash = hash.slice(hash.indexOf('#'));
|
||||||
|
const url = new URL(relativeHash, `${location.protocol}//${location.host}${location.pathname}`);
|
||||||
|
window.location.hash = url.hash;
|
||||||
},
|
},
|
||||||
formatTime(unixTime, timeFormat) {
|
formatTime(unixTime, timeFormat) {
|
||||||
return Moment.utc(unixTime).format(timeFormat);
|
return Moment.utc(unixTime).format(timeFormat);
|
||||||
|
@ -12,11 +12,15 @@
|
|||||||
<div class="c-ne__content">
|
<div class="c-ne__content">
|
||||||
<div :id="entry.id"
|
<div :id="entry.id"
|
||||||
class="c-ne__text"
|
class="c-ne__text"
|
||||||
:class="{'c-ne__input' : !readOnly }"
|
tabindex="0"
|
||||||
|
:class="{ 'c-ne__input' : !readOnly }"
|
||||||
:contenteditable="!readOnly"
|
:contenteditable="!readOnly"
|
||||||
@blur="updateEntryValue($event, entry.id)"
|
@blur="updateEntryValue($event)"
|
||||||
@focus="updateCurrentEntryValue($event, entry.id)"
|
@keydown.enter.exact.prevent
|
||||||
>{{ entry.text }}</div>
|
@keyup.enter.exact.prevent="forceBlur($event)"
|
||||||
|
v-text="entry.text"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
<div class="c-snapshots c-ne__embeds">
|
<div class="c-snapshots c-ne__embeds">
|
||||||
<NotebookEmbed v-for="embed in entry.embeds"
|
<NotebookEmbed v-for="embed in entry.embeds"
|
||||||
:key="embed.id"
|
:key="embed.id"
|
||||||
@ -33,6 +37,7 @@
|
|||||||
>
|
>
|
||||||
<button class="c-icon-button c-icon-button--major icon-trash"
|
<button class="c-icon-button c-icon-button--major icon-trash"
|
||||||
title="Delete this entry"
|
title="Delete this entry"
|
||||||
|
tabindex="-1"
|
||||||
@click="deleteEntry"
|
@click="deleteEntry"
|
||||||
>
|
>
|
||||||
</button>
|
</button>
|
||||||
@ -57,14 +62,14 @@
|
|||||||
|
|
||||||
<script>
|
<script>
|
||||||
import NotebookEmbed from './NotebookEmbed.vue';
|
import NotebookEmbed from './NotebookEmbed.vue';
|
||||||
import { createNewEmbed, getEntryPosById, getNotebookEntries } from '../utils/notebook-entries';
|
import { createNewEmbed } from '../utils/notebook-entries';
|
||||||
import Moment from 'moment';
|
import Moment from 'moment';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct', 'snapshotContainer'],
|
|
||||||
components: {
|
components: {
|
||||||
NotebookEmbed
|
NotebookEmbed
|
||||||
},
|
},
|
||||||
|
inject: ['openmct', 'snapshotContainer'],
|
||||||
props: {
|
props: {
|
||||||
domainObject: {
|
domainObject: {
|
||||||
type: Object,
|
type: Object,
|
||||||
@ -103,11 +108,6 @@ export default {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
data() {
|
|
||||||
return {
|
|
||||||
currentEntryValue: ''
|
|
||||||
};
|
|
||||||
},
|
|
||||||
computed: {
|
computed: {
|
||||||
createdOnDate() {
|
createdOnDate() {
|
||||||
return this.formatTime(this.entry.createdOn, 'YYYY-MM-DD');
|
return this.formatTime(this.entry.createdOn, 'YYYY-MM-DD');
|
||||||
@ -117,10 +117,20 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
this.updateEntries = this.updateEntries.bind(this);
|
|
||||||
this.dropOnEntry = this.dropOnEntry.bind(this);
|
this.dropOnEntry = this.dropOnEntry.bind(this);
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
addNewEmbed(objectPath) {
|
||||||
|
const bounds = this.openmct.time.bounds();
|
||||||
|
const snapshotMeta = {
|
||||||
|
bounds,
|
||||||
|
link: null,
|
||||||
|
objectPath,
|
||||||
|
openmct: this.openmct
|
||||||
|
};
|
||||||
|
const newEmbed = createNewEmbed(snapshotMeta);
|
||||||
|
this.entry.embeds.push(newEmbed);
|
||||||
|
},
|
||||||
cancelEditMode(event) {
|
cancelEditMode(event) {
|
||||||
const isEditing = this.openmct.editor.isEditing();
|
const isEditing = this.openmct.editor.isEditing();
|
||||||
if (isEditing) {
|
if (isEditing) {
|
||||||
@ -132,63 +142,23 @@ export default {
|
|||||||
event.dataTransfer.dropEffect = "copy";
|
event.dataTransfer.dropEffect = "copy";
|
||||||
},
|
},
|
||||||
deleteEntry() {
|
deleteEntry() {
|
||||||
const self = this;
|
this.$emit('deleteEntry', this.entry.id);
|
||||||
const entryPosById = self.entryPosById(self.entry.id);
|
|
||||||
if (entryPosById === -1) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const dialog = this.openmct.overlays.dialog({
|
|
||||||
iconClass: 'alert',
|
|
||||||
message: 'This action will permanently delete this entry. Do you wish to continue?',
|
|
||||||
buttons: [
|
|
||||||
{
|
|
||||||
label: "Ok",
|
|
||||||
emphasis: true,
|
|
||||||
callback: () => {
|
|
||||||
const entries = getNotebookEntries(self.domainObject, self.selectedSection, self.selectedPage);
|
|
||||||
entries.splice(entryPosById, 1);
|
|
||||||
self.updateEntries(entries);
|
|
||||||
dialog.dismiss();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: "Cancel",
|
|
||||||
callback: () => {
|
|
||||||
dialog.dismiss();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
dropOnEntry($event) {
|
dropOnEntry($event) {
|
||||||
event.stopImmediatePropagation();
|
event.stopImmediatePropagation();
|
||||||
|
|
||||||
const snapshotId = $event.dataTransfer.getData('openmct/snapshot/id');
|
const snapshotId = $event.dataTransfer.getData('openmct/snapshot/id');
|
||||||
if (snapshotId.length) {
|
if (snapshotId.length) {
|
||||||
this.moveSnapshot(snapshotId);
|
const snapshot = this.snapshotContainer.getSnapshot(snapshotId);
|
||||||
|
this.snapshotContainer.removeSnapshot(snapshotId);
|
||||||
return;
|
this.entry.embeds.push(snapshot);
|
||||||
}
|
} else {
|
||||||
|
|
||||||
const data = $event.dataTransfer.getData('openmct/domain-object-path');
|
const data = $event.dataTransfer.getData('openmct/domain-object-path');
|
||||||
const objectPath = JSON.parse(data);
|
const objectPath = JSON.parse(data);
|
||||||
const entryPos = this.entryPosById(this.entry.id);
|
this.addNewEmbed(objectPath);
|
||||||
const bounds = this.openmct.time.bounds();
|
}
|
||||||
const snapshotMeta = {
|
|
||||||
bounds,
|
this.$emit('updateEntry', this.entry);
|
||||||
link: null,
|
|
||||||
objectPath,
|
|
||||||
openmct: this.openmct
|
|
||||||
};
|
|
||||||
const newEmbed = createNewEmbed(snapshotMeta);
|
|
||||||
const entries = getNotebookEntries(this.domainObject, this.selectedSection, this.selectedPage);
|
|
||||||
const currentEntryEmbeds = entries[entryPos].embeds;
|
|
||||||
currentEntryEmbeds.push(newEmbed);
|
|
||||||
this.updateEntries(entries);
|
|
||||||
},
|
|
||||||
entryPosById(entryId) {
|
|
||||||
return getEntryPosById(entryId, this.domainObject, this.selectedSection, this.selectedPage);
|
|
||||||
},
|
},
|
||||||
findPositionInArray(array, id) {
|
findPositionInArray(array, id) {
|
||||||
let position = -1;
|
let position = -1;
|
||||||
@ -203,15 +173,12 @@ export default {
|
|||||||
|
|
||||||
return position;
|
return position;
|
||||||
},
|
},
|
||||||
|
forceBlur(event) {
|
||||||
|
event.target.blur();
|
||||||
|
},
|
||||||
formatTime(unixTime, timeFormat) {
|
formatTime(unixTime, timeFormat) {
|
||||||
return Moment.utc(unixTime).format(timeFormat);
|
return Moment.utc(unixTime).format(timeFormat);
|
||||||
},
|
},
|
||||||
moveSnapshot(snapshotId) {
|
|
||||||
const snapshot = this.snapshotContainer.getSnapshot(snapshotId);
|
|
||||||
this.entry.embeds.push(snapshot);
|
|
||||||
this.updateEntry(this.entry);
|
|
||||||
this.snapshotContainer.removeSnapshot(snapshotId);
|
|
||||||
},
|
|
||||||
navigateToPage() {
|
navigateToPage() {
|
||||||
this.$emit('changeSectionPage', {
|
this.$emit('changeSectionPage', {
|
||||||
sectionId: this.result.section.id,
|
sectionId: this.result.section.id,
|
||||||
@ -227,15 +194,8 @@ export default {
|
|||||||
removeEmbed(id) {
|
removeEmbed(id) {
|
||||||
const embedPosition = this.findPositionInArray(this.entry.embeds, id);
|
const embedPosition = this.findPositionInArray(this.entry.embeds, id);
|
||||||
this.entry.embeds.splice(embedPosition, 1);
|
this.entry.embeds.splice(embedPosition, 1);
|
||||||
this.updateEntry(this.entry);
|
|
||||||
},
|
|
||||||
updateCurrentEntryValue($event) {
|
|
||||||
if (this.readOnly) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const target = $event.target;
|
this.$emit('updateEntry', this.entry);
|
||||||
this.currentEntryValue = target ? target.textContent : '';
|
|
||||||
},
|
},
|
||||||
updateEmbed(newEmbed) {
|
updateEmbed(newEmbed) {
|
||||||
this.entry.embeds.some(e => {
|
this.entry.embeds.some(e => {
|
||||||
@ -247,44 +207,14 @@ export default {
|
|||||||
return found;
|
return found;
|
||||||
});
|
});
|
||||||
|
|
||||||
this.updateEntry(this.entry);
|
this.$emit('updateEntry', this.entry);
|
||||||
},
|
},
|
||||||
updateEntry(newEntry) {
|
updateEntryValue($event) {
|
||||||
const entries = getNotebookEntries(this.domainObject, this.selectedSection, this.selectedPage);
|
const value = $event.target.innerText;
|
||||||
entries.some(entry => {
|
if (value !== this.entry.text && value.match(/\S/)) {
|
||||||
const found = (entry.id === newEntry.id);
|
this.entry.text = value;
|
||||||
if (found) {
|
this.$emit('updateEntry', this.entry);
|
||||||
entry = newEntry;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return found;
|
|
||||||
});
|
|
||||||
|
|
||||||
this.updateEntries(entries);
|
|
||||||
},
|
|
||||||
updateEntryValue($event, entryId) {
|
|
||||||
if (!this.domainObject || !this.selectedSection || !this.selectedPage) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const target = $event.target;
|
|
||||||
if (!target) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const entryPos = this.entryPosById(entryId);
|
|
||||||
const value = target.textContent.trim();
|
|
||||||
if (this.currentEntryValue !== value) {
|
|
||||||
target.textContent = value;
|
|
||||||
|
|
||||||
const entries = getNotebookEntries(this.domainObject, this.selectedSection, this.selectedPage);
|
|
||||||
entries[entryPos].text = value;
|
|
||||||
|
|
||||||
this.updateEntries(entries);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
updateEntries(entries) {
|
|
||||||
this.$emit('updateEntries', entries);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -44,29 +44,38 @@ export default {
|
|||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
notebookSnapshot: null,
|
notebookSnapshot: undefined,
|
||||||
notebookTypes: []
|
notebookTypes: []
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
validateNotebookStorageObject();
|
validateNotebookStorageObject();
|
||||||
|
this.getDefaultNotebookObject();
|
||||||
|
|
||||||
this.notebookSnapshot = new Snapshot(this.openmct);
|
this.notebookSnapshot = new Snapshot(this.openmct);
|
||||||
this.setDefaultNotebookStatus();
|
this.setDefaultNotebookStatus();
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
showMenu(event) {
|
async getDefaultNotebookObject() {
|
||||||
const notebookTypes = [];
|
|
||||||
const defaultNotebook = getDefaultNotebook();
|
const defaultNotebook = getDefaultNotebook();
|
||||||
|
const defaultNotebookObject = defaultNotebook && await this.openmct.objects.get(defaultNotebook.notebookMeta.identifier);
|
||||||
|
|
||||||
|
return defaultNotebookObject;
|
||||||
|
},
|
||||||
|
async showMenu(event) {
|
||||||
|
const notebookTypes = [];
|
||||||
const elementBoundingClientRect = this.$el.getBoundingClientRect();
|
const elementBoundingClientRect = this.$el.getBoundingClientRect();
|
||||||
const x = elementBoundingClientRect.x;
|
const x = elementBoundingClientRect.x;
|
||||||
const y = elementBoundingClientRect.y + elementBoundingClientRect.height;
|
const y = elementBoundingClientRect.y + elementBoundingClientRect.height;
|
||||||
|
|
||||||
if (defaultNotebook) {
|
const defaultNotebookObject = await this.getDefaultNotebookObject();
|
||||||
const domainObject = defaultNotebook.domainObject;
|
if (defaultNotebookObject) {
|
||||||
|
const name = defaultNotebookObject.name;
|
||||||
|
|
||||||
if (domainObject.location) {
|
const defaultNotebook = getDefaultNotebook();
|
||||||
const defaultPath = `${domainObject.name} - ${defaultNotebook.section.name} - ${defaultNotebook.page.name}`;
|
const sectionName = defaultNotebook.section.name;
|
||||||
|
const pageName = defaultNotebook.page.name;
|
||||||
|
const defaultPath = `${name} - ${sectionName} - ${pageName}`;
|
||||||
|
|
||||||
notebookTypes.push({
|
notebookTypes.push({
|
||||||
cssClass: 'icon-notebook',
|
cssClass: 'icon-notebook',
|
||||||
@ -76,7 +85,6 @@ export default {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
notebookTypes.push({
|
notebookTypes.push({
|
||||||
cssClass: 'icon-camera',
|
cssClass: 'icon-camera',
|
||||||
|
@ -56,11 +56,11 @@ import { NOTEBOOK_SNAPSHOT_MAX_COUNT } from '../snapshot-container';
|
|||||||
import { EVENT_SNAPSHOTS_UPDATED } from '../notebook-constants';
|
import { EVENT_SNAPSHOTS_UPDATED } from '../notebook-constants';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
inject: ['openmct', 'snapshotContainer'],
|
|
||||||
components: {
|
components: {
|
||||||
NotebookEmbed,
|
NotebookEmbed,
|
||||||
PopupMenu
|
PopupMenu
|
||||||
},
|
},
|
||||||
|
inject: ['openmct', 'snapshotContainer'],
|
||||||
props: {
|
props: {
|
||||||
toggleSnapshot: {
|
toggleSnapshot: {
|
||||||
type: Function,
|
type: Function,
|
||||||
|
@ -69,14 +69,14 @@ export default {
|
|||||||
const divElement = document.querySelector('.l-shell__drawer div');
|
const divElement = document.querySelector('.l-shell__drawer div');
|
||||||
|
|
||||||
this.component = new Vue({
|
this.component = new Vue({
|
||||||
provide: {
|
|
||||||
openmct,
|
|
||||||
snapshotContainer
|
|
||||||
},
|
|
||||||
el: divElement,
|
el: divElement,
|
||||||
components: {
|
components: {
|
||||||
SnapshotContainerComponent
|
SnapshotContainerComponent
|
||||||
},
|
},
|
||||||
|
provide: {
|
||||||
|
openmct,
|
||||||
|
snapshotContainer
|
||||||
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
toggleSnapshot
|
toggleSnapshot
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user