/* classroom.css — Classroom (list + room views).
   Room view: "class register" editorial look — Newsreader headings, ledger
   roster with mono indices, ember pins, ticket invite bar. vb- tokens only. */
.classroom-root { max-width: var(--vb-container-default); margin: 0 auto;
  padding: var(--vb-space-6) var(--vb-space-4) var(--vb-space-12);
  font-family: var(--vb-font-sans); color: var(--vb-text-primary); }

/* ---- entrance: one staggered rise on room load ---- */
@keyframes vb-cls-rise {
  from { opacity: 0; transform: translateY(10px); }
  to   { opacity: 1; transform: none; }
}
.vb-cls-room-head, .vb-cls-cols > .vb-cls-panel, .vb-cls-invitebar, .vb-cls-banner {
  animation: vb-cls-rise var(--vb-dur-slow) var(--vb-ease-out) backwards; }
.vb-cls-cols > .vb-cls-panel--notices { animation-delay: 70ms; }
.vb-cls-cols > .vb-cls-panel:not(.vb-cls-panel--notices) { animation-delay: 140ms; }
.vb-cls-invitebar { animation-delay: 210ms; }
@media (prefers-reduced-motion: reduce) {
  .vb-cls-room-head, .vb-cls-cols > .vb-cls-panel, .vb-cls-invitebar, .vb-cls-banner {
    animation: none; }
}

/* ---- shared: header (list view) + buttons ---- */
.vb-cls-header { display: flex; justify-content: space-between; align-items: center;
  margin-bottom: var(--vb-space-4); }
.vb-cls-header h2 { font-family: var(--vb-font-serif); font-weight: 500;
  font-size: var(--vb-text-xl); }
.vb-cls-btn { background: var(--vb-aubergine); color: var(--vb-text-on-accent);
  border: none; border-radius: var(--vb-radius-md); padding: 7px 14px;
  font-size: var(--vb-text-sm); font-weight: 600; cursor: pointer; }
.vb-cls-btn--ghost { background: transparent; color: var(--vb-aubergine);
  border: 0.5px solid var(--vb-border-accent); }
.vb-cls-btn--danger { background: transparent; color: var(--vb-danger);
  border: 0.5px solid var(--vb-border); font-size: var(--vb-text-xs); padding: 3px 8px; }
.vb-cls-btn:disabled { opacity: .45; cursor: default; }

/* ---- list view cards ---- */
.vb-cls-card { display: flex; justify-content: space-between; align-items: center;
  background: var(--vb-bg-elevated); border: 0.5px solid var(--vb-border);
  border-radius: var(--vb-radius-lg); padding: var(--vb-space-3) var(--vb-space-4);
  margin-bottom: var(--vb-space-2); cursor: pointer;
  transition: transform var(--vb-dur-fast) var(--vb-ease),
              box-shadow var(--vb-dur-fast) var(--vb-ease),
              border-color var(--vb-dur-fast) var(--vb-ease); }
.vb-cls-card:hover { transform: translateY(-1px); border-color: var(--vb-border-strong);
  box-shadow: 0 6px 18px -12px color-mix(in srgb, var(--vb-ink) 35%, transparent); }
.vb-cls-card--frozen { opacity: .6; }
.vb-cls-card strong { font-family: var(--vb-font-serif); font-weight: 500;
  font-size: var(--vb-text-base); }
.vb-cls-card .meta { color: var(--vb-text-tertiary); font-size: var(--vb-text-xs); }

/* ---- badges (roles) ---- */
.vb-cls-badge { font-size: 10px; font-weight: 600; letter-spacing: .08em;
  text-transform: uppercase; border-radius: var(--vb-radius-sm); padding: 2px 7px;
  background: var(--vb-aubergine-soft); color: var(--vb-aubergine);
  margin-left: var(--vb-space-2); vertical-align: 1px;
  white-space: nowrap; /* 좁은 행에서 단어 중간 꺾임 방지 */ }

/* ---- room header ---- */
.vb-cls-backlink { display: inline-block; color: var(--vb-text-tertiary);
  font-size: var(--vb-text-sm); text-decoration: none;
  margin-bottom: var(--vb-space-4); }
