"use strict";
/**
* Provides scrolling lists with large data sets that behave in a very
* performance-optimised fashion by controlling the DOM elements currently
* on screen to ensure that only the visible elements are rendered and
* all other elements are simulated by variable height divs at the top
* and bottom of the scrolling list.
*
* This module requires that the AutoBind module is loaded before it
* will work.
* @class Infinilist
* @requires AutoBind
*/
var Shared = window.ForerunnerDB.shared,
View = Shared.modules.View;
/**
* Creates an infinilist instance.
* @param {Selector} selector A jQuery selector targeting the element that
* will contain the list items.
* @param {Selector} template jQuery selector of the template to use when
* rendering an individual list item.
* @param {Object} options The options object.
* @param {View} view The view to read data from.
* @constructor
*/
var Infinilist = function (selector, template, options, view) {
var self = this;
selector = $(selector);
options = options || {};
self.options = options.infinilist || {};
delete options.infinilist;
self.skip = 0;
self.limit = 0;
self.ignoreScroll = false;
self.previousScrollTop = 0;
self.selector = selector;
self.template = template;
self.view = view;
self.itemTopMargin = $("<div class='il_topMargin'></div>");
self.itemContainer = $("<div class='il_items'></div>");
self.itemBottomMargin = $("<div class='il_bottomMargin'></div>");
self.total = self.view.from().count(self.options.countQuery);
self.itemHeight(self.options.itemHeight);
selector.append(self.itemTopMargin);
selector.append(self.itemContainer);
selector.append(self.itemBottomMargin);
self.resize();
view.link(self.itemContainer, template, options);
selector.on('scroll', function () {
// Debounce scroll event
if (!self.scrollDebouceTimeout) {
self.scrollDebouceTimeout = setTimeout(function () {
self.scroll();
self.scrollDebouceTimeout = 0;
}, 16);
}
});
$(window).on('resize', function () {
// Debounce resize event
if (self.resizeDebouceTimeout) {
clearTimeout(self.resizeDebouceTimeout);
}
self.resizeDebouceTimeout = setTimeout(function () {
self.resize();
}, 16);
});
};
Shared.addModule('Infinilist', Infinilist);
Shared.mixin(Infinilist.prototype, 'Mixin.Events');
Shared.synthesize(Infinilist.prototype, 'itemHeight', function (val) {
var self = this;
if (val !== undefined) {
self.virtualHeight = self.total * val;
self._itemHeight = val;
self.resize();
}
return this.$super.apply(this, arguments);
});
/**
* Handle screen resizing.
*/
Infinilist.prototype.resize = function () {
var self = this,
newHeight = self.selector.height(),
skipCount,
scrollTop = self.selector.scrollTop();
if (self.oldHeight !== newHeight) {
self.oldHeight = newHeight;
// Calculate number of visible items
self.maxItemCount = Math.ceil(newHeight / self._itemHeight);
skipCount = Math.floor(scrollTop / self._itemHeight);
self.skip = skipCount;
self.limit = self.maxItemCount + 1;
self.view.queryOptions(self.currentRange());
self.itemBottomMargin.height(self.virtualHeight - (skipCount * self._itemHeight)- (self.maxItemCount * self._itemHeight));
}
};
Infinilist.prototype.currentRange = function () {
return {
$skip: this.skip,
$limit: this.limit
};
};
Infinilist.prototype.scroll = function () {
var self = this,
delta,
skipCount,
scrollTop = self.selector.scrollTop();
// Get the current scroll position
delta = scrollTop - self.previousScrollTop;
self.previousScrollTop = scrollTop;
// Check if a scroll change occurred
if (delta !== 0) {
// Determine the new item range
skipCount = Math.floor(scrollTop / self._itemHeight);
self.skip = skipCount;
self.view.queryOptions(self.currentRange());
self.itemTopMargin.height(skipCount * self._itemHeight);
self.itemBottomMargin.height(self.virtualHeight - (skipCount * self._itemHeight)- (self.maxItemCount * self._itemHeight));
}
self.emit('scroll');
};
Infinilist.prototype.scrollToQuery = function (query, options, callback) {
var self = this,
result,
index,
orderOp = {
$orderBy: self.view.queryOptions().$orderBy
},
tmpColl,
scrollPos;
if (typeof options === 'function') {
callback = options;
options = undefined;
}
// Ensure options has properties we expect
options = options || {};
options.$inc = options.$inc !== undefined ? options.$inc : 0;
options.$incItem = options.$incItem !== undefined ? options.$incItem : 0;
// Run query and get first matching record (with current sort)
result = self.view.from().findOne(query, orderOp);
// Find the position of the element inside the current view
// based on the sort order
tmpColl = self.view.db().collection('tmpSortCollection');
tmpColl.setData(self.view.from().find(self.view.query()));
index = tmpColl.indexOf(result, orderOp);
tmpColl.drop();
if (index > -1) {
scrollPos = ((index + options.$incItem) * self._itemHeight) + options.$inc;
scrollPos = scrollPos > 0 ? scrollPos : 0;
if (self.selector.scrollTop() !== scrollPos) {
if (callback) {
self.once('scroll', function () {
callback();
});
}
// Scroll the main element to the position of the item
self.selector.scrollTop(scrollPos);
} else {
callback();
}
return true;
}
return false;
};
Infinilist.prototype.drop = function () {
var self = this;
// Unlink the view from the dom
self.view.unlink(self.itemContainer, self.template);
// Set state to dropped
self._state = 'dropped';
// Kill listeners
self.selector.off('scroll');
$(window).off('resize');
// Remove references
delete self.ignoreScroll;
delete self.previousScrollTop;
delete self._itemHeight;
delete self.selector;
delete self.template;
delete self.view;
delete self.itemTopMargin;
delete self.itemContainer;
delete self.itemBottomMargin;
};
View.prototype.infinilist = function (targetSelector, templateSelector, options) {
var target = window.jQuery(targetSelector);
if (templateSelector === undefined) {
return target.data('infinilist');
}
target.data('infinilist', new Infinilist(targetSelector, templateSelector, options, this));
};
View.prototype.unInfinilist = function (targetSelector) {
var target = window.jQuery(targetSelector);
if (target.data('infinilist')) {
target.data('infinilist').drop();
target.removeData('infinilist');
return true;
}
return false;
};
Shared.moduleFinished('AutoBind', function () {
Shared.finishModule('Infinilist');
});
module.exports = Infinilist;