Added a rendering context which replaces the rendered flag on ignite elements so that properties won't return their value if we are rendering. This fixes an issue where lists would get messed up since they reconstruct whenever the list changes.
This commit is contained in:
parent
057960db8a
commit
b0b9a83cc6
@ -1,6 +1,10 @@
|
||||
import { IgniteProperty } from './ignite-html.js';
|
||||
import { IgniteTemplate } from './ignite-template.js';
|
||||
|
||||
/**
|
||||
* The outline of a ignite element that extends the html element
|
||||
* and can be used to create custom components.
|
||||
*/
|
||||
class IgniteElement extends HTMLElement {
|
||||
constructor() {
|
||||
super();
|
||||
@ -9,7 +13,6 @@ class IgniteElement extends HTMLElement {
|
||||
this.template = null;
|
||||
this.elements = [];
|
||||
this.createProperties();
|
||||
this.rendered = false;
|
||||
}
|
||||
|
||||
/**
|
||||
@ -42,7 +45,7 @@ class IgniteElement extends HTMLElement {
|
||||
((propName) => {
|
||||
Object.defineProperty(this, propName, {
|
||||
get: function () {
|
||||
if (this.rendered) {
|
||||
if (IgniteRenderingContext.rendering == false) {
|
||||
return this[`_${propName}`].value;
|
||||
} else {
|
||||
return this[`_${propName}`];
|
||||
@ -98,6 +101,9 @@ class IgniteElement extends HTMLElement {
|
||||
//Add any childNodes we have to the elements list within this
|
||||
this.childNodes.forEach((item) => this.elements.push(item));
|
||||
|
||||
//Enter a rendering context so properties don't expose their values until we are done.
|
||||
IgniteRenderingContext.enter();
|
||||
|
||||
//Make sure the render template is our template, if not, add it as a child.
|
||||
var renderTemplate = this.render();
|
||||
if (renderTemplate !== this.template && renderTemplate) {
|
||||
@ -107,10 +113,14 @@ class IgniteElement extends HTMLElement {
|
||||
}
|
||||
|
||||
//Construct our template.
|
||||
try {
|
||||
this.template.construct(this.parentElement);
|
||||
} catch (error) {
|
||||
console.error(error);
|
||||
}
|
||||
|
||||
//Set rendered to true so that all future properties return their values.
|
||||
this.rendered = true;
|
||||
//Leave the rendering context.
|
||||
IgniteRenderingContext.leave();
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -1,3 +1,7 @@
|
||||
/**
|
||||
* 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.
|
||||
*/
|
||||
class IgniteProperty {
|
||||
constructor() {
|
||||
this.callbacks = [];
|
||||
@ -25,6 +29,18 @@ class IgniteProperty {
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Return the value of the property if we try to convert
|
||||
* the property to a string.
|
||||
*/
|
||||
IgniteProperty.prototype.toString = function () {
|
||||
return this.value.toString();
|
||||
}
|
||||
|
||||
/**
|
||||
* The outline of a ignite property callback that allows
|
||||
* disconnecting of callbacks on demand when they are no longer needed.
|
||||
*/
|
||||
class IgnitePropertyCallback {
|
||||
constructor(property, callback) {
|
||||
this.callback = callback;
|
||||
@ -47,6 +63,38 @@ class IgnitePropertyCallback {
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 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.
|
||||
*/
|
||||
class IgniteRenderingContext {
|
||||
static enter() {
|
||||
if (!IgniteRenderingContext.RenderCount) {
|
||||
IgniteRenderingContext.RenderCount = 0;
|
||||
}
|
||||
|
||||
IgniteRenderingContext.RenderCount++;
|
||||
}
|
||||
|
||||
static leave() {
|
||||
if (IgniteRenderingContext.RenderCount) {
|
||||
IgniteRenderingContext.RenderCount--;
|
||||
}
|
||||
}
|
||||
|
||||
static get rendering() {
|
||||
if (IgniteRenderingContext.RenderCount && IgniteRenderingContext.RenderCount > 0) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
window.IgniteRenderingContext = IgniteRenderingContext;
|
||||
|
||||
export {
|
||||
IgniteProperty
|
||||
};
|
@ -1,5 +1,4 @@
|
||||
import { IgniteProperty } from './ignite-html.js';
|
||||
import { IgniteElement } from './ignite-element.js';
|
||||
|
||||
class IgniteTemplate {
|
||||
constructor(items) {
|
||||
@ -468,7 +467,7 @@ class html extends IgniteTemplate {
|
||||
this.elements.forEach((item) => item.remove());
|
||||
this.elements = [];
|
||||
|
||||
//Reconstruct them html
|
||||
//Reconstruct the html
|
||||
this.construct(null, null);
|
||||
}
|
||||
}
|
||||
@ -479,10 +478,7 @@ class list extends IgniteTemplate {
|
||||
this.tagName = "shadow list";
|
||||
|
||||
if (list instanceof IgniteProperty) {
|
||||
this.callbacks.push(list.attach((oldValue, newValue) => {
|
||||
this.list = newValue;
|
||||
this.construct(null); //If the list changed, reconstruct this template.
|
||||
}));
|
||||
this.callbacks.push(list.attach((oldValue, newValue) => this.onListChanged(oldValue, newValue)));
|
||||
|
||||
this.list = list.value;
|
||||
} else {
|
||||
@ -539,6 +535,18 @@ class list extends IgniteTemplate {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
onListChanged(oldValue, newValue) {
|
||||
this.list = newValue;
|
||||
|
||||
IgniteRenderingContext.enter();
|
||||
|
||||
try {
|
||||
this.construct(null); //The list changed, reconstruct this template.
|
||||
} catch { }
|
||||
|
||||
IgniteRenderingContext.leave();
|
||||
}
|
||||
}
|
||||
|
||||
class slot extends IgniteTemplate {
|
||||
|
Loading…
x
Reference in New Issue
Block a user