/* Veola — Sega-blue palette and component overrides for Tailwind play CDN. */ :root { --bg: #1a2b6d; --surface: #1f3380; --surface-2: #243a93; --accent: #00a4e4; --yellow: #f5c400; --text: #ffffff; --text-2: #a8c0f0; --danger: #e84040; --success: #00e4a4; --border: rgba(255, 255, 255, 0.12); --shadow: 0 4px 16px rgba(0, 0, 80, 0.4); } html, body { background: var(--bg); color: var(--text); font-family: 'Outfit', system-ui, sans-serif; } .font-mono { font-family: 'JetBrains Mono', ui-monospace, monospace; } a { color: var(--accent); } a:hover { text-decoration: underline; } .v-card { position: relative; isolation: isolate; background: linear-gradient(180deg, rgba(36, 58, 147, 0.82), rgba(31, 51, 128, 0.82)); backdrop-filter: blur(10px) saturate(140%); -webkit-backdrop-filter: blur(10px) saturate(140%); border: 1px solid var(--border); border-radius: 10px; box-shadow: var(--shadow); transition: transform 180ms ease, box-shadow 180ms ease, border-color 180ms ease; } .v-card::before { content: ""; position: absolute; inset: 0; border-radius: inherit; padding: 1px; background: linear-gradient(135deg, rgba(0, 164, 228, 0.65) 0%, rgba(245, 196, 0, 0.30) 45%, rgba(255, 255, 255, 0.04) 100%); -webkit-mask: linear-gradient(#000 0 0) content-box, linear-gradient(#000 0 0); -webkit-mask-composite: xor; mask-composite: exclude; pointer-events: none; z-index: -1; } .v-card:hover { transform: translateY(-2px); box-shadow: 0 10px 28px rgba(0, 0, 80, 0.55), 0 0 0 1px rgba(0, 164, 228, 0.30); } .v-card-flat { position: relative; isolation: isolate; background: linear-gradient(180deg, rgba(36, 58, 147, 0.70), rgba(31, 51, 128, 0.70)); backdrop-filter: blur(8px) saturate(130%); -webkit-backdrop-filter: blur(8px) saturate(130%); border: 1px solid var(--border); border-radius: 10px; } .v-divider { border-top: 1px solid var(--border); } .v-btn { background: var(--accent); color: white; border-radius: 6px; padding: 0.5rem 1rem; font-weight: 600; transition: filter 0.1s ease; cursor: pointer; display: inline-flex; align-items: center; gap: 0.4rem; } .v-btn:hover { filter: brightness(1.1); } .v-btn[disabled] { opacity: 0.5; cursor: not-allowed; } .v-btn-ghost { background: transparent; color: var(--text); border: 1px solid var(--border); border-radius: 6px; padding: 0.5rem 1rem; cursor: pointer; } .v-btn-ghost:hover { background: rgba(255,255,255,0.05); } .v-btn-danger { background: var(--danger); color: white; border-radius: 6px; padding: 0.5rem 1rem; cursor: pointer; } .v-input, .v-select, .v-textarea { background: rgba(0, 0, 0, 0.25); color: var(--text); border: 1px solid var(--border); border-radius: 6px; padding: 0.5rem 0.75rem; width: 100%; font-family: inherit; } .v-input:focus, .v-select:focus, .v-textarea:focus { outline: none; border-color: var(--accent); box-shadow: 0 0 0 3px rgba(0, 164, 228, 0.25); } .v-label { display: block; color: var(--text-2); font-size: 0.85rem; margin-bottom: 0.3rem; letter-spacing: 0.01em; } .v-pill { display: inline-block; padding: 0.15rem 0.55rem; border-radius: 999px; font-size: 0.75rem; font-weight: 600; letter-spacing: 0.04em; text-transform: uppercase; } .v-pill-active { background: var(--accent); color: white; } .v-pill-paused { background: rgba(255,255,255,0.1); color: var(--text-2); } .v-pill-error { background: var(--danger); color: white; } .v-price { font-family: 'JetBrains Mono', ui-monospace, monospace; font-size: 1.5rem; } .v-price-target { color: var(--yellow); } .v-price-deal { color: var(--success); } .v-side-nav { background: var(--surface); border-right: 1px solid var(--border); width: 220px; min-height: 100vh; position: sticky; top: 0; } .v-side-nav a { display: flex; align-items: center; gap: 0.6rem; padding: 0.7rem 1rem; color: var(--text-2); text-decoration: none; border-left: 3px solid transparent; } .v-side-nav a.active { background: var(--surface-2); border-left-color: var(--accent); color: white; } .v-side-nav a:hover { color: white; } .v-veola-portrait { background: #f3ead8; border-radius: 12px; padding: 1rem; border: 1px solid var(--border); box-shadow: var(--shadow); } .v-veola-portrait img { display: block; max-width: 100%; height: auto; } .v-badge { display: inline-block; padding: 0.2rem 0.55rem; border-radius: 6px; font-size: 0.75rem; font-weight: 700; } .v-badge-low { background: var(--success); color: #053b2c; } .v-badge-avg { background: var(--accent); color: white; } .v-badge-target { background: var(--yellow); color: #2a2200; } table.v-table { width: 100%; border-collapse: collapse; } table.v-table th { text-align: left; font-size: 0.75rem; letter-spacing: 0.05em; text-transform: uppercase; color: var(--text-2); padding: 0.6rem 0.75rem; border-bottom: 1px solid var(--border); } table.v-table td { padding: 0.7rem 0.75rem; border-bottom: 1px solid var(--border); vertical-align: middle; } table.v-table td { transition: background 140ms ease; } table.v-table tr { transition: transform 140ms ease; } table.v-table tbody tr:hover td { background: rgba(0, 164, 228, 0.08); } .v-error-text { color: var(--danger); font-size: 0.85rem; } .v-muted { color: var(--text-2); } .v-flash { background: rgba(0, 164, 228, 0.15); border: 1px solid var(--accent); border-radius: 6px; padding: 0.6rem 0.9rem; margin-bottom: 1rem; } .v-flash-error { background: rgba(232, 64, 64, 0.15); border: 1px solid var(--danger); border-radius: 6px; padding: 0.6rem 0.9rem; margin-bottom: 1rem; } .v-spinner { display: inline-block; width: 14px; height: 14px; border: 2px solid rgba(255,255,255,0.2); border-top-color: var(--accent); border-radius: 50%; animation: v-spin 0.8s linear infinite; } @keyframes v-spin { to { transform: rotate(360deg); } } /* htmx indicator: hidden by default, visible during in-flight requests. Declared AFTER .v-spinner on purpose: an element carrying both classes (e.g. ) must stay hidden until a request is active, and equal-specificity rules resolve by source order. */ .htmx-indicator { display: none; } .htmx-request .htmx-indicator, .htmx-request.htmx-indicator { display: inline-flex; } /* flair.js adds .v-just-swapped to any htmx swap target for ~400ms, giving refreshed regions a soft fade-in instead of an abrupt content jump. */ @keyframes v-fade-in-up { from { opacity: 0; transform: translateY(6px); } to { opacity: 1; transform: translateY(0); } } .v-just-swapped { animation: v-fade-in-up 240ms ease-out; } /* Ending-soon strip: one global "next auction to close" banner. flair.js keeps the countdown live. Subtle by default; pulses red when inside the last 5 minutes via .v-countdown-critical on the inner counter. */ .v-ending-strip { display: flex; align-items: center; gap: 1rem; padding: 0.7rem 1rem; border-radius: 8px; background: linear-gradient(90deg, rgba(245, 196, 0, 0.12), rgba(0, 164, 228, 0.08)); border: 1px solid rgba(245, 196, 0, 0.35); } .v-ending-label { font-size: 0.7rem; letter-spacing: 0.08em; text-transform: uppercase; font-weight: 700; color: var(--yellow); white-space: nowrap; } .v-ending-title { flex: 1; min-width: 0; } .v-ending-countdown { font-size: 1.15rem; font-weight: 700; color: var(--yellow); } /* Per-row + strip countdown urgency states. Default reads as neutral muted text so 6-day countdowns don't shout; urgency tints kick in only when flair.js flips the class. */ .v-countdown { color: var(--text-2); } .v-countdown-urgent { color: var(--yellow); font-weight: 600; } .v-countdown-critical { color: var(--danger); font-weight: 700; animation: v-pulse 1.2s ease-in-out infinite; } .v-countdown-ended { color: var(--text-2); font-style: italic; } @keyframes v-pulse { 0%, 100% { opacity: 1; } 50% { opacity: 0.55; } } @media (prefers-reduced-motion: reduce) { .v-card { transition: none; } .v-card:hover { transform: none; } table.v-table td, table.v-table tr { transition: none; } .v-just-swapped { animation: none; } .v-countdown-critical { animation: none; } }