Files
mcr/web/static/style.css
Kyle Isom 593da3975d Phases 11, 12: mcrctl CLI tool and mcr-web UI
Phase 11 implements the admin CLI with dual REST/gRPC transport,
global flags (--server, --grpc, --token, --ca-cert, --json), and
all commands: status, repo list/delete, policy CRUD, audit tail,
gc trigger/status/reconcile, and snapshot.

Phase 12 implements the HTMX web UI with chi router, session-based
auth (HttpOnly/Secure/SameSite=Strict cookies), CSRF protection
(HMAC-SHA256 signed double-submit), and pages for dashboard,
repositories, manifest detail, policy management, and audit log.

Security: CSRF via signed double-submit cookie, session cookies
with HttpOnly/Secure/SameSite=Strict, TLS 1.3 minimum on all
connections, form body size limits via http.MaxBytesReader.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-20 10:14:38 -07:00

405 lines
6.2 KiB
CSS

/* MCR Web UI - minimal clean styling */
*,
*::before,
*::after {
box-sizing: border-box;
}
body {
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto,
Oxygen, Ubuntu, Cantarell, "Fira Sans", "Droid Sans",
"Helvetica Neue", sans-serif;
margin: 0;
padding: 0;
color: #1a1a1a;
background: #f5f5f5;
line-height: 1.5;
}
a {
color: #0066cc;
text-decoration: none;
}
a:hover {
text-decoration: underline;
}
/* Navigation */
nav {
background: #1a1a2e;
color: #fff;
padding: 0 1rem;
display: flex;
align-items: center;
gap: 1.5rem;
height: 3.5rem;
}
nav .brand {
font-weight: 700;
font-size: 1.125rem;
color: #fff;
margin-right: 1rem;
}
nav a {
color: #ccc;
font-size: 0.875rem;
}
nav a:hover,
nav a.active {
color: #fff;
text-decoration: none;
}
nav .spacer {
flex: 1;
}
nav .logout {
color: #e57373;
}
/* Main container */
.container {
max-width: 1200px;
margin: 0 auto;
padding: 1.5rem;
}
/* Page header */
h1 {
font-size: 1.5rem;
margin: 0 0 1rem 0;
font-weight: 600;
}
h2 {
font-size: 1.25rem;
margin: 1.5rem 0 0.75rem 0;
font-weight: 600;
}
/* Stats cards */
.stats {
display: flex;
gap: 1rem;
margin-bottom: 1.5rem;
flex-wrap: wrap;
}
.stat-card {
background: #fff;
border: 1px solid #e0e0e0;
border-radius: 6px;
padding: 1rem 1.5rem;
min-width: 180px;
flex: 1;
}
.stat-card .label {
font-size: 0.75rem;
text-transform: uppercase;
color: #666;
letter-spacing: 0.05em;
}
.stat-card .value {
font-size: 1.75rem;
font-weight: 700;
color: #1a1a2e;
}
/* Tables */
table {
width: 100%;
border-collapse: collapse;
background: #fff;
border: 1px solid #e0e0e0;
border-radius: 6px;
overflow: hidden;
margin-bottom: 1.5rem;
}
th,
td {
padding: 0.625rem 0.875rem;
text-align: left;
border-bottom: 1px solid #e0e0e0;
}
th {
background: #f9f9f9;
font-weight: 600;
font-size: 0.8125rem;
text-transform: uppercase;
letter-spacing: 0.03em;
color: #555;
}
tr:last-child td {
border-bottom: none;
}
tr:nth-child(even) td {
background: #fafafa;
}
/* Forms */
form {
margin-bottom: 1rem;
}
label {
display: block;
font-size: 0.875rem;
font-weight: 500;
margin-bottom: 0.25rem;
color: #333;
}
input[type="text"],
input[type="password"],
input[type="number"],
input[type="date"],
select,
textarea {
width: 100%;
padding: 0.5rem 0.75rem;
border: 1px solid #ccc;
border-radius: 4px;
font-size: 0.875rem;
font-family: inherit;
}
input:focus,
select:focus,
textarea:focus {
outline: none;
border-color: #0066cc;
box-shadow: 0 0 0 2px rgba(0, 102, 204, 0.2);
}
/* Buttons */
button,
input[type="submit"] {
padding: 0.5rem 1rem;
border: none;
border-radius: 4px;
font-size: 0.875rem;
font-family: inherit;
cursor: pointer;
background: #0066cc;
color: #fff;
font-weight: 500;
}
button:hover,
input[type="submit"]:hover {
background: #0052a3;
}
button.secondary {
background: #666;
}
button.secondary:hover {
background: #555;
}
button.danger {
background: #d32f2f;
}
button.danger:hover {
background: #b71c1c;
}
button.small {
padding: 0.25rem 0.5rem;
font-size: 0.75rem;
}
/* Login page */
.login-container {
max-width: 400px;
margin: 5rem auto;
padding: 2rem;
background: #fff;
border: 1px solid #e0e0e0;
border-radius: 8px;
}
.login-container h1 {
text-align: center;
margin-bottom: 1.5rem;
}
.login-container .form-group {
margin-bottom: 1rem;
}
.login-container button {
width: 100%;
padding: 0.75rem;
font-size: 1rem;
margin-top: 0.5rem;
}
/* Error and info messages */
.error {
background: #ffebee;
color: #c62828;
padding: 0.75rem 1rem;
border-radius: 4px;
margin-bottom: 1rem;
border: 1px solid #ef9a9a;
font-size: 0.875rem;
}
.info {
background: #e3f2fd;
color: #1565c0;
padding: 0.75rem 1rem;
border-radius: 4px;
margin-bottom: 1rem;
border: 1px solid #90caf9;
font-size: 0.875rem;
}
/* Manifest JSON */
pre {
background: #263238;
color: #eeffff;
padding: 1rem;
border-radius: 6px;
overflow-x: auto;
font-size: 0.8125rem;
line-height: 1.6;
margin-bottom: 1.5rem;
}
code {
font-family: "SF Mono", "Fira Code", "Fira Mono", "Roboto Mono",
"Consolas", monospace;
}
/* Inline form row for policy creation */
.form-row {
display: flex;
gap: 0.75rem;
align-items: flex-end;
flex-wrap: wrap;
margin-bottom: 1rem;
}
.form-row .form-group {
flex: 1;
min-width: 120px;
}
.form-row button {
margin-bottom: 0;
white-space: nowrap;
}
/* Filter form */
.filters {
display: flex;
gap: 0.75rem;
align-items: flex-end;
flex-wrap: wrap;
margin-bottom: 1.5rem;
background: #fff;
padding: 1rem;
border: 1px solid #e0e0e0;
border-radius: 6px;
}
.filters .form-group {
flex: 1;
min-width: 140px;
}
.filters button {
white-space: nowrap;
}
/* Pagination */
.pagination {
display: flex;
gap: 0.5rem;
justify-content: center;
margin-top: 1rem;
}
.pagination a,
.pagination span {
padding: 0.375rem 0.75rem;
border: 1px solid #e0e0e0;
border-radius: 4px;
font-size: 0.875rem;
}
.pagination span {
background: #0066cc;
color: #fff;
border-color: #0066cc;
}
/* Badge */
.badge {
display: inline-block;
padding: 0.125rem 0.5rem;
border-radius: 3px;
font-size: 0.75rem;
font-weight: 600;
}
.badge-allow {
background: #e8f5e9;
color: #2e7d32;
}
.badge-deny {
background: #ffebee;
color: #c62828;
}
.badge-enabled {
background: #e8f5e9;
color: #2e7d32;
}
.badge-disabled {
background: #fafafa;
color: #999;
}
/* Truncated text */
.truncated {
font-family: "SF Mono", "Fira Code", monospace;
font-size: 0.8125rem;
}
/* Responsive */
@media (max-width: 768px) {
.stats {
flex-direction: column;
}
.form-row,
.filters {
flex-direction: column;
}
nav {
flex-wrap: wrap;
height: auto;
padding: 0.5rem 1rem;
}
}