[feat](trx-frontend-http): add F1 keyboard shortcuts overlay

Press F1 to toggle a help overlay listing available keyboard shortcuts.
Dismiss with F1, Escape, or clicking the backdrop. Refactored the
global keydown handler to route all shortcuts through one listener.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
Signed-off-by: Stan Grams <sjg@haxx.space>
This commit is contained in:
2026-03-25 22:35:11 +01:00
parent 892533bdc2
commit d11f7b4876
3 changed files with 135 additions and 3 deletions
@@ -9823,13 +9823,55 @@ function shouldIgnoreGlobalShortcut(target) {
return !!target.closest("[contenteditable='true']");
}
// ── Shortcut help overlay ─────────────────────────────────────────────────────
function toggleShortcutOverlay() {
const el = document.getElementById("shortcut-overlay");
if (!el) return;
el.classList.toggle("is-hidden");
}
function hideShortcutOverlay() {
const el = document.getElementById("shortcut-overlay");
if (el) el.classList.add("is-hidden");
}
function isShortcutOverlayVisible() {
const el = document.getElementById("shortcut-overlay");
return el && !el.classList.contains("is-hidden");
}
document.addEventListener("DOMContentLoaded", () => {
const overlay = document.getElementById("shortcut-overlay");
if (overlay) overlay.addEventListener("click", (e) => {
if (e.target === overlay) hideShortcutOverlay();
});
});
window.addEventListener("keydown", (event) => {
if (event.defaultPrevented || event.repeat || event.isComposing) return;
const key = (event.key || "").toLowerCase();
// F1 — toggle shortcut help
if (event.key === "F1") {
event.preventDefault();
toggleShortcutOverlay();
return;
}
// Escape — close shortcut overlay if open
if (event.key === "Escape" && isShortcutOverlayVisible()) {
event.preventDefault();
hideShortcutOverlay();
return;
}
if (event.ctrlKey || event.metaKey || event.altKey) return;
if (shouldIgnoreGlobalShortcut(event.target)) return;
if ((event.key || "").toLowerCase() !== "s") return;
event.preventDefault();
void captureSpectrumScreenshot();
// S — spectrum screenshot
if (key === "s") {
event.preventDefault();
void captureSpectrumScreenshot();
return;
}
}, { capture: true });
// ── Zoom helpers ──────────────────────────────────────────────────────────────
@@ -1046,6 +1046,19 @@
</div>
</div>
</div>
<div id="shortcut-overlay" class="shortcut-overlay is-hidden" aria-live="polite" aria-atomic="true">
<div class="shortcut-overlay-card">
<div class="shortcut-overlay-title">Keyboard Shortcuts</div>
<table class="shortcut-table">
<tbody>
<tr><td class="shortcut-key"><kbd>S</kbd></td><td>Spectrum screenshot</td></tr>
<tr><td class="shortcut-key"><kbd>F1</kbd></td><td>Toggle this help</td></tr>
<tr><td class="shortcut-key"><kbd>Esc</kbd></td><td>Close overlay / exit fullscreen</td></tr>
</tbody>
</table>
<div class="shortcut-overlay-hint">Press <kbd>F1</kbd> or <kbd>Esc</kbd> to close</div>
</div>
</div>
<div id="decode-history-overlay" class="decode-history-overlay is-hidden" aria-live="polite" aria-atomic="true">
<div class="decode-history-overlay-card">
<div id="decode-history-overlay-title" class="decode-history-overlay-title">Loading decode history…</div>
@@ -1320,6 +1320,83 @@ small { color: var(--text-muted); }
.sub-tab { flex-shrink: 0; background: transparent; border: none; border-bottom: 2px solid transparent; border-radius: 0; padding: 0.35rem 0.75rem; color: var(--text-muted); cursor: pointer; font-size: 0.85rem; height: auto; }
.sub-tab.active { border-bottom-color: var(--accent-green); color: var(--accent-green); font-weight: 600; }
.sub-tab:hover:not(.active) { color: var(--text); }
/* ── Shortcut help overlay (F1) ─────────────────────────────────────── */
.shortcut-overlay {
position: fixed;
inset: 0;
z-index: 9600;
display: flex;
align-items: center;
justify-content: center;
padding: 1.2rem;
background: color-mix(in srgb, var(--bg) 36%, transparent);
backdrop-filter: blur(6px);
-webkit-backdrop-filter: blur(6px);
opacity: 1;
visibility: visible;
transition: opacity 140ms ease, visibility 140ms ease;
}
.shortcut-overlay.is-hidden {
opacity: 0;
visibility: hidden;
pointer-events: none;
}
.shortcut-overlay-card {
min-width: min(22rem, calc(100vw - 2.4rem));
max-width: min(28rem, calc(100vw - 2.4rem));
padding: 1.2rem 1.4rem;
border-radius: 0.9rem;
border: 1px solid color-mix(in srgb, var(--border-light) 72%, transparent);
background: color-mix(in srgb, var(--card-bg) 92%, transparent);
box-shadow: 0 18px 40px rgba(0, 0, 0, 0.22);
}
.shortcut-overlay-title {
font-size: 1.05rem;
font-weight: 800;
color: var(--text-heading);
margin-bottom: 0.8rem;
text-align: center;
}
.shortcut-table {
width: 100%;
border-collapse: collapse;
}
.shortcut-table td {
padding: 0.32rem 0.4rem;
font-size: 0.88rem;
color: var(--text);
border-bottom: 1px solid color-mix(in srgb, var(--border-light) 40%, transparent);
}
.shortcut-table tr:last-child td {
border-bottom: none;
}
.shortcut-key {
width: 5rem;
text-align: right;
padding-right: 0.9rem !important;
}
.shortcut-key kbd,
.shortcut-overlay-hint kbd {
display: inline-block;
min-width: 1.6em;
padding: 0.12em 0.45em;
border: 1px solid var(--border);
border-radius: 0.3rem;
background: var(--bg);
font-family: inherit;
font-size: 0.82rem;
font-weight: 600;
text-align: center;
color: var(--text);
box-shadow: 0 1px 0 color-mix(in srgb, var(--border) 60%, transparent);
}
.shortcut-overlay-hint {
margin-top: 0.7rem;
text-align: center;
font-size: 0.76rem;
color: var(--text-muted);
}
.decode-history-overlay {
position: fixed;
inset: 0;