/* ============================================
   EnrichmentHelper — Design System
   DaisyUI 5 (nord / dim) + custom PIM-style tokens
   Load AFTER daisyui.css and tailwind browser
   ============================================ */

/* -----------------------------------------
   1. FONT + BASE
   ----------------------------------------- */
body {
  font-family: 'Inter', ui-sans-serif, system-ui, -apple-system, sans-serif;
  font-feature-settings: 'cv02', 'cv03', 'cv04', 'cv11';
  -webkit-font-smoothing: antialiased;
  -moz-osx-font-smoothing: grayscale;
  letter-spacing: -0.005em;
}

/* Tighten the Nord and Dim palettes so the app reads as an enterprise
   PIM (flat, neutral, high-contrast) rather than the pastel default. */
[data-theme="nord"] {
  --color-base-100: #ffffff;
  --color-base-200: #f6f7f9;
  --color-base-300: #e7eaef;
  --color-base-content: #1e2530;
  --color-primary: #3b4d6b;
  --color-primary-content: #ffffff;
  --color-secondary: #5e81ac;
  --color-accent: #2f6fd3;
  --color-neutral: #2a3240;
}

[data-theme="dim"] {
  --color-base-100: #1a2030;
  --color-base-200: #151a27;
  --color-base-300: #232a3b;
  --color-base-content: #e6e8ee;
  --color-primary: #7b8fff;
  --color-primary-content: #0f1422;
  --color-accent: #4a8cff;
}

/* -----------------------------------------
   2. APP SHELL LAYOUT (sidebar + topbar)
   Layout spec matches Claude Design prototype:
   240px sidebar, brand → scrollable nav, no
   footer. User moved to top-right of topbar,
   which carries a "Synced Xm ago" live chip.
   ----------------------------------------- */
.app-shell {
  display: grid;
  grid-template-columns: 240px 1fr;
  min-height: 100vh;
  background: var(--color-base-200);
}

@media (max-width: 900px) {
  .app-shell { grid-template-columns: 1fr; }
  .app-sidebar { display: none; }
  .app-sidebar.is-open { display: flex; position: fixed; inset: 0 auto 0 0; width: 260px; z-index: 60; box-shadow: 0 0 0 100vw rgba(15, 20, 34, 0.42); }
}

.app-sidebar {
  background: var(--color-base-100);
  border-right: 1px solid var(--color-base-300);
  display: flex;
  flex-direction: column;
  position: sticky;
  top: 0;
  height: 100vh;
  z-index: 10;
  /* No padding — children carry their own: brand (header height),
     scroll (its own inset). Keeps the brand bar flush with the topbar. */
  padding: 0;
}

/* Brand — gradient square mark + "EnrichmentHelper" wordmark.
   Height matches .app-topbar so the top edge reads as a single band. */
.app-sidebar__brand {
  display: flex;
  align-items: center;
  gap: 0.625rem;
  padding: 0 1.125rem;
  height: 3.25rem;
  border-bottom: 1px solid var(--color-base-300);
  color: var(--color-base-content);
  text-decoration: none;
  flex-shrink: 0;
}

/* Original brand logo — rendered as-is (no gradient tint box). */
.app-sidebar__logo {
  width: 28px;
  height: 28px;
  flex-shrink: 0;
  display: block;
}

.app-sidebar__wordmark {
  font-weight: 600;
  font-size: 0.86rem;
  letter-spacing: -0.015em;
}
.app-sidebar__wordmark em {
  font-style: normal;
  color: color-mix(in oklch, var(--color-base-content) 60%, transparent);
  font-weight: 500;
}

.app-sidebar__scroll {
  flex: 1;
  overflow-y: auto;
  padding: 0.75rem 0.625rem 1rem;
}

.app-sidebar__section + .app-sidebar__section { margin-top: 0.875rem; }

.section-label {
  font-size: 0.68rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  color: color-mix(in oklch, var(--color-base-content) 55%, transparent);
  font-weight: 600;
  padding: 0.375rem 0.625rem 0.25rem;
}

.app-nav-link {
  display: flex;
  align-items: center;
  gap: 0.625rem;
  padding: 0.45rem 0.625rem;
  border-radius: 0.45rem;
  font-size: 0.82rem;
  line-height: 1.3;
  color: color-mix(in oklch, var(--color-base-content) 78%, transparent);
  transition: background-color 0.12s ease, color 0.12s ease;
  text-decoration: none;
  font-weight: 500;
  border: 0;
  background: transparent;
  width: 100%;
  text-align: left;
  cursor: pointer;
}

.app-nav-link:hover {
  background: color-mix(in oklch, var(--color-base-content) 5%, transparent);
  color: var(--color-base-content);
}

.app-nav-link__icon {
  width: 16px;
  height: 16px;
  flex-shrink: 0;
  color: color-mix(in oklch, var(--color-base-content) 60%, transparent);
}

.app-nav-link.is-active {
  background: color-mix(in oklch, var(--color-primary) 14%, transparent);
  color: var(--color-primary);
  font-weight: 600;
}
.app-nav-link.is-active .app-nav-link__icon { color: var(--color-primary); }

.app-nav-link__count {
  margin-left: auto;
  font-size: 0.7rem;
  color: color-mix(in oklch, var(--color-base-content) 55%, transparent);
  font-variant-numeric: tabular-nums;
}
.app-nav-link.is-active .app-nav-link__count { color: var(--color-primary); }

.app-main {
  display: flex;
  flex-direction: column;
  min-width: 0;
}

/* Topbar — sticky band with blur. Layout:
   [mobile toggle] [breadcrumbs] [synced chip] [enrichment status] [user] */
