Files

1115 lines
36 KiB
HTML

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>{% block title %}Server Monitoring Dashboard{% endblock %}</title>
<!-- Apply dark mode before render to avoid flash -->
<script>if(localStorage.getItem('darkMode')==='enabled'){document.documentElement.style.backgroundColor='#121212';}</script>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css">
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.0.0/css/all.min.css">
<style>
body {
background-color: #f8f9fa;
font-family: Arial, sans-serif;
margin: 0;
padding: 0;
}
/* Sidebar Styles */
.sidebar {
position: fixed;
top: 0;
left: 0;
width: 250px;
height: 100vh;
background: linear-gradient(135deg, #2c3e50, #3498db);
color: white;
padding: 20px 0;
box-shadow: 2px 0 10px rgba(0, 0, 0, 0.1);
overflow-y: auto;
z-index: 1000;
}
.sidebar .logo {
text-align: center;
padding: 20px 15px 30px;
border-bottom: 1px solid rgba(255, 255, 255, 0.1);
margin-bottom: 20px;
}
.sidebar .logo h3 {
margin: 0;
font-weight: bold;
color: #fff;
}
.sidebar .logo small {
color: #bdc3c7;
}
.sidebar .nav-menu {
list-style: none;
padding: 0 10px;
margin: 0;
}
/* ── Accordion group header ── */
.nav-group {
margin-bottom: 4px;
}
.nav-group-header {
display: flex;
align-items: center;
padding: 11px 15px;
color: #ecf0f1;
border-radius: 8px;
cursor: pointer;
font-size: 13px;
font-weight: 600;
text-transform: uppercase;
letter-spacing: 0.5px;
user-select: none;
transition: background 0.2s ease;
}
.nav-group-header:hover {
background-color: rgba(255,255,255,0.1);
}
.nav-group-header.open {
background-color: rgba(255,255,255,0.12);
}
.nav-group-header i.group-icon {
width: 20px;
margin-right: 10px;
font-size: 15px;
}
.nav-group-header .group-label {
flex: 1;
}
.nav-group-header .chevron {
font-size: 11px;
transition: transform 0.25s ease;
opacity: 0.7;
}
.nav-group-header.open .chevron {
transform: rotate(90deg);
}
/* Badge on group header (e.g. pending count) */
.nav-group-header .group-badge {
font-size: 11px;
padding: 2px 6px;
border-radius: 10px;
background: #e74c3c;
color: #fff;
margin-right: 6px;
line-height: 1.4;
}
/* ── Collapsible children ── */
.nav-group-children {
overflow: hidden;
max-height: 0;
transition: max-height 0.3s ease;
padding-left: 10px;
}
.nav-group-children.open {
max-height: 600px;
}
/* ── Child nav links ── */
.sidebar .nav-item {
margin-bottom: 2px;
}
.sidebar .nav-link {
display: flex;
align-items: center;
padding: 9px 15px;
color: #dce6f0;
text-decoration: none;
border-radius: 7px;
transition: all 0.2s ease;
font-size: 13.5px;
}
.sidebar .nav-link:hover {
background-color: rgba(255,255,255,0.1);
color: #fff;
transform: translateX(4px);
}
.sidebar .nav-link.active {
background-color: rgba(255,255,255,0.22);
color: #fff;
font-weight: 600;
}
.sidebar .nav-link i {
width: 20px;
margin-right: 10px;
font-size: 14px;
opacity: 0.85;
}
/* ── Admin standalone link ── */
.sidebar .nav-link.admin-link {
color: #ff8a80;
margin-top: 6px;
border-top: 1px solid rgba(255,255,255,0.1);
padding-top: 13px;
font-size: 13.5px;
}
.sidebar .nav-link.admin-link:hover {
background-color: rgba(220,53,69,0.25);
color: #ff6b6b;
transform: translateX(4px);
}
.sidebar .nav-link.admin-link.active {
background-color: rgba(220,53,69,0.35);
color: #ff6b6b;
}
/* Main Content Styles */
.main-content {
margin-left: 250px;
padding: 20px;
min-height: 100vh;
}
.content-header {
background-color: #fff;
padding: 20px 25px;
border-radius: 10px;
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
margin-bottom: 20px;
}
.content-header h1 {
margin: 0;
color: #2c3e50;
font-size: 28px;
font-weight: 600;
}
.content-header .breadcrumb {
background: none;
padding: 0;
margin: 8px 0 0 0;
}
.content-header .breadcrumb-item a {
color: #3498db;
text-decoration: none;
}
.content-body {
background-color: #fff;
border-radius: 10px;
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
padding: 25px;
}
/* Responsive Design */
@media (max-width: 768px) {
.sidebar {
transform: translateX(-100%);
transition: transform 0.3s ease;
}
.sidebar.mobile-open {
transform: translateX(0);
}
.main-content {
margin-left: 0;
}
.mobile-menu-toggle {
display: block !important;
position: fixed;
top: 15px;
left: 15px;
z-index: 1001;
background: #3498db;
color: white;
border: none;
border-radius: 5px;
padding: 8px 12px;
cursor: pointer;
}
}
.mobile-menu-toggle {
display: none;
}
/* Additional Styles */
.table-container {
background-color: #ffffff;
border-radius: 8px;
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
padding: 20px;
margin-bottom: 20px;
}
.card {
border: none;
border-radius: 10px;
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
transition: transform 0.2s ease;
}
.card:hover {
transform: translateY(-2px);
}
.btn-primary {
background-color: #3498db;
border-color: #3498db;
}
.btn-primary:hover {
background-color: #2980b9;
border-color: #2980b9;
}
/* Flash Messages */
.alert {
border-radius: 8px;
border: none;
margin-bottom: 20px;
}
/* ── Dark Mode ── */
body.dark-mode {
background-color: #121212;
color: #e0e0e0;
}
/* Override Bootstrap 5 CSS variables globally in dark mode */
body.dark-mode {
--bs-body-bg: #121212;
--bs-body-color: #e0e0e0;
--bs-card-bg: #2a2a2a;
--bs-card-color: #e0e0e0;
--bs-card-border-color: #444444;
--bs-table-bg: transparent;
--bs-table-color: #e0e0e0;
--bs-table-border-color: #444444;
--bs-table-striped-bg: #232323;
--bs-table-striped-color: #e0e0e0;
--bs-table-hover-bg: #303030;
--bs-table-hover-color: #e0e0e0;
--bs-table-active-bg: #353535;
--bs-border-color: #444444;
--bs-border-color-translucent: rgba(255,255,255,0.1);
--bs-link-color: #64b5f6;
--bs-link-hover-color: #90caf9;
--bs-nav-tabs-link-active-bg: #2a2a2a;
--bs-nav-tabs-link-active-color: #e0e0e0;
--bs-nav-tabs-link-active-border-color: #444444 #444444 #2a2a2a;
--bs-nav-link-color: #aaaaaa;
--bs-nav-link-hover-color: #e0e0e0;
--bs-list-group-bg: #2a2a2a;
--bs-list-group-color: #e0e0e0;
--bs-list-group-border-color: #444444;
--bs-list-group-action-color: #e0e0e0;
--bs-list-group-action-hover-bg: #333333;
--bs-list-group-action-active-bg: #333333;
--bs-input-bg: #2a2a2a;
--bs-input-color: #e0e0e0;
--bs-input-border-color: #555555;
--bs-form-select-bg: #2a2a2a;
}
body.dark-mode .content-header {
background-color: #1e1e1e;
color: #e0e0e0;
box-shadow: 0 2px 10px rgba(0,0,0,0.4);
}
body.dark-mode .content-header h1 {
color: #e0e0e0;
}
body.dark-mode .content-header .breadcrumb-item a {
color: #64b5f6;
}
body.dark-mode .content-header .breadcrumb-item.active {
color: #aaaaaa;
}
body.dark-mode .content-body {
background-color: #1e1e1e;
color: #e0e0e0;
box-shadow: 0 2px 10px rgba(0,0,0,0.4);
}
/* Cards */
body.dark-mode .card {
background-color: #2a2a2a !important;
color: #e0e0e0 !important;
border-color: #444444 !important;
box-shadow: 0 2px 10px rgba(0,0,0,0.4);
}
body.dark-mode .card-body {
background-color: #2a2a2a;
color: #e0e0e0;
}
body.dark-mode .card-header {
background-color: #333333 !important;
border-bottom-color: #444444 !important;
color: #e0e0e0 !important;
}
body.dark-mode .card-footer {
background-color: #2e2e2e !important;
border-top-color: #444444 !important;
color: #cccccc;
}
/* Tables — override Bootstrap table-light/table-dark utility */
body.dark-mode .table {
--bs-table-bg: transparent;
--bs-table-color: #e0e0e0;
--bs-table-border-color: #444444;
--bs-table-hover-bg: #303030;
--bs-table-hover-color: #e0e0e0;
--bs-table-striped-bg: #232323;
--bs-table-striped-color: #e0e0e0;
color: #e0e0e0;
border-color: #444444;
}
body.dark-mode .table > :not(caption) > * > * {
background-color: transparent;
color: #e0e0e0;
border-bottom-color: #444444;
}
body.dark-mode .table thead,
body.dark-mode .table-light,
body.dark-mode .table thead.table-light,
body.dark-mode .table thead > tr > th {
--bs-table-bg: #1a1a1a;
--bs-table-color: #b0b0b0;
--bs-table-border-color: #444444;
background-color: #1a1a1a !important;
color: #b0b0b0 !important;
border-color: #444444 !important;
}
body.dark-mode .table-hover > tbody > tr:hover > * {
background-color: #303030 !important;
color: #e0e0e0 !important;
}
body.dark-mode .table-container {
background-color: #1e1e1e;
box-shadow: 0 4px 8px rgba(0,0,0,0.4);
}
/* List groups */
body.dark-mode .list-group-item {
background-color: #2a2a2a !important;
color: #e0e0e0 !important;
border-color: #444444 !important;
}
body.dark-mode .list-group-item:hover {
background-color: #333333 !important;
}
/* Nav tabs */
body.dark-mode .nav-tabs {
border-bottom-color: #444444;
}
body.dark-mode .nav-tabs .nav-link {
color: #aaaaaa;
border-color: transparent;
}
body.dark-mode .nav-tabs .nav-link:hover {
color: #e0e0e0;
border-color: #555555 #555555 transparent;
}
body.dark-mode .nav-tabs .nav-link.active {
background-color: #2a2a2a !important;
color: #e0e0e0 !important;
border-color: #444444 #444444 #2a2a2a !important;
}
/* Nav pills */
body.dark-mode .nav-pills .nav-link {
color: #aaaaaa;
}
body.dark-mode .nav-pills .nav-link.active {
background-color: #3498db;
color: #fff;
}
/* Forms */
body.dark-mode .form-control,
body.dark-mode .form-select {
background-color: #2a2a2a !important;
border-color: #555555 !important;
color: #e0e0e0 !important;
}
body.dark-mode .form-control:focus,
body.dark-mode .form-select:focus {
background-color: #333333 !important;
border-color: #64b5f6 !important;
color: #e0e0e0 !important;
box-shadow: 0 0 0 0.2rem rgba(100,181,246,0.25);
}
body.dark-mode .form-control::placeholder {
color: #777777;
}
/* File input — native "Choose file" button */
body.dark-mode input[type="file"].form-control::file-selector-button {
background-color: #444444;
color: #e0e0e0;
border: 0;
border-right: 1px solid #555555;
padding: 0.375rem 0.75rem;
margin-right: 0.75rem;
}
body.dark-mode input[type="file"].form-control::file-selector-button:hover {
background-color: #555555;
cursor: pointer;
}
/* "No file chosen" text area colour */
body.dark-mode input[type="file"].form-control {
color: #aaaaaa;
}
/* Helper text below inputs */
body.dark-mode .form-text {
color: #888888 !important;
}
body.dark-mode .form-label,
body.dark-mode label {
color: #cccccc;
}
body.dark-mode .form-check-label {
color: #cccccc;
}
body.dark-mode .input-group-text {
background-color: #333333 !important;
border-color: #555555 !important;
color: #cccccc !important;
}
/* Modals */
body.dark-mode .modal-content {
background-color: #1e1e1e;
color: #e0e0e0;
border-color: #444444;
}
body.dark-mode .modal-header {
border-bottom-color: #444444;
background-color: #252525;
}
body.dark-mode .modal-footer {
border-top-color: #444444;
background-color: #252525;
}
/* Dropdowns */
body.dark-mode .dropdown-menu {
background-color: #2a2a2a;
border-color: #444444;
}
body.dark-mode .dropdown-item {
color: #e0e0e0;
}
body.dark-mode .dropdown-item:hover,
body.dark-mode .dropdown-item:focus {
background-color: #333333;
color: #e0e0e0;
}
body.dark-mode .dropdown-divider {
border-top-color: #444444;
}
/* Badges */
body.dark-mode .badge.bg-secondary {
background-color: #555555 !important;
}
body.dark-mode .badge.bg-light {
background-color: #444444 !important;
color: #e0e0e0 !important;
}
/* Alerts */
body.dark-mode .alert-info {
background-color: #0d3250;
border-color: #1565c0;
color: #90caf9;
}
body.dark-mode .alert-success {
background-color: #0d3b2a;
border-color: #1b5e20;
color: #a5d6a7;
}
body.dark-mode .alert-warning {
background-color: #3e2d00;
border-color: #f57f17;
color: #ffe082;
}
body.dark-mode .alert-danger {
background-color: #3b0d0d;
border-color: #b71c1c;
color: #ef9a9a;
}
/* Buttons */
body.dark-mode .btn-outline-secondary {
color: #aaaaaa;
border-color: #666666;
}
body.dark-mode .btn-outline-secondary:hover {
background-color: #444444;
color: #e0e0e0;
}
body.dark-mode .btn-outline-primary {
color: #64b5f6;
border-color: #64b5f6;
}
body.dark-mode .btn-outline-primary:hover {
background-color: #1565c0;
border-color: #1565c0;
color: #fff;
}
body.dark-mode .btn-outline-danger {
color: #ef9a9a;
border-color: #ef9a9a;
}
body.dark-mode .btn-outline-danger:hover {
background-color: #b71c1c;
border-color: #b71c1c;
color: #fff;
}
body.dark-mode .btn-outline-warning {
color: #ffe082;
border-color: #f57f17;
}
body.dark-mode .btn-outline-warning:hover {
background-color: #e65100;
border-color: #e65100;
color: #fff;
}
body.dark-mode .btn-close {
filter: invert(1);
}
/* Misc */
body.dark-mode hr {
border-color: #444444;
}
body.dark-mode .text-muted {
color: #888888 !important;
}
body.dark-mode .text-dark {
color: #e0e0e0 !important;
}
body.dark-mode .bg-light {
background-color: #2a2a2a !important;
}
body.dark-mode .bg-white {
background-color: #1e1e1e !important;
}
body.dark-mode .border {
border-color: #444444 !important;
}
body.dark-mode pre,
body.dark-mode code {
background-color: #1a1a1a;
color: #80cbc4;
border-color: #444444;
}
body.dark-mode a {
color: #64b5f6;
}
body.dark-mode a:hover {
color: #90caf9;
}
/* Pagination */
body.dark-mode .page-link {
background-color: #2a2a2a;
border-color: #444444;
color: #64b5f6;
}
body.dark-mode .page-link:hover {
background-color: #333333;
border-color: #555555;
color: #90caf9;
}
body.dark-mode .page-item.active .page-link {
background-color: #3498db;
border-color: #3498db;
}
body.dark-mode .page-item.disabled .page-link {
background-color: #1e1e1e;
border-color: #444444;
color: #666666;
}
/* Progress bars */
body.dark-mode .progress {
background-color: #333333;
}
/* Tabs content / tab-pane */
body.dark-mode .tab-content {
background-color: transparent;
}
/* Dark mode toggle button */
.dark-mode-btn {
display: flex;
align-items: center;
width: calc(100% - 20px);
margin: 4px 10px 0;
padding: 10px 15px;
background: rgba(255, 255, 255, 0.08);
border: 1px solid rgba(255,255,255,0.15);
border-radius: 8px;
color: #ecf0f1;
cursor: pointer;
font-size: 14px;
text-align: left;
transition: all 0.3s ease;
}
.dark-mode-btn:hover {
background: rgba(255, 255, 255, 0.18);
color: #fff;
}
.dark-mode-btn i {
width: 20px;
margin-right: 10px;
font-size: 16px;
}
.dark-mode-btn .toggle-track {
margin-left: auto;
width: 36px;
height: 20px;
background: rgba(255,255,255,0.2);
border-radius: 10px;
position: relative;
transition: background 0.3s;
}
.dark-mode-btn .toggle-track::after {
content: '';
position: absolute;
top: 3px;
left: 3px;
width: 14px;
height: 14px;
background: #fff;
border-radius: 50%;
transition: transform 0.3s;
}
body.dark-mode .dark-mode-btn .toggle-track {
background: #3498db;
}
body.dark-mode .dark-mode-btn .toggle-track::after {
transform: translateX(16px);
}
/* Dark mode — accordion sidebar */
body.dark-mode .nav-group-header {
color: #dce6f0;
}
body.dark-mode .nav-group-header:hover {
background-color: rgba(255,255,255,0.07);
}
body.dark-mode .nav-group-header.open {
background-color: rgba(255,255,255,0.09);
}
/* ── Dark mode sidebar ── */
body.dark-mode .sidebar {
background: linear-gradient(160deg, #0d0d0d 0%, #1a1a2e 50%, #16213e 100%);
box-shadow: 2px 0 16px rgba(0,0,0,0.6);
}
body.dark-mode .sidebar .logo {
border-bottom-color: rgba(255,255,255,0.08);
}
body.dark-mode .sidebar .logo small {
color: #7f8c9a;
}
body.dark-mode .sidebar .nav-link {
color: #b0bec5;
}
body.dark-mode .sidebar .nav-link:hover {
background-color: rgba(100,181,246,0.12);
color: #e0f0ff;
}
body.dark-mode .sidebar .nav-link.active {
background-color: rgba(100,181,246,0.2);
color: #e0f0ff;
border-left: 3px solid #64b5f6;
padding-left: 12px;
}
body.dark-mode .sidebar .nav-link i {
color: #64b5f6;
}
body.dark-mode .sidebar .nav-link.active i {
color: #90caf9;
}
body.dark-mode .sidebar .nav-link.admin-link {
color: #ef9a9a;
border-top-color: rgba(255,255,255,0.07);
}
body.dark-mode .sidebar .nav-link.admin-link:hover {
background-color: rgba(239,154,154,0.15);
color: #ffcdd2;
}
body.dark-mode .sidebar .nav-link.admin-link.active {
background-color: rgba(239,154,154,0.22);
color: #ffcdd2;
}
body.dark-mode .sidebar .nav-link.admin-link i {
color: #ef9a9a;
}
body.dark-mode .dark-mode-btn {
border-top-color: rgba(255,255,255,0.07);
color: #b0bec5;
}
</style>
{% block extra_css %}{% endblock %}
</head>
<body>
<!-- Mobile Menu Toggle -->
<button class="mobile-menu-toggle" onclick="toggleSidebar()">
<i class="fas fa-bars"></i>
</button>
<!-- Sidebar Navigation -->
<nav class="sidebar" id="sidebar">
<div class="logo">
<h3><i class="fas fa-server"></i> Monitor</h3>
<small>Server Monitoring v2.0</small>
</div>
<ul class="nav-menu">
<!-- ── WMT group ── -->
<li class="nav-group" id="group-wmt">
<div class="nav-group-header {% if request.endpoint and request.endpoint.startswith('wmt_web') %}open{% endif %}"
onclick="toggleGroup('group-wmt')">
<i class="fas fa-tablet-alt group-icon"></i>
<span class="group-label">WMT</span>
{% if pending_wmt_count > 0 %}<span class="group-badge">{{ pending_wmt_count }}</span>{% endif %}
<i class="fas fa-chevron-right chevron"></i>
</div>
<ul class="nav-group-children {% if request.endpoint and request.endpoint.startswith('wmt_web') %}open{% endif %}" style="list-style:none;padding-left:10px;margin:0;">
<li class="nav-item">
<a href="{{ url_for('wmt_web.index') }}" class="nav-link {% if request.endpoint == 'wmt_web.index' %}active{% endif %}">
<i class="fas fa-tachometer-alt"></i>Dashboard
</a>
</li>
<li class="nav-item">
<a href="{{ url_for('wmt_web.devices') }}" class="nav-link {% if request.endpoint in ['wmt_web.devices','wmt_web.device_new','wmt_web.device_edit'] %}active{% endif %}">
<i class="fas fa-desktop"></i>Devices
</a>
</li>
<li class="nav-item">
<a href="{{ url_for('wmt_web.update_requests') }}" class="nav-link {% if request.endpoint == 'wmt_web.update_requests' %}active{% endif %}">
<i class="fas fa-inbox"></i>Update Requests
{% if pending_wmt_count > 0 %}<span class="badge bg-danger ms-auto">{{ pending_wmt_count }}</span>{% endif %}
</a>
</li>
<li class="nav-item">
<a href="{{ url_for('wmt_web.releases') }}" class="nav-link {% if request.endpoint == 'wmt_web.releases' %}active{% endif %}">
<i class="fas fa-box-open"></i>Client Releases
</a>
</li>
<li class="nav-item">
<a href="{{ url_for('wmt_web.settings') }}" class="nav-link {% if request.endpoint == 'wmt_web.settings' %}active{% endif %}">
<i class="fas fa-cog"></i>Settings
</a>
</li>
</ul>
</li>
<!-- ── Live View group ── -->
{% set lv_active = request.endpoint in ['main.devices','main.device_edit','main.device_detail','main.logs','main.templates','main.stats'] %}
<li class="nav-group" id="group-liveview">
<div class="nav-group-header {% if lv_active %}open{% endif %}"
onclick="toggleGroup('group-liveview')">
<i class="fas fa-heartbeat group-icon"></i>
<span class="group-label">Live View</span>
<i class="fas fa-chevron-right chevron"></i>
</div>
<ul class="nav-group-children {% if lv_active %}open{% endif %}" style="list-style:none;padding-left:10px;margin:0;">
<li class="nav-item">
<a href="{{ url_for('main.devices') }}" class="nav-link {% if request.endpoint in ['main.devices','main.device_edit','main.device_detail'] %}active{% endif %}">
<i class="fas fa-satellite-dish"></i>Device Health
</a>
</li>
<li class="nav-item">
<a href="{{ url_for('main.logs') }}" class="nav-link {% if request.endpoint == 'main.logs' %}active{% endif %}">
<i class="fas fa-list-alt"></i>Logs
</a>
</li>
<li class="nav-item">
<a href="{{ url_for('main.stats') }}" class="nav-link {% if request.endpoint == 'main.stats' %}active{% endif %}">
<i class="fas fa-chart-bar"></i>Statistics
</a>
</li>
<li class="nav-item">
<a href="{{ url_for('main.templates') }}" class="nav-link {% if request.endpoint == 'main.templates' %}active{% endif %}">
<i class="fas fa-file-alt"></i>Templates
</a>
</li>
</ul>
</li>
<!-- ── Automation group ── -->
{% set auto_active = request.endpoint and request.endpoint.startswith('ansible_web') %}
<li class="nav-group" id="group-automation">
<div class="nav-group-header {% if auto_active %}open{% endif %}"
onclick="toggleGroup('group-automation')">
<i class="fas fa-robot group-icon"></i>
<span class="group-label">Automation</span>
<i class="fas fa-chevron-right chevron"></i>
</div>
<ul class="nav-group-children {% if auto_active %}open{% endif %}" style="list-style:none;padding-left:10px;margin:0;">
<li class="nav-item">
<a href="{{ url_for('ansible_web.devices') }}" class="nav-link {% if request.endpoint == 'ansible_web.devices' %}active{% endif %}">
<i class="fas fa-network-wired"></i>Remote Devices
</a>
</li>
<li class="nav-item">
<a href="{{ url_for('ansible_web.playbooks') }}" class="nav-link {% if request.endpoint == 'ansible_web.playbooks' %}active{% endif %}">
<i class="fas fa-book-open"></i>Playbooks
</a>
</li>
<li class="nav-item">
<a href="{{ url_for('ansible_web.execute') }}" class="nav-link {% if request.endpoint == 'ansible_web.execute' %}active{% endif %}">
<i class="fas fa-terminal"></i>Execute
</a>
</li>
<li class="nav-item">
<a href="{{ url_for('ansible_web.ssh_setup') }}" class="nav-link {% if request.endpoint == 'ansible_web.ssh_setup' %}active{% endif %}">
<i class="fas fa-key"></i>SSH Setup
</a>
</li>
<li class="nav-item">
<a href="{{ url_for('ansible_web.failure_reports') }}" class="nav-link {% if request.endpoint == 'ansible_web.failure_reports' %}active{% endif %}">
<i class="fas fa-exclamation-triangle"></i>Failure Reports
</a>
</li>
</ul>
</li>
<!-- ── Admin standalone ── -->
<li class="nav-item" style="padding: 0 0px;">
<a href="{{ url_for('main.admin') }}" class="nav-link admin-link {% if request.endpoint == 'main.admin' %}active{% endif %}">
<i class="fas fa-shield-alt"></i>Admin
</a>
</li>
</ul>
<button class="dark-mode-btn" id="darkModeToggle" onclick="toggleDarkMode()" title="Toggle dark / light mode">
<i class="fas fa-moon" id="darkModeIcon"></i>
<span id="darkModeLabel">Dark Mode</span>
<span class="toggle-track"></span>
</button>
</nav>
<!-- Main Content Area -->
<div class="main-content">
<!-- Content Header -->
<div class="content-header">
<h1>{% block page_title %}Dashboard{% endblock %}</h1>
{% if breadcrumbs %}
<nav aria-label="breadcrumb">
<ol class="breadcrumb">
{% for crumb in breadcrumbs %}
<li class="breadcrumb-item {% if loop.last %}active{% endif %}">
{% if not loop.last %}
<a href="{{ crumb.url }}">{{ crumb.title }}</a>
{% else %}
{{ crumb.title }}
{% endif %}
</li>
{% endfor %}
</ol>
</nav>
{% endif %}
</div>
<!-- Flash Messages -->
{% with messages = get_flashed_messages(with_categories=true) %}
{% if messages %}
<div class="flash-messages">
{% for category, message in messages %}
<div class="alert alert-{% if category == 'error' %}danger{% else %}{{ category }}{% endif %} alert-dismissible fade show" role="alert">
{{ message }}
<button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button>
</div>
{% endfor %}
</div>
{% endif %}
{% endwith %}
<!-- Main Content Body -->
<div class="content-body">
{% block content %}{% endblock %}
</div>
</div>
<!-- Scripts -->
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/js/bootstrap.bundle.min.js"></script>
<script>
// ── Accordion sidebar ──────────────────────────────────────────
function toggleGroup(groupId) {
const group = document.getElementById(groupId);
const header = group.querySelector('.nav-group-header');
const children = group.querySelector('.nav-group-children');
const isOpen = header.classList.contains('open');
header.classList.toggle('open', !isOpen);
children.classList.toggle('open', !isOpen);
// Persist state
const state = JSON.parse(localStorage.getItem('sidebarState') || '{}');
state[groupId] = !isOpen;
localStorage.setItem('sidebarState', JSON.stringify(state));
}
function restoreSidebarState() {
const state = JSON.parse(localStorage.getItem('sidebarState') || '{}');
document.querySelectorAll('.nav-group').forEach(function(group) {
const id = group.id;
// If server already rendered it open (active page), leave it
const header = group.querySelector('.nav-group-header');
const children = group.querySelector('.nav-group-children');
if (header.classList.contains('open')) return; // already active
if (state[id] === true) {
header.classList.add('open');
children.classList.add('open');
}
});
}
// Mobile sidebar toggle
function toggleSidebar() {
document.getElementById('sidebar').classList.toggle('mobile-open');
}
// Close sidebar when clicking outside on mobile
document.addEventListener('click', function(event) {
const sidebar = document.getElementById('sidebar');
const toggle = document.querySelector('.mobile-menu-toggle');
if (window.innerWidth <= 768 &&
!sidebar.contains(event.target) &&
!toggle.contains(event.target)) {
sidebar.classList.remove('mobile-open');
}
});
// ── Dark mode ─────────────────────────────────────────────────
function toggleDarkMode() {
const isDark = document.body.classList.toggle('dark-mode');
localStorage.setItem('darkMode', isDark ? 'enabled' : 'disabled');
updateDarkModeButton(isDark);
}
function updateDarkModeButton(isDark) {
const icon = document.getElementById('darkModeIcon');
const label = document.getElementById('darkModeLabel');
if (!icon || !label) return;
icon.className = isDark ? 'fas fa-sun' : 'fas fa-moon';
label.textContent = isDark ? 'Light Mode' : 'Dark Mode';
}
// ── Init ──────────────────────────────────────────────────────
document.addEventListener('DOMContentLoaded', function() {
// Restore dark mode
if (localStorage.getItem('darkMode') === 'enabled') {
document.body.classList.add('dark-mode');
updateDarkModeButton(true);
}
// Restore sidebar accordion state
restoreSidebarState();
});
</script>
{% block extra_js %}{% endblock %}
</body>
</html>