2016-02-26 19:09:51 +00:00
|
|
|
/*****************************************************************************
|
2016-07-12 23:21:58 +00:00
|
|
|
* Open MCT, Copyright (c) 2014-2016, United States Government
|
2016-02-26 19:09:51 +00:00
|
|
|
* as represented by the Administrator of the National Aeronautics and Space
|
|
|
|
* Administration. All rights reserved.
|
|
|
|
*
|
2016-07-12 23:21:58 +00:00
|
|
|
* Open MCT is licensed under the Apache License, Version 2.0 (the
|
2016-02-26 19:09:51 +00:00
|
|
|
* "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.
|
|
|
|
*
|
2016-07-12 23:21:58 +00:00
|
|
|
* Open MCT includes source code licensed under additional open source
|
2016-02-26 19:09:51 +00:00
|
|
|
* 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(
|
|
|
|
[
|
2016-10-05 01:23:13 +00:00
|
|
|
"zepto",
|
2016-02-26 19:09:51 +00:00
|
|
|
"../../src/controllers/MCTTableController"
|
|
|
|
],
|
2016-10-05 01:23:13 +00:00
|
|
|
function ($, MCTTableController) {
|
|
|
|
|
|
|
|
var MOCK_ELEMENT_TEMPLATE =
|
|
|
|
'<div><div class="l-view-section scrolling">' +
|
|
|
|
'<table class="sizing-table"><tbody></tbody></table>' +
|
|
|
|
'<table class="mct-table"><thead></thead></table>' +
|
|
|
|
'</div></div>';
|
2016-02-26 19:09:51 +00:00
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
describe('The MCTTable Controller', function () {
|
2016-02-26 19:09:51 +00:00
|
|
|
|
|
|
|
var controller,
|
|
|
|
mockScope,
|
|
|
|
watches,
|
|
|
|
mockTimeout,
|
2016-05-25 22:07:17 +00:00
|
|
|
mockElement,
|
|
|
|
mockExportService;
|
2016-02-26 19:09:51 +00:00
|
|
|
|
|
|
|
function promise(value) {
|
|
|
|
return {
|
2016-04-08 23:05:04 +00:00
|
|
|
then: function (callback) {
|
2016-02-26 19:09:51 +00:00
|
|
|
return promise(callback(value));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
beforeEach(function () {
|
2016-02-26 19:09:51 +00:00
|
|
|
watches = {};
|
|
|
|
|
|
|
|
mockScope = jasmine.createSpyObj('scope', [
|
2016-03-09 05:36:33 +00:00
|
|
|
'$watch',
|
|
|
|
'$on',
|
2016-02-26 19:09:51 +00:00
|
|
|
'$watchCollection'
|
|
|
|
]);
|
2016-05-19 18:29:13 +00:00
|
|
|
mockScope.$watchCollection.andCallFake(function (event, callback) {
|
2016-02-26 19:09:51 +00:00
|
|
|
watches[event] = callback;
|
|
|
|
});
|
|
|
|
|
2016-10-05 01:23:13 +00:00
|
|
|
mockElement = $(MOCK_ELEMENT_TEMPLATE);
|
2016-05-25 22:07:17 +00:00
|
|
|
mockExportService = jasmine.createSpyObj('exportService', [
|
|
|
|
'exportCSV'
|
|
|
|
]);
|
|
|
|
|
2016-02-26 19:09:51 +00:00
|
|
|
mockScope.displayHeaders = true;
|
|
|
|
mockTimeout = jasmine.createSpy('$timeout');
|
2016-03-09 05:36:33 +00:00
|
|
|
mockTimeout.andReturn(promise(undefined));
|
2016-02-26 19:09:51 +00:00
|
|
|
|
2016-05-25 22:07:17 +00:00
|
|
|
controller = new MCTTableController(
|
|
|
|
mockScope,
|
|
|
|
mockTimeout,
|
|
|
|
mockElement,
|
|
|
|
mockExportService
|
|
|
|
);
|
2016-05-10 21:57:09 +00:00
|
|
|
spyOn(controller, 'setVisibleRows').andCallThrough();
|
2016-02-26 19:09:51 +00:00
|
|
|
});
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
it('Reacts to changes to filters, headers, and rows', function () {
|
2016-02-26 19:09:51 +00:00
|
|
|
expect(mockScope.$watchCollection).toHaveBeenCalledWith('filters', jasmine.any(Function));
|
2016-03-09 05:36:33 +00:00
|
|
|
expect(mockScope.$watch).toHaveBeenCalledWith('headers', jasmine.any(Function));
|
|
|
|
expect(mockScope.$watch).toHaveBeenCalledWith('rows', jasmine.any(Function));
|
2016-02-26 19:09:51 +00:00
|
|
|
});
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
describe('rows', function () {
|
2016-02-26 19:09:51 +00:00
|
|
|
var testRows = [];
|
2016-05-19 18:29:13 +00:00
|
|
|
beforeEach(function () {
|
2016-02-26 19:09:51 +00:00
|
|
|
testRows = [
|
|
|
|
{
|
|
|
|
'col1': {'text': 'row1 col1 match'},
|
|
|
|
'col2': {'text': 'def'},
|
|
|
|
'col3': {'text': 'row1 col3'}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
'col1': {'text': 'row2 col1 match'},
|
|
|
|
'col2': {'text': 'abc'},
|
|
|
|
'col3': {'text': 'row2 col3'}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
'col1': {'text': 'row3 col1'},
|
|
|
|
'col2': {'text': 'ghi'},
|
|
|
|
'col3': {'text': 'row3 col3'}
|
|
|
|
}
|
|
|
|
];
|
2016-03-14 02:37:08 +00:00
|
|
|
mockScope.rows = testRows;
|
2016-02-26 19:09:51 +00:00
|
|
|
});
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
it('Filters results based on filter input', function () {
|
|
|
|
var filters = {},
|
|
|
|
filteredRows;
|
2016-02-26 19:09:51 +00:00
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
mockScope.filters = filters;
|
2016-02-26 19:09:51 +00:00
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
filteredRows = controller.filterRows(testRows);
|
|
|
|
expect(filteredRows.length).toBe(3);
|
|
|
|
filters.col1 = 'row1';
|
|
|
|
filteredRows = controller.filterRows(testRows);
|
|
|
|
expect(filteredRows.length).toBe(1);
|
|
|
|
filters.col1 = 'match';
|
|
|
|
filteredRows = controller.filterRows(testRows);
|
|
|
|
expect(filteredRows.length).toBe(2);
|
2016-02-26 19:09:51 +00:00
|
|
|
});
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
it('Sets rows on scope when rows change', function () {
|
2016-04-22 16:44:34 +00:00
|
|
|
controller.setRows(testRows);
|
2016-02-26 19:09:51 +00:00
|
|
|
expect(mockScope.displayRows.length).toBe(3);
|
|
|
|
expect(mockScope.displayRows).toEqual(testRows);
|
|
|
|
});
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
it('Supports adding rows individually', function () {
|
|
|
|
var addRowFunc = mockScope.$on.calls[mockScope.$on.calls.length - 2].args[1],
|
2016-03-09 05:36:33 +00:00
|
|
|
row4 = {
|
|
|
|
'col1': {'text': 'row3 col1'},
|
|
|
|
'col2': {'text': 'ghi'},
|
|
|
|
'col3': {'text': 'row3 col3'}
|
|
|
|
};
|
2016-04-22 16:44:34 +00:00
|
|
|
controller.setRows(testRows);
|
2016-03-09 05:36:33 +00:00
|
|
|
expect(mockScope.displayRows.length).toBe(3);
|
2016-03-14 02:37:08 +00:00
|
|
|
testRows.push(row4);
|
2016-03-17 01:26:14 +00:00
|
|
|
addRowFunc(undefined, 3);
|
2016-03-09 05:36:33 +00:00
|
|
|
expect(mockScope.displayRows.length).toBe(4);
|
|
|
|
});
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
it('Supports removing rows individually', function () {
|
|
|
|
var removeRowFunc = mockScope.$on.calls[mockScope.$on.calls.length - 1].args[1];
|
2016-04-22 16:44:34 +00:00
|
|
|
controller.setRows(testRows);
|
2016-03-17 01:26:14 +00:00
|
|
|
expect(mockScope.displayRows.length).toBe(3);
|
|
|
|
removeRowFunc(undefined, 2);
|
|
|
|
expect(mockScope.displayRows.length).toBe(2);
|
|
|
|
expect(controller.setVisibleRows).toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
2016-05-25 22:07:17 +00:00
|
|
|
it("can be exported as CSV", function () {
|
|
|
|
controller.setRows(testRows);
|
|
|
|
controller.setHeaders(Object.keys(testRows[0]));
|
|
|
|
mockScope.exportAsCSV();
|
|
|
|
expect(mockExportService.exportCSV)
|
|
|
|
.toHaveBeenCalled();
|
|
|
|
mockExportService.exportCSV.mostRecentCall.args[0]
|
|
|
|
.forEach(function (row, i) {
|
|
|
|
Object.keys(row).forEach(function (k) {
|
|
|
|
expect(row[k]).toEqual(
|
|
|
|
mockScope.displayRows[i][k].text
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
describe('sorting', function () {
|
2016-02-26 19:09:51 +00:00
|
|
|
var sortedRows;
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
it('Sorts rows ascending', function () {
|
2016-02-26 19:09:51 +00:00
|
|
|
mockScope.sortColumn = 'col1';
|
|
|
|
mockScope.sortDirection = 'asc';
|
|
|
|
|
|
|
|
sortedRows = controller.sortRows(testRows);
|
|
|
|
expect(sortedRows[0].col1.text).toEqual('row1 col1 match');
|
|
|
|
expect(sortedRows[1].col1.text).toEqual('row2 col1' +
|
|
|
|
' match');
|
|
|
|
expect(sortedRows[2].col1.text).toEqual('row3 col1');
|
|
|
|
|
|
|
|
});
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
it('Sorts rows descending', function () {
|
2016-02-26 19:09:51 +00:00
|
|
|
mockScope.sortColumn = 'col1';
|
|
|
|
mockScope.sortDirection = 'desc';
|
|
|
|
|
|
|
|
sortedRows = controller.sortRows(testRows);
|
|
|
|
expect(sortedRows[0].col1.text).toEqual('row3 col1');
|
|
|
|
expect(sortedRows[1].col1.text).toEqual('row2 col1 match');
|
|
|
|
expect(sortedRows[2].col1.text).toEqual('row1 col1 match');
|
|
|
|
});
|
2016-05-19 18:29:13 +00:00
|
|
|
it('Sorts rows descending based on selected sort column', function () {
|
2016-02-26 19:09:51 +00:00
|
|
|
mockScope.sortColumn = 'col2';
|
|
|
|
mockScope.sortDirection = 'desc';
|
|
|
|
|
|
|
|
sortedRows = controller.sortRows(testRows);
|
|
|
|
expect(sortedRows[0].col2.text).toEqual('ghi');
|
|
|
|
expect(sortedRows[1].col2.text).toEqual('def');
|
|
|
|
expect(sortedRows[2].col2.text).toEqual('abc');
|
|
|
|
});
|
2016-03-09 05:36:33 +00:00
|
|
|
|
2016-05-10 21:57:09 +00:00
|
|
|
// https://github.com/nasa/openmct/issues/910
|
|
|
|
it('updates visible rows in scope', function () {
|
|
|
|
var oldRows;
|
|
|
|
mockScope.rows = testRows;
|
|
|
|
controller.setRows(testRows);
|
|
|
|
oldRows = mockScope.visibleRows;
|
|
|
|
mockScope.toggleSort('col2');
|
|
|
|
expect(mockScope.visibleRows).not.toEqual(oldRows);
|
|
|
|
});
|
|
|
|
|
2016-04-22 16:44:34 +00:00
|
|
|
it('correctly sorts rows of differing types', function () {
|
|
|
|
mockScope.sortColumn = 'col2';
|
|
|
|
mockScope.sortDirection = 'desc';
|
|
|
|
|
|
|
|
testRows.push({
|
|
|
|
'col1': {'text': 'row4 col1'},
|
|
|
|
'col2': {'text': '123'},
|
|
|
|
'col3': {'text': 'row4 col3'}
|
|
|
|
});
|
|
|
|
testRows.push({
|
|
|
|
'col1': {'text': 'row5 col1'},
|
|
|
|
'col2': {'text': '456'},
|
|
|
|
'col3': {'text': 'row5 col3'}
|
|
|
|
});
|
|
|
|
testRows.push({
|
|
|
|
'col1': {'text': 'row5 col1'},
|
|
|
|
'col2': {'text': ''},
|
|
|
|
'col3': {'text': 'row5 col3'}
|
|
|
|
});
|
|
|
|
|
|
|
|
sortedRows = controller.sortRows(testRows);
|
|
|
|
expect(sortedRows[0].col2.text).toEqual('ghi');
|
|
|
|
expect(sortedRows[1].col2.text).toEqual('def');
|
|
|
|
expect(sortedRows[2].col2.text).toEqual('abc');
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
expect(sortedRows[sortedRows.length - 3].col2.text).toEqual('456');
|
|
|
|
expect(sortedRows[sortedRows.length - 2].col2.text).toEqual('123');
|
|
|
|
expect(sortedRows[sortedRows.length - 1].col2.text).toEqual('');
|
2016-04-22 16:44:34 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('The sort comparator', function () {
|
|
|
|
it('Correctly sorts different data types', function () {
|
|
|
|
var val1 = "",
|
|
|
|
val2 = "1",
|
|
|
|
val3 = "2016-04-05 18:41:30.713Z",
|
|
|
|
val4 = "1.1",
|
|
|
|
val5 = "8.945520958175627e-13";
|
|
|
|
mockScope.sortDirection = "asc";
|
|
|
|
|
|
|
|
expect(controller.sortComparator(val1, val2)).toEqual(-1);
|
|
|
|
expect(controller.sortComparator(val3, val1)).toEqual(1);
|
|
|
|
expect(controller.sortComparator(val3, val2)).toEqual(1);
|
|
|
|
expect(controller.sortComparator(val4, val2)).toEqual(1);
|
|
|
|
expect(controller.sortComparator(val2, val5)).toEqual(1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('Adding new rows', function () {
|
2016-03-09 05:36:33 +00:00
|
|
|
var row4,
|
|
|
|
row5,
|
|
|
|
row6;
|
|
|
|
|
2016-05-19 18:29:13 +00:00
|
|
|
beforeEach(function () {
|
2016-03-09 05:36:33 +00:00
|
|
|
row4 = {
|
|
|
|
'col1': {'text': 'row5 col1'},
|
|
|
|
'col2': {'text': 'xyz'},
|
|
|
|
'col3': {'text': 'row5 col3'}
|
|
|
|
};
|
|
|
|
row5 = {
|
|
|
|
'col1': {'text': 'row6 col1'},
|
|
|
|
'col2': {'text': 'aaa'},
|
|
|
|
'col3': {'text': 'row6 col3'}
|
|
|
|
};
|
|
|
|
row6 = {
|
|
|
|
'col1': {'text': 'row6 col1'},
|
|
|
|
'col2': {'text': 'ggg'},
|
|
|
|
'col3': {'text': 'row6 col3'}
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
it('Adds new rows at the correct sort position when' +
|
2016-05-19 18:29:13 +00:00
|
|
|
' sorted ', function () {
|
2016-03-09 05:36:33 +00:00
|
|
|
mockScope.sortColumn = 'col2';
|
|
|
|
mockScope.sortDirection = 'desc';
|
|
|
|
|
2016-03-14 02:37:08 +00:00
|
|
|
mockScope.displayRows = controller.sortRows(testRows.slice(0));
|
2016-03-09 05:36:33 +00:00
|
|
|
|
2016-03-14 02:37:08 +00:00
|
|
|
mockScope.rows.push(row4);
|
2016-05-19 18:29:13 +00:00
|
|
|
controller.addRow(undefined, mockScope.rows.length - 1);
|
2016-03-09 05:36:33 +00:00
|
|
|
expect(mockScope.displayRows[0].col2.text).toEqual('xyz');
|
2016-03-14 02:37:08 +00:00
|
|
|
|
|
|
|
mockScope.rows.push(row5);
|
2016-05-19 18:29:13 +00:00
|
|
|
controller.addRow(undefined, mockScope.rows.length - 1);
|
2016-03-09 05:36:33 +00:00
|
|
|
expect(mockScope.displayRows[4].col2.text).toEqual('aaa');
|
2016-03-14 02:37:08 +00:00
|
|
|
|
|
|
|
mockScope.rows.push(row6);
|
2016-05-19 18:29:13 +00:00
|
|
|
controller.addRow(undefined, mockScope.rows.length - 1);
|
2016-03-09 05:36:33 +00:00
|
|
|
expect(mockScope.displayRows[2].col2.text).toEqual('ggg');
|
|
|
|
|
|
|
|
//Add a duplicate row
|
2016-03-14 02:37:08 +00:00
|
|
|
mockScope.rows.push(row6);
|
2016-05-19 18:29:13 +00:00
|
|
|
controller.addRow(undefined, mockScope.rows.length - 1);
|
2016-03-09 05:36:33 +00:00
|
|
|
expect(mockScope.displayRows[2].col2.text).toEqual('ggg');
|
|
|
|
expect(mockScope.displayRows[3].col2.text).toEqual('ggg');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('Adds new rows at the correct sort position when' +
|
2016-05-19 18:29:13 +00:00
|
|
|
' sorted and filtered', function () {
|
2016-03-09 05:36:33 +00:00
|
|
|
mockScope.sortColumn = 'col2';
|
|
|
|
mockScope.sortDirection = 'desc';
|
|
|
|
mockScope.filters = {'col2': 'a'};//Include only
|
|
|
|
// rows with 'a'
|
|
|
|
|
2016-03-14 02:37:08 +00:00
|
|
|
mockScope.displayRows = controller.sortRows(testRows.slice(0));
|
2016-03-09 05:36:33 +00:00
|
|
|
mockScope.displayRows = controller.filterRows(testRows);
|
|
|
|
|
2016-03-14 02:37:08 +00:00
|
|
|
mockScope.rows.push(row5);
|
2016-05-19 18:29:13 +00:00
|
|
|
controller.addRow(undefined, mockScope.rows.length - 1);
|
2016-03-09 05:36:33 +00:00
|
|
|
expect(mockScope.displayRows.length).toBe(2);
|
|
|
|
expect(mockScope.displayRows[1].col2.text).toEqual('aaa');
|
|
|
|
|
2016-03-14 02:37:08 +00:00
|
|
|
mockScope.rows.push(row6);
|
2016-05-19 18:29:13 +00:00
|
|
|
controller.addRow(undefined, mockScope.rows.length - 1);
|
2016-03-09 05:36:33 +00:00
|
|
|
expect(mockScope.displayRows.length).toBe(2);
|
|
|
|
//Row was not added because does not match filter
|
|
|
|
});
|
|
|
|
|
|
|
|
it('Adds new rows at the correct sort position when' +
|
2016-04-22 16:44:34 +00:00
|
|
|
' not sorted ', function () {
|
2016-03-09 05:36:33 +00:00
|
|
|
mockScope.sortColumn = undefined;
|
|
|
|
mockScope.sortDirection = undefined;
|
|
|
|
mockScope.filters = {};
|
|
|
|
|
2016-03-14 02:37:08 +00:00
|
|
|
mockScope.displayRows = testRows.slice(0);
|
2016-03-09 05:36:33 +00:00
|
|
|
|
2016-03-14 02:37:08 +00:00
|
|
|
mockScope.rows.push(row5);
|
2016-05-19 18:29:13 +00:00
|
|
|
controller.addRow(undefined, mockScope.rows.length - 1);
|
2016-03-09 05:36:33 +00:00
|
|
|
expect(mockScope.displayRows[3].col2.text).toEqual('aaa');
|
2016-03-14 02:37:08 +00:00
|
|
|
|
|
|
|
mockScope.rows.push(row6);
|
2016-05-19 18:29:13 +00:00
|
|
|
controller.addRow(undefined, mockScope.rows.length - 1);
|
2016-03-09 05:36:33 +00:00
|
|
|
expect(mockScope.displayRows[4].col2.text).toEqual('ggg');
|
|
|
|
});
|
|
|
|
|
2016-04-22 16:44:34 +00:00
|
|
|
it('Resizes columns if length of any columns in new' +
|
2016-05-19 18:29:13 +00:00
|
|
|
' row exceeds corresponding existing column', function () {
|
2016-04-22 16:44:34 +00:00
|
|
|
var row7 = {
|
|
|
|
'col1': {'text': 'row6 col1'},
|
|
|
|
'col2': {'text': 'some longer string'},
|
|
|
|
'col3': {'text': 'row6 col3'}
|
|
|
|
};
|
|
|
|
|
|
|
|
mockScope.sortColumn = undefined;
|
|
|
|
mockScope.sortDirection = undefined;
|
|
|
|
mockScope.filters = {};
|
|
|
|
|
|
|
|
mockScope.displayRows = testRows.slice(0);
|
|
|
|
|
|
|
|
mockScope.rows.push(row7);
|
2016-05-19 18:29:13 +00:00
|
|
|
controller.addRow(undefined, mockScope.rows.length - 1);
|
2016-04-22 16:44:34 +00:00
|
|
|
expect(controller.$scope.sizingRow.col2).toEqual({text: 'some longer string'});
|
|
|
|
});
|
|
|
|
|
2016-03-09 05:36:33 +00:00
|
|
|
});
|
2016-02-26 19:09:51 +00:00
|
|
|
});
|
2016-03-09 05:36:33 +00:00
|
|
|
|
|
|
|
|
2016-02-26 19:09:51 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|