Simplified callbacks into a single class with a detach function option so we can use them generically. Patached arrays can have multiple on push/on pop callbacks and moved over to this new model. Needs more testing and cleanup.

This commit is contained in:
Matt Mo 2020-08-22 18:11:37 -07:00
parent 56530fc966
commit d4df41e427
3 changed files with 75 additions and 75 deletions

View File

@ -7,10 +7,11 @@ class IgniteProperty {
this.onChangeCallbacks = []; this.onChangeCallbacks = [];
this.onPushCallbacks = []; this.onPushCallbacks = [];
this.onPopCallbacks = []; this.onPopCallbacks = [];
this.callbacks = [];
this._value = val; this._value = val;
//Attempt to patch the value if it's a list. //Attempt to patch the value if it's a list.
this.patchListValue(); this.patchList();
} }
get value() { get value() {
@ -21,28 +22,59 @@ class IgniteProperty {
var old = this._value; var old = this._value;
this._value = val; this._value = val;
//Disconnect any existing callbacks
this.callbacks.forEach(callback => callback.disconnect());
this.callbacks = [];
//Attempt to patch the value if it's a list. //Attempt to patch the value if it's a list.
this.patchListValue(); this.patchList();
//Invoke any callbacks letting them know the value changed. //Invoke any callbacks letting them know the value changed.
this.invokeOnChange(old, val); this.invokeOnChange(old, val);
} }
patchListValue() { patchList() {
if (Array.isArray(this._value)) { if (Array.isArray(this._value) && this._value.onPushCallbacks == undefined) {
let prop = this; this._value.onPushCallbacks = [];
this._value.onPopCallbacks = [];
this._value.push = function () { this._value.push = function () {
var len = Array.prototype.push.apply(this, arguments); var len = Array.prototype.push.apply(this, arguments);
prop.invokeOnPush(); this.onPushCallbacks.forEach((callback) => callback.invoke(arguments[0]));
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);
prop.invokeOnPop(); this.onPopCallbacks.forEach((callback) => callback.invoke());
return len; return len;
} }
this._value.attachOnPush = function (func) {
var callback = new IgniteCallback(func, (detach) => this.detachOnPush(detach));
this.onPushCallbacks.push(callback);
return callback;
}
this._value.attachOnPop = function (func) {
var callback = new IgniteCallback(func, (detach) => this.detachOnPop(detach));
this.onPopCallbacks.push(callback);
return callback;
}
this._value.detachOnPush = function (callback) {
this.onPushCallbacks = this.onPushCallbacks.filter(push => push != callback);
}
this._value.detachOnPop = function (callback) {
this.onPopCallbacks = this.onPopCallbacks.filter(pop => pop != callback);
}
this.callbacks.push(this._value.attachOnPush(() => this.invokeOnPush()));
this.callbacks.push(this._value.attachOnPop(() => this.invokeOnPop()));
} else if (Array.isArray(this._value) && this._value.onPushCallbacks) {
this.callbacks.push(this._value.attachOnPush(() => this.invokeOnPush()));
this.callbacks.push(this._value.attachOnPop(() => this.invokeOnPop()));
} }
} }
@ -65,22 +97,34 @@ class IgniteProperty {
} }
attachOnChange(onChange) { attachOnChange(onChange) {
var callback = new IgnitePropertyOnChangeCallback(this, onChange); 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 IgnitePropertyOnPushCallback(this, onPush); 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 IgnitePropertyOnPopCallback(this, onPop); var callback = new IgniteCallback(onPop, (detach) => this.detachOnPop(detach));
this.onPopCallbacks.push(callback); this.onPopCallbacks.push(callback);
return callback; return callback;
} }
detachOnChange(callback) {
this.onChangeCallbacks = this.onChangeCallbacks.filter(change => change != callback);
}
detachOnPush(callback) {
this.onPushCallbacks = this.onPushCallbacks.filter(push => push != callback);
}
detachOnPop(callback) {
this.onPopCallbacks = this.onPopCallbacks.filter(pop => pop != callback);
}
} }
/** /**
@ -92,79 +136,28 @@ IgniteProperty.prototype.toString = function () {
} }
/** /**
* The outline of a ignite property onchange callback that allows * The outline of a ignite callback that can be invoked and disconnected
* disconnecting of callbacks on demand when they are no longer needed. * to help maintain and cleanup callbacks.
*/ */
class IgnitePropertyOnChangeCallback { class IgniteCallback {
constructor(property, callback) { constructor(callback, detach) {
this.callback = callback; this.callback = callback;
this.property = property; this.detach = detach;
} }
invoke(oldValue, newValue) { invoke(...params) {
console.log("Invoke with params:", params);
if (this.callback) { if (this.callback) {
this.callback(oldValue, newValue); this.callback(...params);
} }
} }
disconnect() { disconnect() {
this.callback = null; this.callback = null;
if (this.property) { if (this.detach) {
this.property.onChangeCallbacks = this.property.onChangeCallbacks.filter(callback => callback != this); this.detach(this);
this.property = null;
}
}
}
/**
* The outline of a ignite property onpush callback that allows
* disconnecting of callbacks on demand when they are no longer needed.
*/
class IgnitePropertyOnPushCallback {
constructor(property, callback) {
this.callback = callback;
this.property = property;
}
invoke(oldValue, newValue) {
if (this.callback) {
this.callback(oldValue, newValue);
}
}
disconnect() {
this.callback = null;
if (this.property) {
this.property.onPushCallbacks = this.property.onPushCallbacks.filter(callback => callback != this);
this.property = null;
}
}
}
/**
* The outline of a ignite property onpop callback that allows
* disconnecting of callbacks on demand when they are no longer needed.
*/
class IgnitePropertyOnPopCallback {
constructor(property, callback) {
this.callback = callback;
this.property = property;
}
invoke(oldValue, newValue) {
if (this.callback) {
this.callback(oldValue, newValue);
}
}
disconnect() {
this.callback = null;
if (this.property) {
this.property.onPopCallbacks = this.property.onPopCallbacks.filter(callback => callback != this);
this.property = null;
} }
} }
} }

