hmc472/firmware/data/style.css
Ryan Malloy 613611d37a Add ESP32-S2 firmware for HMC472A attenuator control
PlatformIO/Arduino firmware for WEMOS/LOLIN S2 Mini:
- 6-bit GPIO control (GPIOs 1-6) with glitch-free register writes
- REST API: /status, /set, /config, /sweep endpoints
- Web UI with PCB green theme, slider, presets, pin visualization
- NVS persistence of attenuation setting across power cycles
- Sweep mode for automated attenuation stepping
- mDNS (attenuator.local), OTA updates, watchdog
2026-02-02 21:24:29 -07:00

420 lines
7.8 KiB
CSS

/* HMC472A Attenuator Web UI - PCB Green Theme */
:root {
--bg: #131009;
--bg-elevated: #1a1610;
--bg-card: #221e17;
--accent-green: #3aaa62;
--accent-green-dim: #2d8a4e;
--accent-green-glow: rgba(58, 170, 98, 0.3);
--accent-gold: #dbb960;
--accent-gold-dim: #b39a4d;
--accent-gold-glow: rgba(219, 185, 96, 0.3);
--text: #e8e2d6;
--text-dim: #9a9488;
--text-muted: #6a645a;
--border: #3a352c;
--error: #d64545;
--font-mono: 'JetBrains Mono', 'Fira Code', 'SF Mono', monospace;
--font-sans: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
}
* {
box-sizing: border-box;
margin: 0;
padding: 0;
}
body {
font-family: var(--font-sans);
background: var(--bg);
color: var(--text);
min-height: 100vh;
line-height: 1.5;
}
.container {
max-width: 480px;
margin: 0 auto;
padding: 1rem;
}
/* Header */
header {
text-align: center;
padding: 1.5rem 0;
border-bottom: 1px solid var(--border);
margin-bottom: 1.5rem;
}
header h1 {
font-family: var(--font-mono);
font-size: 1.75rem;
font-weight: 600;
color: var(--accent-green);
display: flex;
align-items: center;
justify-content: center;
gap: 0.5rem;
}
header h1 .icon {
width: 1.5rem;
height: 1.5rem;
}
.subtitle {
color: var(--text-dim);
font-size: 0.875rem;
margin-top: 0.25rem;
}
/* Sections */
section {
background: var(--bg-card);
border: 1px solid var(--border);
border-radius: 8px;
padding: 1rem;
margin-bottom: 1rem;
}
section h2 {
font-size: 0.75rem;
font-weight: 500;
color: var(--text-muted);
text-transform: uppercase;
letter-spacing: 0.05em;
margin-bottom: 0.75rem;
}
/* dB Display */
.display-section {
text-align: center;
background: var(--bg);
border-color: var(--accent-green-dim);
}
.db-display {
font-family: var(--font-mono);
font-size: 3.5rem;
font-weight: 700;
color: var(--accent-green);
text-shadow: 0 0 20px var(--accent-green-glow);
line-height: 1;
padding: 0.5rem 0;
}
.db-display .db-unit {
font-size: 1.5rem;
color: var(--text-dim);
margin-left: 0.25rem;
}
.step-display {
font-family: var(--font-mono);
font-size: 0.875rem;
color: var(--text-dim);
}
/* Slider */
.slider-section {
padding: 1.25rem 1rem;
}
#db-slider {
-webkit-appearance: none;
appearance: none;
width: 100%;
height: 8px;
background: var(--bg);
border-radius: 4px;
outline: none;
cursor: pointer;
}
#db-slider::-webkit-slider-thumb {
-webkit-appearance: none;
appearance: none;
width: 24px;
height: 24px;
background: var(--accent-green);
border-radius: 50%;
cursor: pointer;
box-shadow: 0 0 10px var(--accent-green-glow);
transition: transform 0.1s ease;
}
#db-slider::-webkit-slider-thumb:hover {
transform: scale(1.1);
}
#db-slider::-moz-range-thumb {
width: 24px;
height: 24px;
background: var(--accent-green);
border: none;
border-radius: 50%;
cursor: pointer;
box-shadow: 0 0 10px var(--accent-green-glow);
}
.slider-labels {
display: flex;
justify-content: space-between;
font-size: 0.75rem;
color: var(--text-muted);
margin-top: 0.5rem;
}
/* Preset Buttons */
.preset-buttons {
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 0.5rem;
}
.preset-btn {
font-family: var(--font-mono);
font-size: 0.875rem;
padding: 0.75rem 0.5rem;
background: var(--bg);
color: var(--text);
border: 1px solid var(--border);
border-radius: 6px;
cursor: pointer;
transition: all 0.15s ease;
}
.preset-btn:hover {
background: var(--bg-elevated);
border-color: var(--accent-green-dim);
}
.preset-btn:active,
.preset-btn.active {
background: var(--accent-green-dim);
border-color: var(--accent-green);
color: var(--bg);
}
/* Pin Visualization */
.pin-grid {
display: grid;
grid-template-columns: repeat(6, 1fr);
gap: 0.5rem;
}
.pin {
text-align: center;
padding: 0.5rem 0.25rem;
background: var(--bg);
border-radius: 6px;
border: 1px solid var(--border);
transition: all 0.15s ease;
}
.pin-indicator {
width: 20px;
height: 20px;
border-radius: 50%;
margin: 0 auto 0.375rem;
background: var(--accent-green);
box-shadow: 0 0 8px var(--accent-green-glow);
transition: all 0.15s ease;
}
.pin.low .pin-indicator {
background: var(--accent-gold);
box-shadow: 0 0 8px var(--accent-gold-glow);
}
.pin-label {
font-family: var(--font-mono);
font-size: 0.75rem;
font-weight: 600;
color: var(--text);
}
.pin-db {
font-size: 0.625rem;
color: var(--text-muted);
}
.pin-legend {
display: flex;
justify-content: center;
gap: 1.5rem;
margin-top: 0.75rem;
padding-top: 0.75rem;
border-top: 1px solid var(--border);
}
.legend-item {
display: flex;
align-items: center;
gap: 0.375rem;
font-size: 0.75rem;
color: var(--text-dim);
}
.legend-dot {
width: 10px;
height: 10px;
border-radius: 50%;
}
.legend-dot.high {
background: var(--accent-green);
}
.legend-dot.low {
background: var(--accent-gold);
}
/* Sweep Controls */
.sweep-controls {
display: flex;
gap: 0.5rem;
margin-bottom: 0.75rem;
}
.sweep-btn {
flex: 1;
display: flex;
align-items: center;
justify-content: center;
gap: 0.375rem;
font-family: var(--font-sans);
font-size: 0.875rem;
padding: 0.625rem 0.75rem;
background: var(--bg);
color: var(--text);
border: 1px solid var(--border);
border-radius: 6px;
cursor: pointer;
transition: all 0.15s ease;
}
.sweep-btn .icon {
width: 16px;
height: 16px;
}
.sweep-btn:hover {
background: var(--bg-elevated);
border-color: var(--accent-green-dim);
}
.sweep-btn.active {
background: var(--accent-green-dim);
border-color: var(--accent-green);
color: var(--bg);
}
.sweep-btn.stop:hover {
border-color: var(--error);
}
.sweep-config {
display: flex;
justify-content: center;
}
.sweep-config label {
display: flex;
align-items: center;
gap: 0.5rem;
font-size: 0.875rem;
color: var(--text-dim);
}
.sweep-config input {
width: 80px;
font-family: var(--font-mono);
font-size: 0.875rem;
padding: 0.375rem 0.5rem;
background: var(--bg);
color: var(--text);
border: 1px solid var(--border);
border-radius: 4px;
text-align: center;
}
.sweep-config input:focus {
outline: none;
border-color: var(--accent-green-dim);
}
.sweep-status {
text-align: center;
font-size: 0.75rem;
color: var(--accent-gold);
min-height: 1.25rem;
margin-top: 0.5rem;
}
/* Footer */
footer {
padding: 1rem 0;
border-top: 1px solid var(--border);
margin-top: 0.5rem;
}
.status-bar {
display: flex;
flex-wrap: wrap;
justify-content: center;
gap: 0.75rem 1.5rem;
font-family: var(--font-mono);
font-size: 0.75rem;
color: var(--text-muted);
}
.ota-section {
text-align: center;
margin-top: 0.75rem;
}
.ota-btn {
font-size: 0.75rem;
padding: 0.375rem 0.75rem;
background: transparent;
color: var(--text-muted);
border: 1px solid var(--border);
border-radius: 4px;
cursor: pointer;
transition: all 0.15s ease;
}
.ota-btn:hover {
color: var(--text);
border-color: var(--text-dim);
}
.ota-btn.enabled {
color: var(--accent-green);
border-color: var(--accent-green-dim);
}
/* Icons */
.icon {
width: 1em;
height: 1em;
vertical-align: middle;
}
/* Responsive */
@media (max-width: 400px) {
.preset-buttons {
grid-template-columns: repeat(2, 1fr);
}
.pin-grid {
grid-template-columns: repeat(3, 1fr);
}
.db-display {
font-size: 2.75rem;
}
}