*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }
body {
font-family: 'Space Grotesk', sans-serif;
background: var(--bg);
color: var(--text);
min-height: 100vh;
overflow-x: hidden;
}
/* ─── HEADER ─────────────────────────────────────────── */
.header {
position: sticky;
top: 0;
z-index: 100;
background: rgba(10,11,14,0.92);
backdrop-filter: blur(12px);
border-bottom: 1px solid var(--border);
display: flex;
align-items: center;
justify-content: space-between;
padding: 0 24px;
height: 56px;
}
.header-left { display: flex; align-items: center; gap: 16px; }
.back-btn {
background: var(--surface2);
border: 1px solid var(--border);
color: var(--muted);
font-family: 'Space Grotesk', sans-serif;
font-size: 13px;
font-weight: 500;
padding: 6px 14px;
border-radius: 6px;
cursor: pointer;
transition: all 0.15s;
letter-spacing: 0.02em;
}
.back-btn:hover { color: var(--text); border-color: var(--muted); }
.header-title {
font-size: 15px;
font-weight: 600;
letter-spacing: 0.04em;
color: var(--text);
}
.header-title span { color: var(--accent); }
.header-right { display: flex; align-items: center; gap: 10px; }
.btn {
font-family: 'Space Grotesk', sans-serif;
font-size: 13px;
font-weight: 500;
padding: 7px 16px;
border-radius: 6px;
cursor: pointer;
border: 1px solid transparent;
transition: all 0.15s;
letter-spacing: 0.02em;
white-space: nowrap;
}
.btn-ghost {
background: var(--surface2);
border-color: var(--border);
color: var(--muted);
}
.btn-ghost:hover { color: var(--text); border-color: var(--muted); }
.btn-accent {
background: rgba(0,229,255,0.1);
border-color: rgba(0,229,255,0.3);
color: var(--accent);
}
.btn-accent:hover { background: rgba(0,229,255,0.18); border-color: var(--accent); }
.btn-green {
background: rgba(0,230,118,0.1);
border-color: rgba(0,230,118,0.3);
color: var(--green);
}
.btn-green:hover { background: rgba(0,230,118,0.18); border-color: var(--green); }
.btn:disabled {
opacity: 0.4;
cursor: not-allowed;
pointer-events: none;
}
/* ─── MAIN ────────────────────────────────────────────── */
.main { max-width: 1100px; margin: 0 auto; padding: 28px 24px 60px; }
/* ─── VPC CONFIG ──────────────────────────────────────── */
.card {
background: var(--surface);
border: 1px solid var(--border);
border-radius: 10px;
padding: 20px 24px;
margin-bottom: 18px;
}
.card-title {
font-size: 11px;
font-weight: 600;
letter-spacing: 0.1em;
text-transform: uppercase;
color: var(--muted);
margin-bottom: 16px;
display: flex;
align-items: center;
gap: 8px;
}
.card-title::after {
content: '';
flex: 1;
height: 1px;
background: var(--border);
}
.vpc-row { display: flex; align-items: flex-end; gap: 16px; flex-wrap: wrap; }
.field { display: flex; flex-direction: column; gap: 6px; }
.field label {
font-size: 11px;
font-weight: 500;
color: var(--muted);
letter-spacing: 0.06em;
text-transform: uppercase;
}
.field input, .field select {
font-family: 'JetBrains Mono', monospace;
font-size: 13px;
background: var(--surface2);
border: 1px solid var(--border);
color: var(--text);
padding: 8px 12px;
border-radius: 6px;
outline: none;
transition: border-color 0.15s;
min-width: 160px;
}
.field input:focus, .field select:focus { border-color: var(--accent); }
.field input.error { border-color: var(--red); }
.field select option { background: var(--surface2); }
.stat-pill {
display: flex;
align-items: center;
gap: 8px;
background: var(--surface2);
border: 1px solid var(--border);
border-radius: 6px;
padding: 8px 14px;
font-family: 'JetBrains Mono', monospace;
font-size: 12px;
}
.stat-pill .stat-label { color: var(--muted); }
.stat-pill .stat-value { color: var(--accent); font-weight: 600; }
/* ─── VISUALIZER BAR ──────────────────────────────────── */
.viz-wrap { margin-bottom: 18px; }
.viz-bar-container {
height: 36px;
background: var(--surface2);
border: 1px solid var(--border);
border-radius: 8px;
overflow: hidden;
display: flex;
position: relative;
}
.viz-segment {
height: 100%;
display: flex;
align-items: center;
justify-content: center;
font-family: 'JetBrains Mono', monospace;
font-size: 10px;
font-weight: 600;
letter-spacing: 0.04em;
color: rgba(0,0,0,0.7);
overflow: hidden;
transition: width 0.3s ease;
cursor: default;
position: relative;
}
.viz-segment-label {
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
padding: 0 4px;
pointer-events: none;
}
.viz-free {
height: 100%;
flex: 1;
background: repeating-linear-gradient(
45deg,
transparent,
transparent 4px,
rgba(90,100,120,0.15) 4px,
rgba(90,100,120,0.15) 8px
);
}
.viz-legend {
display: flex;
gap: 14px;
flex-wrap: wrap;
margin-top: 8px;
}
.viz-legend-item {
display: flex;
align-items: center;
gap: 6px;
font-size: 11px;
color: var(--muted);
font-family: 'JetBrains Mono', monospace;
}
.viz-legend-dot {
width: 8px;
height: 8px;
border-radius: 2px;
flex-shrink: 0;
}
/* ─── SUBNET BUILDER ──────────────────────────────────── */
.subnet-add-row {
display: flex;
align-items: flex-end;
gap: 12px;
flex-wrap: wrap;
padding: 16px;
background: var(--surface2);
border: 1px solid var(--border);
border-radius: 8px;
margin-bottom: 16px;
}
.subnet-add-row .field input { min-width: 180px; }
.subnet-add-row .field select { min-width: 100px; }
.auto-ip-preview {
font-family: 'JetBrains Mono', monospace;
font-size: 11px;
color: var(--muted);
background: var(--surface2);
border: 1px dashed var(--border);
border-radius: 6px;
padding: 8px 12px;
white-space: nowrap;
align-self: flex-end;
line-height: 1;
}
.auto-ip-preview span { color: var(--accent); }
/* ─── SUBNET TABLE ────────────────────────────────────── */
.subnet-table-wrap {
overflow-x: auto;
border-radius: 8px;
border: 1px solid var(--border);
}
table {
width: 100%;
border-collapse: collapse;
font-family: 'JetBrains Mono', monospace;
font-size: 12.5px;
}
thead th {
font-family: 'Space Grotesk', sans-serif;
font-size: 10px;
font-weight: 600;
letter-spacing: 0.1em;
text-transform: uppercase;
color: var(--muted);
padding: 10px 14px;
text-align: left;
background: var(--surface2);
border-bottom: 1px solid var(--border);
white-space: nowrap;
}
tbody tr {
border-bottom: 1px solid var(--border);
transition: background 0.1s;
}
tbody tr:last-child { border-bottom: none; }
tbody tr:hover { background: rgba(255,255,255,0.02); }
tbody td {
padding: 10px 14px;
color: var(--text);
vertical-align: middle;
}
tbody td.muted { color: var(--muted); }
tbody td.mono { font-family: 'JetBrains Mono', monospace; }
.color-swatch {
display: inline-block;
width: 10px;
height: 10px;
border-radius: 2px;
margin-right: 8px;
vertical-align: middle;
}
tr.conflict td { background: rgba(255,68,68,0.07); }
tr.conflict td.mono { color: var(--red); }
.conflict-badge {
font-family: 'Space Grotesk', sans-serif;
font-size: 10px;
font-weight: 600;
letter-spacing: 0.06em;
color: var(--red);
background: rgba(255,68,68,0.12);
border: 1px solid rgba(255,68,68,0.25);
border-radius: 4px;
padding: 2px 6px;
text-transform: uppercase;
margin-left: 8px;
}
.row-actions { display: flex; gap: 6px; align-items: center; }
.action-btn {
background: none;
border: 1px solid var(--border);
color: var(--muted);
font-size: 11px;
font-family: 'Space Grotesk', sans-serif;
padding: 3px 8px;
border-radius: 4px;
cursor: pointer;
transition: all 0.12s;
}
.action-btn:hover { border-color: var(--red); color: var(--red); background: rgba(255,68,68,0.07); }
.empty-state {
text-align: center;
padding: 40px 20px;
color: var(--muted);
font-family: 'Space Grotesk', sans-serif;
font-size: 13px;
}
/* ─── TOASTS ──────────────────────────────────────────── */
.toast-container {
position: fixed;
bottom: 24px;
right: 24px;
z-index: 9999;
display: flex;
flex-direction: column;
gap: 10px;
pointer-events: none;
}
.toast {
font-family: 'Space Grotesk', sans-serif;
font-size: 13px;
font-weight: 500;
padding: 10px 18px;
border-radius: 8px;
border: 1px solid;
backdrop-filter: blur(12px);
pointer-events: none;
animation: toastIn 0.2s ease forwards;
display: flex;
align-items: center;
gap: 8px;
}
.toast.toast-success {
background: rgba(0,230,118,0.12);
border-color: rgba(0,230,118,0.3);
color: var(--green);
}
.toast.toast-info {
background: rgba(0,229,255,0.12);
border-color: rgba(0,229,255,0.3);
color: var(--accent);
}
.toast.toast-out { animation: toastOut 0.2s ease forwards; }
@keyframes toastIn {
from { opacity: 0; transform: translateY(12px) scale(0.95); }
to { opacity: 1; transform: translateY(0) scale(1); }
}
@keyframes toastOut {
from { opacity: 1; transform: translateY(0) scale(1); }
to { opacity: 0; transform: translateY(6px) scale(0.95); }
}
/* ─── SUMMARY ROW ─────────────────────────────────────── */
.summary-row {
display: flex;
gap: 12px;
flex-wrap: wrap;
margin-top: 10px;
}
.summary-pill {
font-family: 'JetBrains Mono', monospace;
font-size: 11px;
background: var(--surface2);
border: 1px solid var(--border);
border-radius: 6px;
padding: 6px 12px;
color: var(--muted);
}
.summary-pill strong { color: var(--text); }
@media (max-width: 640px) {
.header-title { font-size: 13px; }
.btn { font-size: 12px; padding: 6px 10px; }
}
Add Subnet
| # |
Name |
CIDR |
IP Range |
Hosts |
% of VPC |
Status |
|
| No subnets yet — add one above. |