.vb-cls-backlink:hover { color: var(--vb-aubergine); }
.vb-cls-eyebrow { font-size: var(--vb-text-xs); font-weight: 600;
  letter-spacing: .22em; text-transform: uppercase; color: var(--vb-ember);
  margin-bottom: var(--vb-space-1); }
.vb-cls-room-head { display: flex; justify-content: space-between;
  align-items: flex-end; gap: var(--vb-space-4); flex-wrap: wrap;
  padding-bottom: var(--vb-space-4);
  border-bottom: 3px double var(--vb-border-strong);
  margin-bottom: var(--vb-space-6); }
.vb-cls-room-head h2 { font-family: var(--vb-font-serif); font-weight: 500;
  font-size: var(--vb-text-3xl); line-height: var(--vb-leading-tight);
  letter-spacing: -.01em; }
.vb-cls-desc { font-family: var(--vb-font-serif); font-style: italic;
  font-size: var(--vb-text-base); color: var(--vb-text-secondary);
  margin-top: var(--vb-space-1); }
.vb-cls-actions { display: flex; gap: var(--vb-space-2); flex: none; }

/* ---- panels ---- */
.vb-cls-cols { display: flex; gap: var(--vb-space-6); align-items: flex-start; }
.vb-cls-panel { background: var(--vb-bg-elevated);
  border: 0.5px solid var(--vb-border); border-radius: var(--vb-radius-lg);
  padding: var(--vb-space-5); flex: 1;
  box-shadow: 0 1px 2px color-mix(in srgb, var(--vb-ink) 4%, transparent),
              0 8px 24px -18px color-mix(in srgb, var(--vb-ink) 25%, transparent); }
.vb-cls-panel--notices { flex: 1.7; }
.vb-cls-panel h5 { font-size: var(--vb-text-xs); text-transform: uppercase;
  letter-spacing: .18em; color: var(--vb-text-tertiary);
  padding-bottom: var(--vb-space-2); border-bottom: 1px solid var(--vb-border);
  margin-bottom: var(--vb-space-3); }

/* ---- notice board ---- */
.vb-cls-notice { padding: var(--vb-space-4) 0;
  border-bottom: 0.5px solid var(--vb-border-soft); }
.vb-cls-notice:last-of-type { border-bottom: none; }
.vb-cls-notice .meta { display: flex; align-items: center; gap: var(--vb-space-2);
  color: var(--vb-text-tertiary); font-family: var(--vb-font-mono);
  font-size: var(--vb-text-xs); letter-spacing: .08em; text-transform: uppercase;
  margin-bottom: var(--vb-space-2); }
.vb-cls-pin { width: 7px; height: 7px; background: var(--vb-ember);
  transform: rotate(45deg); border-radius: 1px; flex: none; }
.vb-cls-notice .meta a { margin-left: auto; color: var(--vb-text-tertiary);
  text-decoration: none; text-transform: none; letter-spacing: 0; opacity: 0;
  transition: opacity var(--vb-dur-fast) var(--vb-ease); }
.vb-cls-notice:hover .meta a, .vb-cls-notice .meta a:focus-visible { opacity: 1; }
.vb-cls-notice .meta a:hover { color: var(--vb-danger); }
.vb-cls-notice .vb-content { font-family: var(--vb-font-serif);
  font-size: var(--vb-text-reading); line-height: var(--vb-leading-relaxed); }
.vb-cls-notice .vb-content :is(h1, h2, h3) { font-family: var(--vb-font-serif);
  font-weight: 600; font-size: var(--vb-text-lg);
  margin: var(--vb-space-2) 0 var(--vb-space-1); }
.vb-cls-notice .vb-content p { margin: var(--vb-space-2) 0; }
.vb-cls-notice .vb-content p:first-child { margin-top: 0; }
.vb-cls-notice .vb-content a { color: var(--vb-aubergine); }

/* ---- members register ---- */
.vb-cls-panel { counter-reset: vb-roster; }
/* baseline 정렬 — 닉네임(13px)·날짜(11px mono)·kick 텍스트가 같은 글줄에 놓이도록
   (center 정렬은 폰트 크기 차이만큼 베이스라인이 어긋나 보인다) */
