/* 2026-04-22 rebuild — `--cp-*` tokens must be available wherever portal
   markup renders, including the two modals whose partials are rendered as
   siblings to `.cp-portal` (not descendants). Without the modal-root
   selectors below, dialog + body `background: var(--cp-bg-card)` etc.
   silently fail to resolve and the modals render transparent.
   Fix: 2026-04-23 Phase 10 QA finding. */
.cp-portal,
.cp-door-issue-modal,
.cp-portal-user-delete-modal {
  --cp-bg: #f7f5f0;
  --cp-bg-card: #ffffff;
  --cp-bg-hover: #f2efe9;
  --cp-bg-inset: #edeae3;
  --cp-rule: rgba(0, 0, 0, 0.07);
  --cp-rule-strong: rgba(0, 0, 0, 0.11);
  --cp-ink: #1a1815;
  --cp-ink-2: #3d3a34;
  --cp-ink-3: #6b6458;
  --cp-ink-4: #9a9188;
  --cp-accent: #c49510;
  --cp-accent-bg: rgba(196, 149, 16, 0.08);
  --cp-accent-border: rgba(196, 149, 16, 0.2);
  --cp-active-ink: #7a5c00;
  --cp-active-bg: rgba(122, 92, 0, 0.1);
  --cp-active-border: rgba(122, 92, 0, 0.22);
  --cp-green: #1e7a45;
  --cp-green-bg: rgba(30, 122, 69, 0.08);
  --cp-red: #b03030;
  --cp-red-bg: rgba(176, 48, 48, 0.07);
  --cp-red-border: rgba(176, 48, 48, 0.18);
  --cp-blue: #2563a8;
  --cp-blue-bg: rgba(37, 99, 168, 0.08);
  --cp-shadow-sm: 0 1px 3px rgba(0, 0, 0, 0.06), 0 1px 2px rgba(0, 0, 0, 0.04);
  --cp-shadow-md: 0 4px 12px rgba(0, 0, 0, 0.08), 0 2px 4px rgba(0, 0, 0, 0.04);
  --cp-shadow-card: 0 1px 4px rgba(0, 0, 0, 0.06), 0 0 0 1px rgba(0, 0, 0, 0.05);
  color: var(--cp-ink);
}

html[data-theme="dark"] .cp-portal,
html[data-theme="dark"] .cp-door-issue-modal,
html[data-theme="dark"] .cp-portal-user-delete-modal {
  --cp-bg: #0b1220;
  --cp-bg-card: #0f172a;
  --cp-bg-hover: rgba(148, 163, 184, 0.08);
  --cp-bg-inset: #0a0f1c;
  --cp-rule: rgba(148, 163, 184, 0.22);
  --cp-rule-strong: rgba(71, 85, 105, 0.58);
  --cp-ink: rgba(226, 232, 240, 0.95);
  --cp-ink-2: rgba(203, 213, 225, 0.9);
  --cp-ink-3: rgba(148, 163, 184, 0.8);
  --cp-ink-4: rgba(100, 116, 139, 0.8);
  --cp-accent: #e3b008;
  --cp-accent-bg: rgba(227, 176, 8, 0.18);
  --cp-accent-border: rgba(227, 176, 8, 0.3);
  --cp-active-ink: #f4d98b;
  --cp-active-bg: rgba(227, 176, 8, 0.14);
  --cp-active-border: rgba(227, 176, 8, 0.32);
  --cp-green: #4ade80;
  --cp-green-bg: rgba(30, 122, 69, 0.22);
  --cp-red: #f87171;
  --cp-red-bg: rgba(176, 48, 48, 0.22);
  --cp-red-border: rgba(176, 48, 48, 0.35);
  --cp-blue: #60a5fa;
  --cp-blue-bg: rgba(37, 99, 168, 0.22);
  --cp-shadow-sm: none;
  --cp-shadow-md: 0 12px 28px rgba(0, 0, 0, 0.38);
  --cp-shadow-card: 0 0 0 1px rgba(148, 163, 184, 0.18);
}

.cp-portal__banner {
  margin: 18px 0 16px;
}

.cp-portal .trial-banner {
  padding: 12px 16px;
  background: rgba(230, 86, 30, 0.1);
  border: 1px solid rgba(230, 86, 30, 0.3);
  border-radius: 10px;
}

.cp-portal .trial-banner--actions {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 14px;
  position: sticky;
  top: 12px;
  z-index: 20;
}

.cp-portal .trial-banner__content {
  min-width: 240px;
}

.cp-portal .trial-banner__actions {
  display: flex;
  align-items: center;
  gap: 10px;
  flex-wrap: wrap;
  justify-content: flex-end;
}

.cp-portal .trial-banner__actions form {
  margin: 0;
  display: inline;
}

.cp-portal .trial-banner__actions .btn {
  min-height: 40px;
}

.cp-portal .trial-banner__actions .btn--compact {
  padding: 8px 12px;
}

.cp-portal .trial-banner__actions .btn-ghost {
  border-color: rgba(230, 86, 30, 0.35);
}

.cp-portal svg {
  display: block;
}

.cp-page-header {
  padding: 22px 0 0;
  display: flex;
  align-items: flex-start;
  justify-content: space-between;
  gap: 16px;
}

.cp-page-greeting {
  font-size: 11.5px;
  font-weight: 400;
  color: var(--cp-ink-4);
  margin-bottom: 3px;
}

.cp-page-title {
  font-family: var(--font-display);
  font-size: 22px;
  font-weight: 700;
  color: var(--cp-ink);
  letter-spacing: -0.025em;
  margin: 0;
}

.cp-page-header__right {
  display: flex;
  align-items: center;
  gap: 8px;
  flex-wrap: wrap;
  justify-content: flex-end;
}

.cp-page-header__right .btn svg {
  width: 13px;
  height: 13px;
}

.cp-status-strip {
  display: grid;
  grid-template-columns: repeat(4, minmax(0, 1fr));
  gap: 14px;
  /* 2026-04-23: the old `padding: 20px 0` here used to approximate the
     rhythm above + below the strip. That rhythm is now owned by the
     `.cp-main` flex-gap, so the padding stacked with the gap and produced
     36px instead of 16px. Removed. */
}

.cp-status-strip--3 {
  grid-template-columns: repeat(3, minmax(0, 1fr));
}

.cp-status-strip--5 {
  grid-template-columns: repeat(5, minmax(0, 1fr));
}

.cp-status-strip--compact {
  grid-template-columns: repeat(4, minmax(0, 1fr));
}

.cp-stat-card {
  background: var(--cp-bg-card);
  border-radius: 10px;
  padding: 16px 18px;
  box-shadow: var(--cp-shadow-card);
  display: flex;
  flex-direction: column;
  gap: 10px;
  min-width: 0;
}

.cp-stat-card:hover {
  box-shadow: var(--cp-shadow-md);
}

.cp-stat-top {
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 12px;
}

.cp-stat-label {
  font-size: 11px;
  font-weight: 500;
  color: var(--cp-ink-4);
  line-height: 1.3;
}

.cp-stat-icon-wrap {
  width: 28px;
  height: 28px;
  border-radius: 6px;
  display: flex;
  align-items: center;
  justify-content: center;
  background: var(--cp-bg-inset);
  color: var(--cp-ink-4);
  flex-shrink: 0;
}

.cp-stat-icon-wrap--danger {
  background: var(--cp-red-bg);
  color: var(--cp-red);
}

.cp-stat-icon-wrap--success {
  background: var(--cp-green-bg);
  color: var(--cp-green);
}

.cp-stat-icon-wrap svg {
  width: 13px;
  height: 13px;
}

.cp-stat-value {
  font-family: var(--font-display);
  font-size: 28px;
  font-weight: 700;
  color: var(--cp-ink);
  letter-spacing: -0.03em;
  line-height: 1;
}

.cp-stat-value--danger {
  color: var(--cp-red);
}

.cp-stat-value--success {
  color: var(--cp-green);
}

.cp-stat-footer {
  font-size: 11px;
  font-weight: 400;
  color: var(--cp-ink-4);
}

.cp-stat-footer--danger {
  color: var(--cp-red);
}

.cp-stat-footer--success {
  color: var(--cp-green);
}

/* Compact Contact-your-provider KPI — shared by Properties / Inspections /
   Billing / Documents as the 4th stat in `.cp-status-strip`. Matches the
   chrome of `.cp-stat-card`; swaps the big numeric value for a provider
   name + compact email/phone rows so the card is same height as siblings. */
.cp-stat-card--contact {
  gap: 8px;
}
.cp-stat-contact__name {
  font-family: var(--font-display);
  font-size: 16px;
  font-weight: 600;
  color: var(--cp-ink);
  line-height: 1.2;
  letter-spacing: -0.01em;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
  min-width: 0;
}
.cp-stat-contact__links {
  list-style: none;
  padding: 0;
  margin: 0;
  display: flex;
  flex-direction: column;
  gap: 4px;
  min-width: 0;
}
.cp-stat-contact__links li {
  min-width: 0;
  display: block;
}
.cp-stat-contact__links a {
  display: inline-flex;
  align-items: center;
  gap: 6px;
  font-size: 12px;
  color: var(--cp-accent);
  text-decoration: none;
  line-height: 1.3;
  max-width: 100%;
  min-width: 0;
}
.cp-stat-contact__links a:hover {
  text-decoration: underline;
}
.cp-stat-contact__links a svg {
  flex-shrink: 0;
  color: var(--cp-ink-4);
}
.cp-stat-contact__links a span {
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
  min-width: 0;
}
.cp-stat-contact__empty {
  font-size: 12px;
  color: var(--cp-ink-4);
}
.cp-stat-contact__empty a {
  color: var(--cp-accent);
  text-decoration: none;
  font-weight: 500;
}
.cp-stat-contact__empty a:hover {
  text-decoration: underline;
}

.cp-content-grid {
  display: grid;
  grid-template-columns: minmax(0, 1fr) 300px;
  gap: 14px;
  padding: 0 0 28px;
  align-items: start;
}

.cp-content-grid > :last-child {
  position: sticky;
  top: 24px;
}

.cp-content-grid--full {
  grid-template-columns: 1fr;
}

/* Grid main column min-width fix */
.cp-grid-main { min-width: 0; }

/* Stack spacing for all child elements in grid columns */
.cp-grid-main > * + * { margin-top: 14px; }
.cp-content-grid > :last-child > * + * { margin-top: 12px; }

.cp-card--field-work {
  grid-area: field;
}

.cp-card--recent-inspections {
  grid-area: recent;
}

.cp-card--quick-actions {
  grid-area: quick;
}

.cp-card--workspace {
  grid-area: workspace;
}

.cp-card--field-work,
.cp-card--recent-inspections,
.cp-card--quick-actions,
.cp-card--workspace {
  display: flex;
  flex-direction: column;
  gap: 14px;
  min-width: 0;
}

.cp-card {
  background: var(--cp-bg-card);
  border-radius: 12px;
  box-shadow: var(--cp-shadow-card);
  overflow: hidden;
}

.cp-card-header {
  padding: 16px 20px 14px;
  border-bottom: 1px solid var(--cp-rule);
  display: flex;
  align-items: center;
  justify-content: space-between;
  gap: 10px;
}

.cp-card-label {
  font-size: 10px;
  font-weight: 600;
  letter-spacing: 0.1em;
  text-transform: uppercase;
  color: var(--cp-accent);
  margin-bottom: 2px;
  display: block;
}

.cp-card-title {
  font-size: 14px;
  font-weight: 600;
  color: var(--cp-ink);
  letter-spacing: -0.01em;
  margin: 0;
}

