BrickTracker/static/scripts/changer.js

125 lines
3.6 KiB
JavaScript
Raw Permalink Normal View History

// Generic state changer with visual feedback
class BrickChanger {
constructor(prefix, id, url, parent = undefined) {
this.prefix = prefix
this.html_element = document.getElementById(`${prefix}-${id}`);
this.html_status = document.getElementById(`status-${prefix}-${id}`);
this.html_type = this.html_element.getAttribute("type");
this.url = url;
if (parent) {
this.html_parent = document.getElementById(`${parent}-${id}`);
this.parent_dataset = `data-${prefix}`
}
// Register an event depending on the type
if (this.html_type == "checkbox") {
var listener = "change";
} else {
var listener = "click";
}
this.html_element.addEventListener(listener, ((changer) => (e) => {
changer.change();
})(this));
}
// Clean the status
status_clean() {
if (this.html_status) {
const to_remove = Array.from(
this.html_status.classList.values()
).filter(
(name) => name.startsWith('ri-') || name.startsWith('text-') || name.startsWith('bg-')
);
if (to_remove.length) {
this.html_status.classList.remove(...to_remove);
}
}
}
// Set the status to Error
status_error() {
if (this.html_status) {
this.status_clean();
this.html_status.classList.add("ri-alert-line", "text-danger");
}
}
// Set the status to OK
status_ok() {
if (this.html_status) {
this.status_clean();
this.html_status.classList.add("ri-checkbox-circle-line", "text-success");
}
}
// Set the status to Unknown
status_unknown() {
if (this.html_status) {
this.status_clean();
this.html_status.classList.add("ri-question-line", "text-warning");
}
}
async change() {
try {
this.status_unknown();
// Grab the value depending on the type
if (this.html_type == "checkbox") {
var value = this.html_element.checked;
} else {
var value = this.html_element.value;
}
const response = await fetch(this.url, {
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify({
name: this.prefix,
value: value,
})
});
if (!response.ok) {
throw new Error(`Response status: ${response.status}`);
}
const json = await response.json();
if ("error" in json) {
throw new Error(`Error received: ${json.error}`)
}
this.status_ok();
// Update the parent
if (this.html_parent) {
if (this.html_type == "checkbox") {
value = Number(value)
}
// Not going through dataset to avoid converting
this.html_parent.setAttribute(this.parent_dataset, value);
}
} catch (error) {
console.log(error.message);
this.status_error();
}
}
}
// Helper to setup the changer
const setup_changers = () => document.querySelectorAll("*[data-changer-id]").forEach(
el => new BrickChanger(
el.dataset.changerPrefix,
el.dataset.changerId,
el.dataset.changerUrl,
el.dataset.changerParent
)
);