/** * Paginathing * Paginate Everything * * Original author Alfred Crosby * Inspired from http://esimakin.github.io/twbs-pagination/ * Modified to pure JavaScript and specialised to LibreOffice Help by * Ilmari Lauhakangas * * MIT License (Expat) * * Copyright (c) 2018 Alfred Crosby * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ // Polyfill for .after() (function (arr) { arr.forEach(function (item) { if (item.hasOwnProperty('after')) { return; } Object.defineProperty(item, 'after', { configurable: true, enumerable: true, writable: true, value: function after() { var argArr = Array.prototype.slice.call(arguments), docFrag = document.createDocumentFragment(); argArr.forEach(function (argItem) { var isNode = argItem instanceof Node; docFrag.appendChild(isNode ? argItem : document.createTextNode(String(argItem))); }); this.parentNode.insertBefore(docFrag, this.nextSibling); } }); }); })([Element.prototype, CharacterData.prototype, DocumentType.prototype]); var options = { perPage: 20, limitPagination: 6, prevNext: true, firstLast: true, prevText: '←', nextText: '→', firstText: '⇤', lastText: '⇥', containerClass: 'pagination-container', ulClass: 'pagination', liClass: 'page', activeClass: 'active', disabledClass: 'disabled' }; var Paginator = function(element) { var el = element; var startPage = 1; var currentPage = 1; var pageDivision = 0; var totalItems = el.children.length; var limitPagination = options.limitPagination; pageDivision = Math.ceil(totalItems / options.perPage); // let's not display pagination leading nowhere function pagLimit() { if (options.limitPagination >= pageDivision) { limitPagination = pageDivision; } else { limitPagination = options.limitPagination; } return limitPagination; } var totalPages = Math.max(pageDivision, pagLimit()); var existingContainer = document.getElementsByClassName('pagination-container')[0]; if (existingContainer) { parent = existingContainer.parentNode; parent.removeChild(existingContainer); } var container = document.createElement('nav'); container.setAttribute('class', options.containerClass); var ul = document.createElement('ul'); ul.setAttribute('class', options.ulClass); function paginationFunc(type, page) { var li = document.createElement('li'); var a = document.createElement('a'); a.setAttribute('href', '#'); var cssClass = type === 'number' ? options.liClass : type; var text = document.createTextNode(type === 'number' ? page : paginationText(type)); li.classList.add(cssClass); li.setAttribute('data-pagination-type', type); li.setAttribute('data-page', page); a.appendChild(text); li.appendChild(a); return li; } function paginationText(type) { return options[type + 'Text']; } function buildPagination() { var pagination = []; var prev = currentPage - 1 < startPage ? startPage : currentPage - 1; var next = currentPage + 1 > totalPages ? totalPages : currentPage + 1; var start = 0; var end = 0; var limit = limitPagination; if (limit) { if (currentPage <= Math.ceil(limit / 2) + 1) { start = 1; end = limit; } else if (currentPage + Math.floor(limit / 2) >= totalPages) { start = totalPages + 1 - limit; end = totalPages; } else { start = currentPage - Math.ceil(limit / 2); end = currentPage + Math.floor(limit / 2); } } else { start = startPage; end = totalPages; } // "First" button if (options.firstLast) { pagination.push(paginationFunc('first', startPage)); } // "Prev" button if (options.prevNext) { pagination.push(paginationFunc('prev', prev)); } // Pagination for (var i = start; i <= end; i++) { pagination.push(paginationFunc('number', i)); } // "Next" button if (options.prevNext) { pagination.push(paginationFunc('next', next)); } // "Last" button if (options.firstLast) { pagination.push(paginationFunc('last', totalPages)); } return pagination; } function render(page) { // Remove children before re-render (prevent duplicate) while (ul.hasChildNodes()) { ul.removeChild(ul.lastChild); } var paginationBuild = buildPagination(); paginationBuild.forEach(function(item) { ul.appendChild(item); }); // Manage active DOM var startAt = page === 1 ? 0 : (page - 1) * options.perPage; var endAt = page * options.perPage; var domLi = el.children; for (var i = 0, len = domLi.length; i < len; i++) { var item = domLi[i]; if (i >= startAt && i <= endAt) { item.classList.remove('hidden'); } else { item.classList.add('hidden'); } } // Manage active state var ulKids = ul.getElementsByTagName("li"); for (var i = 0, len = ulKids.length; i < len; i++) { var _li = ulKids[i]; var type = _li.getAttribute('data-pagination-type'); switch (type) { case 'number': if (parseInt(_li.getAttribute('data-page'), 10) === page) { _li.classList.add(options.activeClass); } break; case 'first': page === startPage && _li.classList.toggle(options.disabledClass); break; case 'last': page === totalPages && _li.classList.toggle(options.disabledClass); break; case 'prev': (page - 1) < startPage && _li.classList.toggle(options.disabledClass); break; case 'next': (page + 1) > totalPages && _li.classList.toggle(options.disabledClass); break; default: break; } } el.after(container); container.appendChild(ul); } function handle() { var pagLi = container.childNodes[0].childNodes; for (var i = 0, len = pagLi.length; i < len; i++) { (function() { var item = pagLi[i]; item.addEventListener('click', function(e) { e.preventDefault(); var page = parseInt(item.getAttribute('data-page'), 10); currentPage = page; // let's prevent the pagination from flowing to two rows if (currentPage >= 98) { limitPagination = 4; } else { limitPagination = pagLimit(); } show(page); }); }()); } } function show(page) { render(page); handle(); } show(startPage); return; };