Add message sending from web dashboard with channel selector, new POST /api/send endpoint, scheduler support for free-text messages (type field), and modernized dashboard layout with glassmorphism navbar, gradient stat cards, chat bubbles. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
139 lines
7 KiB
HTML
139 lines
7 KiB
HTML
<!DOCTYPE html>
|
|
<html lang="de" data-bs-theme="dark">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>MeshDD-Bot Dashboard</title>
|
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/css/bootstrap.min.css" rel="stylesheet">
|
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.11.3/font/bootstrap-icons.min.css" rel="stylesheet">
|
|
<link rel="stylesheet" href="/static/css/style.css">
|
|
</head>
|
|
<body>
|
|
<!-- Navbar -->
|
|
<nav class="navbar navbar-expand-sm navbar-glass sticky-top">
|
|
<div class="container-fluid">
|
|
<a class="navbar-brand d-flex align-items-center gap-2" href="/">
|
|
<i class="bi bi-broadcast-pin text-info fs-5"></i>
|
|
<span class="fw-bold">MeshDD-Bot</span>
|
|
<span class="badge badge-pill bg-body-secondary text-body-secondary fw-normal small" id="versionLabel"></span>
|
|
</a>
|
|
<div class="d-flex align-items-center gap-2">
|
|
<a href="/scheduler" class="btn btn-outline-info btn-sm rounded-pill px-3">
|
|
<i class="bi bi-clock-history me-1"></i>Scheduler
|
|
</a>
|
|
<a href="/map" target="_blank" class="btn btn-outline-info btn-sm rounded-pill px-3">
|
|
<i class="bi bi-map me-1"></i>Karte
|
|
</a>
|
|
<button class="btn btn-outline-secondary btn-sm rounded-circle" id="themeToggle" title="Theme wechseln" style="width:32px;height:32px;">
|
|
<i class="bi bi-sun-fill" id="themeIcon"></i>
|
|
</button>
|
|
<span class="badge badge-pill bg-body-secondary d-flex align-items-center gap-2 px-3 py-2" id="statusBadge">
|
|
<span class="status-dot" id="statusDot"></span>
|
|
<span class="small" id="statusText">Verbinde...</span>
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</nav>
|
|
|
|
<div class="container-fluid py-3">
|
|
<!-- Stats Cards -->
|
|
<div class="row g-2 mb-2">
|
|
<div class="col-4">
|
|
<div class="card stat-card">
|
|
<div class="stat-accent" style="background: linear-gradient(90deg, var(--bs-info), transparent);"></div>
|
|
<i class="bi bi-router stat-icon text-info"></i>
|
|
<div class="card-body py-2 ps-3">
|
|
<div class="fs-4 fw-bold text-info" id="statNodes">0</div>
|
|
<div class="text-body-secondary small">Nodes gesamt</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="col-4">
|
|
<div class="card stat-card">
|
|
<div class="stat-accent" style="background: linear-gradient(90deg, var(--bs-success), transparent);"></div>
|
|
<i class="bi bi-activity stat-icon text-success"></i>
|
|
<div class="card-body py-2 ps-3">
|
|
<div class="fs-4 fw-bold text-success" id="statNodes24h">0</div>
|
|
<div class="text-body-secondary small">Aktiv (24h)</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="col-4">
|
|
<div class="card stat-card">
|
|
<div class="stat-accent" style="background: linear-gradient(90deg, var(--bs-warning), transparent);"></div>
|
|
<i class="bi bi-terminal stat-icon text-warning"></i>
|
|
<div class="card-body py-2 ps-3">
|
|
<div class="fs-4 fw-bold text-warning" id="statCommands">0</div>
|
|
<div class="text-body-secondary small">Anfragen</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="row g-2 mb-3">
|
|
<div class="col-12">
|
|
<div class="card breakdown-bar">
|
|
<div class="card-body py-2 d-flex align-items-center gap-3 flex-wrap">
|
|
<span class="text-body-secondary small me-1"><i class="bi bi-bar-chart-fill me-1"></i>Anfragen:</span>
|
|
<span id="commandBreakdown" class="d-flex gap-2 flex-wrap"></span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Panels -->
|
|
<div class="row g-3">
|
|
<!-- Nodes Table -->
|
|
<div class="col-lg-7">
|
|
<div class="card panel-card">
|
|
<div class="card-header d-flex align-items-center">
|
|
<i class="bi bi-router me-2 text-info"></i>
|
|
<span class="fw-semibold">Nodes</span>
|
|
<span class="badge badge-pill bg-info bg-opacity-25 text-info ms-auto" id="nodeCountBadge">0</span>
|
|
</div>
|
|
<div class="card-body p-0 table-responsive" style="max-height: 500px; overflow-y: auto;">
|
|
<table class="table table-hover table-sm mb-0 align-middle nodes-table">
|
|
<thead class="table-dark sticky-top">
|
|
<tr>
|
|
<th>Name</th>
|
|
<th>Hardware</th>
|
|
<th class="text-end px-1">SNR</th>
|
|
<th class="px-1">Batterie</th>
|
|
<th class="text-center px-1">Hops</th>
|
|
<th class="text-end px-1">Zuletzt</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody id="nodesTable"></tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Messages -->
|
|
<div class="col-lg-5">
|
|
<div class="card panel-card">
|
|
<div class="card-header d-flex align-items-center">
|
|
<i class="bi bi-chat-dots me-2 text-warning"></i>
|
|
<span class="fw-semibold">Nachrichten</span>
|
|
</div>
|
|
<div class="card-body p-0 msg-scroll" style="max-height: 500px; overflow-y: auto;">
|
|
<div id="messagesList"></div>
|
|
</div>
|
|
<div class="send-bar">
|
|
<div class="input-group input-group-sm">
|
|
<select class="form-select form-select-sm rounded-start-pill" id="sendChannel" style="max-width: 120px;"></select>
|
|
<input type="text" class="form-control form-control-sm" id="sendText" placeholder="Nachricht senden...">
|
|
<button class="btn btn-info btn-sm rounded-end-pill px-3" id="btnSend" type="button">
|
|
<i class="bi bi-send-fill"></i>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/js/bootstrap.bundle.min.js"></script>
|
|
<script src="/static/js/dashboard.js"></script>
|
|
</body>
|
|
</html>
|