[feat](trx-frontend-http): improve mobile UX and fix S0/S9 label offset

- Add viewport meta tag so mobile browsers use device width instead of
  the default 980px desktop viewport
- Add 520px breakpoint: remove controls-tray forced min-width (was
  52–58rem, causing horizontal scroll on phones), stack controls-row
  into a single column with jog wheel first (order: -1)
- Scale frequency DSEG14 display with clamp() on narrow screens
- Make volume sliders responsive width with larger 20px thumb
- Raise spectrum Set/Auto button and input heights to 2.2rem (overrides
  the min-height: 0 that blocked the existing 760px touch-size rule)
- Add overflow-x: auto + flex-shrink: 0 to sub-tab-bar so all six
  plugin tabs are reachable on small screens without clipping
- Swap spectrum hint text based on input device: mouse/keyboard hint
  hidden on touch devices, touch-specific hint shown instead
- Offset S0/S9/S9+ signal history labels 6px below their grid lines

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
Signed-off-by: Stan Grams <sjg@haxx.space>
This commit is contained in:
2026-02-28 01:37:04 +01:00
parent 449663ddd7
commit 4e9516871f
3 changed files with 57 additions and 5 deletions
@@ -654,7 +654,7 @@ function drawOverviewSignalHistory(ctx, w, h, isLight) {
ctx.moveTo(0, y); ctx.moveTo(0, y);
ctx.lineTo(w, y); ctx.lineTo(w, y);
ctx.stroke(); ctx.stroke();
ctx.fillText(marker.label, w - 6, Math.max(8, Math.min(h - 8, y))); ctx.fillText(marker.label, w - 6, Math.max(8, Math.min(h - 8, y + 6)));
} }
ctx.beginPath(); ctx.beginPath();
@@ -2,6 +2,7 @@
<html lang="en"> <html lang="en">
<head> <head>
<meta charset="utf-8"> <meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0, viewport-fit=cover">
<title>trx-rs v{ver}</title> <title>trx-rs v{ver}</title>
<link rel="icon" href="/favicon.ico" /> <link rel="icon" href="/favicon.ico" />
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fontsource/dseg14-classic/400.css" /> <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fontsource/dseg14-classic/400.css" />
@@ -84,7 +85,8 @@
<button id="spectrum-auto-btn" type="button">Auto</button> <button id="spectrum-auto-btn" type="button">Auto</button>
</div> </div>
</div> </div>
<div id="spectrum-hint">Scroll to zoom &middot; Ctrl+Scroll to tune &middot; Drag to pan &middot; Drag BW edges to resize</div> <div id="spectrum-hint" class="spectrum-hint-mouse">Scroll to zoom &middot; Ctrl+Scroll to tune &middot; Drag to pan &middot; Drag BW edges to resize</div>
<div id="spectrum-hint-touch" class="spectrum-hint-touch">Pinch to zoom &middot; Drag to pan &middot; Drag BW edges to resize</div>
</div> </div>
<div class="status"> <div class="status">
<div class="full-row freq-row"> <div class="full-row freq-row">
@@ -605,8 +605,8 @@ small { color: var(--text-muted); }
transition: width 100ms ease; transition: width 100ms ease;
} }
.sub-tab-bar { display: flex; border-bottom: 1px solid var(--border); margin-bottom: 0.75rem; } .sub-tab-bar { display: flex; border-bottom: 1px solid var(--border); margin-bottom: 0.75rem; overflow-x: auto; -webkit-overflow-scrolling: touch; }
.sub-tab { 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 { 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.active { border-bottom-color: var(--accent-green); color: var(--accent-green); font-weight: 600; }
.sub-tab:hover:not(.active) { color: var(--text); } .sub-tab:hover:not(.active) { color: var(--text); }
#aprs-map { min-height: 150px; border-radius: 6px; } #aprs-map { min-height: 150px; border-radius: 6px; }
@@ -830,10 +830,60 @@ button:focus-visible, input:focus-visible, select:focus-visible {
padding: 0 8px; padding: 0 8px;
font-size: 0.73rem; font-size: 0.73rem;
} }
#spectrum-hint { .spectrum-hint-mouse,
.spectrum-hint-touch {
font-size: 0.68rem; font-size: 0.68rem;
color: var(--text-muted); color: var(--text-muted);
text-align: right; text-align: right;
margin-top: 2px; margin-top: 2px;
opacity: 0.6; opacity: 0.6;
} }
/* Show correct hint based on input device */
.spectrum-hint-touch { display: none; }
@media (hover: none) and (pointer: coarse) {
.spectrum-hint-mouse { display: none; }
.spectrum-hint-touch { display: block; }
}
/* ── Phone layout (≤ 520px) ───────────────────────────────────────────── */
@media (max-width: 520px) {
/* Remove forced min-width — no horizontal scroll on phones */
.controls-tray {
width: 100%;
}
/* Single-column controls: jog first, then mode, then power */
.controls-row {
grid-template-columns: 1fr;
}
.controls-col-center {
order: -1;
justify-self: center;
width: auto;
}
.controls-col-center::after { display: none; }
.controls-col.label-below-col .inline,
.controls-col.label-below-col .btn-grid { margin-top: 0; }
/* Scale frequency display to fit narrow screens */
#freq { font-size: clamp(1.3rem, 6vw, 2rem); }
/* Wider volume sliders for touch */
.vol-slider { width: clamp(80px, 35vw, 140px); }
.vol-slider::-webkit-slider-thumb { width: 20px; height: 20px; }
.vol-slider::-moz-range-thumb { width: 20px; height: 20px; }
/* Spectrum control inputs and buttons: meet minimum tap-target size */
#spectrum-bw-input,
#spectrum-floor-input {
height: 2.2rem;
font-size: 0.82rem;
}
#spectrum-bw-set-btn,
#spectrum-auto-btn {
height: 2.2rem;
min-height: 0;
padding: 0 10px;
font-size: 0.82rem;
}
}