2020-07-28 22:23:49 -07:00
|
|
|
import { IgniteProperty } from './ignite-html.js';
|
2020-08-23 11:50:31 -07:00
|
|
|
import { IgniteElement } from './ignite-element.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-30 08:32:44 -07:00
|
|
|
this.events = {};
|
2020-08-11 12:53:57 -07:00
|
|
|
this.styles = {};
|
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) {
|
2020-07-29 14:01:41 -07:00
|
|
|
this.children.push(new html(items[i]));
|
|
|
|
} else if (items[i] instanceof String || typeof items[i] === 'string') {
|
|
|
|
this.children.push(new html(items[i]));
|
2020-07-29 14:03:52 -07:00
|
|
|
} else if (items[i] instanceof Number || typeof items[i] === 'number') {
|
|
|
|
this.children.push(new html(items[i]));
|
2020-07-29 14:01:41 -07:00
|
|
|
} else if (items[i] instanceof IgniteTemplate || items[i].prototype instanceof IgniteTemplate) {
|
2020-07-28 09:04:04 -07:00
|
|
|
this.children.push(items[i]);
|
2020-07-29 14:01:41 -07:00
|
|
|
} else {
|
2020-08-23 11:50:31 -07:00
|
|
|
throw `Attempted to add a child for template: ${this.tagName} which is not supported. Child: ${items[i]}`;
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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.
|
2020-08-23 22:27:12 -07:00
|
|
|
* @param {...any} items
|
2020-07-28 22:23:49 -07:00
|
|
|
*/
|
2020-08-23 22:27:12 -07:00
|
|
|
class(name, converter = null) {
|
|
|
|
if (name instanceof IgniteProperty) {
|
|
|
|
this.callbacks.push(name.attachOnChange((oldValue, newValue) => this.onClassChanged(oldValue, newValue, converter)));
|
|
|
|
this.classes.push(converter != null ? converter(name.value) : name.value);
|
|
|
|
} else {
|
|
|
|
this.classes.push(converter != null ? converter(name) : name);
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
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-08-23 22:27:12 -07:00
|
|
|
attribute(name, value, converter = null) {
|
2020-07-28 09:04:04 -07:00
|
|
|
if (value instanceof IgniteProperty) {
|
2020-08-23 22:27:12 -07:00
|
|
|
this.callbacks.push(value.attachOnChange((oldValue, newValue) => this.onAttributeChanged(oldValue, newValue, name, converter)));
|
|
|
|
this.attributes[name] = converter != null ? converter(value.value) : value.value;
|
2020-07-28 09:04:04 -07:00
|
|
|
} else {
|
2020-08-23 22:27:12 -07:00
|
|
|
this.attributes[name] = converter != null ? converter(value) : value;
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
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-08-23 11:50:31 -07:00
|
|
|
property(name, value, reflect = false) {
|
|
|
|
if (this.properties[name]) {
|
|
|
|
throw `Attempted to set a property twice on a IgniteTemplate: ${this.tagName}. This is not allowed and should be avoided.`;
|
|
|
|
}
|
|
|
|
|
2020-07-28 09:04:04 -07:00
|
|
|
if (value instanceof IgniteProperty) {
|
2020-08-22 16:15:45 -07:00
|
|
|
this.callbacks.push(value.attachOnChange((oldValue, newValue) => this.onPropertyChanged(oldValue, newValue, name)));
|
2020-08-23 11:50:31 -07:00
|
|
|
this.properties[name] = { value: value.value, reflect: (reflect == true ? value : null) };
|
2020-07-28 09:04:04 -07:00
|
|
|
} else {
|
2020-08-23 11:50:31 -07:00
|
|
|
this.properties[name] = { value: value, reflect: null };
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
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) {
|
2020-07-29 14:01:41 -07:00
|
|
|
this.children.push(new html(items[i]));
|
|
|
|
} else if (items[i] instanceof String || typeof items[i] === 'string') {
|
|
|
|
this.children.push(new html(items[i]));
|
2020-07-29 14:03:52 -07:00
|
|
|
} else if (items[i] instanceof Number || typeof items[i] === 'number') {
|
|
|
|
this.children.push(new html(items[i]));
|
2020-07-29 14:01:41 -07:00
|
|
|
} else if (items[i] instanceof IgniteTemplate || items[i].prototype instanceof IgniteTemplate) {
|
2020-07-28 22:23:49 -07:00
|
|
|
this.children.push(items[i]);
|
2020-07-29 14:01:41 -07:00
|
|
|
} else {
|
2020-08-23 11:50:31 -07:00
|
|
|
throw `Attempted to add a child for template: ${this.tagName} which is not supported. Child: ${items[i]}`;
|
2020-07-28 22:23:49 -07:00
|
|
|
}
|
|
|
|
}
|
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) {
|
2020-08-22 16:15:45 -07:00
|
|
|
this.callbacks.push(item.attachOnChange((oldValue, newValue) => this.onRefChanged(oldValue, newValue, item)));
|
2020-07-29 12:24:36 -07:00
|
|
|
this.refs.push((element) => item.value = element);
|
2020-07-28 09:50:26 -07:00
|
|
|
} else {
|
|
|
|
this.refs.push(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2020-07-30 08:32:44 -07:00
|
|
|
/**
|
|
|
|
* Adds a event by its name and the func to invoke once the event fires.
|
|
|
|
* @param {String} eventName
|
|
|
|
* @param {Any} func
|
|
|
|
*/
|
|
|
|
on(eventName, func) {
|
2020-07-30 09:58:47 -07:00
|
|
|
if (!this.events[eventName]) {
|
|
|
|
this.events[eventName] = [];
|
|
|
|
}
|
|
|
|
|
2020-07-30 08:32:44 -07:00
|
|
|
if (func instanceof IgniteProperty) {
|
2020-08-22 16:15:45 -07:00
|
|
|
this.callbacks.push(func.attachOnChange((oldValue, newValue) => this.onEventChanged(oldValue, newValue, eventName)));
|
2020-07-30 09:58:47 -07:00
|
|
|
this.events[eventName].push(func.value);
|
2020-07-30 08:32:44 -07:00
|
|
|
} else {
|
2020-07-30 09:58:47 -07:00
|
|
|
this.events[eventName].push(func);
|
2020-07-30 08:32:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2020-08-23 11:50:31 -07:00
|
|
|
/**
|
|
|
|
* Adds a on click event handler to this template.
|
|
|
|
* @param {Any} func
|
|
|
|
*/
|
2020-08-21 22:13:06 -07:00
|
|
|
onClick(func) {
|
|
|
|
this.on("click", func);
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2020-08-23 11:50:31 -07:00
|
|
|
/**
|
|
|
|
* Adds a on enter key press event handler to this template.
|
|
|
|
* @param {Any} func
|
|
|
|
*/
|
2020-08-21 22:13:06 -07:00
|
|
|
onEnter(func) {
|
|
|
|
var eventName = "keydown";
|
|
|
|
|
|
|
|
if (!this.events[eventName]) {
|
|
|
|
this.events[eventName] = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (func instanceof IgniteProperty) {
|
2020-08-22 16:15:45 -07:00
|
|
|
this.callbacks.push(func.attachOnChange((oldValue, newValue) => {
|
2020-08-21 22:13:06 -07:00
|
|
|
//Create a new wrapped function to check for the enter key being pressed.
|
|
|
|
var wrapped = (e) => {
|
|
|
|
if (e.key === 'Enter') {
|
|
|
|
newValue(e);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-08-21 22:14:57 -07:00
|
|
|
//Store the wrapped function so that it's the old value next time around
|
2020-08-21 22:13:06 -07:00
|
|
|
//and the old event can be removed in the future
|
|
|
|
func._value = wrapped;
|
|
|
|
|
|
|
|
//Invoke the on event changed with the old value and our wrapped value.
|
|
|
|
this.onEventChanged(oldValue, wrapped, eventName)
|
|
|
|
}));
|
|
|
|
|
|
|
|
//Create the initial wrapper
|
2020-08-21 22:14:57 -07:00
|
|
|
var target = func._value;
|
2020-08-21 22:13:06 -07:00
|
|
|
var wrapped = (e) => {
|
|
|
|
if (e.key === 'Enter') {
|
2020-08-21 22:14:57 -07:00
|
|
|
target(e);
|
2020-08-21 22:13:06 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
//Store the wrapped so that it's the old value next time around.
|
|
|
|
func._value = wrapped;
|
|
|
|
|
|
|
|
this.events[eventName].push(wrapped);
|
|
|
|
} else {
|
|
|
|
this.on(eventName, (e) => {
|
|
|
|
if (e.key === 'Enter') {
|
|
|
|
func(e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2020-08-11 12:53:57 -07:00
|
|
|
/**
|
|
|
|
* Sets a css style on this template and adds it to the element
|
|
|
|
* once it's constructed.
|
|
|
|
* @param {String} name
|
|
|
|
* @param {String} value
|
|
|
|
* @param {Boolean} priority
|
|
|
|
*/
|
2020-08-23 22:27:12 -07:00
|
|
|
style(name, value, priority = null, converter = null) {
|
2020-08-11 12:53:57 -07:00
|
|
|
if (value instanceof IgniteProperty) {
|
2020-08-23 22:27:12 -07:00
|
|
|
this.callbacks.push(value.attachOnChange((oldValue, newValue) => this.onCssValueChanged(oldValue, newValue, name, converter)));
|
|
|
|
this.styles[name] = { name: name, value: (converter != null ? converter(value.value) : value.value), priority: priority };
|
2020-08-11 12:53:57 -07:00
|
|
|
} else {
|
2020-08-23 22:27:12 -07:00
|
|
|
this.styles[name] = { name: name, value: (converter != null ? converter(value) : value), priority: priority };
|
2020-08-11 12:53:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
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-29 11:21:02 -07:00
|
|
|
//Don't construct if we have no parent, no sibling and no element.
|
|
|
|
if (!parent && !sibling && !this.element) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2020-08-23 11:50:31 -07:00
|
|
|
//Pass back our template to the element we are creating if it's not a ignite element, since
|
|
|
|
//it will have it's own template automatically.
|
|
|
|
if (!(this.element instanceof IgniteElement)) {
|
|
|
|
this.element.template = this;
|
|
|
|
}
|
2020-07-28 22:23:49 -07:00
|
|
|
|
|
|
|
//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
|
|
|
|
2020-08-23 11:50:31 -07:00
|
|
|
//Invoke any refs we have and pass back the element reference.
|
2020-07-29 12:24:36 -07:00
|
|
|
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++) {
|
2020-08-23 22:27:12 -07:00
|
|
|
if (this.classes[i] !== null && this.classes[i] !== undefined && this.classes[i] !== "") {
|
2020-07-28 09:04:04 -07:00
|
|
|
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]]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-30 08:32:44 -07:00
|
|
|
//Set the events on this element
|
|
|
|
var keys = Object.keys(this.events);
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
2020-07-30 09:58:47 -07:00
|
|
|
this.events[keys[i]].forEach((event) => {
|
|
|
|
if (event !== null && event !== undefined) {
|
|
|
|
this.element.addEventListener(keys[i], event);
|
|
|
|
}
|
|
|
|
});
|
2020-07-30 08:32:44 -07:00
|
|
|
}
|
|
|
|
|
2020-08-11 12:53:57 -07:00
|
|
|
//Set the styles on this element.
|
|
|
|
var keys = Object.keys(this.styles);
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
|
|
|
var style = this.styles[keys[i]];
|
|
|
|
this.element.style.setProperty(style.name, style.value, style.priority);
|
|
|
|
}
|
|
|
|
|
2020-07-28 09:04:04 -07:00
|
|
|
//Set the properties on this element
|
|
|
|
var keys = Object.keys(this.properties);
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
2020-08-23 11:50:31 -07:00
|
|
|
this.element[keys[i]] = this.properties[keys[i]].value;
|
|
|
|
|
|
|
|
if (this.properties[keys[i]].reflect != null) {
|
|
|
|
this.element[keys[i]].reflected.push(this.properties[keys[i]].reflect);
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
//Construct the children under this element
|
|
|
|
for (var i = 0; i < this.children.length; i++) {
|
2020-07-29 14:01:41 -07:00
|
|
|
this.children[i].construct(this.element);
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
2020-07-29 11:21:02 -07:00
|
|
|
|
|
|
|
//If our element has not been added to the dom yet, then add it.
|
|
|
|
if (this.element.isConnected == false && this.element.parentElement == null) {
|
|
|
|
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
|
|
|
/**
|
|
|
|
* Deconstructs this template and cleans up all resources to make sure
|
|
|
|
* there are no memory leaks.
|
|
|
|
*/
|
|
|
|
deconstruct() {
|
2020-07-30 08:32:44 -07:00
|
|
|
//Remove and disconnect all events
|
|
|
|
if (this.element && this.events) {
|
|
|
|
var keys = Object.keys(this.events);
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
|
|
|
this.element.removeEventListener(keys[i], this.events[keys[i]]);
|
|
|
|
}
|
|
|
|
this.events = null;
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
//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;
|
|
|
|
}
|
|
|
|
|
2020-07-29 12:24:36 -07:00
|
|
|
//Disconnect all callbacks
|
2020-07-28 22:23:49 -07:00
|
|
|
if (this.callbacks) {
|
|
|
|
this.callbacks.forEach((item) => item.disconnect());
|
|
|
|
this.callbacks = null;
|
|
|
|
}
|
|
|
|
|
2020-07-29 12:24:36 -07:00
|
|
|
//Remove any refs
|
2020-07-28 22:23:49 -07:00
|
|
|
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-08-23 22:27:12 -07:00
|
|
|
onClassChanged(oldValue, newValue, converter) {
|
|
|
|
if (converter !== null) {
|
|
|
|
oldValue = converter(oldValue);
|
|
|
|
newValue = converter(newValue);
|
|
|
|
}
|
|
|
|
|
2020-07-29 11:21:02 -07:00
|
|
|
if (this.element) {
|
|
|
|
if (oldValue !== null && oldValue !== undefined && oldValue !== "" && oldValue !== " ") {
|
|
|
|
this.element.classList.remove(oldValue);
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
|
2020-07-29 11:21:02 -07:00
|
|
|
if (newValue !== null && newValue !== undefined && newValue !== "" && newValue !== " ") {
|
|
|
|
this.element.classList.add(newValue);
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
2020-07-29 11:21:02 -07:00
|
|
|
|
2020-07-30 10:11:09 -07:00
|
|
|
//Remove the old value
|
2020-07-29 11:21:02 -07:00
|
|
|
this.classes = this.classes.filter(cl => cl != oldValue && cl != newValue);
|
2020-07-30 10:11:09 -07:00
|
|
|
|
|
|
|
//Add the new value if its valid.
|
|
|
|
if (newValue !== null && newValue !== undefined) {
|
|
|
|
this.classes.push(newValue);
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
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-08-23 22:27:12 -07:00
|
|
|
onAttributeChanged(oldValue, newValue, attributeName, converter) {
|
|
|
|
if (converter !== null) {
|
|
|
|
newValue = converter(newValue);
|
|
|
|
}
|
|
|
|
|
2020-07-29 11:21:02 -07:00
|
|
|
if (this.element) {
|
|
|
|
if (newValue == null || newValue == undefined) {
|
|
|
|
this.element.removeAttribute(attributeName);
|
|
|
|
} else {
|
|
|
|
this.element.setAttribute(attributeName, newValue);
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
2020-07-29 11:21:02 -07:00
|
|
|
|
|
|
|
this.attributes[attributeName] = newValue;
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
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) {
|
2020-07-29 11:21:02 -07:00
|
|
|
if (this.element) {
|
2020-08-23 11:50:31 -07:00
|
|
|
//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.
|
|
|
|
this.element[`_${propertyName}`].setValue(newValue, false);
|
2020-07-29 11:21:02 -07:00
|
|
|
}
|
|
|
|
|
2020-08-23 11:50:31 -07:00
|
|
|
this.properties[propertyName].value = newValue;
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
2020-07-29 12:24:36 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Called when a ref was changed and we need to update the refs
|
|
|
|
* value to match this elements reference.
|
|
|
|
* @param {any} oldValue
|
|
|
|
* @param {any} newValue
|
|
|
|
* @param {any} ref
|
|
|
|
*/
|
|
|
|
onRefChanged(oldValue, newValue, ref) {
|
|
|
|
//Only set the reference value to ourself if it's not our element.
|
|
|
|
//Otherwise we will get a never ending loop.
|
|
|
|
if (this.element != newValue) {
|
|
|
|
ref.value = this.element;
|
|
|
|
}
|
|
|
|
}
|
2020-07-30 08:32:44 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Called when a event was changed and we need to update it.
|
|
|
|
* @param {any} oldValue
|
|
|
|
* @param {any} newValue
|
|
|
|
* @param {any} eventName
|
|
|
|
*/
|
|
|
|
onEventChanged(oldValue, newValue, eventName) {
|
|
|
|
if (this.element) {
|
|
|
|
if (oldValue !== null && oldValue !== undefined) {
|
|
|
|
this.element.removeEventListener(eventName, oldValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (newValue !== null && newValue !== undefined) {
|
|
|
|
this.element.addEventListener(eventName, newValue);
|
|
|
|
}
|
|
|
|
|
2020-07-30 09:58:47 -07:00
|
|
|
//Remove the old value from the events
|
|
|
|
this.events[eventName] = this.events[eventName].filter(ev => ev != oldValue);
|
2020-07-30 10:11:09 -07:00
|
|
|
|
|
|
|
//Add the new value if it's needed
|
|
|
|
if (newValue !== null && newValue !== undefined) {
|
|
|
|
this.events[eventName].push(newValue);
|
|
|
|
}
|
2020-07-30 08:32:44 -07:00
|
|
|
}
|
|
|
|
}
|
2020-08-11 12:53:57 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Called when a css value was changed and we need to update the styling.
|
|
|
|
* @param {any} oldValue
|
|
|
|
* @param {any} newValue
|
|
|
|
* @param {any} style
|
|
|
|
*/
|
2020-08-23 22:27:12 -07:00
|
|
|
onCssValueChanged(oldValue, newValue, name, converter) {
|
2020-08-11 12:53:57 -07:00
|
|
|
if (this.element) {
|
2020-08-23 22:27:12 -07:00
|
|
|
this.element.style.setProperty(name, (converter != null ? converter(newValue) : newValue), this.styles[name].priority);
|
2020-08-11 12:53:57 -07:00
|
|
|
}
|
2020-08-23 22:27:12 -07:00
|
|
|
|
|
|
|
this.styles[name].value = (converter != null ? converter(newValue) : newValue);
|
2020-08-11 12:53:57 -07:00
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
class div extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "div";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class a extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "a";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-21 22:13:06 -07:00
|
|
|
class input extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "input";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-23 11:50:31 -07:00
|
|
|
class h1 extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "h1";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class h2 extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "h2";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class h3 extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "h3";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class h4 extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "h4";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class h5 extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "h5";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class p extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "p";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class span extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "span";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class i extends IgniteTemplate {
|
|
|
|
constructor(...items) {
|
|
|
|
super(items);
|
|
|
|
|
|
|
|
this.tagName = "i";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-28 09:04:04 -07:00
|
|
|
class html extends IgniteTemplate {
|
|
|
|
constructor(code) {
|
|
|
|
super([]);
|
2020-07-29 14:01:41 -07:00
|
|
|
|
|
|
|
if (code instanceof IgniteProperty) {
|
2020-08-22 16:15:45 -07:00
|
|
|
this.callbacks.push(code.attachOnChange((oldValue, newValue) => this.onPropertyChanged(oldValue, newValue)));
|
2020-07-29 14:01:41 -07:00
|
|
|
this.code = code.value;
|
|
|
|
} else {
|
|
|
|
this.code = code;
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
this.tagName = "shadow html";
|
2020-07-29 11:21:02 -07:00
|
|
|
this.elements = [];
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
2020-07-29 11:21:02 -07:00
|
|
|
construct(parent, sibling) {
|
|
|
|
//Don't construct if we have no parent, no sibling and no element.
|
|
|
|
if (!parent && !sibling && !this.element) {
|
|
|
|
return;
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
2020-07-29 11:21:02 -07:00
|
|
|
if (!this.element) {
|
|
|
|
this.element = window.document.createTextNode("");
|
|
|
|
|
|
|
|
if (sibling) {
|
|
|
|
sibling.parentElement.insertBefore(this.element, sibling);
|
|
|
|
} else {
|
|
|
|
parent.appendChild(this.element);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-29 14:01:41 -07:00
|
|
|
//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.
|
|
|
|
if (this.elements.length == 0 && this.code) {
|
2020-07-29 11:21:02 -07:00
|
|
|
var template = window.document.createElement("template");
|
|
|
|
template.innerHTML = this.code;
|
|
|
|
while (template.content.childNodes.length > 0) {
|
|
|
|
var item = template.content.childNodes[0];
|
|
|
|
this.element.parentElement.insertBefore(item, this.element);
|
|
|
|
this.elements.push(item);
|
|
|
|
}
|
|
|
|
template.remove();
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
2020-07-29 14:01:41 -07:00
|
|
|
|
|
|
|
onPropertyChanged(oldValue, newValue) {
|
|
|
|
//Update our code to the new value from the property.
|
|
|
|
this.code = newValue;
|
|
|
|
|
|
|
|
//Remove any elements that already exist.
|
|
|
|
this.elements.forEach((item) => item.remove());
|
|
|
|
this.elements = [];
|
|
|
|
|
2020-08-21 21:42:10 -07:00
|
|
|
//Reconstruct the html
|
2020-07-29 14:01:41 -07:00
|
|
|
this.construct(null, null);
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
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-08-22 16:15:45 -07:00
|
|
|
this.callbacks.push(list.attachOnChange((oldValue, newValue) => this.onListChanged(oldValue, newValue)));
|
|
|
|
this.callbacks.push(list.attachOnPush((oldValue, newValue) => this.onListPush(oldValue, newValue)));
|
|
|
|
this.callbacks.push(list.attachOnPop((oldValue, newValue) => this.onListPop(oldValue, newValue)));
|
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-29 11:21:02 -07:00
|
|
|
//Don't construct if we have no parent, no sibling and no element.
|
|
|
|
if (!parent && !sibling && !this.element) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-28 09:04:04 -07:00
|
|
|
if (!this.element) {
|
|
|
|
this.element = window.document.createTextNode(""); //Use a textnode as our placeholder
|
2020-07-29 11:21:02 -07:00
|
|
|
|
|
|
|
if (sibling) {
|
|
|
|
sibling.parentElement.insertBefore(this.element, sibling);
|
|
|
|
} else {
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-08-21 21:42:10 -07:00
|
|
|
|
|
|
|
onListChanged(oldValue, newValue) {
|
|
|
|
this.list = newValue;
|
|
|
|
|
|
|
|
IgniteRenderingContext.enter();
|
|
|
|
|
|
|
|
try {
|
|
|
|
this.construct(null); //The list changed, reconstruct this template.
|
2020-08-23 11:50:31 -07:00
|
|
|
} catch (error) {
|
|
|
|
console.error(error);
|
|
|
|
}
|
2020-08-21 21:42:10 -07:00
|
|
|
|
|
|
|
IgniteRenderingContext.leave();
|
|
|
|
}
|
2020-08-22 16:15:45 -07:00
|
|
|
|
|
|
|
onListPush(oldValue, newValue) {
|
|
|
|
IgniteRenderingContext.enter();
|
|
|
|
|
|
|
|
try {
|
|
|
|
var template = this.forEach(this.list[this.list.length - 1]);
|
2020-08-22 18:11:37 -07:00
|
|
|
|
|
|
|
if (this.elements.length > 0) {
|
|
|
|
template.construct(this.element.parentElement, this.elements[this.elements.length - 1].nextSibling);
|
|
|
|
} else {
|
|
|
|
template.construct(this.element.parentElement, null);
|
|
|
|
}
|
2020-08-22 16:15:45 -07:00
|
|
|
|
|
|
|
this.children.push(template);
|
|
|
|
this.elements.push(template.element);
|
|
|
|
} catch { }
|
|
|
|
|
|
|
|
IgniteRenderingContext.leave();
|
|
|
|
}
|
|
|
|
|
|
|
|
onListPop(oldValue, newValue) {
|
|
|
|
this.children[this.children.length - 1].deconstruct();
|
|
|
|
this.children.pop();
|
|
|
|
this.elements.pop();
|
|
|
|
}
|
2020-07-28 22:23:49 -07:00
|
|
|
}
|
|
|
|
|
2020-07-28 22:46:22 -07:00
|
|
|
class slot extends IgniteTemplate {
|
|
|
|
constructor(element) {
|
|
|
|
super(null);
|
|
|
|
|
|
|
|
this.parent = element;
|
|
|
|
}
|
|
|
|
|
|
|
|
construct(parent, sibling) {
|
2020-07-29 11:21:02 -07:00
|
|
|
//Don't construct if we have no parent, no sibling and no element.
|
|
|
|
if (!parent && !sibling && !this.element) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:46:22 -07:00
|
|
|
if (!this.element) {
|
|
|
|
this.element = window.document.createTextNode("");
|
|
|
|
|
|
|
|
if (sibling) {
|
2020-07-29 11:21:02 -07:00
|
|
|
sibling.parentElement.insertBefore(this.element, sibling);
|
2020-07-28 22:46:22 -07:00
|
|
|
} else {
|
|
|
|
parent.appendChild(this.element);
|
|
|
|
}
|
|
|
|
|
|
|
|
//Add any slot elements after this element.
|
|
|
|
this.parent.elements.forEach((item) => this.element.parentElement.insertBefore(item, this.element));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-28 09:04:04 -07:00
|
|
|
export {
|
|
|
|
IgniteTemplate,
|
|
|
|
div,
|
|
|
|
html,
|
|
|
|
list,
|
2020-07-28 22:46:22 -07:00
|
|
|
a,
|
2020-08-21 22:13:06 -07:00
|
|
|
input,
|
2020-08-23 11:50:31 -07:00
|
|
|
h1,
|
|
|
|
h2,
|
|
|
|
h3,
|
|
|
|
h4,
|
|
|
|
h5,
|
|
|
|
p,
|
|
|
|
span,
|
|
|
|
i,
|
2020-07-28 22:46:22 -07:00
|
|
|
slot
|
2020-07-28 09:04:04 -07:00
|
|
|
};
|