import base64 import mimetypes import threading import time from dataclasses import replace from pathlib import Path from random import Random import gradio as gr from clients import art_client_from_env, card_client_from_env, configure_mode from primitives import School from ui import ( CARD_PANEL_COUNT, HAND_PANEL_COUNT, RunState, board_html, boss_splash_uri, choose_draft_card_loading_steps, collect_ready_battle, collect_ready_pack, draft_screen_html, escape_html, log_html, new_run_shell, pass_turn_steps, play_hand_card_steps, queue_next_pack, refresh_art, ) ASSETS = Path(__file__).parent / "assets" # Encode a bundled image asset as a CSS-ready data URI (HF-Spaces safe, no static paths). def asset_data_uri(name: str, mime: str = "image/jpeg") -> str: encoded = base64.b64encode((ASSETS / name).read_bytes()).decode("ascii") return f"data:{mime};base64,{encoded}" # Encode an optional bundled image asset as a CSS-ready data URI. def optional_asset_data_uri(name: str) -> str: path = optional_asset_path(name) if path is None: return "" mime = mimetypes.guess_type(path.name)[0] or "image/jpeg" encoded = base64.b64encode(path.read_bytes()).decode("ascii") return f"data:{mime};base64,{encoded}" # Return the first matching asset path, accepting common image extensions. def optional_asset_path(name: str) -> Path | None: path = ASSETS / name if path.exists(): return path if path.suffix: for suffix in (".jpg", ".jpeg", ".png", ".webp"): candidate = path.with_suffix(suffix) if candidate.exists(): return candidate return None INN_BG = asset_data_uri("inn_bg.jpg") WOOD_BG = asset_data_uri("wood_board.jpg") HEAD = """ """ CSS = """ .gradio-container { background: linear-gradient(rgba(10, 6, 14, 0.82), rgba(6, 4, 9, 0.9)), url("__INN_BG__") center / cover fixed; color: #f4ead9; max-width: 100% !important; } footer { display: none !important; } .hidden-controls { display: none !important; } *, *::before, *::after { box-sizing: border-box; } gradio-app, body { background: linear-gradient(rgba(10, 6, 14, 0.82), rgba(6, 4, 9, 0.9)), url("__INN_BG__") center / cover fixed !important; overflow-x: hidden; } .gradio-container .block, .gradio-container .form, .gradio-container .gr-group, .gradio-container .styler { background: transparent !important; border: none !important; box-shadow: none !important; } button.primary { background: linear-gradient(180deg, #f2c24d, #a96f17) !important; color: #3a2403 !important; border: 2px solid #f6dd9a !important; font-weight: 800 !important; } #play-now-btn { width: auto !important; min-width: 170px !important; max-width: 220px !important; align-self: center !important; } #play-now-btn button { width: auto !important; min-width: 170px !important; padding: 10px 28px !important; } #start-draft-btn { width: auto !important; min-width: 150px !important; max-width: 200px !important; align-self: center !important; margin: 0 auto !important; } #start-draft-btn button { width: auto !important; min-width: 150px !important; padding: 10px 24px !important; } .tabras-title { display: flex; flex-direction: column; justify-content: center; align-items: center; text-align: center; gap: 26px; padding: 18vh 20px 0; } .tabras-title h1 { font-size: 92px; margin: 0; color: #f6f0ff; text-shadow: 0 0 30px rgba(160, 90, 255, 0.6), 0 3px 18px rgba(0, 0, 0, 0.65); } .tabras-sub { display: flex; flex-direction: column; gap: 10px; color: #efe6ff; text-shadow: 0 2px 10px rgba(0, 0, 0, 0.8); } .tabras-sub .subline { font-size: 21px; line-height: 1.4; white-space: nowrap; } #play-now-btn { margin-top: 26px !important; } .gradio-container .setup-panel { background: rgba(26, 20, 46, 0.82) !important; border: 2px solid rgba(190, 160, 255, 0.3) !important; box-shadow: 0 18px 48px rgba(0, 0, 0, 0.35) !important; padding: 18px !important; border-radius: 14px !important; max-width: 560px; margin: 8vh auto 0 !important; } .step-kicker { color: #ffd35c; font-size: 13px; font-weight: 900; letter-spacing: 0.12em; text-transform: uppercase; margin-bottom: 8px; } .setup-panel h2 { margin: 0 0 12px; color: #f6f0ff; font-size: 28px; } .choice-row button { min-height: 46px; } .selector-grid { display: grid; grid-template-columns: repeat(3, minmax(0, 1fr)); gap: 14px; width: min(980px, 92vw); margin: 0 auto; } .selector-panel { position: relative; min-height: 360px; border: 2px solid rgba(190, 160, 255, 0.34); border-radius: 14px; overflow: hidden; cursor: pointer; background: var(--selector-fallback); background-size: cover; background-position: center; box-shadow: 0 18px 42px rgba(0, 0, 0, 0.42); transition: transform 0.16s ease, border-color 0.16s ease, box-shadow 0.16s ease, filter 0.16s ease; } .selector-image { position: absolute; inset: 0; width: 100%; height: 100%; object-fit: cover; } .selector-panel::after { content: ""; position: absolute; inset: 0; background: linear-gradient(180deg, rgba(12, 8, 30, 0.06), rgba(12, 8, 30, 0.88)); pointer-events: none; } .selector-panel:hover { transform: translateY(-8px); border-color: #ffd35c; box-shadow: 0 0 28px rgba(255, 211, 92, 0.22), 0 24px 58px rgba(0, 0, 0, 0.52); filter: saturate(1.1); } .selector-label { position: absolute; left: 18px; right: 18px; bottom: 18px; color: #f6f0ff; font-size: 26px; font-weight: 900; text-shadow: 0 3px 14px rgba(0, 0, 0, 0.82); z-index: 2; } .selector-copy { position: absolute; inset: 0; display: flex; align-items: flex-end; padding: 18px; color: #f6f0ff; font-size: 17px; font-weight: 700; line-height: 1.35; opacity: 0; background: linear-gradient(180deg, rgba(8, 4, 20, 0.24), rgba(8, 4, 20, 0.88)); text-shadow: 0 2px 10px rgba(0, 0, 0, 0.9); transition: opacity 0.16s ease; z-index: 3; } .selector-panel:hover .selector-copy { opacity: 1; } @media (max-width: 900px) { .selector-grid { grid-template-columns: 1fr; } .selector-panel { min-height: 240px; } } /* ---- card faces ---- */ .tabras-card { position: relative; border-radius: 10px; padding: 26px 10px 8px; background: linear-gradient(180deg, #2a2347, #151028); border: 2px solid #8d7bd6; color: #f0eaff; box-shadow: 0 10px 28px rgba(0, 0, 0, 0.35); display: flex; flex-direction: column; gap: 4px; } .tabras-card.empty { opacity: 0.35; } .card-cost { position: absolute; top: -9px; left: -9px; width: 34px; height: 34px; border-radius: 50%; background: radial-gradient(circle at 35% 30%, #7fd4ff, #1668c9); border: 2px solid #d8f1ff; color: #fff; display: flex; align-items: center; justify-content: center; font-weight: 900; font-size: 17px; text-shadow: 0 1px 2px rgba(0, 0, 0, 0.6); z-index: 2; } .card-name { font-size: 13px; font-weight: 800; line-height: 1.15; min-height: 30px; } .school-fire .card-name { color: #ff9d80; } .school-ice .card-name { color: #8fdfff; } .school-earth .card-name { color: #aaf08a; } .tabras-card.school-fire { border-color: #e06a48; box-shadow: 0 10px 28px rgba(0, 0, 0, 0.35), 0 0 14px rgba(255, 110, 70, 0.22); } .tabras-card.school-ice { border-color: #4fb6e8; box-shadow: 0 10px 28px rgba(0, 0, 0, 0.35), 0 0 14px rgba(90, 200, 255, 0.22); } .tabras-card.school-earth { border-color: #6fc454; box-shadow: 0 10px 28px rgba(0, 0, 0, 0.35), 0 0 14px rgba(130, 230, 110, 0.22); } .card-art { height: 64px; border-radius: 6px; border: 1px solid rgba(200, 170, 255, 0.3); background: linear-gradient(135deg, rgba(80, 60, 140, 0.5), rgba(20, 40, 70, 0.55)); overflow: hidden; } .card-art.generated { border-style: solid; background-color: #0d0a16; background-repeat: no-repeat; background-size: cover; background-position: center; } .pending-art { position: relative; border-color: rgba(190, 160, 255, 0.28); } .pending-art::after { content: ""; position: absolute; inset: 0; background: linear-gradient(100deg, transparent 0%, rgba(255, 255, 255, 0.12) 42%, transparent 70%); transform: translateX(-120%); animation: art-sheen 1.8s ease-in-out infinite; } .school-art-fire { background: radial-gradient(circle at 40% 35%, rgba(255, 155, 80, 0.34), transparent 32%), linear-gradient(135deg, #3a1724, #180d1f 72%); } .school-art-ice { background: radial-gradient(circle at 45% 35%, rgba(125, 220, 255, 0.32), transparent 34%), linear-gradient(135deg, #172c4a, #111328 72%); } .school-art-earth { background: radial-gradient(circle at 45% 38%, rgba(170, 240, 138, 0.28), transparent 34%), linear-gradient(135deg, #20331f, #111328 72%); } @keyframes art-sheen { from { transform: translateX(-120%); } to { transform: translateX(120%); } } .card-rules { font-size: 11.5px; line-height: 1.3; min-height: 44px; color: #ece4ff; } .card-flavor { color: #b9a8e0; font-size: 10px; font-style: italic; border-top: 1px solid rgba(200, 170, 255, 0.2); padding-top: 4px; } /* ---- draft screen ---- */ .draft-board { width: min(1220px, calc(100vw - 56px)); min-height: 82vh; margin: 0 auto; display: flex; flex-direction: column; align-items: center; justify-content: center; gap: 24px; padding: 20px 18px; } .draft-banner { text-align: center; } .draft-banner h2 { margin: 0; color: #ffd35c; letter-spacing: 0.08em; text-transform: uppercase; text-shadow: 0 0 18px rgba(255, 211, 92, 0.45); font-size: 30px; } .draft-banner p { margin: 6px 0 0; color: #b9a8e0; font-size: 17px; } .draft-pack { display: grid; grid-template-columns: repeat(3, minmax(0, 300px)); gap: clamp(16px, 2vw, 30px); justify-content: center; align-items: stretch; width: 100%; } .draft-card { width: 100%; min-width: 0; min-height: 405px; cursor: pointer; transition: transform 0.15s ease, box-shadow 0.15s ease; animation: pack-in 0.5s ease backwards; } .draft-pack .draft-card:nth-child(2) { animation-delay: 0.1s; } .draft-pack .draft-card:nth-child(3) { animation-delay: 0.2s; } .draft-card:hover { transform: translateY(-10px) scale(1.045); box-shadow: 0 0 22px rgba(56, 232, 210, 0.5); } @keyframes pack-in { from { opacity: 0; transform: translateY(28px) scale(0.92); } to { opacity: 1; transform: none; } } .draft-pack .draft-card.fading { animation: pack-out 0.7s ease forwards; animation-delay: 0s; pointer-events: none; cursor: default; } .draft-pack .draft-card.picked { animation: picked-out 0.7s ease forwards; animation-delay: 0s; pointer-events: none; cursor: default; } @keyframes pack-out { from { opacity: 1; } to { opacity: 0; transform: translateY(-14px) scale(0.94); filter: grayscale(0.7); } } @keyframes picked-out { 30% { opacity: 1; transform: scale(1.08); box-shadow: 0 0 44px rgba(242, 194, 77, 0.95); } to { opacity: 0; transform: translateY(-34px) scale(1.02); box-shadow: 0 0 30px rgba(242, 194, 77, 0.6); } } .draft-card .card-name { font-size: 17px; min-height: 44px; } .draft-card .card-art { height: 112px; } .draft-card .card-rules { font-size: 13px; min-height: 64px; } .draft-card .card-flavor { font-size: 12px; } .starter-board { min-height: 82vh; gap: 22px; } .starter-grid { display: grid; grid-template-columns: repeat(3, minmax(190px, 1fr)); gap: 18px; width: min(920px, 92vw); } .starter-card { min-height: 260px; } .starter-card .card-name { font-size: 17px; min-height: 40px; } .starter-card .card-art { height: 70px; } .starter-card .card-rules { font-size: 13.5px; min-height: 46px; } .starter-card .card-flavor { font-size: 12px; } .loading-board { gap: 30px; } .draft-loading { width: min(520px, 86vw); display: flex; flex-direction: column; align-items: center; gap: 12px; padding: 20px 22px; border: 1px solid rgba(255, 211, 92, 0.35); border-radius: 12px; background: rgba(12, 8, 30, 0.6); box-shadow: 0 0 24px rgba(255, 211, 92, 0.12); } .loading-title { color: #ffd35c; font-size: 20px; font-weight: 900; letter-spacing: 0.08em; text-transform: uppercase; } .loading-subtitle { color: #b9a8e0; font-size: 13px; text-align: center; } .rules-screen { min-height: 74vh; display: flex; align-items: center; justify-content: center; padding: 24px; } .rules-card { width: min(780px, 92vw); border: 2px solid rgba(255, 211, 92, 0.32); border-radius: 16px; background: rgba(16, 10, 34, 0.82); box-shadow: 0 24px 70px rgba(0, 0, 0, 0.45), 0 0 28px rgba(255, 211, 92, 0.12); padding: 30px; } .rules-card h1 { margin: 0 0 12px; color: #ffd35c; letter-spacing: 0.08em; text-transform: uppercase; } .rules-card p { color: #d9d0f0; font-size: 16px; line-height: 1.45; } .reveal-screen { min-height: 64vh; display: flex; flex-direction: column; align-items: center; justify-content: center; gap: 12px; padding-top: 12px; text-align: center; animation: reveal-in 0.6s ease backwards; } #reveal-next-btn { width: auto !important; min-width: 150px !important; max-width: 200px !important; align-self: center !important; margin: 4px auto 0 !important; } #reveal-next-btn button { width: auto !important; min-width: 150px !important; padding: 9px 26px !important; } @keyframes reveal-in { from { opacity: 0; transform: scale(0.97); } to { opacity: 1; transform: none; } } .reveal-kicker { font-size: 15px; font-weight: 900; letter-spacing: 0.22em; text-transform: uppercase; color: #f06bff; text-shadow: 0 0 18px rgba(232, 91, 255, 0.55); } .reveal-art { width: min(440px, 86vw); height: min(440px, 60vh); border-radius: 16px; border: 3px solid #b14ad0; background-size: cover; background-position: center top; background-color: #160e26; box-shadow: 0 0 48px rgba(177, 74, 208, 0.45), 0 24px 60px rgba(0, 0, 0, 0.6); } .reveal-name { font-size: 40px; font-weight: 900; color: #f6f0ff; letter-spacing: 0.04em; text-shadow: 0 0 26px rgba(232, 91, 255, 0.5), 0 3px 12px rgba(0, 0, 0, 0.7); } .reveal-quote { font-size: 19px; font-style: italic; color: #d9c4ee; max-width: 540px; } .versus { display: flex; align-items: center; justify-content: center; gap: 28px; margin: 6px 0 14px; } .versus-side { display: flex; flex-direction: column; align-items: center; gap: 6px; } .versus-face { width: 132px; height: 132px; border-radius: 50% 50% 46% 46%; border: 4px solid #b08a4f; background-size: cover; background-position: center top; background-color: #1b1430; box-shadow: 0 12px 30px rgba(0, 0, 0, 0.55); } .versus-side:last-child .versus-face { border-color: #b14ad0; box-shadow: 0 0 26px rgba(232, 91, 255, 0.4), 0 12px 30px rgba(0, 0, 0, 0.55); } .versus-face-text { display: flex; align-items: center; justify-content: center; color: #f4d69b; font-weight: 900; font-size: 22px; } .versus-name { font-weight: 800; color: #f6f0ff; font-size: 18px; } .versus-tag { font-size: 12px; color: #b9a8e0; letter-spacing: 0.04em; } .versus-vs { font-size: 30px; font-weight: 900; color: #ffd35c; text-shadow: 0 0 16px rgba(255, 211, 92, 0.5); } .villain-story { text-align: center; color: #d9d0f0; font-style: italic; max-width: 540px; margin: 0 auto; } .rules-grid { display: grid; grid-template-columns: repeat(2, minmax(0, 1fr)); gap: 12px; margin-top: 18px; } .rule-tile { border: 1px solid rgba(190, 160, 255, 0.28); border-radius: 10px; padding: 14px; background: rgba(42, 35, 71, 0.62); } .rule-tile b { display: block; color: #f6f0ff; margin-bottom: 6px; } .rule-tile span { color: #b9a8e0; font-size: 14px; line-height: 1.35; } .loading-bar { width: 100%; height: 12px; overflow: hidden; border-radius: 999px; background: rgba(20, 14, 50, 0.95); border: 1px solid rgba(95, 220, 255, 0.35); } .loading-bar span { display: block; width: 40%; height: 100%; border-radius: inherit; background: linear-gradient(90deg, #38e8d2, #ffd35c, #ff9d80); animation: loading-sweep 1.15s ease-in-out infinite; } @keyframes loading-sweep { 0% { transform: translateX(-110%); } 100% { transform: translateX(260%); } } .deck-strip { display: flex; flex-wrap: wrap; gap: 7px; justify-content: center; max-width: 1100px; align-items: center; } .deck-strip-label { font-weight: 800; color: #f2c24d; margin-right: 6px; font-size: 18px; } .deck-chip { background: rgba(10, 6, 30, 0.5); border: 1px solid #5b4a8f; border-radius: 12px; padding: 3px 12px; font-size: 13px; } .deck-chip b { color: #38e8d2; } .deck-chip.anchor { border-color: #ffd35c; box-shadow: 0 0 8px rgba(255, 211, 92, 0.4); } @media (max-width: 900px) { .draft-board { min-height: auto; justify-content: flex-start; } .draft-pack { grid-template-columns: 1fr; max-width: 320px; } .draft-card { width: min(300px, 92vw); min-height: 405px; } .starter-grid { grid-template-columns: repeat(2, minmax(150px, 1fr)); } } /* ---- battle board ---- */ .board { position: relative; border: 4px solid #7a5630; border-radius: 22px; padding: 8px 14px; background: linear-gradient(rgba(24, 13, 5, 0.42), rgba(14, 7, 2, 0.6)), url("__WOOD_BG__") center / cover; box-shadow: inset 0 0 70px rgba(0, 0, 0, 0.62), 0 0 30px rgba(150, 95, 40, 0.25), 0 24px 60px rgba(0, 0, 0, 0.55); overflow: hidden; } .zone { position: relative; display: flex; align-items: center; justify-content: center; } .zone-center { display: flex; flex-direction: column; align-items: center; gap: 6px; flex: 1; } .piles { position: absolute; right: 8px; top: 50%; transform: translateY(-50%); display: flex; flex-direction: column; gap: 8px; } .pile { width: 64px; text-align: center; border: 2px solid #5b4a8f; border-radius: 8px; padding: 6px 0 3px; background: repeating-linear-gradient(45deg, #1c1535 0, #1c1535 6px, #251c45 6px, #251c45 12px); } .pile span { font-weight: 900; font-size: 18px; color: #ffd35c; } .pile label { display: block; font-size: 9px; letter-spacing: 0.12em; text-transform: uppercase; color: #9d8fc7; } .fatigue-warn { color: #ff8d7a; font-size: 10px; font-weight: 800; } .hero { display: flex; flex-direction: column; align-items: center; gap: 3px; } .hero-frame { position: relative; width: 124px; height: 124px; } .hero-face { width: 100%; height: 100%; border-radius: 50% 50% 46% 46%; border: 4px solid #b08a4f; background: radial-gradient(circle at 50% 30%, #4a3e72, #1b1430 75%); display: flex; align-items: center; justify-content: center; font-weight: 900; font-size: 22px; color: #f4d69b; box-shadow: 0 10px 26px rgba(0, 0, 0, 0.5); } .hero-face.portrait { background-size: cover; background-position: center top; } .hp-gem, .block-gem, .ward-gem { position: absolute; width: 42px; height: 42px; border-radius: 50%; display: flex; align-items: center; justify-content: center; font-weight: 900; font-size: 19px; color: #fff; text-shadow: 0 1px 2px #000; border: 3px solid #f3d492; } .hp-gem { right: -10px; bottom: -6px; background: radial-gradient(circle at 35% 30%, #e0524d, #8f1612); } .block-gem { left: -10px; bottom: -6px; background: radial-gradient(circle at 35% 30%, #c9cdd4, #5a626e); border-color: #eef2f7; } .ward-gem { left: -10px; top: -6px; background: radial-gradient(circle at 35% 30%, #8fd0ff, #1d4d8f); border-color: #d8f1ff; } .hero-name { font-weight: 800; color: #f4d69b; text-shadow: 0 2px 6px #000; } .hero-frame.hit { animation: hero-shake 0.45s ease; } .hero-frame.hit .hp-gem { animation: gem-flash 0.7s ease; } @keyframes hero-shake { 0%, 100% { transform: translateX(0); } 20% { transform: translateX(-7px) rotate(-2deg); } 40% { transform: translateX(6px) rotate(2deg); } 60% { transform: translateX(-4px); } 80% { transform: translateX(3px); } } @keyframes gem-flash { 0%, 100% { filter: none; } 30% { filter: brightness(2.2) saturate(1.6); transform: scale(1.25); } } .dmg-pop { position: absolute; left: 50%; top: 22%; transform: translateX(-50%); font-size: 34px; font-weight: 900; color: #ff5340; text-shadow: 0 0 12px rgba(255, 60, 30, 0.9), 0 2px 3px #000; pointer-events: none; z-index: 50; animation: dmg-float 1.4s ease-out forwards; } @keyframes dmg-float { from { opacity: 0; transform: translateX(-50%) translateY(16px) scale(0.5); } 25% { opacity: 1; transform: translateX(-50%) translateY(-8px) scale(1.3); } to { opacity: 0; transform: translateX(-50%) translateY(-54px) scale(1); } } .chips { display: flex; gap: 6px; min-height: 18px; } .chip { font-size: 10px; font-weight: 700; border-radius: 10px; padding: 1px 8px; border: 1px solid; } .chip.charge { color: #ffd98c; border-color: #c9982f; background: rgba(120, 85, 10, 0.35); } .chip.weak { color: #a8c6ff; border-color: #4a6fb5; background: rgba(30, 55, 110, 0.35); } .chip.vuln { color: #ffb3c2; border-color: #b54a64; background: rgba(110, 25, 45, 0.35); } .enemy-hand { display: flex; justify-content: center; margin-bottom: -10px; } .enemy-card-back { width: 52px; height: 72px; border-radius: 6px; border: 2px solid #7a5fc0; margin: 0 -10px; background: repeating-linear-gradient(45deg, #2a1745 0, #2a1745 7px, #371d5c 7px, #371d5c 14px); box-shadow: 0 6px 14px rgba(0, 0, 0, 0.4); transform: rotate(3deg); } .enemy-card-back:nth-child(odd) { transform: rotate(-3deg); } .battlefield { position: relative; min-height: 118px; margin: 8px 36px; border-radius: 14px; background: rgba(10, 6, 30, 0.35); box-shadow: inset 0 0 30px rgba(0, 0, 0, 0.35); display: flex; flex-direction: column; justify-content: space-between; padding: 6px 150px 6px 16px; } .round-banner { position: absolute; left: 16px; top: 50%; transform: translateY(-50%); font-weight: 800; letter-spacing: 0.12em; text-transform: uppercase; color: #cfc3f2; font-size: 13px; } .end-turn { position: absolute; right: 14px; top: 50%; transform: translateY(-50%); padding: 13px 24px; border-radius: 26px; border: 3px solid #f6dd9a; background: linear-gradient(180deg, #f2c24d, #a96f17); color: #3a2403; font-weight: 900; letter-spacing: 0.06em; cursor: pointer; box-shadow: 0 6px 18px rgba(0, 0, 0, 0.45); } .end-turn:hover { filter: brightness(1.12); } .end-turn.pulse { animation: end-pulse 1.2s ease-in-out infinite; } @keyframes end-pulse { 0%, 100% { box-shadow: 0 6px 18px rgba(0, 0, 0, 0.45); } 50% { box-shadow: 0 0 26px rgba(246, 221, 154, 0.9), 0 6px 18px rgba(0, 0, 0, 0.45); } } .board.quake { animation: board-quake 0.5s ease; } @keyframes board-quake { 0%, 100% { transform: translate(0, 0); } 20% { transform: translate(-6px, 3px); } 40% { transform: translate(5px, -4px); } 60% { transform: translate(-4px, 2px); } 80% { transform: translate(3px, -2px); } } .pending-row { display: flex; gap: 8px; min-height: 34px; justify-content: center; } .showcase { display: flex; gap: 18px; justify-content: center; align-items: flex-start; min-height: 0; perspective: 900px; } .play-slot { display: flex; flex-direction: column; align-items: center; gap: 4px; } .play-slot.fresh { animation: play-in 0.5s cubic-bezier(0.2, 1.4, 0.4, 1) backwards; } .play-label { font-size: 10px; font-weight: 800; letter-spacing: 0.14em; text-transform: uppercase; padding: 1px 10px; border-radius: 10px; } .play-label.you { color: #7df5e2; background: rgba(20, 90, 80, 0.45); border: 1px solid #2aa896; } .play-label.boss { color: #f0b8ff; background: rgba(110, 30, 130, 0.45); border: 1px solid #b14ad0; } .played-card { width: 132px; } .played-card.player-play { box-shadow: 0 0 22px rgba(56, 232, 210, 0.6); } .played-card.enemy-play { box-shadow: 0 0 22px rgba(232, 91, 255, 0.6); border-color: #b14ad0; } @keyframes play-in { from { transform: translateY(120px) rotateY(90deg) scale(0.4); opacity: 0; } 60% { transform: translateY(-6px) rotateY(0deg) scale(1.12); opacity: 1; } to { transform: none; opacity: 1; } } .token { display: flex; flex-direction: column; align-items: center; padding: 2px 10px; border-radius: 10px; font-size: 11px; border: 2px solid; line-height: 1.1; } .token b { font-size: 16px; } .token.bomb { border-color: #e0524d; background: rgba(120, 20, 10, 0.5); color: #ffb9a8; } .token.burn { border-color: #f2913d; background: rgba(140, 70, 10, 0.45); color: #ffd9a8; } .mana-bar { display: flex; align-items: center; gap: 6px; } .mana { width: 19px; height: 19px; transform: rotate(45deg); border-radius: 5px; background: #1d2c3c; border: 2px solid #2f4a66; } .mana.filled { background: radial-gradient(circle at 35% 30%, #7fd4ff, #1668c9); border-color: #bfe9ff; box-shadow: 0 0 8px rgba(80, 170, 255, 0.8); } .mana-count { margin-left: 8px; font-weight: 800; color: #bfe9ff; } .hand-fan { display: flex; justify-content: center; align-items: flex-end; padding: 26px 0 6px; min-height: 240px; } .hand-card { width: 150px; margin: 0 -16px; transform: rotate(var(--rot)) translateY(var(--ty)); transform-origin: 50% 130%; transition: transform 0.15s ease, box-shadow 0.15s ease; cursor: pointer; box-shadow: 0 0 14px rgba(56, 232, 210, 0.55); border-color: #45e8c8; } .hand-card:hover { transform: rotate(0deg) translateY(-48px) scale(1.45); z-index: 99 !important; box-shadow: 0 0 24px rgba(56, 232, 210, 0.5), 0 24px 50px rgba(0, 0, 0, 0.6); } .hand-card.unplayable { filter: grayscale(0.8) brightness(0.7); cursor: default; box-shadow: 0 10px 28px rgba(0, 0, 0, 0.35); border-color: #4a3f6e; } .hand-card.unplayable:hover { box-shadow: 0 24px 50px rgba(0, 0, 0, 0.6); } .hand-card.launching { transform: translateY(-170px) scale(1.08) rotate(0deg) !important; opacity: 0; transition: transform 0.3s ease-in, opacity 0.3s ease-in; pointer-events: none; } .round-splash { position: absolute; inset: 0; display: flex; flex-direction: column; gap: 12px; align-items: center; justify-content: center; z-index: 150; pointer-events: none; border-radius: 18px; animation: splash-bg 1.9s ease forwards; } @keyframes splash-bg { 0% { background: rgba(6, 3, 18, 0); } 18% { background: rgba(6, 3, 18, 0.78); } 72% { background: rgba(6, 3, 18, 0.78); } 100% { background: rgba(6, 3, 18, 0); } } .splash-round { font-size: 50px; font-weight: 900; letter-spacing: 0.3em; color: #ffd35c; text-shadow: 0 0 34px rgba(255, 211, 92, 0.9), 0 4px 8px #000; animation: splash 1.9s ease forwards; } .splash-initiative { font-size: 24px; font-weight: 900; letter-spacing: 0.26em; animation: splash 1.65s ease 0.25s both; } .splash-initiative.you { color: #38e8d2; text-shadow: 0 0 24px rgba(56, 232, 210, 0.85); } .splash-initiative.boss { color: #f06bff; text-shadow: 0 0 24px rgba(232, 91, 255, 0.85); } @keyframes splash { 0% { opacity: 0; transform: scale(2.2); } 25% { opacity: 1; transform: scale(1); } 70% { opacity: 1; } 100% { opacity: 0; transform: scale(0.92); } } .boss-thinking { position: absolute; left: 50%; bottom: -6px; transform: translateX(-50%); z-index: 60; padding: 5px 16px; border-radius: 14px; background: rgba(20, 10, 36, 0.94); border: 1px solid #b14ad0; color: #f0c4ff; font-weight: 700; font-size: 12px; letter-spacing: 0.08em; white-space: nowrap; animation: think-pulse 1.1s ease-in-out infinite; } @keyframes think-pulse { 0%, 100% { opacity: 0.55; transform: translateX(-50%) scale(1); } 50% { opacity: 1; transform: translateX(-50%) scale(1.06); } } .board.thinking .hero.enemy .hero-face { box-shadow: 0 0 28px rgba(232, 91, 255, 0.6); } .winner-banner { position: absolute; inset: 0; display: flex; flex-direction: column; gap: 16px; align-items: center; justify-content: center; background: rgba(8, 5, 22, 0.78); z-index: 200; border-radius: 18px; } .winner-banner h1 { font-size: 64px; letter-spacing: 0.2em; margin: 0; animation: banner-in 0.7s cubic-bezier(0.2, 1.6, 0.4, 1) backwards; } @keyframes banner-in { from { transform: scale(2.6); opacity: 0; letter-spacing: 0.6em; } to { transform: scale(1); opacity: 1; letter-spacing: 0.2em; } } .winner-banner.victory h1 { color: #ffd98c; text-shadow: 0 0 30px rgba(255, 200, 90, 0.7); } .winner-banner.defeat h1 { color: #ff7a6a; text-shadow: 0 0 30px rgba(255, 70, 40, 0.6); } .winner-banner.draw h1 { color: #b9a8e0; } .new-run { padding: 12px 30px; border-radius: 24px; border: 3px solid #f6dd9a; background: linear-gradient(180deg, #f2c24d, #a96f17); font-weight: 900; cursor: pointer; color: #3a2403; } .game-over .hand-fan, .game-over .end-turn { pointer-events: none; } .log-panel { background: rgba(18, 14, 38, 0.85) !important; border: 2px solid #5b4a8f; border-radius: 14px; padding: 8px; } .log-scroll { display: flex; flex-direction: column-reverse; gap: 4px; max-height: 600px; overflow-y: auto; font-size: 11.5px; color: #d9d0f0; } .log-line { padding: 4px 8px; border-left: 3px solid transparent; border-radius: 4px; line-height: 1.35; } .log-line b { color: #ffd35c; font-size: 12.5px; } .log-owner { display: block; font-size: 9px; letter-spacing: 0.12em; text-transform: uppercase; opacity: 0.7; } .log-rules { display: block; font-size: 10.5px; opacity: 0.85; } .log-play.log-you { border-left-color: #38e8d2; background: rgba(30, 120, 110, 0.22); } .log-play.log-boss { border-left-color: #e85bff; background: rgba(120, 40, 140, 0.22); } .log-round { text-align: center; color: #ffd35c; font-weight: 800; letter-spacing: 0.06em; margin-top: 8px; border-top: 1px solid rgba(255, 211, 92, 0.35); padding-top: 6px; } .log-winner { text-align: center; color: #ffd35c; font-weight: 900; font-size: 13px; border: 1px solid #ffd35c; background: rgba(255, 211, 92, 0.14); } .log-muted { opacity: 0.55; font-style: italic; } .log-draft { color: #aaf08a; } .log-scroll > .log-line:first-child { box-shadow: inset 0 0 0 1px rgba(255, 211, 92, 0.35); } """ CSS = CSS.replace("__INN_BG__", INN_BG).replace("__WOOD_BG__", WOOD_BG) # Build all Gradio components for the Tabras app. def build_app() -> gr.Blocks: with gr.Blocks(title="Tabras") as app: state = gr.State(None) screen_state = gr.State("title") with gr.Group(visible=True, elem_id="title-screen") as title_group: gr.HTML( "