.vb-cls-member { display: flex; align-items: baseline; gap: var(--vb-space-2);
  /* 좁은 패널(모바일·2열 우측)에서 이름을 쥐어짜는 대신 날짜·버튼이 다음 줄로 —
     이름 min-width 와 한 쌍 (이것 없이는 anywhere 가 한 글자 폭까지 수축) */
  flex-wrap: wrap; row-gap: var(--vb-space-1);
  padding: var(--vb-space-2) 0; border-bottom: 0.5px solid var(--vb-border-soft);
  font-size: var(--vb-text-sm); }
.vb-cls-member:last-child { border-bottom: none; }
.vb-cls-member:not(.vb-cls-member--owner) { counter-increment: vb-roster; }
.vb-cls-member:not(.vb-cls-member--owner)::before {
  content: counter(vb-roster, decimal-leading-zero);
  font-family: var(--vb-font-mono); font-size: var(--vb-text-xs);
  color: var(--vb-text-tertiary); width: 20px; flex: none; }
.vb-cls-member--owner { border-bottom: 1px solid var(--vb-border-strong);
  padding-bottom: var(--vb-space-3); margin-bottom: var(--vb-space-1); }
.vb-cls-member--owner .vb-cls-member-name { font-family: var(--vb-font-serif);
  font-size: var(--vb-text-base); font-weight: 500; }
.vb-cls-member-name { flex: 1; min-width: 12ch; overflow-wrap: anywhere; }
.vb-cls-member .meta { color: var(--vb-text-tertiary);
  font-family: var(--vb-font-mono); font-size: var(--vb-text-xs); flex: none; }
/* kick 은 행 호버에서만 — 터치 기기는 항상 노출 */
.vb-cls-member .vb-cls-btn--danger { opacity: 0;
  transition: opacity var(--vb-dur-fast) var(--vb-ease); }
.vb-cls-member:hover .vb-cls-btn--danger,
.vb-cls-member .vb-cls-btn--danger:focus-visible { opacity: 1; }
/* 터치 기기 + 좁은 뷰포트(좁힌 데스크톱 창 포함)에선 상시 노출 —
   hover:none 만으론 좁은 데스크톱 창에서 kick 이 안 보여 빈 공간만 남는다 */
@media (hover: none), (max-width: 720px) {
  .vb-cls-member .vb-cls-btn--danger, .vb-cls-notice .meta a { opacity: 1; }
}
/* ---- invite ticket (owner) ---- */
.vb-cls-invitebar { display: flex; align-items: center; gap: var(--vb-space-4);
  flex-wrap: wrap; position: relative; margin-top: var(--vb-space-6);
  background: var(--vb-bg-tint); border: 1px dashed var(--vb-border-strong);
  border-radius: var(--vb-radius-lg);
  padding: var(--vb-space-3) var(--vb-space-6); font-size: var(--vb-text-sm); }
/* 티켓 천공 노치 — 페이지 배경색 원으로 펀칭 효과 */
.vb-cls-invitebar::before, .vb-cls-invitebar::after { content: '';
  position: absolute; top: 50%; width: 14px; height: 14px; border-radius: 50%;
  background: var(--vb-bg-page); border: 1px dashed var(--vb-border-strong);
  transform: translateY(-50%); }
.vb-cls-invitebar::before { left: -8px; }
.vb-cls-invitebar::after { right: -8px; }
.vb-cls-invite-label { font-size: var(--vb-text-xs); font-weight: 600;
  letter-spacing: .18em; text-transform: uppercase;
  color: var(--vb-text-tertiary); }
.vb-cls-invitebar code { font-family: var(--vb-font-mono);
  font-size: var(--vb-text-lg); font-weight: 500; letter-spacing: .22em;
  color: var(--vb-ember); }
.vb-cls-invite-actions { margin-left: auto; display: flex; gap: var(--vb-space-2); }

/* ---- banner / composer / empty ---- */
.vb-cls-banner { background: var(--vb-highlight-unlearned);
  border-left: 3px solid var(--vb-warning); border-radius: var(--vb-radius-md);
  color: var(--vb-text-secondary); padding: var(--vb-space-3) var(--vb-space-4);
  font-size: var(--vb-text-sm); margin-bottom: var(--vb-space-4); }
.vb-cls-textarea { width: 100%; box-sizing: border-box; min-height: 90px;
  border: 0.5px solid var(--vb-border-strong); border-radius: var(--vb-radius-md);
  background: var(--vb-bg-elevated); color: var(--vb-text-primary);
  font-family: var(--vb-font-sans); font-size: var(--vb-text-sm);
  padding: var(--vb-space-2); margin: var(--vb-space-3) 0 var(--vb-space-2); }