View File

@ -600,13 +600,15 @@ class list extends IgniteTemplate {
} }
onListChanged(oldValue, newValue) { onListChanged(oldValue, newValue) {
console.log("On List changed:", newValue);
this.list = newValue; this.list = newValue;
IgniteRenderingContext.enter(); IgniteRenderingContext.enter();
try { try {
this.construct(null); //The list changed, reconstruct this template. this.construct(null); //The list changed, reconstruct this template.
} catch { } } catch (error) { console.error(error); }
IgniteRenderingContext.leave(); IgniteRenderingContext.leave();
} }
@ -616,7 +618,12 @@ class list extends IgniteTemplate {
try { try {
var template = this.forEach(this.list[this.list.length - 1]); var template = this.forEach(this.list[this.list.length - 1]);
template.construct(this.element.parentElement, this.elements[this.elements.length - 1].nextSibling);
if (this.elements.length > 0) {
template.construct(this.element.parentElement, this.elements[this.elements.length - 1].nextSibling);
} else {
template.construct(this.element.parentElement, null);
}
this.children.push(template); this.children.push(template);
this.elements.push(template.element); this.elements.push(template.element);

View File

@ -26,7 +26,7 @@ class MainApp extends IgniteElement {
.class(this.sheetClass) .class(this.sheetClass)
.child(new html(`<h3>Im a child for sheet!</h3>`)) .child(new html(`<h3>Im a child for sheet!</h3>`))
) )
.child(new Sheet()) .child(new Sheet().property("items", this.items))
.child( .child(
new div( new div(
new html(`<h1>This is a slot!`), new html(`<h1>This is a slot!`),