.cp-card-action {
  display: inline-flex;
  align-items: center;
  gap: 4px;
  font-size: 12px;
  font-weight: 500;
  color: var(--cp-ink-4);
  text-decoration: none;
  padding: 5px 10px;
  border-radius: 6px;
  border: 1px solid var(--cp-rule-strong);
  background: var(--cp-bg-card);
  transition: all 0.15s;
  box-shadow: var(--cp-shadow-sm);
  white-space: nowrap;
}

.cp-card-action:hover {
  background: var(--cp-bg-hover);
  color: var(--cp-ink-3);
}

.cp-card-action:focus-visible,
.cp-qa-item:focus-visible,
.cp-status-strip:focus-visible {
  outline: 2px solid var(--cp-accent);
  outline-offset: 2px;
}

.cp-table-wrap:focus-visible {
  outline: none;
  box-shadow: inset 0 0 0 2px var(--cp-accent);
}

@media (forced-colors: active) {
  .cp-table-wrap:focus-visible {
    outline: 2px solid Highlight;
    outline-offset: -2px;
    box-shadow: none;
  }
}

.cp-card-action svg,
.cp-qa-arr svg {
  width: 9px;
  height: 9px;
}

.cp-card-body { padding: 16px 20px; }
.cp-card-body--padded { padding: 16px 20px; }
.cp-card-body--flush { padding: 0; }

/* Tables provide their own cell padding, so remove wrapper padding when table is inside */
.cp-card-body > .cp-table-wrap:first-child,
.cp-card-body > .cp-table:first-child,
.cp-card > .cp-table-wrap,
.cp-card > .cp-table {
  margin: 0;
}
.cp-card-body:has(> .cp-table-wrap:only-child),
.cp-card-body:has(> .cp-table:only-child) {
  padding: 0;
}

/* Portal-specific class aliases and additions */
.cp-td-name { font-size: 13px; font-weight: 500; color: var(--cp-ink); display: block; }
.cp-td-sub { font-size: 11px; color: var(--cp-ink-4); display: block; }
a.cp-td-name { text-decoration: none; }
a.cp-td-name:hover { text-decoration: underline; }
.cp-stat-icon-wrap--accent { background: var(--cp-accent-bg); color: var(--cp-accent); }
.cp-stat-icon-wrap--info { background: var(--cp-blue-bg); color: var(--cp-blue); }
.cp-status-chip--danger { background: var(--cp-red-bg); color: var(--cp-red); border: 1px solid var(--cp-red-border); }
.cp-status-chip--success { background: var(--cp-green-bg); color: var(--cp-green); border: 1px solid rgba(30, 122, 69, 0.15); }
.cp-status-chip--muted { background: var(--cp-bg-inset); color: var(--cp-ink-4); border: 1px solid var(--cp-rule-strong); }
.cp-status-chip--warning { background: var(--cp-accent-bg); color: var(--cp-accent); border: 1px solid var(--cp-accent-border); }
.cp-card-title-group { }
.cp-field-hint { font-size: 11.5px; color: var(--cp-ink-4); margin-top: 2px; }
.cp-form-row { display: flex; gap: 10px; align-items: flex-end; }
.cp-form-row--2col { display: grid; grid-template-columns: repeat(2, 1fr); gap: 14px; }
.cp-floorplan__meta { margin-top: 6px; }
.cp-invoice-actions { padding: 16px 20px; border-top: 1px solid var(--cp-rule); display: flex; flex-direction: column; gap: 8px; }
.cp-pay-hero {
  display: grid;
  grid-template-columns: 1fr auto;
  align-items: center;
  gap: 16px;
  padding: 18px 20px;
  border: 1px solid var(--cp-accent-border);
  border-radius: 12px;
  background: var(--cp-accent-bg);
  box-shadow: var(--cp-shadow-sm);
}
.cp-pay-hero__body { display: flex; flex-direction: column; gap: 4px; }
.cp-pay-hero__eyebrow { margin: 0; font-size: 10px; font-weight: 700; letter-spacing: 0.1em; text-transform: uppercase; color: var(--cp-accent); }
.cp-pay-hero__title { margin: 0; font-family: var(--font-display, inherit); font-size: 22px; font-weight: 600; color: var(--cp-ink); display: inline-flex; align-items: center; gap: 10px; flex-wrap: wrap; }
.cp-pay-hero__sub { margin: 0; font-size: 12.5px; color: var(--cp-ink-3); }
.cp-pay-hero__actions { display: flex; justify-content: flex-end; }
.cp-pay-hero__actions form { margin: 0; }
.cp-pay-hero__btn { display: inline-flex; align-items: center; gap: 8px; min-height: 42px; padding: 10px 18px; font-size: 14px; font-weight: 600; white-space: nowrap; }
.cp-pay-hero__warning { grid-column: 1 / -1; margin: 4px 0 0; padding: 10px 12px; border-radius: 8px; background: var(--cp-red-bg); color: var(--cp-red); font-size: 12px; display: flex; align-items: flex-start; gap: 8px; border: 1px solid var(--cp-red-border); }
.cp-pay-hero__warning svg { flex-shrink: 0; margin-top: 1px; }
@media (max-width: 900px) {
  .cp-pay-hero { grid-template-columns: 1fr; padding: 16px; gap: 12px; }
  .cp-pay-hero__actions { justify-content: stretch; }
  .cp-pay-hero__actions form { width: 100%; }
  .cp-pay-hero__btn { width: 100%; justify-content: center; min-height: 46px; font-size: 15px; }
  .cp-pay-hero__title { font-size: 20px; }
}
.cp-search-item__title-row { display: flex; align-items: center; gap: 8px; flex-wrap: wrap; }
.cp-search-item__text { font-size: 13px; font-weight: 600; color: var(--cp-ink); }
.cp-search-item__subtext { font-size: 12px; color: var(--cp-ink-4); margin-top: 1px; }
.cp-security-note { font-size: 11.5px; color: var(--cp-ink-4); margin-top: 4px; }
.cp-evidence-list { list-style: none; padding: 0; margin: 4px 0 0; font-size: 12px; }
.cp-evidence-list li { padding: 2px 0; }
.cp-evidence-list a { color: var(--cp-accent); text-decoration: none; }
.cp-evidence-list a:hover { text-decoration: underline; }
.cp-update-history { margin-top: 12px; }
.cp-update-list { list-style: none; padding: 0; margin: 0; }
.cp-update-list li { padding: 8px 0; border-bottom: 1px solid var(--cp-rule); font-size: 13px; }
.cp-update-list li:last-child { border-bottom: none; }
.cp-auth-section__list { list-style: none; padding: 0; margin: 0 0 12px; font-size: 13px; color: var(--cp-ink-3); }
.cp-auth-section__list li { padding: 3px 0; }
.cp-auth-section__list li::before { content: "\2022"; margin-right: 6px; color: var(--cp-ink-4); }
.cp-field--checkbox { flex-direction: row; align-items: center; gap: 8px; }
.cp-field--checkbox .cp-field-label { text-transform: none; letter-spacing: 0; font-size: 13px; font-weight: 400; color: var(--cp-ink-2); }

.cp-fu-card {
  background: var(--cp-bg);
  border: 1px solid var(--cp-red-border);
  border-radius: 8px;
  padding: 12px;
}

.cp-fu-address {
  font-size: 12.5px;
  font-weight: 600;
  color: var(--cp-ink);
  margin-bottom: 2px;
}

.cp-fu-sub {
  font-size: 11px;
  color: var(--cp-ink-4);
  margin-bottom: 8px;
  line-height: 1.4;
}

.cp-fu-tags {
  display: flex;
  flex-wrap: wrap;
  gap: 4px;
  margin-bottom: 10px;
}

.cp-fu-tag {
  font-size: 10px;
  font-weight: 500;
  padding: 2px 7px;
  border-radius: 4px;
  white-space: nowrap;
}

.cp-fu-tag--overdue {
  background: var(--cp-red-bg);
  color: var(--cp-red);
  border: 1px solid var(--cp-red-border);
}

.cp-fu-tag--sched {
  background: var(--cp-blue-bg);
  color: var(--cp-blue);
  border: 1px solid rgba(37, 99, 168, 0.15);
}

.cp-fu-tag--active {
  background: var(--cp-active-bg);
  color: var(--cp-active-ink);
  border: 1px solid var(--cp-active-border);
}

.cp-fu-tag--done {
  background: var(--cp-green-bg);
  color: var(--cp-green);
  border: 1px solid rgba(30, 122, 69, 0.15);
}

.cp-fu-tag--cancelled {
  background: var(--cp-bg-inset);
  color: var(--cp-ink-3);
  border: 1px solid var(--cp-rule-strong);
}

.cp-fu-tag--id {
  background: var(--cp-bg-inset);
  color: var(--cp-ink-4);
  border: 1px solid var(--cp-rule-strong);
}

.cp-fu-btn {
  width: 100%;
  justify-content: center;
}

.cp-fu-btn:hover {
  text-decoration: none;
}

.cp-table {
  width: 100%;
  border-collapse: collapse;
}

.cp-table thead th {
  padding: 10px 20px;
  text-align: left;
  font-size: 10px;
  font-weight: 600;
  letter-spacing: 0.08em;
  text-transform: uppercase;
  color: var(--cp-ink-4);
  border-bottom: 1px solid var(--cp-rule);
  white-space: nowrap;
}

.cp-table tbody tr {
  border-bottom: 1px solid var(--cp-rule);
  transition: background 0.1s;
}

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

/* Only rows that actually contain a link get the clickable affordance —
 * hover bg + pointer cursor. Rows without any <a href> (e.g. invoice
 * line items) stay visually inert so users aren't led to click something
 * that doesn't respond. Uses :has() — supported in all evergreen browsers
 * since 2023. */
.cp-table tbody tr:has(a[href]) {
  cursor: pointer;
}

.cp-table tbody tr:has(a[href]):hover {
  background: var(--cp-bg-hover);
}

.cp-table tbody tr:has(a[href]):focus-within {
  background: var(--cp-bg-hover);
  outline: 2px solid var(--cp-accent);
  outline-offset: -2px;
}

.cp-table tbody td {
  padding: 11px 20px;
  color: var(--cp-ink-3);
  font-size: 13px;
  font-weight: 300;
  vertical-align: middle;
}

.cp-td-prop-name {
  font-size: 13px;
  font-weight: 500;
  color: var(--cp-ink);
  display: block;
}

.cp-td-prop-sub {
  font-size: 11px;
  color: var(--cp-ink-4);
  display: block;
}

.cp-date-chip {
  display: inline-flex;
  padding: 3px 9px;
  border-radius: 5px;
  font-size: 11.5px;
  font-weight: 400;
  background: var(--cp-bg-inset);
  color: var(--cp-ink-3);
  border: 1px solid var(--cp-rule-strong);
  white-space: nowrap;
}

.cp-status-chip {
  display: inline-flex;
  align-items: center;
  gap: 5px;
  padding: 3px 9px;
  border-radius: 100px;
  font-size: 11.5px;
  font-weight: 500;
  white-space: nowrap;
}

.cp-status-chip--sched {
  background: var(--cp-blue-bg);
  color: var(--cp-blue);
  border: 1px solid rgba(37, 99, 168, 0.15);
}

.cp-status-chip--active {
  background: var(--cp-active-bg);
  color: var(--cp-active-ink);
  border: 1px solid var(--cp-active-border);
}

.cp-status-chip--overdue {
  background: var(--cp-red-bg);
  color: var(--cp-red);
  border: 1px solid var(--cp-red-border);
}

.cp-status-chip--done {
  background: var(--cp-green-bg);
  color: var(--cp-green);
  border: 1px solid rgba(30, 122, 69, 0.15);
}

.cp-status-chip--cancelled {
  background: var(--cp-bg-inset);
  color: var(--cp-ink-3);
  border: 1px solid var(--cp-rule-strong);
}

.cp-table-scroll-hint,
.cp-status-strip-hint {
  display: none;
}

