2020-08-21 21:42:10 -07:00
|
|
|
/**
|
|
|
|
* The outline of a ignite property which is a managed property that
|
|
|
|
* can be used to invoke call back functions when the value of the property changes.
|
2020-08-30 21:55:21 -07:00
|
|
|
* @ignore
|
2020-08-21 21:42:10 -07:00
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
class IgniteProperty {
|
2023-04-17 12:09:21 -07:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {Any} val Starting value of this property.
|
|
|
|
* @param {{ onChange, onPush, onPop, onShift, onUnshift, onSplice }} options
|
|
|
|
*/
|
2021-03-28 20:34:35 -07:00
|
|
|
constructor(val, options = null) {
|
2020-08-22 16:15:45 -07:00
|
|
|
this.onChangeCallbacks = [];
|
|
|
|
this.onPushCallbacks = [];
|
|
|
|
this.onPopCallbacks = [];
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onShiftCallbacks = [];
|
|
|
|
this.onUnshiftCallbacks = [];
|
|
|
|
this.onSpliceCallbacks = [];
|
2020-08-23 11:50:31 -07:00
|
|
|
this.arrayCallbacks = [];
|
|
|
|
this.reflected = [];
|
2020-08-22 16:15:45 -07:00
|
|
|
this._value = val;
|
2020-08-23 11:50:31 -07:00
|
|
|
this.ignoreValueChange = false;
|
2021-05-25 22:27:23 -07:00
|
|
|
this.name = null;
|
2020-08-22 16:15:45 -07:00
|
|
|
|
2021-03-28 20:34:35 -07:00
|
|
|
//If we were passed options, add any callbacks needed.
|
|
|
|
if (options) {
|
|
|
|
if (options.onChange) {
|
|
|
|
this.attachOnChange(options.onChange);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.onPush) {
|
|
|
|
this.attachOnPush(options.onPush);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.onPop) {
|
|
|
|
this.attachOnPop(options.onPop);
|
|
|
|
}
|
2020-09-25 09:36:39 -07:00
|
|
|
|
2021-03-28 20:34:35 -07:00
|
|
|
if (options.onShift) {
|
|
|
|
this.attachOnShift(options.onShift);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.onUnshift) {
|
|
|
|
this.attachOnUnshift(options.onUnshift);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.onSplice) {
|
|
|
|
this.attachOnSplice(options.onSplice);
|
|
|
|
}
|
|
|
|
}
|
2021-05-25 22:27:23 -07:00
|
|
|
|
2020-08-22 16:15:45 -07:00
|
|
|
//Attempt to patch the value if it's a list.
|
2020-08-23 11:50:31 -07:00
|
|
|
this.patchArray();
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
2022-09-13 21:10:42 -07:00
|
|
|
/**
|
|
|
|
* Creates a new ignite property on the target object that responds to rendering states.
|
|
|
|
* @param {Any} target
|
|
|
|
* @param {String} name
|
|
|
|
* @param {Any} value
|
|
|
|
* @returns {Any} The target passed to the function.
|
|
|
|
*/
|
|
|
|
static create(target, name, value) {
|
|
|
|
var property = new IgniteProperty(value);
|
|
|
|
|
|
|
|
Object.defineProperty(target, name, {
|
|
|
|
get: () => { return (IgniteRendering.rendering ? property : property.value); },
|
|
|
|
|
|
|
|
set: (value) => { property.value = value; }
|
|
|
|
});
|
|
|
|
|
|
|
|
return target;
|
|
|
|
}
|
|
|
|
|
2021-12-28 22:32:19 -08:00
|
|
|
/**
|
|
|
|
* Gets the value of this IgniteProperty.
|
|
|
|
* @returns {Any} The value of this IgniteProperty.
|
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
get value() {
|
|
|
|
return this._value;
|
|
|
|
}
|
|
|
|
|
2021-12-28 22:32:19 -08:00
|
|
|
/**
|
|
|
|
* Sets the value of this IgniteProperty, this will reflect the value to any listeners on this property.
|
|
|
|
* @param {Any} val The value to set.
|
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
set value(val) {
|
2020-08-23 11:50:31 -07:00
|
|
|
this.setValue(val, true);
|
|
|
|
}
|
|
|
|
|
2021-12-28 22:32:19 -08:00
|
|
|
/**
|
|
|
|
* Sets the value of this IngiteProperty with control of reflection.
|
|
|
|
* @param {Any} val The value to set for this IgniteProperty.
|
|
|
|
* @param {Boolean} reflect If true, this will reflect the value onto any listeners on this property.
|
|
|
|
*/
|
2020-08-23 11:50:31 -07:00
|
|
|
setValue(val, reflect) {
|
|
|
|
//If the ignore value change flag is set exit.
|
|
|
|
if (this.ignoreValueChange) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-02-12 18:53:09 -08:00
|
|
|
//If the current value is the same as the old value don't do anything.
|
|
|
|
if (this._value === val) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-10-26 15:30:11 -07:00
|
|
|
//Get the old value
|
2020-07-28 09:04:04 -07:00
|
|
|
var old = this._value;
|
2020-10-26 15:30:11 -07:00
|
|
|
|
|
|
|
//Based on the old value, see if we need to convert the new value to match the original type.
|
|
|
|
if (typeof old === typeof true) {
|
|
|
|
val = val != null && val != undefined ? val.toString().toLowerCase().trim() : val;
|
|
|
|
val = val == "true" || val == "1" || val == "yes" || val == "t" || val == "y";
|
|
|
|
} else if (typeof old === typeof 0) {
|
|
|
|
val = Number(val != null && val != undefined ? val.toString().trim() : "0");
|
|
|
|
val = isNaN(val) ? 0 : val;
|
|
|
|
} else if (typeof old === typeof "") {
|
|
|
|
val = val != null && val != undefined ? val.toString() : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
//Set the new value
|
2020-07-28 09:04:04 -07:00
|
|
|
this._value = val;
|
|
|
|
|
2020-08-23 11:50:31 -07:00
|
|
|
//Attempt to patch the value if it's an array.
|
|
|
|
this.patchArray();
|
2020-08-22 16:15:45 -07:00
|
|
|
|
2020-12-09 20:41:11 -08:00
|
|
|
//Invoke any callbacks letting them know the value changed.
|
2021-02-12 18:53:09 -08:00
|
|
|
this.invokeOnChange(old, val, reflect);
|
2020-08-22 16:15:45 -07:00
|
|
|
}
|
|
|
|
|
2020-08-23 11:50:31 -07:00
|
|
|
patchArray() {
|
|
|
|
//Disconnect any existing array callbacks
|
|
|
|
if (this.arrayCallbacks.length > 0) {
|
|
|
|
this.arrayCallbacks.forEach(callback => callback.disconnect());
|
|
|
|
this.arrayCallbacks = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
//If our value is an array and it hasn't been patched, then patch it.
|
2020-08-22 18:11:37 -07:00
|
|
|
if (Array.isArray(this._value) && this._value.onPushCallbacks == undefined) {
|
|
|
|
this._value.onPushCallbacks = [];
|
|
|
|
this._value.onPopCallbacks = [];
|
2020-10-07 08:30:03 -07:00
|
|
|
this._value.onShiftCallbacks = [];
|
|
|
|
this._value.onUnshiftCallbacks = [];
|
|
|
|
this._value.onSpliceCallbacks = [];
|
2020-08-22 16:15:45 -07:00
|
|
|
|
|
|
|
this._value.push = function () {
|
|
|
|
var len = Array.prototype.push.apply(this, arguments);
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onPushCallbacks.forEach(callback => callback.invoke(Array.from(arguments)));
|
2020-08-22 16:15:45 -07:00
|
|
|
return len;
|
|
|
|
};
|
|
|
|
|
|
|
|
this._value.pop = function () {
|
|
|
|
var len = Array.prototype.pop.apply(this, arguments);
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onPopCallbacks.forEach(callback => callback.invoke());
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._value.unshift = function () {
|
|
|
|
var len = Array.prototype.unshift.apply(this, arguments);
|
|
|
|
this.onUnshiftCallbacks.forEach(callback => callback.invoke(Array.from(arguments)));
|
2020-08-22 16:15:45 -07:00
|
|
|
return len;
|
|
|
|
}
|
2020-08-22 18:11:37 -07:00
|
|
|
|
2020-10-07 08:30:03 -07:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2020-08-22 18:11:37 -07:00
|
|
|
this._value.attachOnPush = function (func) {
|
2020-10-07 08:30:03 -07:00
|
|
|
var callback = new IgniteCallback(func, detach => this.detachOnPush(detach));
|
2020-08-22 18:11:37 -07:00
|
|
|
this.onPushCallbacks.push(callback);
|
|
|
|
return callback;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._value.attachOnPop = function (func) {
|
2020-10-07 08:30:03 -07:00
|
|
|
var callback = new IgniteCallback(func, detach => this.detachOnPop(detach));
|
2020-08-22 18:11:37 -07:00
|
|
|
this.onPopCallbacks.push(callback);
|
|
|
|
return callback;
|
|
|
|
}
|
|
|
|
|
2020-10-07 08:30:03 -07:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2021-02-12 18:53:09 -08:00
|
|
|
this._value.attachOnSplice = function (func) {
|
|
|
|
var callback = new IgniteCallback(func, detach => this.detachOnSplice(detach));
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onSpliceCallbacks.push(callback);
|
|
|
|
return callback;
|
|
|
|
}
|
|
|
|
|
2020-08-22 18:11:37 -07:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2020-10-07 08:30:03 -07:00
|
|
|
this._value.detachOnUnshift = function (callback) {
|
|
|
|
this.onUnshiftCallbacks = this.onUnshiftCallbacks.filter(unshift => unshift != callback);
|
|
|
|
}
|
|
|
|
|
|
|
|
this._value.detachOnShift = function (callback) {
|
|
|
|
this.onShiftCallbacks = this.onShiftCallbacks.filter(shift => shift != callback);
|
|
|
|
}
|
|
|
|
|
2021-02-12 18:53:09 -08:00
|
|
|
this._value.detachOnSplice = function (callback) {
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onSpliceCallbacks = this.onSpliceCallbacks.filter(slice => slice != callback);
|
|
|
|
}
|
|
|
|
}
|
2020-10-20 13:18:26 -07:00
|
|
|
|
2020-10-07 08:30:03 -07:00
|
|
|
if (Array.isArray(this._value) && this._value.onPushCallbacks) {
|
2020-08-23 11:50:31 -07:00
|
|
|
//This array has already been patched but attach to it so we get callbacks.
|
2020-10-07 08:30:03 -07:00
|
|
|
this.arrayCallbacks.push(this._value.attachOnPush((items) => this.invokeOnPush(items)));
|
2020-08-23 11:50:31 -07:00
|
|
|
this.arrayCallbacks.push(this._value.attachOnPop(() => this.invokeOnPop()));
|
2020-10-07 08:30:03 -07:00
|
|
|
this.arrayCallbacks.push(this._value.attachOnShift(() => this.invokeOnShift()));
|
|
|
|
this.arrayCallbacks.push(this._value.attachOnUnshift((items) => this.invokeOnUnshift(items)));
|
2021-02-12 18:53:09 -08:00
|
|
|
this.arrayCallbacks.push(this._value.attachOnSplice((start, deleteCount, items) => this.invokeOnSplice(start, deleteCount, items)));
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
}
|
2020-07-28 22:23:49 -07:00
|
|
|
|
2021-02-12 18:53:09 -08:00
|
|
|
reflect() {
|
|
|
|
this.ignoreValueChange = true; //Ignore changes incase we are connected to any reflected properties.
|
|
|
|
this.reflected.forEach(reflect => reflect instanceof Function ? reflect(this._value) : (reflect.value = this._value));
|
|
|
|
this.ignoreValueChange = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
invokeOnChange(oldValue, newValue, reflect = true) {
|
2020-12-09 20:41:11 -08:00
|
|
|
//Enter a new rendering context since this event may contain code that expects a new context.
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.push();
|
2021-02-12 18:53:09 -08:00
|
|
|
if (reflect) {
|
|
|
|
this.reflect();
|
|
|
|
}
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onChangeCallbacks.forEach(callback => callback.invoke(oldValue, newValue));
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.pop();
|
2020-08-22 16:15:45 -07:00
|
|
|
}
|
|
|
|
|
2021-02-12 18:53:09 -08:00
|
|
|
invokeOnPush(items, reflect = true) {
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.push();
|
2021-02-12 18:53:09 -08:00
|
|
|
if (reflect) {
|
|
|
|
this.reflect();
|
|
|
|
}
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onPushCallbacks.forEach(callback => callback.invoke(this._value, items));
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.pop();
|
2020-08-22 16:15:45 -07:00
|
|
|
}
|
|
|
|
|
2021-02-12 18:53:09 -08:00
|
|
|
invokeOnPop(reflect = true) {
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.push();
|
2021-02-12 18:53:09 -08:00
|
|
|
if (reflect) {
|
|
|
|
this.reflect();
|
|
|
|
}
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onPopCallbacks.forEach(callback => callback.invoke(this._value));
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.pop();
|
2020-10-07 08:30:03 -07:00
|
|
|
}
|
|
|
|
|
2021-02-12 18:53:09 -08:00
|
|
|
invokeOnShift(reflect = true) {
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.push();
|
2021-02-12 18:53:09 -08:00
|
|
|
if (reflect) {
|
|
|
|
this.reflect();
|
|
|
|
}
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onShiftCallbacks.forEach(callback => callback.invoke(this._value));
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.pop();
|
2020-10-07 08:30:03 -07:00
|
|
|
}
|
|
|
|
|
2021-02-12 18:53:09 -08:00
|
|
|
invokeOnUnshift(items, reflect = true) {
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.push();
|
2021-02-12 18:53:09 -08:00
|
|
|
if (reflect) {
|
|
|
|
this.reflect();
|
|
|
|
}
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onUnshiftCallbacks.forEach(callback => callback.invoke(this._value, items));
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.pop();
|
2020-10-07 08:30:03 -07:00
|
|
|
}
|
|
|
|
|
2021-02-12 18:53:09 -08:00
|
|
|
invokeOnSplice(start, deleteCount, items, reflect = true) {
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.push();
|
2021-02-12 18:53:09 -08:00
|
|
|
if (reflect) {
|
|
|
|
this.reflect();
|
|
|
|
}
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onSpliceCallbacks.forEach(callback => callback.invoke(this._value, start, deleteCount, items));
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.pop();
|
2020-08-22 16:15:45 -07:00
|
|
|
}
|
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Attaches a callback function to be invoked when this property value changes.
|
|
|
|
* @param {Function(oldvalue, newValue)} onChange Callback function to be invoked.
|
|
|
|
* @returns {IgniteCallback} A new ignite callback.
|
|
|
|
*/
|
2020-08-22 16:15:45 -07:00
|
|
|
attachOnChange(onChange) {
|
2020-10-07 08:30:03 -07:00
|
|
|
var callback = new IgniteCallback(onChange, detach => this.detachOnChange(detach));
|
2020-08-22 16:15:45 -07:00
|
|
|
this.onChangeCallbacks.push(callback);
|
|
|
|
return callback;
|
|
|
|
}
|
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Attaches a callback function to be invoked when a set of items are pushed to this properties value.
|
|
|
|
* @param {Function(array, items)} onPush Callback function to be invoked.
|
|
|
|
* @returns {IgniteCallback} A new ignite callback.
|
|
|
|
*/
|
2020-08-22 16:15:45 -07:00
|
|
|
attachOnPush(onPush) {
|
2020-10-07 08:30:03 -07:00
|
|
|
var callback = new IgniteCallback(onPush, detach => this.detachOnPush(detach));
|
2020-08-22 16:15:45 -07:00
|
|
|
this.onPushCallbacks.push(callback);
|
|
|
|
return callback;
|
|
|
|
}
|
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Attaches a callback function to be invoked when an item is popped from this properties value.
|
|
|
|
* @param {Function} onPop Callback function to be invoked.
|
|
|
|
* @returns {IgniteCallback} A new ignite callback.
|
|
|
|
*/
|
2020-08-22 16:15:45 -07:00
|
|
|
attachOnPop(onPop) {
|
2020-10-07 08:30:03 -07:00
|
|
|
var callback = new IgniteCallback(onPop, detach => this.detachOnPop(detach));
|
2020-08-22 16:15:45 -07:00
|
|
|
this.onPopCallbacks.push(callback);
|
2020-07-28 22:23:49 -07:00
|
|
|
return callback;
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Attaches a callback function to be invoked when items are shifted from this properties value.
|
|
|
|
* @param {Function} onShift Callback function to be invoked.
|
|
|
|
* @returns {IgniteCallback} A new ignite callback.
|
|
|
|
*/
|
2020-10-07 08:30:03 -07:00
|
|
|
attachOnShift(onShift) {
|
|
|
|
var callback = new IgniteCallback(onShift, detach => this.detachOnShift(detach));
|
|
|
|
this.onShiftCallbacks.push(callback);
|
|
|
|
return callback;
|
|
|
|
}
|
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Attaches a callback function to be invoked when items are unshifted to this properties value.
|
|
|
|
* @param {Function(array, items)} onUnshift Callback function to be invoked.
|
|
|
|
* @returns {IgniteCallback} A new ignite callback.
|
|
|
|
*/
|
2020-10-07 08:30:03 -07:00
|
|
|
attachOnUnshift(onUnshift) {
|
|
|
|
var callback = new IgniteCallback(onUnshift, detach => this.detachOnUnshift(detach));
|
|
|
|
this.onUnshiftCallbacks.push(callback);
|
|
|
|
return callback;
|
|
|
|
}
|
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Attaches a callback function to be invoked when items are spliced on this properties value.
|
|
|
|
* @param {Function(array, start, deleteCount, items)} onSplice Callback function to be invoked.
|
|
|
|
* @returns {IgniteCallback} A new ignite callback.
|
|
|
|
*/
|
2020-10-07 08:30:03 -07:00
|
|
|
attachOnSplice(onSplice) {
|
2021-02-12 18:53:09 -08:00
|
|
|
var callback = new IgniteCallback(onSplice, detach => this.detachOnSplice(detach));
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onSpliceCallbacks.push(callback);
|
|
|
|
return callback;
|
|
|
|
}
|
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Removes an ignite callback for the OnChange event.
|
|
|
|
* @param {IgniteCallback} callback
|
|
|
|
*/
|
2020-08-22 18:11:37 -07:00
|
|
|
detachOnChange(callback) {
|
|
|
|
this.onChangeCallbacks = this.onChangeCallbacks.filter(change => change != callback);
|
2020-08-22 16:15:45 -07:00
|
|
|
}
|
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Removes an ignite callback for the OnPush event.
|
|
|
|
* @param {Ignitec} callback
|
|
|
|
*/
|
2020-08-22 18:11:37 -07:00
|
|
|
detachOnPush(callback) {
|
|
|
|
this.onPushCallbacks = this.onPushCallbacks.filter(push => push != callback);
|
2020-08-22 16:15:45 -07:00
|
|
|
}
|
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Removes an ignite callback for the OnPop event.
|
|
|
|
* @param {IgniteCallback} callback
|
|
|
|
*/
|
2020-08-22 18:11:37 -07:00
|
|
|
detachOnPop(callback) {
|
|
|
|
this.onPopCallbacks = this.onPopCallbacks.filter(pop => pop != callback);
|
2020-08-22 16:15:45 -07:00
|
|
|
}
|
2020-10-07 08:30:03 -07:00
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Removes an ignite callback for the OnShift event.
|
|
|
|
* @param {IgniteCallback} callback
|
|
|
|
*/
|
2020-10-07 08:30:03 -07:00
|
|
|
detachOnShift(callback) {
|
|
|
|
this.onShiftCallbacks = this.onShiftCallbacks.filter(shift => shift != callback);
|
|
|
|
}
|
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Removes an ignite callback for the OnUnshift event.
|
|
|
|
* @param {IgniteCallback} callback
|
|
|
|
*/
|
2020-10-07 08:30:03 -07:00
|
|
|
detachOnUnshift(callback) {
|
|
|
|
this.onUnshiftCallbacks = this.onUnshiftCallbacks.filter(unshift => unshift != callback);
|
|
|
|
}
|
|
|
|
|
2023-04-18 15:38:49 -07:00
|
|
|
/**
|
|
|
|
* Removes an ignite callback for the OnSplice event.
|
|
|
|
* @param {IgniteCallback} callback
|
|
|
|
*/
|
2021-02-12 18:53:09 -08:00
|
|
|
detachOnSplice(callback) {
|
2020-10-07 08:30:03 -07:00
|
|
|
this.onSpliceCallbacks = this.onSpliceCallbacks.filter(slice => slice != callback);
|
|
|
|
}
|
2020-08-22 16:15:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-08-22 18:11:37 -07:00
|
|
|
* Return the value of the property if we try to convert
|
|
|
|
* the property to a string.
|
2020-08-22 16:15:45 -07:00
|
|
|
*/
|
2020-08-22 18:11:37 -07:00
|
|
|
IgniteProperty.prototype.toString = function () {
|
2022-10-24 08:39:41 -07:00
|
|
|
if (this.value) {
|
|
|
|
return this.value.toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
2020-08-22 16:15:45 -07:00
|
|
|
}
|
|
|
|
|
2020-09-25 09:36:39 -07:00
|
|
|
/**
|
|
|
|
* Add a prototype to help get property values from an array
|
|
|
|
*/
|
|
|
|
Array.prototype.getPropertyValues = function () {
|
|
|
|
var ret = [];
|
2020-10-20 13:18:26 -07:00
|
|
|
this.forEach(prop => {
|
|
|
|
if (prop instanceof IgniteProperty) {
|
|
|
|
ret.push(prop.value);
|
|
|
|
} else {
|
|
|
|
ret.push(prop);
|
|
|
|
}
|
|
|
|
});
|
2020-09-25 09:36:39 -07:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add a prototype to help get old property values from an array
|
|
|
|
*/
|
|
|
|
Array.prototype.getOldPropertyValues = function (property, oldValue) {
|
|
|
|
var ret = [];
|
|
|
|
this.forEach(prop => {
|
2023-04-15 08:27:11 -07:00
|
|
|
if (prop === property) {
|
2020-09-25 09:36:39 -07:00
|
|
|
ret.push(oldValue);
|
|
|
|
} else {
|
2020-10-20 13:18:26 -07:00
|
|
|
if (prop instanceof IgniteProperty) {
|
|
|
|
ret.push(prop.value);
|
|
|
|
} else {
|
|
|
|
ret.push(prop);
|
|
|
|
}
|
2020-09-25 09:36:39 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-05-25 22:27:23 -07:00
|
|
|
/**
|
|
|
|
* The outline of an IgniteObject which contains IgniteProperties.
|
|
|
|
*/
|
|
|
|
class IgniteObject {
|
|
|
|
/**
|
|
|
|
* Creates a new IgniteObject from an object and returns it.
|
|
|
|
* @param {Any} obj The object to create an IgniteObject out of.
|
|
|
|
* @returns {IgniteObject} The ignite object created.
|
|
|
|
*/
|
|
|
|
constructor(obj) {
|
|
|
|
//Only do this if the object is not an ignite object already.
|
|
|
|
if (!(obj instanceof IgniteObject)) {
|
|
|
|
Object.keys(obj).forEach(name => {
|
|
|
|
var prop = new IgniteProperty(obj[name]);
|
|
|
|
Object.defineProperty(this, name, {
|
2022-08-26 08:34:20 -07:00
|
|
|
get: () => { return (IgniteRendering.rendering ? prop : prop.value); },
|
2021-05-25 22:27:23 -07:00
|
|
|
set: (value) => { prop.value = value; }
|
|
|
|
});
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks this IgniteObject for any properties not converted and automatically
|
|
|
|
* converts them.
|
|
|
|
*/
|
|
|
|
update() {
|
|
|
|
Object.keys(this).forEach(name => {
|
|
|
|
if (!(this[name] instanceof IgniteProperty)) {
|
|
|
|
var prop = new IgniteProperty(this[name]);
|
|
|
|
delete this[name];
|
|
|
|
Object.defineProperty(this, name, {
|
2022-08-26 08:34:20 -07:00
|
|
|
get: () => { return (IgniteRendering.rendering ? prop : prop.value); },
|
2021-05-25 22:27:23 -07:00
|
|
|
set: (value) => { prop.value = value; }
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-09-22 09:43:18 -07:00
|
|
|
/**
|
|
|
|
* Sets the values of properties on this ignite object.
|
|
|
|
* @param {Object|IgniteObject} props The properties to set onto this ignite object.
|
|
|
|
*/
|
|
|
|
setProperties(props) {
|
|
|
|
if (props == null || props == undefined) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (props instanceof IgniteObject) {
|
|
|
|
props.update();
|
|
|
|
Object.getOwnPropertyNames(props).forEach(name => this[name] = props[name]);
|
|
|
|
} else {
|
|
|
|
Object.keys(props).forEach(name => this[name] = props[name]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-25 22:27:23 -07:00
|
|
|
/**
|
|
|
|
* Creates an IgniteObject or Array of IgniteObjects and returns it.
|
|
|
|
* @param {Object|Array} obj The object to create an IgniteObject from.
|
|
|
|
* @returns {IgniteObject} The created IgniteObject.
|
|
|
|
*/
|
|
|
|
static create(obj) {
|
|
|
|
if (obj instanceof Array) {
|
|
|
|
var converted = [];
|
|
|
|
obj.forEach(item => converted.push(new IgniteObject(item)));
|
|
|
|
return converted;
|
|
|
|
} else if (!(obj instanceof IgniteObject)) {
|
|
|
|
return new IgniteObject(obj);
|
|
|
|
} else {
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-22 16:15:45 -07:00
|
|
|
/**
|
2020-08-22 18:11:37 -07:00
|
|
|
* The outline of a ignite callback that can be invoked and disconnected
|
|
|
|
* to help maintain and cleanup callbacks.
|
2020-08-30 21:55:21 -07:00
|
|
|
* @ignore
|
2020-08-21 21:42:10 -07:00
|
|
|
*/
|
2020-08-22 18:11:37 -07:00
|
|
|
class IgniteCallback {
|
2022-05-23 09:12:23 -07:00
|
|
|
/**
|
|
|
|
* Creates a new IgniteCallback with a callback and optional detach callback.
|
|
|
|
* @param {Function(...)} callback The callback function to invoke when the callback is invoked.
|
|
|
|
* @param {Function(IgniteCallback)} detach The detach function that is called when the callback is disconnected.
|
|
|
|
*/
|
|
|
|
constructor(callback, detach = null) {
|
2020-07-28 22:23:49 -07:00
|
|
|
this.callback = callback;
|
2020-08-22 18:11:37 -07:00
|
|
|
this.detach = detach;
|
2020-07-28 22:23:49 -07:00
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
|
2022-10-24 08:39:41 -07:00
|
|
|
/**
|
|
|
|
* Invokes this callback.
|
|
|
|
* @param {...any} params Any params to pass to the target function.
|
|
|
|
*/
|
2021-12-06 09:32:31 -08:00
|
|
|
async invoke(...params) {
|
2020-07-28 22:23:49 -07:00
|
|
|
if (this.callback) {
|
2021-12-06 09:32:31 -08:00
|
|
|
await this.callback(...params);
|
2020-07-28 22:23:49 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-24 08:39:41 -07:00
|
|
|
/**
|
|
|
|
* Invokes the detach callback for this callback if there is one.
|
|
|
|
*/
|
2020-07-28 22:23:49 -07:00
|
|
|
disconnect() {
|
|
|
|
this.callback = null;
|
|
|
|
|
2020-08-22 18:11:37 -07:00
|
|
|
if (this.detach) {
|
|
|
|
this.detach(this);
|
2020-07-28 22:23:49 -07:00
|
|
|
}
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
2020-08-21 21:42:10 -07:00
|
|
|
/**
|
|
|
|
* The outline of a simple rendering context which allows us to
|
|
|
|
* know if we are currently rendering anything ignite related. This works
|
|
|
|
* because Javascript is single threaded, if events could break the current execution
|
|
|
|
* this would fail. But it's safe since events cannot do that.
|
2020-08-30 21:55:21 -07:00
|
|
|
* @ignore
|
2020-08-21 21:42:10 -07:00
|
|
|
*/
|
2022-08-26 08:34:20 -07:00
|
|
|
class IgniteRendering {
|
2022-10-24 08:39:41 -07:00
|
|
|
/**
|
|
|
|
* Increments the rendering counter. Must call leave() for each enter().
|
|
|
|
*/
|
2020-08-21 21:42:10 -07:00
|
|
|
static enter() {
|
2022-08-26 08:34:20 -07:00
|
|
|
if (!IgniteRendering.RenderCount) {
|
|
|
|
IgniteRendering.RenderCount = 0;
|
2020-08-21 21:42:10 -07:00
|
|
|
}
|
|
|
|
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.RenderCount++;
|
2020-08-21 21:42:10 -07:00
|
|
|
}
|
|
|
|
|
2022-10-24 08:39:41 -07:00
|
|
|
/**
|
|
|
|
* Decrements the rendering counter, if 0, rendering() will return false.
|
|
|
|
*/
|
2020-08-21 21:42:10 -07:00
|
|
|
static leave() {
|
2022-08-26 08:34:20 -07:00
|
|
|
if (IgniteRendering.RenderCount) {
|
|
|
|
IgniteRendering.RenderCount--;
|
2020-08-21 21:42:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-24 08:39:41 -07:00
|
|
|
/**
|
|
|
|
* Saves the current rendering state.
|
|
|
|
*/
|
2020-09-25 09:36:39 -07:00
|
|
|
static push() {
|
2022-08-26 08:34:20 -07:00
|
|
|
if (IgniteRendering.Stack == null) {
|
|
|
|
IgniteRendering.Stack = [];
|
2020-09-25 09:36:39 -07:00
|
|
|
}
|
|
|
|
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.Stack.push(IgniteRendering.RenderCount);
|
|
|
|
IgniteRendering.RenderCount = 0;
|
2020-09-25 09:36:39 -07:00
|
|
|
}
|
|
|
|
|
2022-10-24 08:39:41 -07:00
|
|
|
/**
|
|
|
|
* Restores the last rendering state.
|
|
|
|
*/
|
2020-09-25 09:36:39 -07:00
|
|
|
static pop() {
|
2022-08-26 08:34:20 -07:00
|
|
|
if (IgniteRendering.Stack && IgniteRendering.Stack.length > 0) {
|
|
|
|
IgniteRendering.RenderCount = IgniteRendering.Stack.pop();
|
2020-09-25 09:36:39 -07:00
|
|
|
}
|
|
|
|
}
|
2020-09-28 09:02:57 -07:00
|
|
|
|
2022-10-24 08:39:41 -07:00
|
|
|
/**
|
|
|
|
* Registers a callback function that will be invoked once rendering is complete.
|
|
|
|
* @param {IgniteCallback|Function} callback The callback function to invoke once rendering is complete.
|
|
|
|
*/
|
2020-09-11 07:57:37 -07:00
|
|
|
static ready(callback) {
|
|
|
|
//Setup the callbacks if it's not init'd.
|
2022-08-26 08:34:20 -07:00
|
|
|
if (!IgniteRendering.ReadyCallbacks) {
|
|
|
|
IgniteRendering.ReadyCallbacks = [];
|
2020-09-11 07:57:37 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
//Add this ignite callback.
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.ReadyCallbacks.push(callback);
|
2020-09-11 07:57:37 -07:00
|
|
|
|
|
|
|
//Clear the existing timer if there is one.
|
2022-08-26 08:34:20 -07:00
|
|
|
if (IgniteRendering.ReadyTimer && !IgniteRendering.ReadyTimerRunning) {
|
|
|
|
clearTimeout(IgniteRendering.ReadyTimer);
|
2020-09-11 07:57:37 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
//Set a new timeout, it will only run once all elements are ready because
|
|
|
|
//of the way single threaded timers work.
|
2022-08-26 08:34:20 -07:00
|
|
|
if (!IgniteRendering.ReadyTimerRunning) {
|
|
|
|
IgniteRendering.ReadyTimer = setTimeout(async () => {
|
|
|
|
IgniteRendering.ReadyTimerRunning = true;
|
|
|
|
while (IgniteRendering.ReadyCallbacks.length > 0) {
|
2022-10-24 08:39:41 -07:00
|
|
|
var callback = IgniteRendering.ReadyCallbacks.shift();
|
|
|
|
|
|
|
|
if (callback instanceof IgniteCallback) {
|
|
|
|
callback.invoke();
|
|
|
|
} else if (callback instanceof Function) {
|
|
|
|
callback();
|
|
|
|
}
|
2022-03-26 02:11:27 -07:00
|
|
|
}
|
|
|
|
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering.ReadyCallbacks = [];
|
|
|
|
IgniteRendering.ReadyTimer = null;
|
|
|
|
IgniteRendering.ReadyTimerRunning = false;
|
2022-03-26 02:11:27 -07:00
|
|
|
}, 1);
|
|
|
|
}
|
2020-09-11 07:57:37 -07:00
|
|
|
}
|
|
|
|
|
2022-10-24 08:39:41 -07:00
|
|
|
/**
|
|
|
|
* Returns whether or not we are currently in a rendering state.
|
|
|
|
*/
|
2020-08-21 21:42:10 -07:00
|
|
|
static get rendering() {
|
2022-10-24 08:39:41 -07:00
|
|
|
return IgniteRendering.RenderCount && IgniteRendering.RenderCount > 0;
|
2020-08-21 21:42:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-26 08:34:20 -07:00
|
|
|
/**
|
|
|
|
* Main helper class to access ignite html functions.
|
|
|
|
*/
|
|
|
|
class IgniteHtml {
|
|
|
|
/**
|
|
|
|
* Registers a new element to be rendered when render() is called.
|
|
|
|
* @param {String} name The tag name of the element to register.
|
|
|
|
* @param {Any} definition The class definition of the element.
|
|
|
|
*/
|
|
|
|
static register(name, definition) {
|
|
|
|
if (!IgniteHtml.registered) {
|
|
|
|
IgniteHtml.registered = { };
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IgniteHtml.registered[name]) {
|
|
|
|
IgniteHtml.registered[name] = definition;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Begins rendering all registered elements.
|
|
|
|
*/
|
|
|
|
static render() {
|
|
|
|
if (IgniteHtml.registered) {
|
|
|
|
var elements = Object.keys(IgniteHtml.registered);
|
|
|
|
|
|
|
|
for (var i = 0; i < elements.length; i++) {
|
|
|
|
customElements.define(elements[i], IgniteHtml.registered[elements[i]]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
window.IgniteProperty = IgniteProperty;
|
|
|
|
window.IgniteRendering = IgniteRendering;
|
2021-05-25 22:27:23 -07:00
|
|
|
window.IgniteObject = IgniteObject;
|
2022-03-26 02:11:27 -07:00
|
|
|
window.IgniteCallback = IgniteCallback;
|
2022-08-26 08:34:20 -07:00
|
|
|
window.IgniteHtml = IgniteHtml;
|
2020-08-21 21:42:10 -07:00
|
|
|
|
2020-07-28 09:04:04 -07:00
|
|
|
export {
|
2020-08-23 11:50:31 -07:00
|
|
|
IgniteProperty,
|
2021-05-25 22:27:23 -07:00
|
|
|
IgniteObject,
|
2022-08-26 08:34:20 -07:00
|
|
|
IgniteRendering,
|
|
|
|
IgniteCallback,
|
|
|
|
IgniteHtml
|
2020-07-28 09:04:04 -07:00
|
|
|
};
|