.vb-cls-textarea:focus { outline: none; border-color: var(--vb-border-accent);
  box-shadow: var(--vb-focus-ring); }
.vb-cls-empty { color: var(--vb-text-tertiary); font-family: var(--vb-font-serif);
  font-style: italic; font-size: var(--vb-text-base); text-align: center;
  padding: var(--vb-space-8) 0; }

/* ---- tabs: Notice | Members — 전 환경, 한 패널씩 (2열 병렬 폐지) ---- */
.vb-cls-tabbar { display: flex; gap: var(--vb-space-1);
  margin-bottom: var(--vb-space-3);
  border-bottom: 1px solid var(--vb-border-strong); }
.vb-cls-tab { appearance: none; background: none; border: none; cursor: pointer;
  padding: var(--vb-space-2) var(--vb-space-3);
  font-size: var(--vb-text-sm); font-weight: 600; letter-spacing: .02em;
  color: var(--vb-text-tertiary);
  border-bottom: 2px solid transparent; margin-bottom: -1px; }
.vb-cls-tab.is-active { color: var(--vb-aubergine);
  border-bottom-color: var(--vb-aubergine); }
/* 탭 전환: 기본=Notice, .vb-cls-cols--members 수식자=Members (JS 토글) */
.vb-cls-cols > .vb-cls-panel--members { display: none; }
.vb-cls-cols--members > .vb-cls-panel--members { display: block; }
.vb-cls-cols--members > .vb-cls-panel--notices { display: none; }

@media (max-width: 720px) {
  /* column 방향에선 align-items 가 가로축 — flex-start 면 패널이 내용 폭으로 수축 */
  .vb-cls-cols { flex-direction: column; gap: var(--vb-space-4); align-items: stretch; }
  .vb-cls-room-head { gap: var(--vb-space-3); padding-bottom: var(--vb-space-3);
    margin-bottom: var(--vb-space-4); }
  .vb-cls-room-head h2 { font-size: var(--vb-text-xl); }
  /* 액션은 우측 정렬 + 콤팩트 칩 — 제목 아래 왼쪽 큰 버튼 덩어리(폼처럼 보임) 방지 */
  .vb-cls-actions { width: 100%; justify-content: flex-end; }
  .vb-cls-actions .vb-cls-btn { padding: 5px 11px; font-size: var(--vb-text-xs); }
  /* 티켓 내부: 라벨 → 코드(크게) → 버튼 2개 균등 — 세로 중앙 정렬 */
  .vb-cls-invitebar { flex-direction: column; align-items: center;
    gap: var(--vb-space-2); text-align: center;
    padding: var(--vb-space-5) var(--vb-space-4); }
  .vb-cls-invitebar code { font-size: var(--vb-text-xl); }
  .vb-cls-invite-actions { margin-left: 0; width: 100%; justify-content: center;
    margin-top: var(--vb-space-1); }
  .vb-cls-invite-actions .vb-cls-btn { flex: 1; max-width: 160px; }
  /* 풀-블리드 — root 좌우 16px 패딩을 상쇄해 헤더 괘선·패널·티켓이 화면 폭을
     꽉 채운다. 내부 패딩으로 텍스트 정렬선은 유지. */
  .vb-cls-room-head {
    margin-left: calc(-1 * var(--vb-space-4));
    margin-right: calc(-1 * var(--vb-space-4));
    padding-left: var(--vb-space-4); padding-right: var(--vb-space-4); }
  .vb-cls-panel { padding: var(--vb-space-4);
    margin-left: calc(-1 * var(--vb-space-4));
    margin-right: calc(-1 * var(--vb-space-4));
    border-left: none; border-right: none; border-radius: 0; box-shadow: none; }
  .vb-cls-invitebar { margin-top: var(--vb-space-4);
    margin-left: calc(-1 * var(--vb-space-4));
    margin-right: calc(-1 * var(--vb-space-4));
    border-left: none; border-right: none; border-radius: 0; }
  /* 풀-블리드에선 천공 노치가 화면 밖으로 돌출해 가로 스크롤 유발 → 숨김 */
  .vb-cls-invitebar::before, .vb-cls-invitebar::after { display: none; }
}

