1
0
mirror of https://github.com/coalaura/whiskr.git synced 2025-09-09 09:19:54 +00:00
Files
whiskr/static/js/lib.js

225 lines
3.6 KiB
JavaScript
Raw Normal View History

2025-08-11 00:15:58 +02:00
/** biome-ignore-all lint/correctness/noUnusedVariables: utility */
function storeValue(key, value = false) {
if (value === null || value === undefined || value === false) {
2025-08-09 21:16:24 +02:00
localStorage.removeItem(key);
return;
}
localStorage.setItem(key, JSON.stringify(value));
}
function loadValue(key, fallback = false) {
const raw = localStorage.getItem(key);
2025-08-11 00:15:58 +02:00
if (raw === null) {
2025-08-09 21:16:24 +02:00
return fallback;
}
try {
const value = JSON.parse(raw);
2025-08-11 00:15:58 +02:00
if (value === null) {
2025-08-09 21:16:24 +02:00
throw new Error("no value");
}
return value;
} catch {}
return fallback;
}
2025-08-10 15:53:30 +02:00
function schedule(cb) {
if (document.visibilityState === "visible") {
requestAnimationFrame(cb);
return;
}
setTimeout(cb, 80);
}
2025-08-09 21:16:24 +02:00
function uid() {
return `${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 8)}`;
}
function make(tag, ...classes) {
const el = document.createElement(tag);
2025-08-10 16:38:02 +02:00
if (classes.length) {
el.classList.add(...classes);
}
2025-08-09 21:16:24 +02:00
return el;
}
2025-08-18 04:46:17 +02:00
function fillSelect($select, options, callback) {
$select.innerHTML = "";
for (const option of options) {
const el = document.createElement("option");
callback(el, option);
$select.appendChild(el);
}
}
2025-08-25 23:36:46 +02:00
function wait(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
2025-08-09 21:16:24 +02:00
function escapeHtml(text) {
2025-08-18 03:47:37 +02:00
return text.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/>/g, "&gt;");
2025-08-09 21:16:24 +02:00
}
2025-08-11 15:43:00 +02:00
function formatMilliseconds(ms) {
if (ms < 1000) {
return `${ms}ms`;
} else if (ms < 10000) {
return `${(ms / 1000).toFixed(1)}s`;
}
return `${Math.round(ms / 1000)}s`;
}
function fixed(num, decimals = 0) {
return num.toFixed(decimals).replace(/\.?0+$/m, "");
}
2025-08-16 14:54:27 +02:00
2025-08-23 18:07:53 +02:00
function formatMoney(num) {
if (num === 0) {
return "0ct";
}
if (num < 1) {
let decimals = 1;
if (num < 0.0001) {
decimals = 3;
} else if (num < 0.001) {
decimals = 2;
}
return `${fixed(num * 100, decimals)}ct`;
}
return `$${fixed(num, 2)}`;
}
2025-08-23 17:15:03 +02:00
function clamp(num, min, max) {
return Math.min(Math.max(num, min), max);
}
2025-08-16 14:54:27 +02:00
function download(name, type, data) {
let blob;
if (data instanceof Blob) {
blob = data;
} else {
blob = new Blob([data], {
type: type,
});
}
const a = document.createElement("a"),
url = URL.createObjectURL(blob);
a.setAttribute("download", name);
a.style.display = "none";
a.href = url;
document.body.appendChild(a);
a.click();
document.body.removeChild(a);
URL.revokeObjectURL(url);
}
2025-08-18 03:47:37 +02:00
function lines(text) {
let count = 0,
index = 0;
while (index < text.length) {
index = text.indexOf("\n", index);
if (index === -1) {
break;
}
count++;
index++;
}
return count + 1;
}
2025-08-26 00:48:46 +02:00
function readFile(file, handler, onError = false) {
return new Promise(resolve => {
const reader = new FileReader();
reader.onload = () => {
try {
const result = {
name: file.name,
content: reader.result,
};
handler(result);
resolve(result);
} catch (err) {
onError?.(`${file.name}: ${err.message}`);
resolve(false);
}
};
reader.onerror = () => resolve(false);
reader.readAsText(file);
});
}
function selectFile(accept, multiple, handler, onError = false) {
2025-08-18 03:47:37 +02:00
return new Promise(resolve => {
2025-08-16 14:54:27 +02:00
const input = make("input");
input.type = "file";
input.accept = accept;
2025-08-26 00:48:46 +02:00
input.multiple = multiple;
2025-08-16 14:54:27 +02:00
2025-08-26 00:48:46 +02:00
input.onchange = async () => {
const files = input.files;
2025-08-16 14:54:27 +02:00
2025-08-26 00:48:46 +02:00
if (!files.length) {
2025-08-16 14:54:27 +02:00
resolve(false);
return;
}
2025-08-26 00:48:46 +02:00
const results = [];
2025-08-16 14:54:27 +02:00
2025-08-26 00:48:46 +02:00
for (const file of files) {
const result = await readFile(file, handler, onError);
2025-08-18 03:47:37 +02:00
2025-08-26 00:48:46 +02:00
if (result) {
results.push(result);
2025-08-16 14:54:27 +02:00
}
2025-08-26 00:48:46 +02:00
}
2025-08-18 03:47:37 +02:00
2025-08-26 00:48:46 +02:00
if (!results.length) {
resolve(false);
2025-08-16 14:54:27 +02:00
2025-08-26 00:48:46 +02:00
return;
}
2025-08-16 14:54:27 +02:00
2025-08-26 00:48:46 +02:00
resolve(multiple ? results : results[0]);
2025-08-16 14:54:27 +02:00
};
input.click();
});
}