human/demo/menu.js

244 lines
9.9 KiB
JavaScript
Raw Normal View History

2020-10-18 18:12:09 +02:00
let instance = 0;
2020-10-18 02:59:43 +02:00
const css = `
2020-10-30 15:29:50 +01:00
.menu { position: fixed; top: 0rem; right: 0; width: fit-content; padding: 0 0.8rem 0 0.8rem; line-height: 1.8rem; z-index: 10; max-height: calc(100% - 4rem); box-shadow: 0 0 8px dimgrey; background: darkslategray; border-radius: 1rem; }
2020-10-18 18:12:09 +02:00
.menu:hover { box-shadow: 0 0 8px lightgrey; }
.menu-container { display: block; max-height: 100vh; }
.menu-container-fadeout { max-height: 0; overflow: hidden; transition: max-height, 0.5s ease; }
.menu-container-fadein { max-height: 100vh; overflow: hidden; transition: max-height, 0.5s ease; }
.menu-item { display: flex; white-space: nowrap; background: darkslategray; padding: 0.2rem; width: max-content; }
2020-10-18 14:07:45 +02:00
.menu-title { text-align: right; cursor: pointer; }
2020-10-30 15:23:49 +01:00
.menu-hr { margin: 0.2rem; border: 1px solid rgba(0, 0, 0, 0.5); }
2020-10-18 14:07:45 +02:00
.menu-label { padding: 0; }
2020-10-18 02:59:43 +02:00
2020-10-30 15:23:49 +01:00
.menu-list { margin-right: 0.8rem; }
select:focus { outline: none; }
.menu-list-item { background: black; color: white; border: none; padding: 0.2rem; font-family: inherit; font-variant: inherit; border-radius: 1rem; }
2020-10-18 02:59:43 +02:00
.menu-chart-title { align-items: center; }
.menu-chart-canvas { background: transparent; height: 40px; width: 180px; margin: 0.2rem 0.2rem 0.2rem 1rem; }
2020-10-30 15:29:50 +01:00
.menu-button { border: 0; background: lightblue; width: -webkit-fill-available; padding: 8px; margin: 8px 0 8px 0; cursor: pointer; box-shadow: 4px 4px 4px 0 dimgrey; border-radius: 1rem; justify-content: center; }
2020-10-18 14:07:45 +02:00
.menu-button:hover { background: lightgreen; box-shadow: 4px 4px 4px 0 black; }
.menu-button:focus { outline: none; }
2020-10-18 02:59:43 +02:00
.menu-checkbox { width: 2.8rem; height: 1rem; background: black; margin: 0.5rem 0.8rem 0 0; position: relative; border-radius: 1rem; }
.menu-checkbox:after { content: 'OFF'; color: lightcoral; position: absolute; right: 0.2rem; top: -0.4rem; font-weight: 800; font-size: 0.5rem; }
.menu-checkbox:before { content: 'ON'; color: lightgreen; position: absolute; left: 0.3rem; top: -0.4rem; font-weight: 800; font-size: 0.5rem; }
2020-10-18 14:07:45 +02:00
.menu-checkbox-label { width: 1.3rem; height: 0.8rem; cursor: pointer; position: absolute; top: 0.1rem; left: 0.1rem; z-index: 1; background: lightcoral; border-radius: 1rem; transition: left 0.6s ease; }
2020-10-18 02:59:43 +02:00
input[type=checkbox] { visibility: hidden; }
input[type=checkbox]:checked + label { left: 1.4rem; background: lightgreen; }
.menu-range { margin: 0 0.8rem 0 0; width: 5rem; background: transparent; color: lightblue; }
.menu-range:before { content: attr(value); color: white; margin: 0 0.4rem 0 0; font-weight: 800; font-size: 0.6rem; position: relative; top: 0.3rem; }
input[type=range] { -webkit-appearance: none; }
input[type=range]::-webkit-slider-runnable-track { width: 100%; height: 1rem; cursor: pointer; background: black; border-radius: 1rem; border: 1px; }
input[type=range]::-webkit-slider-thumb { border: 1px solid #000000; margin-top: 0.05rem; height: 0.9rem; width: 1.5rem; border-radius: 1rem; background: lightblue; cursor: pointer; -webkit-appearance: none; }
`;
function createCSS() {
const el = document.createElement('style');
el.innerHTML = css;
document.getElementsByTagName('head')[0].appendChild(el);
}
2020-10-18 18:12:09 +02:00
function createMenu(parent, title, position = { top: null, left: null, bottom: null, right: null }) {
2020-10-18 02:59:43 +02:00
const el = document.createElement('div');
2020-10-18 18:12:09 +02:00
el.id = `menu-${instance}`;
el.className = 'menu';
if (position) {
if (position.top) el.style.top = position.top;
if (position.bottom) el.style.bottom = position.bottom;
if (position.left) el.style.left = position.left;
if (position.right) el.style.right = position.right;
}
const elContainer = document.createElement('div');
elContainer.id = `menu-container-${instance}`;
elContainer.className = 'menu-container menu-container-fadein';
const elTitle = document.createElement('div');
elTitle.className = 'menu-title';
elTitle.id = `menu-title-${instance}`;
elTitle.innerHTML = title;
el.appendChild(elTitle);
elTitle.addEventListener('click', () => {
elContainer.classList.toggle('menu-container-fadeout');
elContainer.classList.toggle('menu-container-fadein');
});
el.appendChild(elContainer);
2020-10-18 02:59:43 +02:00
if (typeof parent === 'object') parent.appendChild(el);
else document.getElementById(parent).appendChild(el);
2020-10-18 18:12:09 +02:00
return [el, elContainer];
2020-10-18 02:59:43 +02:00
}
class Menu {
2020-10-18 18:12:09 +02:00
constructor(parent, title, position) {
2020-10-18 02:59:43 +02:00
createCSS();
2020-10-18 18:12:09 +02:00
[this.menu, this.container] = createMenu(parent, title, position);
this.id = 0;
this.instance = instance;
instance++;
2020-10-18 02:59:43 +02:00
this._maxFPS = 0;
2020-10-18 14:07:45 +02:00
this.hidden = 0;
2020-10-18 02:59:43 +02:00
}
get newID() {
2020-10-18 18:12:09 +02:00
this.id++;
return `menu-${this.instance}-${this.id}`;
2020-10-18 02:59:43 +02:00
}
get ID() {
2020-10-18 18:12:09 +02:00
return `menu-${this.instance}-${this.id}`;
2020-10-18 02:59:43 +02:00
}
get width() {
return this.menu.offsetWidth;
}
get height() {
return this.menu.offsetHeight;
}
2020-10-18 14:07:45 +02:00
async addTitle(title) {
const el = document.createElement('div');
el.className = 'menu-title';
el.id = this.newID;
el.innerHTML = title;
this.menu.appendChild(el);
el.addEventListener('click', () => {
this.hidden = !this.hidden;
const all = document.getElementsByClassName('menu');
2020-10-18 18:12:09 +02:00
for (const item of all) {
item.style.display = this.hidden ? 'none' : 'flex';
}
2020-10-18 14:07:45 +02:00
});
}
2020-10-18 02:59:43 +02:00
async addLabel(title) {
const el = document.createElement('div');
2020-10-18 18:12:09 +02:00
el.className = 'menu-item menu-label';
2020-10-18 02:59:43 +02:00
el.id = this.newID;
el.innerHTML = title;
2020-10-18 18:12:09 +02:00
this.container.appendChild(el);
2020-10-18 02:59:43 +02:00
}
async addBool(title, object, variable, callback) {
const el = document.createElement('div');
2020-10-18 18:12:09 +02:00
el.className = 'menu-item';
2020-10-18 14:07:45 +02:00
el.innerHTML = `<div class="menu-checkbox"><input class="menu-checkbox" type="checkbox" id="${this.newID}" ${object[variable] ? 'checked' : ''}/><label class="menu-checkbox-label" for="${this.ID}"></label></div>${title}`;
2020-10-18 18:12:09 +02:00
this.container.appendChild(el);
2020-10-18 14:07:45 +02:00
el.addEventListener('change', (evt) => {
2020-10-18 02:59:43 +02:00
object[variable] = evt.target.checked;
if (callback) callback(evt.target.checked);
});
}
2020-10-30 15:23:49 +01:00
async addList(title, items, selected, callback) {
const el = document.createElement('div');
el.className = 'menu-item';
let options = '';
for (const item of items) {
const def = item === selected ? 'selected' : '';
options += `<option value="${item}" ${def}>${item}</option>`;
}
el.innerHTML = `<div class="menu-list"><select name="${this.ID}" class="menu-list-item">${options}</select><label for="${this.ID}"></label></div>${title}`;
el.style.fontFamily = document.body.style.fontFamily;
el.style.fontSize = document.body.style.fontSize;
el.style.fontVariant = document.body.style.fontVariant;
this.container.appendChild(el);
el.addEventListener('change', (evt) => {
if (callback) callback(items[evt.target.selectedIndex]);
});
}
2020-10-18 02:59:43 +02:00
async addRange(title, object, variable, min, max, step, callback) {
const el = document.createElement('div');
2020-10-18 18:12:09 +02:00
el.className = 'menu-item';
2020-10-27 19:20:21 +01:00
const arr = Array.isArray(variable);
el.innerHTML = `<input class="menu-range" type="range" id="${this.newID}" min=${min} max=${max} step=${step} value=${arr ? object[variable[0]] : object[variable]}>${title}`;
2020-10-18 18:12:09 +02:00
this.container.appendChild(el);
2020-10-18 14:07:45 +02:00
el.addEventListener('change', (evt) => {
2020-10-27 19:20:21 +01:00
const int = parseInt(evt.target.value) === parseFloat(evt.target.value);
const val = Array.isArray(variable) ? variable : [variable];
for (const item of val) {
object[item] = int ? parseInt(evt.target.value) : parseFloat(evt.target.value);
}
2020-10-18 02:59:43 +02:00
evt.target.setAttribute('value', evt.target.value);
if (callback) callback(evt.target.value);
});
}
async addHTML(html) {
const el = document.createElement('div');
2020-10-18 18:12:09 +02:00
el.className = 'menu-item';
2020-10-18 02:59:43 +02:00
el.id = this.newID;
if (html) el.innerHTML = html;
2020-10-18 18:12:09 +02:00
this.container.appendChild(el);
2020-10-18 02:59:43 +02:00
}
async addButton(titleOn, titleOff, callback) {
const el = document.createElement('button');
2020-10-18 18:12:09 +02:00
el.className = 'menu-item menu-button';
2020-10-18 14:07:45 +02:00
el.style.fontFamily = document.body.style.fontFamily;
el.style.fontSize = document.body.style.fontSize;
el.style.fontVariant = document.body.style.fontVariant;
2020-10-18 02:59:43 +02:00
el.type = 'button';
el.id = this.newID;
el.innerText = titleOn;
2020-10-18 18:12:09 +02:00
this.container.appendChild(el);
2020-10-18 14:07:45 +02:00
el.addEventListener('click', () => {
2020-10-18 02:59:43 +02:00
if (el.innerText === titleOn) el.innerText = titleOff;
else el.innerText = titleOn;
if (callback) callback(el.innerText !== titleOn);
});
}
async addValue(title, val) {
const el = document.createElement('div');
2020-10-18 18:12:09 +02:00
el.className = 'menu-item';
2020-10-18 02:59:43 +02:00
el.id = title;
el.innerText = `${title}: ${val}`;
2020-10-18 18:12:09 +02:00
this.contaner.appendChild(el);
2020-10-18 02:59:43 +02:00
}
// eslint-disable-next-line class-methods-use-this
async updateValue(title, val) {
const el = document.getElementById(title);
el.innerText = `${title}: ${val}`;
}
async addChart(title, id) {
const el = document.createElement('div');
2020-10-18 18:12:09 +02:00
el.className = 'menu-item menu-chart-title';
2020-10-18 02:59:43 +02:00
el.id = this.newID;
el.innerHTML = `${title}<canvas id="menu-canvas-${id}" class="menu-chart-canvas" width="180px" height="40px"></canvas>`;
2020-10-18 18:12:09 +02:00
this.container.appendChild(el);
2020-10-18 02:59:43 +02:00
}
// eslint-disable-next-line class-methods-use-this
async updateChart(id, values) {
if (!values || (values.length === 0)) return;
const canvas = document.getElementById(`menu-canvas-${id}`);
if (!canvas) return;
const ctx = canvas.getContext('2d');
ctx.fillStyle = 'darkslategray';
ctx.fillRect(0, 0, canvas.width, canvas.height);
const width = canvas.width / values.length;
const max = 1 + Math.max(...values);
const height = canvas.height / max;
for (const i in values) {
const gradient = ctx.createLinearGradient(0, (max - values[i]) * height, 0, 0);
gradient.addColorStop(0.1, 'lightblue');
gradient.addColorStop(0.4, 'darkslategray');
ctx.fillStyle = gradient;
ctx.fillRect(i * width, 0, width - 4, canvas.height);
ctx.fillStyle = 'black';
ctx.font = '12px "Segoe UI"';
ctx.fillText(Math.round(values[i]), i * width, canvas.height - 2, width);
}
}
}
export default Menu;