/* ===== 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; }