/* ---- Phase B: worksheet post card ---- */
/* (--ws 좌측 보더 제거 — 사용자 요청 §230 후속. 클래스는 JS 훅으로 잔존) */
.vb-cls-ws-title { font-family: var(--vb-font-serif); font-weight: 600;
  font-size: var(--vb-text-base); margin: var(--vb-space-1) 0; }
.vb-cls-ws-meta { display: flex; flex-wrap: wrap; align-items: center;
  gap: var(--vb-space-1); margin-bottom: var(--vb-space-2); }
.vb-cls-ws-chip { font-size: 10px; font-weight: 600; letter-spacing: .08em;
  text-transform: uppercase; border-radius: var(--vb-radius-sm); padding: 2px 7px;
  background: var(--vb-aubergine-soft); color: var(--vb-aubergine); }
.vb-cls-ws-info { font-size: var(--vb-text-xs); color: var(--vb-text-tertiary);
  margin-left: var(--vb-space-1); }
.vb-cls-ws-actions { display: flex; gap: var(--vb-space-2);
  margin-top: var(--vb-space-2); }

/* ---- live chat (Phase C) ---- */
.vb-cls-panel--chat { margin-top: 16px; }
.vb-cls-chat-log {
  display: flex; flex-direction: column; gap: 6px;
  max-height: 320px; min-height: 48px; overflow-y: auto;
  padding: 8px 0;
}
.vb-cls-chat-msg { display: flex; align-items: baseline; gap: 8px; }
.vb-cls-chat-nick { font-weight: 600; flex: none; }
.vb-cls-chat-body { overflow-wrap: anywhere; }
.vb-cls-chat-del {
  flex: none; margin-left: auto;
  border: none; background: none; cursor: pointer;
  font-size: 14px; line-height: 1; opacity: .55;
  color: var(--vb-text-tertiary);
}
.vb-cls-chat-del:hover { opacity: 1; color: var(--vb-danger); }
.vb-cls-chat-inputrow { display: flex; gap: 8px; margin-top: 8px; }
.vb-cls-chat-inputrow input { flex: 1; min-width: 0;
  border: 0.5px solid var(--vb-border-strong); border-radius: var(--vb-radius-md);
  background: var(--vb-bg-elevated); color: var(--vb-text-primary);
  font-family: var(--vb-font-sans); font-size: var(--vb-text-sm);
  padding: var(--vb-space-2); }
.vb-cls-chat-inputrow input:focus { outline: none; border-color: var(--vb-border-accent);
  box-shadow: var(--vb-focus-ring); }
#vb-cls-chat-close-btn { margin-top: 8px; }   /* inputrow 와 동일 리듬 — 입력창에 붙지 않게 */
/* hidden 속성이 display 지정에 지지 않도록 (known trap: hidden vs display CSS) */
#vb-cls-chat-closed[hidden], #vb-cls-chat-live[hidden] { display: none; }

/* 학생 로그인 카드 (Add student / Reset PW 결과 모달) */
.vb-cls-logincard { border: 1px solid var(--vb-border-strong); border-radius: 8px;
                    padding: 12px 14px; margin-bottom: 8px; }
.vb-cls-logincard-row { display: flex; justify-content: space-between; gap: 12px;
                        padding: 4px 0; font-size: 0.92rem; }
.vb-cls-logincard-row code { font-family: var(--vb-font-mono, monospace);
                             font-size: 1rem; user-select: all; }
.vb-modal__check { display: flex; gap: 8px; align-items: flex-start;
                   font-size: 0.9rem; margin-top: 10px; }
.vb-modal__check input { width: auto; margin-top: 2px; }
.vb-modal__hint { color: var(--vb-text-tertiary); font-size: 0.85rem; margin-top: 8px; }

/* 온라인 시험 오버레이 (Phase D) */
/* 풀스크린 오버레이 (spec §4-③ — 몰입감). 모달 자체를 fixed inset:0 으로 펼치고
   내부 콘텐츠는 읽기 폭(760px) 중앙 칼럼 — 풀블리드 + 적정 행길이.
   부수효과(의도): backdrop 클릭 dismiss 불가 — 응시 중 오클릭으로 닫힘 방지. */
