ignite-html-tabs/ignite-html-tabs.js

187 lines
6.8 KiB
JavaScript

import { IgniteCallback } from "../ignite-html/ignite-html.js";
import { IgniteTemplate, slot, div, html } from "../ignite-html/ignite-template.js";
/**
* Makes this IgniteTemplate a tab that will become active or inactive based on the state of the tab.
* @param {String|String[]} name The name of the tab. If an array is passed, this tab can activate is more than one tab is active.
* @param {String} group The tab group to put this tab under, by default this is null.
* @param {Boolean} active Whether or not this tab is active by default.
* @param {Function} showCallback Called when this tab is changed to the shown state.
* @param {Function} hideCallback Called when this tab is changed to hidden state.
* @returns {IgniteTemplate} Returns this ignite template.
*/
IgniteTemplate.prototype.tab = function(name, group = null, active = false, showCallback = null, hideCallback = null) {
//Add the tab class to this element.
this.class("tab")
//If active, make this tab active.
if (active) {
this.class("active");
Tabs.change(name, group);
}
//What we do is create an IgniteCallback that will get called when
//the tabchange event is raised. Upon disconnecting the callback we remove
//the event listener.
var callback = new IgniteCallback(event => {
if (this.element) {
if (event.group == group && Array.isArray(name) && name.includes(event.name)) {
if (!this.element.classList.contains("active")) {
this.element.classList.add("active");
//Invoke the show callback if there is one.
if (showCallback) {
showCallback();
}
}
} else if (event.name == name && event.group == group) {
if (!this.element.classList.contains("active")) {
this.element.classList.add("active");
//Invoke the show callback if there is one.
if (showCallback) {
showCallback();
}
}
} else if (event.group == group) {
this.element.classList.remove("active");
//Invoke the hide callback if there is one.
if (hideCallback) {
hideCallback();
}
}
}
}, () => {
window.removeEventListener("tabchange", callback.callback);
});
//Register our callback to the tabchange event.
window.addEventListener("tabchange", callback.callback);
//Add this callback to our template so that upon deconstruction our callback is destroyed correctly.
this._callbacks.push(callback);
return this;
}
/**
* Makes this IgniteTemplate a tab button that will become active or inactive based on the current tab and change tabs when clicked.
* @param {String} name The name of the tab that this button will set to active.
* @param {String} group The tab group the tab belongs to.
* @param {Boolean} active Whether or not this button is active, indicating the tab is active.
* @returns {IgniteTemplate} Returns this ignite template.
*/
IgniteTemplate.prototype.tabButton = function(name, group = null, active = false) {
//Add the tab-btn class to this element.
this.class("tab-btn");
//If active, make the button active.
if (active) {
this.class("active");
}
//What we do is create an IgniteCallback that will get called when
//the tabchange event is raised. Upon disconnecting the callback we remove
//the event listener.
var callback = new IgniteCallback(event => {
if (this.element) {
//See if the group matches.
if (event.group == group) {
if (event.name == name) {
//Mark this button as active.
if (!this.element.classList.contains("active")) {
this.element.classList.add("active");
}
} else {
//Mark this button as inactive.
this.element.classList.remove("active");
}
}
}
}, () => {
window.removeEventListener("tabchange", callback.callback);
});
//Register our callback to the tabchange event.
window.addEventListener("tabchange", callback.callback);
//Add this callback to our template so that upon deconstruction our callback is destroyed correctly.
this._callbacks.push(callback);
//Setup our on click to change tabs.
this.onClick(() => Tabs.change(name, group));
return this;
}
/**
* A api class to help easily work with tabs.
*/
class Tabs {
/**
* Changes the given tab to active.
* @param {String} name The name of the tab to change to.
* @param {String} group The name of the group this tab belongs to if any. Default is null.
* @param {Boolean} active Whether or not to make this tab active. Default is true.
*/
static change(name, group = null, active = true) {
//Init the groups if needed.
if (Tabs.groups[group] == undefined) {
Tabs.groups[group] = { };
}
//Init the tabs if needed.
if (group == null && Tabs.tabs[name] == undefined) {
Tabs.tabs[name] = false;
}
//If this tab is supposed to be active make that happen.
if (active) {
if (group) {
//Reset all tabs
Object.keys(Tabs.groups[group]).forEach(key => Tabs.groups[group][key] = false);
//Set this tab to active.
Tabs.groups[group][name] = true;
} else {
//Reset all tabs
Object.keys(Tabs.tabs).forEach(key => Tabs.tabs[key] = false);
//Set this tab to active.
Tabs.groups[group][name] = true;
}
} else {
//Set this tab to inactive.
if (group) {
Tabs.groups[group][name] = false;
} else {
Tabs.tabs[name] = false;
}
}
//Create a tab change event and fire it.
var changeEvent = new Event("tabchange");
changeEvent.name = (active ? name : "");
changeEvent.group = group;
window.dispatchEvent(changeEvent);
}
/**
* Returns whether or not the given tab is active.
* @param {String} name The name of the tab.
* @param {String} group The name of the group this tab belongs to. Default is null.
*/
static isActive(name, group = null) {
return ((group && Tabs.groups[group] != undefined && Tabs.groups[group][name]) || (group == null && Tabs.tabs[name] != undefined && Tabs.tabs[name])) ? true : false;
}
}
Tabs.groups = { };
Tabs.tabs = { };
window.Tabs = Tabs;
export {
Tabs
}