617 lines
18 KiB
CSS
617 lines
18 KiB
CSS
/* ============================================================
|
|
TELEMETRY DASHBOARD — live ops for jaeswift.xyz
|
|
============================================================ */
|
|
|
|
:root {
|
|
--tm-green: #00cc33;
|
|
--tm-green-dim: #00cc33aa;
|
|
--tm-green-glow: rgba(0,204,51,0.4);
|
|
--tm-amber: #c9a227;
|
|
--tm-red: #8b0000;
|
|
--tm-red-bright: #ff2d2d;
|
|
--tm-sol: #14F195;
|
|
--tm-bg: #0a0a0a;
|
|
--tm-panel: #111111;
|
|
--tm-panel-alt: #161616;
|
|
--tm-border: #1c1c1c;
|
|
--tm-border-hot: #00cc3333;
|
|
--tm-text: #c9c9c9;
|
|
--tm-muted: #6b6b6b;
|
|
}
|
|
|
|
/* ─── Boot animation ──────────────────────────────── */
|
|
.tm-boot {
|
|
position: fixed; inset: 0;
|
|
background: #000;
|
|
z-index: 9999;
|
|
display: flex; flex-direction: column;
|
|
align-items: center; justify-content: center;
|
|
font-family: 'JetBrains Mono', monospace;
|
|
color: var(--tm-green);
|
|
gap: 1.2rem;
|
|
text-shadow: 0 0 8px var(--tm-green-glow);
|
|
transition: opacity 0.5s;
|
|
}
|
|
.tm-boot.hidden { opacity: 0; pointer-events: none; }
|
|
.tm-boot-line {
|
|
font-size: 1rem;
|
|
opacity: 0;
|
|
letter-spacing: 0.12em;
|
|
animation: tm-bootfade 0.4s forwards;
|
|
}
|
|
.tm-boot-line::before { content: '> '; color: var(--tm-green-dim); }
|
|
@keyframes tm-bootfade {
|
|
from { opacity: 0; transform: translateY(6px); }
|
|
to { opacity: 1; transform: none; }
|
|
}
|
|
.tm-boot-bar {
|
|
width: 300px; height: 3px;
|
|
background: #0a2a0a;
|
|
border: 1px solid var(--tm-border-hot);
|
|
position: relative;
|
|
overflow: hidden;
|
|
}
|
|
.tm-boot-bar::after {
|
|
content: ''; position: absolute;
|
|
top: 0; left: 0; height: 100%; width: 0;
|
|
background: var(--tm-green);
|
|
box-shadow: 0 0 10px var(--tm-green);
|
|
animation: tm-bootbar 1.3s forwards;
|
|
}
|
|
@keyframes tm-bootbar { to { width: 100%; } }
|
|
|
|
/* ─── Dashboard container ─────────────────────────── */
|
|
.tm-dashboard {
|
|
padding: 0 1.5rem 4rem;
|
|
max-width: 1800px;
|
|
margin: 0 auto;
|
|
font-family: 'JetBrains Mono', monospace;
|
|
color: var(--tm-text);
|
|
position: relative;
|
|
}
|
|
|
|
/* ─── Alert banner ────────────────────────────────── */
|
|
.tm-alerts {
|
|
position: sticky; top: calc(var(--nav-height) + 8px);
|
|
z-index: 100;
|
|
display: none;
|
|
margin-bottom: 1rem;
|
|
border: 1px solid var(--tm-red);
|
|
background: rgba(139,0,0,0.15);
|
|
padding: 0.5rem 0.8rem;
|
|
overflow: hidden;
|
|
font-size: 0.82rem;
|
|
letter-spacing: 0.08em;
|
|
}
|
|
.tm-alerts.visible { display: block; animation: tm-pulse-red 2s infinite; }
|
|
.tm-alerts.amber {
|
|
border-color: var(--tm-amber);
|
|
background: rgba(201,162,39,0.1);
|
|
animation: none;
|
|
}
|
|
@keyframes tm-pulse-red {
|
|
0%,100% { box-shadow: 0 0 0 0 rgba(255,45,45,0.3); }
|
|
50% { box-shadow: 0 0 0 6px rgba(255,45,45,0); }
|
|
}
|
|
.tm-alerts-content {
|
|
display: inline-block;
|
|
white-space: nowrap;
|
|
animation: tm-marquee 30s linear infinite;
|
|
padding-left: 100%;
|
|
}
|
|
.tm-alert-item {
|
|
display: inline-block;
|
|
margin-right: 2.5rem;
|
|
}
|
|
.tm-alert-item.red { color: var(--tm-red-bright); }
|
|
.tm-alert-item.amber { color: var(--tm-amber); }
|
|
.tm-alert-item.info { color: var(--tm-green); }
|
|
.tm-alert-item::before { content: '▲ '; }
|
|
@keyframes tm-marquee {
|
|
from { transform: translateX(0); }
|
|
to { transform: translateX(-100%); }
|
|
}
|
|
|
|
/* ─── Top status bar (LIVE indicator) ─────────────── */
|
|
.tm-statusbar {
|
|
display: flex;
|
|
justify-content: space-between;
|
|
align-items: center;
|
|
padding: 0.6rem 0.9rem;
|
|
background: var(--tm-panel);
|
|
border: 1px solid var(--tm-border);
|
|
margin-bottom: 1.2rem;
|
|
font-size: 0.8rem;
|
|
}
|
|
.tm-live {
|
|
display: flex; align-items: center; gap: 0.6rem;
|
|
letter-spacing: 0.15em;
|
|
}
|
|
.tm-live-dot {
|
|
width: 10px; height: 10px; border-radius: 50%;
|
|
background: var(--tm-green);
|
|
box-shadow: 0 0 12px var(--tm-green);
|
|
animation: tm-blink 1.2s infinite;
|
|
}
|
|
.tm-live-dot.amber { background: var(--tm-amber); box-shadow: 0 0 8px var(--tm-amber); }
|
|
.tm-live-dot.red { background: var(--tm-red-bright); box-shadow: 0 0 12px var(--tm-red-bright); }
|
|
@keyframes tm-blink {
|
|
0%,100% { opacity: 1; }
|
|
50% { opacity: 0.25; }
|
|
}
|
|
.tm-visitors {
|
|
display: flex; align-items: center; gap: 0.8rem;
|
|
color: var(--tm-muted);
|
|
}
|
|
.tm-visitors strong {
|
|
color: var(--tm-sol);
|
|
font-size: 1.1rem;
|
|
text-shadow: 0 0 8px rgba(20,241,149,0.5);
|
|
}
|
|
.tm-sound-toggle {
|
|
background: none; border: 1px solid var(--tm-border);
|
|
color: var(--tm-muted); cursor: pointer;
|
|
padding: 0.3rem 0.6rem;
|
|
font-family: inherit; font-size: 0.75rem;
|
|
transition: all 0.2s;
|
|
}
|
|
.tm-sound-toggle:hover { color: var(--tm-green); border-color: var(--tm-green); }
|
|
.tm-sound-toggle.on { color: var(--tm-green); border-color: var(--tm-green); }
|
|
|
|
/* ─── Grid layout ─────────────────────────────────── */
|
|
.tm-row {
|
|
display: grid;
|
|
gap: 1rem;
|
|
margin-bottom: 1rem;
|
|
}
|
|
.tm-row.r1 { grid-template-columns: repeat(4, 1fr); }
|
|
.tm-row.r2 { grid-template-columns: 1.2fr 1fr; }
|
|
.tm-row.r3 { grid-template-columns: 1fr; }
|
|
.tm-row.r4 { grid-template-columns: 1.3fr 1fr; }
|
|
.tm-row.r5 { grid-template-columns: 1fr 1.3fr; }
|
|
.tm-row.r6 { grid-template-columns: 1.3fr 1fr; }
|
|
.tm-row.r7 { grid-template-columns: 1fr 1.3fr; }
|
|
|
|
@media (max-width: 1200px) {
|
|
.tm-row.r1 { grid-template-columns: repeat(2, 1fr); }
|
|
}
|
|
@media (max-width: 768px) {
|
|
.tm-dashboard { padding: 0 0.6rem 3rem; }
|
|
.tm-row,
|
|
.tm-row.r1, .tm-row.r2, .tm-row.r4,
|
|
.tm-row.r5, .tm-row.r6, .tm-row.r7 {
|
|
grid-template-columns: 1fr;
|
|
}
|
|
.tm-alerts { font-size: 0.72rem; }
|
|
}
|
|
|
|
/* ─── Panel base ──────────────────────────────────── */
|
|
.tm-panel {
|
|
background: var(--tm-panel);
|
|
border: 1px solid var(--tm-border);
|
|
padding: 0.8rem 1rem 1rem;
|
|
position: relative;
|
|
min-height: 0;
|
|
overflow: hidden;
|
|
transition: border-color 0.3s;
|
|
}
|
|
.tm-panel:hover { border-color: var(--tm-border-hot); }
|
|
.tm-panel-title {
|
|
font-size: 0.72rem;
|
|
letter-spacing: 0.2em;
|
|
color: var(--tm-green-dim);
|
|
margin-bottom: 0.7rem;
|
|
padding-bottom: 0.4rem;
|
|
border-bottom: 1px dashed var(--tm-border);
|
|
display: flex;
|
|
justify-content: space-between;
|
|
align-items: center;
|
|
}
|
|
.tm-panel-title::before { content: '▸ '; color: var(--tm-green); }
|
|
.tm-panel-title .badge {
|
|
background: rgba(0,204,51,0.1);
|
|
color: var(--tm-green);
|
|
padding: 1px 6px;
|
|
font-size: 0.65rem;
|
|
letter-spacing: 0.1em;
|
|
border: 1px solid var(--tm-border-hot);
|
|
}
|
|
|
|
/* ─── Gauges (CPU/MEM) ────────────────────────────── */
|
|
.tm-gauge-wrap {
|
|
display: flex;
|
|
flex-direction: column;
|
|
align-items: center;
|
|
gap: 0.4rem;
|
|
}
|
|
.tm-gauge-canvas {
|
|
width: 140px; height: 140px;
|
|
max-width: 100%;
|
|
}
|
|
.tm-gauge-value {
|
|
font-family: 'Orbitron', sans-serif;
|
|
font-size: 1.6rem;
|
|
font-weight: 700;
|
|
color: var(--tm-green);
|
|
text-shadow: 0 0 10px var(--tm-green-glow);
|
|
letter-spacing: 0.05em;
|
|
}
|
|
.tm-gauge-value.warn { color: var(--tm-amber); text-shadow: 0 0 10px rgba(201,162,39,0.5); }
|
|
.tm-gauge-value.crit { color: var(--tm-red-bright); text-shadow: 0 0 10px rgba(255,45,45,0.6); }
|
|
.tm-gauge-label {
|
|
font-size: 0.7rem;
|
|
color: var(--tm-muted);
|
|
letter-spacing: 0.12em;
|
|
}
|
|
.tm-spark {
|
|
width: 100%; height: 40px;
|
|
margin-top: 0.5rem;
|
|
background: #000;
|
|
border: 1px solid var(--tm-border);
|
|
}
|
|
|
|
/* ─── Disk bars ──────────────────────────────────── */
|
|
.tm-disks { display: flex; flex-direction: column; gap: 0.7rem; }
|
|
.tm-disk {
|
|
display: flex; flex-direction: column; gap: 0.2rem;
|
|
font-size: 0.72rem;
|
|
}
|
|
.tm-disk-head {
|
|
display: flex; justify-content: space-between;
|
|
color: var(--tm-text);
|
|
}
|
|
.tm-disk-head .mount { color: var(--tm-green); }
|
|
.tm-disk-bar {
|
|
font-family: 'JetBrains Mono', monospace;
|
|
color: var(--tm-green);
|
|
letter-spacing: -0.5px;
|
|
font-size: 0.9rem;
|
|
line-height: 1;
|
|
}
|
|
.tm-disk-bar.warn { color: var(--tm-amber); }
|
|
.tm-disk-bar.crit { color: var(--tm-red-bright); }
|
|
|
|
/* ─── Network ────────────────────────────────────── */
|
|
.tm-net {
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 0.3rem;
|
|
font-size: 0.8rem;
|
|
}
|
|
.tm-net-row {
|
|
display: flex; justify-content: space-between;
|
|
align-items: baseline;
|
|
padding: 0.2rem 0;
|
|
}
|
|
.tm-net-row .label { color: var(--tm-muted); font-size: 0.7rem; letter-spacing: 0.15em; }
|
|
.tm-net-row .val {
|
|
font-family: 'Orbitron', sans-serif;
|
|
font-size: 1.25rem;
|
|
color: var(--tm-sol);
|
|
text-shadow: 0 0 8px rgba(20,241,149,0.4);
|
|
}
|
|
.tm-net-row .val.tx { color: var(--tm-green); text-shadow: 0 0 8px var(--tm-green-glow); }
|
|
|
|
/* ─── Tables (services, crons) ─────────────────────── */
|
|
.tm-table {
|
|
width: 100%;
|
|
border-collapse: collapse;
|
|
font-size: 0.74rem;
|
|
}
|
|
.tm-table th {
|
|
text-align: left;
|
|
padding: 0.35rem 0.4rem;
|
|
border-bottom: 1px solid var(--tm-border);
|
|
color: var(--tm-muted);
|
|
letter-spacing: 0.1em;
|
|
font-weight: 400;
|
|
}
|
|
.tm-table td {
|
|
padding: 0.4rem;
|
|
border-bottom: 1px dashed #1a1a1a;
|
|
}
|
|
.tm-table tr:hover td { background: rgba(0,204,51,0.04); }
|
|
.tm-table .svc-name { color: var(--tm-text); font-weight: 500; }
|
|
.tm-table .dot {
|
|
display: inline-block;
|
|
width: 8px; height: 8px; border-radius: 50%;
|
|
margin-right: 6px;
|
|
vertical-align: middle;
|
|
}
|
|
.dot.up { background: var(--tm-green); box-shadow: 0 0 8px var(--tm-green); animation: tm-blink 1.6s infinite; }
|
|
.dot.down { background: var(--tm-red-bright); box-shadow: 0 0 8px var(--tm-red-bright); }
|
|
.dot.unknown { background: var(--tm-muted); }
|
|
.dot.warn { background: var(--tm-amber); box-shadow: 0 0 6px var(--tm-amber); }
|
|
|
|
.tm-table .cron-status.ok { color: var(--tm-green); }
|
|
.tm-table .cron-status.fail { color: var(--tm-red-bright); }
|
|
.tm-table .cron-status.unknown { color: var(--tm-muted); }
|
|
|
|
.tm-expandable { cursor: pointer; }
|
|
.tm-expanded-row {
|
|
background: #0d0d0d;
|
|
padding: 0.6rem 0.8rem;
|
|
color: var(--tm-muted);
|
|
font-size: 0.68rem;
|
|
white-space: pre-wrap;
|
|
max-height: 160px;
|
|
overflow: auto;
|
|
}
|
|
|
|
/* ─── Nginx stats ────────────────────────────────── */
|
|
.tm-nginx-grid {
|
|
display: grid;
|
|
grid-template-columns: repeat(5, 1fr);
|
|
gap: 0.8rem;
|
|
margin-bottom: 0.8rem;
|
|
}
|
|
@media (max-width: 768px) { .tm-nginx-grid { grid-template-columns: repeat(2, 1fr); } }
|
|
.tm-stat {
|
|
background: var(--tm-panel-alt);
|
|
padding: 0.6rem 0.8rem;
|
|
border-left: 2px solid var(--tm-green);
|
|
text-align: left;
|
|
}
|
|
.tm-stat .v {
|
|
font-family: 'Orbitron', sans-serif;
|
|
font-size: 1.15rem;
|
|
color: var(--tm-green);
|
|
text-shadow: 0 0 6px var(--tm-green-glow);
|
|
display: block;
|
|
}
|
|
.tm-stat .l {
|
|
font-size: 0.65rem;
|
|
color: var(--tm-muted);
|
|
letter-spacing: 0.1em;
|
|
margin-top: 0.2rem;
|
|
}
|
|
.tm-stat.err { border-left-color: var(--tm-red-bright); }
|
|
.tm-stat.err .v { color: var(--tm-red-bright); text-shadow: 0 0 6px rgba(255,45,45,0.4); }
|
|
|
|
.tm-toppages {
|
|
display: grid;
|
|
grid-template-columns: 1fr 1fr;
|
|
gap: 0.6rem;
|
|
font-size: 0.72rem;
|
|
}
|
|
@media (max-width: 768px) { .tm-toppages { grid-template-columns: 1fr; } }
|
|
.tm-toppages ul { list-style: none; padding: 0; margin: 0; }
|
|
.tm-toppages li {
|
|
display: flex; justify-content: space-between;
|
|
padding: 0.25rem 0;
|
|
border-bottom: 1px dashed #1a1a1a;
|
|
}
|
|
.tm-toppages li .path {
|
|
color: var(--tm-text);
|
|
overflow: hidden; text-overflow: ellipsis;
|
|
white-space: nowrap;
|
|
max-width: 70%;
|
|
}
|
|
.tm-toppages li .c {
|
|
color: var(--tm-sol);
|
|
font-family: 'Orbitron', sans-serif;
|
|
}
|
|
.tm-toppages h4 {
|
|
color: var(--tm-green-dim);
|
|
font-size: 0.7rem;
|
|
letter-spacing: 0.15em;
|
|
margin: 0 0 0.3rem;
|
|
font-weight: 400;
|
|
}
|
|
|
|
/* ─── Security / SSL panels ───────────────────────── */
|
|
.tm-security-grid {
|
|
display: grid;
|
|
grid-template-columns: repeat(4, 1fr);
|
|
gap: 0.8rem;
|
|
margin-bottom: 0.8rem;
|
|
}
|
|
.tm-jails {
|
|
font-size: 0.72rem;
|
|
margin-top: 0.5rem;
|
|
}
|
|
.tm-jails .jail {
|
|
display: flex; justify-content: space-between;
|
|
padding: 0.2rem 0;
|
|
border-bottom: 1px dashed #1a1a1a;
|
|
}
|
|
.tm-ssl { font-size: 0.74rem; }
|
|
.tm-ssl .ssl-row {
|
|
display: flex; justify-content: space-between;
|
|
padding: 0.4rem 0;
|
|
border-bottom: 1px dashed #1a1a1a;
|
|
}
|
|
.tm-ssl .days {
|
|
font-family: 'Orbitron', sans-serif;
|
|
color: var(--tm-green);
|
|
}
|
|
.tm-ssl .days.warn { color: var(--tm-amber); }
|
|
.tm-ssl .days.crit { color: var(--tm-red-bright); }
|
|
.tm-ssl .days.fail { color: var(--tm-muted); }
|
|
|
|
/* ─── Stack inventory ─────────────────────────────── */
|
|
.tm-stack {
|
|
display: grid;
|
|
grid-template-columns: repeat(auto-fill, minmax(140px, 1fr));
|
|
gap: 0.5rem;
|
|
}
|
|
.tm-stack-item {
|
|
background: var(--tm-panel-alt);
|
|
border: 1px solid var(--tm-border);
|
|
padding: 0.45rem 0.6rem;
|
|
font-size: 0.72rem;
|
|
}
|
|
.tm-stack-item .k {
|
|
color: var(--tm-muted); font-size: 0.62rem;
|
|
letter-spacing: 0.1em; display: block;
|
|
}
|
|
.tm-stack-item .v {
|
|
color: var(--tm-green);
|
|
font-family: 'Orbitron', sans-serif;
|
|
overflow: hidden; text-overflow: ellipsis;
|
|
white-space: nowrap;
|
|
display: block;
|
|
margin-top: 0.15rem;
|
|
}
|
|
|
|
/* ─── Repos ───────────────────────────────────────── */
|
|
.tm-repo {
|
|
background: var(--tm-panel-alt);
|
|
padding: 0.6rem 0.8rem;
|
|
border-left: 2px solid var(--tm-sol);
|
|
margin-bottom: 0.5rem;
|
|
font-size: 0.75rem;
|
|
}
|
|
.tm-repo-head {
|
|
display: flex; justify-content: space-between;
|
|
align-items: center;
|
|
margin-bottom: 0.3rem;
|
|
}
|
|
.tm-repo-head .name { color: var(--tm-sol); font-weight: 600; }
|
|
.tm-repo-head .sha {
|
|
font-family: 'JetBrains Mono', monospace;
|
|
color: var(--tm-amber);
|
|
background: rgba(201,162,39,0.08);
|
|
padding: 1px 6px;
|
|
font-size: 0.68rem;
|
|
}
|
|
.tm-repo .msg { color: var(--tm-text); margin-bottom: 0.2rem; }
|
|
.tm-repo .meta { color: var(--tm-muted); font-size: 0.68rem; }
|
|
.tm-repo .dirty-badge {
|
|
color: var(--tm-red-bright);
|
|
margin-left: 0.5rem;
|
|
animation: tm-blink 1.4s infinite;
|
|
}
|
|
|
|
/* ─── Geo map ─────────────────────────────────────── */
|
|
.tm-geo {
|
|
width: 100%;
|
|
min-height: 280px;
|
|
position: relative;
|
|
}
|
|
.tm-geo svg { width: 100%; height: auto; display: block; }
|
|
.tm-geo .country { fill: #1a2a1a; stroke: #0a0a0a; stroke-width: 0.5; transition: fill 0.3s; }
|
|
.tm-geo .country.hot { fill: var(--tm-green); }
|
|
.tm-geo .country:hover { fill: var(--tm-sol); cursor: pointer; }
|
|
.tm-geo-legend {
|
|
position: absolute;
|
|
bottom: 0.5rem; left: 0.5rem;
|
|
font-size: 0.65rem;
|
|
color: var(--tm-muted);
|
|
background: rgba(0,0,0,0.7);
|
|
padding: 0.3rem 0.5rem;
|
|
border: 1px solid var(--tm-border);
|
|
}
|
|
.tm-geo-empty {
|
|
text-align: center;
|
|
padding: 3rem 1rem;
|
|
color: var(--tm-muted);
|
|
font-size: 0.75rem;
|
|
}
|
|
.tm-geo-top {
|
|
margin-top: 0.6rem;
|
|
font-size: 0.7rem;
|
|
display: grid;
|
|
grid-template-columns: repeat(auto-fill, minmax(120px, 1fr));
|
|
gap: 0.3rem;
|
|
}
|
|
.tm-geo-top .c-item {
|
|
display: flex; justify-content: space-between;
|
|
padding: 0.15rem 0.4rem;
|
|
background: rgba(0,204,51,0.05);
|
|
border-left: 2px solid var(--tm-green);
|
|
}
|
|
.tm-geo-top .c-item .n { color: var(--tm-green); font-family: 'Orbitron', sans-serif; }
|
|
|
|
/* ─── Terminal tail ────────────────────────────────── */
|
|
.tm-tail {
|
|
background: #000;
|
|
border: 1px solid var(--tm-border);
|
|
padding: 0.6rem;
|
|
min-height: 280px;
|
|
max-height: 400px;
|
|
overflow-y: auto;
|
|
font-size: 0.7rem;
|
|
line-height: 1.4;
|
|
font-family: 'JetBrains Mono', monospace;
|
|
}
|
|
.tm-tail::-webkit-scrollbar { width: 6px; }
|
|
.tm-tail::-webkit-scrollbar-thumb { background: #1a2a1a; }
|
|
.tm-tail-line {
|
|
opacity: 0;
|
|
animation: tm-fadein 0.4s forwards;
|
|
padding: 1px 0;
|
|
border-bottom: 1px dotted #111;
|
|
white-space: nowrap;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
}
|
|
@keyframes tm-fadein {
|
|
from { opacity: 0; transform: translateX(-8px); }
|
|
to { opacity: 1; transform: none; }
|
|
}
|
|
.tm-tail-line .st-2xx { color: var(--tm-green); }
|
|
.tm-tail-line .st-3xx { color: var(--tm-sol); }
|
|
.tm-tail-line .st-4xx { color: var(--tm-amber); }
|
|
.tm-tail-line .st-5xx { color: var(--tm-red-bright); }
|
|
.tm-tail-line .method { color: var(--tm-green-dim); }
|
|
.tm-tail-line .path { color: var(--tm-text); }
|
|
.tm-tail-line .ip { color: var(--tm-muted); }
|
|
.tm-tail-line .time { color: #3a3a3a; font-size: 0.62rem; }
|
|
|
|
/* ─── Burn rate / deploys ─────────────────────────── */
|
|
.tm-burn {
|
|
font-size: 0.85rem;
|
|
color: var(--tm-text);
|
|
line-height: 1.9;
|
|
}
|
|
.tm-burn .big {
|
|
font-family: 'Orbitron', sans-serif;
|
|
color: var(--tm-amber);
|
|
font-size: 1.1rem;
|
|
text-shadow: 0 0 6px rgba(201,162,39,0.4);
|
|
}
|
|
.tm-burn .label { color: var(--tm-muted); font-size: 0.72rem; letter-spacing: 0.1em; display: block; }
|
|
|
|
.tm-ticker {
|
|
overflow: hidden;
|
|
white-space: nowrap;
|
|
font-size: 0.75rem;
|
|
padding: 0.4rem 0;
|
|
}
|
|
.tm-ticker-inner {
|
|
display: inline-block;
|
|
animation: tm-marquee 60s linear infinite;
|
|
padding-left: 100%;
|
|
}
|
|
.tm-ticker-item {
|
|
display: inline-block;
|
|
margin-right: 2rem;
|
|
color: var(--tm-text);
|
|
}
|
|
.tm-ticker-item .sha {
|
|
color: var(--tm-amber);
|
|
font-family: 'JetBrains Mono', monospace;
|
|
margin-right: 0.5rem;
|
|
}
|
|
.tm-ticker-item .iso { color: var(--tm-muted); margin-left: 0.6rem; font-size: 0.65rem; }
|
|
|
|
/* ─── Loading & error states ──────────────────────── */
|
|
.tm-loading {
|
|
color: var(--tm-muted);
|
|
font-size: 0.72rem;
|
|
padding: 1rem 0;
|
|
text-align: center;
|
|
letter-spacing: 0.1em;
|
|
}
|
|
.tm-loading::after {
|
|
content: '...';
|
|
animation: tm-dots 1.4s infinite;
|
|
}
|
|
@keyframes tm-dots {
|
|
0%,20% { content: '.'; }
|
|
40% { content: '..'; }
|
|
60% { content: '...'; }
|
|
80%,100% { content: '....'; }
|
|
}
|
|
.tm-err { color: var(--tm-red-bright); font-size: 0.72rem; padding: 0.5rem 0; }
|