Files
ionic-framework/test/unit/angular/service/collectionRepeatManager.unit.js

507 lines
17 KiB
JavaScript

describe('collectionRepeatManager service', function() {
beforeEach(module('ionic'));
var afThrottle;
beforeEach(function() {
afThrottle = ionic.animationFrameThrottle;
ionic.animationFrameThrottle = function(cb) {
return function() {
cb.apply(this, arguments);
};
};
});
afterEach(function() {
ionic.animationFrameThrottle = afThrottle;
});
function setup(options, dataSourceOptions, scrollViewOptions) {
var manager;
inject(function($collectionDataSource, $rootScope, $compile, $collectionRepeatManager) {
var dataSource = new $collectionDataSource(angular.extend({
scope: $rootScope.$new(),
transcludeParent: angular.element('<div>'),
transcludeFn: function(scope, cb) {
cb($compile('<div></div>')(scope));
},
keyExpr: 'key',
listExpr: 'list',
heightGetter: function() {
return 1;
},
widthGetter: function() {
return 1;
}
}, dataSourceOptions || {}));
var el = angular.element('<div scroll-container>')
.append(angular.element('<div scroll-content>'));
options = angular.extend({
dataSource: dataSource,
element: el,
scrollView: new ionic.views.Scroll(angular.extend({
el: el[0]
}, scrollViewOptions || {})),
}, options || {});
manager = new $collectionRepeatManager(options);
});
return manager;
}
it('should have properties', function() {
var manager = setup();
expect(manager.dataSource).toBeTruthy();
expect(manager.scrollView).toBeTruthy();
expect(manager.element[0]).toBe(manager.scrollView.__container);
});
it('should be isVertical if scrollingY', function() {
var manager = setup();
expect(manager.scrollView.options.scrollingY).toBe(true);
expect(manager.isVertical).toBe(true);
});
it('should be !isVertical if scrollingX', function() {
var manager = setup({}, {}, {
scrollingX: true,
scrollingY: false
});
expect(manager.scrollView.options.scrollingX).toBe(true);
expect(manager.scrollView.options.scrollingY).toBe(false);
expect(manager.isVertical).toBe(false);
});
describe('isVertical', function() {
it('should set getContentHeight', function() {
var manager = setup();
manager.viewportSize = 55;
expect(manager.scrollView.options.getContentHeight()).toBe(55);
});
it('.scrollValue()', function() {
var manager = setup();
manager.scrollView.__scrollTop = 33;
expect(manager.scrollValue()).toBe(33);
});
it('.scrollMaxValue()', function() {
var manager = setup();
manager.scrollView.__maxScrollTop = 123;
expect(manager.scrollMaxValue()).toBe(123);
});
it('.scrollSize()', function() {
var manager = setup();
manager.scrollView.__clientHeight = 12;
expect(manager.scrollSize()).toBe(12);
});
it('.secondaryScrollSize()', function() {
var manager = setup();
manager.scrollView.__clientWidth = 999;
expect(manager.secondaryScrollSize()).toBe(999);
});
it('.transformString()', function() {
var manager = setup();
expect(manager.transformString(3,6)).toBe('translate3d(6px,3px,0)');
});
it('.primaryDimension()', function() {
var manager = setup();
expect(manager.primaryDimension({height: 100})).toBe(100);
});
it('.secondaryDimension()', function() {
var manager = setup();
expect(manager.secondaryDimension({width: 99})).toBe(99);
});
});
describe('!isVertical', function() {
it('should set getContentWidth', function() {
var manager = setup({}, {}, { scrollingX: true, scrollingY: false });
manager.viewportSize = 55;
expect(manager.scrollView.options.getContentWidth()).toBe(55);
});
it('.scrollValue()', function() {
var manager = setup({}, {}, { scrollingX: true, scrollingY: false });
manager.scrollView.__scrollLeft = 33;
expect(manager.scrollValue()).toBe(33);
});
it('.scrollMaxValue()', function() {
var manager = setup({}, {}, { scrollingX: true, scrollingY: false });
manager.scrollView.__maxScrollLeft = 123;
expect(manager.scrollMaxValue()).toBe(123);
});
it('.scrollSize()', function() {
var manager = setup({}, {}, { scrollingX: true, scrollingY: false });
manager.scrollView.__clientWidth = 12;
expect(manager.scrollSize()).toBe(12);
});
it('.secondaryScrollSize()', function() {
var manager = setup({}, {}, { scrollingX: true, scrollingY: false });
manager.scrollView.__clientHeight = 999;
expect(manager.secondaryScrollSize()).toBe(999);
});
it('.transformString()', function() {
var manager = setup({}, {}, { scrollingX: true, scrollingY: false });
expect(manager.transformString(3,6)).toBe('translate3d(3px,6px,0)');
});
it('.primaryDimension()', function() {
var manager = setup({}, {}, { scrollingX: true, scrollingY: false });
expect(manager.primaryDimension({width: 100})).toBe(100);
});
it('.secondaryDimension()', function() {
var manager = setup({}, {}, { scrollingX: true, scrollingY: false });
expect(manager.secondaryDimension({height: 99})).toBe(99);
});
});
describe('.calculateDimensions()', function() {
it('should work with 1 item per space', function() {
var manager = setup();
manager.dataSource.dimensions = [
{ width: 100, height: 20 },
{ width: 100, height: 30 },
{ width: 100, height: 40 },
{ width: 100, height: 50 }
];
manager.secondaryScrollSize = function() {
return 100;
};
var result = manager.calculateDimensions();
expect(result[0].primarySize).toBe(20);
expect(result[0].secondarySize).toBe(100);
expect(result[0].primaryPos).toBe(0);
expect(result[0].secondaryPos).toBe(0);
expect(result[1].primarySize).toBe(30);
expect(result[1].secondarySize).toBe(100);
expect(result[1].primaryPos).toBe(20);
expect(result[1].secondaryPos).toBe(0);
expect(result[2].primarySize).toBe(40);
expect(result[2].secondarySize).toBe(100);
expect(result[2].primaryPos).toBe(50);
expect(result[2].secondaryPos).toBe(0);
expect(result[3].primarySize).toBe(50);
expect(result[3].secondarySize).toBe(100);
expect(result[3].primaryPos).toBe(90);
expect(result[3].secondaryPos).toBe(0);
});
it('should work with a grid', function() {
var manager = setup();
manager.dataSource.dimensions = [
{ width: 30, height: 30 },
{ width: 35, height: 30 },
{ width: 40, height: 30 },
{ width: 30, height: 30 },
{ width: 20, height: 30 },
{ width: 35, height: 30 },
];
manager.secondaryScrollSize = function() {
return 90;
};
var result = manager.calculateDimensions();
expect(result[0].primarySize).toBe(30);
expect(result[0].secondarySize).toBe(30);
expect(result[0].primaryPos).toBe(0);
expect(result[0].secondaryPos).toBe(0);
expect(result[1].primarySize).toBe(30);
expect(result[1].secondarySize).toBe(35);
expect(result[1].primaryPos).toBe(0);
expect(result[1].secondaryPos).toBe(30);
expect(result[2].primarySize).toBe(30);
expect(result[2].secondarySize).toBe(40);
expect(result[2].primaryPos).toBe(30);
expect(result[2].secondaryPos).toBe(0);
expect(result[3].primarySize).toBe(30);
expect(result[3].secondarySize).toBe(30);
expect(result[3].primaryPos).toBe(30);
expect(result[3].secondaryPos).toBe(40);
expect(result[4].primarySize).toBe(30);
expect(result[4].secondarySize).toBe(20);
expect(result[4].primaryPos).toBe(30);
expect(result[4].secondaryPos).toBe(70);
expect(result[5].primarySize).toBe(30);
expect(result[5].secondarySize).toBe(35);
expect(result[5].primaryPos).toBe(60);
expect(result[5].secondaryPos).toBe(0);
});
});
describe('.resize()', function() {
it('should work without data', function() {
var manager = setup();
spyOn(manager, 'render');
spyOn(manager, 'calculateDimensions').andReturn([]);
spyOn(manager, 'setCurrentIndex');
manager.resize();
expect(manager.dimensions).toEqual([]);
expect(manager.viewportSize).toBe(0);
expect(manager.setCurrentIndex).toHaveBeenCalledWith(0);
expect(manager.render).toHaveBeenCalledWith(true);
});
it('should work with data', function() {
var manager = setup();
spyOn(manager, 'calculateDimensions').andReturn([{
primaryPos: 100, primarySize: 30
}]);
manager.resize();
expect(manager.viewportSize).toBe(130);
});
});
describe('.setCurrentIndex()', function() {
it('without prev or next', function() {
var manager = setup();
spyOn(manager.dataSource, 'getLength').andReturn(0);
manager.setCurrentIndex(0);
expect(manager.currentIndex).toBe(0);
expect(manager.hasPrevIndex).toBe(false);
expect(manager.previousPos).toBeFalsy();
expect(manager.hasNextIndex).toBe(false);
expect(manager.nextPos).toBeFalsy();
});
it('with next', function() {
var manager = setup();
spyOn(manager.dataSource, 'getLength').andReturn(2);
manager.dimensions = [{ primaryPos: 0 }, { primaryPos: 25 }];
manager.setCurrentIndex(0);
expect(manager.currentIndex).toBe(0);
expect(manager.hasPrevIndex).toBe(false);
expect(manager.previousPos).toBeFalsy();
expect(manager.hasNextIndex).toBe(true);
expect(manager.nextPos).toBe(25);
});
it('with prev', function() {
var manager = setup();
spyOn(manager.dataSource, 'getLength').andReturn(2);
manager.dimensions = [{ primaryPos: 0 }, { primaryPos: 25 }];
manager.setCurrentIndex(1);
expect(manager.currentIndex).toBe(1);
expect(manager.hasPrevIndex).toBe(true);
expect(manager.previousPos).toBe(0);
expect(manager.hasNextIndex).toBe(false);
expect(manager.nextPos).toBeFalsy();
});
it('with next and prev', function() {
var manager = setup();
spyOn(manager.dataSource, 'getLength').andReturn(3);
manager.dimensions = [{ primaryPos: 0 }, { primaryPos: 25 }, { primaryPos: 50 }];
manager.setCurrentIndex(1);
expect(manager.currentIndex).toBe(1);
expect(manager.hasPrevIndex).toBe(true);
expect(manager.previousPos).toBe(0);
expect(manager.hasNextIndex).toBe(true);
expect(manager.nextPos).toBe(50);
});
});
describe('.renderScroll()', function() {
it('should pass the values to __$callback', function() {
var manager = setup();
spyOn(manager.scrollView, '__$callback');
manager.renderScroll(1, 2, 3, 4);
expect(manager.scrollView.__$callback).toHaveBeenCalledWith(1, 2, 3, 4);
});
});
describe('.renderIfNeeded()', function() {
it('should render if >= nextPos', function() {
var manager = setup();
spyOn(manager, 'render');
manager.hasNextIndex = true;
manager.nextPos = 30;
manager.renderIfNeeded(20);
expect(manager.render).not.toHaveBeenCalled();
manager.renderIfNeeded(30);
expect(manager.render).toHaveBeenCalled();
});
it('should render if < previousPos', function() {
var manager = setup();
spyOn(manager, 'render');
manager.hasPrevIndex = true;
manager.previousPos = 50;
manager.renderIfNeeded(60);
expect(manager.render).not.toHaveBeenCalled();
manager.renderIfNeeded(50);
expect(manager.render).not.toHaveBeenCalled();
manager.renderIfNeeded(49);
expect(manager.render).toHaveBeenCalled();
});
});
describe('.getIndexForScrollValue()', function() {
it('should scroll until it finds right position', function() {
var manager = setup();
manager.dimensions = [
{ primaryPos: 0 },
{ primaryPos: 100 },
{ primaryPos: 200 },
{ primaryPos: 300 },
{ primaryPos: 400 },
{ primaryPos: 500 }
];
expect(manager.getIndexForScrollValue(3, 50)).toBe(1);
expect(manager.getIndexForScrollValue(1, 450)).toBe(4);
//bounds
expect(manager.getIndexForScrollValue(0, 1000)).toBe(5);
expect(manager.getIndexForScrollValue(2, -100)).toBe(0);
});
});
describe('.render()', function() {
it('currentIndex >= length should remove all and return', function() {
var manager = setup();
manager.renderedItems = {'a':1, 'b':1};
spyOn(manager, 'removeItem');
manager.render();
expect(manager.removeItem).toHaveBeenCalledWith('a');
expect(manager.removeItem).toHaveBeenCalledWith('b');
});
it('shouldRedrawAll should remove all', function() {
var manager = setup();
manager.renderedItems = {'a':1, 'b':1};
spyOn(manager, 'removeItem');
manager.render(true);
expect(manager.removeItem).toHaveBeenCalledWith('a');
expect(manager.removeItem).toHaveBeenCalledWith('b');
});
function mockRendering(options) {
var manager = setup({}, {
keyExpr: 'item',
listExpr: 'items',
heightGetter: function() {
return options.itemHeight;
},
widthGetter: function() {
return options.itemWidth;
}
});
spyOn(manager, 'scrollSize').andReturn(options.scrollHeight);
spyOn(manager, 'secondaryScrollSize').andReturn(options.scrollWidth);
spyOn(manager, 'renderItem').andCallFake(function(i) {
manager.renderedItems[i] = true;
});
spyOn(manager, 'removeItem').andCallFake(function(i) {
delete manager.renderedItems[i];
});
var data = [];
for (var i = 0; i < 100; i++) {
data.push(i);
}
manager.dataSource.setData(data);
return manager;
}
it('should render the first items that fit on screen', function() {
var manager = mockRendering({
itemWidth: 3,
itemHeight: 20,
scrollWidth: 10,
scrollHeight: 100
});
manager.resize(); //triggers render
//it should render (items that fit * items per row) with one extra row at end
expect(Object.keys(manager.renderedItems).length).toBe(18);
for (var i = 0; i < 18; i++) {
expect(manager.renderedItems[i]).toBe(true);
}
expect(manager.renderedItems[18]).toBeUndefined();
});
it('should render items in the middle of the screen', function() {
var manager = mockRendering({
itemWidth: 3,
itemHeight: 20,
scrollWidth: 10,
scrollHeight: 100
});
spyOn(manager, 'scrollValue').andReturn(111);
manager.resize();
var startIndex = 17;
var bufferStartIndex = 14; //one row of buffer before the start
var bufferEndIndex = 35; //start + 17 + 6
expect(Object.keys(manager.renderedItems).length).toBe(22);
for (var i = bufferStartIndex; i <= bufferEndIndex; i++) {
expect(manager.renderedItems[i]).toBe(true);
}
expect(manager.renderedItems[bufferStartIndex - 1]).toBeUndefined();
expect(manager.renderedItems[bufferEndIndex + 1]).toBeUndefined();
});
it('should remove items outside the range', function() {
var manager = mockRendering({
itemWidth: 3,
itemHeight: 20,
scrollWidth: 10,
scrollHeight: 100
});
manager.resize();
manager.removeItem.reset();
manager.renderedItems = { 17: true, 18: true, 19: true };
//resize() re-renders everything, need to just do a normal rerender
manager.render();
expect(manager.removeItem.callCount).toBe(2);
expect(manager.removeItem).toHaveBeenCalledWith('18');
expect(manager.removeItem).toHaveBeenCalledWith('19');
});
});
describe('.renderItem()', function() {
it('should attachItemAtIndex and set the element transform', function() {
var manager = setup();
var item = {
element: angular.element('<div>')
};
spyOn(item.element, 'css');
spyOn(manager.dataSource, 'attachItemAtIndex').andReturn(item);
manager.renderItem(0, 33, 44);
expect(manager.dataSource.attachItemAtIndex).toHaveBeenCalledWith(0);
expect(item.element.css).toHaveBeenCalledWith(
ionic.CSS.TRANSFORM,
manager.transformString(33, 44)
);
expect(manager.renderedItems[0]).toBe(item);
});
});
describe('.removeItem()', function() {
it('should detachItem', function() {
var manager = setup();
var item = {};
manager.renderedItems[0] = item;
spyOn(manager, 'removeItem').andCallThrough();
spyOn(manager.dataSource, 'detachItem');
manager.removeItem(0);
expect(manager.renderedItems).toEqual({});
});
});
});