2020-07-28 22:23:49 -07:00
|
|
|
import { IgniteProperty } from './ignite-html.js';
|
2020-07-28 09:04:04 -07:00
|
|
|
|
|
|
|
class IgniteTemplate {
|
|
|
|
constructor(items) {
|
|
|
|
this.siblings = [];
|
|
|
|
this.children = [];
|
|
|
|
this.attributes = {};
|
|
|
|
this.classes = [];
|
|
|
|
this.tagName = null;
|
|
|
|
this.element = null;
|
|
|
|
this.properties = {};
|
2020-07-28 09:50:26 -07:00
|
|
|
this.refs = [];
|
2020-07-28 22:23:49 -07:00
|
|
|
this.callbacks = [];
|
2020-07-28 09:04:04 -07:00
|
|
|
|
|
|
|
if (items) {
|
|
|
|
for (var i = 0; i < items.length; i++) {
|
2020-07-28 22:23:49 -07:00
|
|
|
if (items[i] instanceof IgniteProperty) {
|
|
|
|
this.children.push(new property(items[i]));
|
2020-07-28 09:04:04 -07:00
|
|
|
} else {
|
|
|
|
this.children.push(items[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Adds a single or series of classes to this template
|
|
|
|
* to be added once this template is constructed.
|
|
|
|
* @param {...any} items
|
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
class(...items) {
|
|
|
|
for (var i = 0; i < items.length; i++) {
|
|
|
|
if (items[i] instanceof IgniteProperty) {
|
2020-07-28 22:23:49 -07:00
|
|
|
this.callbacks.push(items[i].attach(this.onClassChanged));
|
2020-07-28 09:04:04 -07:00
|
|
|
this.classes.push(items[i].value);
|
|
|
|
} else {
|
|
|
|
this.classes.push(items[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Adds a attribute to this template to be added once this template is constructed.
|
|
|
|
* @param {string} name
|
|
|
|
* @param {any} value
|
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
attribute(name, value) {
|
|
|
|
if (value instanceof IgniteProperty) {
|
2020-07-28 22:23:49 -07:00
|
|
|
this.callbacks.push(value.attach((oldValue, newValue) => this.onAttributeChanged(oldValue, newValue, name)));
|
2020-07-28 09:04:04 -07:00
|
|
|
this.attributes[name] = value.value;
|
|
|
|
} else {
|
|
|
|
this.attributes[name] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Adds a property to this template to be added once this template is constructed.
|
|
|
|
* @param {string} name
|
|
|
|
* @param {any} value
|
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
property(name, value) {
|
|
|
|
if (value instanceof IgniteProperty) {
|
2020-07-28 22:23:49 -07:00
|
|
|
this.callbacks.push(value.attach((oldValue, newValue) => this.onAttributeChanged(oldValue, newValue, name)));
|
2020-07-28 09:04:04 -07:00
|
|
|
this.properties[name] = value.value;
|
|
|
|
} else {
|
|
|
|
this.properties[name] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Adds a single or series of children to be added once this template
|
|
|
|
* is constructed.
|
|
|
|
* @param {...any} items
|
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
child(...items) {
|
2020-07-28 22:23:49 -07:00
|
|
|
for (var i = 0; i < items.length; i++) {
|
|
|
|
if (items[i] instanceof IgniteProperty) {
|
|
|
|
this.children.push(new property(items[i]));
|
|
|
|
} else {
|
|
|
|
this.children.push(items[i]);
|
|
|
|
}
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Adds a ref callback function to be invoked once this template is constructed.
|
|
|
|
* @param {function} item
|
|
|
|
*/
|
2020-07-28 09:50:26 -07:00
|
|
|
ref(item) {
|
|
|
|
if (item instanceof IgniteProperty) {
|
|
|
|
this.refs.push((element) => {
|
|
|
|
item.value = element;
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
this.refs.push(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Constructs this template and adds it to the DOM if this template
|
|
|
|
* has not already been constructed.
|
|
|
|
* @param {HTMLElement} parent
|
|
|
|
* @param {HTMLElement} sibling
|
|
|
|
*/
|
|
|
|
construct(parent, sibling) {
|
2020-07-28 09:04:04 -07:00
|
|
|
//Construct this element if we haven't already
|
|
|
|
if (!this.element) {
|
|
|
|
this.element = window.document.createElement(this.tagName);
|
2020-07-28 22:23:49 -07:00
|
|
|
|
|
|
|
if (sibling) {
|
|
|
|
parent.insertBefore(this.element, sibling);
|
|
|
|
} else {
|
|
|
|
parent.appendChild(this.element);
|
|
|
|
}
|
|
|
|
|
|
|
|
//If the element has a onDisconnected function, attach to it
|
|
|
|
//(This way if a custom element is removed we can deconstruct and cleanup)
|
|
|
|
if (this.element.onDisconnected !== undefined) {
|
|
|
|
this.element.onDisconnected = () => this.deconstruct();
|
|
|
|
}
|
2020-07-28 09:50:26 -07:00
|
|
|
|
|
|
|
//Invoke any refs we have
|
|
|
|
this.refs.forEach((ref) => { ref(this.element); });
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
//Set the classes on this element
|
|
|
|
for (var i = 0; i < this.classes.length; i++) {
|
|
|
|
if (this.classes[i] !== null && this.classes[i] !== undefined) {
|
|
|
|
this.element.classList.add(this.classes[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//Set the attributes on this element
|
|
|
|
var keys = Object.keys(this.attributes);
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
|
|
|
if (this.attributes[keys[i]] !== null && this.attributes[keys[i]] !== undefined) {
|
|
|
|
this.element.setAttribute(keys[i], this.attributes[keys[i]]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//Set the properties on this element
|
|
|
|
var keys = Object.keys(this.properties);
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
|
|
|
this.element[keys[i]] = this.properties[keys[i]];
|
|
|
|
}
|
|
|
|
|
|
|
|
//Construct the children under this element
|
|
|
|
for (var i = 0; i < this.children.length; i++) {
|
|
|
|
if (this.children[i] instanceof String || typeof this.children[i] === 'string') {
|
|
|
|
this.element.appendChild(document.createTextNode(this.children[i]));
|
|
|
|
} else if (this.children[i] instanceof IgniteTemplate || this.children[i].prototype instanceof IgniteTemplate) {
|
|
|
|
this.children[i].construct(this.element);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Deconstructs this template and cleans up all resources to make sure
|
|
|
|
* there are no memory leaks.
|
|
|
|
*/
|
|
|
|
deconstruct() {
|
|
|
|
console.log(`Deconstructing ${this.tagName}`);
|
|
|
|
//Remove our element if we have one.
|
|
|
|
if (this.element) {
|
|
|
|
this.element.remove();
|
|
|
|
this.element = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
//Deconstruct all children elements.
|
|
|
|
if (this.children) {
|
|
|
|
for (var i = 0; i < this.children.length; i++) {
|
|
|
|
if (this.children[i] instanceof IgniteTemplate || this.children[i].prototype instanceof IgniteTemplate) {
|
|
|
|
this.children[i].deconstruct();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.children = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
//disconnect all callbacks
|
|
|
|
if (this.callbacks) {
|
|
|
|
this.callbacks.forEach((item) => item.disconnect());
|
|
|
|
this.callbacks = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
//remove any refs
|
|
|
|
if (this.refs) {
|
|
|
|
this.refs = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called when a class on this template was changed and needs to be updated
|
|
|
|
* on the template's element.
|
|
|
|
* @param {any} oldValue
|
|
|
|
* @param {any} newValue
|
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
onClassChanged(oldValue, newValue) {
|
|
|
|
console.log(`Class changed, oldValue: ${oldValue} newValue: ${newValue}`);
|
|
|
|
if (oldValue !== null && oldValue !== undefined && oldValue !== "" && oldValue !== " ") {
|
|
|
|
this.element.classList.remove(oldValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (newValue !== null && newValue !== undefined && newValue !== "" && newValue !== " ") {
|
|
|
|
this.element.classList.add(newValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Called when a attribute on this template was changed and needs to be updated
|
|
|
|
* on the template's element.
|
|
|
|
* @param {any} oldValue
|
|
|
|
* @param {any} newValue
|
|
|
|
* @param {string} attributeName
|
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
onAttributeChanged(oldValue, newValue, attributeName) {
|
|
|
|
console.log(`Attribute changed, oldValue: ${oldValue} newValue: ${newValue} attribute: ${attributeName}`);
|
|
|
|
if (newValue == null || newValue == undefined) {
|
|
|
|
this.element.removeAttribute(attributeName);
|
|
|
|
} else {
|
|
|
|
this.element.setAttribute(attributeName, newValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Called when a property on this template was changed and needs to be updated
|
|
|
|
* on the template's element.
|
|
|
|
* @param {any} oldValue
|
|
|
|
* @param {any} newValue
|
|
|
|
* @param {string} propertyName
|
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
onPropertyChanged(oldValue, newValue, propertyName) {
|
|
|
|
console.log(`Property changed, oldValue: ${oldValue} newValue: ${newValue} property: ${propertyName}`);
|
|
|
|
this.properties[propertyName] = newValue;
|
|
|
|
this.element[propertyName] = newValue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class div extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "div";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class a extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "a";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class html extends IgniteTemplate {
|
|
|
|
constructor(code) {
|
|
|
|
super([]);
|
|
|
|
this.code = code;
|
2020-07-28 22:23:49 -07:00
|
|
|
this.tagName = "shadow html";
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
construct(parent) {
|
|
|
|
if (!parent) {
|
|
|
|
parent = window.document.body;
|
|
|
|
}
|
|
|
|
|
|
|
|
parent.insertAdjacentHTML('beforeend', this.code);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class list extends IgniteTemplate {
|
|
|
|
constructor(list, forEach) {
|
|
|
|
super([]);
|
2020-07-28 22:23:49 -07:00
|
|
|
this.tagName = "shadow list";
|
2020-07-28 09:04:04 -07:00
|
|
|
|
|
|
|
if (list instanceof IgniteProperty) {
|
2020-07-28 22:23:49 -07:00
|
|
|
this.callbacks.push(list.attach((oldValue, newValue) => {
|
|
|
|
this.list = newValue;
|
|
|
|
this.construct(null); //If the list changed, reconstruct this template.
|
|
|
|
}));
|
2020-07-28 09:04:04 -07:00
|
|
|
|
|
|
|
this.list = list.value;
|
|
|
|
} else {
|
|
|
|
this.list = list;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.forEach = forEach;
|
|
|
|
this.elements = [];
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
construct(parent, sibling) {
|
2020-07-28 09:04:04 -07:00
|
|
|
if (!this.element) {
|
|
|
|
this.element = window.document.createTextNode(""); //Use a textnode as our placeholder
|
|
|
|
parent.appendChild(this.element);
|
2020-07-28 22:23:49 -07:00
|
|
|
} else {
|
|
|
|
parent = this.element.parentElement;
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
//If we already have elements we created, destroy them.
|
|
|
|
if (this.elements.length > 0) {
|
|
|
|
for (var i = 0; i < this.elements.length; i++) {
|
|
|
|
this.elements[i].remove();
|
|
|
|
}
|
|
|
|
this.elements = [];
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
//If we already have children elements, deconstruct them.
|
|
|
|
//(Because we are about to recreate them)
|
|
|
|
if (this.children.length > 0) {
|
|
|
|
for (var i = 0; i < this.children.length; i++) {
|
|
|
|
this.children[i].deconstruct();
|
|
|
|
}
|
|
|
|
this.children = [];
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
|
|
|
|
//Construct all the items in our list and use the container
|
|
|
|
if (this.list) {
|
|
|
|
for (var i = 0; i < this.list.length; i++) {
|
2020-07-28 22:23:49 -07:00
|
|
|
var template = this.forEach(this.list[i]);
|
|
|
|
template.construct(parent, this.element);
|
|
|
|
|
|
|
|
this.children.push(template);
|
|
|
|
this.elements.push(template.element);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class property extends IgniteTemplate {
|
|
|
|
constructor(prop) {
|
|
|
|
super(null);
|
|
|
|
|
|
|
|
this.property = prop;
|
|
|
|
this.elements = [];
|
|
|
|
this.callbacks.push(this.property.attach((oldValue, newValue) => this.onPropertyChanged(oldValue, newValue)));
|
|
|
|
}
|
|
|
|
|
|
|
|
construct(parent, sibling) {
|
|
|
|
if (!this.element) {
|
|
|
|
this.element = window.document.createTextNode("");
|
|
|
|
|
|
|
|
if (sibling) {
|
|
|
|
parent.insertBefore(this.element, sibling);
|
|
|
|
} else {
|
|
|
|
parent.appendChild(this.element);
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
//Remove any elements that already exist.
|
|
|
|
this.elements.forEach((item) => item.remove());
|
|
|
|
|
|
|
|
//Create a template to hold the elements that will be created from the
|
|
|
|
//properties value and then add them to the DOM and store their pointer.
|
|
|
|
var template = window.document.createElement("template");
|
|
|
|
template.innerHTML = this.property.value;
|
|
|
|
while (template.content.childNodes.length > 0) {
|
|
|
|
var item = template.content.childNodes[0];
|
|
|
|
this.element.parentElement.insertBefore(item, this.element);
|
|
|
|
this.elements.push(item);
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
2020-07-28 22:23:49 -07:00
|
|
|
template.remove();
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
onPropertyChanged(oldValue, newValue) {
|
|
|
|
this.construct(null, null);
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export {
|
|
|
|
IgniteTemplate,
|
|
|
|
div,
|
|
|
|
html,
|
|
|
|
list,
|
|
|
|
a
|
|
|
|
};
|