secubox-openwrt/package/secubox/luci-app-exposure/htdocs/luci-static/resources/exposure/dashboard.css
CyberMind-FR 26daa57a4b fix(multi): HAProxy duplicate server, Streamlit headless, dashboard optimization
Fixes:
- HAProxy: Prevent duplicate server names when both inline and separate
  server UCI sections exist for same backend
- Streamlit: Force --server.headless=true in start script (required for server)
- Dashboard: Optimize get_dashboard_data RPC call (6.56s → 0.09s) by using
  fast catalog counting instead of slow appstore list command
- Exposure: Add themed dashboard with SecuBox styling
- ACL: Add missing RPCD permissions for various LuCI apps

Version bumps:
- luci-app-exposure: 1.0.0-r3
- secubox-core: 0.10.0-r5
- secubox-app-haproxy: 1.0.0-r18
- secubox-app-streamlit: 1.0.0-r2
- Portal: v0.15.51

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-26 11:04:02 +01:00

870 lines
16 KiB
CSS

/* SecuBox Service Exposure Manager - Dashboard Styles */
/* Unified theme matching SecuBox HAProxy dashboard */
:root {
--exp-bg-primary: #0d1117;
--exp-bg-secondary: #161b22;
--exp-bg-tertiary: #1a1a2e;
--exp-border: #30363d;
--exp-text-primary: #e6edf3;
--exp-text-secondary: #8892b0;
--exp-text-muted: #6e7681;
--exp-accent: #64ffda;
--exp-tor: #9b59b6;
--exp-ssl: #27ae60;
--exp-success: #22c55e;
--exp-warning: #f97316;
--exp-danger: #ef4444;
}
.exposure-dashboard {
padding: 0;
max-width: 1400px;
margin: 0 auto;
}
/* Page Header */
.exp-page-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 24px;
padding-bottom: 16px;
border-bottom: 1px solid var(--exp-border);
}
.exp-page-title {
font-size: 28px;
font-weight: 700;
color: var(--exp-text-primary);
display: flex;
align-items: center;
gap: 12px;
margin: 0;
}
.exp-page-title-icon {
font-size: 32px;
}
.exp-page-subtitle {
color: var(--exp-text-secondary);
font-size: 14px;
margin: 4px 0 0 0;
}
.exp-header-badges {
display: flex;
gap: 12px;
}
.exp-header-badge {
background: var(--exp-bg-tertiary);
border: 1px solid var(--exp-border);
padding: 8px 16px;
border-radius: 8px;
font-size: 14px;
color: var(--exp-text-secondary);
display: flex;
align-items: center;
gap: 6px;
}
/* Stats Grid */
.exp-stats-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(220px, 1fr));
gap: 16px;
margin-bottom: 24px;
}
.exp-stat-card {
background: linear-gradient(135deg, #1a1a2e 0%, #16213e 100%);
border: 1px solid var(--exp-border);
border-radius: 12px;
padding: 20px;
text-align: center;
transition: transform 0.2s, box-shadow 0.2s;
}
.exp-stat-card:hover {
transform: translateY(-2px);
box-shadow: 0 8px 25px rgba(0, 0, 0, 0.3);
}
.exp-stat-card.exp-stat-tor {
border-color: rgba(155, 89, 182, 0.4);
}
.exp-stat-card.exp-stat-ssl {
border-color: rgba(39, 174, 96, 0.4);
}
.exp-stat-icon {
font-size: 32px;
margin-bottom: 8px;
}
.exp-stat-value {
font-size: 36px;
font-weight: 700;
color: var(--exp-accent);
margin-bottom: 4px;
}
.exp-stat-card.exp-stat-tor .exp-stat-value {
color: var(--exp-tor);
}
.exp-stat-card.exp-stat-ssl .exp-stat-value {
color: var(--exp-ssl);
}
.exp-stat-label {
font-size: 14px;
color: var(--exp-text-secondary);
margin-bottom: 4px;
}
.exp-stat-trend {
font-size: 12px;
color: var(--exp-text-muted);
}
/* Cards */
.exp-card {
background: var(--exp-bg-secondary);
border: 1px solid var(--exp-border);
border-radius: 12px;
margin-bottom: 20px;
overflow: hidden;
}
.exp-card.exp-warning-card {
border-left: 4px solid var(--exp-warning);
}
.exp-card.exp-suggestions-card {
border-left: 4px solid var(--exp-accent);
}
.exp-card-header {
display: flex;
justify-content: space-between;
align-items: center;
padding: 16px 20px;
border-bottom: 1px solid var(--exp-border);
background: rgba(255, 255, 255, 0.02);
}
.exp-card-title {
font-size: 16px;
font-weight: 600;
color: var(--exp-text-primary);
display: flex;
align-items: center;
gap: 10px;
}
.exp-card-title-icon {
font-size: 20px;
}
.exp-card-body {
padding: 20px;
}
.exp-card-body.no-padding {
padding: 0;
}
/* Row layout */
.exp-row {
display: flex;
gap: 20px;
margin-bottom: 20px;
}
@media (max-width: 900px) {
.exp-row {
flex-direction: column;
}
}
/* Empty state */
.exp-empty {
text-align: center;
padding: 40px 20px;
color: var(--exp-text-muted);
}
.exp-empty-icon {
font-size: 48px;
margin-bottom: 12px;
opacity: 0.5;
}
.exp-empty-text {
font-size: 16px;
color: var(--exp-text-secondary);
margin-bottom: 8px;
}
.exp-empty-hint {
font-size: 13px;
color: var(--exp-text-muted);
}
/* Suggestions Grid */
.exp-suggestions-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
gap: 12px;
}
.exp-suggestion-item {
display: flex;
align-items: center;
gap: 12px;
padding: 12px 16px;
background: var(--exp-bg-tertiary);
border: 1px solid var(--exp-border);
border-radius: 8px;
transition: border-color 0.2s, background 0.2s;
}
.exp-suggestion-item:hover {
border-color: var(--exp-accent);
background: rgba(100, 255, 218, 0.05);
}
.exp-suggestion-icon {
font-size: 28px;
min-width: 40px;
text-align: center;
}
.exp-suggestion-info {
flex: 1;
min-width: 0;
}
.exp-suggestion-name {
font-weight: 600;
color: var(--exp-text-primary);
font-size: 14px;
}
.exp-suggestion-port {
font-size: 12px;
color: var(--exp-text-muted);
font-family: monospace;
}
.exp-suggestion-actions {
display: flex;
gap: 6px;
}
/* Services list */
.exp-services-list {
display: flex;
flex-direction: column;
gap: 8px;
}
.exp-service-item {
display: flex;
align-items: center;
gap: 12px;
padding: 12px 16px;
background: var(--exp-bg-tertiary);
border-radius: 8px;
transition: background 0.2s;
}
.exp-service-item:hover {
background: rgba(255, 255, 255, 0.05);
}
.exp-service-icon {
font-size: 24px;
min-width: 32px;
text-align: center;
}
.exp-service-info {
flex: 1;
min-width: 0;
}
.exp-service-name {
font-weight: 600;
color: var(--exp-text-primary);
font-size: 14px;
}
.exp-service-detail {
font-size: 12px;
font-family: monospace;
word-break: break-all;
}
.exp-service-detail.exp-onion {
color: var(--exp-tor);
}
.exp-service-detail.exp-domain {
color: var(--exp-ssl);
}
/* Buttons */
.exp-btn {
display: inline-flex;
align-items: center;
justify-content: center;
gap: 6px;
padding: 8px 16px;
border-radius: 6px;
font-size: 14px;
font-weight: 500;
cursor: pointer;
border: 1px solid transparent;
transition: all 0.2s;
text-decoration: none;
}
.exp-btn:hover {
transform: translateY(-1px);
}
.exp-btn-sm {
padding: 6px 12px;
font-size: 13px;
}
.exp-btn-xs {
padding: 4px 8px;
font-size: 16px;
min-width: 32px;
}
.exp-btn-primary {
background: linear-gradient(135deg, #64ffda, #4fc3f7);
color: #0d1117;
border: none;
}
.exp-btn-primary:hover {
box-shadow: 0 4px 15px rgba(100, 255, 218, 0.4);
}
.exp-btn-secondary {
background: transparent;
color: var(--exp-text-secondary);
border-color: var(--exp-border);
}
.exp-btn-secondary:hover {
background: rgba(255, 255, 255, 0.05);
border-color: var(--exp-text-secondary);
}
.exp-btn-tor {
background: rgba(155, 89, 182, 0.2);
color: var(--exp-tor);
border-color: var(--exp-tor);
}
.exp-btn-tor:hover {
background: var(--exp-tor);
color: #fff;
}
.exp-btn-ssl {
background: rgba(39, 174, 96, 0.2);
color: var(--exp-ssl);
border-color: var(--exp-ssl);
}
.exp-btn-ssl:hover {
background: var(--exp-ssl);
color: #fff;
}
.exp-btn-danger {
background: rgba(239, 68, 68, 0.2);
color: var(--exp-danger);
border-color: var(--exp-danger);
}
.exp-btn-danger:hover {
background: var(--exp-danger);
color: #fff;
}
/* Quick Actions */
.exp-quick-actions {
display: flex;
flex-wrap: wrap;
gap: 12px;
}
.exp-action-btn {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
gap: 8px;
padding: 16px 24px;
background: var(--exp-bg-tertiary);
border: 1px solid var(--exp-border);
border-radius: 10px;
cursor: pointer;
transition: all 0.2s;
text-decoration: none;
color: inherit;
min-width: 100px;
}
.exp-action-btn:hover {
background: rgba(100, 255, 218, 0.1);
border-color: var(--exp-accent);
transform: translateY(-2px);
}
.exp-action-icon {
font-size: 24px;
}
.exp-action-label {
font-size: 12px;
color: var(--exp-text-secondary);
font-weight: 500;
}
/* Table styles */
.exp-table {
width: 100%;
border-collapse: collapse;
}
.exp-table th,
.exp-table td {
padding: 12px 16px;
text-align: left;
border-bottom: 1px solid var(--exp-border);
}
.exp-table th {
color: var(--exp-text-muted);
font-weight: 500;
font-size: 12px;
text-transform: uppercase;
letter-spacing: 0.5px;
}
.exp-table td {
color: var(--exp-text-primary);
}
.exp-table tr:hover td {
background: rgba(100, 255, 218, 0.03);
}
/* Badge styles */
.exp-badge {
display: inline-block;
padding: 4px 10px;
border-radius: 20px;
font-size: 11px;
font-weight: 600;
text-transform: uppercase;
}
.exp-badge-success {
background: rgba(34, 197, 94, 0.2);
color: var(--exp-success);
}
.exp-badge-warning {
background: rgba(249, 115, 22, 0.2);
color: var(--exp-warning);
}
.exp-badge-danger {
background: rgba(239, 68, 68, 0.2);
color: var(--exp-danger);
}
.exp-badge-info {
background: rgba(100, 255, 218, 0.2);
color: var(--exp-accent);
}
.exp-badge-tor {
background: rgba(155, 89, 182, 0.2);
color: var(--exp-tor);
}
.exp-badge-ssl {
background: rgba(39, 174, 96, 0.2);
color: var(--exp-ssl);
}
/* Monospace text */
.exp-mono {
font-family: 'SF Mono', 'Monaco', 'Inconsolata', 'Fira Mono', monospace;
}
/* Toast notification */
.exp-toast {
animation: slideInRight 0.3s ease-out;
}
@keyframes slideInRight {
from {
transform: translateX(100%);
opacity: 0;
}
to {
transform: translateX(0);
opacity: 1;
}
}
/* Toggle switches (from services.js) */
.toggle-switch {
position: relative;
display: inline-block;
width: 50px;
height: 26px;
}
.toggle-switch input {
opacity: 0;
width: 0;
height: 0;
}
.toggle-slider {
position: absolute;
cursor: pointer;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: #333;
transition: 0.3s;
border-radius: 26px;
}
.toggle-slider:before {
position: absolute;
content: "";
height: 20px;
width: 20px;
left: 3px;
bottom: 3px;
background-color: #666;
transition: 0.3s;
border-radius: 50%;
}
input:checked + .toggle-slider {
background-color: #1a1a2e;
}
input:checked + .toggle-slider:before {
transform: translateX(24px);
}
input:checked + .tor-slider {
background-color: rgba(155, 89, 182, 0.3);
border: 1px solid #9b59b6;
}
input:checked + .tor-slider:before {
background-color: #9b59b6;
}
input:checked + .ssl-slider {
background-color: rgba(39, 174, 96, 0.3);
border: 1px solid #27ae60;
}
input:checked + .ssl-slider:before {
background-color: #27ae60;
}
.toggle-slider:hover {
border: 1px solid #555;
}
/* === Progress Modal Styles === */
.exp-progress-modal {
min-width: 400px;
}
.exp-progress-header {
text-align: center;
margin-bottom: 24px;
padding-bottom: 16px;
border-bottom: 1px solid var(--exp-border);
}
.exp-progress-title {
font-size: 18px;
font-weight: 600;
color: var(--exp-text-primary);
margin-bottom: 4px;
}
.exp-progress-subtitle {
font-size: 13px;
color: var(--exp-text-muted);
}
.exp-progress-steps {
display: flex;
flex-direction: column;
gap: 12px;
}
.exp-progress-step {
display: flex;
align-items: flex-start;
gap: 12px;
padding: 12px;
background: var(--exp-bg-tertiary);
border-radius: 8px;
border-left: 3px solid var(--exp-border);
transition: all 0.3s ease;
}
.exp-progress-step[data-status="pending"] {
opacity: 0.5;
}
.exp-progress-step[data-status="active"] {
border-left-color: #3b82f6;
background: rgba(59, 130, 246, 0.1);
}
.exp-progress-step[data-status="complete"] {
border-left-color: var(--exp-success);
}
.exp-progress-step[data-status="error"] {
border-left-color: var(--exp-danger);
background: rgba(239, 68, 68, 0.1);
}
.exp-step-indicator {
min-width: 32px;
height: 32px;
border-radius: 50%;
background: var(--exp-border);
display: flex;
align-items: center;
justify-content: center;
position: relative;
}
.exp-progress-step[data-status="active"] .exp-step-indicator {
background: #3b82f6;
}
.exp-progress-step[data-status="complete"] .exp-step-indicator {
background: var(--exp-success);
}
.exp-progress-step[data-status="error"] .exp-step-indicator {
background: var(--exp-danger);
}
.exp-step-number {
font-size: 14px;
font-weight: 600;
color: var(--exp-text-secondary);
}
.exp-progress-step[data-status="active"] .exp-step-number,
.exp-progress-step[data-status="complete"] .exp-step-number,
.exp-progress-step[data-status="error"] .exp-step-number {
color: #fff;
}
.exp-progress-step[data-status="complete"] .exp-step-number::before {
content: '\2713';
}
.exp-progress-step[data-status="complete"] .exp-step-number {
font-size: 0;
}
.exp-progress-step[data-status="complete"] .exp-step-number::before {
font-size: 16px;
}
.exp-progress-step[data-status="error"] .exp-step-number::before {
content: '\2717';
}
.exp-progress-step[data-status="error"] .exp-step-number {
font-size: 0;
}
.exp-progress-step[data-status="error"] .exp-step-number::before {
font-size: 16px;
}
.exp-progress-step[data-status="active"] .exp-step-indicator::after {
content: '';
position: absolute;
width: 100%;
height: 100%;
border-radius: 50%;
border: 2px solid #3b82f6;
animation: pulse 1.5s ease-in-out infinite;
}
@keyframes pulse {
0% { transform: scale(1); opacity: 1; }
50% { transform: scale(1.3); opacity: 0; }
100% { transform: scale(1); opacity: 0; }
}
.exp-step-content {
flex: 1;
min-width: 0;
}
.exp-step-label {
font-size: 14px;
font-weight: 500;
color: var(--exp-text-primary);
margin-bottom: 2px;
}
.exp-step-detail {
font-size: 12px;
color: var(--exp-text-muted);
word-break: break-word;
}
.exp-progress-step[data-status="active"] .exp-step-detail {
color: #93c5fd;
}
.exp-progress-step[data-status="error"] .exp-step-detail {
color: #fca5a5;
}
/* Progress Result */
.exp-progress-result {
margin-top: 20px;
padding: 16px;
border-radius: 8px;
text-align: center;
}
.exp-progress-result.success {
background: rgba(34, 197, 94, 0.15);
border: 1px solid var(--exp-success);
}
.exp-progress-result.error {
background: rgba(239, 68, 68, 0.15);
border: 1px solid var(--exp-danger);
}
.exp-result-icon {
font-size: 32px;
margin-bottom: 8px;
}
.exp-result-message {
font-size: 16px;
font-weight: 600;
color: var(--exp-text-primary);
margin-bottom: 8px;
}
.exp-result-details {
font-size: 13px;
color: var(--exp-text-secondary);
}
/* === Loading Skeleton === */
.exp-skeleton {
background: linear-gradient(90deg, var(--exp-bg-tertiary) 25%, var(--exp-bg-secondary) 50%, var(--exp-bg-tertiary) 75%);
background-size: 200% 100%;
animation: skeleton-shimmer 1.5s infinite;
border-radius: 8px;
}
@keyframes skeleton-shimmer {
0% { background-position: 200% 0; }
100% { background-position: -200% 0; }
}
.exp-skeleton-stat {
height: 120px;
}
.exp-skeleton-card {
height: 200px;
}
.exp-skeleton-text {
height: 20px;
margin-bottom: 8px;
}
.exp-skeleton-text.short {
width: 60%;
}
/* === Fade-in Animation === */
.exp-fade-in {
animation: fadeIn 0.3s ease-out;
}
@keyframes fadeIn {
from { opacity: 0; transform: translateY(10px); }
to { opacity: 1; transform: translateY(0); }
}
/* === Loading State === */
.exp-loading-overlay {
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(13, 17, 23, 0.8);
display: flex;
align-items: center;
justify-content: center;
z-index: 10;
border-radius: 12px;
}
.exp-loading-spinner {
width: 40px;
height: 40px;
border: 3px solid var(--exp-border);
border-top-color: var(--exp-accent);
border-radius: 50%;
animation: spin 1s linear infinite;
}
@keyframes spin {
to { transform: rotate(360deg); }
}