Added pagination template and population template to help create pagination components. Allowed the use of non properties in property arrays for class, style ect.

This commit is contained in:
Matt Mo 2020-10-20 13:18:26 -07:00
parent 08f4f9006f
commit c172cb5599
2 changed files with 443 additions and 17 deletions

View File

@ -152,7 +152,7 @@ class IgniteProperty {
this.onSpliceCallbacks = this.onSpliceCallbacks.filter(slice => slice != callback);
}
}
if (Array.isArray(this._value) && this._value.onPushCallbacks) {
//This array has already been patched but attach to it so we get callbacks.
this.arrayCallbacks.push(this._value.attachOnPush((items) => this.invokeOnPush(items)));
@ -261,7 +261,13 @@ IgniteProperty.prototype.toString = function () {
*/
Array.prototype.getPropertyValues = function () {
var ret = [];
this.forEach(prop => ret.push(prop.value));
this.forEach(prop => {
if (prop instanceof IgniteProperty) {
ret.push(prop.value);
} else {
ret.push(prop);
}
});
return ret;
}
@ -274,7 +280,11 @@ Array.prototype.getOldPropertyValues = function (property, oldValue) {
if (prop == property) {
ret.push(oldValue);
} else {
ret.push(prop.value);
if (prop instanceof IgniteProperty) {
ret.push(prop.value);
} else {
ret.push(prop);
}
}
});
return ret;

View File

@ -83,12 +83,14 @@ class IgniteTemplate {
//Attack a callback for all the properties
name.forEach(prop => {
this.callbacks.push(prop.attachOnChange((oldValue, newValue) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, oldValue)), converter(...name.getPropertyValues()))));
this.callbacks.push(prop.attachOnPush((list, items) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues()))));
this.callbacks.push(prop.attachOnUnshift((list, items) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues()))));
this.callbacks.push(prop.attachOnPop((list) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues()))));
this.callbacks.push(prop.attachOnShift((list) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues()))));
this.callbacks.push(prop.attachOnSplice((list, start, deleteCount, items) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues()))));
if (prop instanceof IgniteProperty) {
this.callbacks.push(prop.attachOnChange((oldValue, newValue) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, oldValue)), converter(...name.getPropertyValues()))));
this.callbacks.push(prop.attachOnPush((list, items) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues()))));
this.callbacks.push(prop.attachOnUnshift((list, items) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues()))));
this.callbacks.push(prop.attachOnPop((list) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues()))));
this.callbacks.push(prop.attachOnShift((list) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues()))));
this.callbacks.push(prop.attachOnSplice((list, start, deleteCount, items) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues()))));
}
});
var value = converter(...name.getPropertyValues());
@ -467,12 +469,14 @@ class IgniteTemplate {
//Attack a callback for all the properties
value.forEach(prop => {
this.callbacks.push(prop.attachOnChange((oldValue, newValue) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
this.callbacks.push(prop.attachOnPush((list, items) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
this.callbacks.push(prop.attachOnUnshift((list, items) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
this.callbacks.push(prop.attachOnPop((list) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
this.callbacks.push(prop.attachOnShift((list) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
this.callbacks.push(prop.attachOnSplice((list, start, deleteCount, items) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
if (prop instanceof IgniteProperty) {
this.callbacks.push(prop.attachOnChange((oldValue, newValue) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
this.callbacks.push(prop.attachOnPush((list, items) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
this.callbacks.push(prop.attachOnUnshift((list, items) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
this.callbacks.push(prop.attachOnPop((list) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
this.callbacks.push(prop.attachOnShift((list) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
this.callbacks.push(prop.attachOnSplice((list, start, deleteCount, items) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
}
});
this.styles[name] = { name: name, value: converter(...value.getPropertyValues()), priority: priority };
@ -487,7 +491,7 @@ class IgniteTemplate {
/**
* Hides the element this template is constructing if the value is true.
* @param {Boolean} value If true hides the element this template is constructing. If an IgniteProperty is passed it's value will auto update this.
* @param {Boolean|IgniteProperty} value If true hides the element this template is constructing. If an IgniteProperty is passed it's value will auto update this.
* @param {Function} converter An optional function to convert the value if needed.
*/
hide(value, converter = null) {
@ -496,6 +500,17 @@ class IgniteTemplate {
});
}
/**
* Shows the element this template is constructing if the value is true.
* @param {Boolean|IgniteProperty} value
* @param {Function} converter
*/
show(value, converter = null) {
return this.style("display", value, true, (...params) => {
return ((converter != null && converter(...params)) || (converter == null && params[0])) ? null : "none";
});
}
/**
* Sets the id attribute of the element to be constructed by this template.
* @param {String|IgniteProperty} value The value to set for the id attribute of the element this template will construct.
@ -1623,6 +1638,405 @@ class slot extends IgniteTemplate {
}
}
/**
* A pagination is a template that segments a list of items into pages based
* on the items, page size, current page.
*/
class pagination extends IgniteTemplate {
constructor(list, pageSize, currentPage, forEach) {
super();
if (list instanceof IgniteProperty) {
this.callbacks.push(list.attachOnChange((oldValue, newValue) => this.onListChanged(oldValue, newValue)));
this.callbacks.push(list.attachOnPush((list, items) => this.onListPush(list, items)));
this.callbacks.push(list.attachOnUnshift((list, items) => this.onListUnshift(list, items)));
this.callbacks.push(list.attachOnPop(list => this.onListPop(list)));
this.callbacks.push(list.attachOnShift(list => this.onListShift(list)));
this.callbacks.push(list.attachOnSplice((list, start, deleteCount, items) => this.onListSplice(list, start, deleteCount, items)));
this.list = list.value;
} else {
this.list = list;
}
if (pageSize instanceof IgniteProperty) {
this.callbacks.push(pageSize.attachOnChange((oldValue, newValue) => this.onPageSizeChanged(newValue)));
this.pageSize = pageSize.value;
} else {
this.pageSize = pageSize;
}
if (currentPage instanceof IgniteProperty) {
this.callbacks.push(currentPage.attachOnChange((oldValue, newValue) => this.onCurrentPageChanged(newValue)));
this.currentPage = currentPage.value;
} else {
this.currentPage = currentPage;
}
this.forEach = forEach;
this.elements = [];
this.tagName = "shadow pagination";
this.pages = [];
}
recalculate() {
//Hide the elements in the current page.
this.pages[this.currentPage].forEach(item => item.style.setProperty("display", "none", "important"));
//Recreate the pages.
var pages = parseInt((this.list.length / this.pageSize)) + (this.list.length % this.pageSize > 0 ? 1 : 0);
this.pages = [];
for (var i = 0; i < pages; i++) {
this.pages.push([]);
}
//Add the elements to the correct pages.
for (var i = 0; i < this.elements.length; i++) {
var page = parseInt(i / this.pageSize);
this.pages[page].push(this.elements[i]);
}
//Adjust the current page if it's now incorrect.
if (this.currentPage >= pages) {
this.currentPage = pages - 1;
}
//Show the elements in the current page
this.pages[this.currentPage].forEach(item => item.style.removeProperty("display"));
}
construct(parent, sibling) {
//Don't construct if we have no parent, no sibling and no element.
if (!parent && !sibling && !this.element) {
return;
}
if (!this.element) {
this.element = window.document.createTextNode(""); //Use a textnode as our placeholder
if (sibling) {
sibling.parentElement.insertBefore(this.element, sibling);
} else {
parent.appendChild(this.element);
}
} else {
parent = this.element.parentElement;
}
//If we already have elements we created, destroy them.
if (this.elements.length > 0) {
for (var i = 0; i < this.elements.length; i++) {
this.elements[i].remove();
}
this.elements = [];
}
//If we already have children elements, deconstruct them.
//(Because we are about to recreate them)
if (this.children.length > 0) {
for (var i = 0; i < this.children.length; i++) {
this.children[i].deconstruct();
}
this.children = [];
}
//Init our pages to put elements into.
var pages = parseInt((this.list.length / this.pageSize)) + (this.list.length % this.pageSize > 0 ? 1 : 0);
this.pages = [];
for (var i = 0; i < pages; i++) {
this.pages.push([]);
}
//Construct all the items in our list and use the container
if (this.list) {
for (var i = 0; i < this.list.length; i++) {
var template = this.forEach(this.list[i]);
template.construct(parent, this.element);
var page = parseInt(i / this.pageSize);
if (page != this.currentPage) {
template.element.style.setProperty("display", "none", "important");
} else {
template.element.style.removeProperty("display");
}
this.pages[page].push(template.element);
this.children.push(template);
this.elements.push(template.element);
}
}
}
onPageSizeChanged(newValue) {
//Set the new page size
this.pageSize = newValue;
//Recalculate the pagination.
this.recalculate();
}
onCurrentPageChanged(newValue) {
//If the new page is invalid don't do this.
if (newValue >= this.pages.length) {
return;
} else if (newValue < 0) {
return;
}
//Hide all the elements in the current page
this.pages[this.currentPage].forEach(item => item.style.setProperty("display", "none", "important"));
//Set the new current page.
this.currentPage = newValue;
//Show the elements in the next page
this.pages[this.currentPage].forEach(item => item.style.removeProperty("display"));
}
onListChanged(oldValue, newValue) {
this.list = newValue;
IgniteRenderingContext.enter();
try {
this.construct(null); //The list changed, reconstruct this template.
} catch (error) {
console.error("An error occurred during onListChanged:", error);
}
IgniteRenderingContext.leave();
}
onListPush(list, items) {
IgniteRenderingContext.enter();
try {
items.forEach(item => {
var template = this.forEach(item);
if (this.elements.length > 0) {
template.construct(this.element.parentElement, this.elements[this.elements.length - 1].nextSibling);
} else {
template.construct(this.element.parentElement, this.element);
}
this.children.push(template);
this.elements.push(template.element);
});
//Recalculate the pagination.
this.recalculate();
} catch (error) {
console.error("An error occurred during onListPush:", error);
}
IgniteRenderingContext.leave();
}
onListUnshift(list, items) {
IgniteRenderingContext.enter();
try {
items.reverse();
items.forEach(item => {
var template = this.forEach(item);
if (this.elements.length > 0) {
template.construct(this.element.parentElement, this.elements[0]);
} else {
template.construct(this.element.parentElement, this.element);
}
this.children.unshift(template);
this.elements.unshift(template.element);
});
//Recalculate the pagination.
this.recalculate();
} catch (error) {
console.error("An error occurred during onListUnshift:", error);
}
IgniteRenderingContext.leave();
}
onListPop(list) {
if (this.children.length > 0) {
this.children[this.children.length - 1].deconstruct();
this.children.pop();
this.elements.pop();
}
//Recalculate the pagination.
this.recalculate();
}
onListShift(list) {
if (this.children.length > 0) {
this.children[0].deconstruct();
this.children.shift();
this.elements.shift();
}
//Recalculate the pagination.
this.recalculate();
}
onListSplice(list, start, deleteCount, items) {
IgniteRenderingContext.enter();
//Remove any items that are needed.
if (deleteCount > 0 && this.children.length > 0) {
for (var i = start; i < Math.min(this.children.length, start + deleteCount); i++) {
this.children[i].deconstruct();
}
this.children.splice(start, deleteCount);
this.elements.splice(start, deleteCount);
}
//If the start is greater than the length of the items adjust it.
if (start > this.children.length) {
start = Math.max(this.children.length - 1, 0);
}
//Append any new items if there are any.
if (items) {
items.forEach(item => {
var template = this.forEach(item);
if (this.elements.length > 0) {
template.construct(this.element.parentElement, this.elements[start]);
} else {
template.construct(this.element.parentElement, this.element);
}
this.children.splice(start, 0, template);
this.elements.splice(start, 0, template.element);
start += 1;
});
}
//Recalculate the pagination.
this.recalculate();
IgniteRenderingContext.leave();
}
}
/**
* An ignite template that can construct a population of items
* based on a count.
*/
class population extends IgniteTemplate {
/**
* Creates a new population with the number of items in it, a converter if needed, and a foreach function.
* @param {Number|IgniteProperty} count The number of items in this population.
* @param {Function} forEach A function to generate items in the population.
* @param {Function?} converter A converter to be used to convert the count if needed.
*/
constructor(count, forEach, converter = null) {
super();
if (count instanceof IgniteProperty) {
this.callbacks.push(count.attachOnChange((oldValue, newValue) => this.onCountChange(converter != null ? converter(newValue) : newValue)));
this.count = count.value;
} else if (Array.isArray(count) && count.length > 0 && count[0] instanceof IgniteProperty) {
//There must be a converter for this to work correctly
if (!converter) {
throw "Cannot pass an array of properties without using a converter!";
}
//Attack a callback for all the properties
count.forEach(prop => {
if (prop instanceof IgniteProperty) {
this.callbacks.push(prop.attachOnChange((oldValue, newValue) => this.onCountChange(converter(...count.getPropertyValues()))));
this.callbacks.push(prop.attachOnPush((list, items) => this.onCountChange(converter(...count.getPropertyValues()))));
this.callbacks.push(prop.attachOnUnshift((list, items) => this.onCountChange(converter(...count.getPropertyValues()))));
this.callbacks.push(prop.attachOnPop((list) => this.onCountChange(converter(...count.getPropertyValues()))));
this.callbacks.push(prop.attachOnShift((list) => this.onCountChange(converter(...count.getPropertyValues()))));
this.callbacks.push(prop.attachOnSplice((list, start, deleteCount, items) => this.onCountChange(converter(...count.getPropertyValues()))));
}
});
this.count = converter(...count.getPropertyValues());
} else {
this.count = (converter != null ? converter(count) : count);
}
this.forEach = forEach;
this.elements = [];
this.tagName = "shadow population";
}
construct(parent, sibling) {
//Don't construct if we have no parent, no sibling and no element.
if (!parent && !sibling && !this.element) {
return;
}
if (!this.element) {
this.element = window.document.createTextNode(""); //Use a textnode as our placeholder
if (sibling) {
sibling.parentElement.insertBefore(this.element, sibling);
} else {
parent.appendChild(this.element);
}
} else {
parent = this.element.parentElement;
}
//If we already have elements we created, destroy them.
if (this.elements.length > 0) {
this.elements.forEach(item => item.remove());
this.elements = [];
}
//Construct all the elements for this population.
for (var i = 0; i < this.count; i++) {
var template = this.forEach(i);
if (this.elements.length > 0) {
template.construct(this.element.parentElement, this.elements[this.elements.length - 1].nextSibling);
} else {
template.construct(this.element.parentElement, this.element);
}
this.elements.push(template.element);
}
}
onCountChange(newValue) {
IgniteRenderingContext.enter();
if (newValue != this.elements.length) {
//Remove all our existing elements.
this.elements.forEach(item => item.remove());
this.elements = [];
//Create new elements.
for (var i = 0; i < newValue; i++) {
var template = this.forEach(i);
if (this.elements.length > 0) {
template.construct(this.element.parentElement, this.elements[this.elements.length - 1].nextSibling);
} else {
template.construct(this.element.parentElement, this.element);
}
this.elements.push(template.element);
}
//Set the new count
this.count = newValue;
}
IgniteRenderingContext.leave();
}
}
export {
IgniteTemplate,
div,
@ -1650,5 +2064,7 @@ export {
select,
option,
script,
slot
slot,
pagination,
population
};