.cp-status-dot {
  width: 5px;
  height: 5px;
  border-radius: 50%;
  background: currentColor;
}

.cp-client-pill {
  display: inline-flex;
  padding: 2px 9px;
  border-radius: 4px;
  font-size: 11.5px;
  background: rgba(26, 24, 21, 0.035);
  color: var(--cp-ink-3);
  border: 1px solid var(--cp-rule);
}

.cp-client-pill--unassigned {
  background: var(--cp-bg-inset);
  color: var(--cp-ink-4);
  border: 1px solid var(--cp-rule-strong);
}

.cp-empty {
  padding: 20px;
  color: var(--cp-ink-4);
}

.cp-qa-list {
  display: flex;
  flex-direction: column;
  padding: 0 0 4px;
}

.cp-qa-item {
  display: flex;
  align-items: center;
  gap: 12px;
  padding: 13px 20px;
  border-bottom: 1px solid var(--cp-rule);
  text-decoration: none;
  transition: background 0.12s;
}

.cp-qa-item:last-child {
  border-bottom: none;
}

.cp-qa-item:hover {
  background: var(--cp-bg-hover);
}

.cp-qa-item:focus-visible {
  position: relative;
  z-index: 1;
  background: var(--cp-bg-hover);
}

.cp-qa-icon {
  width: 34px;
  height: 34px;
  flex-shrink: 0;
  border-radius: 8px;
  border: 1px solid var(--cp-rule-strong);
  background: var(--cp-bg);
  display: flex;
  align-items: center;
  justify-content: center;
  transition: all 0.15s;
  color: var(--cp-ink-3);
}

.cp-qa-item:hover .cp-qa-icon {
  border-color: var(--cp-accent-border);
  background: var(--cp-accent-bg);
  color: var(--cp-accent);
}

.cp-qa-item:focus-visible .cp-qa-icon {
  border-color: var(--cp-accent-border);
  background: var(--cp-accent-bg);
  color: var(--cp-accent);
}

.cp-qa-icon svg {
  width: 15px;
  height: 15px;
}

.cp-qa-body {
  flex: 1;
  min-width: 0;
}

.cp-qa-title {
  font-size: 13px;
  font-weight: 500;
  color: var(--cp-ink);
  margin-bottom: 1px;
}

.cp-qa-sub {
  font-size: 11.5px;
  font-weight: 300;
  color: var(--cp-ink-4);
}

.cp-qa-arr {
  width: 14px;
  height: 14px;
  color: var(--cp-ink-4);
  flex-shrink: 0;
  transition: color 0.15s, transform 0.15s;
}

.cp-qa-item:hover .cp-qa-arr {
  color: var(--cp-accent);
  transform: translateX(2px);
}

.cp-team-list {
  padding: 4px 16px 16px;
  display: flex;
  flex-direction: column;
  gap: 2px;
}

.cp-team-member {
  display: flex;
  align-items: center;
  gap: 10px;
  padding: 7px 8px;
  border-radius: 7px;
  transition: background 0.12s;
}

.cp-team-member:hover {
  background: var(--cp-bg-hover);
}

.cp-team-avatar {
  width: 28px;
  height: 28px;
  border-radius: 50%;
  display: flex;
  align-items: center;
  justify-content: center;
  font-size: 10px;
  font-weight: 700;
  color: #fff;
  flex-shrink: 0;
}

.cp-team-avatar--gold {
  background: #c49510;
}

.cp-team-avatar--blue {
  background: #2563a8;
}

.cp-team-avatar--green {
  background: #1e7a45;
}

.cp-team-avatar--stone {
  background: #9a9188;
}

.cp-team-name {
  font-size: 12.5px;
  font-weight: 500;
  color: var(--cp-ink-2);
  flex: 1;
  min-width: 0;
}

.cp-team-role {
  font-size: 11px;
  color: var(--cp-ink-4);
  white-space: nowrap;
}

.cp-team-status {
  display: inline-flex;
  align-items: center;
  gap: 5px;
  font-size: 11px;
  font-weight: 500;
  white-space: nowrap;
  flex-shrink: 0;
  padding: 2px 8px;
  border-radius: 999px;
  border: 1px solid var(--cp-rule-strong);
  background: var(--cp-bg-inset);
  color: var(--cp-ink-3);
}

.cp-team-status-text {
  color: var(--cp-ink-4);
}

.cp-team-dot {
  width: 7px;
  height: 7px;
  border-radius: 50%;
  flex-shrink: 0;
  background: currentColor;
  opacity: 0.65;
}

.cp-team-status--recent {
  background: rgba(26, 24, 21, 0.035);
  color: var(--cp-ink-2);
  border-color: rgba(26, 24, 21, 0.08);
}

.cp-team-status--recent .cp-team-dot {
  opacity: 0.75;
}

.cp-team-status--stale {
  color: var(--cp-ink-3);
}

.cp-team-status--unknown,
.cp-team-status--inactive {
  color: var(--cp-ink-4);
  opacity: 0.92;
}

.cp-team-status--inactive .cp-team-dot,
.cp-team-status--unknown .cp-team-dot,
.cp-team-status--stale .cp-team-dot {
  opacity: 0.5;
}

.cp-team-empty {
  padding: 12px 8px 4px;
  color: var(--cp-ink-4);
  font-size: 12px;
}

/* 2026-04-23: the strip now always carries 4 cards (3 KPIs + Contact), so
   we keep 4 columns down to the 900px mobile carousel instead of downshifting
   to 3 at 1366px (which would wrap the Contact card onto a second row). */

@media (max-width: 1024px) {
  .cp-table-wrap {
    overflow-x: auto;
  }

  .cp-table-scroll-hint {
    display: block;
    margin: 0 0 10px;
    font-size: 11px;
    color: var(--cp-ink-4);
  }
}

@media (max-width: 900px) {
  .cp-page-header {
    padding-top: 18px;
    flex-direction: column;
    align-items: stretch;
  }

  .cp-page-header__right {
    justify-content: flex-start;
  }

  .cp-status-strip {
    display: flex;
    gap: 12px;
    padding: 18px 0 20px;
    overflow-x: auto;
    overflow-y: hidden;
    -webkit-overflow-scrolling: touch;
    scrollbar-width: thin;
    scrollbar-color: var(--cp-rule-strong) transparent;
    scroll-snap-type: x proximity;
  }

  .cp-status-strip::-webkit-scrollbar {
    height: 7px;
  }

  .cp-status-strip::-webkit-scrollbar-thumb {
    background: var(--cp-rule-strong);
    border-radius: 999px;
  }

  .cp-status-strip::-webkit-scrollbar-track {
    background: transparent;
  }

  .cp-status-strip .cp-stat-card {
    flex: 0 0 min(240px, 76vw);
    scroll-snap-align: start;
  }

  .cp-status-strip-hint {
    display: block;
    margin: -8px 0 18px;
    font-size: 11px;
    color: var(--cp-ink-4);
  }

  .cp-content-grid {
    display: flex;
    flex-direction: column;
    align-items: stretch;
    gap: 14px;
    width: 100%;
    padding: 0;
  }

  /* Force all grid children + their card descendants to be full-width on mobile */
  .cp-content-grid > * {
    width: 100%;
    max-width: 100%;
    min-width: 0;
    box-sizing: border-box;
  }
  .cp-content-grid .cp-card {
    width: 100%;
    max-width: 100%;
    box-sizing: border-box;
  }

  /* Quick access column (right-hand sticky on desktop) comes first on mobile
     — matches tenant dashboard pattern: KPIs → quick actions → main content. */
  .cp-content-grid > :last-child {
    order: 1;
    position: static !important;
    top: auto;
  }
  .cp-grid-main {
    order: 2;
    width: 100%;
  }

  .cp-card--quick-actions {
    order: 1;
  }

  .cp-card--field-work {
    order: 2;
  }

  .cp-card--workspace {
    order: 3;
  }

  .cp-card--recent-inspections {
    order: 4;
  }

  .cp-card-header {
    flex-direction: column;
    align-items: flex-start;
  }

  /* Table mobile behaviour is owned by the comprehensive block below
     (card-ification with data-label pseudo-labels). Old conflicting rules
     that un-card-ified tables with display: table-* + min-width: 640px have
     been removed 2026-04-17 — they fought the new card-ification and created
     horizontal scroll. */

  .cp-team-member {
    flex-wrap: wrap;
    align-items: center;
  }

  .cp-team-name {
    flex: 1 1 auto;
    min-width: 0;
  }
}

@media (max-width: 900px) {
  .cp-portal .trial-banner--actions {
    flex-direction: column;
    align-items: stretch;
    position: static;
  }

  .cp-portal .trial-banner__content {
    min-width: 0;
    width: 100%;
  }

  .cp-portal .trial-banner__actions {
    width: 100%;
    justify-content: flex-start;
  }

  .cp-portal .trial-banner__actions form {
    display: block;
    width: 100%;
  }

  .cp-portal .trial-banner__actions .btn,
  .cp-portal .trial-banner__actions > .btn {
    width: 100%;
    justify-content: center;
  }
}

/* --- Inspector subtitle --- */
.cp-page-sub {
  font-size: 13px;
  color: var(--cp-ink-4);
  margin: 4px 0 0;
}

/* --- Clickable stat cards (inspector links) --- */
a.cp-stat-card {
  text-decoration: none;
  color: inherit;
  cursor: pointer;
  transition: box-shadow 0.15s ease, border-color 0.15s ease;
}

a.cp-stat-card:hover {
  box-shadow: var(--cp-shadow-md);
  border-color: var(--cp-rule-strong);
}

/* --- Inspector mobile card order: field work first --- */
@media (max-width: 900px) {
  .cp-content-grid--compact .cp-card--field-work { order: 1; }
  .cp-content-grid--compact .cp-card--quick-actions { order: 2; }
  .cp-content-grid--compact .cp-card--recent-inspections { order: 3; }
}

/* --- Dark mode: mobile table cards (moved from 600px → 900px 2026-04-17) --- */
@media (max-width: 900px) {
  html[data-theme="dark"] .cp-table tbody tr {
    background: var(--card);
    border-color: var(--border);
  }
  html[data-theme="dark"] .cp-table tbody td {
    border: none;
    color: var(--text);
  }
  html[data-theme="dark"] a.cp-stat-card:hover {
    border-color: var(--border-strong, var(--border));
  }
  html[data-theme="dark"] .cp-sidebar {
    box-shadow: 4px 0 24px rgba(0, 0, 0, 0.4);
  }
  html[data-theme="dark"] .cp-nav-overlay {
    background: rgba(2, 6, 23, 0.62);
  }
}

/* --- Dark mode: dashboard table cards on mobile --- */
@media (max-width: 900px) {
  html[data-theme="dark"] .cp-table-wrap {
    scrollbar-color: var(--border) transparent;
  }

  html[data-theme="dark"] .cp-status-strip {
    scrollbar-color: var(--border-strong, var(--border)) transparent;
  }

  html[data-theme="dark"] .cp-table tbody td {
    border-bottom-color: rgba(71, 85, 105, 0.4);
    color: var(--text);
  }
}

/* ==========================================================================
   PORTAL-SPECIFIC ADDITIONS (not in dashboard.css)
   ========================================================================== */

