fix: use leading edge debounce to sync up transitions

This commit is contained in:
dragonwocky 2024-01-24 12:50:13 +11:00
parent 26c9638013
commit ff002044aa
Signed by: dragonwocky
GPG Key ID: 7998D08F7D7BD7A8
2 changed files with 31 additions and 17 deletions

View File

@ -6,19 +6,28 @@
"use strict";
// convenient util: batch event handling
// every ___ ms to avoid over-handling &
// any conflicts / perf.issues that may
// otherwise result. a wait time of ~200ms
// is recommended (the avg. human visual
// batch event callbacks to avoid over-handling
// and any conflicts / perf.issues that may
// otherwise result. initial call is immediate,
// following calls are delayed. a wait time of
// ~200ms is recommended (the avg. human visual
// reaction time is ~180-200ms)
const debounce = (callback, wait = 200) => {
let timeoutId;
return (...args) => {
clearTimeout(timeoutId);
timeoutId = setTimeout(() => callback(...args), wait);
const sleep = async (ms) => {
return new Promise((res, rej) => setTimeout(res, ms));
},
debounce = (callback, ms = 200) => {
let delay, update;
const next = () =>
sleep(ms).then(() => {
if (!update) return (delay = undefined);
update(), (update = undefined);
delay = next();
});
return (...args) => {
if (delay) update = callback.bind(this, ...args);
return delay || ((delay = next()), callback(...args));
};
};
};
// provides basic key/value reactivity:
// this is shared between all active mods,
@ -156,6 +165,7 @@ document.addEventListener("keydown", (event) => {
});
Object.assign((globalThis.__enhancerApi ??= {}), {
sleep,
debounce,
setState,
useState,

View File

@ -8,7 +8,7 @@
import { Button } from "./Button.mjs";
function Footer({ categories }) {
function Footer({ categories, transitionDuration = 150 }) {
const { html, setState, useState, reloadApp } = globalThis.__enhancerApi,
$reload = html`<${Button}
class="ml-auto"
@ -33,12 +33,16 @@ function Footer({ categories }) {
useState(["view"], ([view]) => {
let [footerOpen] = useState(["databaseUpdated"]);
for (const [ids, $btn] of $categories) {
const modInCategory = ids.some((id) => id === view);
if (modInCategory) footerOpen = true;
$btn.style.display = modInCategory ? "" : "none";
}
footerOpen ||= $categories.some(([ids]) => ids.some((id) => id === view));
setState({ footerOpen });
if (!footerOpen) return;
// only toggle buttons if footer is open,
// otherwise leave as is during transition
for (const [ids, $btn] of $categories) {
const viewInCategory = ids.some((id) => id === view);
$btn.style.display = viewInCategory ? "" : "none";
}
});
useState(["databaseUpdated"], ([databaseUpdated]) => {
$reload.style.display = databaseUpdated ? "" : "none";