mirror of
https://github.com/nasa/openmct.git
synced 2025-01-04 20:34:09 +00:00
433dee0314
* Updated Karma and Jasmine versions * Added DOMObserver class. Supports promise-based testing of DOM changes Update asynchronous test specs to use promises or done() instead of waitsFor/runs * Modified ActionCapability to duplicate context object properties as own properties for better object equality comparisons * Global find + replace to fix syntax issues * Fixed various issues caused by non-deterministic runtime order of tests in Jasmine 3. Fixed issues caused by changes to determination of object equality * Addressed review comments * Resolved merge conflicts with master * Fixed style errors * Use spy.calls.count() instead of manually tracking
332 lines
13 KiB
JavaScript
332 lines
13 KiB
JavaScript
/*****************************************************************************
|
|
* Open MCT, Copyright (c) 2014-2018, 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.
|
|
*****************************************************************************/
|
|
|
|
/**
|
|
* MCTRepresentationSpec. Created by vwoeltje on 11/6/14.
|
|
*/
|
|
define(
|
|
["../src/MCTRepresentation"],
|
|
function (MCTRepresentation) {
|
|
|
|
var JQLITE_FUNCTIONS = ["on", "off", "attr", "removeAttr"],
|
|
LOG_FUNCTIONS = ["error", "warn", "info", "debug"],
|
|
DOMAIN_OBJECT_METHODS = ["getId", "getModel", "getCapability", "hasCapability", "useCapability"];
|
|
|
|
describe("The mct-representation directive", function () {
|
|
var testRepresentations,
|
|
testViews,
|
|
testUrls,
|
|
mockRepresenters,
|
|
mockMutationCapability,
|
|
mockQ,
|
|
mockLinker,
|
|
mockLog,
|
|
mockChangeTemplate,
|
|
mockScope,
|
|
mockElement,
|
|
mockDomainObject,
|
|
testModel,
|
|
mctRepresentation;
|
|
|
|
function mockPromise(value) {
|
|
return (value && value.then) ? value : {
|
|
then: function (callback) {
|
|
return mockPromise(callback(value));
|
|
}
|
|
};
|
|
}
|
|
|
|
function fireWatch(expr, value) {
|
|
mockScope.$watch.calls.all().forEach(function (call) {
|
|
if (call.args[0] === expr) {
|
|
call.args[1](value);
|
|
}
|
|
});
|
|
}
|
|
|
|
beforeEach(function () {
|
|
testUrls = {};
|
|
|
|
testRepresentations = [
|
|
{
|
|
key: "abc",
|
|
bundle: { path: "a", resources: "b" },
|
|
templateUrl: "c/template.html"
|
|
},
|
|
{
|
|
key: "def",
|
|
bundle: { path: "d", resources: "e" },
|
|
templateUrl: "f/template.html",
|
|
uses: ["testCapability", "otherTestCapability"]
|
|
}
|
|
];
|
|
|
|
testViews = [
|
|
{
|
|
key: "uvw",
|
|
bundle: { path: "u", resources: "v" },
|
|
templateUrl: "w/template.html",
|
|
gestures: ["testGesture", "otherTestGesture"]
|
|
},
|
|
{
|
|
key: "xyz",
|
|
bundle: { path: "x", resources: "y" },
|
|
templateUrl: "z/template.html"
|
|
}
|
|
];
|
|
|
|
testModel = { someKey: "some value" };
|
|
|
|
testUrls = {};
|
|
testViews.concat(testRepresentations).forEach(function (t, i) {
|
|
testUrls[t.key] = "some URL " + String(i);
|
|
});
|
|
|
|
mockRepresenters = ["A", "B"].map(function (name) {
|
|
var constructor = jasmine.createSpy("Representer" + name),
|
|
representer = jasmine.createSpyObj(
|
|
"representer" + name,
|
|
["represent", "destroy"]
|
|
);
|
|
constructor.and.returnValue(representer);
|
|
return constructor;
|
|
});
|
|
|
|
mockQ = { when: mockPromise };
|
|
mockLinker = jasmine.createSpyObj(
|
|
'templateLinker',
|
|
['link', 'getPath']
|
|
);
|
|
mockChangeTemplate = jasmine.createSpy('changeTemplate');
|
|
mockLog = jasmine.createSpyObj("$log", LOG_FUNCTIONS);
|
|
|
|
mockMutationCapability =
|
|
jasmine.createSpyObj("mutation", ["listen"]);
|
|
|
|
mockScope = jasmine.createSpyObj("scope", ["$watch", "$on"]);
|
|
mockElement = jasmine.createSpyObj("element", JQLITE_FUNCTIONS);
|
|
mockDomainObject = jasmine.createSpyObj("domainObject", DOMAIN_OBJECT_METHODS);
|
|
|
|
mockDomainObject.getModel.and.returnValue(testModel);
|
|
mockLinker.link.and.returnValue(mockChangeTemplate);
|
|
mockLinker.getPath.and.callFake(function (ext) {
|
|
return testUrls[ext.key];
|
|
});
|
|
|
|
mockDomainObject.getCapability.and.callFake(function (c) {
|
|
return c === 'mutation' && mockMutationCapability;
|
|
});
|
|
|
|
mctRepresentation = new MCTRepresentation(
|
|
testRepresentations,
|
|
testViews,
|
|
mockRepresenters,
|
|
mockQ,
|
|
mockLinker,
|
|
mockLog
|
|
);
|
|
mctRepresentation.link(mockScope, mockElement);
|
|
});
|
|
|
|
it("is restricted to elements", function () {
|
|
expect(mctRepresentation.restrict).toEqual("E");
|
|
});
|
|
|
|
it("exposes templates via the templateLinker", function () {
|
|
expect(mockLinker.link)
|
|
.toHaveBeenCalledWith(mockScope, mockElement);
|
|
});
|
|
|
|
it("watches scope when linked", function () {
|
|
expect(mockScope.$watch).toHaveBeenCalledWith(
|
|
"key",
|
|
jasmine.any(Function)
|
|
);
|
|
expect(mockScope.$watch).toHaveBeenCalledWith(
|
|
"domainObject",
|
|
jasmine.any(Function)
|
|
);
|
|
});
|
|
|
|
it("recognizes keys for representations", function () {
|
|
mockScope.key = "abc";
|
|
mockScope.domainObject = mockDomainObject;
|
|
|
|
// Trigger the watch
|
|
fireWatch('key', mockScope.key);
|
|
fireWatch('domainObject', mockDomainObject);
|
|
|
|
expect(mockChangeTemplate)
|
|
.toHaveBeenCalledWith(testRepresentations[0]);
|
|
});
|
|
|
|
it("recognizes keys for views", function () {
|
|
mockScope.key = "xyz";
|
|
mockScope.domainObject = mockDomainObject;
|
|
|
|
// Trigger the watches
|
|
fireWatch('key', mockScope.key);
|
|
fireWatch('domainObject', mockDomainObject);
|
|
|
|
expect(mockChangeTemplate)
|
|
.toHaveBeenCalledWith(testViews[1]);
|
|
});
|
|
|
|
it("does not load templates until there is an object", function () {
|
|
mockScope.key = "xyz";
|
|
|
|
// Trigger the watch
|
|
fireWatch('key', mockScope.key);
|
|
|
|
expect(mockChangeTemplate)
|
|
.not.toHaveBeenCalledWith(jasmine.any(Object));
|
|
|
|
mockScope.domainObject = mockDomainObject;
|
|
fireWatch('domainObject', mockDomainObject);
|
|
|
|
expect(mockChangeTemplate)
|
|
.toHaveBeenCalledWith(jasmine.any(Object));
|
|
});
|
|
|
|
it("loads declared capabilities", function () {
|
|
mockScope.key = "def";
|
|
mockScope.domainObject = mockDomainObject;
|
|
|
|
// Trigger the watch
|
|
mockScope.$watch.calls.all()[0].args[1]();
|
|
|
|
expect(mockDomainObject.useCapability)
|
|
.toHaveBeenCalledWith("testCapability");
|
|
expect(mockDomainObject.useCapability)
|
|
.toHaveBeenCalledWith("otherTestCapability");
|
|
});
|
|
|
|
it("logs when no representation is available for a key", function () {
|
|
mockScope.key = "someUnknownThing";
|
|
|
|
// Verify precondition
|
|
expect(mockLog.warn).not.toHaveBeenCalled();
|
|
|
|
// Trigger the watch
|
|
mockScope.$watch.calls.all()[0].args[1]();
|
|
|
|
// Should have gotten a warning - that's an unknown key
|
|
expect(mockLog.warn).toHaveBeenCalled();
|
|
});
|
|
|
|
it("clears out obsolete properties from scope", function () {
|
|
mockScope.key = "def";
|
|
mockScope.domainObject = mockDomainObject;
|
|
mockDomainObject.useCapability.and.returnValue("some value");
|
|
|
|
// Trigger the watch
|
|
mockScope.$watch.calls.all()[0].args[1]();
|
|
expect(mockScope.testCapability).toBeDefined();
|
|
|
|
// Change the view
|
|
mockScope.key = "xyz";
|
|
|
|
// Trigger the watch again; should clear capability from scope
|
|
mockScope.$watch.calls.all()[0].args[1]();
|
|
expect(mockScope.testCapability).toBeUndefined();
|
|
});
|
|
|
|
describe("when a domain object has been observed", function () {
|
|
var mockContext,
|
|
mockContext2,
|
|
mockLink,
|
|
mockParent;
|
|
|
|
beforeEach(function () {
|
|
mockContext = jasmine.createSpyObj('context', ['getPath']);
|
|
mockContext2 = jasmine.createSpyObj('context', ['getPath']);
|
|
mockLink = jasmine.createSpyObj(
|
|
'linkedObject',
|
|
DOMAIN_OBJECT_METHODS
|
|
);
|
|
mockParent = jasmine.createSpyObj(
|
|
'parentObject',
|
|
DOMAIN_OBJECT_METHODS
|
|
);
|
|
|
|
mockDomainObject.getCapability.and.callFake(function (c) {
|
|
return {
|
|
context: mockContext,
|
|
mutation: mockMutationCapability
|
|
}[c];
|
|
});
|
|
mockLink.getCapability.and.callFake(function (c) {
|
|
return {
|
|
context: mockContext2,
|
|
mutation: mockMutationCapability
|
|
}[c];
|
|
});
|
|
mockDomainObject.hasCapability.and.callFake(function (c) {
|
|
return c === 'context';
|
|
});
|
|
mockLink.hasCapability.and.callFake(function (c) {
|
|
return c === 'context';
|
|
});
|
|
mockLink.getModel.and.returnValue({});
|
|
|
|
mockContext.getPath.and.returnValue([mockDomainObject]);
|
|
mockContext2.getPath.and.returnValue([mockParent, mockLink]);
|
|
|
|
mockLink.getId.and.returnValue('test-id');
|
|
mockDomainObject.getId.and.returnValue('test-id');
|
|
|
|
mockParent.getId.and.returnValue('parent-id');
|
|
|
|
mockScope.key = "abc";
|
|
mockScope.domainObject = mockDomainObject;
|
|
|
|
mockScope.$watch.calls.all()[0].args[1]();
|
|
});
|
|
|
|
it("listens for mutation of that object", function () {
|
|
expect(mockMutationCapability.listen)
|
|
.toHaveBeenCalledWith(jasmine.any(Function));
|
|
});
|
|
|
|
it("detects subsequent changes among linked instances", function () {
|
|
var callCount = mockChangeTemplate.calls.count();
|
|
|
|
mockScope.domainObject = mockLink;
|
|
mockScope.$watch.calls.all()[0].args[1]();
|
|
|
|
expect(mockChangeTemplate.calls.count())
|
|
.toEqual(callCount + 1);
|
|
});
|
|
|
|
it("does not trigger excess template changes for same instances", function () {
|
|
var callCount = mockChangeTemplate.calls.count();
|
|
mockScope.$watch.calls.all()[0].args[1]();
|
|
expect(mockChangeTemplate.calls.count()).toEqual(callCount);
|
|
});
|
|
|
|
});
|
|
|
|
|
|
});
|
|
}
|
|
);
|