Merge pull request #808 from nasa/revert-open-793

Revert "Merge remote-tracking branch 'origin/open793'"
This commit is contained in:
Andrew Henry 2016-03-31 16:37:18 -07:00
commit 2040abb768
6 changed files with 100 additions and 187 deletions

View File

@ -39,7 +39,6 @@ define(
function TableConfiguration(domainObject, telemetryFormatter) { function TableConfiguration(domainObject, telemetryFormatter) {
this.domainObject = domainObject; this.domainObject = domainObject;
this.columns = []; this.columns = [];
this.columnConfiguration = {};
this.telemetryFormatter = telemetryFormatter; this.telemetryFormatter = telemetryFormatter;
} }
@ -145,30 +144,16 @@ define(
{}).columns || {}; {}).columns || {};
}; };
function configEqual(obj1, obj2) {
var obj1Keys = Object.keys(obj1),
obj2Keys = Object.keys(obj2);
return (obj1Keys.length === obj2Keys.length) &&
obj1Keys.every(function (key) {
return obj1[key] === obj2[key];
});
}
/** /**
* Set the established configuration on the domain object. Will noop * Set the established configuration on the domain object
* if configuration is unchanged
* @private * @private
*/ */
TableConfiguration.prototype.saveColumnConfiguration = function (columnConfig) { TableConfiguration.prototype.saveColumnConfiguration = function (columnConfig) {
var self = this; this.domainObject.useCapability('mutation', function (model) {
if (!configEqual(this.columnConfiguration, columnConfig)) { model.configuration = model.configuration || {};
this.domainObject.useCapability('mutation', function (model) { model.configuration.table = model.configuration.table || {};
model.configuration = model.configuration || {}; model.configuration.table.columns = columnConfig;
model.configuration.table = model.configuration.table || {}; });
model.configuration.table.columns = columnConfig;
self.columnConfiguration = columnConfig;
});
}
}; };
/** /**

View File

@ -67,8 +67,8 @@ define(
$scope.$watchCollection('filters', function () { $scope.$watchCollection('filters', function () {
self.updateRows($scope.rows); self.updateRows($scope.rows);
}); });
$scope.$watch('rows', this.updateRows.bind(this));
$scope.$watch('headers', this.updateHeaders.bind(this)); $scope.$watch('headers', this.updateHeaders.bind(this));
$scope.$watch('rows', this.updateRows.bind(this));
/* /*
* Listen for rows added individually (eg. for real-time tables) * Listen for rows added individually (eg. for real-time tables)
@ -101,19 +101,13 @@ define(
*/ */
MCTTableController.prototype.newRow = function (event, rowIndex) { MCTTableController.prototype.newRow = function (event, rowIndex) {
var row = this.$scope.rows[rowIndex]; var row = this.$scope.rows[rowIndex];
//If rows.length === 1 we need to calculate column widths etc. //Add row to the filtered, sorted list of all rows
// so do the updateRows logic, rather than the 'add row' logic if (this.filterRows([row]).length > 0) {
if (this.$scope.rows.length === 1){ this.insertSorted(this.$scope.displayRows, row);
this.updateRows(this.$scope.rows);
} else {
//Add row to the filtered, sorted list of all rows
if (this.filterRows([row]).length > 0) {
this.insertSorted(this.$scope.displayRows, row);
}
this.$timeout(this.setElementSizes.bind(this))
.then(this.scrollToBottom.bind(this));
} }
this.$timeout(this.setElementSizes.bind(this))
.then(this.scrollToBottom.bind(this));
}; };
/** /**

View File

@ -69,36 +69,53 @@ define(
RTTelemetryTableController.prototype = Object.create(TableController.prototype); RTTelemetryTableController.prototype = Object.create(TableController.prototype);
/** /**
* Override the subscribe function defined on the parent controller in
order to handle realtime telemetry instead of historical.
*/ */
RTTelemetryTableController.prototype.addHistoricalData = function () { RTTelemetryTableController.prototype.subscribe = function () {
//Noop for realtime table var self = this;
}; self.$scope.rows = undefined;
(this.subscriptions || []).forEach(function (unsubscribe){
unsubscribe();
});
/**
* Handling for real-time data
*/
RTTelemetryTableController.prototype.updateRealtime = function () {
var datum,
row,
self = this;
if (this.handle) { if (this.handle) {
this.handle.getTelemetryObjects().forEach(function (telemetryObject) { this.handle.unsubscribe();
}
function updateData(){
var datum,
row;
self.handle.getTelemetryObjects().forEach(function (telemetryObject){
datum = self.handle.getDatum(telemetryObject); datum = self.handle.getDatum(telemetryObject);
if (datum) { if (datum) {
row = self.table.getRowValues(telemetryObject, datum); row = self.table.getRowValues(telemetryObject, datum);
self.$scope.rows.push(row); if (!self.$scope.rows){
self.$scope.rows = [row];
self.$scope.$digest();
} else {
self.$scope.rows.push(row);
if (self.$scope.rows.length > self.maxRows) { if (self.$scope.rows.length > self.maxRows) {
self.$scope.$broadcast('remove:row', 0); self.$scope.$broadcast('remove:row', 0);
self.$scope.rows.shift(); self.$scope.rows.shift();
}
self.$scope.$broadcast('add:row',
self.$scope.rows.length - 1);
} }
self.$scope.$broadcast('add:row',
self.$scope.rows.length - 1);
} }
}); });
} }
this.handle = this.$scope.domainObject && this.telemetryHandler.handle(
this.$scope.domainObject,
updateData,
true // Lossless
);
this.setup();
}; };
return RTTelemetryTableController; return RTTelemetryTableController;

