2022-11-22 22:24:55 +00:00
|
|
|
function makeDraggable(element) {
|
2022-11-21 11:24:35 +00:00
|
|
|
const startbb = element.getBoundingClientRect();
|
|
|
|
let dragging = false;
|
|
|
|
let offset = {x: 0, y: 0};
|
|
|
|
|
|
|
|
element.style.top = startbb.y + "px";
|
|
|
|
element.style.left = startbb.x + "px";
|
|
|
|
|
|
|
|
mouse.listen.window.left.onpaintstart.on((evn) => {
|
|
|
|
if (
|
|
|
|
element.contains(evn.target) &&
|
|
|
|
evn.target.classList.contains("draggable")
|
|
|
|
) {
|
|
|
|
const bb = element.getBoundingClientRect();
|
|
|
|
offset.x = evn.x - bb.x;
|
|
|
|
offset.y = evn.y - bb.y;
|
|
|
|
dragging = true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
mouse.listen.window.left.onpaint.on((evn) => {
|
|
|
|
if (dragging) {
|
|
|
|
element.style.top = evn.y - offset.y + "px";
|
|
|
|
element.style.left = evn.x - offset.x + "px";
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
mouse.listen.window.left.onpaintend.on((evn) => {
|
|
|
|
dragging = false;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-11-22 22:24:55 +00:00
|
|
|
document.querySelectorAll(".floating-window").forEach((w) => {
|
|
|
|
makeDraggable(w);
|
|
|
|
});
|
2022-11-21 11:24:35 +00:00
|
|
|
|
2022-11-19 21:48:12 +00:00
|
|
|
var coll = document.getElementsByClassName("collapsible");
|
|
|
|
for (var i = 0; i < coll.length; i++) {
|
2022-11-25 23:27:44 +00:00
|
|
|
let active = false;
|
2022-11-20 23:10:03 +00:00
|
|
|
coll[i].addEventListener("click", function () {
|
|
|
|
var content = this.nextElementSibling;
|
2022-11-25 23:27:44 +00:00
|
|
|
|
|
|
|
if (!active) {
|
|
|
|
this.classList.add("active");
|
|
|
|
content.classList.add("active");
|
|
|
|
} else {
|
|
|
|
this.classList.remove("active");
|
|
|
|
content.classList.remove("active");
|
|
|
|
}
|
|
|
|
|
|
|
|
const observer = new ResizeObserver(() => {
|
|
|
|
if (active) content.style.maxHeight = content.scrollHeight + "px";
|
|
|
|
});
|
|
|
|
|
|
|
|
Array.from(content.children).forEach((child) => {
|
|
|
|
observer.observe(child);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (active) {
|
2022-11-20 23:10:03 +00:00
|
|
|
content.style.maxHeight = null;
|
2022-11-25 23:27:44 +00:00
|
|
|
active = false;
|
2022-11-20 23:10:03 +00:00
|
|
|
} else {
|
|
|
|
content.style.maxHeight = content.scrollHeight + "px";
|
2022-11-25 23:27:44 +00:00
|
|
|
active = true;
|
2022-11-20 23:10:03 +00:00
|
|
|
}
|
|
|
|
});
|
2022-11-20 21:39:24 +00:00
|
|
|
}
|
2022-11-24 04:53:14 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Slider Inputs
|
|
|
|
*/
|
|
|
|
function createSlider(name, wrapper, options = {}) {
|
|
|
|
defaultOpt(options, {
|
|
|
|
valuecb: null,
|
|
|
|
min: 0,
|
|
|
|
max: 1,
|
|
|
|
step: 0.1,
|
|
|
|
defaultValue: 0.7,
|
|
|
|
});
|
|
|
|
|
|
|
|
let value = options.defaultValue;
|
|
|
|
|
|
|
|
// Use phantom range element for rounding
|
|
|
|
const phantomRange = document.createElement("input");
|
|
|
|
phantomRange.type = "range";
|
|
|
|
phantomRange.min = options.min;
|
|
|
|
phantomRange.max = options.max;
|
|
|
|
phantomRange.step = options.step;
|
|
|
|
|
|
|
|
// Build slider element
|
|
|
|
const underEl = document.createElement("div");
|
|
|
|
underEl.classList.add("under");
|
|
|
|
const textEl = document.createElement("input");
|
|
|
|
textEl.type = "text";
|
|
|
|
textEl.classList.add("text");
|
|
|
|
|
|
|
|
const overEl = document.createElement("div");
|
|
|
|
overEl.classList.add("over");
|
|
|
|
|
|
|
|
wrapper.classList.add("slider-wrapper");
|
|
|
|
wrapper.appendChild(underEl);
|
|
|
|
wrapper.appendChild(textEl);
|
|
|
|
wrapper.appendChild(overEl);
|
|
|
|
|
|
|
|
const bar = document.createElement("div");
|
|
|
|
bar.classList.add("slider-bar");
|
|
|
|
underEl.appendChild(bar);
|
|
|
|
underEl.appendChild(document.createElement("div"));
|
|
|
|
|
|
|
|
// Set value
|
|
|
|
const setValue = (val) => {
|
|
|
|
phantomRange.value = val;
|
|
|
|
value = parseFloat(phantomRange.value);
|
|
|
|
bar.style.width = `${
|
2022-11-24 14:54:30 +00:00
|
|
|
100 * ((value - options.min) / (options.max - options.min))
|
|
|
|
}%`;
|
2022-11-24 04:53:14 +00:00
|
|
|
textEl.value = `${name}: ${value}`;
|
|
|
|
options.valuecb && options.valuecb(value);
|
|
|
|
};
|
|
|
|
|
|
|
|
setValue(options.defaultValue);
|
|
|
|
|
|
|
|
// Events
|
|
|
|
textEl.addEventListener("blur", () => {
|
|
|
|
overEl.style.pointerEvents = "auto";
|
|
|
|
textEl.value = `${name}: ${value}`;
|
|
|
|
});
|
|
|
|
textEl.addEventListener("focus", () => {
|
2022-11-24 15:30:13 +00:00
|
|
|
overEl.style.pointerEvents = "none";
|
2022-11-24 04:53:14 +00:00
|
|
|
textEl.value = value;
|
|
|
|
});
|
|
|
|
|
|
|
|
textEl.addEventListener("change", () => {
|
|
|
|
try {
|
|
|
|
if (Number.isNaN(parseFloat(textEl.value))) setValue(value);
|
|
|
|
else setValue(parseFloat(textEl.value));
|
|
|
|
} catch (e) {}
|
|
|
|
});
|
|
|
|
|
|
|
|
keyboard.listen.onkeyclick.on((evn) => {
|
|
|
|
if (evn.target === textEl && evn.code === "Enter") {
|
|
|
|
textEl.blur();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
mouse.listen.window.left.onclick.on((evn) => {
|
|
|
|
if (evn.target === overEl) {
|
|
|
|
textEl.select();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
mouse.listen.window.left.ondrag.on((evn) => {
|
|
|
|
if (evn.target === overEl) {
|
|
|
|
setValue(
|
|
|
|
Math.max(
|
|
|
|
options.min,
|
|
|
|
Math.min(
|
|
|
|
options.max,
|
|
|
|
(evn.evn.layerX / wrapper.offsetWidth) *
|
|
|
|
(options.max - options.min) +
|
|
|
|
options.min
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return {
|
|
|
|
set value(val) {
|
|
|
|
setValue(val);
|
|
|
|
},
|
|
|
|
get value() {
|
|
|
|
return value;
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|