2022-12-30 16:54:38 -05:00
|
|
|
class InterfaceElement {
|
2022-12-30 16:18:53 -05:00
|
|
|
constructor({id, tag}) {
|
|
|
|
this.id = id;
|
|
|
|
if (tag !== undefined) {
|
|
|
|
this.handle = tag;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-30 14:32:27 -05:00
|
|
|
#hidden = false;
|
|
|
|
get hidden() {
|
|
|
|
return this.#hidden;
|
|
|
|
}
|
|
|
|
set hidden(x) {
|
|
|
|
this.#hidden = x;
|
|
|
|
|
|
|
|
this.handle.hidden = this.hidden;
|
|
|
|
if (this.label !== undefined) {
|
|
|
|
this.label.hidden = this.hidden;
|
|
|
|
}
|
2022-12-30 16:18:53 -05:00
|
|
|
|
|
|
|
if (this.hidden === true) this.clearAlerts();
|
2022-12-30 14:32:27 -05:00
|
|
|
}
|
2022-12-30 16:18:53 -05:00
|
|
|
|
2022-12-30 14:32:27 -05:00
|
|
|
}
|
|
|
|
|
2022-12-30 16:54:38 -05:00
|
|
|
class Form extends InterfaceElement {
|
2022-12-30 16:18:53 -05:00
|
|
|
constructor({id, tag}) {
|
|
|
|
super(id, tag);
|
2022-12-29 19:22:50 -05:00
|
|
|
|
2022-12-30 16:18:53 -05:00
|
|
|
if (tag === undefined) {
|
2022-12-29 19:22:50 -05:00
|
|
|
this.handle = document.createElement("div");
|
|
|
|
}
|
|
|
|
|
2022-12-29 21:58:19 -05:00
|
|
|
this.elements = new Map();
|
2022-12-30 16:18:53 -05:00
|
|
|
|
|
|
|
this.clearAlerts = this.clearAlerts.bind(this);
|
2022-12-29 21:58:19 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#advanced = false;
|
|
|
|
get advanced() {
|
|
|
|
return this.#advanced;
|
|
|
|
}
|
|
|
|
set advanced(x) {
|
|
|
|
this.#advanced = x;
|
|
|
|
for (const [id, element] of this.elements.entries()) {
|
|
|
|
if (element.advanced === true) {
|
|
|
|
if (this.advanced === true) {
|
|
|
|
element.hidden = false;
|
|
|
|
} else {
|
|
|
|
element.hidden = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-12-29 19:22:50 -05:00
|
|
|
}
|
2022-12-30 16:18:53 -05:00
|
|
|
|
|
|
|
clearAlerts() {
|
|
|
|
for (const [id, element] of this.elements.entries()) {
|
|
|
|
element.clearAlerts();
|
|
|
|
}
|
|
|
|
}
|
2022-12-29 19:22:50 -05:00
|
|
|
}
|
|
|
|
|
2022-12-29 21:58:19 -05:00
|
|
|
function b64ToBuf (b64) {
|
|
|
|
let ascii = atob(b64);
|
|
|
|
let buf = new ArrayBuffer(ascii.length);
|
|
|
|
let bytes = new Uint8Array(buf);
|
|
|
|
for (var i = 0; i < ascii.length; i++) {
|
|
|
|
bytes[i] = ascii.charCodeAt(i);
|
|
|
|
}
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
function bufToB64 (buf) {
|
|
|
|
let bytes = new Uint8Array(buf);
|
|
|
|
let ascii = ''
|
|
|
|
for (var i = 0; i < bytes.byteLength; i++) {
|
|
|
|
ascii += String.fromCharCode(bytes[i]);
|
|
|
|
}
|
|
|
|
return btoa(ascii);
|
|
|
|
}
|
|
|
|
|
2022-12-30 16:54:38 -05:00
|
|
|
class FormElement extends InterfaceElement {
|
2022-12-29 19:22:50 -05:00
|
|
|
#enabled = true;
|
|
|
|
get enabled() {
|
|
|
|
return this.#enabled;
|
|
|
|
}
|
|
|
|
set enabled(x) {
|
|
|
|
this.#enabled = x;
|
|
|
|
this.handle.disabled = !this.#enabled;
|
|
|
|
}
|
|
|
|
|
2022-12-30 16:18:53 -05:00
|
|
|
constructor({id, type, form, tag, label="", dataType="plaintext", advanced=false, enabled=true}) {
|
|
|
|
super(id);
|
2022-12-29 19:22:50 -05:00
|
|
|
this.id = id;
|
2022-12-29 21:58:19 -05:00
|
|
|
|
2022-12-29 19:22:50 -05:00
|
|
|
this.advanced = advanced;
|
|
|
|
this.type = type;
|
2022-12-29 21:58:19 -05:00
|
|
|
|
2022-12-30 16:18:53 -05:00
|
|
|
this.clearAlerts = this.clearAlerts.bind(this);
|
|
|
|
|
2022-12-29 19:22:50 -05:00
|
|
|
switch (type) {
|
|
|
|
case "textbox":
|
|
|
|
this.handle = document.createElement("input");
|
|
|
|
break;
|
|
|
|
case "password":
|
|
|
|
this.handle = document.createElement("input");
|
|
|
|
this.handle.setAttribute("type", "password");
|
|
|
|
break;
|
|
|
|
case "textarea":
|
|
|
|
this.handle = document.createElement("textarea");
|
|
|
|
break;
|
|
|
|
case "button":
|
|
|
|
this.handle = document.createElement("button");
|
2022-12-30 16:18:53 -05:00
|
|
|
this.handle.innerHTML = label;
|
|
|
|
label = "";
|
2022-12-29 21:58:19 -05:00
|
|
|
dataType = "none"
|
2022-12-29 19:22:50 -05:00
|
|
|
break;
|
|
|
|
case "output":
|
|
|
|
this.handle = document.createElement("textarea");
|
|
|
|
this.handle.setAttribute("readonly", true);
|
|
|
|
break;
|
|
|
|
default:
|
2022-12-29 21:58:19 -05:00
|
|
|
throw `Unknown input type: ${type}`;
|
2022-12-29 19:22:50 -05:00
|
|
|
}
|
|
|
|
|
2022-12-30 16:18:53 -05:00
|
|
|
if (label !== "") {
|
|
|
|
this.label = document.createElement("label");
|
|
|
|
this.label.innerHTML = label;
|
|
|
|
}
|
|
|
|
|
2022-12-29 21:58:19 -05:00
|
|
|
this.dataType = dataType;
|
|
|
|
|
2022-12-29 19:22:50 -05:00
|
|
|
this.enabled = enabled;
|
2022-12-29 21:58:19 -05:00
|
|
|
this.handle.id = this.id;
|
|
|
|
|
|
|
|
if (this.advanced === true) this.hidden = true;
|
2022-12-29 19:22:50 -05:00
|
|
|
|
|
|
|
if (form !== undefined) {
|
2022-12-29 21:58:19 -05:00
|
|
|
this.form = form;
|
2022-12-29 19:22:50 -05:00
|
|
|
if (this.label !== undefined) {
|
2022-12-29 21:58:19 -05:00
|
|
|
this.label.setAttribute("for", this.id);
|
2022-12-29 19:22:50 -05:00
|
|
|
form.handle.appendChild(this.label);
|
|
|
|
}
|
|
|
|
form.handle.appendChild(this.handle);
|
2022-12-29 21:58:19 -05:00
|
|
|
form.elements.set(id, this);
|
|
|
|
|
|
|
|
if (this.advanced === true) this.hidden = !form.advanced;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// plaintext is string data
|
|
|
|
// b64 is raw ArrayBuffer data
|
|
|
|
// or none, which gives undefined
|
|
|
|
#dataType = "none";
|
|
|
|
get dataType() {
|
|
|
|
return this.#dataType;
|
|
|
|
}
|
|
|
|
set dataType(x) {
|
|
|
|
function err(type, x) {
|
|
|
|
throw `'${type}' element can not support '${x}' data type`;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (x) {
|
|
|
|
case "plaintext":
|
|
|
|
case "b64":
|
|
|
|
let valid = ["textbox", "password", "textarea", "output"];
|
|
|
|
if (!valid.includes(this.type)) err(this.type, x);
|
|
|
|
break;
|
|
|
|
case "none":
|
|
|
|
if (this.type !== "button") err(this.type, x);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw `Unknown data type: ${x}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.#dataType = x;
|
|
|
|
}
|
|
|
|
|
|
|
|
get value() {
|
2022-12-30 17:31:28 -05:00
|
|
|
this.clearAlerts();
|
2022-12-29 21:58:19 -05:00
|
|
|
switch (this.dataType) {
|
|
|
|
case "plaintext":
|
|
|
|
return this.handle.value;
|
|
|
|
case "b64":
|
2022-12-30 14:32:27 -05:00
|
|
|
try {
|
|
|
|
return b64ToBuf(this.handle.value);
|
|
|
|
} catch (e) {
|
2022-12-30 17:31:28 -05:00
|
|
|
this.alertBox("alert-error", "Invalid base64 value.");
|
|
|
|
return;
|
2022-12-30 14:32:27 -05:00
|
|
|
}
|
2022-12-29 21:58:19 -05:00
|
|
|
case "none":
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
set value(x) {
|
|
|
|
switch (this.dataType) {
|
|
|
|
case "plaintext":
|
|
|
|
this.handle.value = x;
|
|
|
|
case "b64":
|
|
|
|
this.handle.value = bufToB64(x);
|
2022-12-29 19:22:50 -05:00
|
|
|
}
|
|
|
|
}
|
2022-12-30 16:18:53 -05:00
|
|
|
|
|
|
|
alerts = [];
|
|
|
|
alertBox(type, message, title) {
|
|
|
|
// type is alert-error or alert-info
|
|
|
|
|
|
|
|
if (this.handle === undefined) {
|
|
|
|
throw `can not add alert for '${this.id}': still undefined`;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.hidden === true) {
|
|
|
|
throw `can not add alert for '${this.id}': hidden`;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (title === undefined) {
|
|
|
|
switch (type) {
|
|
|
|
case "alert-info":
|
|
|
|
title = "Info: ";
|
|
|
|
break;
|
|
|
|
case "alert-error":
|
|
|
|
title = "Error: ";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
title = "";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let box = document.createElement("div");
|
|
|
|
box.classList.add(type);
|
|
|
|
box.classList.add("alert");
|
|
|
|
box.innerHTML = message;
|
|
|
|
|
|
|
|
if (title !== "") {
|
|
|
|
let titleTag = document.createElement("strong");
|
|
|
|
titleTag.innerHTML = title;
|
|
|
|
box.prepend(titleTag);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.handle.after(box);
|
|
|
|
this.alerts.push(box);
|
|
|
|
}
|
|
|
|
clearAlerts() {
|
|
|
|
for (const box of this.alerts) {
|
|
|
|
box.remove();
|
|
|
|
}
|
|
|
|
this.alerts = [];
|
|
|
|
}
|
2022-12-29 19:22:50 -05:00
|
|
|
}
|