Multiple bug fixes and improvements. Shift, Unshift, and Splice are now supported to do inline array modifications without losing the original array reference across elements.

This commit is contained in:
Matt Mo 2020-10-07 08:30:03 -07:00
parent b49a513831
commit 08f4f9006f
2 changed files with 241 additions and 43 deletions

View File

@ -8,6 +8,10 @@ class IgniteProperty {
this.onChangeCallbacks = []; this.onChangeCallbacks = [];
this.onPushCallbacks = []; this.onPushCallbacks = [];
this.onPopCallbacks = []; this.onPopCallbacks = [];
this.onShiftCallbacks = [];
this.onUnshiftCallbacks = [];
this.onSpliceCallbacks = [];
this.arrayCallbacks = []; this.arrayCallbacks = [];
this.reflected = []; this.reflected = [];
this._value = val; this._value = val;
@ -64,31 +68,70 @@ class IgniteProperty {
if (Array.isArray(this._value) && this._value.onPushCallbacks == undefined) { if (Array.isArray(this._value) && this._value.onPushCallbacks == undefined) {
this._value.onPushCallbacks = []; this._value.onPushCallbacks = [];
this._value.onPopCallbacks = []; this._value.onPopCallbacks = [];
this._value.onShiftCallbacks = [];
this._value.onUnshiftCallbacks = [];
this._value.onSpliceCallbacks = [];
this._value.push = function () { this._value.push = function () {
var len = Array.prototype.push.apply(this, arguments); var len = Array.prototype.push.apply(this, arguments);
this.onPushCallbacks.forEach((callback) => callback.invoke(arguments[0])); this.onPushCallbacks.forEach(callback => callback.invoke(Array.from(arguments)));
return len; return len;
}; };
this._value.pop = function () { this._value.pop = function () {
var len = Array.prototype.pop.apply(this, arguments); var len = Array.prototype.pop.apply(this, arguments);
this.onPopCallbacks.forEach((callback) => callback.invoke()); this.onPopCallbacks.forEach(callback => callback.invoke());
return len; return len;
} }
this._value.unshift = function () {
var len = Array.prototype.unshift.apply(this, arguments);
this.onUnshiftCallbacks.forEach(callback => callback.invoke(Array.from(arguments)));
return len;
}
this._value.shift = function () {
var len = Array.prototype.shift.apply(this, arguments);
this.onShiftCallbacks.forEach(callback => callback.invoke());
return len;
};
this._value.splice = function (start, deleteCount = 0, ...items) {
var removed = Array.prototype.splice.apply(this, arguments);
this.onSpliceCallbacks.forEach(callback => callback.invoke(start, deleteCount, items));
return removed;
}
this._value.attachOnPush = function (func) { this._value.attachOnPush = function (func) {
var callback = new IgniteCallback(func, (detach) => this.detachOnPush(detach)); var callback = new IgniteCallback(func, detach => this.detachOnPush(detach));
this.onPushCallbacks.push(callback); this.onPushCallbacks.push(callback);
return callback; return callback;
} }
this._value.attachOnPop = function (func) { this._value.attachOnPop = function (func) {
var callback = new IgniteCallback(func, (detach) => this.detachOnPop(detach)); var callback = new IgniteCallback(func, detach => this.detachOnPop(detach));
this.onPopCallbacks.push(callback); this.onPopCallbacks.push(callback);
return callback; return callback;
} }
this._value.attachOnUnshift = function (func) {
var callback = new IgniteCallback(func, detach => this.detachOnUnshift(detach));
this.onUnshiftCallbacks.push(callback);
return callback;
}
this._value.attachOnShift = function (func) {
var callback = new IgniteCallback(func, detach => this.detachOnShift(detach));
this.onShiftCallbacks.push(callback);
return callback;
}
this._value.attachonSplice = function (func) {
var callback = new IgniteCallback(func, detach => this.detachonSplice(detach));
this.onSpliceCallbacks.push(callback);
return callback;
}
this._value.detachOnPush = function (callback) { this._value.detachOnPush = function (callback) {
this.onPushCallbacks = this.onPushCallbacks.filter(push => push != callback); this.onPushCallbacks = this.onPushCallbacks.filter(push => push != callback);
} }
@ -97,51 +140,89 @@ class IgniteProperty {
this.onPopCallbacks = this.onPopCallbacks.filter(pop => pop != callback); this.onPopCallbacks = this.onPopCallbacks.filter(pop => pop != callback);
} }
this.arrayCallbacks.push(this._value.attachOnPush(() => this.invokeOnPush())); this._value.detachOnUnshift = function (callback) {
this.arrayCallbacks.push(this._value.attachOnPop(() => this.invokeOnPop())); this.onUnshiftCallbacks = this.onUnshiftCallbacks.filter(unshift => unshift != callback);
} else if (Array.isArray(this._value) && this._value.onPushCallbacks) { }
this._value.detachOnShift = function (callback) {
this.onShiftCallbacks = this.onShiftCallbacks.filter(shift => shift != callback);
}
this._value.detachonSplice = function (callback) {
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 array has already been patched but attach to it so we get callbacks.
this.arrayCallbacks.push(this._value.attachOnPush(() => this.invokeOnPush())); this.arrayCallbacks.push(this._value.attachOnPush((items) => this.invokeOnPush(items)));
this.arrayCallbacks.push(this._value.attachOnPop(() => this.invokeOnPop())); this.arrayCallbacks.push(this._value.attachOnPop(() => this.invokeOnPop()));
this.arrayCallbacks.push(this._value.attachOnShift(() => this.invokeOnShift()));
this.arrayCallbacks.push(this._value.attachOnUnshift((items) => this.invokeOnUnshift(items)));
this.arrayCallbacks.push(this._value.attachonSplice((start, deleteCount, items) => this.invokeonSplice(start, deleteCount, items)));
} }
} }
invokeOnChange(oldValue, newValue) { invokeOnChange(oldValue, newValue) {
for (var i = 0; i < this.onChangeCallbacks.length; i++) { this.onChangeCallbacks.forEach(callback => callback.invoke(oldValue, newValue));
this.onChangeCallbacks[i].invoke(oldValue, newValue);
}
} }
invokeOnPush() { invokeOnPush(items) {
for (var i = 0; i < this.onPushCallbacks.length; i++) { this.onPushCallbacks.forEach(callback => callback.invoke(this._value, items));
this.onPushCallbacks[i].invoke(null, this._value);
}
} }
invokeOnPop() { invokeOnPop() {
for (var i = 0; i < this.onPopCallbacks.length; i++) { this.onPopCallbacks.forEach(callback => callback.invoke(this._value));
this.onPopCallbacks[i].invoke(null, this._value); }
}
invokeOnShift() {
this.onShiftCallbacks.forEach(callback => callback.invoke(this._value));
}
invokeOnUnshift(items) {
this.onUnshiftCallbacks.forEach(callback => callback.invoke(this._value, items));
}
invokeonSplice(start, deleteCount, items) {
this.onSpliceCallbacks.forEach(callback => callback.invoke(this._value, start, deleteCount, items));
} }
attachOnChange(onChange) { attachOnChange(onChange) {
var callback = new IgniteCallback(onChange, (detach) => this.detachOnChange(detach)); var callback = new IgniteCallback(onChange, detach => this.detachOnChange(detach));
this.onChangeCallbacks.push(callback); this.onChangeCallbacks.push(callback);
return callback; return callback;
} }
attachOnPush(onPush) { attachOnPush(onPush) {
var callback = new IgniteCallback(onPush, (detach) => this.detachOnPush(detach)); var callback = new IgniteCallback(onPush, detach => this.detachOnPush(detach));
this.onPushCallbacks.push(callback); this.onPushCallbacks.push(callback);
return callback; return callback;
} }
attachOnPop(onPop) { attachOnPop(onPop) {
var callback = new IgniteCallback(onPop, (detach) => this.detachOnPop(detach)); var callback = new IgniteCallback(onPop, detach => this.detachOnPop(detach));
this.onPopCallbacks.push(callback); this.onPopCallbacks.push(callback);
return callback; return callback;
} }
attachOnShift(onShift) {
var callback = new IgniteCallback(onShift, detach => this.detachOnShift(detach));
this.onShiftCallbacks.push(callback);
return callback;
}
attachOnUnshift(onUnshift) {
var callback = new IgniteCallback(onUnshift, detach => this.detachOnUnshift(detach));
this.onUnshiftCallbacks.push(callback);
return callback;
}
attachOnSplice(onSplice) {
var callback = new IgniteCallback(onSplice, detach => this.detachonSplice(detach));
this.onSpliceCallbacks.push(callback);
return callback;
}
detachOnChange(callback) { detachOnChange(callback) {
this.onChangeCallbacks = this.onChangeCallbacks.filter(change => change != callback); this.onChangeCallbacks = this.onChangeCallbacks.filter(change => change != callback);
} }
@ -153,6 +234,18 @@ class IgniteProperty {
detachOnPop(callback) { detachOnPop(callback) {
this.onPopCallbacks = this.onPopCallbacks.filter(pop => pop != callback); this.onPopCallbacks = this.onPopCallbacks.filter(pop => pop != callback);
} }
detachOnShift(callback) {
this.onShiftCallbacks = this.onShiftCallbacks.filter(shift => shift != callback);
}
detachOnUnshift(callback) {
this.onUnshiftCallbacks = this.onUnshiftCallbacks.filter(unshift => unshift != callback);
}
detachonSplice(callback) {
this.onSpliceCallbacks = this.onSpliceCallbacks.filter(slice => slice != callback);
}
} }
/** /**

View File

@ -39,7 +39,9 @@ class IgniteTemplate {
if (children) { if (children) {
for (var i = 0; i < children.length; i++) { for (var i = 0; i < children.length; i++) {
if (children[i] instanceof IgniteProperty) { if (children[i] === undefined || children[i] === null) {
continue; //Skip undefined or null children.
} else if (children[i] instanceof IgniteProperty) {
this.children.push(new html(children[i])); this.children.push(new html(children[i]));
} else if (children[i] instanceof String || typeof children[i] === 'string') { } else if (children[i] instanceof String || typeof children[i] === 'string') {
this.children.push(new html(children[i])); this.children.push(new html(children[i]));
@ -82,8 +84,11 @@ class IgniteTemplate {
//Attack a callback for all the properties //Attack a callback for all the properties
name.forEach(prop => { name.forEach(prop => {
this.callbacks.push(prop.attachOnChange((oldValue, newValue) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, oldValue)), converter(...name.getPropertyValues())))); this.callbacks.push(prop.attachOnChange((oldValue, newValue) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, oldValue)), converter(...name.getPropertyValues()))));
this.callbacks.push(prop.attachOnPush((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.attachOnPop((oldValue, newValue) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, oldValue)), 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()); var value = converter(...name.getPropertyValues());
@ -230,7 +235,9 @@ class IgniteTemplate {
child(...items) { child(...items) {
if (items) { if (items) {
for (var i = 0; i < items.length; i++) { for (var i = 0; i < items.length; i++) {
if (items[i] instanceof IgniteProperty) { if (items[i] === undefined || items[i] === null) {
continue; //Skip undefined or null items.
} else if (items[i] instanceof IgniteProperty) {
this.children.push(new html(items[i])); this.children.push(new html(items[i]));
} else if (items[i] instanceof String || typeof items[i] === 'string') { } else if (items[i] instanceof String || typeof items[i] === 'string') {
this.children.push(new html(items[i])); this.children.push(new html(items[i]));
@ -298,7 +305,7 @@ class IgniteTemplate {
/** /**
* Adds a onblur event handler to this template. * Adds a onblur event handler to this template.
* @param {Function|IgniteProperty} eventCallback THe callback function to be invoked by the event once it fires. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires.
* @returns This ignite template so function calls can be chained. * @returns This ignite template so function calls can be chained.
*/ */
onBlur(eventCallback) { onBlur(eventCallback) {
@ -307,13 +314,22 @@ class IgniteTemplate {
/** /**
* Adds a onfocus event handler to this template. * Adds a onfocus event handler to this template.
* @param {Function|IgniteProperty} eventCallback THe callback function to be invoked by the event once it fires. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires.
* @returns This ignite template so function calls can be chained. * @returns This ignite template so function calls can be chained.
*/ */
onFocus(eventCallback) { onFocus(eventCallback) {
return this.on("focus", eventCallback); return this.on("focus", eventCallback);
} }
/**
* Adds a onchange event handler to this template.
* @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires.
* @returns This ignite template so function calls can be chained.
*/
onChange(eventCallback) {
return this.on("change", eventCallback);
}
/** /**
* Adds a on enter key press event handler to this template. * Adds a on enter key press event handler to this template.
* @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires.
@ -452,8 +468,11 @@ class IgniteTemplate {
//Attack a callback for all the properties //Attack a callback for all the properties
value.forEach(prop => { value.forEach(prop => {
this.callbacks.push(prop.attachOnChange((oldValue, newValue) => this.onStyleChanged(name, converter(...value.getPropertyValues())))); this.callbacks.push(prop.attachOnChange((oldValue, newValue) => this.onStyleChanged(name, converter(...value.getPropertyValues()))));
this.callbacks.push(prop.attachOnPush((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.attachOnPop((oldValue, newValue) => 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 }; this.styles[name] = { name: name, value: converter(...value.getPropertyValues()), priority: priority };
@ -1263,8 +1282,11 @@ class list extends IgniteTemplate {
if (list instanceof IgniteProperty) { if (list instanceof IgniteProperty) {
this.callbacks.push(list.attachOnChange((oldValue, newValue) => this.onListChanged(oldValue, newValue))); this.callbacks.push(list.attachOnChange((oldValue, newValue) => this.onListChanged(oldValue, newValue)));
this.callbacks.push(list.attachOnPush((oldValue, newValue) => this.onListPush(oldValue, newValue))); this.callbacks.push(list.attachOnPush((list, items) => this.onListPush(list, items)));
this.callbacks.push(list.attachOnPop((oldValue, newValue) => this.onListPop(oldValue, newValue))); 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; this.list = list.value;
} else { } else {
@ -1331,38 +1353,121 @@ class list extends IgniteTemplate {
try { try {
this.construct(null); //The list changed, reconstruct this template. this.construct(null); //The list changed, reconstruct this template.
} catch (error) { } catch (error) {
console.error(error); console.error("An error occurred during onListChanged:", error);
} }
IgniteRenderingContext.leave(); IgniteRenderingContext.leave();
} }
onListPush(oldValue, newValue) { onListPush(list, items) {
IgniteRenderingContext.enter(); IgniteRenderingContext.enter();
try { try {
var template = this.forEach(this.list[this.list.length - 1]); items.forEach(item => {
var template = this.forEach(item);
if (this.elements.length > 0) { if (this.elements.length > 0) {
template.construct(this.element.parentElement, this.elements[this.elements.length - 1].nextSibling); template.construct(this.element.parentElement, this.elements[this.elements.length - 1].nextSibling);
} else { } else {
template.construct(this.element.parentElement, this.element); template.construct(this.element.parentElement, this.element);
} }
this.children.push(template); this.children.push(template);
this.elements.push(template.element); this.elements.push(template.element);
} catch { } });
} catch (error) {
console.error("An error occurred during onListPush:", error);
}
IgniteRenderingContext.leave(); IgniteRenderingContext.leave();
} }
onListPop(oldValue, newValue) { 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);
});
} catch (error) {
console.error("An error occurred during onListUnshift:", error);
}
IgniteRenderingContext.leave();
}
onListPop(list) {
if (this.children.length > 0) { if (this.children.length > 0) {
this.children[this.children.length - 1].deconstruct(); this.children[this.children.length - 1].deconstruct();
this.children.pop(); this.children.pop();
this.elements.pop(); this.elements.pop();
} }
} }
onListShift(list) {
if (this.children.length > 0) {
this.children[0].deconstruct();
this.children.shift();
this.elements.shift();
}
}
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;
});
}
IgniteRenderingContext.leave();
}
onStyleChanged(name, newValue) {
this.elements.forEach((element) => {
element.style.setProperty(name, newValue, this.styles[name].priority);
});
this.styles[name].value = newValue;
}
} }
/** /**