Added IgniteObject to support full end to end properties.

This commit is contained in:
MattMo 2021-05-25 22:27:23 -07:00
parent 67b4bd60a0
commit d6142ea250
3 changed files with 104 additions and 60 deletions

View File

@ -1,4 +1,4 @@
import { IgniteProperty, IgniteCallback } from './ignite-html.js'; import { IgniteProperty, IgniteCallback, IgniteRenderingContext } from './ignite-html.js';
import { IgniteTemplate } from './ignite-template.js'; import { IgniteTemplate } from './ignite-template.js';
/** /**
@ -139,12 +139,8 @@ class IgniteElement extends HTMLElement {
*/ */
createVariables() { createVariables() {
var vars = this.variables; var vars = this.variables;
if (vars != null) { if (vars != null) {
var keys = Object.keys(vars); Object.keys(vars).forEach(name => this[name] = vars[name]);
for (var i = 0; i < keys.length; i++) {
this[keys[i]] = vars[keys[i]];
}
} }
} }
@ -154,15 +150,23 @@ class IgniteElement extends HTMLElement {
*/ */
resetVariables() { resetVariables() {
var vars = this.variables; var vars = this.variables;
if (vars != null) { if (vars != null) {
var keys = Object.keys(vars); Object.keys(vars).forEach(name => this[name] = vars[name]);
for (var i = 0; i < keys.length; i++) {
this[keys[i]] = vars[keys[i]];
}
} }
} }
/**
* Gets all the property values from this element and returns it.
*/
getProperties() {
var ret = {};
var props = this.properties;
IgniteRenderingContext.push();
Object.keys(props).forEach(name => ret[name] = this[name]);
IgniteRenderingContext.pop();
return ret;
}
/** /**
* Creates the getters/setters for properties in this * Creates the getters/setters for properties in this
* ignite element and initializes everything. * ignite element and initializes everything.
@ -170,37 +174,14 @@ class IgniteElement extends HTMLElement {
*/ */
createProperties() { createProperties() {
var props = this.properties; var props = this.properties;
if (props != null) { if (props != null) {
var keys = Object.keys(props); Object.keys(props).forEach(name => {
for (var i = 0; i < keys.length; i++) { var prop = (props[name] instanceof IgniteProperty ? props[name] : new IgniteProperty(props[name]));
let propValue = props[keys[i]]; Object.defineProperty(this, name, {
let propName = keys[i]; get: () => { return (IgniteRenderingContext.rendering ? prop : prop.value); },
set: (value) => { prop.value = value; }
//Create a new property, if the propValue is a property, use that instead.
var prop = null;
if (propValue instanceof IgniteProperty) {
prop = propValue;
} else {
prop = new IgniteProperty(propValue);
}
this[`_${propName}`] = prop;
Object.defineProperty(this, propName, {
get: () => {
if (IgniteRenderingContext.rendering == false) {
return this[`_${propName}`].value;
} else {
return this[`_${propName}`];
}
},
set: (value) => {
this[`_${propName}`].value = value;
}
}); });
} });
} }
} }
@ -210,12 +191,8 @@ class IgniteElement extends HTMLElement {
*/ */
resetProperties() { resetProperties() {
var props = this.properties; var props = this.properties;
if (props != null) { if (props != null) {
var keys = Object.keys(props); Object.keys(props).forEach(name => this[name] = props[name]);
for (var i = 0; i < keys.length; i++) {
this[keys[i]] = props[keys[i]];
}
} }
} }

View File

@ -11,11 +11,11 @@ class IgniteProperty {
this.onShiftCallbacks = []; this.onShiftCallbacks = [];
this.onUnshiftCallbacks = []; this.onUnshiftCallbacks = [];
this.onSpliceCallbacks = []; this.onSpliceCallbacks = [];
this.arrayCallbacks = []; this.arrayCallbacks = [];
this.reflected = []; this.reflected = [];
this._value = val; this._value = val;
this.ignoreValueChange = false; this.ignoreValueChange = false;
this.name = null;
//If we were passed options, add any callbacks needed. //If we were passed options, add any callbacks needed.
if (options) { if (options) {
@ -43,7 +43,7 @@ class IgniteProperty {
this.attachOnSplice(options.onSplice); this.attachOnSplice(options.onSplice);
} }
} }
//Attempt to patch the value if it's a list. //Attempt to patch the value if it's a list.
this.patchArray(); this.patchArray();
} }
@ -361,6 +361,65 @@ Array.prototype.getOldPropertyValues = function (property, oldValue) {
return ret; return ret;
} }
/**
* 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, {
get: () => { return (IgniteRenderingContext.rendering ? prop : prop.value); },
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, {
get: () => { return (IgniteRenderingContext.rendering ? prop : prop.value); },
set: (value) => { prop.value = value; }
});
}
});
}
/**
* 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;
}
}
}
/** /**
* The outline of a ignite callback that can be invoked and disconnected * The outline of a ignite callback that can be invoked and disconnected
* to help maintain and cleanup callbacks. * to help maintain and cleanup callbacks.
@ -457,9 +516,11 @@ class IgniteRenderingContext {
} }
window.IgniteRenderingContext = IgniteRenderingContext; window.IgniteRenderingContext = IgniteRenderingContext;
window.IgniteObject = IgniteObject;
export { export {
IgniteProperty, IgniteProperty,
IgniteObject,
IgniteRenderingContext, IgniteRenderingContext,
IgniteCallback IgniteCallback
}; };

View File

@ -1,4 +1,4 @@
import { IgniteProperty } from './ignite-html.js'; import { IgniteObject, IgniteProperty, IgniteRenderingContext } from './ignite-html.js';
import { IgniteElement } from './ignite-element.js'; import { IgniteElement } from './ignite-element.js';
/** /**
@ -318,28 +318,27 @@ class IgniteTemplate {
/** /**
* Adds a set of properties from an object to be added to this template once it's constructed. * Adds a set of properties from an object to be added to this template once it's constructed.
* @param {Object} props The object value that property names/values will be pulled from. * @param {Object|IgniteObject} props The object value that property names/values will be pulled from.
* @returns This ignite template so function calls can be chained. * @returns This ignite template so function calls can be chained.
*/ */
properties(props) { properties(props) {
IgniteRenderingContext.push();
//Make sure we have a valid props. //Make sure we have a valid props.
if (props == null || props == undefined) { if (props == null || props == undefined) {
return; return;
} }
else if (!(typeof props === 'object')) {
if (!(typeof props === 'object')) {
throw `Cannot set properties with a non object set of properties: ${props}`; throw `Cannot set properties with a non object set of properties: ${props}`;
} }
var propNames = Object.keys(props); if (props instanceof IgniteObject) {
props.update();
propNames.forEach(name => { Object.getOwnPropertyNames(props).forEach(name => this.property(name, props[name], true));
this.property(name, props[name], false, null); }
}); else {
Object.keys(props).forEach(name => this.property(name, props[name], false, null));
}
IgniteRenderingContext.pop();
return this; return this;
} }
@ -907,7 +906,8 @@ class IgniteTemplate {
for (var i = 0; i < keys.length; i++) { for (var i = 0; i < keys.length; i++) {
this.element[keys[i]] = this._properties[keys[i]].value; this.element[keys[i]] = this._properties[keys[i]].value;
if (this._properties[keys[i]].reflect != null) { //Reflect the property if it can be on the element.
if (this._properties[keys[i]].reflect != null && this.element[keys[i]] instanceof IgniteProperty) {
this.element[keys[i]].reflected.push(this._properties[keys[i]].reflect); this.element[keys[i]].reflected.push(this._properties[keys[i]].reflect);
} }
} }
@ -1136,7 +1136,13 @@ class IgniteTemplate {
if (this.element) { if (this.element) {
//Use the set value function and don't reflect the change because it came from above which //Use the set value function and don't reflect the change because it came from above which
//would be the reflected property, this reduces some functions being called twice that don't need to be. //would be the reflected property, this reduces some functions being called twice that don't need to be.
this.element[`_${propertyName}`].setValue(newValue, false); IgniteRenderingContext.enter();
if (this.element[propertyName] instanceof IgniteProperty) {
this.element[propertyName].setValue(newValue, false);
} else {
this.element[propertyName] = newValue;
}
IgniteRenderingContext.leave();
} }
this._properties[propertyName].value = newValue; this._properties[propertyName].value = newValue;