/* --- Page background --- */
.page-section.client-portal { background: #f7f5f0; padding: 0; }

/* --- Reset global form + input styling from style.css --- */
.cp-portal form,
.cp-auth form {
  background: transparent;
  padding: 0;
  border-radius: 0;
  display: block;
  gap: 0;
}

.cp-portal input:not([type="checkbox"]):not([type="radio"]):not([type="hidden"]),
.cp-portal select,
.cp-portal textarea,
.cp-auth input:not([type="checkbox"]):not([type="radio"]):not([type="hidden"]),
.cp-auth select,
.cp-auth textarea {
  width: auto;
  border: 1px solid var(--cp-rule-strong);
  border-radius: 8px;
  padding: 10px 13px;
  font-size: 13.5px;
  background: var(--cp-bg-card);
  color: var(--cp-ink);
  font-family: inherit;
}

/* --- Shell: grid layout matching tenant app-shell --- */
.cp-shell {
  display: grid;
  grid-template-rows: auto 1fr;
  grid-template-columns: 56px minmax(0, 1fr);
  min-height: 100vh;
}

/* Nav spans full width across both columns */
.cp-nav {
  grid-column: 1 / -1;
  grid-row: 1;
}

/* Sidebar fills left column, full height */
.cp-sidebar {
  grid-column: 1;
  grid-row: 2;
}

/* Main content fills right column */
.cp-main {
  grid-column: 2;
  grid-row: 2;
  min-width: 0;
  /* Bottom padding gives the last section room to breathe instead of sitting
     flush with the viewport edge. Fixed 2026-04-23 Phase 10 QA finding. */
  padding: 0 24px 48px 18px;
  /* Flex column with a fixed gap owns the rhythm between direct children
     (page header, stats strip, flash, tabs, filter shell, cards). Previously
     each element tried to carry its own margin/padding which produced
     inconsistent spacing between adjacent sections (or no spacing at all
     when two cards stacked). Fixed 2026-04-23 Phase 10 QA finding. */
  display: flex;
  flex-direction: column;
  gap: 16px;
  /* 2026-04-24: opacity-only fade-in. The previous `transform: translateY`
     entrance created a new containing block on `.cp-main` (any non-`none`
     transform does), which reparented every `position: fixed` descendant
     to `.cp-main` instead of the viewport. On mobile `.cp-main` is taller
     than the viewport (header + stats + cards stack vertically), so the
     filter drawer's `inset: 0` landed at the bottom of the page column
     and users had to scroll down to see it. Dropping transform fixes
     the drawer without losing the entrance feel. */
  opacity: 0; animation: cp-fade-in 220ms ease-out 40ms forwards;
}
@keyframes cp-fade-in { from { opacity: 0; } to { opacity: 1; } }
@media (prefers-reduced-motion: reduce) { .cp-main { animation: none !important; opacity: 1 !important; } }

/* --- Sidebar --- */
.cp-sidebar {
  padding: 12px 6px 20px;
  background: var(--cp-bg-card);
  border-right: 1px solid var(--cp-rule);
  display: flex; flex-direction: column;
  align-items: center; gap: 8px;
  overflow-y: auto; overflow-x: hidden;
}
.cp-sidebar__spacer { flex: 1; min-height: 12px; }
.cp-sidebar__link {
  width: 32px; height: 32px; border-radius: 999px;
  display: flex; align-items: center; justify-content: center;
  color: var(--cp-ink-4); text-decoration: none; background: transparent;
  transition: background 0.12s, color 0.12s; position: relative;
}
.cp-sidebar__link:hover { background: var(--cp-bg-hover); }
.cp-sidebar__link:focus-visible { outline: 2px solid var(--cp-accent); outline-offset: 2px; }
/* 2026-04-23 Phase 10 QA: active state softened from a bold `var(--cp-ink)`
   (near-black) pill with white text. The old treatment read as jarring on
   the mobile drawer rows where the pill stretches full-width. Replaced
   with a soft accent-tinted background + accent text; reads clearly at
   both the 32px desktop rail and the full-width mobile drawer row. */
.cp-sidebar__link.is-active { background: var(--cp-accent-bg); color: var(--cp-accent); }
.cp-sidebar__link svg { width: 18px; height: 18px; }
/* 2026-04-22 rebuild: 4 primary items. Accent colour per destination keeps the
   rail scannable at 32px desktop width. */
.cp-sidebar__link--properties { color: var(--cp-green); }
.cp-sidebar__link--inspections { color: var(--cp-blue); }
.cp-sidebar__link--billing { color: var(--cp-accent); }
.cp-sidebar__link--documents { color: var(--cp-accent); }
.cp-sidebar__link--settings { color: var(--cp-blue); }
/* Force accent colour on active regardless of per-item base colour. Source
   order matters: this must come AFTER the per-item colour rules above. */
.cp-sidebar__link.is-active { color: var(--cp-accent); }
.cp-sidebar__tip {
  position: absolute; left: 44px; top: 50%; transform: translateY(-50%);
  padding: 4px 10px; border-radius: 999px; background: var(--cp-ink); color: #fff;
  font-size: 12px; white-space: nowrap; box-shadow: var(--cp-shadow-md);
  display: none; pointer-events: none;
}
.cp-sidebar__link:hover .cp-sidebar__tip { display: block; }

/* Drawer-only shortcuts (Notifications / Help / Sign out) — desktop hides these,
   the topbar `.cp-nav__btn` cluster owns those actions at ≥901px. */
.cp-sidebar__link--mobile-only { display: none; }
.cp-sidebar__badge { display: none; }
/* Neutralise the global `form { background; padding; border-radius; display: grid }`
   set in style.css for the logout form in the drawer. */
.cp-sidebar__logout {
  display: none; margin: 0; padding: 0; gap: 0;
  background: transparent; border-radius: 0;
}
.cp-sidebar__link--logout {
  width: 100%; background: transparent; border: 0; cursor: pointer;
  font: inherit; color: inherit; text-align: left;
}

/* --- Mobile nav elements (desktop: hidden) --- */
.cp-nav__hamburger {
  display: none;
  align-items: center; justify-content: center;
  width: 40px; height: 40px;
  background: transparent; border: 1px solid transparent;
  border-radius: 9px; color: var(--cp-ink);
  cursor: pointer; padding: 0;
  flex-shrink: 0;
}
.cp-nav__hamburger:hover { background: var(--cp-bg-hover); }
.cp-nav__hamburger:focus-visible { outline: 2px solid var(--cp-accent); outline-offset: 2px; }
.cp-sidebar__close {
  display: none;
  align-items: center; justify-content: center;
  width: 36px; height: 36px;
  background: transparent; border: 1px solid var(--cp-rule);
  border-radius: 9px; color: var(--cp-ink);
  cursor: pointer; padding: 0;
  align-self: flex-end;
  margin-bottom: 8px;
}
.cp-sidebar__close:hover { background: var(--cp-bg-hover); }
.cp-sidebar__label {
  display: none;
  font-size: 14px; font-weight: 500;
  color: var(--cp-ink-2);
}
.cp-nav-overlay {
  display: none;
  position: fixed; inset: 0;
  background: rgba(26, 24, 21, 0.42);
  z-index: 2110;
}

/* --- Navigation bar --- */
.cp-nav {
  display: flex; align-items: center; justify-content: space-between; gap: 12px;
  padding: 10px 20px; height: 56px;
  background: var(--cp-bg-card); border-bottom: 1px solid var(--cp-rule);
}
.cp-nav__brand { display: flex; align-items: center; gap: 10px; min-width: 0; }
.cp-nav__logo { max-height: 30px; width: auto; border-radius: 4px; }
.cp-nav__brand .logo-placeholder--compact svg { height: 30px; width: auto; }
.cp-nav__name { font-size: 13px; font-weight: 600; color: var(--cp-ink); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
.cp-nav__badge { font-size: 9px; font-weight: 600; text-transform: uppercase; letter-spacing: 0.12em; color: var(--cp-ink-4); display: block; }
.cp-nav__tabs {
  display: flex; align-items: center; gap: 2px; padding: 3px;
  background: var(--cp-bg-inset); border: 1px solid var(--cp-rule-strong); border-radius: 8px;
}
.cp-nav__tab {
  padding: 6px 14px; border-radius: 6px; font-size: 12.5px; font-weight: 500;
  color: var(--cp-ink-4); text-decoration: none; white-space: nowrap;
  transition: background 0.12s, color 0.12s;
}
.cp-nav__tab:hover { background: var(--cp-bg-card); color: var(--cp-ink-3); }
.cp-nav__tab.is-active { background: var(--cp-bg-card); color: var(--cp-ink); font-weight: 600; box-shadow: var(--cp-shadow-sm); }
.cp-nav__actions { display: flex; align-items: center; gap: 8px; }
.cp-nav__search-form { margin: 0; }
.cp-nav__search { position: relative; display: inline-block; }
.cp-nav__search-input.cp-nav__search-input {
  padding: 7px 12px 7px 32px !important; border-radius: 8px;
  border: 1px solid var(--cp-rule-strong); background: var(--cp-bg);
  font-size: 12.5px; color: var(--cp-ink); width: 200px;
  box-sizing: border-box;
}
.cp-nav__search-input::placeholder { color: var(--cp-ink-4); }
.cp-nav__search-input:focus { outline: none; border-color: var(--cp-accent); box-shadow: 0 0 0 3px var(--cp-accent-bg); }
.cp-nav__search-icon { position: absolute; left: 10px; top: 50%; transform: translateY(-50%); width: 14px; height: 14px; color: var(--cp-ink-4); pointer-events: none; z-index: 1; }
.cp-nav__btn {
  width: 32px; height: 32px; border-radius: 8px;
  border: 1px solid var(--cp-rule-strong); background: var(--cp-bg-card);
  color: var(--cp-ink-4); cursor: pointer; display: flex; align-items: center; justify-content: center;
  transition: background 0.12s, color 0.12s;
}
.cp-nav__btn:hover { background: var(--cp-bg-hover); color: var(--cp-ink-3); }
/* 2026-04-23 Phase 10 QA: match the sidebar active-state softening — the
   old bold `--cp-ink` + white pill was jarring, especially on mobile. */
.cp-nav__btn.is-active { background: var(--cp-accent-bg); color: var(--cp-accent); border-color: var(--cp-accent-border); }
.cp-nav__btn svg { width: 15px; height: 15px; }
.cp-nav__logout { margin: 0; display: inline; }
.cp-nav__logout .btn { font-size: 12px; padding: 6px 10px; border-radius: 8px; border: 1px solid var(--cp-rule-strong); color: var(--cp-ink-4); background: var(--cp-bg-card); }
.cp-nav__logout .btn:hover { background: var(--cp-bg-hover); }

/* --- Tabs (pill bar) --- */
/* 2026-04-23 Phase 10 QA: gap / padding / border-radius aligned to the
   tenant canonical `.ipi-tabs` container so cross-app tab bars look
   identical. */
.cp-tabs {
  display: inline-flex; gap: 4px; padding: 4px;
  background: var(--cp-bg-inset); border: 1px solid var(--cp-rule-strong);
  border-radius: 9px; box-shadow: var(--cp-shadow-sm);
  /* 2026-04-23: margin-bottom dropped — `.cp-main` flex-gap owns rhythm. */
  /* 2026-04-24: `.cp-main` is display:flex; flex-direction:column which
     defaults children to align-items:stretch, so `inline-flex` alone
     still rendered the bar full-width. Opt out of the column stretch to
     match the tenant canonical `.ipi-tabs` (inspections_shell.css:308). */
  align-self: flex-start;
}
/* 2026-04-23 Phase 10 QA: sizing aligned to the tenant canonical
   `.ipi-tab` (padding 7px 16px, font-size 13px). Keeps tab rhythm
   consistent across the tenant dashboard + client portal. */
.cp-tab {
  padding: 7px 16px; border-radius: 6px; font-size: 13px; font-weight: 500;
  color: var(--cp-ink-4); cursor: pointer; text-decoration: none;
  border: none; background: none; transition: background 0.12s, color 0.12s;
}
.cp-tab:hover { background: var(--cp-bg-card); color: var(--cp-ink-3); }
.cp-tab.is-active, .cp-tab--active { background: var(--cp-bg-card); color: var(--cp-ink); font-weight: 600; box-shadow: var(--cp-shadow-sm); }

/* 2026-04-22 rebuild: count badges on primary tabs, used on the per-property
   hub to surface open-remedials / document counts without a second request. */
.cp-tab-badge {
  display: inline-flex; align-items: center; justify-content: center;
  min-width: 18px; height: 18px; padding: 0 6px; margin-left: 6px;
  border-radius: 999px; background: var(--cp-bg-inset); color: var(--cp-ink-3);
  font-size: 10.5px; font-weight: 600; line-height: 1;
}
.cp-tab-badge--danger { background: var(--cp-red-bg); color: var(--cp-red); }
.cp-tab--active .cp-tab-badge { background: var(--cp-bg-inset); color: var(--cp-ink-2); }

/* Secondary tab bar — used for in-tab filter switching (e.g. Remedials tab's
   Open / Completed / All). Smaller type + lighter background than primary. */
.cp-tabs--secondary {
  display: inline-flex; gap: 2px; padding: 3px;
  background: transparent; border: 1px solid var(--cp-rule);
  border-radius: 8px; box-shadow: none;
  /* 2026-04-23: margin-bottom dropped — `.cp-main` flex-gap owns rhythm. */
  /* 2026-04-24: sits at content width inside the `.cp-main` flex column
     — matches the primary `.cp-tabs` fix above. */
  align-self: flex-start;
}
.cp-tabs--secondary .cp-tab { font-size: 12px; padding: 5px 12px; }

/* Status chip extensions used by the per-property hub visit type column. */
.cp-status-chip--info { background: var(--cp-blue-bg); color: var(--cp-blue); border: 1px solid rgba(37, 99, 168, 0.2); }
.cp-status-chip--accent { background: var(--cp-accent-bg); color: var(--cp-active-ink); border: 1px solid var(--cp-accent-border); }

/* Table-cell text helpers. */
.cp-td-muted { color: var(--cp-ink-4); }
.cp-td-danger { color: var(--cp-red); }
.cp-td-actions { display: flex; gap: 6px; flex-wrap: wrap; justify-content: flex-end; }

/* 2026-04-23 Phase 10 QA: clickable table rows on Properties + Billing.
   The JS controller in cp-row-click.js listens at document level and
   navigates on plain clicks that aren't handled by a nested anchor,
   button, or form control. CSS adds the visual affordance:
   pointer cursor + subtle hover bg. */
.cp-table tbody tr[data-cp-row-href] {
  cursor: pointer;
  transition: background 0.12s;
}
.cp-table tbody tr[data-cp-row-href]:hover {
  background: var(--cp-bg-hover);
}

/* Inline content helpers — used in the inspection_show door register to stack
   component lists + issue detail panels under a single table cell. */
.cp-td-stack { margin-top: 6px; display: flex; flex-direction: column; gap: 4px; }
.cp-td-details { margin-top: 6px; }
.cp-td-details__summary {
  cursor: pointer; color: var(--cp-red); font-size: 11.5px;
  list-style: none;
}
.cp-td-details__summary::-webkit-details-marker { display: none; }
.cp-td-details__list { list-style: none; padding: 4px 0 0; margin: 0; display: flex; flex-direction: column; gap: 3px; }
.cp-td-issues { min-width: 180px; }

/* 2026-04-22 rebuild: Door-issue modal. Uses the canonical .modal--sheet
   pattern from style.css (full-screen on mobile, centred dialog on
   desktop). Portal-scoped tweaks to the dialog body padding + form layout
   so inputs feel native to the cp-* type scale. */
.cp-door-issue-modal .modal__dialog {
  max-width: min(560px, calc(100vw - 32px));
  background: var(--cp-bg-card);
  color: var(--cp-ink);
}
.cp-door-issue-modal .modal__header {
  padding: 18px 20px 12px;
  border-bottom: 1px solid var(--cp-rule);
}
.cp-door-issue-modal .modal__body {
  padding: 18px 20px 20px;
  background: var(--cp-bg);
  display: flex; flex-direction: column; gap: 14px;
}
.cp-door-issue-modal .modal__footer {
  padding: 12px 20px 14px;
  border-top: 1px solid var(--cp-rule);
  display: flex; justify-content: flex-end; gap: 8px;
}
.cp-door-issue-modal .modal__title {
  margin: 0; font-size: 17px; color: var(--cp-ink);
}
.cp-door-issue-modal .eyebrow {
  font-size: 10px; font-weight: 600; letter-spacing: 0.08em;
  text-transform: uppercase; color: var(--cp-accent);
  margin: 0 0 4px;
}
.cp-door-issue-modal .text-muted {
  margin: 6px 0 0; font-size: 12.5px; color: var(--cp-ink-3);
}
body.cp-door-issue-modal-open { overflow: hidden; }

html[data-theme="dark"] .cp-door-issue-modal .modal__dialog {
  background: var(--cp-bg-card);
}
html[data-theme="dark"] .cp-door-issue-modal .modal__body {
  background: var(--cp-bg);
}

/* --- Filters --- */
.cp-portal .cp-filters { display: flex; flex-wrap: wrap; gap: 8px; align-items: center; padding: 14px 20px; }
.cp-filter-pill {
  padding: 5px 12px; border-radius: 6px; border: 1px solid var(--cp-rule-strong);
  background: var(--cp-bg-card); font-size: 12px; font-weight: 500; color: var(--cp-ink-4);
  text-decoration: none; cursor: pointer;
}
.cp-filter-pill:hover { background: var(--cp-bg-hover); }
.cp-filter-pill.is-active { background: var(--cp-ink); color: #fff; border-color: var(--cp-ink); }
.cp-filter-input {
  padding: 7px 12px; border-radius: 8px; border: 1px solid var(--cp-rule-strong);
  background: var(--cp-bg); font-size: 12.5px; color: var(--cp-ink); min-width: 180px; font-family: inherit;
}
.cp-filter-input:focus { outline: none; border-color: var(--cp-accent); box-shadow: 0 0 0 3px var(--cp-accent-bg); }
.cp-filter-select {
  padding: 7px 12px; border-radius: 8px; border: 1px solid var(--cp-rule-strong);
  background: var(--cp-bg-card); font-size: 12.5px; color: var(--cp-ink); min-width: 140px; font-family: inherit;
}
.cp-filter-select:focus { outline: none; border-color: var(--cp-accent); box-shadow: 0 0 0 3px var(--cp-accent-bg); }

/* Filter-shell mobile pattern: the rail + drawer ship in the DOM on every
   page that wraps its filters in .cp-filter-shell, but they're a mobile-only
   affordance. Hide on desktop so the inline .cp-filters form owns filtering
   at ≥901px. The mobile @media block below flips these back on with their
   own display values (flex for the rail, none/flex toggle for the drawer
   via .is-open). Without these base rules the rail + drawer render as
   default unstyled block elements on desktop — confirmed via Sidekick
   2026-04-23. */
.cp-filter-shell .cp-filter-rail,
.cp-filter-shell .cp-filters-drawer {
  display: none;
}

/* --- Aside cards --- */
.cp-aside-card {
  background: var(--cp-bg-card); border-radius: 10px; padding: 14px 16px; box-shadow: var(--cp-shadow-card);
}
.cp-aside-card + .cp-aside-card { margin-top: 12px; }
.cp-aside-card__label { font-size: 10px; font-weight: 600; letter-spacing: 0.08em; text-transform: uppercase; color: var(--cp-ink-4); margin-bottom: 8px; }
.cp-aside-card__row { display: flex; justify-content: space-between; gap: 8px; font-size: 12.5px; padding: 4px 0; color: var(--cp-ink-3); }
.cp-aside-card__key { font-size: 11px; font-weight: 500; color: var(--cp-ink-4); white-space: nowrap; }
.cp-aside-card__val { font-size: 12.5px; color: var(--cp-ink); text-align: right; word-break: break-word; }
.cp-aside-card__row strong { color: var(--cp-ink); font-weight: 500; }

/* --- Flash messages --- */
.cp-flash { padding: 10px 14px; border-radius: 8px; font-size: 13px; /* 2026-04-23: margin-bottom dropped — .cp-main flex-gap owns rhythm */ }
.cp-flash--success { background: var(--cp-green-bg); color: var(--cp-green); border: 1px solid rgba(30, 122, 69, 0.15); }
.cp-flash--danger, .cp-flash--error { background: var(--cp-red-bg); color: var(--cp-red); border: 1px solid var(--cp-red-border); }
.cp-flash--warning { background: var(--cp-accent-bg); color: var(--cp-accent); border: 1px solid var(--cp-accent-border); }
.cp-flash--info { background: var(--cp-blue-bg); color: var(--cp-blue); border: 1px solid rgba(37, 99, 168, 0.15); }

/* --- Forms --- */
.cp-portal .cp-form,
.cp-auth .cp-form { display: flex; flex-direction: column; gap: 14px; }
.cp-form-grid { display: grid; grid-template-columns: repeat(2, 1fr); gap: 14px; }
.cp-form-actions { display: flex; gap: 8px; justify-content: flex-end; padding-top: 6px; }
.cp-field { display: flex; flex-direction: column; gap: 5px; }
.cp-field-label { font-size: 10px; font-weight: 600; letter-spacing: 0.08em; text-transform: uppercase; color: var(--cp-ink-4); }
.cp-input {
  min-height: 40px; padding: 10px 13px; border-radius: 8px;
  border: 1px solid var(--cp-rule-strong); background: var(--cp-bg-card);
  font-size: 13.5px; color: var(--cp-ink); font-family: inherit; width: 100%; box-sizing: border-box;
}
.cp-input:focus { outline: none; border-color: var(--cp-accent); box-shadow: 0 0 0 3px var(--cp-accent-bg); }
.cp-checkbox-group { display: flex; flex-direction: column; gap: 8px; margin-top: 6px; }
.cp-checkbox-label { display: flex; align-items: center; gap: 8px; font-size: 13px; color: var(--cp-ink-2); cursor: pointer; }
.cp-checkbox-label input[type="checkbox"] { width: 16px; height: 16px; accent-color: var(--cp-accent); }

/* --- Auth pages (centred card) --- */
.cp-auth { min-height: 80vh; display: flex; align-items: center; justify-content: center; padding: 32px 16px; }
.cp-auth-card { width: 100%; max-width: 420px; background: var(--cp-bg-card); border-radius: 16px; box-shadow: var(--cp-shadow-md); padding: 32px 28px; position: relative; }
.cp-auth-card__logo { display: block; max-height: 48px; width: auto; margin: 0 auto 12px; border-radius: 6px; }
.cp-auth-card__logo-placeholder { display: flex; justify-content: center; margin: 0 auto 12px; }
.cp-auth-card__logo-placeholder .logo-placeholder--default svg { height: 48px; width: auto; }
.cp-auth-card__workspace { text-align: center; font-size: 14px; font-weight: 600; color: var(--cp-ink); margin: 0 0 4px; }
.cp-auth-card__badge { text-align: center; font-size: 9px; font-weight: 600; text-transform: uppercase; letter-spacing: 0.12em; color: var(--cp-ink-4); margin: 0 0 20px; display: block; }
.cp-auth-card h2 { font-family: var(--font-display); font-size: 20px; font-weight: 700; color: var(--cp-ink); text-align: center; margin: 0 0 6px; }
.cp-auth-card__sub { text-align: center; font-size: 13px; color: var(--cp-ink-3); margin: 0 0 20px; }
/* 2026-04-22 Phase 7/9 — replaces inline `style="margin-top:Npx"` on the
   auth-card sub text + the form siblings below submit buttons. One spacing
   value covers all auth-card rhythm because the original values (8px, 12px,
   16px) were inconsistent by accident, not design. */
.cp-auth-card__sub--spaced { margin: 12px 0 0; }
.cp-auth-form--spaced { margin-top: 12px; }
.cp-auth-help { position: absolute; top: 12px; right: 12px; width: 28px; height: 28px; border-radius: 6px; border: 1px solid var(--cp-rule-strong); background: var(--cp-bg-card); display: flex; align-items: center; justify-content: center; cursor: pointer; color: var(--cp-ink-4); font-size: 13px; }
.cp-auth-help:hover { background: var(--cp-bg-hover); }
.cp-auth-footer { text-align: center; font-size: 12px; color: var(--cp-ink-4); margin-top: 16px; }
.cp-auth-footer a { color: var(--cp-accent); text-decoration: none; }
.cp-auth-footer a:hover { text-decoration: underline; }
.cp-auth-footer__legal { font-size: 11px; color: var(--cp-ink-4); margin-top: 8px; }
.cp-auth-footer__legal a { color: var(--cp-accent); }
.cp-auth-section { margin-top: 16px; padding-top: 16px; border-top: 1px solid var(--cp-rule); }
.cp-auth-section h3 { font-size: 14px; font-weight: 600; color: var(--cp-ink); margin: 0 0 8px; }
.cp-auth-section p { font-size: 13px; color: var(--cp-ink-3); margin: 0 0 10px; }
.cp-auth-qr { text-align: center; margin: 12px 0; }
.cp-auth-qr img { max-width: 200px; border-radius: 8px; }
.cp-auth-qr code { display: block; margin-top: 8px; font-size: 12px; color: var(--cp-ink-3); word-break: break-all; }
.cp-account-list { border: none; padding: 0; margin: 0; display: grid; gap: 8px; }
.cp-account-row { display: flex; align-items: center; gap: 10px; padding: 10px 14px; border-radius: 8px; border: 1px solid var(--cp-rule-strong); cursor: pointer; transition: background 0.12s; }
.cp-account-row:hover { background: var(--cp-bg-hover); }
.cp-account-row__meta { display: grid; gap: 1px; min-width: 0; }
.cp-account-row__name { font-size: 13px; font-weight: 600; color: var(--cp-ink); }
.cp-account-row__sub { font-size: 11.5px; color: var(--cp-ink-4); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
.cp-password-wrap { position: relative; }
.cp-password-toggle { position: absolute; right: 8px; top: 50%; transform: translateY(-50%); background: none; border: none; padding: 4px; cursor: pointer; color: var(--cp-ink-4); border-radius: 4px; }
.cp-password-toggle:hover { color: var(--cp-ink-2); }
.cp-password-strength { margin-top: 4px; height: 3px; border-radius: 999px; background: var(--cp-bg-inset); overflow: hidden; }
.cp-password-strength__bar { height: 100%; border-radius: 999px; transition: width 0.2s, background 0.2s; width: 0; }
.cp-password-strength.is-weak .cp-password-strength__bar { width: 33%; background: var(--cp-red); }
.cp-password-strength.is-ok .cp-password-strength__bar { width: 66%; background: var(--cp-accent); }
.cp-password-strength.is-strong .cp-password-strength__bar { width: 100%; background: var(--cp-green); }
.cp-password-match { font-size: 12px; margin: 4px 0 0; }
.cp-password-match.is-bad { color: var(--cp-red); }
.cp-password-match.is-good { color: var(--cp-green); }
.cp-password-caps { font-size: 12px; color: var(--cp-accent); margin: 4px 0 0; }

/* --- Pagination --- */
.cp-pagination { display: flex; align-items: center; justify-content: space-between; gap: 10px; padding: 12px 20px; border-top: 1px solid var(--cp-rule); font-size: 12px; color: var(--cp-ink-4); }
.cp-pagination__btns { display: flex; gap: 6px; }

/* --- Notifications dropdown --- */
[data-client-notifications-dropdown].notifications-modal { position: fixed; inset: 0; display: flex; align-items: flex-start; justify-content: flex-end; z-index: 2135; }
[data-client-notifications-dropdown].notifications-modal[hidden] { display: none !important; }
[data-client-notifications-dropdown].notifications-modal .notifications-modal__backdrop { position: absolute; inset: 0; background: rgba(26, 24, 21, 0.32); }
[data-client-notifications-dropdown].notifications-modal .notifications-modal__panel { position: relative; margin: 66px 24px 24px; max-width: 360px; width: 100%; background: var(--cp-bg-card); border-radius: 12px; box-shadow: var(--cp-shadow-md); padding: 14px; display: flex; flex-direction: column; }
[data-client-notifications-dropdown].notifications-modal .notifications-header { display: flex; align-items: center; justify-content: space-between; gap: 8px; margin-bottom: 6px; }
[data-client-notifications-dropdown].notifications-modal .notifications-header h3 { margin: 0; font-size: 14px; color: var(--cp-ink); }
[data-client-notifications-dropdown].notifications-modal .notifications-header__actions { display: inline-flex; align-items: center; gap: 4px; }
[data-client-notifications-dropdown].notifications-modal .notifications-mark-all { border: none; background: transparent; color: var(--cp-ink-4); font-size: 12px; cursor: pointer; padding: 4px 8px; border-radius: 6px; }
[data-client-notifications-dropdown].notifications-modal .notifications-mark-all:hover { background: var(--cp-bg-hover); }
[data-client-notifications-dropdown].notifications-modal .notifications-close { border: none; background: transparent; color: var(--cp-ink-4); font-size: 1.1rem; cursor: pointer; padding: 2px 6px; border-radius: 6px; }
[data-client-notifications-dropdown].notifications-modal .notifications-close:hover { background: var(--cp-bg-hover); }
[data-client-notifications-dropdown].notifications-modal .notifications-list { list-style: none; margin: 0; padding: 4px 0 0; max-height: 320px; overflow-y: auto; }
[data-client-notifications-dropdown].notifications-modal .notifications-item { padding: 8px; border-radius: 8px; }
[data-client-notifications-dropdown].notifications-modal .notifications-item + .notifications-item { margin-top: 2px; }
[data-client-notifications-dropdown].notifications-modal .notifications-item--unread { background: var(--cp-accent-bg); }
[data-client-notifications-dropdown].notifications-modal .notifications-item__title { font-size: 13px; font-weight: 600; color: var(--cp-accent); }
[data-client-notifications-dropdown].notifications-modal .notifications-item__link { color: inherit; text-decoration: none; }
[data-client-notifications-dropdown].notifications-modal .notifications-item__link:hover { text-decoration: underline; }
[data-client-notifications-dropdown].notifications-modal .notifications-item__body { font-size: 12px; color: var(--cp-ink-3); margin-top: 2px; }
[data-client-notifications-dropdown].notifications-modal .notifications-item__meta { font-size: 11px; color: var(--cp-ink-4); margin-top: 2px; }
[data-client-notifications-dropdown].notifications-modal .notifications-empty { padding: 12px 4px; font-size: 13px; color: var(--cp-ink-3); }

/* --- Compliance legend --- */
.cp-legend { display: flex; flex-wrap: wrap; gap: 8px; margin: 0 0 14px; }
.cp-legend__item { display: inline-flex; align-items: center; gap: 4px; font-size: 11px; color: var(--cp-ink-3); }
.cp-legend__dot { width: 6px; height: 6px; border-radius: 999px; }
.cp-legend__dot--ok { background: var(--cp-green); }
.cp-legend__dot--overdue { background: var(--cp-red); }
.cp-legend__dot--never { background: var(--cp-accent); }

/* --- Documents hub grid --- */
.cp-doc-grid { display: grid; grid-template-columns: repeat(3, minmax(0, 1fr)); gap: 14px; padding: 14px 0; }
.cp-doc-card {
  background: var(--cp-bg-card); border-radius: 12px; box-shadow: var(--cp-shadow-card);
  padding: 20px; display: flex; flex-direction: column; gap: 12px;
  text-decoration: none; color: inherit; transition: box-shadow 0.15s;
}
.cp-doc-card:hover { box-shadow: var(--cp-shadow-md); }
.cp-doc-card__icon { width: 36px; height: 36px; border-radius: 8px; display: flex; align-items: center; justify-content: center; }
.cp-doc-card__icon svg { width: 18px; height: 18px; }
.cp-doc-card__title { font-size: 15px; font-weight: 600; color: var(--cp-ink); margin: 0; }
.cp-doc-card__count { font-family: var(--font-display); font-size: 26px; font-weight: 700; color: var(--cp-ink); letter-spacing: -0.02em; }
.cp-doc-card__meta { font-size: 12px; color: var(--cp-ink-4); }

/* --- Floorplan + timeline (inspection show) --- */
.cp-floorplan-grid { display: grid; grid-template-columns: repeat(auto-fit, minmax(260px, 1fr)); gap: 14px; }
.cp-floorplan__canvas { position: relative; border-radius: 10px; overflow: hidden; background: var(--cp-ink); }
.cp-floorplan__canvas--empty { display: flex; align-items: center; justify-content: center; min-height: 160px; background: var(--cp-bg-inset); }
.cp-floorplan__image { display: block; width: 100%; height: auto; }
.cp-floorplan__markers { position: absolute; inset: 0; pointer-events: none; }
.cp-floorplan-marker { position: absolute; transform: translate(-50%, -50%); pointer-events: auto; }
.cp-floorplan-marker__dot { width: 12px; height: 12px; border-radius: 999px; border: 2px solid var(--cp-ink); background: var(--cp-green); box-shadow: 0 0 0 2px rgba(26, 24, 21, 0.25); }
.cp-floorplan-marker--faulty .cp-floorplan-marker__dot { background: var(--cp-red); }
.cp-floorplan__title { font-weight: 600; font-size: 13px; color: var(--cp-ink); margin: 6px 0 2px; }
.cp-floorplan__hint { margin: 0; font-size: 11.5px; color: var(--cp-ink-4); }
.cp-timeline { list-style: none; padding: 0; margin: 0; }
.cp-timeline__item { display: grid; grid-template-columns: 16px 1fr; gap: 4px 10px; align-items: flex-start; padding: 8px 0; }
.cp-timeline__item + .cp-timeline__item { border-top: 1px solid var(--cp-rule); }
.cp-timeline__dot { width: 8px; height: 8px; border-radius: 999px; background: var(--cp-blue); margin-top: 5px; }
.cp-timeline__title { font-size: 13px; font-weight: 500; color: var(--cp-ink); margin: 0; }
.cp-timeline__meta { font-size: 11.5px; color: var(--cp-ink-4); }

/* --- Quote actions dropdown --- */
.cp-quote-actions { display: flex; justify-content: flex-end; gap: 6px; flex-wrap: wrap; }
.cp-quote-actions__more { position: relative; display: inline-block; }
.cp-quote-actions__more > summary::-webkit-details-marker { display: none; }
.cp-quote-actions__menu { position: absolute; right: 0; top: calc(100% + 4px); min-width: 260px; max-width: 340px; padding: 6px; border-radius: 10px; background: var(--cp-bg-card); border: 1px solid var(--cp-rule-strong); box-shadow: var(--cp-shadow-md); z-index: 2500; text-align: left; }
.cp-quote-actions__more--up .cp-quote-actions__menu { top: auto; bottom: calc(100% + 4px); }
.cp-quote-actions__item { list-style: none; display: flex; align-items: center; justify-content: space-between; width: 100%; padding: 8px 10px; border-radius: 8px; cursor: pointer; font-size: 13px; color: var(--cp-ink); }
.cp-quote-actions__item:hover { background: var(--cp-bg-hover); }
.cp-quote-actions__section { border-top: 1px solid var(--cp-rule); padding-top: 4px; margin-top: 4px; }
.cp-quote-actions__section:first-child { border-top: 0; padding-top: 0; margin-top: 0; }
.cp-quote-actions__section-body { padding: 6px 10px; max-height: 240px; overflow: auto; }
.cp-quote-actions__accept-form { display: grid; gap: 8px; padding: 6px 10px; }
.cp-quote-actions__input { width: 100%; padding: 7px 10px; border-radius: 6px; border: 1px solid var(--cp-rule-strong); background: var(--cp-bg-card); font-size: 12.5px; color: var(--cp-ink); font-family: inherit; }
.cp-quote-actions__input:focus { outline: none; border-color: var(--cp-accent); box-shadow: 0 0 0 3px var(--cp-accent-bg); }

/* --- File upload --- */
.cp-file-upload { border: 2px dashed var(--cp-rule-strong); border-radius: 10px; padding: 16px; text-align: center; cursor: pointer; transition: border-color 0.12s, background 0.12s; }
.cp-file-upload:hover { border-color: var(--cp-accent); background: var(--cp-accent-bg); }
.cp-file-upload input[type="file"] { display: none; }
.cp-file-upload__label { display: flex; align-items: center; justify-content: center; gap: 6px; color: var(--cp-ink-3); font-size: 13px; cursor: pointer; }
.cp-file-upload__label svg { width: 16px; height: 16px; color: var(--cp-ink-4); }

/* --- Search --- */
.cp-search-form { display: flex; gap: 8px; margin-bottom: 14px; }
.cp-search-form .cp-input { flex: 1; }
.cp-search-summary { font-size: 13px; color: var(--cp-ink-3); margin-bottom: 14px; }
.cp-search-item { display: flex; align-items: flex-start; gap: 12px; padding: 10px 14px; text-decoration: none; color: var(--cp-ink); border-radius: 8px; transition: background 0.12s; }
.cp-search-item:hover { background: var(--cp-bg-hover); }
.cp-search-item__icon { flex: 0 0 auto; width: 32px; height: 32px; border-radius: 8px; display: flex; align-items: center; justify-content: center; }
.cp-search-item__icon svg { width: 15px; height: 15px; }
.cp-search-item__copy { flex: 1; min-width: 0; }
.cp-search-item__title { font-size: 13px; font-weight: 600; color: var(--cp-ink); }
.cp-search-item__sub { font-size: 12px; color: var(--cp-ink-4); margin-top: 1px; }
.cp-search-item__cta { flex: 0 0 auto; font-size: 12px; color: var(--cp-accent); text-decoration: none; white-space: nowrap; }
.cp-search-item__cta:hover { text-decoration: underline; }

/* --- Inline utilities --- */
.cp-row-actions { display: flex; gap: 6px; justify-content: flex-end; flex-wrap: wrap; }
.cp-inline-form { display: inline; margin: 0; }
.cp-breadcrumbs { display: flex; align-items: center; gap: 8px; font-size: 12px; color: var(--cp-ink-4); margin-bottom: 14px; }
.cp-breadcrumbs a { color: var(--cp-accent); text-decoration: none; }
.cp-breadcrumbs a:hover { text-decoration: underline; }
.cp-def-list { display: grid; grid-template-columns: 120px 1fr; gap: 6px 14px; font-size: 13px; padding: 16px 20px; }
.cp-def-list dt { color: var(--cp-ink-4); font-weight: 500; }
.cp-def-list dd { margin: 0; color: var(--cp-ink); }
.cp-invite-form { margin-top: 16px; padding-top: 16px; border-top: 1px solid var(--cp-rule); }

/* 2026-04-22 Phase 7 — Settings page invite-form + contact card primitives. */
.cp-fieldset { border: 0; padding: 0; margin: 0; display: flex; flex-direction: column; gap: 12px; }
.cp-fieldset legend { margin-bottom: 6px; }
.cp-checkbox {
  display: flex; align-items: center; gap: 8px;
  font-size: 13px; color: var(--cp-ink-2); cursor: pointer;
}
.cp-checkbox input[type="checkbox"] {
  width: 16px; height: 16px; accent-color: var(--cp-accent);
}

/* Portal-user delete modal — canonical .modal--sheet shell, cp-scoped chrome
   so it inherits full-screen-on-mobile from style.css but matches the rest
   of the portal's type/colour scale on desktop. */
.cp-portal-user-delete-modal .modal__dialog {
  max-width: min(480px, calc(100vw - 32px));
  background: var(--cp-bg-card);
  color: var(--cp-ink);
}
.cp-portal-user-delete-modal .modal__header {
  padding: 18px 20px 12px;
  border-bottom: 1px solid var(--cp-rule);
}
.cp-portal-user-delete-modal .modal__body {
  padding: 18px 20px 20px;
  background: var(--cp-bg);
  display: flex; flex-direction: column; gap: 10px;
}
.cp-portal-user-delete-modal .modal__footer {
  padding: 12px 20px 14px;
  border-top: 1px solid var(--cp-rule);
  display: flex; justify-content: flex-end; gap: 8px;
}
.cp-portal-user-delete-modal .modal__title {
  margin: 0; font-size: 17px; color: var(--cp-ink);
}
.cp-portal-user-delete-modal .eyebrow {
  font-size: 10px; font-weight: 600; letter-spacing: 0.08em;
  text-transform: uppercase; color: var(--cp-red);
  margin: 0 0 4px;
}
.cp-portal-user-delete-modal .text-muted {
  margin: 6px 0 0; font-size: 12.5px; color: var(--cp-ink-3);
}
body.cp-portal-user-delete-modal-open { overflow: hidden; }

html[data-theme="dark"] .cp-portal-user-delete-modal .modal__dialog {
  background: var(--cp-bg-card);
}
html[data-theme="dark"] .cp-portal-user-delete-modal .modal__body {
  background: var(--cp-bg);
}

/* --- Mobile: sidebar + nav (Phase 1-5 mobile refactor 2026-04-17) --- */
@media (max-width: 900px) {
  /* Shell / main */
  .cp-shell { display: flex; flex-direction: column; min-height: auto; }
  .cp-main {
    grid-column: auto !important;
    grid-row: auto !important;
    /* Bottom padding + iOS safe-area respect so the last section never sits
       flush with the viewport edge or behind the home indicator. `!important`
       wins over any earlier desktop override inside the same media context.
       Fixed 2026-04-23 Phase 10 QA finding. */
    padding: 0 14px calc(48px + env(safe-area-inset-bottom, 0px)) !important;
    width: 100%;
    max-width: 100%;
    box-sizing: border-box;
  }

  /* Phase 1: Sidebar becomes slide-out drawer instead of hidden */
  .cp-sidebar {
    position: fixed;
    top: 0; left: 0; bottom: 0;
    width: min(280px, 82vw);
    height: 100vh; height: 100dvh;
    z-index: 2120;
    padding: 14px 12px calc(14px + env(safe-area-inset-bottom, 0px));
    background: var(--cp-bg-card);
    border-right: 1px solid var(--cp-rule);
    box-shadow: 4px 0 24px rgba(0, 0, 0, 0.08);
    transform: translateX(-110%);
    transition: transform 220ms ease;
    align-items: stretch;
    overflow-y: auto;
  }
  body.cp-nav-open .cp-sidebar { transform: translateX(0); }
  body.cp-nav-open { overflow: hidden; }

  /* Mobile sidebar elements */
  .cp-sidebar__close { display: inline-flex; }
  .cp-sidebar__tip { display: none !important; }
  .cp-sidebar__link {
    width: auto; height: auto;
    min-height: 44px;
    justify-content: flex-start;
    gap: 12px;
    padding: 10px 14px;
    border-radius: 9px;
  }
  .cp-sidebar__link:hover .cp-sidebar__tip { display: none; }
  .cp-sidebar__label { display: inline; }
  .cp-sidebar__spacer { min-height: 18px; }

  /* Overlay (shown via JS removing [hidden]) */
  .cp-nav-overlay:not([hidden]) { display: block; }

  /* Phase 2: Compact header — single row on mobile, hamburger drawer owns nav */
  .cp-nav {
    flex-direction: row !important;
    align-items: center !important;
    justify-content: space-between !important;
    flex-wrap: nowrap !important;
    gap: 8px !important;
    padding: 10px 12px !important;
    height: auto !important;
    min-height: 56px !important;
  }
  .cp-nav__hamburger { display: inline-flex !important; }
  .cp-nav__brand { flex: 1 1 auto; min-width: 0; overflow: hidden; }
  .cp-nav__name { font-size: 13px; }
  /* Hide top tabs on mobile — drawer owns primary nav, no redundant double nav */
  .cp-nav__tabs { display: none !important; }
  /* Hide inline search on mobile — accessed via the sidebar drawer */
  .cp-nav__search-form { display: none !important; }
  .cp-nav__actions { flex-shrink: 0; gap: 6px; }
  /* Non-demo action icons (notifications / help / settings) move to the drawer
     on mobile — topbar was clipping the brand text at 390px. Demo CTAs
     (.cp-nav__demo-btn) stay in the topbar because they're marketing CTAs. */
  .cp-nav__actions .cp-nav__btn { display: none !important; }
  .cp-nav__logout { display: none !important; }
  /* Collapse demo tour button to icon-only; keep trial CTA with label */
  .cp-nav__demo-btn.demo-tour-launch { padding: 0; width: 40px; height: 40px; }
  .cp-nav__demo-btn.demo-tour-launch span { display: none; }
  .cp-nav__demo-btn.demo-trial-cta { padding: 8px 12px; font-size: 12.5px; }

  /* Drawer-only action rows — visible alongside primary nav items */
  .cp-sidebar__link--mobile-only { display: flex; }
  .cp-sidebar__logout { display: block; width: 100%; }
  .cp-sidebar__link--notifications { color: var(--cp-accent); }
  .cp-sidebar__link--help { color: var(--cp-ink-3); }
  .cp-sidebar__link--logout { color: var(--cp-red); }
  .cp-sidebar__link--logout:hover { background: var(--cp-red-bg); }
  .cp-sidebar__badge {
    display: inline-flex; align-items: center; justify-content: center;
    min-width: 20px; height: 20px; padding: 0 6px; margin-left: auto;
    border-radius: 999px;
    background: var(--cp-red); color: #fff;
    font-size: 11px; font-weight: 600; line-height: 1;
  }
  .cp-sidebar__badge[hidden] { display: none; }

  /* Phase 3: Stats strip horizontal scroll (was 4-col grid overflow) */
  .cp-status-strip,
  .cp-status-strip--3,
  .cp-status-strip--5,
  .cp-status-strip--compact {
    display: flex;
    flex-wrap: nowrap;
    overflow-x: auto;
    scrollbar-width: none;
    -ms-overflow-style: none;
    gap: 10px;
    scroll-snap-type: x mandatory;
    padding-inline: 2px;
  }
  .cp-status-strip::-webkit-scrollbar,
  .cp-status-strip--3::-webkit-scrollbar,
  .cp-status-strip--5::-webkit-scrollbar,
  .cp-status-strip--compact::-webkit-scrollbar { display: none; }
  .cp-stat-card {
    flex: 0 0 min(240px, 76vw);
    scroll-snap-align: start;
  }

  /* Phase 4: Table card-ification at 900px (was 600px) */
  .cp-table-wrap { overflow-x: visible; }
  .cp-table {
    min-width: 0;
    table-layout: auto;
  }
  .cp-table thead {
    position: absolute;
    width: 1px; height: 1px;
    padding: 0; margin: -1px;
    overflow: hidden; clip: rect(0, 0, 0, 0);
    border: 0;
  }
  .cp-table tbody {
    display: flex;
    flex-direction: column;
    gap: 8px;
  }
  .cp-table tbody tr {
    display: flex;
    flex-direction: column;
    border: 1px solid var(--cp-rule);
    border-radius: 10px;
    padding: 12px 14px;
    background: var(--cp-bg-card);
    transition: background 0.12s, border-color 0.12s, transform 0.12s;
  }
  .cp-table tbody tr:has(a[href]) { border-color: var(--cp-rule-strong, var(--cp-rule)); }
  .cp-table tbody tr:has(a[href]):active {
    background: var(--cp-bg-hover);
    transform: scale(0.995);
  }
  .cp-table tbody td {
    display: flex;
    align-items: baseline;
    gap: 8px;
    padding: 4px 0;
    border: none;
    width: auto;
  }
  .cp-table tbody td[data-label]::before {
    content: attr(data-label);
    display: inline;
    font-size: 11px; font-weight: 600;
    text-transform: uppercase; letter-spacing: 0.05em;
    color: var(--cp-ink-4);
    min-width: 70px;
    flex-shrink: 0;
    margin-right: 8px;
  }
  .cp-table-scroll-hint { display: none; }
  .cp-td-prop-name { font-weight: 600; }
  .cp-td-prop-sub { display: block; font-size: 12px; color: var(--cp-ink-4); }

  /* Phase 5 base: filter rail becomes visible on mobile (per-page wrappers add cp-filter-shell) */
  .cp-filter-shell .cp-filters { display: none; }
  .cp-filter-shell .cp-filter-rail {
    display: flex;
    flex-direction: column;
    gap: 10px;
    padding: 14px 16px;
    background: var(--cp-bg-card);
    border: 1px solid var(--cp-rule);
    border-radius: 12px;
    box-shadow: var(--cp-shadow-card, 0 1px 4px rgba(26, 24, 21, 0.06));
  }
  .cp-filter-rail__copy { display: flex; flex-direction: column; gap: 2px; }
  .cp-filter-rail__eyebrow {
    font-size: 10px; font-weight: 600;
    text-transform: uppercase; letter-spacing: 0.08em;
    color: var(--cp-ink-4);
    margin: 0;
  }
  .cp-filter-rail__meta {
    font-size: 13px; color: var(--cp-ink-2);
    margin: 0;
  }
  .cp-filter-rail__actions {
    display: grid;
    grid-template-columns: 1fr auto;
    gap: 8px;
  }
  .cp-filter-railbtn {
    min-height: 40px;
    padding: 8px 14px;
    border-radius: 9px;
    border: 1px solid var(--cp-rule);
    background: var(--cp-bg-inset);
    color: var(--cp-ink);
    font-size: 13px; font-weight: 500;
    cursor: pointer;
    display: inline-flex;
    align-items: center;
    justify-content: center;
    gap: 6px;
  }
  .cp-filter-railbtn:hover { background: var(--cp-bg-hover); }
  .cp-filter-railbtn--primary {
    background: var(--cp-ink);
    color: #fff;
    border-color: var(--cp-ink);
  }
  .cp-filter-railbtn--primary:hover { background: var(--cp-ink-2); }
  .cp-filters-drawer {
    position: fixed; inset: 0;
    display: none;
    z-index: 2120;
  }
  .cp-filters-drawer.is-open {
    display: flex;
    flex-direction: column;
    justify-content: flex-end;
  }
  .cp-filters-drawer__overlay {
    position: absolute; inset: 0;
    background: rgba(26, 24, 21, 0.32);
    backdrop-filter: blur(2px);
  }
  .cp-filters-drawer__panel {
    position: relative;
    background: var(--cp-bg-card);
    border-radius: 20px 20px 0 0;
    /* 2026-04-24: min-height keeps the panel at a genuine half-sheet
       footprint even on pages with only 1–2 filter fields (Properties,
       Inspections, Billing). Without it, content-sizing shrinks the
       panel to ~250px which feels like a cramped strip vs. the tenant
       drawer's half-screen weight. `max-height: 86dvh` matches the
       tenant canonical `.tnt-filter-drawer__panel`. */
    min-height: min(55dvh, 460px);
    max-height: min(86dvh, 760px);
    display: flex;
    flex-direction: column;
    box-shadow: 0 -18px 40px rgba(15, 23, 42, 0.18);
  }
  .cp-filters-drawer__header {
    display: flex;
    align-items: center;
    justify-content: space-between;
    padding: 16px 18px 12px;
    border-bottom: 1px solid var(--cp-rule);
    flex-shrink: 0;
  }
  .cp-filters-drawer__title { margin: 0; font-size: 16px; font-weight: 600; color: var(--cp-ink); }
  .cp-filters-drawer__close {
    width: 36px; height: 36px;
    border: 1px solid var(--cp-rule); border-radius: 9px;
    background: transparent; color: var(--cp-ink);
    cursor: pointer; padding: 0;
    display: inline-flex; align-items: center; justify-content: center;
  }
  /* 2026-04-24: `.cp-filters-drawer__body` is applied to a `<form>`
     element in every CP filter view. The `.cp-portal form, .cp-auth form`
     reset at line 1235 has specificity 0,1,1 and zeroes padding on every
     form inside the portal shell. Without scoping the body rule under
     `.cp-portal` (to reach 0,2,0) the reset wins and the drawer content
     touches the panel edges on every side. Match tenant `.tnt-filter-
     drawer__body` padding rhythm (16px all round + safe-area bottom). */
  .cp-portal .cp-filters-drawer__body,
  .cp-filters-drawer__body {
    padding: 16px 16px calc(18px + env(safe-area-inset-bottom, 0px));
    overflow-y: auto;
    flex: 1 1 auto;
    min-height: 0;
    /* Flex column owns rhythm between the .cp-field rows + the actions row;
       the `.cp-portal form` reset zeroes gap + display, so re-assert both
       here with higher specificity. */
    display: flex;
    flex-direction: column;
    gap: 12px;
  }
  .cp-filters-drawer__body .cp-filters {
    display: flex;
    flex-direction: column;
    align-items: stretch;
    gap: 12px;
  }
  .cp-filters-drawer__body .cp-filter-input,
  .cp-filters-drawer__body .cp-filter-select {
    width: 100%;
    min-height: 44px;
    font-size: 16px;
  }
  .cp-filters-drawer__actions {
    display: grid;
    grid-template-columns: repeat(2, minmax(0, 1fr));
    gap: 8px;
    /* 2026-04-24: rhythm above the actions row is now owned by the body's
       flex-gap (12px). Previous margin-top: 14px stacked on top of that
       and made the Clear/Apply row drift away from the last filter. */
    margin-top: auto;
  }

  /* Original mobile rules preserved */
  .cp-filters { flex-direction: column; align-items: stretch; }
  .cp-filter-input, .cp-filter-select { min-width: 0; width: 100%; min-height: 44px; font-size: 16px; }
  .cp-form-grid { grid-template-columns: 1fr; }
  .cp-input { min-height: 44px; font-size: 16px; }
  .cp-auth { padding: 16px 12px; min-height: auto; }
  .cp-auth-card { padding: 24px 20px; max-width: none; }
  /* 2026-04-24: previous `width: 100%; display: flex` forced the mobile
     tab bar to stretch the full viewport, overriding the base rule's
     `inline-flex + align-self: flex-start`. That made CP tabs look
     different from tenant `.ipi-tabs` (which stays content-width at
     every breakpoint). Keep only overflow-x: auto as a safety net for
     pages that ever ship with enough tabs to need horizontal scroll —
     but leave width/display to the base rule so the bar sizes to content. */
  .cp-tabs { overflow-x: auto; scrollbar-width: none; max-width: 100%; }
  .cp-tabs::-webkit-scrollbar { display: none; }
  .cp-row-actions { flex-direction: column; }
  .cp-doc-grid { grid-template-columns: 1fr; }
  .cp-def-list { grid-template-columns: 1fr; gap: 2px 0; padding: 14px 16px; }
  .cp-def-list dt { font-size: 10px; margin-top: 8px; }
  .cp-def-list dt:first-child { margin-top: 0; }
  .cp-pagination { flex-direction: column; align-items: stretch; gap: 8px; text-align: center; }

  /* Client-portal modal: full-screen on mobile */
  .cp-modal { align-items: stretch; justify-content: stretch; padding: 0; }
  .cp-modal__dialog {
    width: 100%;
    max-width: 100%;
    height: 100vh;
    height: 100dvh;
    max-height: 100vh;
    max-height: 100dvh;
    min-height: 0;
    margin: 0;
    border: none;
    border-radius: 0;
    box-shadow: none;
  }
  .cp-modal__header { padding-top: calc(20px + env(safe-area-inset-top, 0px)); flex-shrink: 0; }
  .cp-modal__body { flex: 1 1 auto; min-height: 0; overflow-y: auto; -webkit-overflow-scrolling: touch; overscroll-behavior: contain; }
  .cp-modal__footer { padding-bottom: calc(13px + env(safe-area-inset-bottom, 0px)); flex-shrink: 0; }

  [data-client-notifications-dropdown].notifications-modal { align-items: stretch; justify-content: stretch; padding: 0; }
  [data-client-notifications-dropdown].notifications-modal .notifications-modal__panel {
    width: 100%;
    max-width: none;
    height: 100dvh;
    max-height: 100dvh;
    margin: 0;
    border-radius: 0;
    box-shadow: none;
  }
}

/* --- Dark mode: portal-specific --- */
html[data-theme="dark"] .page-section.client-portal { background: #0b1220; padding: 18px 24px 0; }
html[data-theme="dark"] .cp-sidebar__tip { background: var(--cp-bg-card); border: 1px solid var(--cp-rule-strong); }
html[data-theme="dark"] .cp-nav__search-input { background: rgba(148, 163, 184, 0.08); border-color: var(--cp-rule-strong); color: var(--cp-ink); }
html[data-theme="dark"] .cp-input { background: rgba(148, 163, 184, 0.08); border-color: var(--cp-rule-strong); color: var(--cp-ink); }
html[data-theme="dark"] .cp-filter-input, html[data-theme="dark"] .cp-filter-select { background: rgba(148, 163, 184, 0.08); border-color: var(--cp-rule-strong); color: var(--cp-ink); }
html[data-theme="dark"] .cp-filter-pill.is-active { background: var(--cp-bg-card); color: var(--cp-ink); border-color: var(--cp-ink); }
html[data-theme="dark"] .cp-auth-card { box-shadow: 0 4px 24px rgba(0, 0, 0, 0.4); }
html[data-theme="dark"] .cp-quote-actions__menu { background: var(--cp-bg-card); border-color: var(--cp-rule-strong); box-shadow: 0 12px 30px rgba(0, 0, 0, 0.55); }
html[data-theme="dark"] .cp-quote-actions__item:hover { background: var(--cp-bg-hover); }
html[data-theme="dark"] [data-client-notifications-dropdown].notifications-modal .notifications-modal__panel { background: var(--cp-bg-card); box-shadow: 0 22px 60px rgba(0, 0, 0, 0.55); }
html[data-theme="dark"] [data-client-notifications-dropdown].notifications-modal .notifications-mark-all:hover { background: rgba(255, 255, 255, 0.06); }
html[data-theme="dark"] [data-client-notifications-dropdown].notifications-modal .notifications-close:hover { background: rgba(255, 255, 255, 0.06); }
html[data-theme="dark"] [data-client-notifications-dropdown].notifications-modal .notifications-item--unread { background: rgba(59, 130, 246, 0.12); }
html[data-theme="dark"] [data-client-notifications-dropdown].notifications-modal .notifications-item__body,
html[data-theme="dark"] [data-client-notifications-dropdown].notifications-modal .notifications-item__meta,
html[data-theme="dark"] [data-client-notifications-dropdown].notifications-modal .notifications-empty { color: var(--cp-ink-4); }

/* 2026-04-22 Phase 8 dark-mode override for `.cp-filter-railbtn--primary` —
   background `var(--cp-ink)` flips to near-white in dark mode; `#fff` text
   becomes invisible-on-invisible. Swap to canvas.
   (Sidebar + nav__btn active-state overrides from Phase 8 removed 2026-04-23
   when those selectors were softened to accent-tinted backgrounds — the
   `--cp-accent*` tokens already cascade to appropriate dark values and no
   override is needed.) */
html[data-theme="dark"] .cp-filter-railbtn--primary { color: var(--cp-bg); }
