/* tools/canciones-boda — curated wedding-songs player *//* Override the layout's default 1100px container width on this tool only,
   so the song grid fits 5–6 cards per row on wide screens instead of 3.
   :has() targets the wrapping <section class="container"> in layout.php
   without needing a body class. */main > section.container:has(.cb-tool){max-width:1600px}.cb-tool{max-width:1600px;padding:28px}@media (max-width: 640px){.cb-tool{padding:20px}}/* Keep filters within reach as the user scrolls through 100+ song cards.
   The shared .tools-nav is sticky by default (theme.css:127); we drop that
   on this tool so the filters get the prime sticky slot instead. */body:has(.cb-tool) .tools-nav{position:static}.cb-controls{display:flex;flex-direction:column;gap:12px;margin-bottom:20px;position:sticky;top:0;z-index:50;background:var(--bg-card);padding:14px 6px 12px;margin-left:-6px;margin-right:-6px;border-radius:12px;box-shadow:0 6px 18px -14px rgba(0, 0, 0, 0.18)}.cb-filter-row{display:flex;align-items:center;gap:12px;flex-wrap:wrap}.cb-filter-label{font-size:0.78rem;text-transform:uppercase;letter-spacing:0.06em;color:var(--text-muted);font-weight:600;min-width:70px}.cb-tabs{display:flex;flex-wrap:wrap;gap:6px}.cb-tab{background:transparent;border:1px solid var(--border-medium);color:var(--text-body);padding:8px 14px;border-radius:999px;font-size:0.88rem;font-weight:500;cursor:pointer;font-family:Inter,sans-serif;transition:all 0.18s ease;white-space:nowrap}.cb-tab:hover{border-color:var(--primary);color:var(--primary)}.cb-tab.is-active{background:linear-gradient(135deg, var(--gradient-start), var(--gradient-end));color:#fff;border-color:transparent}.cb-meta{color:var(--text-muted);font-size:0.86rem;margin:4px 0 16px}.cb-empty{text-align:center;color:var(--text-muted);padding:40px 16px}.cb-grid{display:grid;grid-template-columns:repeat(auto-fill, minmax(260px, 1fr));gap:16px}.cb-card{background:var(--bg-card);border:1px solid var(--border-light);border-radius:14px;overflow:hidden;display:flex;flex-direction:column;transition:background-color 0.18s ease,border-color 0.18s ease,box-shadow 0.18s ease,transform 0.18s ease}.cb-card:hover{background:var(--primary-soft);box-shadow:0 8px 24px -16px rgba(255, 107, 53, 0.35);transform:translateY(-1px)}.cb-card.is-playing{border-color:var(--primary);box-shadow:0 10px 28px -14px rgba(255, 107, 53, 0.4)}.cb-card-thumb-wrap{position:relative}.cb-card-thumb{position:relative;aspect-ratio:16/9;background:#f5f0ec;border:0;padding:0;cursor:pointer;overflow:hidden;display:block;width:100%}.cb-card-img{width:100%;height:100%;object-fit:cover;display:block}.cb-card-play-icon{position:absolute;inset:0;display:flex;align-items:center;justify-content:center;background:rgba(0, 0, 0, 0.18);opacity:0;transition:opacity 0.2s ease}.cb-card-play-icon::before{content:'';width:0;height:0;border-left:18px solid #fff;border-top:11px solid transparent;border-bottom:11px solid transparent;margin-left:4px;filter:drop-shadow(0 2px 6px rgba(0,0,0,0.4))}.cb-card:hover .cb-card-play-icon,.cb-card.is-playing .cb-card-play-icon,.cb-card.is-loading .cb-card-play-icon{opacity:1}.cb-card.is-playing .cb-card-play-icon{background:rgba(255, 107, 53, 0.45)}.cb-card.is-playing .cb-card-play-icon::before{border-left:0;border-top:0;border-bottom:0;width:5px;height:18px;margin-left:0;background:#fff;box-shadow:9px 0 0 #fff}/* Loader — spinner overlay shown while we wait for YouTube to actually
   begin playback (mirrors editor-invitation.css → .inv-song.is-loading). */.cb-card-loader{position:absolute;inset:0;display:none;align-items:center;justify-content:center;background:rgba(0, 0, 0, 0.45);pointer-events:none}.cb-card.is-loading .cb-card-loader{display:flex}.cb-card.is-loading .cb-card-play-icon{display:none}.cb-card-loader::before{content:'';width:36px;height:36px;border-radius:50%;border:3px solid rgba(255, 255, 255, 0.35);border-top-color:#fff;animation:cbSpin 0.85s linear infinite}@keyframes cbSpin{to{transform:rotate(360deg)}}/* Equalizer "wave" overlay — animated bars in the lower-left of the
   thumbnail while the song is playing, so the active card reads as alive
   from across the grid. Same visual language as .inv-song-icon--bars in
   the invitation editor. */.cb-card-wave{position:absolute;left:12px;bottom:12px;display:none;align-items:flex-end;gap:3px;height:22px;padding:4px 8px;border-radius:999px;background:rgba(0, 0, 0, 0.45);backdrop-filter:blur(4px);-webkit-backdrop-filter:blur(4px);pointer-events:none}.cb-card.is-playing .cb-card-wave{display:inline-flex}.cb-card-wave span{display:block;width:3px;height:100%;border-radius:2px;background:#fff;transform-origin:bottom center;animation:cbWaveBar 1s ease-in-out infinite}.cb-card-wave span:nth-child(1){animation-delay:0s}.cb-card-wave span:nth-child(2){animation-delay:0.12s}.cb-card-wave span:nth-child(3){animation-delay:0.24s}.cb-card-wave span:nth-child(4){animation-delay:0.36s}.cb-card-wave span:nth-child(5){animation-delay:0.48s}@keyframes cbWaveBar{0%,100%{transform:scaleY(0.25)}50%{transform:scaleY(1)}}@media (prefers-reduced-motion: reduce){.cb-card-wave span{animation:none;transform:scaleY(0.6)}.cb-card-loader::before{animation-duration:1.6s}}.cb-card-body{padding:14px 16px 16px;display:flex;flex-direction:column;gap:8px;flex:1}.cb-card-title{font-family:'Rubik',sans-serif;font-size:1.05rem;color:var(--text-heading);line-height:1.3;margin:0}.cb-card-artist{color:var(--text-muted);font-size:0.86rem;margin:0}.cb-card-actions{display:flex;align-items:center;gap:8px;margin-top:auto;padding-top:8px;flex-wrap:wrap;opacity:0;transition:opacity 0.15s ease}.cb-card:hover .cb-card-actions,.cb-card:focus-within .cb-card-actions{opacity:1}@media (hover: none){.cb-card-actions{opacity:1}}.cb-card-add{background:transparent;border:1px solid var(--border-medium);color:var(--text-body);padding:6px 12px;border-radius:999px;font-size:0.78rem;font-weight:600;cursor:pointer;font-family:Inter,sans-serif;transition:all 0.15s ease}.cb-card-add:hover{border-color:var(--primary);color:var(--primary)}.cb-card-add[aria-pressed="true"]{background:var(--primary-soft);color:var(--primary);border-color:var(--primary-soft-border)}.cb-card-add .cb-card-add-on{display:none}.cb-card-add[aria-pressed="true"] .cb-card-add-default{display:none}.cb-card-add[aria-pressed="true"] .cb-card-add-on{display:inline}.cb-card-use{color:var(--primary);text-decoration:none;font-size:0.82rem;font-weight:600;margin-left:auto}.cb-card-use:hover{text-decoration:underline}/* Spotify chip — sits on the thumbnail (top-right) so it's visually
   detached from the "Usar en invitación" CTA in the actions row.
   Labelled "Spotify" so the icon's purpose is unambiguous. */.cb-card-spotify{position:absolute;top:8px;right:8px;display:inline-flex;align-items:center;gap:5px;padding:5px 10px 5px 8px;background:rgba(0, 0, 0, 0.62);color:#fff;border-radius:999px;font-size:0.72rem;font-weight:600;text-decoration:none;line-height:1;backdrop-filter:blur(4px);-webkit-backdrop-filter:blur(4px);transition:background 0.15s ease,transform 0.15s ease;z-index:2}.cb-card-spotify svg{color:#1DB954;flex-shrink:0}.cb-card-spotify:hover{background:rgba(0, 0, 0, 0.78);transform:translateY(-1px)}#cb-yt-host{position:fixed;width:1px;height:1px;overflow:hidden;opacity:0;pointer-events:none;bottom:0;right:0}/* ── Playlist modal (Spotify-style track list) ───────────────────────── */.cb-playlist-backdrop[hidden],.cb-playlist[hidden]{display:none}.cb-playlist-backdrop{position:fixed;inset:0;background:rgba(20, 14, 10, 0.45);backdrop-filter:blur(2px);-webkit-backdrop-filter:blur(2px);z-index:79;animation:cbPlBackdropIn 0.18s ease forwards}@keyframes cbPlBackdropIn{from{opacity:0}to{opacity:1}}.cb-playlist{position:fixed;top:50%;left:50%;transform:translate(-50%, -50%);width:min(520px, calc(100vw - 32px));max-height:min(80vh, 720px);background:var(--bg-card, #fff);border-radius:20px;box-shadow:0 40px 80px -20px rgba(0,0,0,0.35),0 0 0 1px rgba(0,0,0,0.04);z-index:80;display:flex;flex-direction:column;overflow:hidden;animation:cbPlIn 0.22s cubic-bezier(0.22, 1, 0.36, 1) forwards}@keyframes cbPlIn{from{opacity:0;transform:translate(-50%, -48%) scale(0.97)}to{opacity:1;transform:translate(-50%, -50%) scale(1)}}.cb-playlist-header{display:flex;align-items:center;justify-content:space-between;padding:20px 22px 14px;border-bottom:1px solid var(--border-light)}.cb-playlist-title{margin:0;font-family:'Rubik',sans-serif;color:var(--text-heading);font-size:1.25rem;font-weight:600;display:flex;align-items:center;gap:10px}.cb-playlist-count{background:var(--primary-soft);color:var(--primary);font-size:0.78rem;padding:2px 9px;border-radius:999px;font-weight:700}.cb-playlist-close{background:transparent;border:0;color:var(--text-muted);cursor:pointer;line-height:0;padding:6px;border-radius:8px;transition:background 0.15s ease,color 0.15s ease}.cb-playlist-close:hover{background:var(--bg-section-warm);color:var(--text-heading)}.cb-playlist-list{list-style:none;margin:0;padding:6px 10px;overflow-y:auto;flex:1;counter-reset:cbpl;scrollbar-width:thin}.cb-playlist-list::-webkit-scrollbar{width:8px}.cb-playlist-list::-webkit-scrollbar-thumb{background:rgba(0,0,0,0.12);border-radius:4px}.cb-playlist-empty{margin:0;padding:36px 20px;text-align:center;color:var(--text-muted);font-size:0.92rem}/* ── Track row ────────────────────────────────────────────────────────── */.cb-pl-track{counter-increment:cbpl;display:grid;grid-template-columns:28px 44px 1fr auto auto;align-items:center;gap:12px;padding:6px 10px;border-radius:10px;transition:background 0.15s ease;position:relative;cursor:pointer;user-select:none}.cb-pl-track:hover{background:var(--bg-section-warm)}.cb-pl-track:focus-visible{outline:2px solid var(--primary);outline-offset:2px}.cb-pl-track.is-playing{background:var(--primary-soft)}/* Use ::before via the index span for the number, swap to play/pause on hover. */.cb-pl-track-index{grid-column:1;width:28px;text-align:center;color:var(--text-muted);font-size:0.95rem;font-variant-numeric:tabular-nums;font-weight:500}.cb-pl-track-index::before{content:counter(cbpl)}.cb-pl-track.is-playing .cb-pl-track-index{color:var(--primary);font-weight:600}.cb-pl-track-thumb-wrap{grid-column:2;position:relative;width:44px;height:44px;border-radius:6px;overflow:hidden;background:#f1ebe6;flex-shrink:0}.cb-pl-track-thumb{width:100%;height:100%;object-fit:cover;display:block}.cb-pl-track-overlay{position:absolute;inset:0;background:rgba(0, 0, 0, 0.5);opacity:0;transition:opacity 0.15s ease}.cb-pl-track:hover .cb-pl-track-overlay,.cb-pl-track.is-playing .cb-pl-track-overlay,.cb-pl-track.is-loading .cb-pl-track-overlay{opacity:1}.cb-pl-track-icon{position:absolute;inset:0;display:none;align-items:center;justify-content:center;color:#fff;pointer-events:none}.cb-pl-track:hover .cb-pl-track-icon-play{display:flex}.cb-pl-track.is-playing .cb-pl-track-icon-play,.cb-pl-track.is-loading .cb-pl-track-icon-play,.cb-pl-track:hover.is-playing .cb-pl-track-icon-play{display:none}.cb-pl-track.is-playing .cb-pl-track-icon-pause{display:flex}.cb-pl-track.is-playing.is-loading .cb-pl-track-icon-pause{display:none}.cb-pl-track-loader{display:none}.cb-pl-track.is-loading .cb-pl-track-loader{display:flex}.cb-pl-track-loader::before{content:'';width:18px;height:18px;border-radius:50%;border:2px solid rgba(255,255,255,0.4);border-top-color:#fff;animation:cbSpin 0.85s linear infinite}.cb-pl-track-meta{grid-column:3;min-width:0;display:flex;flex-direction:column;gap:2px;line-height:1.25}.cb-pl-track-title{color:var(--text-heading);font-size:0.92rem;font-weight:600;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.cb-pl-track.is-playing .cb-pl-track-title{color:var(--primary)}.cb-pl-track-artist{color:var(--text-muted);font-size:0.8rem;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.cb-pl-track-wave{grid-column:4;display:none;align-items:flex-end;gap:2px;height:16px;width:18px}.cb-pl-track.is-playing .cb-pl-track-wave{display:inline-flex}.cb-pl-track-wave span{display:block;width:2px;height:100%;border-radius:1px;background:var(--primary);transform-origin:bottom center;animation:cbWaveBar 1s ease-in-out infinite}.cb-pl-track-wave span:nth-child(1){animation-delay:0s}.cb-pl-track-wave span:nth-child(2){animation-delay:0.15s}.cb-pl-track-wave span:nth-child(3){animation-delay:0.3s}.cb-pl-track-wave span:nth-child(4){animation-delay:0.45s}.cb-pl-track-remove{grid-column:5;background:transparent;border:0;color:var(--text-muted);cursor:pointer;padding:6px;border-radius:6px;line-height:0;opacity:0;transition:opacity 0.15s ease,background 0.15s ease,color 0.15s ease}.cb-pl-track:hover .cb-pl-track-remove,.cb-pl-track:focus-within .cb-pl-track-remove{opacity:1}.cb-pl-track-remove:hover{background:rgba(0,0,0,0.06);color:var(--primary)}@media (hover: none){.cb-pl-track-remove{opacity:1}}@media (prefers-reduced-motion: reduce){.cb-pl-track-wave span{animation:none;transform:scaleY(0.6)}.cb-pl-track-loader::before{animation-duration:1.6s}.cb-playlist,.cb-playlist-backdrop{animation:none}}.cb-playlist-actions{display:flex;flex-direction:column;gap:8px;padding:14px 18px 18px;border-top:1px solid var(--border-light)}.cb-playlist-actions .btn-ghost,.cb-playlist-actions .btn-primary{padding:12px 16px;font-size:0.92rem;text-align:center}.cb-playlist-secondary{text-align:center;color:var(--text-muted);font-size:0.82rem;text-decoration:none;padding:2px 0}.cb-playlist-secondary:hover{color:var(--primary);text-decoration:underline}.cb-playlist-toggle{position:fixed;right:16px;bottom:16px;background:linear-gradient(135deg, var(--gradient-start), var(--gradient-end));color:#fff;width:56px;height:56px;border-radius:50%;border:0;cursor:pointer;display:flex;align-items:center;justify-content:center;flex-direction:column;box-shadow:0 12px 30px -12px rgba(255, 107, 53, 0.5);z-index:70;font-family:Inter,sans-serif}.cb-playlist-toggle-icon{font-size:1.4rem;line-height:1}.cb-playlist-toggle-count{font-size:0.7rem;font-weight:700;background:rgba(255,255,255,0.25);padding:1px 6px;border-radius:999px;margin-top:1px}@media (max-width: 540px){.cb-playlist{width:calc(100vw - 16px);max-height:88vh;border-radius:16px}.cb-playlist-header{padding:16px 16px 12px}.cb-playlist-actions{padding:12px 14px 14px}}/* ── Article: per-moment static song lists for SEO ───────────────────── */.cb-article-songs{list-style:none;margin:16px 0 24px;padding:0;display:grid;grid-template-columns:repeat(auto-fill, minmax(260px, 1fr));gap:6px 18px}.cb-article-songs li{border-bottom:1px dashed var(--border-light);padding:8px 0;font-size:0.95rem}.cb-article-songs li strong{color:var(--text-heading)}.cb-article-songs li span{color:var(--text-muted)}/* Canva-style compact toast (mirrors the app's .toast--compact) */.cb-toast{position:fixed;top:16px;left:50%;transform:translate(-50%, -20px);background:#1f2937;color:#fff;border-radius:12px;padding:10px 14px;box-shadow:0 10px 30px rgba(0, 0, 0, 0.25);display:flex;align-items:center;gap:10px;font-size:0.85rem;font-weight:600;line-height:1.25;z-index:10002;opacity:0;pointer-events:none;max-width:360px}.cb-toast.is-show{animation:cbToastIn 0.32s cubic-bezier(0.22, 1, 0.36, 1) forwards}.cb-toast.is-hide{animation:cbToastOut 0.24s ease forwards}.cb-toast-icon{width:18px;height:18px;flex:0 0 18px;color:#34d399}@keyframes cbToastIn{to{transform:translate(-50%, 0);opacity:1}}@keyframes cbToastOut{from{transform:translate(-50%, 0);opacity:1}to{transform:translate(-50%, -20px);opacity:0}}