feat(admin): implement full admin dashboard structure
This commit is contained in:
732
static/css/admin.css
Normal file
732
static/css/admin.css
Normal file
@@ -0,0 +1,732 @@
|
||||
/* ===========================
|
||||
Admin Shell / Frame
|
||||
=========================== */
|
||||
.admin-shell {
|
||||
width: 100%;
|
||||
min-height: 100vh;
|
||||
display: flex;
|
||||
justify-content: flex-start;
|
||||
align-items: center;
|
||||
flex-direction: column;
|
||||
padding: 10px 16px 34px;
|
||||
gap: 10px;
|
||||
}
|
||||
|
||||
.admin-frame {
|
||||
width: min(var(--admin-frame-width, 1320px), 100%);
|
||||
display: grid;
|
||||
grid-template-rows: auto auto;
|
||||
gap: 10px;
|
||||
align-items: start;
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Admin Taskbar (top nav)
|
||||
=========================== */
|
||||
.admin-taskbar {
|
||||
width: 100%;
|
||||
display: grid;
|
||||
grid-template-columns: auto minmax(0, 1fr) auto;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
color: #000000;
|
||||
background-color: var(--w98-gray);
|
||||
background-image: linear-gradient(180deg, rgba(255,255,255,.36), rgba(0,0,0,.08)), repeating-linear-gradient(45deg, rgba(255,255,255,.12) 0 1px, transparent 1px 5px);
|
||||
border-top: 2px solid #ffffff;
|
||||
border-left: 2px solid #ffffff;
|
||||
border-right: 2px solid #000000;
|
||||
border-bottom: 2px solid #000000;
|
||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf, 4px 4px 0 rgba(0,0,0,.45);
|
||||
padding: 3px;
|
||||
position: sticky;
|
||||
top: 0;
|
||||
z-index: 50;
|
||||
transition: box-shadow 120ms steps(2, end), filter 120ms steps(2, end);
|
||||
}
|
||||
|
||||
.admin-taskbar.is-scrolled {
|
||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf, 0 5px 0 rgba(0,0,0,.55), 0 11px 0 rgba(0,0,0,.18);
|
||||
filter: brightness(1.02);
|
||||
}
|
||||
|
||||
.admin-taskbar.is-scrolled::after {
|
||||
content: "";
|
||||
position: absolute;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: -10px;
|
||||
height: 10px;
|
||||
pointer-events: none;
|
||||
background: linear-gradient(to bottom, rgba(0,0,0,.46), rgba(0,0,0,0));
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Start Button
|
||||
=========================== */
|
||||
.admin-start-button {
|
||||
min-width: 108px;
|
||||
height: 24px;
|
||||
display: inline-grid;
|
||||
grid-template-columns: 18px 1fr;
|
||||
align-items: center;
|
||||
gap: 5px;
|
||||
padding: 0 8px;
|
||||
color: #000000;
|
||||
background: var(--w98-gray);
|
||||
border-top: 2px solid #ffffff;
|
||||
border-left: 2px solid #ffffff;
|
||||
border-right: 2px solid #000000;
|
||||
border-bottom: 2px solid #000000;
|
||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf;
|
||||
font-weight: bold;
|
||||
text-decoration: none;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.admin-start-button:active {
|
||||
border-top-color: #000000;
|
||||
border-left-color: #000000;
|
||||
border-right-color: #ffffff;
|
||||
border-bottom-color: #ffffff;
|
||||
box-shadow: inset -1px -1px 0 #dfdfdf, inset 1px 1px 0 #808080;
|
||||
padding-top: 1px;
|
||||
}
|
||||
|
||||
.admin-start-logo {
|
||||
width: 16px;
|
||||
height: 16px;
|
||||
display: grid;
|
||||
place-items: center;
|
||||
color: #ffffff;
|
||||
background: #000078;
|
||||
border: 1px solid #ffffff;
|
||||
box-shadow: inset -5px 0 0 #0f80cd, inset 0 -5px 0 #4c1ca0;
|
||||
font-size: 10px;
|
||||
line-height: 10px;
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Taskbar Nav Buttons
|
||||
=========================== */
|
||||
.admin-taskbar-nav {
|
||||
min-width: 0;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 4px;
|
||||
overflow-x: auto;
|
||||
scrollbar-width: thin;
|
||||
padding-bottom: 1px;
|
||||
}
|
||||
|
||||
.admin-taskbar-button {
|
||||
height: 24px;
|
||||
min-width: 76px;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
gap: 5px;
|
||||
padding: 0 8px;
|
||||
color: #000000;
|
||||
background: var(--w98-gray);
|
||||
border-top: 1px solid #ffffff;
|
||||
border-left: 1px solid #ffffff;
|
||||
border-right: 1px solid #808080;
|
||||
border-bottom: 1px solid #808080;
|
||||
text-decoration: none;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.admin-taskbar-button:active {
|
||||
border-top-color: #000000;
|
||||
border-left-color: #000000;
|
||||
border-right-color: #ffffff;
|
||||
border-bottom-color: #ffffff;
|
||||
box-shadow: inset -1px -1px 0 #dfdfdf, inset 1px 1px 0 #808080;
|
||||
padding-top: 1px;
|
||||
}
|
||||
|
||||
.admin-taskbar-button.is-active {
|
||||
color: #ffffff;
|
||||
background: #000078;
|
||||
border-top-color: #000000;
|
||||
border-left-color: #000000;
|
||||
border-right-color: #ffffff;
|
||||
border-bottom-color: #ffffff;
|
||||
}
|
||||
|
||||
.admin-taskbar-button:hover {
|
||||
color: #ffffff;
|
||||
background: #000078;
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Taskbar Session Chips
|
||||
=========================== */
|
||||
.admin-taskbar-session {
|
||||
min-width: 0;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: flex-end;
|
||||
gap: 5px;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.admin-session-chip,
|
||||
.admin-alert-chip {
|
||||
height: 24px;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 5px;
|
||||
padding: 0 8px;
|
||||
background: #dfdfdf;
|
||||
border-top: 1px solid #808080;
|
||||
border-left: 1px solid #808080;
|
||||
border-right: 1px solid #ffffff;
|
||||
border-bottom: 1px solid #ffffff;
|
||||
color: #000000;
|
||||
text-decoration: none;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.admin-alert-chip.is-ok { background: #e8ffe8; border-color: #008000 #ffffff #ffffff #008000; }
|
||||
.admin-alert-chip.is-info { background: #d8e5f8; }
|
||||
.admin-alert-chip.is-warning {
|
||||
background: #ffffcc;
|
||||
border: 3px solid transparent;
|
||||
border-image: repeating-linear-gradient(45deg, #111111 0 8px, #ffcc00 8px 16px) 3;
|
||||
}
|
||||
.admin-alert-chip.is-danger {
|
||||
color: #ffffff;
|
||||
background: #800000;
|
||||
border: 3px solid transparent;
|
||||
border-image: repeating-linear-gradient(45deg, #ffcccc 0 8px, #300000 8px 16px) 3;
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Dashboard Window
|
||||
=========================== */
|
||||
.admin-dashboard-window {
|
||||
width: 100%;
|
||||
min-height: 0;
|
||||
padding: 0;
|
||||
overflow: visible;
|
||||
color: #000000;
|
||||
background-color: var(--w98-gray);
|
||||
background-image: linear-gradient(180deg, rgba(255,255,255,.24), rgba(0,0,0,.06));
|
||||
}
|
||||
|
||||
.admin-dashboard-window > .win98-titlebar {
|
||||
margin: 2px 2px 0;
|
||||
}
|
||||
|
||||
.admin-dashboard-window > .menu-bar {
|
||||
flex: 0 0 auto;
|
||||
height: auto;
|
||||
min-height: 24px;
|
||||
margin: 0 2px;
|
||||
padding: 1px 6px;
|
||||
color: #000000;
|
||||
background: var(--w98-gray);
|
||||
border-top: 1px solid #ffffff;
|
||||
border-left: 1px solid #ffffff;
|
||||
border-right: 1px solid #808080;
|
||||
border-bottom: 1px solid #808080;
|
||||
z-index: 30;
|
||||
}
|
||||
|
||||
.admin-dashboard-window > .menu-bar .menu-button {
|
||||
color: #000000;
|
||||
}
|
||||
|
||||
.admin-dashboard-window > .dashboard-body {
|
||||
flex: 1 1 auto;
|
||||
margin-top: 0;
|
||||
padding: 0 10px 10px;
|
||||
background-color: var(--w98-gray);
|
||||
background-image: linear-gradient(180deg, rgba(255,255,255,.18), rgba(0,0,0,.05));
|
||||
}
|
||||
|
||||
.admin-dashboard-statusbar {
|
||||
grid-template-columns: minmax(0, 1fr) 160px 210px;
|
||||
height: 28px;
|
||||
padding: 3px 4px 4px;
|
||||
background: var(--w98-gray);
|
||||
font-size: 12px;
|
||||
line-height: 14px;
|
||||
}
|
||||
|
||||
.admin-dashboard-statusbar span {
|
||||
min-height: 19px;
|
||||
align-items: center;
|
||||
padding: 1px 6px;
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Menu Bar (toolbar)
|
||||
=========================== */
|
||||
.admin-menu-bar {
|
||||
position: relative;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 2px;
|
||||
min-height: 24px;
|
||||
padding: 1px 6px;
|
||||
font-size: 13px;
|
||||
line-height: 13px;
|
||||
z-index: 20;
|
||||
}
|
||||
|
||||
.admin-menu-item {
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.admin-menu-button {
|
||||
height: 20px;
|
||||
min-width: 54px;
|
||||
padding: 0 8px;
|
||||
color: #000000;
|
||||
background: transparent;
|
||||
border: 1px solid transparent;
|
||||
font-family: inherit;
|
||||
font-size: 13px;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
.admin-menu-button:hover,
|
||||
.admin-menu-button:focus-visible {
|
||||
border-top: 1px solid #ffffff;
|
||||
border-left: 1px solid #ffffff;
|
||||
border-right: 1px solid #808080;
|
||||
border-bottom: 1px solid #808080;
|
||||
outline: none;
|
||||
}
|
||||
|
||||
.admin-menu-popup {
|
||||
position: absolute;
|
||||
top: 22px;
|
||||
left: 0;
|
||||
min-width: 220px;
|
||||
padding: 2px;
|
||||
background: var(--w98-gray);
|
||||
border-top: 2px solid #ffffff;
|
||||
border-left: 2px solid #ffffff;
|
||||
border-right: 2px solid #000000;
|
||||
border-bottom: 2px solid #000000;
|
||||
box-shadow: 3px 3px 0 rgba(0,0,0,.35);
|
||||
display: none;
|
||||
z-index: 60;
|
||||
}
|
||||
|
||||
.admin-menu-item.is-open .admin-menu-popup {
|
||||
display: block;
|
||||
}
|
||||
|
||||
.admin-menu-action {
|
||||
width: 100%;
|
||||
min-height: 22px;
|
||||
display: grid;
|
||||
grid-template-columns: 20px minmax(0, 1fr) auto;
|
||||
gap: 8px;
|
||||
align-items: center;
|
||||
padding: 2px 6px;
|
||||
color: #000000;
|
||||
background: transparent;
|
||||
border: 0;
|
||||
font-family: inherit;
|
||||
font-size: 12px;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
.admin-menu-action:hover,
|
||||
.admin-menu-action:focus-visible {
|
||||
color: #ffffff;
|
||||
background: #000078;
|
||||
outline: none;
|
||||
}
|
||||
|
||||
.admin-menu-separator {
|
||||
height: 1px;
|
||||
margin: 3px 2px;
|
||||
background: #808080;
|
||||
border-bottom: 1px solid #ffffff;
|
||||
}
|
||||
|
||||
.admin-menu-action .shortcut {
|
||||
color: #555555;
|
||||
}
|
||||
|
||||
.admin-menu-action:hover .shortcut {
|
||||
color: #ffffff;
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Hero Section
|
||||
=========================== */
|
||||
.admin-hero {
|
||||
display: grid;
|
||||
grid-template-columns: minmax(0, 1fr) 330px;
|
||||
gap: 10px;
|
||||
padding: 9px;
|
||||
align-items: stretch;
|
||||
}
|
||||
|
||||
.admin-hero-copy h2 {
|
||||
margin: 0 0 5px;
|
||||
font-size: 22px;
|
||||
line-height: 24px;
|
||||
}
|
||||
|
||||
.admin-hero-copy p {
|
||||
margin: 0;
|
||||
color: #333333;
|
||||
font-size: 13px;
|
||||
line-height: 15px;
|
||||
}
|
||||
|
||||
.admin-hero-status {
|
||||
display: grid;
|
||||
gap: 4px;
|
||||
align-content: center;
|
||||
padding: 7px;
|
||||
background: #ffffff;
|
||||
border-top: 1px solid #808080;
|
||||
border-left: 1px solid #808080;
|
||||
border-right: 1px solid #ffffff;
|
||||
border-bottom: 1px solid #ffffff;
|
||||
font-size: 12px;
|
||||
line-height: 13px;
|
||||
}
|
||||
|
||||
.admin-hero-status-row {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.admin-status-ok { color: #008000; }
|
||||
.admin-status-warn { color: #8a6200; }
|
||||
.admin-status-danger { color: #800000; }
|
||||
|
||||
/* ===========================
|
||||
Stats Grid
|
||||
=========================== */
|
||||
.admin-stats-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(4, minmax(0, 1fr));
|
||||
gap: 10px;
|
||||
}
|
||||
|
||||
.admin-stat-card {
|
||||
position: relative;
|
||||
min-height: 122px;
|
||||
padding: 10px 11px 10px 14px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
/* Left accent bar */
|
||||
.admin-stat-card::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
inset: 0 auto 0 0;
|
||||
width: 7px;
|
||||
border-left: 7px solid #000078;
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
/* Severity color states */
|
||||
.admin-stat-card.is-ok { background: linear-gradient(180deg, #eeffee, #ffffff); }
|
||||
.admin-stat-card.is-ok::before { border-left-color: #008000; }
|
||||
.admin-stat-card.is-info { background: linear-gradient(180deg, #edf4ff, #ffffff); }
|
||||
.admin-stat-card.is-info::before { border-left-color: #000078; }
|
||||
.admin-stat-card.is-warning { background: linear-gradient(180deg, #ffffcc, #ffffff); }
|
||||
.admin-stat-card.is-warning::before { border-left-color: #ffcc00; }
|
||||
.admin-stat-card.is-danger {
|
||||
color: #000000;
|
||||
background: repeating-linear-gradient(45deg, #fff2f2 0 6px, #ffe1e1 6px 12px);
|
||||
}
|
||||
.admin-stat-card.is-danger::before { border-left-color: #800000; }
|
||||
|
||||
.admin-stat-label {
|
||||
margin: 0 0 6px;
|
||||
color: #333333;
|
||||
font-size: 13px;
|
||||
line-height: 13px;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.admin-stat-value {
|
||||
margin: 0 0 7px;
|
||||
font-size: 32px;
|
||||
line-height: 32px;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.admin-stat-note {
|
||||
display: flex;
|
||||
gap: 4px;
|
||||
flex-wrap: wrap;
|
||||
margin: 0;
|
||||
color: #222222;
|
||||
font-size: 12px;
|
||||
line-height: 14px;
|
||||
}
|
||||
|
||||
.admin-stat-note-pill {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
min-height: 18px;
|
||||
padding: 1px 6px;
|
||||
background: #dfdfdf;
|
||||
border-top: 1px solid #ffffff;
|
||||
border-left: 1px solid #ffffff;
|
||||
border-right: 1px solid #808080;
|
||||
border-bottom: 1px solid #808080;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Main Grid / Section Windows
|
||||
=========================== */
|
||||
.admin-main-grid {
|
||||
display: grid;
|
||||
grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
|
||||
gap: 12px;
|
||||
align-items: start;
|
||||
}
|
||||
|
||||
.admin-span-2 {
|
||||
grid-column: 1 / -1;
|
||||
}
|
||||
|
||||
.admin-section-window {
|
||||
min-height: 0;
|
||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf, 3px 4px 0 rgba(0,0,0,.38);
|
||||
}
|
||||
|
||||
.admin-section-body {
|
||||
margin: 0 6px 6px;
|
||||
padding: 8px;
|
||||
min-height: 0;
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Quick Actions
|
||||
=========================== */
|
||||
.admin-link-list {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
list-style: none;
|
||||
display: grid;
|
||||
gap: 6px;
|
||||
}
|
||||
|
||||
.admin-link-list li {
|
||||
display: grid;
|
||||
grid-template-columns: auto minmax(0, 1fr);
|
||||
gap: 8px;
|
||||
align-items: center;
|
||||
color: #000000;
|
||||
font-size: 13px;
|
||||
line-height: 13px;
|
||||
}
|
||||
|
||||
.admin-link-button {
|
||||
min-width: 112px;
|
||||
height: 24px;
|
||||
display: inline-grid;
|
||||
place-items: center;
|
||||
padding: 0 10px;
|
||||
color: #000000;
|
||||
background: var(--w98-gray);
|
||||
border-top: 1px solid #ffffff;
|
||||
border-left: 1px solid #ffffff;
|
||||
border-right: 1px solid #000000;
|
||||
border-bottom: 1px solid #000000;
|
||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf;
|
||||
font-size: 12px;
|
||||
line-height: 12px;
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.admin-link-button:hover {
|
||||
filter: brightness(1.06);
|
||||
}
|
||||
|
||||
/* Titlebar action links (Show all) */
|
||||
.titlebar-actions {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 2px;
|
||||
margin-left: 8px;
|
||||
}
|
||||
|
||||
.titlebar-link-button {
|
||||
height: 18px;
|
||||
min-width: 64px;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
padding: 0 7px;
|
||||
color: #000000;
|
||||
background: var(--w98-gray);
|
||||
border-top: 1px solid #ffffff;
|
||||
border-left: 1px solid #ffffff;
|
||||
border-right: 1px solid #000000;
|
||||
border-bottom: 1px solid #000000;
|
||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf;
|
||||
text-decoration: none;
|
||||
font-size: 12px;
|
||||
line-height: 12px;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.titlebar-link-button:hover {
|
||||
filter: brightness(1.08);
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Compact Mode
|
||||
=========================== */
|
||||
body.is-compact .admin-dashboard-body {
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
body.is-compact .admin-section-body {
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Responsive: Medium (tablets)
|
||||
=========================== */
|
||||
@media (max-width: 1180px) {
|
||||
.admin-taskbar {
|
||||
grid-template-columns: auto minmax(0, 1fr);
|
||||
}
|
||||
|
||||
.admin-taskbar-session {
|
||||
grid-column: 1 / -1;
|
||||
justify-content: flex-start;
|
||||
overflow-x: auto;
|
||||
}
|
||||
|
||||
.admin-stats-grid {
|
||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
||||
}
|
||||
|
||||
.admin-hero {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.admin-main-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.admin-span-2 {
|
||||
grid-column: auto;
|
||||
}
|
||||
}
|
||||
|
||||
/* ===========================
|
||||
Responsive: Small (mobile)
|
||||
=========================== */
|
||||
@media (max-width: 760px) {
|
||||
.admin-shell {
|
||||
padding: 0 0 18px;
|
||||
align-items: stretch;
|
||||
}
|
||||
|
||||
.admin-frame {
|
||||
width: 100%;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.admin-taskbar {
|
||||
grid-template-columns: 1fr;
|
||||
border-left: 0;
|
||||
border-right: 0;
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.admin-start-button {
|
||||
width: 100%;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.admin-taskbar-nav {
|
||||
width: 100%;
|
||||
overflow-x: auto;
|
||||
padding-bottom: 3px;
|
||||
}
|
||||
|
||||
.admin-taskbar-button {
|
||||
min-width: 92px;
|
||||
}
|
||||
|
||||
.admin-taskbar-session {
|
||||
width: 100%;
|
||||
overflow-x: auto;
|
||||
padding-bottom: 3px;
|
||||
}
|
||||
|
||||
.admin-session-chip,
|
||||
.admin-alert-chip {
|
||||
flex: 0 0 auto;
|
||||
}
|
||||
|
||||
.admin-dashboard-window {
|
||||
min-height: 100dvh;
|
||||
border-left: 0;
|
||||
border-right: 0;
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.admin-dashboard-body {
|
||||
padding: 6px;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.admin-stats-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.admin-stat-card {
|
||||
min-height: 112px;
|
||||
}
|
||||
|
||||
.admin-menu-popup {
|
||||
position: fixed;
|
||||
left: 6px;
|
||||
right: 6px;
|
||||
top: 74px;
|
||||
min-width: 0;
|
||||
}
|
||||
|
||||
.titlebar-actions {
|
||||
margin-left: 4px;
|
||||
}
|
||||
|
||||
.titlebar-link-button {
|
||||
min-width: 58px;
|
||||
padding: 0 5px;
|
||||
}
|
||||
|
||||
.admin-dashboard-statusbar {
|
||||
grid-template-columns: 1fr;
|
||||
height: auto;
|
||||
min-height: 70px;
|
||||
}
|
||||
|
||||
.win98-titlebar h1,
|
||||
.win98-titlebar h2 {
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.win98-window-controls {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
|
||||
/* Override global main layout on admin pages since admin uses its own shell */
|
||||
body:has(.admin-shell) main {
|
||||
display: contents;
|
||||
}
|
||||
289
static/css/dashboard.css
Normal file
289
static/css/dashboard.css
Normal file
@@ -0,0 +1,289 @@
|
||||
/* ==============================================
|
||||
Dashboard-specific styles (shared with admin)
|
||||
Reusable across account dashboard pages
|
||||
============================================== */
|
||||
|
||||
/* Hero section */
|
||||
.dashboard-hero {
|
||||
display: grid;
|
||||
grid-template-columns: minmax(0, 1fr) 330px;
|
||||
gap: 10px;
|
||||
padding: 9px;
|
||||
align-items: stretch;
|
||||
}
|
||||
|
||||
.hero-copy h2 { margin: 0 0 5px; font-size: 22px; line-height: 24px; }
|
||||
.hero-copy p { margin: 0; color: #333; font-size: 13px; line-height: 15px; }
|
||||
|
||||
.hero-status {
|
||||
display: grid;
|
||||
gap: 4px;
|
||||
align-content: center;
|
||||
padding: 7px;
|
||||
background: #ffffff;
|
||||
border-top: 1px solid #808080;
|
||||
border-left: 1px solid #808080;
|
||||
border-right: 1px solid #ffffff;
|
||||
border-bottom: 1px solid #ffffff;
|
||||
font-size: 12px;
|
||||
line-height: 13px;
|
||||
}
|
||||
|
||||
.hero-status-row { display: flex; justify-content: space-between; gap: 8px; }
|
||||
.status-ok { color: #008000; }
|
||||
.status-warn { color: #8a6200; }
|
||||
.status-danger { color: #800000; }
|
||||
|
||||
/* Stats grid */
|
||||
.stats-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(4, minmax(0, 1fr));
|
||||
gap: 10px;
|
||||
}
|
||||
|
||||
.stat-card {
|
||||
position: relative;
|
||||
min-height: 122px;
|
||||
padding: 10px 11px 10px 14px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.stat-card::before {
|
||||
content: "";
|
||||
position: absolute;
|
||||
inset: 0 auto 0 0;
|
||||
width: 7px;
|
||||
border-left: 7px solid #000078;
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
.stat-card.is-ok { background: linear-gradient(180deg, #eeffee, #ffffff); }
|
||||
.stat-card.is-ok::before { border-left-color: #008000; }
|
||||
.stat-card.is-info { background: linear-gradient(180deg, #edf4ff, #ffffff); }
|
||||
.stat-card.is-info::before { border-left-color: #000078; }
|
||||
.stat-card.is-warning { background: linear-gradient(180deg, #ffffcc, #ffffff); }
|
||||
.stat-card.is-warning::before { border-left-color: #ffcc00; }
|
||||
.stat-card.is-danger {
|
||||
color: #000;
|
||||
background: repeating-linear-gradient(45deg, #fff2f2 0 6px, #ffe1e1 6px 12px);
|
||||
}
|
||||
.stat-card.is-danger::before { border-left-color: #800000; }
|
||||
|
||||
.stat-label { margin: 0 0 6px; color: #333; font-size: 13px; line-height: 13px; font-weight: bold; }
|
||||
.stat-value { margin: 0 0 7px; font-size: 32px; line-height: 32px; font-weight: bold; }
|
||||
.stat-note { display: flex; gap: 4px; flex-wrap: wrap; margin: 0; color: #222; font-size: 12px; line-height: 14px; }
|
||||
.stat-note-pill {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
min-height: 18px;
|
||||
padding: 1px 6px;
|
||||
background: #dfdfdf;
|
||||
border-top: 1px solid #ffffff;
|
||||
border-left: 1px solid #ffffff;
|
||||
border-right: 1px solid #808080;
|
||||
border-bottom: 1px solid #808080;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
/* Main two-column grid */
|
||||
.dashboard-main-grid {
|
||||
display: grid;
|
||||
grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
|
||||
gap: 12px;
|
||||
align-items: start;
|
||||
}
|
||||
|
||||
.dashboard-span-2 { grid-column: 1 / -1; }
|
||||
|
||||
/* Dashboard body */
|
||||
.dashboard-body {
|
||||
display: grid;
|
||||
gap: 12px;
|
||||
padding: 10px;
|
||||
}
|
||||
|
||||
/* Section windows */
|
||||
.section-window { min-height: 0; box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf, 3px 4px 0 rgba(0,0,0,.38); }
|
||||
.section-body { margin: 0 6px 6px; padding: 8px; min-height: 0; }
|
||||
|
||||
/* Scroll panels */
|
||||
.scroll-panel { overflow: auto; background: #ffffff; border-top: 2px solid #606060; border-left: 2px solid #606060; border-right: 2px solid #ffffff; border-bottom: 2px solid #ffffff; }
|
||||
.alerts-scroll { height: 326px; }
|
||||
.boxes-scroll { height: 352px; }
|
||||
.activity-scroll { height: 326px; }
|
||||
|
||||
/* Alerts */
|
||||
.alert-list { display: grid; min-width: 0; }
|
||||
.alert-row {
|
||||
display: grid;
|
||||
grid-template-columns: 72px minmax(0, 1fr) auto;
|
||||
gap: 8px;
|
||||
align-items: start;
|
||||
min-height: 74px;
|
||||
padding: 7px;
|
||||
color: #000;
|
||||
border-bottom: 1px solid #dfdfdf;
|
||||
background: #ffffff;
|
||||
}
|
||||
.alert-row:nth-child(even) { background: #f5f8ff; }
|
||||
.alert-row.is-dismissed { display: none; }
|
||||
.alert-severity {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
min-width: 60px;
|
||||
min-height: 20px;
|
||||
padding: 2px 5px;
|
||||
text-transform: uppercase;
|
||||
font-weight: bold;
|
||||
background: #dfdfdf;
|
||||
border-top: 1px solid #ffffff;
|
||||
border-left: 1px solid #ffffff;
|
||||
border-right: 1px solid #808080;
|
||||
border-bottom: 1px solid #808080;
|
||||
}
|
||||
.alert-row[data-severity="low"] .alert-severity { color: #000078; }
|
||||
.alert-row[data-severity="medium"] .alert-severity { color: #8a6200; background: #ffffcc; }
|
||||
.alert-row[data-severity="high"] .alert-severity { color: #ffffff; background: #800000; }
|
||||
.alert-title { margin: 0 0 3px; font-weight: bold; font-size: 14px; line-height: 15px; }
|
||||
.alert-desc { margin: 0 0 3px; color: #333; font-size: 12px; line-height: 14px; }
|
||||
.alert-trace { margin: 0; color: #555; font-family: 'MonoCraft', 'Courier New', monospace; font-size: 10px; line-height: 13px; overflow-wrap: anywhere; }
|
||||
.alert-actions { display: flex; gap: 5px; flex-wrap: wrap; justify-content: flex-end; }
|
||||
|
||||
/* Boxes table */
|
||||
.box-table {
|
||||
width: 100%;
|
||||
min-width: 900px;
|
||||
border-collapse: collapse;
|
||||
color: #000;
|
||||
font-size: 12px;
|
||||
line-height: 14px;
|
||||
}
|
||||
.box-table th, .box-table td { padding: 6px 7px; border-bottom: 1px solid #dfdfdf; text-align: left; vertical-align: middle; }
|
||||
.box-table th { position: sticky; top: 0; z-index: 5; background: #dfdfdf; border-bottom: 1px solid #808080; }
|
||||
.box-table tr:nth-child(even) td { background: #f5f8ff; }
|
||||
.box-actions { display: flex; gap: 5px; flex-wrap: nowrap; }
|
||||
.box-action-button { min-width: 62px; height: 22px; padding: 0 6px; font-size: 12px; line-height: 12px; }
|
||||
|
||||
/* Activity */
|
||||
.activity-list { display: grid; }
|
||||
.activity-row {
|
||||
display: grid;
|
||||
grid-template-columns: 56px minmax(0, 1fr) auto;
|
||||
gap: 9px;
|
||||
align-items: center;
|
||||
min-height: 48px;
|
||||
padding: 6px 8px;
|
||||
border-bottom: 1px solid #dfdfdf;
|
||||
background: #ffffff;
|
||||
color: #000;
|
||||
}
|
||||
.activity-row:nth-child(even) { background: #f5f8ff; }
|
||||
.activity-time { font-weight: bold; color: #000078; }
|
||||
.activity-title { margin: 0 0 2px; font-weight: bold; }
|
||||
.activity-meta { margin: 0; color: #555; font-size: 12px; line-height: 13px; }
|
||||
|
||||
/* Modal / Popup */
|
||||
.modal-backdrop {
|
||||
position: fixed;
|
||||
inset: 0;
|
||||
display: none;
|
||||
background: rgba(128, 128, 128, .42);
|
||||
z-index: 70;
|
||||
}
|
||||
.modal-backdrop.is-visible { display: block; }
|
||||
|
||||
.popup-window {
|
||||
position: fixed;
|
||||
left: 50%;
|
||||
top: 50%;
|
||||
transform: translate(calc(-50% - 1px), -50%);
|
||||
width: min(760px, calc(100vw - 24px));
|
||||
max-height: min(760px, calc(100vh - 24px));
|
||||
display: none;
|
||||
z-index: 80;
|
||||
}
|
||||
.popup-window.is-visible { display: flex; animation: popup-open 160ms steps(5, end); }
|
||||
@keyframes popup-open {
|
||||
from { transform: translate(calc(-50% - 1px), calc(-50% + 10px)) scale(.97); opacity: .45; }
|
||||
to { transform: translate(calc(-50% - 1px), -50%) scale(1); opacity: 1; }
|
||||
}
|
||||
.popup-body { margin: 0 6px 6px; padding: 10px; max-height: calc(100vh - 90px); overflow: auto; color: #000; }
|
||||
.metadata-pre {
|
||||
min-height: 240px;
|
||||
margin: 0;
|
||||
padding: 10px;
|
||||
overflow: auto;
|
||||
color: #b7ffc8;
|
||||
background: #030403;
|
||||
background-image: repeating-linear-gradient(transparent 0 4px, rgba(0,255,102,.018) 4px 6px);
|
||||
font-family: 'MonoCraft', 'Courier New', monospace;
|
||||
font-size: 12px;
|
||||
line-height: 16px;
|
||||
white-space: pre-wrap;
|
||||
}
|
||||
|
||||
/* Tiny button (for alerts / boxes) */
|
||||
.tiny-button {
|
||||
min-width: 56px;
|
||||
height: 22px;
|
||||
display: inline-grid;
|
||||
place-items: center;
|
||||
padding: 0 7px;
|
||||
color: #000;
|
||||
background: var(--w98-gray);
|
||||
border-top: 1px solid #ffffff;
|
||||
border-left: 1px solid #ffffff;
|
||||
border-right: 1px solid #000000;
|
||||
border-bottom: 1px solid #000000;
|
||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf;
|
||||
font-size: 12px;
|
||||
line-height: 12px;
|
||||
text-decoration: none;
|
||||
}
|
||||
.tiny-button:hover { filter: brightness(1.06); }
|
||||
|
||||
/* Compact mode */
|
||||
body.is-compact .dashboard-body { gap: 8px; }
|
||||
body.is-compact .section-body { padding: 5px; }
|
||||
body.is-compact .alerts-scroll,
|
||||
body.is-compact .boxes-scroll { height: 280px; }
|
||||
body.is-compact .activity-scroll { height: 280px; }
|
||||
body.is-compact .alert-row { min-height: 62px; }
|
||||
body.is-compact .activity-row { min-height: 42px; }
|
||||
|
||||
/* Responsive: medium */
|
||||
@media (max-width: 1180px) {
|
||||
.stats-grid { grid-template-columns: repeat(2, minmax(0, 1fr)); }
|
||||
.dashboard-hero { grid-template-columns: 1fr; }
|
||||
.dashboard-main-grid { grid-template-columns: 1fr; }
|
||||
.dashboard-span-2 { grid-column: auto; }
|
||||
.alerts-scroll, .boxes-scroll { height: 310px; }
|
||||
.activity-scroll { height: 310px; }
|
||||
}
|
||||
|
||||
/* Responsive: small (mobile) */
|
||||
@media (max-width: 760px) {
|
||||
.dashboard-body { padding: 6px; gap: 8px; }
|
||||
.stats-grid { grid-template-columns: 1fr; }
|
||||
.stat-card { min-height: 112px; }
|
||||
.alert-row { grid-template-columns: 1fr; min-height: 0; }
|
||||
.alert-actions { justify-content: flex-start; }
|
||||
.alerts-scroll, .boxes-scroll, .activity-scroll { height: 320px; }
|
||||
.boxes-scroll { overflow-x: auto; }
|
||||
.activity-row { grid-template-columns: 48px minmax(0, 1fr); }
|
||||
.activity-row .tag { grid-column: 2; justify-self: start; }
|
||||
.popup-window {
|
||||
left: 0;
|
||||
top: 0;
|
||||
transform: none;
|
||||
width: 100vw;
|
||||
height: 100dvh;
|
||||
max-height: none;
|
||||
border: 0;
|
||||
box-shadow: none;
|
||||
}
|
||||
.popup-window.is-visible { animation: popup-open-mobile 150ms steps(5, end); }
|
||||
@keyframes popup-open-mobile { from { transform: translateY(10px); opacity: .35; } to { transform: translateY(0); opacity: 1; } }
|
||||
.popup-body { max-height: calc(100dvh - 40px); }
|
||||
}
|
||||
@@ -128,3 +128,81 @@
|
||||
font-size: 13px;
|
||||
line-height: 13px;
|
||||
}
|
||||
|
||||
/* Raised panel - appears to sit above the surface */
|
||||
.raised-panel {
|
||||
background: #dfdfdf;
|
||||
border-top: 1px solid #ffffff;
|
||||
border-left: 1px solid #ffffff;
|
||||
border-right: 1px solid #808080;
|
||||
border-bottom: 1px solid #808080;
|
||||
box-shadow: inset 1px 1px 0 #f7f7f7, inset -1px -1px 0 #b0b0b0;
|
||||
}
|
||||
|
||||
/* Sunken panel - appears to be inset into the surface */
|
||||
.sunken-panel {
|
||||
background-color: #ffffff;
|
||||
background-image:
|
||||
linear-gradient(180deg, rgba(255,255,255,.9), rgba(238,238,238,.58)),
|
||||
repeating-linear-gradient(0deg, rgba(0,0,0,.025) 0 1px, transparent 1px 6px);
|
||||
border-top: 2px solid #606060;
|
||||
border-left: 2px solid #606060;
|
||||
border-right: 2px solid #ffffff;
|
||||
border-bottom: 2px solid #ffffff;
|
||||
}
|
||||
|
||||
/* Scroll panel - used for scrollable content areas within windows */
|
||||
.scroll-panel {
|
||||
overflow: auto;
|
||||
background: #ffffff;
|
||||
border-top: 2px solid #606060;
|
||||
border-left: 2px solid #606060;
|
||||
border-right: 2px solid #ffffff;
|
||||
border-bottom: 2px solid #ffffff;
|
||||
}
|
||||
|
||||
/* Meter track for progress bars */
|
||||
.meter-track {
|
||||
display: block;
|
||||
height: 14px;
|
||||
margin-top: 9px;
|
||||
background-color: #ffffff;
|
||||
background-image: repeating-linear-gradient(to right, rgba(0,0,0,.06) 0 1px, transparent 1px 18px);
|
||||
border-top: 2px solid #808080;
|
||||
border-left: 2px solid #808080;
|
||||
border-right: 2px solid #ffffff;
|
||||
border-bottom: 2px solid #ffffff;
|
||||
}
|
||||
|
||||
.meter-bar {
|
||||
display: block;
|
||||
height: 100%;
|
||||
width: var(--meter, 0%);
|
||||
background-color: #000078;
|
||||
background-image: repeating-linear-gradient(to right, rgba(255,255,255,.13) 0 1px, transparent 1px 18px);
|
||||
}
|
||||
|
||||
/* Tag styles for status indicators */
|
||||
.tag {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
min-height: 17px;
|
||||
margin: 1px 2px 1px 0;
|
||||
padding: 1px 5px;
|
||||
color: #000000;
|
||||
background: #dfdfdf;
|
||||
border: 1px solid #808080;
|
||||
box-shadow: inset 1px 1px 0 #ffffff;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.tag.ok { color: #008000; background: #eeffee; }
|
||||
.tag.info { color: #000078; background: #edf4ff; }
|
||||
.tag.warn { color: #8a6200; background: #ffffcc; }
|
||||
.tag.danger { color: #ffffff; background: #800000; }
|
||||
|
||||
/* Titlebar animation - gradient drift */
|
||||
@keyframes titlebar-drift {
|
||||
from { background-position: 0% 50%; }
|
||||
to { background-position: 100% 50%; }
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user