/* .vb-modal 기본 룰(index.html 인라인)과 동일 특이도면 문서 순서상 인라인이 이김 —
   두 클래스로 특이도 상승 필수. */
.vb-modal.vb-modal--exam { position: fixed; inset: 0; width: auto; max-width: none;
                  max-height: none; border-radius: 0; border: none;
                  padding: 24px 0;   /* 좌우 0 — 스크롤바가 화면 맨 우측 가장자리에 */
                  /* 읽기 칼럼: 와이드=760px, 좁으면 좌우 여백 확보 */
                  --vb-exam-col: min(760px, calc(100% - 2 * clamp(16px, 5vw, 48px)));
                  display: flex; flex-direction: column; }
.vb-modal--exam .vb-modal__title,
.vb-modal--exam .vb-exam-tabs,
.vb-modal--exam .vb-modal__actions { width: var(--vb-exam-col); max-width: none;
                                     margin-left: auto; margin-right: auto; }
.vb-exam-body { overflow-y: auto; flex: 1;
                /* 기본 스크롤바가 풀스크린에서 거슬림 — thin+테마 토큰 (라이트/다크 공용) */
                scrollbar-width: thin;
                scrollbar-color: var(--vb-border-strong) transparent; }
/* 스크롤 컨테이너는 풀폭 (스크롤바 = 화면 우측 끝), 콘텐츠만 칼럼 폭 중앙 */
.vb-modal--exam .vb-exam-body { width: 100%; }
.vb-modal--exam .vb-exam-body > * { width: var(--vb-exam-col); max-width: none;
                                    margin-left: auto; margin-right: auto; }
.vb-exam-q { padding: 10px 0; border-bottom: 1px solid var(--vb-border); }
.vb-exam-prompt { font-weight: 600; }
.vb-exam-context { margin: 6px 0; color: var(--vb-text-secondary);
                   font-style: italic; }
.vb-exam-choices { display: flex; flex-wrap: wrap; gap: 8px; margin-top: 8px; }
.vb-exam-chip { border: 1px solid var(--vb-border-strong); border-radius: 999px;
                padding: 6px 14px; background: none; cursor: pointer;
                font-size: 0.92rem; color: var(--vb-text-primary); }
.vb-exam-chip.is-on { background: var(--vb-aubergine);
                      color: var(--vb-text-on-accent);
                      border-color: transparent; }
.vb-exam-footer { display: flex; align-items: center; gap: 12px;
                  padding-top: 12px;
                  width: var(--vb-exam-col, 100%); margin-left: auto; margin-right: auto; }
.vb-exam-footer span { margin-right: auto; color: var(--vb-text-secondary);
                       font-size: 0.9rem; }
.vb-exam-mark { display: inline-block; width: 1.4em; font-weight: 700; }
.vb-exam-q--done .vb-exam-mark { color: var(--vb-aubergine); }

/* 교사 결과 패널 */
.vb-exam-tabs { display: flex; gap: 4px; margin-bottom: 8px; }
.vb-exam-tab { border: none; background: none; padding: 6px 12px;
               cursor: pointer; color: var(--vb-text-secondary);
               border-bottom: 2px solid transparent; }
.vb-exam-tab.is-on { color: var(--vb-text-primary);
                     border-bottom-color: var(--vb-aubergine); }
.vb-exam-row { display: flex; align-items: center; gap: 10px; flex-wrap: wrap;
               padding: 8px 0; border-bottom: 1px solid var(--vb-border);
               cursor: pointer; }
.vb-exam-row-meta { color: var(--vb-text-secondary); font-size: 0.9rem; }
.vb-exam-detail { flex-basis: 100%; padding: 6px 0 0 12px; }
.vb-exam-detail[hidden] { display: none; }   /* hidden vs flex 무력화 가드 */
.vb-exam-detail-row { display: flex; gap: 8px; padding: 2px 0;
                      font-size: 0.9rem; }
.vb-exam-rate { position: relative; height: 18px; margin-top: 6px;
                background: var(--vb-bg-tint);
                border-radius: 4px; overflow: hidden; }
.vb-exam-rate-bar { height: 100%; background: var(--vb-aubergine);
                    opacity: 0.45; }
.vb-exam-rate span { position: absolute; right: 6px; top: 0;
                     font-size: 0.78rem; line-height: 18px; }
