2015-11-05 12:45:51 -08:00
|
|
|
/*****************************************************************************
|
|
|
|
* Open MCT Web, Copyright (c) 2009-2015, United States Government
|
|
|
|
* as represented by the Administrator of the National Aeronautics and Space
|
|
|
|
* Administration. All rights reserved.
|
|
|
|
*
|
|
|
|
* Open MCT Web 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 Web 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.
|
|
|
|
*****************************************************************************/
|
2015-09-14 16:45:38 -07:00
|
|
|
|
|
|
|
define(
|
|
|
|
['../../../src/controllers/drag/TimelineDragHandler'],
|
|
|
|
function (TimelineDragHandler) {
|
|
|
|
|
|
|
|
describe("A Timeline drag handler", function () {
|
|
|
|
var mockLoader,
|
|
|
|
mockSelection,
|
|
|
|
testConfiguration,
|
|
|
|
mockDomainObject,
|
|
|
|
mockDomainObjects,
|
|
|
|
mockTimespans,
|
|
|
|
mockMutations,
|
|
|
|
mockPersists,
|
|
|
|
mockCallback,
|
|
|
|
handler;
|
|
|
|
|
|
|
|
function asPromise(value) {
|
|
|
|
return (value || {}).then ? value : {
|
|
|
|
then: function (callback) {
|
|
|
|
return asPromise(callback(value));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function subgraph(domainObject, objects) {
|
|
|
|
function lookupSubgraph(id) {
|
|
|
|
return subgraph(objects[id], objects);
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
domainObject: domainObject,
|
|
|
|
composition: (domainObject.getModel().composition || [])
|
|
|
|
.map(lookupSubgraph)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function makeMockDomainObject(id, composition) {
|
2016-05-20 11:39:49 -07:00
|
|
|
var mockDomainObj = jasmine.createSpyObj(
|
2015-09-14 16:45:38 -07:00
|
|
|
'domainObject-' + id,
|
|
|
|
['getId', 'getModel', 'getCapability', 'useCapability']
|
|
|
|
);
|
|
|
|
|
2016-05-20 11:39:49 -07:00
|
|
|
mockDomainObj.getId.andReturn(id);
|
|
|
|
mockDomainObj.getModel.andReturn({ composition: composition });
|
|
|
|
mockDomainObj.useCapability.andReturn(asPromise(mockTimespans[id]));
|
|
|
|
mockDomainObj.getCapability.andCallFake(function (c) {
|
2015-09-14 16:45:38 -07:00
|
|
|
return {
|
|
|
|
persistence: mockPersists[id],
|
|
|
|
mutation: mockMutations[id]
|
|
|
|
}[c];
|
|
|
|
});
|
|
|
|
|
2016-05-20 11:39:49 -07:00
|
|
|
return mockDomainObj;
|
2015-09-14 16:45:38 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
beforeEach(function () {
|
|
|
|
mockTimespans = {};
|
|
|
|
mockPersists = {};
|
|
|
|
mockMutations = {};
|
|
|
|
['a', 'b', 'c', 'd', 'e', 'f'].forEach(function (id, index) {
|
|
|
|
mockTimespans[id] = jasmine.createSpyObj(
|
|
|
|
'timespan-' + id,
|
2016-05-19 11:29:13 -07:00
|
|
|
['getStart', 'getEnd', 'getDuration', 'setStart', 'setEnd', 'setDuration']
|
2015-09-14 16:45:38 -07:00
|
|
|
);
|
|
|
|
mockPersists[id] = jasmine.createSpyObj(
|
|
|
|
'persistence-' + id,
|
2016-05-19 11:29:13 -07:00
|
|
|
['persist']
|
2015-09-14 16:45:38 -07:00
|
|
|
);
|
|
|
|
mockMutations[id] = jasmine.createSpyObj(
|
|
|
|
'mutation-' + id,
|
2016-05-19 11:29:13 -07:00
|
|
|
['mutate']
|
2015-09-14 16:45:38 -07:00
|
|
|
);
|
|
|
|
mockTimespans[id].getStart.andReturn(index * 1000);
|
|
|
|
mockTimespans[id].getDuration.andReturn(4000 + index);
|
|
|
|
mockTimespans[id].getEnd.andReturn(4000 + index + index * 1000);
|
|
|
|
});
|
|
|
|
|
|
|
|
mockLoader = jasmine.createSpyObj('objectLoader', ['load']);
|
|
|
|
mockDomainObject = makeMockDomainObject('a', ['b', 'c']);
|
|
|
|
mockDomainObjects = {
|
|
|
|
a: mockDomainObject,
|
|
|
|
b: makeMockDomainObject('b', ['d']),
|
|
|
|
c: makeMockDomainObject('c', ['e', 'f']),
|
|
|
|
d: makeMockDomainObject('d', []),
|
|
|
|
e: makeMockDomainObject('e', []),
|
|
|
|
f: makeMockDomainObject('f', [])
|
|
|
|
};
|
|
|
|
mockSelection = jasmine.createSpyObj('selection', ['get', 'select']);
|
|
|
|
mockCallback = jasmine.createSpy('callback');
|
|
|
|
|
|
|
|
testConfiguration = {};
|
|
|
|
|
|
|
|
mockLoader.load.andReturn(asPromise(
|
|
|
|
subgraph(mockDomainObject, mockDomainObjects)
|
|
|
|
));
|
|
|
|
|
|
|
|
handler = new TimelineDragHandler(
|
|
|
|
mockDomainObject,
|
|
|
|
mockLoader
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("uses the loader to find subgraph", function () {
|
|
|
|
expect(mockLoader.load).toHaveBeenCalledWith(
|
|
|
|
mockDomainObject,
|
|
|
|
'timespan'
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("reports available object identifiers", function () {
|
|
|
|
expect(handler.ids())
|
|
|
|
.toEqual(Object.keys(mockDomainObjects).sort());
|
|
|
|
});
|
|
|
|
|
|
|
|
it("exposes start/end/duration from timespan capabilities", function () {
|
|
|
|
expect(handler.start('a')).toEqual(0);
|
|
|
|
expect(handler.start('b')).toEqual(1000);
|
|
|
|
expect(handler.start('c')).toEqual(2000);
|
|
|
|
expect(handler.duration('a')).toEqual(4000);
|
|
|
|
expect(handler.duration('b')).toEqual(4001);
|
|
|
|
expect(handler.duration('c')).toEqual(4002);
|
|
|
|
expect(handler.end('a')).toEqual(4000);
|
|
|
|
expect(handler.end('b')).toEqual(5001);
|
|
|
|
expect(handler.end('c')).toEqual(6002);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("accepts objects instead of identifiers for start/end/duration calls", function () {
|
|
|
|
Object.keys(mockDomainObjects).forEach(function (id) {
|
|
|
|
expect(handler.start(mockDomainObjects[id])).toEqual(handler.start(id));
|
|
|
|
expect(handler.duration(mockDomainObjects[id])).toEqual(handler.duration(id));
|
|
|
|
expect(handler.end(mockDomainObjects[id])).toEqual(handler.end(id));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("mutates objects", function () {
|
|
|
|
handler.start('a', 123);
|
|
|
|
expect(mockTimespans.a.setStart).toHaveBeenCalledWith(123);
|
|
|
|
handler.duration('b', 42);
|
|
|
|
expect(mockTimespans.b.setDuration).toHaveBeenCalledWith(42);
|
|
|
|
handler.end('c', 12321);
|
|
|
|
expect(mockTimespans.c.setEnd).toHaveBeenCalledWith(12321);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("disallows negative starts, durations", function () {
|
|
|
|
handler.start('a', -100);
|
|
|
|
handler.duration('b', -1000);
|
|
|
|
expect(mockTimespans.a.setStart).toHaveBeenCalledWith(0);
|
|
|
|
expect(mockTimespans.b.setDuration).toHaveBeenCalledWith(0);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("disallows starts greater than ends violations", function () {
|
|
|
|
handler.start('a', 5000);
|
|
|
|
handler.end('b', 500);
|
|
|
|
expect(mockTimespans.a.setStart).toHaveBeenCalledWith(4000); // end time
|
|
|
|
expect(mockTimespans.b.setEnd).toHaveBeenCalledWith(1000); // start time
|
|
|
|
});
|
|
|
|
|
|
|
|
it("moves objects in groups", function () {
|
|
|
|
handler.move('b', 42);
|
|
|
|
expect(mockTimespans.b.setStart).toHaveBeenCalledWith(1042);
|
|
|
|
expect(mockTimespans.b.setEnd).toHaveBeenCalledWith(5043);
|
|
|
|
expect(mockTimespans.d.setStart).toHaveBeenCalledWith(3042);
|
|
|
|
expect(mockTimespans.d.setEnd).toHaveBeenCalledWith(7045);
|
|
|
|
// Verify no other interactions
|
|
|
|
['a', 'c', 'e', 'f'].forEach(function (id) {
|
|
|
|
expect(mockTimespans[id].setStart).not.toHaveBeenCalled();
|
|
|
|
expect(mockTimespans[id].setEnd).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("moves whole subtrees", function () {
|
|
|
|
handler.move('a', 12321);
|
|
|
|
// We verify the math in the previous test, so just verify
|
|
|
|
// that the whole tree is effected here.
|
|
|
|
Object.keys(mockTimespans).forEach(function (id) {
|
|
|
|
expect(mockTimespans[id].setStart).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("prevents bulk moves past 0", function () {
|
|
|
|
// Have a start later; new lowest start is b, at 1000
|
|
|
|
mockTimespans.a.getStart.andReturn(10000);
|
|
|
|
handler.move('a', -10000);
|
|
|
|
// Verify that move was stopped at 0, for b, even though
|
|
|
|
// move was initiated at a
|
|
|
|
expect(mockTimespans.a.setStart).toHaveBeenCalledWith(9000);
|
|
|
|
expect(mockTimespans.b.setStart).toHaveBeenCalledWith(0);
|
|
|
|
expect(mockTimespans.c.setStart).toHaveBeenCalledWith(1000);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("persists mutated objects", function () {
|
|
|
|
handler.start('a', 20);
|
|
|
|
handler.end('b', 50);
|
|
|
|
handler.duration('c', 30);
|
|
|
|
handler.persist();
|
|
|
|
expect(mockPersists.a.persist).toHaveBeenCalled();
|
|
|
|
expect(mockPersists.b.persist).toHaveBeenCalled();
|
|
|
|
expect(mockPersists.c.persist).toHaveBeenCalled();
|
|
|
|
expect(mockPersists.d.persist).not.toHaveBeenCalled();
|
|
|
|
expect(mockPersists.e.persist).not.toHaveBeenCalled();
|
|
|
|
expect(mockPersists.f.persist).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
});
|
|
|
|
}
|
2016-05-19 11:29:13 -07:00
|
|
|
);
|