asset_browser/public/styles/main.css
2026-03-27 09:18:39 -04:00

932 lines
20 KiB
CSS
Executable file

/* ===== Reset ===== */
*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }
html { height: 100%; }
body {
font-family: var(--font);
background: var(--gray-100);
color: var(--gray-900);
height: 100%;
display: flex;
flex-direction: column;
overflow-x: hidden;
}
/* ===== Header ===== */
#app-header {
background: var(--primary-dark);
color: var(--white);
display: grid;
grid-template-columns: auto 1fr auto;
align-items: center;
gap: 20px;
padding: 10px 24px;
min-height: 68px;
box-shadow: 0 2px 8px rgba(0,0,0,.3);
position: sticky;
top: 0;
z-index: 100;
border-bottom: 3px solid var(--accent);
}
.header-left {
display: flex;
align-items: center;
gap: 12px;
flex-shrink: 0;
}
.header-logo {
height: 44px;
width: 44px;
object-fit: contain;
border-radius: 50%;
background: var(--white);
padding: 2px;
}
.header-title {
display: flex;
flex-direction: column;
line-height: 1.2;
}
.header-logo-link { line-height: 0; border-radius: 50%; }
.header-logo-link:focus-visible { outline: 2px solid rgba(255,255,255,.7); outline-offset: 2px; }
.title-main {
font-size: 1.15rem;
font-weight: 700;
color: var(--white);
letter-spacing: -.01em;
background: none;
border: none;
padding: 0;
cursor: pointer;
text-align: left;
font-family: inherit;
}
.title-main:hover { text-decoration: none; }
.title-sub {
font-size: 0.7rem;
color: rgba(255,255,255,.55);
letter-spacing: .02em;
text-transform: uppercase;
text-decoration: none;
}
.title-sub:hover { color: rgba(255,255,255,.55); }
/* ===== Scan Input (Header Center) ===== */
.header-center { flex: 1; min-width: 0; }
.scan-input-wrap {
position: relative;
display: flex;
align-items: center;
}
.scan-input-icon {
position: absolute;
left: 14px;
top: 50%;
transform: translateY(-50%);
color: rgba(255,255,255,.5);
pointer-events: none;
display: flex;
}
#scan-input {
width: 100%;
background: rgba(255,255,255,.1);
border: 2px solid rgba(255,255,255,.2);
border-radius: var(--radius);
color: var(--white);
font-size: 1rem;
font-family: var(--font);
padding: 10px 42px 10px 44px;
outline: none;
transition: background .15s, border-color .15s;
}
#scan-input::placeholder { color: rgba(255,255,255,.4); }
#scan-input:focus {
background: rgba(255,255,255,.15);
border-color: var(--accent-light);
box-shadow: 0 0 0 3px rgba(196,98,42,.25);
}
.scan-clear-btn {
position: absolute;
right: 10px;
background: none;
border: none;
color: rgba(255,255,255,.5);
cursor: pointer;
padding: 4px 6px;
border-radius: var(--radius-sm);
font-size: 0.85rem;
transition: color .15s;
line-height: 1;
}
.scan-clear-btn:hover { color: var(--white); }
/* ===== Mode Nav ===== */
.header-right { flex-shrink: 0; }
.mode-nav {
display: flex;
gap: 8px;
}
.mode-btn {
display: flex;
align-items: center;
gap: 6px;
background: rgba(255,255,255,.1);
border: 1px solid rgba(255,255,255,.15);
border-radius: var(--radius);
color: rgba(255,255,255,.7);
cursor: pointer;
font-size: 0.82rem;
font-family: var(--font);
font-weight: 600;
padding: 8px 14px;
transition: background .15s, color .15s, border-color .15s;
white-space: nowrap;
}
.mode-btn:hover {
background: rgba(255,255,255,.18);
color: var(--white);
}
.mode-btn.active {
background: var(--accent);
border-color: var(--accent);
color: var(--white);
box-shadow: 0 2px 6px rgba(196,98,42,.4);
}
#btn-menu-toggle[aria-expanded="true"] {
background: rgba(255,255,255,.25);
border-color: rgba(255,255,255,.45);
box-shadow: inset 0 1px 3px rgba(0,0,0,.2);
}
.mode-btn svg { width: 15px; height: 15px; flex-shrink: 0; }
/* ===== Main Content ===== */
#app-main {
flex: 1;
display: flex;
flex-direction: column;
padding: 24px;
min-width: 0; /* prevent flex blowout */
overflow-y: auto; /* main content scrolls independently from sidebar */
}
/* ===== View Sections ===== */
.view { display: none; }
.view.active { display: flex; flex-direction: column; flex: 1; }
/* ===== Idle / Ready State ===== */
#view-idle {
align-items: center;
justify-content: center;
flex: 1;
}
.idle-content {
text-align: center;
animation: fadeIn .4s ease;
}
.scan-indicator {
position: relative;
width: 120px;
height: 120px;
margin: 0 auto 28px;
display: flex;
align-items: center;
justify-content: center;
}
.scan-pulse {
position: absolute;
inset: 0;
border-radius: 50%;
border: 3px solid var(--primary);
animation: pulse 2s ease-in-out infinite;
}
@keyframes pulse {
0%, 100% { transform: scale(1); opacity: .8; }
50% { transform: scale(1.12); opacity: .3; }
}
.scan-icon-large {
width: 64px;
height: 64px;
color: var(--primary);
position: relative;
z-index: 1;
}
.idle-content h2 {
font-size: 1.8rem;
font-weight: 700;
color: var(--gray-800);
margin-bottom: 8px;
}
.idle-content p {
color: var(--gray-500);
font-size: 1rem;
}
/* ===== Loading State ===== */
#view-loading {
align-items: center;
justify-content: center;
}
.loading-content {
text-align: center;
animation: fadeIn .2s ease;
}
.spinner {
width: 48px;
height: 48px;
border: 4px solid var(--gray-200);
border-top-color: var(--primary);
border-radius: 50%;
animation: spin .7s linear infinite;
margin: 0 auto 16px;
}
@keyframes spin { to { transform: rotate(360deg); } }
.loading-content p {
color: var(--gray-500);
font-size: 1rem;
}
/* ===== Error State ===== */
.error-card {
background: var(--red-bg);
border: 1px solid #fca5a5;
border-radius: var(--radius-lg);
padding: 28px;
text-align: center;
max-width: 480px;
margin: 60px auto;
animation: fadeIn .3s ease;
}
.error-card .error-icon {
font-size: 2.5rem;
margin-bottom: 12px;
}
.error-card h3 {
font-size: 1.2rem;
font-weight: 700;
color: var(--red);
margin-bottom: 8px;
}
.error-card p {
color: var(--gray-700);
margin-bottom: 20px;
font-size: 0.92rem;
}
/* ===== Search Results ===== */
.search-results-wrap {
animation: fadeIn .3s ease;
}
.search-results-wrap h3 {
font-size: 1rem;
font-weight: 600;
color: var(--gray-700);
margin-bottom: 12px;
}
.search-result-list {
display: flex;
flex-direction: column;
gap: 8px;
}
.search-result-item {
background: var(--white);
border: 1px solid var(--gray-200);
border-radius: var(--radius);
padding: 14px 18px;
cursor: pointer;
display: flex;
justify-content: space-between;
align-items: center;
transition: border-color .15s, box-shadow .15s;
}
.search-result-item:hover {
border-color: var(--primary);
box-shadow: var(--shadow);
}
.search-result-name {
font-weight: 600;
font-size: 0.95rem;
color: var(--gray-900);
}
.search-result-meta {
font-size: 0.8rem;
color: var(--gray-500);
margin-top: 2px;
}
/* ===== Toast Notifications ===== */
#toast-container {
position: fixed;
bottom: 24px;
right: 24px;
display: flex;
flex-direction: column;
gap: 8px;
z-index: 9999;
}
.toast {
background: var(--gray-900);
color: var(--white);
border-radius: var(--radius);
padding: 12px 20px;
font-size: 0.88rem;
font-weight: 500;
min-width: 220px;
max-width: 360px;
box-shadow: var(--shadow-lg);
animation: slideUp .25s ease;
display: flex;
align-items: center;
gap: 10px;
}
.toast.success { background: var(--green); }
.toast.error { background: var(--red); }
.toast.info { background: var(--primary); }
@keyframes slideUp {
from { transform: translateY(12px); opacity: 0; }
to { transform: translateY(0); opacity: 1; }
}
/* ===== Utility ===== */
@keyframes fadeIn {
from { opacity: 0; transform: translateY(6px); }
to { opacity: 1; transform: translateY(0); }
}
@keyframes menuOpen {
from { clip-path: inset(0 0 100% 0 round 8px); }
to { clip-path: inset(0 0 0% 0 round 8px); }
}
.btn {
display: inline-flex;
align-items: center;
gap: 6px;
border: none;
border-radius: var(--radius-sm);
cursor: pointer;
font-family: var(--font);
font-weight: 600;
padding: 9px 18px;
font-size: 0.88rem;
transition: background .15s, transform .1s, box-shadow .15s;
}
.btn:active { transform: scale(.97); }
.btn-primary {
background: var(--primary);
color: var(--white);
}
.btn-primary:hover { background: var(--primary-light); }
.btn-accent {
background: var(--accent);
color: var(--white);
}
.btn-accent:hover { background: var(--accent-light); }
.btn-ghost {
background: var(--gray-100);
color: var(--gray-700);
border: 1px solid var(--gray-200);
}
.btn-ghost:hover { background: var(--gray-200); }
.btn-danger {
background: var(--red-bg);
color: var(--red);
border: 1px solid #fca5a5;
}
.btn-danger:hover { background: #fecaca; }
/* ===== Label Generation View ===== */
.label-gen-wrap {
max-width: 680px;
width: 100%;
margin: 0 auto;
animation: fadeIn .3s ease;
}
.label-gen-wrap h2 {
font-size: 1.4rem;
font-weight: 700;
margin-bottom: 20px;
color: var(--gray-800);
}
.label-search-row {
display: flex;
gap: 10px;
margin-bottom: 24px;
}
.label-search-row input {
flex: 1;
border: 1.5px solid var(--gray-300);
border-radius: var(--radius);
padding: 10px 14px;
font-size: 0.95rem;
font-family: var(--font);
outline: none;
transition: border-color .15s;
}
.label-search-row input:focus { border-color: var(--primary); }
/* ===== User Menu (header-right) ===== */
.header-right {
display: flex;
align-items: center;
gap: 12px;
flex-shrink: 0;
}
.user-menu {
display: flex;
align-items: center;
gap: 10px;
padding-right: 14px;
border-right: 1px solid rgba(255,255,255,.15);
}
.user-info {
display: flex;
flex-direction: column;
align-items: flex-end;
gap: 2px;
}
.user-name {
font-size: 0.85rem;
font-weight: 600;
color: rgba(255,255,255,.9);
white-space: nowrap;
}
.user-role-badge {
font-size: 0.68rem;
font-weight: 700;
text-transform: uppercase;
letter-spacing: .04em;
padding: 1px 7px;
border-radius: 99px;
background: rgba(255,255,255,.15);
color: rgba(255,255,255,.7);
white-space: nowrap;
}
/* Distinct colour per role */
.user-role-badge[data-role="superduperadmin"] { background: var(--accent); color: #fff; }
.user-role-badge[data-role="admin"] { background: var(--primary-light); color: #fff; }
.user-role-badge[data-role="tech"] { background: rgba(255,255,255,.2); color: rgba(255,255,255,.85); }
.user-role-badge[data-role="client"] { background: rgba(255,255,255,.1); color: rgba(255,255,255,.6); }
.btn-logout {
display: flex;
align-items: center;
gap: 5px;
background: rgba(255,255,255,.1);
border: 1px solid rgba(255,255,255,.15);
border-radius: var(--radius-sm);
color: rgba(255,255,255,.7);
cursor: pointer;
font-size: 0.8rem;
font-family: var(--font);
font-weight: 600;
padding: 6px 11px;
transition: background .15s, color .15s, border-color .15s;
white-space: nowrap;
}
.btn-logout svg { width: 14px; height: 14px; flex-shrink: 0; }
.btn-logout:hover {
background: rgba(185,28,28,.3);
border-color: rgba(252,165,165,.3);
color: #fca5a5;
}
/* ===== Header button divider ===== */
.header-btn-divider {
display: block;
width: 1px;
height: 24px;
background: rgba(255,255,255,.15);
flex-shrink: 0;
}
/* ===== App Menu Dropdown ===== */
.app-menu-wrap {
position: relative;
filter: drop-shadow(0 12px 28px rgba(0,0,0,.18)) drop-shadow(0 3px 8px rgba(0,0,0,.10));
}
.app-menu {
position: absolute;
top: calc(100% + 10px);
right: 0;
background: var(--white);
border-radius: var(--radius-lg);
min-width: 210px;
z-index: 500;
overflow: hidden;
animation: menuOpen .2s ease;
}
.app-menu[hidden] { display: none; }
.app-menu-user {
padding: 14px 16px;
background: var(--primary);
border-bottom: 2px solid var(--accent);
display: flex;
flex-direction: column;
gap: 5px;
}
.app-menu-user .user-name {
font-size: 0.9rem;
font-weight: 600;
color: var(--white);
}
.app-menu-divider {
height: 1px;
background: var(--gray-100);
margin: 4px 0;
}
.app-menu-item {
display: flex;
align-items: center;
gap: 10px;
width: 100%;
padding: 10px 16px;
background: none;
border: none;
font-family: var(--font);
font-size: 0.88rem;
font-weight: 500;
color: var(--gray-700);
cursor: pointer;
text-align: left;
text-decoration: none;
transition: background .1s, color .1s;
}
.app-menu-item:hover { background: var(--gray-50); color: var(--gray-900); }
.app-menu-item svg {
width: 15px;
height: 15px;
color: var(--gray-400);
flex-shrink: 0;
transition: color .1s;
}
.app-menu-item:hover svg { color: var(--gray-600); }
.app-menu-item-danger { color: var(--red); }
.app-menu-item-danger svg { color: var(--red); opacity: .7; }
.app-menu-item-danger:hover { background: var(--red-bg); color: var(--red); }
.app-menu-item-danger:hover svg { color: var(--red); opacity: 1; }
/* Active state inside the menu (e.g. timer ON) */
.app-menu-item.active { color: var(--accent); font-weight: 600; }
.app-menu-item.active svg { color: var(--accent); opacity: 1; }
.app-menu-item.active:hover { background: var(--accent-50); color: var(--accent-dark); }
/* ===== Search Autocomplete Dropdown ===== */
.search-autocomplete {
position: absolute;
top: calc(100% + 6px);
left: 0;
right: 0;
z-index: 1000;
background: var(--white);
border: 1px solid var(--gray-200);
border-radius: var(--radius);
box-shadow: 0 8px 28px rgba(0,0,0,.22), 0 2px 8px rgba(0,0,0,.12);
overflow: hidden;
max-height: 360px;
overflow-y: auto;
animation: fadeIn .12s ease;
}
.search-autocomplete[hidden] { display: none; }
.ac-item {
display: flex;
align-items: center;
gap: 10px;
padding: 9px 14px;
cursor: pointer;
transition: background .1s;
border-bottom: 1px solid var(--gray-100);
}
.ac-item:last-child { border-bottom: none; }
.ac-item:hover,
.ac-item.ac-active {
background: var(--primary-50);
}
.ac-icon {
width: 15px;
height: 15px;
flex-shrink: 0;
color: var(--gray-400);
}
.ac-item-body { min-width: 0; }
.ac-item-name {
font-size: 0.88rem;
font-weight: 600;
color: var(--gray-900);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.ac-item-meta {
font-size: 0.75rem;
color: var(--gray-500);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
margin-top: 1px;
}
.ac-item-syncro {
font-size: 0.85rem;
color: var(--gray-600);
background: var(--gray-50);
border-top: 1px solid var(--gray-200);
}
.ac-item-syncro:hover,
.ac-item-syncro.ac-active {
background: var(--accent-50);
color: var(--accent-dark);
}
.ac-item-syncro .ac-icon { color: var(--gray-400); }
.ac-item-syncro em {
font-style: normal;
font-weight: 600;
color: var(--accent);
}
@media (max-width: 900px) {
.user-info { display: none; }
}
/* ── Camera scan overlay ──────────────────────────────────────────────────── */
.camera-overlay {
position: fixed;
inset: 0;
background: rgba(0, 0, 0, 0.85);
z-index: 1000;
display: flex;
align-items: center;
justify-content: center;
}
.camera-overlay[hidden] {
display: none;
}
.camera-overlay-inner {
background: #1a1a1a;
border-radius: var(--radius-lg);
overflow: hidden;
width: min(92vw, 480px);
display: flex;
flex-direction: column;
}
.camera-header {
display: flex;
justify-content: space-between;
align-items: center;
padding: 12px 16px;
border-bottom: 1px solid #333;
}
.camera-title {
color: #fff;
font-weight: 600;
font-size: 1rem;
}
.camera-close-btn {
background: none;
border: none;
color: #aaa;
font-size: 1.5rem;
cursor: pointer;
line-height: 1;
padding: 0 4px;
}
.camera-close-btn:hover {
color: #fff;
}
.camera-viewport {
position: relative;
background: #000;
aspect-ratio: 4 / 3;
}
#camera-video {
width: 100%;
height: 100%;
object-fit: cover;
display: block;
}
/* Targeting reticle — pure CSS, no images */
.camera-reticle {
position: absolute;
inset: 20%;
border: 1px solid rgba(255, 255, 255, 0.2);
border-radius: var(--radius);
pointer-events: none;
}
/* Corner accents — top-left and bottom-right */
.camera-reticle::before,
.camera-reticle::after {
content: '';
position: absolute;
width: 22px;
height: 22px;
border-style: solid;
border-color: rgba(255, 255, 255, 0.75);
transition: border-color 0.15s;
}
.camera-reticle::before {
top: -1px; left: -1px;
border-width: 3px 0 0 3px;
border-radius: 4px 0 0 0;
}
.camera-reticle::after {
bottom: -1px; right: -1px;
border-width: 0 3px 3px 0;
border-radius: 0 0 4px 0;
}
/* Scan line — hidden by default, shown on detection */
.camera-scan-line {
display: none;
}
/* Flash green when a barcode is detected */
.camera-overlay.detected .camera-reticle {
border-color: rgba(74, 222, 128, 0.5);
}
.camera-overlay.detected .camera-reticle::before,
.camera-overlay.detected .camera-reticle::after {
border-color: #4ade80;
}
.camera-overlay.detected .camera-scan-line {
display: block;
position: absolute;
left: 4px; right: 4px;
height: 2px;
top: 50%;
background: linear-gradient(90deg, transparent 0%, #4ade80 50%, transparent 100%);
border-radius: 1px;
}
.camera-status {
padding: 10px 16px;
color: #ccc;
font-size: 0.875rem;
text-align: center;
min-height: 2.5em;
margin: 0;
}
.camera-status.error {
color: #ff6b6b;
}
/* ── Client role: hide write controls and scan-only UI ──────────────────── */
.role-client #btn-scan-mode,
.role-client #btn-scan-mode-toggle,
.role-client #btn-timer-toggle,
.role-client #btn-label-center,
.role-client #sidebar-refresh,
.role-client #action-toggle-possession,
.role-client #action-lifecycle,
.role-client #action-change-owner,
.role-client #action-remove-user,
.role-client #action-sign-out,
.role-client #action-infrastructure,
.role-client #action-print-label,
.role-client #action-add-to-queue { display: none; }
/* Show Quick View button only for client role */
#btn-quick-view { display: none; }
.role-client #btn-quick-view { display: inline-flex; }
/* Search bar: show magnifying glass icon for clients, barcode for staff */
.icon-search { display: none; }
.role-client .icon-barcode { display: none; }
.role-client .icon-search { display: inline; }
/* Quick View dashboard layout */
#view-quick-view { padding: 1.5rem; overflow-y: auto; }
/* Card — matches .admin-section from admin panel */
.qv-section {
background: var(--white);
border: 1px solid var(--gray-200);
border-radius: var(--radius-lg);
box-shadow: var(--shadow-sm);
margin-bottom: 1.5rem;
overflow: hidden;
}
.qv-card-header {
display: flex;
align-items: center;
justify-content: space-between;
padding: 18px 24px;
border-bottom: 1px solid var(--gray-200);
}
.qv-card-header h2 { font-size: 1rem; font-weight: 700; color: var(--gray-800); margin: 0; }
.qv-card-count { font-size: 0.8rem; color: var(--gray-500); }
/* Table */
.qv-table { width: 100%; border-collapse: collapse; font-size: 0.875rem; }
.qv-table th {
text-align: center; padding: 0.5rem 0.75rem;
background: var(--gray-50); color: var(--gray-600);
font-weight: 600; border-bottom: 1px solid var(--gray-200);
}
.qv-table th:first-child { text-align: left; padding-left: 24px; }
.qv-table td {
text-align: center; padding: 0.5rem 0.75rem;
border-bottom: 1px solid var(--gray-200);
}
.qv-table td:first-child { text-align: left; font-weight: 500; color: var(--gray-800); padding-left: 24px; }
.qv-table tbody tr:last-child td { border-bottom: none; }
.qv-table tfoot td {
font-weight: 600; border-top: 2px solid var(--gray-200);
border-bottom: none; background: var(--gray-50);
padding-top: 0.6rem; padding-bottom: 0.6rem;
}
.qv-cell-link {
display: inline-block; min-width: 2rem; padding: 0.2rem 0.4rem;
border-radius: 4px; cursor: pointer; transition: background 0.15s;
color: var(--accent, #3b82f6);
}
.qv-cell-link:hover { background: var(--gray-100); text-decoration: underline; }
.qv-cell-zero { color: var(--gray-400); }
.qv-empty { padding: 1.5rem 24px; color: var(--gray-500); font-size: 0.875rem; }
.qv-loading { padding: 1.5rem 24px; color: var(--gray-400); font-size: 0.875rem; }