import { IgniteProperty } from './ignite-html.js'; import { IgniteElement } from './ignite-element.js'; /** * The outline of a ignite template. Templates are a blueprint that specify's * how to construct an element and then can be used to construct the element. Everything * starts with a template. * * @example * //You can easily create a template to construct any html element. See the following: * class div extends IgniteTemplate { * constructor(...items) { * super("div", items); * } * } */ class IgniteTemplate { /** * Creates a new IgniteTemplate with the tag name of the element that will be constructed and an * array of child elements. * @param {String} tagName The tag name of the element this template will construct. * @param {String|Number|IgniteProperty|IgniteTemplate} children An array of child elements to be added to this template. */ constructor(tagName = null, children = null) { this.children = []; this.tagName = tagName; this.tagNamespace = null; this.element = null; this._attributes = {}; this._classes = []; this._properties = {}; this._reflecting = {}; this._refs = []; this._callbacks = []; this._events = {}; this._styles = {}; this._elementValue = null; this._elementInnerHTML = null; this._resizeObserverCallback = null; this._resizeObserver = null; if (children) { for (var i = 0; i < children.length; i++) { if (children[i] === undefined || children[i] === null) { continue; //Skip undefined or null children. } else if (children[i] instanceof IgniteProperty) { this.children.push(new html(children[i])); } else if (children[i] instanceof String || typeof children[i] === 'string') { this.children.push(new html(children[i])); } else if (children[i] instanceof Number || typeof children[i] === 'number') { this.children.push(new html(children[i])); } else if (children[i] instanceof IgniteTemplate || children[i].prototype instanceof IgniteTemplate) { this.children.push(children[i]); } else { throw `Attempted to add a child for template: ${this.tagName} which is not supported. Child: ${children[i]}`; } } } } /** * Adds a CSS class to be added once this template is constructed. * @param {String|IgniteProperty} name Name of the CSS class to add. Multiple CSS classes are supported if they are separated by a space. * @param {Function} converter Optional function that can convert the class name into a different one. * @example * .class("row justify-content-center") * @returns This ignite template so function calls can be chained. */ class(name, converter = null) { IgniteRenderingContext.push(); if (name instanceof IgniteProperty) { this._callbacks.push(name.attachOnChange((oldValue, newValue) => this.onClassChanged((converter != null ? converter(oldValue) : oldValue), (converter != null ? converter(newValue) : newValue)))); var value = (converter != null ? converter(name.value) : name.value); (value != null ? value.toString().split(" ") : []).forEach(cl => { if (cl.length > 0) { this._classes.push(cl); } }); } else if (Array.isArray(name) && name.length > 0 && name[0] instanceof IgniteProperty) { //There must be a converter for this to work correctly if (!converter) { throw "Cannot pass an array of properties without using a converter!"; } //Attack a callback for all the properties name.forEach(prop => { if (prop instanceof IgniteProperty) { this._callbacks.push(prop.attachOnChange((oldValue, newValue) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, oldValue)), converter(...name.getPropertyValues())))); this._callbacks.push(prop.attachOnPush((list, items) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues())))); this._callbacks.push(prop.attachOnUnshift((list, items) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues())))); this._callbacks.push(prop.attachOnPop((list) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues())))); this._callbacks.push(prop.attachOnShift((list) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues())))); this._callbacks.push(prop.attachOnSplice((list, start, deleteCount, items) => this.onClassChanged(converter(...name.getOldPropertyValues(prop, list)), converter(...name.getPropertyValues())))); } }); var value = converter(...name.getPropertyValues()); (value != null ? value.toString().split(" ") : []).forEach(cl => { if (cl.length > 0) { this._classes.push(cl); } }); } else { var value = (converter != null ? converter(name) : name); (value != null ? value.toString().split(" ") : []).forEach(cl => { if (cl.length > 0) { this._classes.push(cl); } }); } IgniteRenderingContext.pop(); return this; } /** * Adds a html element attribute to this template to be added once this template is constructed. * @param {String} name The name of the attribute to add * @param {String|IgniteProperty} value The value of the attribute to set, can be anything. If Property is passed it will auto update. * @param {Function} converter Optional function that can convert the value if needed. * @returns This ignite template so function calls can be chained. */ attribute(name, value, converter = null) { IgniteRenderingContext.push(); if (value instanceof IgniteProperty) { this._callbacks.push(value.attachOnChange((oldValue, newValue) => this.onAttributeChanged(oldValue, newValue, name, converter))); this._attributes[name] = converter != null ? converter(value.value) : value.value; } else { this._attributes[name] = converter != null ? converter(value) : value; } IgniteRenderingContext.pop(); return this; } /** * Sets the value of the element this template is constructing with the option to reflect changes * to the value. * @param {String|IgniteProperty} value The value to set on the element. * @param {Boolean} reflect Whether or not to reflect changes to the value of the element back to the property if one was used. * @param {Function} converter Optional function that can convert the value if needed. * @returns This ignite template so function calls can be chained. */ value(value, reflect = false, converter = null) { IgniteRenderingContext.push(); if (reflect && converter != null) { throw `Cannot set a value on an IgniteTemplate: ${this.tagName} with reflect and a converter used at the same time.`; } if (value instanceof IgniteProperty) { this._callbacks.push(value.attachOnChange((oldValue, newValue) => this.onValueChanged(oldValue, newValue, converter))); if (reflect) { this.on("change", e => { if (this.element.hasAttribute("type") && this.element.getAttribute("type").toLowerCase().trim() == "checkbox") { value.setValue(this.element.checked, true); } else if (this.element.hasAttribute("contenteditable") && this.element.getAttribute("contenteditable").toLowerCase().trim() == "true") { value.setValue(this.element.textContent, true); } else { value.setValue(this.element.value, true); } }); this.on("keyup", e => { if (this.element.hasAttribute("type") && this.element.getAttribute("type").toLowerCase().trim() == "checkbox") { value.setValue(this.element.checked, true); } else if (this.element.hasAttribute("contenteditable") && this.element.getAttribute("contenteditable").toLowerCase().trim() == "true") { value.setValue(this.element.textContent, true); } else { value.setValue(this.element.value, true); } }); } this._elementValue = (converter != null ? converter(value.value) : value.value); } else { this._elementValue = (converter != null ? converter(value) : value); } IgniteRenderingContext.pop(); return this; } /** * Adds a property to this template to be added once this template is constructed. * @param {String} name Name of the property to set. * @param {Any|IgniteProperty} value Value of the property to use. If a Property is passed the value will auto update. * @param {Boolean} reflect If true whenever this property is changed it's value will be passed back to the Property that was passed as value if one was passed. * @param {Function} converter Optional function that can be used to convert the value if needed. * @returns This ignite template so function calls can be chained. */ property(name, value, reflect = false, converter = null) { IgniteRenderingContext.push(); if (this._properties[name]) { throw `Attempted to set a property twice on a IgniteTemplate: ${this.tagName}. This is not allowed and should be avoided.`; } if (reflect && converter != null) { throw `Cannot add a property to an IgniteTemplate: ${this.tagName} with reflect and a converter used at the same time.`; } if (value instanceof IgniteProperty) { this._callbacks.push(value.attachOnChange((oldValue, newValue) => this.onPropertyChanged(oldValue, newValue, name, converter))); this._properties[name] = { value: (converter != null ? converter(value.value) : value.value), reflect: (reflect == true ? value : null) }; } else { this._properties[name] = { value: (converter != null ? converter(value) : value), reflect: null }; } IgniteRenderingContext.pop(); return this; } /** * Makes a property on this template reflect its value back to the given target. * (You can reflect a property more than once if it's needed.) * @param {String} name Name of the property to reflect. * @param {IgniteProperty|Function} target The target for the value to be reflected to. */ reflect(name, target) { IgniteRenderingContext.push(); if (this._reflecting[name] == undefined || this._reflecting[name] == null) { this._reflecting[name] = []; } if (target instanceof IgniteProperty) { this._reflecting[name].push(target); } else if (target instanceof Function) { this._reflecting[name].push(target); } IgniteRenderingContext.pop(); return this; } /** * 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. * @returns This ignite template so function calls can be chained. */ properties(props) { IgniteRenderingContext.push(); //Make sure we have a valid props. if (props == null || props == undefined) { return; } if (!(typeof props === 'object')) { throw `Cannot set properties with a non object set of properties: ${props}`; } var propNames = Object.keys(props); propNames.forEach(name => { this.property(name, props[name], false, null); }); IgniteRenderingContext.pop(); return this; } /** * Sets the inner html of the element to be constructed by this template. * @param {String|IgniteProperty} value InnerHTML to set for element. If a property is passed the html will auto update. * @param {Function} converter Optional function that can be used to convert the value if needed. * @returns This ignite template so funciton calls can be chained. */ innerHTML(value, converter = null) { IgniteRenderingContext.push(); if (value instanceof IgniteProperty) { this._callbacks.push(value.attachOnChange((oldValue, newValue) => this.onInnerHTMLChanged(oldValue, newValue, converter))); this._elementInnerHTML = (converter != null ? converter(value.value) : value.value); } else { this._elementInnerHTML = (converter != null ? converter(value) : value); } IgniteRenderingContext.pop(); return this; } /** * Adds a single or series of children to be added once this template * is constructed. Numbers, Strings, and Properties passed will be added as HTML child elements. * @param {...Number|String|IgniteProperty|IgniteTemplate} items A series of children to be added to this template. * @returns This ignite template so function calls can be chained. */ child(...items) { if (items) { for (var i = 0; i < items.length; i++) { if (items[i] === undefined || items[i] === null) { continue; //Skip undefined or null items. } else if (items[i] instanceof IgniteProperty) { this.children.push(new html(items[i])); } else if (items[i] instanceof String || typeof items[i] === 'string') { this.children.push(new html(items[i])); } else if (items[i] instanceof Number || typeof items[i] === 'number') { this.children.push(new html(items[i])); } else if (items[i] instanceof IgniteTemplate || items[i].prototype instanceof IgniteTemplate) { this.children.push(items[i]); } else { throw `Attempted to add a child for template: ${this.tagName} which is not supported. Child: ${items[i]}`; } } } return this; } /** * Adds a reference callback function to be invoked once this template is constructed. The function will be invoked * with the constructed HTMLElement. If an IgniteProperty is passed it's value will be set to the constructed HTMLElement once * the template is constructed. * @param {Function|IgniteProperty} refCallback The callback to be invoked with the HTMLElement of the element this template constructed. * @returns This ignite template so function calls can be chained. */ ref(refCallback) { if (refCallback instanceof IgniteProperty) { this._callbacks.push(refCallback.attachOnChange((oldValue, newValue) => this.onRefChanged(oldValue, newValue, refCallback))); this._refs.push(element => refCallback.value = element); } else { this._refs.push(refCallback); } return this; } /** * Adds an event by its name and the function to invoke once the event fires. Properties may be used * for the function, but their value must be a valid function in order to get a proper event callback. * @param {String} eventName The name of the event to add. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires. * @returns This ignite template so function calls can be chained. */ on(eventName, eventCallback) { IgniteRenderingContext.push(); if (!this._events[eventName]) { this._events[eventName] = []; } if (eventCallback instanceof IgniteProperty) { this._callbacks.push(eventCallback.attachOnChange((oldValue, newValue) => this.onEventChanged(oldValue, newValue, eventName))); this._events[eventName].push(eventCallback.value); } else { this._events[eventName].push(eventCallback); } IgniteRenderingContext.pop(); return this; } /** * Adds a click event handler to this template. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires. * @returns This ignite template so function calls can be chained. */ onClick(eventCallback) { return this.on("click", eventCallback); } /** * Adds a touch event handler to this template. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires. * @returns This ignite template so function calls can be chained. */ onTouch(eventCallback) { return this.on("touch", eventCallback); } /** * Adds a onblur event handler to this template. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires. * @returns This ignite template so function calls can be chained. */ onBlur(eventCallback) { return this.on("blur", eventCallback); } /** * Adds a onfocus event handler to this template. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires. * @returns This ignite template so function calls can be chained. */ onFocus(eventCallback) { return this.on("focus", eventCallback); } /** * Adds a onchange event handler to this template. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires. * @returns This ignite template so function calls can be chained. */ onChange(eventCallback) { return this.on("change", eventCallback); } /** * Adds a on enter key press event handler to this template. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires. * @returns This ignite template so function calls can be chained. */ onEnter(eventCallback) { var eventName = "keydown"; if (!this._events[eventName]) { this._events[eventName] = []; } if (eventCallback instanceof IgniteProperty) { this._callbacks.push(eventCallback.attachOnChange((oldValue, newValue) => { //Create a new wrapped function to check for the enter key being pressed. var wrapped = (e) => { if (e.key === 'Enter') { newValue(e); } }; eventCallback._value = wrapped; //Store the wrapped function into the property so we can remove it later this.onEventChanged(oldValue, wrapped, eventName); //Invoke event changed with the old value and wrapped one. })); //Create the initial wrapper var target = eventCallback._value; var wrapped = (e) => { if (e.key === 'Enter') { target(e); } }; //Store the wrapped so that it's the old value next time around. eventCallback._value = wrapped; this._events[eventName].push(wrapped); } else { this.on(eventName, (e) => { if (e.key === 'Enter') { eventCallback(e); } }); } return this; } /** * Adds a on backspace key press event handler to this template. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires. * @returns This ignite template so function calls can be chained. */ onBackspace(eventCallback) { var eventName = "keydown"; if (!this._events[eventName]) { this._events[eventName] = []; } if (eventCallback instanceof IgniteProperty) { this._callbacks.push(eventCallback.attachOnChange((oldValue, newValue) => { //Create a new wrapped function to check for the enter key being pressed. var wrapped = (e) => { if (e.key === 'Backspace') { newValue(e); } }; eventCallback._value = wrapped; //Store the wrapped function into the property so we can remove it later this.onEventChanged(oldValue, wrapped, eventName); //Invoke event changed with the old value and wrapped one. })); //Create the initial wrapper var target = eventCallback._value; var wrapped = (e) => { if (e.key === 'Backspace') { target(e); } }; //Store the wrapped so that it's the old value next time around. eventCallback._value = wrapped; this._events[eventName].push(wrapped); } else { this.on(eventName, (e) => { if (e.key === 'Backspace') { eventCallback(e); } }); } return this; } /** * Adds a special on resize event handler to this template that will * fire anytime the element is resized by using a resize observer. * @param {Function|IgniteProperty} eventCallback The callback function to be invoked by the event once it fires. * @returns This ignite template so function calls can be chained. */ onResize(eventCallback) { IgniteRenderingContext.push(); if (eventCallback instanceof IgniteProperty) { this._resizeObserverCallback = eventCallback.value; } else if (eventCallback instanceof Function) { this._resizeObserverCallback = eventCallback; } IgniteRenderingContext.pop(); return this; } /** * Adds a CSS property to this template with a value and priority. * @param {String} name The name of the CSS property to set. * @param {String|IgniteProperty} value The value to set for the property. If an IgniteProperty is used it will auto update this style. * @param {String} priority If set to "important" then the style will be marked with !important. Acceptable values: important, !important, true, false, null * @param {Function} converter Optional function to convert the value if needed. * @returns This ignite template so function calls can be chained. */ style(name, value, priority = null, converter = null) { IgniteRenderingContext.push(); //If the name has a : remove it. if (name && typeof name === "string" && name.includes(":")) { name = name.replace(":", ""); } //If the priority starts with a ! remove it. if (priority && typeof priority === "string" && priority.trim().startsWith("!")) { priority = priority.split("!")[1].trim(); } else if (priority && typeof priority === "boolean") { priority = "important"; //If priority is true, set it to important } else if (!priority && typeof priority === "boolean") { priority = null; //If priority is false, set it to null. } //If the value has a ; remove it. if (value && typeof value === "string" && value.includes(";")) { value = value.replace(";", ""); } if (value instanceof IgniteProperty) { this._callbacks.push(value.attachOnChange((oldValue, newValue) => this.onStyleChanged(name, (converter ? converter(newValue) : newValue)))); this._styles[name] = { name: name, value: (converter ? converter(value.value) : value.value), priority: priority }; } else if (Array.isArray(value) && value.length > 0 && value[0] instanceof IgniteProperty) { //There must be a converter for this to work correctly if (!converter) { throw "Cannot pass an array of properties without using a converter!"; } //Attack a callback for all the properties value.forEach(prop => { if (prop instanceof IgniteProperty) { this._callbacks.push(prop.attachOnChange((oldValue, newValue) => this.onStyleChanged(name, converter(...value.getPropertyValues())))); this._callbacks.push(prop.attachOnPush((list, items) => this.onStyleChanged(name, converter(...value.getPropertyValues())))); this._callbacks.push(prop.attachOnUnshift((list, items) => this.onStyleChanged(name, converter(...value.getPropertyValues())))); this._callbacks.push(prop.attachOnPop((list) => this.onStyleChanged(name, converter(...value.getPropertyValues())))); this._callbacks.push(prop.attachOnShift((list) => this.onStyleChanged(name, converter(...value.getPropertyValues())))); this._callbacks.push(prop.attachOnSplice((list, start, deleteCount, items) => this.onStyleChanged(name, converter(...value.getPropertyValues())))); } }); this._styles[name] = { name: name, value: converter(...value.getPropertyValues()), priority: priority }; } else { this._styles[name] = { name: name, value: (converter != null ? converter(value) : value), priority: priority }; } IgniteRenderingContext.pop(); return this; } /** * Hides the element this template is constructing if the value is true. * @param {Boolean|IgniteProperty} value If true hides the element this template is constructing. If an IgniteProperty is passed it's value will auto update this. * @param {Function} converter An optional function to convert the value if needed. */ hide(value, converter = null) { return this.style("display", value, true, (...params) => { return ((converter != null && converter(...params)) || (converter == null && params[0])) ? "none" : null; }); } /** * Shows the element this template is constructing if the value is true. * @param {Boolean|IgniteProperty} value * @param {Function} converter */ show(value, converter = null) { return this.style("display", value, true, (...params) => { return ((converter != null && converter(...params)) || (converter == null && params[0])) ? null : "none"; }); } /** * Sets the id attribute of the element to be constructed by this template. * @param {String|IgniteProperty} value The value to set for the id attribute of the element this template will construct. * @param {Function} converter An optional function that can convert the value if needed. * @returns This ignite template so function calls can be chained. */ id(value, converter = null) { return this.attribute("id", value, converter); } /** * Sets the for attribute of the element to be constructed by this template. * @param {String|IgniteProperty} value The value to set for the for attribute of the element this template will construct. * @param {Function} converter An optional function that can convert the value if needed. * @returns This ignite template so function calls can be chained. */ for(value, converter = null) { return this.attribute("for", value, converter); } /** * Adds a checked attribute to this template. * @param {Boolean|IgniteProperty} value The value to set for the checked attribute. * @param {*} converter Optional function that can convert the value if needed. * @returns This ignite template so function calls can be chained. */ checked(value, converter = null) { return this.attribute("checked", value, converter); } /** * Sets the type attribute of the element to be constructed by this template. * @param {String|IgniteProperty} value The value to set for the type attribute of the element this template will construct. * @param {Function} converter An optional function that can convert the value if needed. * @returns This ignite template so function calls can be chained. */ type(value, converter = null) { return this.attribute("type", value, converter); } /** * Sets a data attribute on the element to be constructed by this template. * @param {String} name The name of the data attribute to set on the element this template will construct. * @param {String|IgniteProperty} value The value to set for the data attribute of the element this template will construct. * @param {*} converter An optional function that can convert the value if needed. * @returns This ignite template so function calls can be chained. */ data(name, value, converter = null) { return this.attribute(`data-${name}`, value, converter); } /** * Sets the value attribute of the element to be constructed by this template. * @param {String|IgniteProperty} value The value to set for the src attribute of the element to be constructed by this template. * @param {Function} converter An optional function that can convert the value if needed. * @returns This ignite template so function calls can be chained. */ src(value, converter = null) { return this.attribute("src", value, converter); } /** * Sets the value attribute of the element to be constructed by this template. * @param {String|IgniteProeprty} value The value to set for the href attribute of the element to be constructed by this template. * @param {Function} converter An optional function that can convert the value if needed. * @returns This ignite template so function calls can be chained. */ href(value, converter = null) { return this.attribute("href", value, converter); } /** * Sets the name attribute of the element to be constructed by this template. * @param {String|IgniteProperty} value The value to set for the name attribute of the element to be constructed by this template. * @param {Function} converter An optional function that can convert the value if needed. * @returns This ignite template so function calls can be chained. */ name(value, converter = null) { return this.attribute("name", value, converter); } /** * Sets the placeholder attribute of the element to be constructed by this template. * @param {String|IgniteProperty} value The value to set for the placeholder attribute of the element. * @param {Function} converter An optional function that can convert the value if needed. * @returns This ignite template so function calls can be chained. */ placeholder(value, converter = null) { return this.attribute("placeholder", value, converter); } /** * Constructs this template and adds it to the DOM if this template * has not already been constructed. * @param {HTMLElement} parent Parent element that will contain the constructed element. * @param {HTMLElement} sibling Optional sibling element that can be used to add the element adjacantly. */ construct(parent, sibling) { //Don't construct if we have no parent, no sibling and no element. if (!parent && !sibling && !this.element) { return; } //Construct this element if we haven't already if (!this.element) { //If we have a tag namespace use that to construct this element. This is mainly //for svg elements since they have to be created this way. if (this.tagNamespace) { this.element = window.document.createElementNS(this.tagNamespace, this.tagName); } else { this.element = window.document.createElement(this.tagName); } //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; } //Invoke any refs we have and pass back the element reference. this._refs.forEach((ref) => ref(this.element)); } //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._classes[i] !== "") { 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 events on this element var keys = Object.keys(this._events); for (var i = 0; i < keys.length; i++) { this._events[keys[i]].forEach((event) => { if (event !== null && event !== undefined) { this.element.addEventListener(keys[i], event); } }); } //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); } //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]].value; if (this._properties[keys[i]].reflect != null) { this.element[keys[i]].reflected.push(this._properties[keys[i]].reflect); } } //Setup any reflecting properties on this element var keys = Object.keys(this._reflecting); for (var i = 0; i < keys.length; i++) { this._reflecting[keys[i]].forEach(value => this.element[keys[i]].reflected.push(value)); } //Set the elements inner html if it was set if (this._elementInnerHTML != null) { this.element.innerHTML = this._elementInnerHTML; } //Construct the children under this element for (var i = 0; i < this.children.length; i++) { this.children[i].construct(this.element); } //Set the elements value if there is one. if (this._elementValue != null) { if (this.element.hasAttribute("type") && this.element.getAttribute("type").toLowerCase().trim() == "checkbox") { this.element.checked = this._elementValue; } else if (this.element.hasAttribute("contenteditable") && this.element.getAttribute("contenteditable").toLowerCase().trim() == "true") { this.element.textContent = this._elementValue.toString(); } else { this.element.value = this._elementValue; } } //Setup a resize observer if needed if (this._resizeObserverCallback !== null) { this._resizeObserver = new ResizeObserver(e => this._resizeObserverCallback(e)); this._resizeObserver.observe(this.element); } //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); } } } /** * Deconstructs this template and cleans up all resources to make sure * there are no memory leaks. */ deconstruct() { //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; } //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; } //Stop observing resize events if we need to. if (this._resizeObserver != null) { this._resizeObserver.disconnect(); this._resizeObserver = null; this._resizeObserverCallback = null; } //Remove any refs if (this._refs) { //Pass null to our refs so that the reference is updated. this._refs.forEach(ref => ref(null)); 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 * @param {Function} converter Optional converter for the value if needed. * @ignore */ onClassChanged(oldValue, newValue) { var oldClasses = (oldValue != null && oldValue != "" ? oldValue.toString().split(" ") : []); var newClasses = (newValue != null && newValue != "" ? newValue.toString().split(" ") : []); if (this.element) { oldClasses.forEach((cl) => this.element.classList.remove(cl)); newClasses.forEach((cl) => this.element.classList.add(cl)); } //Remove the old values from the template, but only remove one copy. oldClasses.forEach((cl) => this._classes.splice(this._classes.indexOf(cl), 1)); //Add the new classes to the template. newClasses.forEach((cl) => this._classes.push(cl)); //For any classes that are missing on the element, add them. If we have duplicates this //can happen. this._classes.forEach((cl) => { if (!this.element.classList.contains(cl)) { this.element.classList.add(cl); } }); } /** * 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 * @ignore */ onAttributeChanged(oldValue, newValue, attributeName, converter) { if (converter !== null) { IgniteRenderingContext.push(); newValue = converter(newValue); IgniteRenderingContext.pop(); } if (this.element) { if (newValue == null || newValue == undefined) { this.element.removeAttribute(attributeName); } else { this.element.setAttribute(attributeName, newValue); } } this._attributes[attributeName] = newValue; } /** * Called when a value for this template was changed and needs to be updated on the template's element. * @param {any} oldValue * @param {any} newValue * @param {Function} converter * @ignore */ onValueChanged(oldValue, newValue, converter) { if (converter !== null) { IgniteRenderingContext.push(); newValue = converter(newValue); IgniteRenderingContext.pop(); } //Only update the elements value if it actually changed. //This is to prevent endless looping potentially. if (this.element) { if (this.element.hasAttribute("type") && this.element.getAttribute("type").toLowerCase().trim() == "checkbox") { if (this.element.checked != newValue) { this.element.checked = newValue; } } else if (this.element.hasAttribute("contenteditable") && this.element.getAttribute("contenteditable").toLowerCase().trim() == "true") { if (this.element.textContent != newValue.toString()) { this.element.textContent = newValue.toString(); } } else { if (this.element.value != newValue) { this.element.value = newValue; } } } this._elementValue = newValue; } /** * 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 * @param {Function} converter * @ignore */ onPropertyChanged(oldValue, newValue, propertyName, converter) { if (converter !== null) { IgniteRenderingContext.push(); newValue = converter(newValue); IgniteRenderingContext.pop(); } if (this.element) { //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); } this._properties[propertyName].value = newValue; } /** * Called when the inner html for this template was changed and needs to be updated * on the template's element. * @param {any} oldValue * @param {any} newValue * @param {Function} converter * @ignore */ onInnerHTMLChanged(oldValue, newValue, converter) { if (converter !== null) { IgniteRenderingContext.push(); newValue = converter(newValue); IgniteRenderingContext.pop(); } if (this.element) { this.element.innerHTML = newValue; } this._elementInnerHTML = newValue; } /** * 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 * @ignore */ 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; } } /** * Called when a event was changed and we need to update it. * @param {any} oldValue * @param {any} newValue * @param {any} eventName * @ignore */ 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); } //Remove the old value from the events this._events[eventName] = this._events[eventName].filter(ev => ev != oldValue); //Add the new value if it's needed if (newValue !== null && newValue !== undefined) { this._events[eventName].push(newValue); } } } /** * Called when a css value was changed and we need to update the styling. * @param {String} name * @param {any} newValue * @ignore */ onStyleChanged(name, newValue) { //Remove the ; from the value if there is one. if (newValue && typeof newValue === "string" && newValue.includes(";")) { newValue = newValue.replace(";", ""); } if (this.element) { this.element.style.setProperty(name, newValue, this._styles[name].priority); } this._styles[name].value = newValue; } } /** * An ignite template that can be used to construct a div element. */ class div extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("div", children); } } /** * An ignite template that can be used to construct a hyperlink element. */ class a extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("a", children); } } /** * An ignite template that can be used to construct a input element. */ class input extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("input", children); } } /** * An ignite template that can be used to construct a textarea element. */ class textarea extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("textarea", children); } } /** * An ignite template that can be used to construct a button element. */ class button extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("button", children); this.type("button"); } } /** * An ignite template that can be used to construct a h1 element. */ class h1 extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("h1", children); } } /** * An ignite template that can be used to construct a h2 element. */ class h2 extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("h2", children); } } /** * An ignite template that can be used to construct a h3 element. */ class h3 extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("h3", children); } } /** * An ignite template that can be used to construct a h4 element. */ class h4 extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("h4", children); } } /** * An ignite template that can be used to construct a h5 element. */ class h5 extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("h5", children); } } /** * An ignite template that can be used to construct a h6 element. */ class h6 extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("h6", children); } } /** * An ignite template that can be used to construct a hr element. */ class hr extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("hr", children); } } /** * An ignite template that can be used to construct a p element. */ class p extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("p", children); } } /** * An ignite template that can be used to construct a ul element. */ class ul extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("ul", children); } } /** * An ignite template that can be used to construct a li element. */ class li extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("li", children); } } /** * An ignite template that can be used to construct a span element. */ class span extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("span", children); } } /** * An ignite template that can be used to construct an i element. */ class i extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("i", children); } } /** * An ignite template that can be used to construct a table element. */ class table extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("table", children); } } /** * An ignite template that can be used to construct a td element. */ class td extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("td", children); } } /** * An ignite template that can be used to construct a th element. */ class th extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("th", children); } } /** * An ignite template that can be used to construct a tr element. */ class tr extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("tr", children); } } /** * An ignite template that can be used to construct a thead element. */ class thead extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("thead", children); } } /** * An ignite template that can be used to construct a tbody element. */ class tbody extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("tbody", children); } } /** * An ignite template that can be used to construct a br element. */ class br extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("br", children); } } /** * An ignite template that can be used to construct a img element. */ class img extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("img", children); } } /** * An ignite template that can be used to construct a label element. */ class label extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("label", children); } } /** * An ignite template that can be used to construct a select element. */ class select extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("select", children); } } /** * An ignite template that can be used to construct a option element. */ class option extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("option", children); } } /** * An ignite template that can be used to construct a script element. */ class script extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("script", children); } } /** * An ignite template that can be used to construct a form element. */ class form extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("form", children); } } /** * An ignite template that can be used to construct a progress element. */ class progress extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("progress", children); } } /** * An ignite template that can be used to construct a svg element. */ class svg extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("svg", children); this.tagNamespace = "http://www.w3.org/2000/svg"; } } /** * An ignite template that can be used to construct a g element. */ class g extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("g", children); this.tagNamespace = "http://www.w3.org/2000/svg"; } } /** * An ignite template that can be used to construct a path element. */ class path extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("path", children); this.tagNamespace = "http://www.w3.org/2000/svg"; } } /** * An ignite template that can be used to construct a circle element. */ class circle extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("circle", children); this.tagNamespace = "http://www.w3.org/2000/svg"; } } /** * An ignite template that can be used to construct a line element. */ class line extends IgniteTemplate { /** * @param {...String|Number|IgniteProperty|IgniteTemplate} children A series of children to be added to this template. */ constructor(...children) { super("line", children); this.tagNamespace = "http://www.w3.org/2000/svg"; } } /** * Html is a special template that can construct raw html or properties into the dom and automatically * update the dom if the property changes. * * @example * new html(`