2020-07-28 09:04:04 -07:00
|
|
|
import { IgniteProperty } from './ignite-html.js';
|
|
|
|
import { IgniteTemplate } from './ignite-template.js';
|
|
|
|
|
2020-08-21 21:42:10 -07:00
|
|
|
/**
|
2020-08-30 21:55:21 -07:00
|
|
|
* The outline of a Ignite Element that extends the html element
|
|
|
|
* and can be used to create custom components. Ignite Element's use an Ignite Template
|
|
|
|
* for the render function.
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
*
|
|
|
|
* class MainApp extends IgniteElement {
|
|
|
|
* constructor() {
|
|
|
|
* super();
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* get properties() {
|
|
|
|
* return {
|
|
|
|
* };
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* render() {
|
|
|
|
* return this.template
|
|
|
|
* .child(
|
|
|
|
* new h1(`<i class="fad fa-fire-alt" style="--fa-primary-color: #FFC107; --fa-secondary-color: #FF5722; --fa-secondary-opacity: 1.0;"></i> Ignite HTML`),
|
|
|
|
* new h4(`Adding more fire to the web.`)
|
|
|
|
* );
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* customElements.define("main-app", MainApp);
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
* //If you want to easily use an Ignite Element with templates see the following which can be added
|
|
|
|
* //to any component file
|
|
|
|
* class MainAppTemplate extends IgniteTemplate {
|
|
|
|
* constructor(...children) {
|
|
|
|
* super("main-app", children);
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* export {
|
|
|
|
* MainAppTemplate as MainApp
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
* //If a template was created for a Ignite Element (like the previous example) it can be used just like any other template:
|
|
|
|
* new MainApp()
|
2020-08-21 21:42:10 -07:00
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
class IgniteElement extends HTMLElement {
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
this.onDisconnected = null;
|
2020-07-28 09:04:04 -07:00
|
|
|
this.template = null;
|
2020-07-28 22:46:22 -07:00
|
|
|
this.elements = [];
|
2020-07-28 09:04:04 -07:00
|
|
|
this.createProperties();
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
2020-08-30 21:55:21 -07:00
|
|
|
* Returns an object with all the properties for this ignite element. If null or empty
|
|
|
|
* then no properties will be created. To change a property and read it's current value see below.
|
|
|
|
*
|
|
|
|
* @returns An object with properties that will be assigned to this ignite element.
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
* get properties() {
|
|
|
|
* return {
|
|
|
|
* show: false,
|
|
|
|
* title: "This is a title",
|
|
|
|
* items: [1, 2, 3]
|
|
|
|
* };
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
* //To change a property access it via this.PROP_NAME
|
|
|
|
* this.show = false;
|
|
|
|
*
|
|
|
|
* //To get a properties value access if via this.PROP_NAME
|
|
|
|
* console.log(this.title);
|
2020-07-28 22:23:49 -07:00
|
|
|
*/
|
|
|
|
get properties() {
|
2020-07-29 11:21:02 -07:00
|
|
|
return {};
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
2020-08-11 13:45:54 -07:00
|
|
|
/**
|
2020-08-30 21:55:21 -07:00
|
|
|
* Returns any CSS styling code for this ignite element. If this returns a non null
|
|
|
|
* value the CSS will auto be injected into the current HTML page once and reused for the life
|
|
|
|
* of the ignite element.
|
|
|
|
*
|
|
|
|
* @returns A string containing CSS code to be used with this ignite element.
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
* get styles() {
|
|
|
|
* return `
|
|
|
|
* h1 {
|
|
|
|
* color: black;
|
|
|
|
* }
|
|
|
|
* `;
|
|
|
|
* }
|
2020-08-11 13:45:54 -07:00
|
|
|
*/
|
|
|
|
get styles() {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Creates the getters/setters for properties in this
|
|
|
|
* ignite element and initializes everything.
|
2020-08-30 21:55:21 -07:00
|
|
|
* @ignore
|
2020-07-28 22:23:49 -07:00
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
createProperties() {
|
2020-07-28 22:23:49 -07:00
|
|
|
var props = this.properties;
|
2020-07-28 09:04:04 -07:00
|
|
|
|
2020-08-23 17:06:18 -07:00
|
|
|
if (props != null) {
|
|
|
|
var keys = Object.keys(props);
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
|
|
|
let prop = new IgniteProperty(props[keys[i]]);
|
|
|
|
this[`_${keys[i]}`] = prop;
|
2020-07-28 09:04:04 -07:00
|
|
|
|
2020-08-23 17:06:18 -07:00
|
|
|
((propName) => {
|
|
|
|
Object.defineProperty(this, propName, {
|
|
|
|
get: function () {
|
|
|
|
if (IgniteRenderingContext.rendering == false) {
|
|
|
|
return this[`_${propName}`].value;
|
|
|
|
} else {
|
|
|
|
return this[`_${propName}`];
|
|
|
|
}
|
|
|
|
},
|
2020-07-28 09:04:04 -07:00
|
|
|
|
2020-08-23 17:06:18 -07:00
|
|
|
set: function (value) {
|
|
|
|
this[`_${propName}`].value = value;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
})(keys[i]);
|
|
|
|
}
|
2020-07-29 11:21:02 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resets the properties for this element back to their original default
|
|
|
|
* value.
|
|
|
|
*/
|
|
|
|
resetProperties() {
|
|
|
|
var props = this.properties;
|
|
|
|
|
2020-08-23 17:06:18 -07:00
|
|
|
if (props != null) {
|
|
|
|
var keys = Object.keys(props);
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
|
|
|
this[keys[i]] = props[keys[i]];
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Setups this ignite element and constructs it's template when
|
|
|
|
* this function is called by the DOM upon this element being created somewhere.
|
2020-08-30 21:55:21 -07:00
|
|
|
* @ignore
|
2020-07-28 22:23:49 -07:00
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
connectedCallback() {
|
2020-08-11 13:45:54 -07:00
|
|
|
//See if a styling sheet has been created for this element if it's needed.
|
|
|
|
if (this.styles !== null && this.styles !== "") {
|
|
|
|
if (document.getElementsByClassName(`_${this.tagName}_styling_`).length == 0) {
|
|
|
|
var styleEl = document.createElement("style");
|
|
|
|
styleEl.classList.add(`_${this.tagName}_styling_`);
|
|
|
|
styleEl.innerHTML = this.styles;
|
|
|
|
document.body.prepend(styleEl);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-29 11:21:02 -07:00
|
|
|
//If we don't already have a template, make sure we create one,
|
|
|
|
//this can happen if this element was constructed in the DOM instead of within a template.
|
|
|
|
if (!this.template) {
|
|
|
|
this.template = new IgniteTemplate();
|
|
|
|
this.template.element = this;
|
|
|
|
this.template.tagName = this.tagName;
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
|
2020-07-28 22:46:22 -07:00
|
|
|
//Add any childNodes we have to the elements list within this
|
|
|
|
this.childNodes.forEach((item) => this.elements.push(item));
|
|
|
|
|
2020-08-21 21:42:10 -07:00
|
|
|
//Enter a rendering context so properties don't expose their values until we are done.
|
|
|
|
IgniteRenderingContext.enter();
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
//Make sure the render template is our template, if not, add it as a child.
|
|
|
|
var renderTemplate = this.render();
|
|
|
|
if (renderTemplate !== this.template && renderTemplate) {
|
|
|
|
this.template.child(renderTemplate);
|
|
|
|
} else if (!renderTemplate) {
|
2020-08-23 11:50:31 -07:00
|
|
|
throw `RenderTemplate was null for template: ${this.tagName}. Is render() returning null or not returning a template?`;
|
2020-07-28 22:23:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
//Construct our template.
|
2020-08-21 21:42:10 -07:00
|
|
|
try {
|
|
|
|
this.template.construct(this.parentElement);
|
|
|
|
} catch (error) {
|
|
|
|
console.error(error);
|
|
|
|
}
|
2020-07-29 12:32:16 -07:00
|
|
|
|
2020-08-21 21:42:10 -07:00
|
|
|
//Leave the rendering context.
|
|
|
|
IgniteRenderingContext.leave();
|
2020-07-28 22:23:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Cleanups this element and deconstructs everything when this element is removed from
|
|
|
|
* the DOM.
|
2020-08-30 21:55:21 -07:00
|
|
|
* @ignore
|
2020-07-28 22:23:49 -07:00
|
|
|
*/
|
|
|
|
disconnectedCallback() {
|
|
|
|
//If we still have a reference to our template, deconstruct it.
|
|
|
|
if (this.template) {
|
|
|
|
this.template.deconstruct();
|
|
|
|
}
|
|
|
|
|
|
|
|
//If we have a onDisconnected callback, call it and then remove the reference.
|
|
|
|
if (this.onDisconnected) {
|
|
|
|
this.onDisconnected();
|
|
|
|
this.onDisconnected = null;
|
|
|
|
}
|
2020-07-28 09:04:04 -07:00
|
|
|
}
|
|
|
|
|
2020-07-28 22:23:49 -07:00
|
|
|
/**
|
|
|
|
* Returns the template to be rendered for this element.
|
2020-08-30 21:55:21 -07:00
|
|
|
*
|
|
|
|
* @returns An ignite template to be used to construct this ignite element.
|
|
|
|
*
|
|
|
|
* @see this.template is automatically created for each IgniteElement and must be used in the render() function.
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
* render() {
|
|
|
|
* return this.template
|
|
|
|
* .child(
|
|
|
|
* new h1(`<i class="fad fa-fire-alt" style="--fa-primary-color: #FFC107; --fa-secondary-color: #FF5722; --fa-secondary-opacity: 1.0;"></i> Ignite HTML`),
|
|
|
|
* new h4(`Adding more fire to the web.`)
|
|
|
|
* );
|
|
|
|
* }
|
2020-07-28 22:23:49 -07:00
|
|
|
*/
|
2020-07-28 09:04:04 -07:00
|
|
|
render() {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export {
|
|
|
|
IgniteElement
|
|
|
|
};
|