/* rSplat — Gaussian Splat Viewer */ :root { --splat-bg: #0f172a; --splat-surface: #1e293b; --splat-border: #334155; --splat-text: #e2e8f0; --splat-text-muted: #94a3b8; --splat-accent: #818cf8; --splat-accent-hover: #6366f1; --splat-card-bg: rgba(30, 41, 59, 0.8); } /* ── Gallery ── */ .splat-gallery { padding: 2rem; max-width: 1200px; margin: 0 auto; color: var(--splat-text); min-height: calc(100vh - 56px); background: var(--splat-bg); } .splat-gallery h1 { font-size: 1.5rem; font-weight: 600; margin: 0 0 0.5rem; } .splat-gallery__subtitle { color: var(--splat-text-muted); font-size: 0.875rem; margin: 0 0 2rem; } .splat-grid { display: grid; grid-template-columns: repeat(auto-fill, minmax(280px, 1fr)); gap: 1.5rem; } /* ── Card ── */ .splat-card { background: var(--splat-card-bg); border: 1px solid var(--splat-border); border-radius: 12px; overflow: hidden; text-decoration: none; color: inherit; transition: transform 0.2s, border-color 0.2s, box-shadow 0.2s; display: flex; flex-direction: column; backdrop-filter: blur(8px); } .splat-card:hover { transform: translateY(-2px); border-color: var(--splat-accent); box-shadow: 0 8px 24px rgba(129, 140, 248, 0.15); } .splat-card__preview { height: 160px; background: linear-gradient(135deg, #1e1b4b 0%, #312e81 50%, #1e293b 100%); display: flex; align-items: center; justify-content: center; font-size: 3rem; position: relative; } .splat-card__body { padding: 1rem; flex: 1; display: flex; flex-direction: column; gap: 0.5rem; } .splat-card__title { font-weight: 600; font-size: 1rem; line-height: 1.3; } .splat-card__meta { display: flex; align-items: center; gap: 0.75rem; font-size: 0.75rem; color: var(--splat-text-muted); } .splat-badge { display: inline-block; padding: 0.125rem 0.5rem; border-radius: 9999px; font-size: 0.7rem; font-weight: 600; text-transform: uppercase; letter-spacing: 0.05em; } .splat-badge--ply { background: #4338ca; color: #e0e7ff; } .splat-badge--splat { background: #7c3aed; color: #ede9fe; } .splat-badge--spz { background: #2563eb; color: #dbeafe; } /* ── Processing status badges ── */ .splat-badge--pending { background: #92400e; color: #fef3c7; } .splat-badge--processing { background: #1e40af; color: #dbeafe; } .splat-badge--failed { background: #991b1b; color: #fecaca; } /* ── Processing card variants ── */ .splat-card--pending, .splat-card--processing, .splat-card--failed { cursor: default; } .splat-card--pending:hover, .splat-card--processing:hover, .splat-card--failed:hover { transform: none; border-color: var(--splat-border); box-shadow: none; } .splat-card--pending .splat-card__preview { background: linear-gradient(135deg, #451a03 0%, #78350f 50%, #1e293b 100%); } .splat-card--processing .splat-card__preview { background: linear-gradient(135deg, #1e1b4b 0%, #1e3a5f 50%, #1e293b 100%); } .splat-card--failed .splat-card__preview { background: linear-gradient(135deg, #450a0a 0%, #7f1d1d 50%, #1e293b 100%); } /* ── Processing overlay ── */ .splat-card__overlay { position: absolute; inset: 0; display: flex; flex-direction: column; align-items: center; justify-content: center; background: rgba(0, 0, 0, 0.5); backdrop-filter: blur(4px); gap: 0.5rem; z-index: 2; } .splat-card__spinner { width: 28px; height: 28px; border: 2px solid rgba(147, 197, 253, 0.3); border-top-color: #93c5fd; border-radius: 50%; animation: splat-spin 0.8s linear infinite; } /* ── Upload ── */ .splat-upload { margin-top: 2rem; border: 2px dashed var(--splat-border); border-radius: 12px; padding: 2rem; text-align: center; transition: border-color 0.2s, background 0.2s; } .splat-upload:hover, .splat-upload--dragover { border-color: var(--splat-accent); background: rgba(129, 140, 248, 0.05); } .splat-upload__icon { font-size: 2.5rem; margin-bottom: 0.75rem; } .splat-upload__text { color: var(--splat-text-muted); font-size: 0.875rem; margin: 0 0 1rem; } .splat-upload__text strong { color: var(--splat-accent); cursor: pointer; } /* ── Upload mode toggle ── */ .splat-upload__toggle { display: inline-flex; border-radius: 9999px; overflow: hidden; border: 1px solid var(--splat-border); margin-bottom: 1.5rem; } .splat-upload__toggle-btn { padding: 0.5rem 1.25rem; border: none; background: transparent; color: var(--splat-text-muted); font-size: 0.8rem; font-weight: 500; cursor: pointer; transition: background 0.2s, color 0.2s; } .splat-upload__toggle-btn:hover { color: var(--splat-text); } .splat-upload__toggle-btn.active { background: var(--splat-accent); color: white; } /* ── Upload modes ── */ .splat-upload__mode { /* shown/hidden via JS display toggle */ } .splat-upload__form { display: none; flex-direction: column; gap: 0.75rem; max-width: 400px; margin: 1rem auto 0; } .splat-upload__form.active { display: flex; } .splat-upload__form input, .splat-upload__form textarea { padding: 0.5rem 0.75rem; border-radius: 6px; border: 1px solid var(--splat-border); background: var(--splat-surface); color: var(--splat-text); font-size: 0.875rem; } .splat-upload__form textarea { resize: vertical; min-height: 60px; } .splat-upload__btn { padding: 0.625rem 1.25rem; border-radius: 8px; border: none; background: var(--splat-accent); color: white; font-weight: 600; font-size: 0.875rem; cursor: pointer; transition: background 0.2s; } .splat-upload__btn:hover { background: var(--splat-accent-hover); } .splat-upload__btn:disabled { opacity: 0.5; cursor: not-allowed; } .splat-upload__status { font-size: 0.8rem; color: var(--splat-text-muted); } .splat-upload__login { color: var(--splat-text-muted); font-size: 0.875rem; } .splat-upload__login a { color: var(--splat-accent); } /* ── Media file count ── */ .splat-upload__selected { min-height: 0; } .splat-upload__file-count { display: inline-block; padding: 0.375rem 0.75rem; border-radius: 6px; background: rgba(129, 140, 248, 0.15); color: var(--splat-accent); font-size: 0.8rem; font-weight: 500; margin-bottom: 0.5rem; } /* ── Viewer ── */ .splat-viewer { position: relative; width: 100%; height: calc(100vh - 56px); background: #111827; overflow: hidden; } .splat-viewer__canvas { width: 100%; height: 100%; display: block; } .splat-viewer__controls { position: absolute; top: 1rem; left: 1rem; z-index: 10; display: flex; gap: 0.5rem; } .splat-viewer__back { display: inline-flex; align-items: center; gap: 0.375rem; padding: 0.5rem 0.75rem; border-radius: 8px; background: rgba(30, 41, 59, 0.85); backdrop-filter: blur(8px); color: var(--splat-text); text-decoration: none; font-size: 0.8rem; border: 1px solid var(--splat-border); transition: background 0.2s; } .splat-viewer__back:hover { background: rgba(51, 65, 85, 0.9); } .splat-viewer__info { position: absolute; bottom: 1rem; left: 1rem; z-index: 10; padding: 0.75rem 1rem; border-radius: 8px; background: rgba(30, 41, 59, 0.85); backdrop-filter: blur(8px); border: 1px solid var(--splat-border); color: var(--splat-text); font-size: 0.8rem; max-width: 320px; } .splat-viewer__title { font-weight: 600; margin: 0 0 0.25rem; } .splat-viewer__desc { color: var(--splat-text-muted); margin: 0; } /* ── Loading ── */ .splat-loading { position: absolute; inset: 0; display: flex; flex-direction: column; align-items: center; justify-content: center; background: #111827; z-index: 20; transition: opacity 0.4s; } .splat-loading.hidden { opacity: 0; pointer-events: none; } .splat-loading__spinner { width: 48px; height: 48px; border: 3px solid var(--splat-border); border-top-color: var(--splat-accent); border-radius: 50%; animation: splat-spin 0.8s linear infinite; } .splat-loading__text { margin-top: 1rem; color: var(--splat-text-muted); font-size: 0.875rem; } @keyframes splat-spin { to { transform: rotate(360deg); } } /* ── Empty state ── */ .splat-empty { text-align: center; padding: 3rem 1rem; color: var(--splat-text-muted); } .splat-empty__icon { font-size: 3rem; margin-bottom: 1rem; } .splat-empty h3 { font-size: 1.125rem; color: var(--splat-text); margin: 0 0 0.5rem; } .splat-empty p { font-size: 0.875rem; margin: 0; } /* ── Responsive ── */ @media (max-width: 640px) { .splat-gallery { padding: 1rem; } .splat-grid { grid-template-columns: 1fr; } .splat-viewer__info { left: 0.5rem; right: 0.5rem; max-width: none; } .splat-upload__toggle-btn { padding: 0.375rem 0.75rem; font-size: 0.75rem; } }