.app-topbar {
  position: sticky;
  top: 0;
  z-index: 20;
  height: 3.25rem;
  background: color-mix(in oklch, var(--color-base-100) 82%, transparent);
  backdrop-filter: saturate(150%) blur(10px);
  -webkit-backdrop-filter: saturate(150%) blur(10px);
  border-bottom: 1px solid var(--color-base-300);
  padding: 0 1.25rem;
  display: flex;
  align-items: center;
  gap: 1rem;
}

.app-topbar__breadcrumbs {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  min-width: 0;
  font-size: 0.82rem;
  color: color-mix(in oklch, var(--color-base-content) 60%, transparent);
}
.app-topbar__breadcrumbs .crumb-current {
  color: var(--color-base-content);
  font-weight: 500;
}
.app-topbar__breadcrumbs svg { opacity: 0.55; flex-shrink: 0; }

/* Synced chip — live-dot pulse + "Synced Xm ago" text. Sits on the
   right side of the topbar so it stays visible while the breadcrumbs
   take any overflow on narrow viewports. */
/* Slot wrapper — pushes the chip to the right even when HTMX hasn't
   yet populated it, so the topbar layout doesn't jump when the
   fragment arrives. */
.app-topbar__sync-slot {
  margin-left: auto;
  display: flex;
  align-items: center;
  min-height: 24px;
  flex-shrink: 0;
}

.app-topbar__sync {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  font-size: 0.72rem;
  color: color-mix(in oklch, var(--color-base-content) 60%, transparent);
  padding: 0.25rem 0.625rem 0.25rem 0.5rem;
  border-radius: 999px;
  border: 1px solid var(--color-base-300);
  background: var(--color-base-200);
  flex-shrink: 0;
}

/* Never-synced variant: amber icon+text signals "no Akeneo pull has
   completed in this workspace yet — trigger one from /products". */
