Tau.Acuvim/firmware/data/app.css
Renier Forster 84a0668c54 Initial commit: Tau Acuvim IoT monitoring system
Complete IoT monitoring platform for Acuvim II power meters via ESP32.

Firmware (Phases 1-7):
- ESP32-WROVER-B (TTGO T-Call v1.4) with RS485 Modbus RTU
- WiFi STA+AP concurrent mode with GSM/GPRS failover
- Transport abstraction layer with 4 priority modes
- MQTT protocol with 20 commands, LWT, QoS, exponential backoff
- SD card offline buffering with JSONL rotation and non-blocking drain
- OTA firmware updates with dual partition rollback protection
- Watchdog timer, crash loop detection, Acuvim health monitoring
- Captive portal provisioning with AP mode

Console backend (Phase 8):
- .NET 10 minimal API with PostgreSQL + EF Core
- JWT authentication, SignalR real-time updates
- MQTTnet 5.x bridge service with health monitoring
- Device, telemetry, firmware, alert, group management
- Rate limiting, security headers, Swagger/OpenAPI

Frontend (Phase 9):
- React 18 + TypeScript + Vite with Ant Design 5
- ECharts telemetry visualization, TanStack Query
- SignalR live updates, device management UI
- Dashboard, fleet management, firmware deployment

Testing & Production (Phase 10):
- 28 firmware unit tests (Modbus, JSON, config, version)
- 23 xUnit backend tests (device, telemetry, command, alert)
- Docker Compose with nginx, TLS MQTT, PostgreSQL
- Production deployment, commissioning, and troubleshooting docs

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-05-16 19:05:32 +02:00

82 lines
3.7 KiB
CSS

*{box-sizing:border-box;margin:0;padding:0}
:root{
--bg:#f4f5f7;--card:#fff;--text:#1a1a2e;--muted:#888;
--border:#ddd;--accent:#0066cc;--accent-hover:#0052a3;
--green:#28a745;--red:#dc3545;--yellow:#ffc107;
--radius:8px;--shadow:0 1px 3px rgba(0,0,0,.1);
}
body{font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,sans-serif;
background:var(--bg);color:var(--text);max-width:600px;margin:0 auto;
padding:0 0 24px;font-size:14px;line-height:1.5}
header{display:flex;align-items:center;justify-content:space-between;
padding:12px 16px;background:var(--accent);color:#fff}
header h1{font-size:16px;font-weight:600}
nav{display:flex;overflow-x:auto;background:#fff;
border-bottom:2px solid var(--border);-webkit-overflow-scrolling:touch}
.tab{flex:1;padding:10px 6px;border:none;background:none;
font-size:13px;font-weight:500;color:var(--muted);cursor:pointer;
white-space:nowrap;border-bottom:2px solid transparent;margin-bottom:-2px}
.tab.active{color:var(--accent);border-bottom-color:var(--accent)}
main{padding:12px}
.panel{display:none}
.panel.active{display:block}
.card{background:var(--card);border-radius:var(--radius);
padding:16px;margin-bottom:12px;box-shadow:var(--shadow)}
.card h2{font-size:15px;margin-bottom:10px;padding-bottom:6px;border-bottom:1px solid var(--border)}
.card h3{font-size:13px;margin:10px 0 6px;display:flex;align-items:center;gap:8px}
.kv{display:flex;justify-content:space-between;padding:4px 0;font-size:13px}
.kv span:first-child{color:var(--muted)}
.dot{width:10px;height:10px;border-radius:50%;display:inline-block;flex-shrink:0}
.dot.on{background:var(--green)}
.dot.off{background:var(--red)}
.dot.warn{background:var(--yellow)}
.form-group{margin:8px 0}
.form-group label{display:block;font-size:12px;color:var(--muted);margin-bottom:2px}
.form-group input,.form-group select{width:100%;padding:8px 10px;border:1px solid var(--border);
border-radius:var(--radius);font-size:14px;background:#fff}
.form-group input:focus,.form-group select:focus{outline:none;border-color:var(--accent)}
.cb{display:flex;align-items:center;gap:8px;padding:6px 0;font-size:13px;cursor:pointer}
.cb input{width:16px;height:16px}
.btn{display:inline-block;padding:10px 20px;border:none;border-radius:var(--radius);
background:var(--accent);color:#fff;font-size:14px;font-weight:500;
cursor:pointer;margin-top:8px;width:100%;text-align:center}
.btn:hover{background:var(--accent-hover)}
.btn.warn{background:var(--yellow);color:#333}
.btn.danger{background:var(--red);margin-top:8px}
.btn-sm{padding:4px 12px;font-size:12px;border:1px solid var(--border);
border-radius:var(--radius);background:#fff;cursor:pointer}
.btn-sm:hover{background:var(--bg)}
.net-list{max-height:180px;overflow-y:auto;border:1px solid var(--border);
border-radius:var(--radius);margin-bottom:8px}
.net-item{display:flex;justify-content:space-between;align-items:center;
padding:8px 10px;border-bottom:1px solid var(--border);cursor:pointer;font-size:13px}
.net-item:last-child{border-bottom:none}
.net-item:hover{background:var(--bg)}
.net-item .ssid{font-weight:500}
.net-item .meta{color:var(--muted);font-size:12px}
.readings-grid{display:grid;grid-template-columns:1fr 1fr 1fr;gap:6px}
.readings-grid .rd{text-align:center;padding:6px;background:var(--bg);border-radius:4px}
.readings-grid .rd .lbl{font-size:11px;color:var(--muted)}
.readings-grid .rd .val{font-size:15px;font-weight:600}
.muted{color:var(--muted);font-size:13px;padding:12px 0;text-align:center}
.toast{position:fixed;bottom:20px;left:50%;transform:translateX(-50%);
padding:10px 24px;border-radius:var(--radius);color:#fff;font-size:13px;
z-index:999;transition:opacity .3s;box-shadow:0 2px 8px rgba(0,0,0,.2)}
.toast.ok{background:var(--green)}
.toast.err{background:var(--red)}
.toast.hidden{opacity:0;pointer-events:none}