filestash/public/assets/components/notification.js
2025-08-01 14:23:15 +10:00

90 lines
4.1 KiB
JavaScript

import { createElement } from "../lib/skeleton/index.js";
import { safe } from "../lib/dom.js";
import { ApplicationError } from "../lib/error.js";
import { animate, slideYIn, slideYOut } from "../lib/animate.js";
import { loadCSS } from "../helpers/loader.js";
const createNotification = async(msg, type) => createElement(`
<span class="component_notification" role="alert">
<div class="no-select">
<div class="component_notification--container ${type}">
<div class="message">${safe(msg)}</div>
<div class="close">
<img class="component_icon" draggable="false" src="" alt="close">
</div>
</div>
</div>
</span>
`);
class NotificationComponent extends HTMLElement {
buffer = [];
async connectedCallback() {
await loadCSS(import.meta.url, "./notification.css");
}
async trigger(message, type) {
if (this.buffer.length > 20) this.buffer.pop(); // failsafe
this.buffer.push({ message, type });
if (this.buffer.length !== 1) {
const $close = this.querySelector(".close");
if (!($close instanceof HTMLElement) || !$close.onclick) return;
$close.onclick(new PointerEvent("mousedown"));
return;
}
await this.run();
}
async run() {
if (this.buffer.length === 0) return;
const { message, type } = this.buffer[0];
const $notification = await createNotification(message, type);
this.replaceChildren($notification);
await animate($notification, {
keyframes: slideYIn(50),
time: 100,
});
const ids = [];
await Promise.race([
new Promise((done) => ids.push(setTimeout(() => {
done(new MouseEvent("mousedown"));
}, this.buffer.length === 1 ? 8000 : 800))),
new Promise((done) => ids.push(setTimeout(() => {
const $close = $notification.querySelector(".close");
if (!($close instanceof HTMLElement)) throw new ApplicationError("INTERNAL_ERROR", "assumption failed: notification close button missing");
$close.onclick = done;
}, 1000))),
]);
ids.forEach((id) => clearTimeout(id));
await animate($notification, {
keyframes: slideYOut(10),
time: 200,
});
$notification.remove();
this.buffer.shift();
await this.run();
}
}
customElements.define("component-notification", NotificationComponent);
function find() {
const $dom = document.body.querySelector("component-notification");
if (!($dom instanceof NotificationComponent)) throw new ApplicationError("INTERNAL_ERROR", "assumption failed: wrong type notification component");
return $dom;
}
export default class Notification {
static info(msg) {
find().trigger(msg, "info");
}
static success(msg) {
find().trigger(msg, "success");
}
static error(msg) {
find().trigger(msg, "error");
}
}