View File

@ -58,14 +58,14 @@ define(
telemetryFormatter); telemetryFormatter);
this.changeListeners = []; this.changeListeners = [];
$scope.rows = []; $scope.rows = undefined;
// Subscribe to telemetry when a domain object becomes available // Subscribe to telemetry when a domain object becomes available
this.$scope.$watch('domainObject', function(domainObject){ this.$scope.$watch('domainObject', function(domainObject){
if (!domainObject) if (!domainObject)
return; return;
self.subscribe(domainObject); self.subscribe();
self.registerChangeListeners(); self.registerChangeListeners();
}); });
@ -79,28 +79,14 @@ define(
* @private * @private
*/ */
TelemetryTableController.prototype.registerChangeListeners = function () { TelemetryTableController.prototype.registerChangeListeners = function () {
var self = this;
this.changeListeners.forEach(function (listener) { this.changeListeners.forEach(function (listener) {
return listener && listener(); return listener && listener();
}); });
this.changeListeners = []; this.changeListeners = [];
// When composition changes, re-subscribe to the various
/** // telemetry subscriptions
* Listen to all children for model mutation events that might this.changeListeners.push(this.$scope.$watchCollection(
* indicate that metadata is available, or that composition has 'domainObject.getModel().composition', this.subscribe.bind(this)));
* changed.
*/
if (this.$scope.domainObject.hasCapability('composition')) {
this.$scope.domainObject.useCapability('composition').then(function (composees) {
composees.forEach(function (composee) {
self.changeListeners.push(composee.getCapability('mutation').listen(self.setup.bind(self)));
});
});
}
//Register mutation listener for the parent itself
self.changeListeners.push(self.$scope.domainObject.getCapability('mutation').listen(this.setup.bind(this)));
//Change of bounds in time conductor //Change of bounds in time conductor
this.changeListeners.push(this.$scope.$on('telemetry:display:bounds', this.changeListeners.push(this.$scope.$on('telemetry:display:bounds',
@ -124,38 +110,26 @@ define(
*/ */
TelemetryTableController.prototype.subscribe = function () { TelemetryTableController.prototype.subscribe = function () {
var self = this; var self = this;
this.$scope.rows = [];
if (this.handle) { if (this.handle) {
this.handle.unsubscribe(); this.handle.unsubscribe();
} }
//Noop because not supporting realtime data right now //Noop because not supporting realtime data right now
function update(){ function noop(){
//Is there anything to show?
if (self.table.columns.length > 0){
self.updateRealtime();
}
} }
this.handle = this.$scope.domainObject && this.telemetryHandler.handle( this.handle = this.$scope.domainObject && this.telemetryHandler.handle(
self.$scope.domainObject, this.$scope.domainObject,
update, noop,
true // Lossless true // Lossless
); );
this.handle.request({}).then(this.addHistoricalData.bind(this)); this.handle.request({}).then(this.addHistoricalData.bind(this));
//Call setup at least once
this.setup(); this.setup();
}; };
/**
* Override this method to define handling for realtime data.
*/
TelemetryTableController.prototype.updateRealtime = function () {
//Noop in an historical table
};
/** /**
* Populates historical data on scope when it becomes available * Populates historical data on scope when it becomes available
* @private * @private
@ -183,52 +157,45 @@ define(
*/ */
TelemetryTableController.prototype.setup = function () { TelemetryTableController.prototype.setup = function () {
var handle = this.handle, var handle = this.handle,
domainObject = this.$scope.domainObject,
table = this.table, table = this.table,
self = this, self = this;
metadatas = [];
function addMetadata(object) { if (handle) {
if (object.hasCapability('telemetry') && handle.promiseTelemetryObjects().then(function () {
object.getCapability('telemetry').getMetadata()){ table.buildColumns(handle.getMetadata());
metadatas.push(object.getCapability('telemetry').getMetadata());
}
}
function buildAndFilterColumns(){
if (metadatas && metadatas.length > 0){
self.$scope.rows = [];
table.buildColumns(metadatas);
self.filterColumns(); self.filterColumns();
}
}
//Add telemetry metadata (if any) for parent object // When table column configuration changes, (due to being
addMetadata(domainObject); // selected or deselected), filter columns appropriately.
self.changeListeners.push(self.$scope.$watchCollection(
//If object is composed of multiple objects, also add 'domainObject.getModel().configuration.table.columns',
// telemetry metadata from children self.filterColumns.bind(self)
if (domainObject.hasCapability('composition')) { ));
domainObject.useCapability('composition').then(function (composition) {
composition.forEach(addMetadata);
}).then(function () {
//Build columns based on available metadata
buildAndFilterColumns();
}); });
} else {
//Build columns based on collected metadata
buildAndFilterColumns();
} }
}; };
/**
* @private
* @param object The object for which data is available (table may
* be composed of multiple objects)
* @param datum The data received from the telemetry source
*/
TelemetryTableController.prototype.updateRows = function (object, datum) {
this.$scope.rows.push(this.table.getRowValues(object, datum));
};
/** /**
* When column configuration changes, update the visible headers * When column configuration changes, update the visible headers
* accordingly. * accordingly.
* @private * @private
*/ */
TelemetryTableController.prototype.filterColumns = function () { TelemetryTableController.prototype.filterColumns = function (columnConfig) {
var columnConfig = this.table.getColumnConfiguration(); if (!columnConfig){
this.table.saveColumnConfiguration(columnConfig); columnConfig = this.table.getColumnConfiguration();
this.table.saveColumnConfiguration(columnConfig);
}
//Populate headers with visible columns (determined by configuration) //Populate headers with visible columns (determined by configuration)
this.$scope.headers = Object.keys(columnConfig).filter(function (column) { this.$scope.headers = Object.keys(columnConfig).filter(function (column) {
return columnConfig[column]; return columnConfig[column];

View File

@ -89,24 +89,20 @@ define(
mockDomainObject= jasmine.createSpyObj('domainObject', [ mockDomainObject= jasmine.createSpyObj('domainObject', [
'getCapability', 'getCapability',
'hasCapability',
'useCapability', 'useCapability',
'getModel' 'getModel'
]); ]);
mockDomainObject.getModel.andReturn({}); mockDomainObject.getModel.andReturn({});
mockDomainObject.hasCapability.andReturn(true);
mockDomainObject.getCapability.andReturn( mockDomainObject.getCapability.andReturn(
{ {
getMetadata: function (){ getMetadata: function (){
return {ranges: [{format: 'string'}]}; return {ranges: [{format: 'string'}]};
} }
}); });
mockDomainObject.useCapability.andReturn(promise([]));
mockScope.domainObject = mockDomainObject; mockScope.domainObject = mockDomainObject;
mockTelemetryHandle = jasmine.createSpyObj('telemetryHandle', [ mockTelemetryHandle = jasmine.createSpyObj('telemetryHandle', [
'request',
'getMetadata', 'getMetadata',
'unsubscribe', 'unsubscribe',
'getDatum', 'getDatum',
@ -117,7 +113,6 @@ define(
// used by mocks // used by mocks
mockTelemetryHandle.getTelemetryObjects.andReturn([{}]); mockTelemetryHandle.getTelemetryObjects.andReturn([{}]);
mockTelemetryHandle.promiseTelemetryObjects.andReturn(promise(undefined)); mockTelemetryHandle.promiseTelemetryObjects.andReturn(promise(undefined));
mockTelemetryHandle.request.andReturn(promise(undefined));
mockTelemetryHandle.getDatum.andReturn({}); mockTelemetryHandle.getDatum.andReturn({});
mockTelemetryHandler = jasmine.createSpyObj('telemetryHandler', [ mockTelemetryHandler = jasmine.createSpyObj('telemetryHandler', [
@ -128,8 +123,6 @@ define(
controller = new TableController(mockScope, mockTelemetryHandler, mockTelemetryFormatter); controller = new TableController(mockScope, mockTelemetryHandler, mockTelemetryFormatter);
controller.table = mockTable; controller.table = mockTable;
controller.handle = mockTelemetryHandle; controller.handle = mockTelemetryHandle;
spyOn(controller, 'updateRealtime');
controller.updateRealtime.andCallThrough();
}); });
it('registers for streaming telemetry', function () { it('registers for streaming telemetry', function () {
@ -137,16 +130,14 @@ define(
expect(mockTelemetryHandler.handle).toHaveBeenCalledWith(jasmine.any(Object), jasmine.any(Function), true); expect(mockTelemetryHandler.handle).toHaveBeenCalledWith(jasmine.any(Object), jasmine.any(Function), true);
}); });
describe('when receiving new telemetry', function () { describe('receives new telemetry', function () {
beforeEach(function() { beforeEach(function() {
controller.subscribe(); controller.subscribe();
mockScope.rows = []; mockScope.rows = [];
mockTable.columns = ['a', 'b'];
}); });
it('updates table with new streaming telemetry', function () { it('updates table with new streaming telemetry', function () {
mockTelemetryHandler.handle.mostRecentCall.args[1](); mockTelemetryHandler.handle.mostRecentCall.args[1]();
expect(controller.updateRealtime).toHaveBeenCalled();
expect(mockScope.$broadcast).toHaveBeenCalledWith('add:row', 0); expect(mockScope.$broadcast).toHaveBeenCalledWith('add:row', 0);
}); });
it('observes the row limit', function () { it('observes the row limit', function () {

View File

@ -33,13 +33,7 @@ define(
mockTelemetryHandler, mockTelemetryHandler,
mockTelemetryHandle, mockTelemetryHandle,
mockTelemetryFormatter, mockTelemetryFormatter,
mockTelemetryCapability,
mockDomainObject, mockDomainObject,
mockChild,
mockMutationCapability,
mockCompositionCapability,
childMutationCapability,
capabilities = {},
mockTable, mockTable,
mockConfiguration, mockConfiguration,
watches, watches,
@ -70,17 +64,6 @@ define(
mockScope.$watchCollection.andCallFake(function (expression, callback){ mockScope.$watchCollection.andCallFake(function (expression, callback){
watches[expression] = callback; watches[expression] = callback;
}); });
mockTelemetryCapability = jasmine.createSpyObj('telemetryCapability',
['getMetadata']
);
mockTelemetryCapability.getMetadata.andReturn({});
capabilities.telemetry = mockTelemetryCapability;
mockCompositionCapability = jasmine.createSpyObj('compositionCapability',
['invoke']
);
mockCompositionCapability.invoke.andReturn(promise([]));
capabilities.composition = mockCompositionCapability;
mockConfiguration = { mockConfiguration = {
'range1': true, 'range1': true,
@ -98,36 +81,13 @@ define(
); );
mockTable.columns = []; mockTable.columns = [];
mockTable.getColumnConfiguration.andReturn(mockConfiguration); mockTable.getColumnConfiguration.andReturn(mockConfiguration);
mockMutationCapability = jasmine.createSpyObj('mutationCapability', [
"listen"
]);
capabilities.mutation = mockMutationCapability;
mockDomainObject = jasmine.createSpyObj('domainObject', [ mockDomainObject= jasmine.createSpyObj('domainObject', [
'getCapability', 'getCapability',
'hasCapability',
'useCapability', 'useCapability',
'getModel' 'getModel'
]); ]);
mockChild = jasmine.createSpyObj('domainObject', [
'getCapability'
]);
childMutationCapability = jasmine.createSpyObj('childMutationCapability', [
"listen"
]);
mockChild.getCapability.andReturn(childMutationCapability);
mockDomainObject.getModel.andReturn({}); mockDomainObject.getModel.andReturn({});
mockDomainObject.hasCapability.andCallFake(function (name){
return typeof capabilities[name] !== 'undefined';
});
mockDomainObject.useCapability.andCallFake(function (capability){
return capabilities[capability] && capabilities[capability].invoke && capabilities[capability].invoke();
});
mockDomainObject.getCapability.andCallFake(function (name){
return capabilities[name];
});
mockScope.domainObject = mockDomainObject; mockScope.domainObject = mockDomainObject;
@ -143,7 +103,6 @@ define(
mockTelemetryHandle.promiseTelemetryObjects.andReturn(promise(undefined)); mockTelemetryHandle.promiseTelemetryObjects.andReturn(promise(undefined));
mockTelemetryHandle.request.andReturn(promise(undefined)); mockTelemetryHandle.request.andReturn(promise(undefined));
mockTelemetryHandle.getTelemetryObjects.andReturn([]); mockTelemetryHandle.getTelemetryObjects.andReturn([]);
mockTelemetryHandle.getMetadata.andReturn(["a", "b", "c"]);
mockTelemetryHandler = jasmine.createSpyObj('telemetryHandler', [ mockTelemetryHandler = jasmine.createSpyObj('telemetryHandler', [
'handle' 'handle'
@ -168,6 +127,7 @@ define(
}); });
describe('the controller makes use of the table', function () { describe('the controller makes use of the table', function () {
it('to create column definitions from telemetry' + it('to create column definitions from telemetry' +
' metadata', function () { ' metadata', function () {
controller.setup(); controller.setup();
@ -239,6 +199,14 @@ define(
expect(controller.subscribe).toHaveBeenCalled(); expect(controller.subscribe).toHaveBeenCalled();
}); });
it('triggers telemetry subscription update when domain' +
' object composition changes', function () {
controller.registerChangeListeners();
expect(watches['domainObject.getModel().composition']).toBeDefined();
watches['domainObject.getModel().composition']();
expect(controller.subscribe).toHaveBeenCalled();
});
it('triggers telemetry subscription update when time' + it('triggers telemetry subscription update when time' +
' conductor bounds change', function () { ' conductor bounds change', function () {
controller.registerChangeListeners(); controller.registerChangeListeners();
@ -246,21 +214,12 @@ define(
watches['telemetry:display:bounds'](); watches['telemetry:display:bounds']();
expect(controller.subscribe).toHaveBeenCalled(); expect(controller.subscribe).toHaveBeenCalled();
}); });
it('Listens for changes to object model', function () {
controller.registerChangeListeners();
expect(mockMutationCapability.listen).toHaveBeenCalled();
});
it('Listens for changes to child model', function () { it('triggers refiltering of the columns when configuration' +
mockCompositionCapability.invoke.andReturn(promise([mockChild])); ' changes', function () {
controller.registerChangeListeners(); controller.setup();
expect(childMutationCapability.listen).toHaveBeenCalled(); expect(watches['domainObject.getModel().configuration.table.columns']).toBeDefined();
}); watches['domainObject.getModel().configuration.table.columns']();
it('Recalculates columns when model changes occur', function () {
controller.registerChangeListeners();
expect(mockMutationCapability.listen).toHaveBeenCalled();
mockMutationCapability.listen.mostRecentCall.args[0]();
expect(controller.filterColumns).toHaveBeenCalled(); expect(controller.filterColumns).toHaveBeenCalled();
}); });