.app-topbar__sync--warn {
  color: var(--color-warning, #d97706);
  border-color: color-mix(in oklch, var(--color-warning, #d97706) 35%, transparent);
  background: color-mix(in oklch, var(--color-warning, #d97706) 10%, var(--color-base-200));
}

/* Green pulsing dot — shared with the Activity Feed "Live" meta label. */
@keyframes live-pulse {
  0%   { box-shadow: 0 0 0 0 rgba(98, 192, 138, 0.55); }
  70%  { box-shadow: 0 0 0 6px rgba(98, 192, 138, 0); }
  100% { box-shadow: 0 0 0 0 rgba(98, 192, 138, 0); }
}
.live-dot {
  width: 7px;
  height: 7px;
  border-radius: 50%;
  background: var(--color-success, #62c08a);
  display: inline-block;
  animation: live-pulse 2s infinite;
  flex-shrink: 0;
}

/* User pill in topbar: email + avatar; clicking opens the dropdown. */
.app-topbar__user {
  display: inline-flex;
  align-items: center;
  gap: 0.5rem;
  padding: 0.2rem 0.25rem 0.2rem 0.6rem;
  border-radius: 999px;
  border: 1px solid transparent;
  background: transparent;
  cursor: pointer;
  color: var(--color-base-content);
  line-height: 1;
  flex-shrink: 0;
  transition: background 0.12s ease, border-color 0.12s ease;
}
.app-topbar__user:hover {
  background: color-mix(in oklch, var(--color-base-content) 5%, transparent);
  border-color: var(--color-base-300);
}
.app-topbar__user-email {
  font-size: 0.72rem;
  color: color-mix(in oklch, var(--color-base-content) 60%, transparent);
  max-width: 180px;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}
@media (max-width: 640px) {
  .app-topbar__user-email { display: none; }
}
.app-topbar__avatar {
  width: 28px;
  height: 28px;
  border-radius: 50%;
  background: linear-gradient(135deg, #5e81ac, #7b8fff);
  color: white;
  display: grid;
  place-items: center;
  font-size: 0.68rem;
  font-weight: 600;
  flex-shrink: 0;
  letter-spacing: 0;
}

.app-content {
  padding: 1.5rem 1.75rem 3rem;
  max-width: 1400px;
  width: 100%;
  margin: 0 auto;
}

.app-mobile-toggle {
  display: none;
}

@media (max-width: 900px) {
  .app-mobile-toggle { display: inline-flex; }
  .app-content { padding: 1rem 1rem 2.5rem; }
  /* On narrow viewports collapse the sync chip label — keep the dot so
     the page still signals that it's live. */
  .app-topbar__sync span:not(.live-dot) { display: none; }
}

/* -----------------------------------------
   3. TABLE POLISH
   ----------------------------------------- */
.table-zebra tbody tr {
  transition: background-color 0.15s ease;
}

.table-zebra tbody tr:hover {
  background-color: color-mix(in oklch, var(--color-primary) 8%, transparent);
}

.table thead th {
  font-size: 0.7rem;
  text-transform: uppercase;
  letter-spacing: 0.06em;
  color: color-mix(in oklch, var(--color-base-content) 55%, transparent);
  font-weight: 600;
  background: color-mix(in oklch, var(--color-base-200) 50%, var(--color-base-100));
  border-bottom: 1px solid var(--color-base-300);
}

/* -----------------------------------------
   4. GLOBAL TRANSITIONS + COMPONENT RESETS
   ----------------------------------------- */
.btn, .card, .badge, .input, .select, .textarea, .toggle, .checkbox {
  transition: all 0.18s ease;
}

/* Modern PIMs use flat cards — no lift. Keep a subtle shadow only on
   explicit .card-elevated. The old global lift caused jitter on dense
   review screens. */
.card {
  box-shadow: none;
  border: 1px solid var(--color-base-300);
  background: var(--color-base-100);
  border-radius: 0.625rem;
}

.card:hover { transform: none; box-shadow: none; }

.card-elevated {
  box-shadow: 0 1px 2px rgba(15, 20, 34, 0.04), 0 2px 6px rgba(15, 20, 34, 0.04);
}

/* Buttons: tighten radius, flatten default */
.btn {
  border-radius: 0.5rem;
  font-weight: 500;
  letter-spacing: 0;
}

.btn-primary {
  box-shadow: 0 1px 0 color-mix(in oklch, var(--color-primary) 60%, transparent);
}

/* Inputs / selects: slimmer, more PIM-like */
.input, .select, .textarea {
  border-radius: 0.5rem;
  background: var(--color-base-100);
}

.input-bordered, .select-bordered, .textarea-bordered {
  border-color: var(--color-base-300);
}

.input-bordered:focus, .select-bordered:focus, .textarea-bordered:focus {
  outline: 2px solid color-mix(in oklch, var(--color-primary) 35%, transparent);
  outline-offset: 1px;
  border-color: color-mix(in oklch, var(--color-primary) 40%, var(--color-base-300));
}

/* Smooth link underlines */
a:not(.btn):not(.nav-active):not([class*="menu"]):not(.app-nav-link):not(.app-sidebar__brand) {
  transition: color 0.15s ease;
}

/* -----------------------------------------
   5. HTMX SWAP TRANSITIONS
   ----------------------------------------- */
.htmx-added { animation: htmxFadeIn 0.25s ease-out both; }
.htmx-swapping { opacity: 0; transition: opacity 0.15s ease-out; }
.htmx-settling { animation: htmxFadeIn 0.2s ease-out both; }

@keyframes htmxFadeIn {
  from { opacity: 0; transform: translateY(4px); }
  to   { opacity: 1; transform: translateY(0); }
}

/* -----------------------------------------
   6. MICRO-INTERACTION KEYFRAMES
   ----------------------------------------- */
@keyframes slideInRight {
  from { transform: translateX(100%); opacity: 0; }
  to   { transform: translateX(0); opacity: 1; }
}
@keyframes slideUp {
  from { transform: translateY(100%); opacity: 0; }
  to   { transform: translateY(0); opacity: 1; }
}
@keyframes fadeIn {
  from { opacity: 0; }
  to   { opacity: 1; }
}
@keyframes skeletonPulse {
  0%, 100% { opacity: 0.4; }
  50%      { opacity: 1; }
}

.anim-slide-in-right   { animation: slideInRight 0.3s ease-out both; }
.anim-slide-up         { animation: slideUp 0.3s ease-out both; }
.anim-fade-in          { animation: fadeIn 0.25s ease-in both; }
.anim-skeleton-pulse   { animation: skeletonPulse 1.5s ease-in-out infinite; }

/* -----------------------------------------
   7. VERTICAL TIMELINE (Trace Viewer)
   ----------------------------------------- */
.trace-timeline {
  position: relative;
  padding-left: 2rem;
  margin-left: 0.75rem;
}

.trace-timeline::before {
  content: "";
  position: absolute;
  top: 0;
  bottom: 0;
  left: 0.75rem;
  width: 2px;
  background-color: var(--color-base-300);
}

.trace-step { position: relative; padding-bottom: 1.25rem; }

.trace-step::before {
  content: "";
  position: absolute;
  left: -1.6rem;
  top: 0.35rem;
  width: 0.75rem;
  height: 0.75rem;
  border-radius: 50%;
  background-color: var(--color-primary);
  border: 2px solid var(--color-base-100);
  z-index: 1;
}

.trace-step:last-child { padding-bottom: 0; }

.trace-step.active::before {
  box-shadow: 0 0 0 3px color-mix(in oklch, var(--color-primary) 30%, transparent);
}

.trace-step--reasoning::before { background-color: var(--color-base-content); opacity: 0.4; }
.trace-step--tool::before      { background-color: var(--color-primary); }
.trace-step--decision::before  { background-color: var(--color-success); }

/* -----------------------------------------
   8. ACCENT BORDER HELPERS
   ----------------------------------------- */
.border-l-primary { border-left: 4px solid var(--color-primary); }

/* -----------------------------------------
   9. DESIGN SYSTEM — page chrome
   ----------------------------------------- */

/* Page-header legacy classes (hyphen BEM) — still used by the home
   ``index.html`` and a couple of settings pages. Kept here verbatim;
   the new dashboard spec in section 15 uses the ``.page-header__*``
   double-underscore variant instead and both coexist. */
.page-header-title {
  font-size: 1.625rem;
  line-height: 1.15;
  font-weight: 700;
  letter-spacing: -0.022em;
  margin-bottom: 0.25rem;
  color: var(--color-base-content);
}

.page-header-subtitle {
  color: color-mix(in oklch, var(--color-base-content) 60%, transparent);
  font-size: 0.875rem;
  line-height: 1.4;
  max-width: 64ch;
}

/* Avatar circle (user initial) */
.avatar-initial {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 1.875rem;
  height: 1.875rem;
  border-radius: 9999px;
  background: color-mix(in oklch, var(--color-primary) 18%, transparent);
  color: var(--color-primary);
  font-weight: 600;
  font-size: 0.8rem;
  letter-spacing: 0.01em;
}

/* Legacy .stat-tile-{label,value,hint} (hyphen BEM) — retained because
   the Products page and a couple of settings screens still reference
   these. The dashboard uses the double-underscore BEM variants from
   section 15 instead. The shared ``.stat-tile`` base is defined once
   down there; this block exposes only the legacy element classes. */
.stat-tile-label {
  font-size: 0.7rem;
  text-transform: uppercase;
  letter-spacing: 0.06em;
  color: color-mix(in oklch, var(--color-base-content) 55%, transparent);
  font-weight: 600;
}

.stat-tile-value {
  font-size: 1.625rem;
  line-height: 1.1;
  font-weight: 700;
  font-variant-numeric: tabular-nums;
  letter-spacing: -0.015em;
}

.stat-tile-hint {
  font-size: 0.7rem;
  color: color-mix(in oklch, var(--color-base-content) 50%, transparent);
}

/* Content card: shared container for product list table, etc. */
.content-card {
  background: var(--color-base-100);
  border: 1px solid var(--color-base-300);
  border-radius: 0.625rem;
  overflow: hidden;
}

.content-card-header {
  padding: 0.875rem 1rem;
  border-bottom: 1px solid var(--color-base-300);
  background: color-mix(in oklch, var(--color-base-200) 50%, var(--color-base-100));
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 1rem;
  flex-wrap: wrap;
}

.content-card-body { padding: 0; }

/* Status pill */
.status-pill {
  display: inline-flex;
  align-items: center;
  gap: 0.375rem;
  padding: 0.125rem 0.625rem;
  border-radius: 9999px;
  font-size: 0.75rem;
  font-weight: 500;
  white-space: nowrap;
}

.status-pill::before {
  content: "";
  width: 0.45rem;
  height: 0.45rem;
  border-radius: 9999px;
  background: currentColor;
}

.status-pill--success { background: color-mix(in oklch, var(--color-success) 14%, transparent); color: var(--color-success); }
.status-pill--warning { background: color-mix(in oklch, var(--color-warning) 18%, transparent); color: var(--color-warning); }
.status-pill--error   { background: color-mix(in oklch, var(--color-error) 14%, transparent); color: var(--color-error); }
.status-pill--info    { background: color-mix(in oklch, var(--color-info) 14%, transparent); color: var(--color-info); }
.status-pill--accent  { background: color-mix(in oklch, var(--color-accent) 14%, transparent); color: var(--color-accent); }
.status-pill--neutral { background: color-mix(in oklch, var(--color-base-content) 9%, transparent); color: color-mix(in oklch, var(--color-base-content) 70%, transparent); }

/* Section label (validate review, etc) */
.section-label {
  display: inline-flex;
  align-items: center;
  gap: 0.5rem;
  font-size: 0.72rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  font-weight: 600;
  color: color-mix(in oklch, var(--color-base-content) 65%, transparent);
  padding-bottom: 0.5rem;
  border-bottom: 1px solid var(--color-base-300);
  margin-bottom: 0.75rem;
  width: 100%;
}

.section-label-hint {
  text-transform: none;
  letter-spacing: normal;
  font-weight: 400;
  font-size: 0.7rem;
  color: color-mix(in oklch, var(--color-base-content) 40%, transparent);
  font-style: italic;
}

/* Side-by-side field comparison */
.compare-current {
  border-left: 3px solid color-mix(in oklch, var(--color-info) 45%, transparent);
  padding-left: 0.75rem;
}

.compare-enriched {
  border-left: 3px solid color-mix(in oklch, var(--color-primary) 55%, transparent);
  padding-left: 0.75rem;
}

.compare-arrow {
  display: none;
  align-items: center;
  justify-content: center;
  color: color-mix(in oklch, var(--color-base-content) 35%, transparent);
}

@media (min-width: 1024px) {
  .compare-arrow { display: flex; }
}

/* Hero Approve button */
.btn-approve-hero {
  font-size: 0.95rem;
  font-weight: 600;
  padding: 0 1.25rem;
  height: 2.5rem;
  box-shadow: 0 2px 0 color-mix(in oklch, var(--color-success) 60%, transparent);
}

/* -----------------------------------------
   10. LANDING / HOME TILES
   ----------------------------------------- */
.launch-tile {
  display: flex;
  flex-direction: column;
  gap: 0.5rem;
  padding: 1.25rem;
  background: var(--color-base-100);
  border: 1px solid var(--color-base-300);
  border-radius: 0.75rem;
  text-decoration: none;
  color: var(--color-base-content);
  transition: border-color 0.2s ease, transform 0.2s ease, box-shadow 0.2s ease;
  position: relative;
  overflow: hidden;
}

.launch-tile::after {
  content: "";
  position: absolute;
  inset: auto -2rem -2rem auto;
  width: 6rem;
  height: 6rem;
  background: color-mix(in oklch, var(--color-primary) 10%, transparent);
  border-radius: 9999px;
  opacity: 0.6;
  transition: transform 0.3s ease, opacity 0.3s ease;
  pointer-events: none;
}

.launch-tile:hover {
  border-color: color-mix(in oklch, var(--color-primary) 45%, var(--color-base-300));
  transform: translateY(-1px);
  box-shadow: 0 6px 24px rgba(15, 20, 34, 0.06);
}

.launch-tile:hover::after {
  transform: scale(1.15);
  opacity: 0.8;
}

.launch-tile-icon {
  width: 2.25rem;
  height: 2.25rem;
  border-radius: 0.5rem;
  background: color-mix(in oklch, var(--color-primary) 14%, transparent);
  color: var(--color-primary);
  display: inline-flex;
  align-items: center;
  justify-content: center;
  position: relative;
  z-index: 1;
}

.launch-tile-title {
  font-size: 1rem;
  font-weight: 600;
  letter-spacing: -0.01em;
  position: relative;
  z-index: 1;
}

.launch-tile-desc {
  font-size: 0.8rem;
  color: color-mix(in oklch, var(--color-base-content) 55%, transparent);
  line-height: 1.4;
  position: relative;
  z-index: 1;
}

.launch-tile-meta {
  margin-top: auto;
  font-size: 0.7rem;
  text-transform: uppercase;
  letter-spacing: 0.06em;
  color: color-mix(in oklch, var(--color-primary) 70%, var(--color-base-content));
  font-weight: 600;
  position: relative;
  z-index: 1;
}

/* -----------------------------------------
   11. STATUS TAB STRIP (page-level filter)
   ----------------------------------------- */
.status-tabs {
  display: flex;
  align-items: stretch;
  gap: 0.25rem;
  border-bottom: 1px solid var(--color-base-300);
  margin-bottom: 0.875rem;
  overflow-x: auto;
  scrollbar-width: none;
}
.status-tabs::-webkit-scrollbar { display: none; }

.status-tab {
  display: inline-flex;
  align-items: center;
  gap: 0.4rem;
  padding: 0.5rem 0.875rem;
  font-size: 0.82rem;
  font-weight: 500;
  color: color-mix(in oklch, var(--color-base-content) 65%, transparent);
  border-bottom: 2px solid transparent;
  margin-bottom: -1px;
  white-space: nowrap;
  text-decoration: none;
  transition: color 0.15s ease, border-color 0.15s ease;
}

.status-tab:hover {
  color: var(--color-base-content);
}

.status-tab.is-active {
  color: var(--color-primary);
  border-bottom-color: var(--color-primary);
  font-weight: 600;
}

.status-tab-count {
  display: inline-flex;
  align-items: center;
  padding: 0 0.45rem;
  min-width: 1.4rem;
  height: 1.15rem;
  border-radius: 9999px;
  background: color-mix(in oklch, var(--color-base-content) 8%, transparent);
  color: color-mix(in oklch, var(--color-base-content) 70%, transparent);
  font-size: 0.68rem;
  font-weight: 600;
  font-variant-numeric: tabular-nums;
  line-height: 1;
}

.status-tab.is-active .status-tab-count {
  background: color-mix(in oklch, var(--color-primary) 18%, transparent);
  color: var(--color-primary);
}

/* -----------------------------------------
   12. COMPACT FILTER BAR
   ----------------------------------------- */
.filter-bar {
  display: flex;
  align-items: center;
  gap: 0.5rem;
  flex-wrap: wrap;
  width: 100%;
}

.filter-bar .search-wrap {
  position: relative;
  flex: 1;
  min-width: 220px;
  max-width: 420px;
}

.filter-bar .search-wrap svg {
  position: absolute;
  left: 0.625rem;
  top: 50%;
  transform: translateY(-50%);
  width: 1rem;
  height: 1rem;
  color: color-mix(in oklch, var(--color-base-content) 45%, transparent);
  pointer-events: none;
}

.filter-bar .search-wrap input {
  padding-left: 2rem;
}

.filter-bar .spacer { flex: 1; }

/* -----------------------------------------
   13. PRODUCT ROW THUMBNAIL
   ----------------------------------------- */
.product-thumb {
  width: 2.25rem;
  height: 2.25rem;
  border-radius: 0.45rem;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  font-weight: 700;
  font-size: 0.75rem;
  letter-spacing: 0.02em;
  flex-shrink: 0;
  text-transform: uppercase;
  border: 1px solid transparent;
}

.product-thumb-img {
  object-fit: cover;
  background: var(--color-base-200);
  border: 1px solid var(--color-base-300);
}

/* Eight-color palette cycled by SKU length % 8 */
.thumb-c0 { background: color-mix(in oklch, #2f6fd3 18%, transparent); color: #2f6fd3; }
.thumb-c1 { background: color-mix(in oklch, #6c5ce7 18%, transparent); color: #6c5ce7; }
.thumb-c2 { background: color-mix(in oklch, #00a884 20%, transparent); color: #008a6d; }
.thumb-c3 { background: color-mix(in oklch, #e27d3b 20%, transparent); color: #c95f1a; }
.thumb-c4 { background: color-mix(in oklch, #d4487e 18%, transparent); color: #bb2d65; }
.thumb-c5 { background: color-mix(in oklch, #2d8ab0 20%, transparent); color: #1f6f91; }
.thumb-c6 { background: color-mix(in oklch, #7a6042 22%, transparent); color: #7a6042; }
.thumb-c7 { background: color-mix(in oklch, #4a5568 18%, transparent); color: #4a5568; }

[data-theme="dim"] .thumb-c0,
[data-theme="dim"] .thumb-c1,
[data-theme="dim"] .thumb-c2,
[data-theme="dim"] .thumb-c3,
[data-theme="dim"] .thumb-c4,
[data-theme="dim"] .thumb-c5,
[data-theme="dim"] .thumb-c6,
[data-theme="dim"] .thumb-c7 {
  color-scheme: dark;
  filter: brightness(1.15) saturate(1.1);
}

.product-cell {
  display: flex;
  align-items: center;
  gap: 0.625rem;
  min-width: 0;
}

.product-cell-body {
  display: flex;
  flex-direction: column;
  min-width: 0;
  gap: 0.1rem;
}

.product-cell-primary {
  font-size: 0.82rem;
  font-weight: 600;
  color: var(--color-base-content);
  line-height: 1.2;
  white-space: nowrap;
  overflow: hidden;
  text-overflow: ellipsis;
  max-width: 320px;
}

.product-cell-secondary {
  font-size: 0.7rem;
  color: color-mix(in oklch, var(--color-base-content) 55%, transparent);
  font-family: ui-monospace, SFMono-Regular, monospace;
  line-height: 1.1;
}

.product-cell-brand {
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
  font-size: 0.7rem;
  color: color-mix(in oklch, var(--color-base-content) 65%, transparent);
  font-weight: 500;
}

/* Enabled/disabled indicator on product thumbnails. Uses a real child
   <span class="product-thumb-dot"> rather than ::after — because ::after
   doesn't render on <img> tags, and using a real element sidesteps any
   pseudo-element quirks entirely. Green = live in Akeneo, red = disabled. */
.product-thumb-box {
  position: relative;
  display: inline-block;
  flex-shrink: 0;
  line-height: 0;
}
.product-thumb-dot {
  position: absolute;
  right: -3px;
  top: -3px;
  width: 0.6rem;
  height: 0.6rem;
  border-radius: 9999px;
  border: 1.5px solid var(--color-base-100);
  pointer-events: none;
  z-index: 1;
}
.product-thumb-box.is-enabled  .product-thumb-dot { background: var(--color-success); }
.product-thumb-box.is-disabled .product-thumb-dot { background: var(--color-error); }
.product-thumb-box.is-disabled .product-thumb {
  border-color: color-mix(in oklch, var(--color-error) 45%, transparent);
}

/* -----------------------------------------
   14. FLOATING SELECTION BAR
   ----------------------------------------- */
.selection-bar {
  position: fixed;
  bottom: 1.25rem;
  left: 50%;
  transform: translateX(-50%) translateY(0);
  display: flex;
  align-items: center;
  gap: 0.625rem;
  padding: 0.5rem 0.5rem 0.5rem 1rem;
  background: var(--color-base-100);
  border: 1px solid var(--color-base-300);
  border-radius: 9999px;
  box-shadow: 0 10px 36px rgba(15, 20, 34, 0.14), 0 2px 8px rgba(15, 20, 34, 0.08);
  z-index: 40;
  max-width: calc(100vw - 2rem);
  min-width: min(560px, 92vw);
  transition: transform 0.25s ease, opacity 0.25s ease;
}

.selection-bar.is-hidden {
  transform: translateX(-50%) translateY(200%);
  opacity: 0;
  pointer-events: none;
}

.selection-bar-count {
  display: inline-flex;
  align-items: center;
  gap: 0.5rem;
  font-size: 0.85rem;
  color: var(--color-base-content);
  font-weight: 500;
}

.selection-bar-count strong {
  font-weight: 700;
  font-variant-numeric: tabular-nums;
}

.selection-bar-divider {
  width: 1px;
  height: 1.5rem;
  background: var(--color-base-300);
}

/* Data-density table tweaks so product-cell rows breathe right */
.table-products thead th {
  font-size: 0.68rem;
  padding: 0.5rem 0.6rem;
}

/* Sortable column headers — rendered by the sort_header macro in
   _table.html. The whole <th> is clickable via the nested <a.sort-link>;
   we keep the link block-level so the hover target matches the cell. */
.sortable-th { padding: 0 !important; }

.sortable-th .sort-link {
  display: inline-flex;
  align-items: center;
  gap: 0.3rem;
  padding: 0.5rem 0.6rem;
  width: 100%;
  color: inherit;
  text-decoration: none;
  cursor: pointer;
  transition: background-color 0.12s ease, color 0.12s ease;
  user-select: none;
}

.sortable-th.text-right .sort-link {
  justify-content: flex-end;
}

.sortable-th .sort-link:hover {
  background: color-mix(in oklch, var(--color-base-content) 6%, transparent);
  color: var(--color-base-content);
}

.sortable-th.is-sorted {
  color: var(--color-primary);
}

.sortable-th.is-sorted .sort-link {
  color: var(--color-primary);
  background: color-mix(in oklch, var(--color-primary) 7%, transparent);
}

.sort-icon {
  width: 0.8rem;
  height: 0.8rem;
  flex-shrink: 0;
}

.sort-icon.is-idle {
  opacity: 0.35;
}

.sortable-th .sort-link:hover .sort-icon.is-idle {
  opacity: 0.7;
}

.table-products tbody td {
  padding: 0.55rem 0.6rem;
  vertical-align: middle;
}

.table-products tbody tr {
  border-bottom: 1px solid color-mix(in oklch, var(--color-base-300) 65%, transparent);
}

.table-products tbody tr:last-child { border-bottom: none; }

/* Legacy horizontal-nav active indicator (retained for any sub-menus) */
.nav-active {
  background-color: color-mix(in oklch, var(--color-primary) 12%, transparent);
  color: var(--color-primary);
  border-radius: 0.375rem;
  font-weight: 600;
}

/* Keep .navbar component usable on auth pages (login form) */
.navbar {
  background: var(--color-base-100);
  border-bottom: 1px solid var(--color-base-300);
}

/* -----------------------------------------
   15. DASHBOARD OPS — stat tiles, pills,
       source rows, section dividers.
   Feeds the Activity Feed / People Panel /
   Pipeline Health / System Health cards.
   ----------------------------------------- */

.tnum { font-variant-numeric: tabular-nums; }
.mono { font-family: ui-monospace, SFMono-Regular, Menlo, Consolas, monospace; }
.text-muted { color: color-mix(in oklch, var(--color-base-content) 65%, transparent); }
.text-mute2 { color: color-mix(in oklch, var(--color-base-content) 40%, transparent); }

/* Page header — matches the prototype's design-system spec. */
.page-header {
  display: flex;
  align-items: flex-start;
  justify-content: space-between;
  gap: 1rem;
  margin-bottom: 1.25rem;
}
.page-header__title {
  font-size: 1.4rem;
  font-weight: 700;
  letter-spacing: -0.015em;
  margin: 0;
}
.page-header__subtitle {
  margin: 0.2rem 0 0;
  color: color-mix(in oklch, var(--color-base-content) 65%, transparent);
  font-size: 0.84rem;
}
.page-header__actions {
  display: flex;
  gap: 0.5rem;
  flex-shrink: 0;
}

/* Dashboard multi-column layouts — defined in pure CSS (no Tailwind
   utility dependency) because the browser-JIT for Tailwind has a
   visible first-paint delay and we want these to be right on initial
   render. Shared breakpoint matches the app-shell mobile cutover at
   900px; the `lg` variant jumps to a 2-col grid at ≥1024px. */
.dash-stack { display: flex; flex-direction: column; gap: 1.25rem; }

.dash-2col {
  display: grid;
  grid-template-columns: 1fr;
  gap: 1.25rem;
}
@media (min-width: 1024px) {
  .dash-2col { grid-template-columns: 1fr 1fr; }
  .dash-2col--wide-left { grid-template-columns: 1.1fr 1fr; }
}

/* Stat tiles — compact KPI row under the page header. 4 across at
   wide widths, auto-fit-stack below 720px. Explicit media query
   beats auto-fit when the container briefly gets very narrow during
   sidebar drawer animations. */
.stat-tile-grid {
  display: grid;
  grid-template-columns: 1fr;
  gap: 0.85rem;
  margin-bottom: 1.25rem;
}
@media (min-width: 640px) {
  .stat-tile-grid { grid-template-columns: repeat(2, 1fr); }
}
@media (min-width: 1024px) {
  .stat-tile-grid { grid-template-columns: repeat(4, 1fr); }
}
.stat-tile {
  background: var(--color-base-100);
  border: 1px solid var(--color-base-300);
  border-radius: 10px;
  padding: 0.9rem 1rem;
  position: relative;
  overflow: hidden;
}
.stat-tile::before {
  content: "";
  position: absolute;
  left: 0; top: 0; bottom: 0;
  width: 3px;
  background: var(--tile-accent, var(--color-primary));
  opacity: 0.75;
}
.stat-tile--info    { --tile-accent: var(--color-info, #4a8cff); }
.stat-tile--accent  { --tile-accent: var(--color-accent); }
.stat-tile--primary { --tile-accent: var(--color-primary); }
.stat-tile--success { --tile-accent: var(--color-success, #22c55e); }
.stat-tile--warning { --tile-accent: var(--color-warning, #d97706); }
.stat-tile--error   { --tile-accent: var(--color-error, #dc2626); }

.stat-tile__label {
  font-size: 0.7rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  color: color-mix(in oklch, var(--color-base-content) 60%, transparent);
  margin-bottom: 0.3rem;
  font-weight: 600;
}
.stat-tile__value {
  font-size: 1.55rem;
  font-weight: 700;
  letter-spacing: -0.02em;
  font-variant-numeric: tabular-nums;
  line-height: 1.1;
}
.stat-tile__hint {
  font-size: 0.72rem;
  color: color-mix(in oklch, var(--color-base-content) 55%, transparent);
  margin-top: 0.25rem;
}

/* Status pill — sizing + semantic variants. DaisyUI has .badge but we
   want tighter radii + the neutral/accent vocabulary the design uses. */
.status-pill {
  display: inline-flex;
  align-items: center;
  padding: 0.1rem 0.5rem;
  border-radius: 999px;
  font-size: 0.68rem;
  font-weight: 600;
  line-height: 1.45;
  letter-spacing: 0.01em;
  text-transform: lowercase;
  border: 1px solid transparent;
  white-space: nowrap;
}
.status-pill--success { background: color-mix(in oklch, var(--color-success, #22c55e) 18%, transparent); color: var(--color-success, #22c55e); border-color: color-mix(in oklch, var(--color-success, #22c55e) 30%, transparent); }
.status-pill--error   { background: color-mix(in oklch, var(--color-error, #dc2626) 18%, transparent); color: var(--color-error, #dc2626); border-color: color-mix(in oklch, var(--color-error, #dc2626) 30%, transparent); }
.status-pill--warning { background: color-mix(in oklch, var(--color-warning, #d97706) 20%, transparent); color: var(--color-warning, #d97706); border-color: color-mix(in oklch, var(--color-warning, #d97706) 30%, transparent); }
.status-pill--info    { background: color-mix(in oklch, var(--color-info, #4a8cff) 18%, transparent); color: var(--color-info, #4a8cff); border-color: color-mix(in oklch, var(--color-info, #4a8cff) 30%, transparent); }
.status-pill--accent  { background: color-mix(in oklch, var(--color-accent) 18%, transparent); color: var(--color-accent); border-color: color-mix(in oklch, var(--color-accent) 30%, transparent); }
.status-pill--neutral { background: color-mix(in oklch, var(--color-base-content) 10%, transparent); color: color-mix(in oklch, var(--color-base-content) 75%, transparent); border-color: var(--color-base-300); }

/* Labeled hairline rule inside cards (used between sections of a card). */
.section-divider {
  display: flex;
  align-items: center;
  gap: 0.75rem;
  margin: 1.1rem 0 0.55rem;
  font-size: 0.68rem;
  text-transform: uppercase;
  letter-spacing: 0.09em;
  color: color-mix(in oklch, var(--color-base-content) 55%, transparent);
  font-weight: 600;
}
.section-divider__rule {
  flex: 1;
  height: 1px;
  background: var(--color-base-300);
}

/* Source-row bar — reused by Pipeline Health for rejection reasons +
   most-edited fields (same shape the dashboard already uses for the
   Source Coverage card). */
.source-row {
  display: grid;
  grid-template-columns: 1fr 2fr 56px;
  align-items: center;
  gap: 0.65rem;
}
.source-row + .source-row { margin-top: 0.3rem; }
.source-row__bar-track {
  height: 6px;
  background: var(--color-base-300);
  border-radius: 999px;
  overflow: hidden;
}
.source-row__bar-fill {
  height: 100%;
  border-radius: 999px;
  background: var(--color-primary);
  transition: width 0.2s ease;
}
.source-row__pct {
  text-align: right;
  font-size: 0.78rem;
  color: color-mix(in oklch, var(--color-base-content) 75%, transparent);
}

/* Avatar bubble used in the activity feed + people panel. */
.avatar-bubble {
  width: 26px;
  height: 26px;
  border-radius: 50%;
  background: linear-gradient(135deg, #5e81ac, #7b8fff);
  color: white;
  display: grid;
  place-items: center;
  font-size: 0.66rem;
  font-weight: 600;
  flex-shrink: 0;
  letter-spacing: 0;
}
.avatar-bubble--system {
  background: var(--color-base-300);
  color: color-mix(in oklch, var(--color-base-content) 60%, transparent);
}

/* Activity-feed row shell — grid layout for when / who+action / drill-in. */
.activity-row {
  display: grid;
  grid-template-columns: 72px 1fr auto;
  align-items: center;
  gap: 0.75rem;
  padding: 0.55rem 1rem;
  font-size: 0.82rem;
  border-bottom: 1px solid color-mix(in oklch, var(--color-base-300) 65%, transparent);
}
.activity-row:last-child { border-bottom: none; }
.activity-row__when { font-size: 0.72rem; color: color-mix(in oklch, var(--color-base-content) 55%, transparent); }
.activity-row__body { display: flex; align-items: center; gap: 0.6rem; min-width: 0; }
.activity-row__headline {
  display: flex;
  align-items: center;
  gap: 0.4rem;
  flex-wrap: wrap;
}
.activity-row__who { font-size: 0.84rem; font-weight: 500; }
.activity-row__meta { font-size: 0.72rem; color: color-mix(in oklch, var(--color-base-content) 60%, transparent); margin-top: 0.15rem; }

/* Error-log tail — monospace with level coloring. */
.error-log {
  background: var(--color-base-200);
  border: 1px solid var(--color-base-300);
  border-radius: 7px;
  padding: 0.6rem 0.7rem;
  font-family: ui-monospace, SFMono-Regular, Menlo, Consolas, monospace;
  font-size: 0.72rem;
  line-height: 1.55;
  overflow: auto;
  max-height: 280px;
}
.error-log__row {
  display: grid;
  grid-template-columns: 54px 58px 1fr;
  gap: 0.5rem;
  padding: 0.25rem 0;
  border-bottom: 1px dashed var(--color-base-300);
}
.error-log__row:last-child { border-bottom: none; }
.error-log__level--error { color: var(--color-error, #dc2626); font-weight: 600; }
.error-log__level--warn  { color: var(--color-warning, #d97706); font-weight: 600; }
.error-log__level--info  { color: color-mix(in oklch, var(--color-base-content) 70%, transparent); font-weight: 600; }

/* Cost anomaly callout — warning-tinted band above the two-column System
   Health content. */
.cost-anomaly-banner {
  display: flex;
  align-items: center;
  gap: 0.85rem;
  padding: 0.75rem 0.9rem;
  border: 1px solid color-mix(in oklch, var(--color-warning, #d97706) 30%, transparent);
  background: color-mix(in oklch, var(--color-warning, #d97706) 14%, transparent);
  border-radius: 8px;
  margin-bottom: 1rem;
}
.cost-anomaly-banner__spark { flex-shrink: 0; }

/* Cost breakdown card — donut on the left, legend + total on the
   right, optional stats strip below. Matches the design's
   Cost-breakdown tile layout. */
.cost-breakdown__body {
  display: grid;
  grid-template-columns: 200px 1fr;
  gap: 1.5rem;
  align-items: center;
}
@media (max-width: 720px) {
  .cost-breakdown__body { grid-template-columns: 1fr; justify-items: center; }
}

.cost-breakdown__donut {
  width: 180px;
  height: 180px;
  flex-shrink: 0;
}

.cost-breakdown__legend {
  display: flex;
  flex-direction: column;
  gap: 0.05rem;
  min-width: 0;
}

.cost-breakdown__row {
  display: grid;
  grid-template-columns: 12px 1fr auto auto;
  gap: 0.75rem;
  align-items: center;
  padding: 0.3rem 0;
  font-size: 0.82rem;
}
.cost-breakdown__row--total { padding-top: 0.5rem; }

.cost-breakdown__dot {
  width: 10px;
  height: 10px;
  border-radius: 50%;
  background: var(--color-primary);
  flex-shrink: 0;
}

.cost-breakdown__label {
  min-width: 0;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}

.cost-breakdown__calls {
  color: color-mix(in oklch, var(--color-base-content) 55%, transparent);
  font-size: 0.75rem;
}

.cost-breakdown__cost {
  font-weight: 600;
  text-align: right;
  min-width: 4.5rem;
}

.cost-breakdown__divider {
  height: 1px;
  background: var(--color-base-300);
  margin: 0.35rem 0 0.25rem;
  grid-column: 1 / -1;
}

/* Secondary stats strip — three compact cells below the donut. */
.cost-breakdown__stats {
  display: grid;
  grid-template-columns: repeat(3, 1fr);
  gap: 1rem;
  margin-top: 1rem;
  padding-top: 1rem;
  border-top: 1px solid var(--color-base-300);
}
.cost-breakdown__stat-label {
  font-size: 0.65rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  color: color-mix(in oklch, var(--color-base-content) 55%, transparent);
  font-weight: 600;
  margin-bottom: 0.2rem;
}
.cost-breakdown__stat-value {
  font-size: 1.05rem;
  font-weight: 700;
  letter-spacing: -0.015em;
  line-height: 1.1;
}
.cost-breakdown__stat-hint {
  font-size: 0.7rem;
  color: color-mix(in oklch, var(--color-base-content) 50%, transparent);
  margin-top: 0.15rem;
}

/* Tight table density used by People panel + System Health. */
.table-ops {
  width: 100%;
  border-collapse: collapse;
  font-size: 0.82rem;
}
.table-ops thead th {
  text-align: left;
  font-size: 0.68rem;
  text-transform: uppercase;
  letter-spacing: 0.08em;
  padding: 0.5rem 0.6rem;
  color: color-mix(in oklch, var(--color-base-content) 55%, transparent);
  border-bottom: 1px solid var(--color-base-300);
  font-weight: 600;
}
.table-ops tbody td {
  padding: 0.55rem 0.6rem;
  vertical-align: middle;
  border-bottom: 1px solid color-mix(in oklch, var(--color-base-300) 65%, transparent);
}
.table-ops tbody tr:last-child td { border-bottom: none; }
.table-ops .text-right { text-align: right; }
