'use strict'; const router = require('../router.js'); const views = require('../util/views.js'); const holderTemplate = views.getTemplate('endless-pager'); const pageTemplate = views.getTemplate('endless-pager-page'); function isScrolledIntoView(element) { let top = 0; do { top += element.offsetTop || 0; element = element.offsetParent; } while(element); return ( (top >= window.scrollY) && (top <= window.scrollY + window.innerHeight)); } class EndlessPageView { constructor(ctx) { this._hostNode = document.getElementById('content-holder'); views.replaceContent(this._hostNode, holderTemplate()); } run(ctx) { this._destroy(); this._active = true; this._runningRequests = 0; this._initialPageLoad = true; this.clearMessages(); views.emptyContent(this._pagesHolderNode); this.minOffsetShown = null; this.maxOffsetShown = null; this.totalRecords = null; this.currentOffset = 0; this.defaultLimit = parseInt(ctx.parameters.limit || ctx.defaultLimit); const initialOffset = parseInt(ctx.parameters.offset || 0); this._loadPage(ctx, initialOffset, this.defaultLimit, true) .then(pageNode => { if (initialOffset !== 0) { pageNode.scrollIntoView(); } }); this._timeout = window.setInterval(() => { window.requestAnimationFrame(() => { this._probePageLoad(ctx); this._syncUrl(ctx); }); }, 250); views.monitorNodeRemoval(this._pagesHolderNode, () => this._destroy()); } get pageHeaderHolderNode() { return this._hostNode.querySelector('.page-header-holder'); } get topPageGuardNode() { return this._hostNode.querySelector('.page-guard.top'); } get bottomPageGuardNode() { return this._hostNode.querySelector('.page-guard.bottom'); } get _pagesHolderNode() { return this._hostNode.querySelector('.pages-holder'); } _destroy() { window.clearInterval(this._timeout); this._active = false; } _syncUrl(ctx) { let topPageNode = null; let element = document.elementFromPoint( window.innerWidth / 2, window.innerHeight / 2); while (element.parentNode !== null) { if (element.classList.contains('page')) { topPageNode = element; break; } element = element.parentNode; } if (!topPageNode) { return; } let topOffset = parseInt(topPageNode.getAttribute('data-offset')); let topLimit = parseInt(topPageNode.getAttribute('data-limit')); if (topOffset !== this.currentOffset) { router.replace( ctx.getClientUrlForPage( topOffset, topLimit === ctx.defaultLimit ? null : topLimit), ctx.state, false); this.currentOffset = topOffset; } } _probePageLoad(ctx) { if (!this._active || this._runningRequests) { return; } if (this.totalRecords === null) { return; } if (this.minOffsetShown > 0 && isScrolledIntoView(this.topPageGuardNode)) { this._loadPage( ctx, this.minOffsetShown - this.defaultLimit, this.defaultLimit, false); } if (this.maxOffsetShown < this.totalRecords && isScrolledIntoView(this.bottomPageGuardNode)) { this._loadPage( ctx, this.maxOffsetShown, this.defaultLimit, true); } } _loadPage(ctx, offset, limit, append) { this._runningRequests++; return new Promise((resolve, reject) => { ctx.requestPage(offset, limit).then(response => { if (!this._active) { this._runningRequests--; return Promise.reject(); } window.requestAnimationFrame(() => { let pageNode = this._renderPage(ctx, append, response); this._runningRequests--; resolve(pageNode); }); }, error => { this.showError(error.message); this._runningRequests--; reject(); }); }); } _renderPage(ctx, append, response) { let pageNode = null; if (response.total) { pageNode = pageTemplate({ totalPages: Math.ceil(response.total / response.limit), page: Math.ceil( (response.offset + response.limit) / response.limit), }); pageNode.setAttribute('data-offset', response.offset); pageNode.setAttribute('data-limit', response.limit); ctx.pageRenderer({ parameters: ctx.parameters, response: response, hostNode: pageNode.querySelector('.page-content-holder'), }); this.totalRecords = response.total; if (response.offset < this.minOffsetShown || this.minOffsetShown === null) { this.minOffsetShown = response.offset; } if (response.offset + response.results.length > this.maxOffsetShown || this.maxOffsetShown === null) { this.maxOffsetShown = response.offset + response.results.length; } response.results.addEventListener('remove', e => { this.maxOffsetShown--; this.totalRecords--; }); if (append) { this._pagesHolderNode.appendChild(pageNode); if (this._initialPageLoad && response.offset > 0) { window.scroll(0, pageNode.getBoundingClientRect().top); } } else { this._pagesHolderNode.prependChild(pageNode); window.scroll( window.scrollX, window.scrollY + pageNode.offsetHeight); } } else if (!response.results.length) { this.showInfo('No data to show'); } this._initialPageLoad = false; return pageNode; } clearMessages() { views.clearMessages(this._hostNode); } showSuccess(message) { views.showSuccess(this._hostNode, message); } showError(message) { views.showError(this._hostNode, message); } showInfo(message) { views.showInfo(this._hostNode, message); } } module.exports = EndlessPageView;