// Global error capture for bug reports window.capturedErrors = []; window.onerror = function (message, source, lineno, colno, error) { window.capturedErrors.push({ type: 'error', message: message, source: source, line: lineno, col: colno, stack: error ? error.stack : null, timestamp: new Date().toISOString() }); // Keep only last 20 errors if (window.capturedErrors.length > 20) window.capturedErrors.shift(); }; window.onunhandledrejection = function (event) { window.capturedErrors.push({ type: 'unhandledrejection', message: event.reason ? event.reason.message || String(event.reason) : 'Unknown', stack: event.reason ? event.reason.stack : null, timestamp: new Date().toISOString() }); if (window.capturedErrors.length > 20) window.capturedErrors.shift(); }; // Preload all texticon images at startup for instant tooltip display const isStaticHost = window.location.hostname.includes('github.io'); // Helper to get the base URL (handles GitHub Pages subdirectories) const getAppBaseUrl = () => { const loc = window.location; // If on github.io and has a path like /reponame/, use that if (loc.hostname.includes('github.io')) { const parts = loc.pathname.split('/'); if (parts.length > 2 && parts[1]) { return `/${parts[1]}/`; } } return '/'; }; const fixImg = (path) => { if (!path) return 'img/icon_blade.png'; let url = path; // Normalize: remove leading slash if (url.startsWith('/')) url = url.substring(1); // Check if it already starts with img/ to avoid double prefixing if (!url.startsWith('img/') && !url.startsWith('http')) { url = 'img/' + url; } // Robust static host detection const isGithub = window.location.hostname.includes('github') || window.location.hostname.includes('lovecasim'); if (isGithub && url.toLowerCase().endsWith('.png')) { url = url.replace(/\.png$/i, '.webp'); } // Prefix with application base URL to handle subdirectories correctly const base = getAppBaseUrl(); if (base !== '/' && !url.startsWith('http')) { url = base + url; } return url; }; const ICON_DATA_URIs = { "center": "", "heart_00": "", "heart_01": "", "heart_02": "", "heart_03": "", "heart_04": "", "heart_05": "", "heart_06": "", "icon_all": "", "icon_blade": "", "icon_b_all": "", "icon_draw": "", "icon_energy": "", "icon_score": "", "jidou": "", "jyouji": "", "kidou": "", "live_start": "", "live_success": "", "toujyou": "", "turn1": "" }; // Phase enum from game_state.py // Phase enum from logic.rs (Rust) const Phase = { SETUP: -2, "-2": "SETUP", MULLIGAN_P1: -1, "-1": "MULLIGAN_P1", MULLIGAN_P2: 0, "0": "MULLIGAN_P2", ACTIVE: 1, "1": "ACTIVE", ENERGY: 2, "2": "ENERGY", DRAW: 3, "3": "DRAW", MAIN: 4, "4": "MAIN", LIVE_SET: 5, "5": "LIVE_SET", PERFORMANCE_P1: 6, "6": "PERFORMANCE_P1", PERFORMANCE_P2: 7, "7": "PERFORMANCE_P2", LIVE_RESULT: 8, "8": "LIVE_RESULT", TERMINAL: 9, "9": "TERMINAL" }; let offlineMode = false; let wasmAdapter = null; let selectedIndices = []; // For multiphase selection (Mulligan, Choice) let selectedPerfTurn = -1; // -1 means current/latest, otherwise specific turn number let currentLang = 'jp'; // Default Japanese let hotseatMode = false; let perspectivePlayer = 0; // 0 or 1 let roomCode = localStorage.getItem('lovelive_room_code'); let sessionToken = null; let showFriendlyAbilities = localStorage.getItem('lovelive_friendly_abilities') !== 'false'; // Default ON /** * Centrally manages how ability text is displayed based on Language and "Friendly" settings. */ function getEffectiveAbilityText(card) { if (!card) return ""; const rawText = card.text || ""; const originalText = card.original_text; // Do not fallback to rawText here yet let effectiveText = ""; if (currentLang === 'en') { // English mode: Always show simplified pseudocode (translated) effectiveText = window.translateAbility ? window.translateAbility(rawText, 'en') : rawText; } else if (showFriendlyAbilities) { // Japanese mode (Friendly ON): Show simplified pseudocode (translated to Japanese) effectiveText = window.translateAbility ? window.translateAbility(rawText, 'jp') : rawText; } else { // Japanese mode (Friendly OFF): Show official original Japanese text if available if (originalText) { effectiveText = originalText; } else { // Fallback: If no official text, force friendly translation to avoid raw pseudocode effectiveText = window.translateAbility ? window.translateAbility(rawText, 'jp') : rawText; } } // Apply icon enrichment (e.g. [Auto] -> ) return enrichAbilityText(effectiveText); } function getEffectiveActionText(action) { if (!action) return ""; const rawText = action.raw_text || action.text || ""; let effectiveText = rawText; // If we want friendly/English, use the translator on the raw pseudocode if ((currentLang === 'en' || showFriendlyAbilities) && window.translateAbility) { effectiveText = window.translateAbility(rawText, currentLang); } // Otherwise in Japanese mode without friendly abilities, try to get original Japanese text else if (currentLang === 'jp') { const srcCard = resolveCardData(action.source_card_id); if (srcCard && srcCard.original_text) { effectiveText = srcCard.original_text; } else { // Fallback: Force translation if no original text found (to avoid raw variable names) if (window.translateAbility) { effectiveText = window.translateAbility(rawText, 'jp'); } } } // Apply icon enrichment let text = enrichAbilityText(effectiveText); // Final cleanup: strip technical prefixes if they persist in the summary text = text.replace(/TRIGGER:\s*/g, ''); text = text.replace(/\[TRIGGER\]\s*/g, ''); return text; } function getActionTags(action, vertical = false) { if (!action || !action.triggers) return ""; const tags = []; if (action.triggers.includes(1)) tags.push(`[登場時]`); if (action.triggers.includes(2)) tags.push(`[開始時]`); if (action.triggers.includes(7)) tags.push(`[起動]`); if (action.triggers.includes(6)) tags.push(`[常時]`); if (tags.length === 0) return ""; if (vertical) { return `
${tags.join('')}
`; } return `
${tags.join('')}
`; } // Helper to get common headers function getHeaders() { const headers = { 'Content-Type': 'application/json' }; if (roomCode) { headers['X-Room-Id'] = roomCode; // console.log(`[DEBUG] Sending X-Room-Id: ${roomCode}, X-Player-Idx: ${perspectivePlayer}`); } else { console.warn("[DEBUG] No roomCode set for getHeaders!"); } headers['X-Player-Idx'] = perspectivePlayer; return headers; } function updateRoomDisplay() { const display = document.getElementById('room-code-header'); if (display) display.innerText = roomCode || "---"; } function saveSession(room, sessionData) { if (!sessionData) return; const key = `lovelive_session_${room}`; localStorage.setItem(key, sessionData.session_id); sessionToken = sessionData.session_id; // Auto-set perspective if (sessionData.player_id !== undefined && sessionData.player_id !== -1) { perspectivePlayer = sessionData.player_id; console.log(`Assigned as Player ${perspectivePlayer}`); } } function loadSession(room) { const key = `lovelive_session_${room}`; sessionToken = localStorage.getItem(key); } async function createRoom(mode = 'pve') { console.log(`[DEBUG] Creating room with mode: ${mode}`); const isPublic = document.getElementById('public-room-check')?.checked || false; try { const res = await fetch('api/rooms/create', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ mode: mode, public: isPublic }) }); if (!res.ok) { const text = await res.text(); alert("Server Error: " + res.status + " - " + text); return; } const data = await res.json(); console.log("[DEBUG] Create Response:", data); if (data.success) { setRoom(data.room_id); // Host is always P1 (Idx 0) perspectivePlayer = 0; document.getElementById('room-modal').style.display = 'none'; // Auto-refresh to get state fetchState(); } else { alert("Failed to create room: " + data.error); } } catch (e) { console.error(e); alert("Network error creating room: " + e.message); } } async function joinRoom(code = null) { if (!code) { const input = document.getElementById('room-code-input'); code = input.value.trim().toUpperCase(); } if (!code) return; console.log(`[DEBUG] Joining room: ${code}`); try { const res = await fetch('api/rooms/join', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ room_id: code }) }); const data = await res.json(); console.log("[DEBUG] Join Response:", data); if (data.success) { setRoom(code); document.getElementById('room-modal').style.display = 'none'; if (data.mode === 'pvp') { console.log("[DEBUG] PvP Mode detected, showing perspective modal"); // Joiner defaults to P2 (Idx 1) perspectivePlayer = 1; document.getElementById('perspective-modal').style.display = 'flex'; } else { console.log("[DEBUG] PvE Mode, fetching state directly"); fetchState(); } } else { alert("Room not found or invalid: " + data.error); } } catch (e) { console.error(e); alert("Network error joining room: " + e.message); } } async function fetchPublicRooms() { const listEl = document.getElementById('public-rooms-list'); if (!listEl) return; if (isStaticHost) { listEl.innerHTML = '
Public rooms are only available in Online Mode.
Please use Start Offline below.
'; return; } listEl.innerHTML = '
Loading...
'; try { const res = await fetch('api/rooms/list'); const data = await res.json(); if (data.success) { const rooms = data.rooms || []; if (rooms.length === 0) { listEl.innerHTML = '
No public rooms found.
'; return; } let html = ''; rooms.forEach(r => { const modeLabel = r.mode === 'pvp' ? '⚔️ PvP' : '🤖 PvE'; const timeStr = r.created_at ? new Date(r.created_at).toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' }) : ''; const phaseStr = r.phase === '?' ? '' : `(T${r.turn})`; html += `
${r.room_id} ${modeLabel}
${timeStr} ${phaseStr} • ${r.players} Player(s)
`; }); listEl.innerHTML = html; } else { listEl.innerHTML = '
Failed to load list.
'; } } catch (e) { console.error("Fetch rooms error", e); listEl.innerHTML = '
Network Error
'; } } function setRoom(code) { roomCode = code; localStorage.setItem('lovelive_room_code', code); loadSession(code); updateRoomDisplay(); } function leaveRoom() { if (!confirm(currentLang === 'jp' ? "ロビーに戻りますか?現在のゲーム状態は失われませんが、再参加にはルームコードが必要です。" : "Return to lobby? The game state will be preserved, but you will need the code to re-join.")) return; roomCode = null; state = null; localStorage.removeItem('lovelive_room_code'); updateRoomDisplay(); document.getElementById('room-modal').style.display = 'flex'; fetchPublicRooms(); } window.startOffline = async function (userInitiated = true) { console.log("[OFFLINE] Starting Offline Mode. userInitiated:", userInitiated, "location:", window.location.href); offlineMode = true; hotseatMode = true; // Offline implies local hotseat or PvE document.getElementById('room-display').style.display = 'none'; // Load WASM Adapter dynamically - use absolute path from base for reliability try { const base = getAppBaseUrl(); console.log(`[OFFLINE] Importing ${base}js/wasm_adapter.js...`); // Using base ensures this works on both localhost and GitHub Pages subfolders const mod = await import(`${base}js/wasm_adapter.js`); wasmAdapter = mod.wasmAdapter; console.log("[OFFLINE] Initializing wasmAdapter..."); await wasmAdapter.init(); if (userInitiated) { document.getElementById('room-modal').style.display = 'none'; } fetchState(); } catch (e) { console.error("[OFFLINE] Failed to load engine:", e); alert("Failed to load Offline Engine: " + e + "\n(Check console for more details)"); offlineMode = false; } } // Initial Check window.addEventListener('DOMContentLoaded', () => { updateLanguage(); // Auto-detect static hosting (GitHub Pages) and fallback to Offline WASM if (isStaticHost && !roomCode) { console.log("[ENV] Static host detected, auto-starting Offline Mode."); startOffline(false); // Initialize but don't hide lobby } fetchAndPopulateDecks(); // Fetch dynamic decks if (roomCode) { loadSession(roomCode); document.getElementById('room-modal').style.display = 'none'; updateRoomDisplay(); fetchState(); } else { document.getElementById('room-modal').style.display = 'flex'; fetchPublicRooms(); } }); // --- Game Setup / Deck Selection Logic --- let setupMode = 'pve'; let deckPresets = []; let pvpJoinPid = 1; async function fetchAndPopulateDecks() { console.log("Fetching decks..."); try { let data = null; // Try API first if NOT on a static host (like GitHub Pages) if (!isStaticHost) { try { let response = await fetch('api/get_test_deck'); if (response.ok) { data = await response.json(); } } catch (apiErr) { console.warn("API deck fetch failed, falling back to static:", apiErr); } } if (!data) { // Fallback to static JSON console.log("Using static decks/list.json"); const response = await fetch('decks/list.json'); if (response.ok) { data = await response.json(); } else { console.error("Failed to load static deck list:", response.status); } } if (!data || data.error) { console.error("Error fetching decks:", data ? data.error : "No data"); return; } const mainDecks = data.available_decks || data.main_decks || []; // const energyDecks = data.energy_decks || []; // Not used in UI yet populateDeckSelect('p0-deck-select', mainDecks); populateDeckSelect('p1-deck-select', mainDecks); } catch (e) { console.error("Failed to fetch decks:", e); } } function populateDeckSelect(elementId, decks) { const select = document.getElementById(elementId); if (!select) return; // Clear and rebuild options select.innerHTML = ''; // Add Random Option const randomOption = document.createElement('option'); randomOption.value = 'random'; randomOption.textContent = '🎲 Random Deck'; select.appendChild(randomOption); // Add Paste Option const pasteOption = document.createElement('option'); pasteOption.value = 'paste'; pasteOption.textContent = '📋 Paste Deck List...'; select.appendChild(pasteOption); decks.forEach(deck => { const option = document.createElement('option'); option.value = deck; option.textContent = deck; select.appendChild(option); }); } // Global handler for select changes window.onDeckSelectChange = function (playerId) { const selectId = playerId === 0 ? 'p0-deck-select' : 'p1-deck-select'; const select = document.getElementById(selectId); if (!select) return; const val = select.value; const pasteArea = document.getElementById(playerId === 0 ? 'p0-paste-area' : 'p1-paste-area'); if (val === 'paste') { if (pasteArea) pasteArea.style.display = 'block'; } else { if (pasteArea) pasteArea.style.display = 'none'; console.log(`Player ${playerId} Deck Selected: ${val}`); } }; window.openGameSetup = function (mode) { setupMode = mode; document.getElementById('setup-mode-display').textContent = mode === 'pve' ? 'Mode: Solo (PvE)' : 'Mode: PvP (Multiplayer)'; document.getElementById('setup-modal').style.display = 'flex'; document.getElementById('room-modal').style.display = 'none'; // Hide lobby const p0Col = document.querySelector('.setup-column:nth-child(1)'); // Player 1 const p1Col = document.getElementById('p2-setup-column'); // Player 2 const startBtn = document.querySelector('#setup-modal .btn'); // Reset styles p0Col.style.display = 'block'; p1Col.style.display = 'block'; startBtn.onclick = submitGameSetup; startBtn.textContent = mode === 'pve' ? '🚀 Start Game' : '🚀 Create Room'; if (mode === 'pve') { p1Col.style.opacity = '1'; p1Col.style.pointerEvents = 'auto'; p1Col.querySelector('h4').textContent = '🤖 Player 2 (AI)'; } else { // In PvP Create, only P1 sets their deck p1Col.style.opacity = '0.3'; p1Col.style.pointerEvents = 'none'; p1Col.querySelector('h4').textContent = '👤 Player 2 (Waiting...)'; } // Init Dropdowns fetchAndPopulateDecks(); }; window.closeSetupModal = function () { document.getElementById('setup-modal').style.display = 'none'; if (!roomCode) { document.getElementById('room-modal').style.display = 'flex'; // Show lobby if not in game } }; window.onDeckSelectChange = function (pid) { const select = document.getElementById(`p${pid}-deck-select`); const pasteArea = document.getElementById(`p${pid}-paste-area`); const val = select.value; if (val === 'paste') { pasteArea.style.display = 'block'; } else { pasteArea.style.display = 'none'; } const preview = document.getElementById(`p${pid}-deck-preview`); const namedDecks = { 'muse_cup': "μ's Cup tournament deck", 'aqours_cup': "Aqours Cup tournament deck", 'nijigaku_cup': "Nijigaku Cup tournament deck", 'liella_cup': "Liella! Cup tournament deck", 'hasunosora_cup': "Hasunosora Cup tournament deck" }; if (namedDecks[val]) { preview.textContent = namedDecks[val]; } else if (val.startsWith('preset-')) { const idx = parseInt(val.split('-')[1]); const p = deckPresets[idx]; preview.textContent = p ? p.description : ''; } else if (val === 'random') { preview.textContent = "A random valid deck will be generated."; } else if (val === 'paste') { preview.textContent = "Paste a list of PL! IDs below."; } else { // Any other value is likely a named deck from ai/decks preview.textContent = `Deck file: ${val}.txt`; } }; function getDeckConfig(pid) { const select = document.getElementById(`p${pid}-deck-select`); const val = select.value; if (val === 'paste') { const text = document.getElementById(`p${pid}-deck-paste`).value; if (!text.trim()) { return { main: [], energy: [], type: 'random' }; // Fallback to empty (random) if empty paste } // Extract PL! IDs const matches = text.match(/(PL![A-Za-z0-9\-]+)/g); if (!matches || matches.length === 0) { alert(`No valid card IDs found in Player ${pid + 1} deck paste.`); return null; } return { main: matches, energy: [] }; } if (val.startsWith('preset-')) { const idx = parseInt(val.split('-')[1]); const p = deckPresets[idx]; return { main: p.cards, energy: [] }; } if (val === 'random') { return { main: [], energy: [], type: 'random' }; } // Default to 'named' for anything else (assumed from ai/decks/) return { main: [], energy: [], type: 'named', deckName: val }; } window.submitGameSetup = async function () { const p0Deck = getDeckConfig(0); if (!p0Deck) return; // Resolve named decks async function resolveDeck(config) { if (config.type === 'named' && config.deckName) { try { // Try API first let res = await fetch(`api/get_test_deck?deck=${config.deckName}`); let data; if (res.ok) { data = await res.json(); } else { // Fallback to static console.log(`API not found for deck ${config.deckName}, trying static decks/${config.deckName}.txt`); res = await fetch(`decks/${config.deckName}.txt`); if (res.ok) { const text = await res.text(); const matches = text.match(/(PL![A-Za-z0-9\-]+)/g); data = { success: true, main_deck: matches || [], energy_deck: [] }; } else { data = { success: false, error: "Static deck file not found" }; } } if (data.success && data.main_deck) { return { main: data.main_deck, energy: data.energy_deck || [] }; } else { alert(`Failed to load deck '${config.deckName}': ${data.error || 'Unknown error'}`); return null; } } catch (e) { alert(`Error fetching deck '${config.deckName}': ${e.message}`); return null; } } return config; } const resolvedP0 = await resolveDeck(p0Deck); if (!resolvedP0) return; const payload = { mode: setupMode, public: document.getElementById('public-room-check')?.checked || false, decks: { 0: resolvedP0 } }; if (setupMode === 'pve') { const p1Deck = getDeckConfig(1); if (!p1Deck) return; const resolvedP1 = await resolveDeck(p1Deck); if (!resolvedP1) return; payload.decks[1] = resolvedP1; } console.log("Creating room with decks:", payload); if (offlineMode && wasmAdapter) { console.log("[OFFLINE] Creating local game..."); try { // Ensure WASM is ready before proceeding await wasmAdapter.init(); const p0Config = await wasmAdapter.resolveDeckList(resolvedP0.main); const p1Config = payload.decks[1] ? await wasmAdapter.resolveDeckList(payload.decks[1].main) : null; const res = await wasmAdapter.createGameWithDecks(p0Config, p1Config); if (res.success) { state = res.state; document.getElementById('setup-modal').style.display = 'none'; render(); } else { alert("Offline game creation failed: " + res.error); } } catch (e) { console.error("Offline setup error:", e); alert("Error in Offline Engine: " + e.message); } return; // Important: Don't fall through to API call } try { const res = await fetch('api/rooms/create', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(payload) }); const data = await res.json(); if (data.success) { setRoom(data.room_id); // Assuming Creator is always P0 perspectivePlayer = 0; saveSession(data.room_id, data.session); document.getElementById('setup-modal').style.display = 'none'; fetchState(); } else { alert("Failed to start game: " + data.error); } } catch (e) { console.error(e); alert("Error starting game: " + e.message); } }; window.openDeckSelectionForPvP = function (pid) { pvpJoinPid = pid; document.getElementById('setup-mode-display').textContent = 'PvP: Select Your Deck'; document.getElementById('setup-modal').style.display = 'flex'; document.getElementById('room-modal').style.display = 'none'; const p0Col = document.querySelector('.setup-column:nth-child(1)'); const p1Col = document.getElementById('p2-setup-column'); const startBtn = document.querySelector('#setup-modal .btn'); // Hide the other player's column if (pid === 0) { p0Col.style.display = 'block'; p1Col.style.display = 'none'; document.querySelector('#p0-deck-select').closest('div').querySelector('h4').textContent = '👤 Your Deck'; } else { p0Col.style.display = 'none'; p1Col.style.display = 'block'; p1Col.style.opacity = '1'; p1Col.style.pointerEvents = 'auto'; p1Col.querySelector('h4').textContent = '👤 Your Deck'; } startBtn.textContent = '✅ Submit Deck & Join'; startBtn.onclick = submitPvPDeck; fetchAndPopulateDecks(); }; window.submitPvPDeck = async function () { const deck = getDeckConfig(pvpJoinPid); if (!deck) return; console.log(`Submitting deck for P${pvpJoinPid}...`); // Call set_deck try { const res = await fetch('api/set_deck', { method: 'POST', headers: { 'Content-Type': 'application/json', 'X-Room-Id': roomCode }, body: JSON.stringify({ player: pvpJoinPid, deck: deck.main, energy_deck: deck.energy }) }); const data = await res.json(); if (data.status === 'ok') { document.getElementById('setup-modal').style.display = 'none'; fetchState(); // Maybe show a waiting toast? alert("Deck Submitted! Waiting for host to start/reset game."); } else { alert("Failed to submit deck: " + (data.error || "Unknown error")); } } catch (e) { console.error(e); alert("Error submitting deck: " + e.message); } }; function openSettingsModal() { document.getElementById('settings-modal').style.display = 'flex'; } function closeSettingsModal() { document.getElementById('settings-modal').style.display = 'none'; } const translations = { 'jp': { 'turn': 'ターン', 'phase': 'フェーズ', 'score': 'スコア', 'update': '🔄 更新', 'reset': 'リセット', 'deck_creator': '🛠 デッキ作成', 'set_deck': '🎴 デッキ設定', 'friendly_abilities': '読みやすい能力テキスト', 'report': '不具合報告', 'live_watch': '🔴 ライブ監視', 'open_file': '📂 ファイルを開く', 'paste': '📋 貼り付け', 'load_server': '☁️ サーバーから読込', 'replay_mode': '🎬 リプレイモード', 'live_guide': '🌟 ライブガイド', 'looked_cards': '👁️ 確認済みカード', 'rule_log': '📜 ルールログ', 'logs': '📜 ログ', 'last_perf': '前回のパフォーマンス', 'god_mode': '🔧 デバッグモード', 'force': '強制実行', 'exec': '実行', 'wait': '待機', 'mulligan_you': 'マリガン (自分)', 'mulligan_opp': 'マリガン (相手)', 'active': '通常', 'energy': 'エナジー', 'draw': 'ドロー', 'main': 'メイン', 'live_set': 'ライブセット', 'perf_p1': 'パフォーマンス (P1)', 'perf_p2': 'パフォーマンス (P2)', 'live_result': 'ライブ結果', 'setup': 'セットアップ', 'act_ability': '能力発動', 'perform_live': 'ライブ開催' }, 'en': { 'turn': 'Turn', 'phase': 'Phase', 'score': 'Score', 'update': '🔄 Update', 'reset': 'RESET', 'deck_creator': '🛠 Deck Creator', 'set_deck': '🎴 Set Deck', 'report': 'Report Issue', 'live_watch': '🔴 Live Monitor', 'open_file': '📂 Open File', 'paste': '📋 Paste', 'load_server': '☁️ Load Server', 'replay_mode': '🎬 Replay Mode', 'live_guide': '🌟 Live Guide', 'looked_cards': '👁️ Looked Cards', 'rule_log': '📜 Rule Log', 'logs': '📜 Logs', 'last_perf': 'Last Performance', 'god_mode': '🔧 God Mode', 'force': 'Force Exec', 'exec': 'Run Code', 'wait': 'Wait', 'mulligan_you': 'Mulligan (You)', 'mulligan_opp': 'Mulligan (Opp)', 'active': 'Active', 'energy': 'Energy', 'draw': 'Draw', 'main': 'Main', 'live_set': 'Live Set', 'perf_p1': 'Performance (P1)', 'perf_p2': 'Performance (P2)', 'live_result': 'Live Result', 'setup': 'Setup', 'friendly_abilities': 'Friendly Abilities', 'act_ability': 'Activate Ability', 'perform_live': 'Perform Live' } }; function toggleLang() { currentLang = currentLang === 'jp' ? 'en' : 'jp'; updateLanguage(); } function toggleHotseat() { hotseatMode = !hotseatMode; fetchState(); } function toggleFriendlyAbilities() { showFriendlyAbilities = !showFriendlyAbilities; localStorage.setItem('lovelive_friendly_abilities', showFriendlyAbilities); updateLanguage(); render(); } window.setPerspective = function (idx) { perspectivePlayer = idx; document.getElementById('perspective-modal').style.display = 'none'; const btn = document.getElementById('switch-btn'); if (btn) btn.textContent = `View: P${perspectivePlayer + 1}`; // Refresh headers and state getHeaders(); render(); fetchState(); }; function togglePerspective() { perspectivePlayer = 1 - perspectivePlayer; render(); } function updateLanguage() { const t = translations[currentLang]; document.querySelectorAll('[data-i18n]').forEach(el => { const key = el.getAttribute('data-i18n'); if (t[key]) { if (key === 'live_watch') { let text = t[key]; if (isLiveWatchOn) { text = text.replace('🔴', '🟢').replace('OFF', 'ON'); } el.textContent = text; } else if (key === 'friendly_abilities') { let text = t[key]; text += showFriendlyAbilities ? ': ON' : ': OFF'; el.textContent = text; } else { el.textContent = t[key]; } } }); // Toggle Button Text const btn = document.getElementById('lang-btn'); if (btn) btn.textContent = currentLang === 'jp' ? 'English' : '日本語'; // Refresh render if state exists if (state) render(); } let state = null; let lastStateJson = null; // To avoid redundant renders let logs = []; let lastPerformanceData = null; // Store for re-viewing let selectedTurn = -1; // -1 means "All" let selectedHandIdx = -1; // -1 means no selection let showingFullLog = false; let lastPerformanceTurn = -1; // Track popup state let fullLogData = null; let lastActionsHash = null; // To avoid redundant action list wipes const feedItems = []; function log(msg, type = 'normal') { const timestamp = new Date().toLocaleTimeString(); logs.unshift({ timestamp, msg, type }); if (logs.length > 100) logs.pop(); // Also add to game feed if it's an interesting human action if (type === 'action' || type === 'score') { addToFeed(msg, type); } } function addToFeed(msg, type) { const icons = { 'action': '🎫', 'score': '✨', 'effect': '🪄', 'turn': '📅' }; feedItems.unshift({ msg, icon: icons[type] || '📝', timestamp: Date.now() }); if (feedItems.length > 20) feedItems.shift(); renderFeed(); } function renderFeed() { const feedEl = document.getElementById('game-feed'); if (!feedEl) return; feedEl.innerHTML = feedItems.map(item => `
${item.icon} ${item.msg}
`).join(''); feedEl.scrollTop = 0; } async function fetchState() { try { if (replayMode) return; // Stop polling while watching replay if (offlineMode) { if (!wasmAdapter) return; const res = await wasmAdapter.fetchState(); if (res.success) { // Check change const raw = JSON.stringify(res.state); if (raw === lastStateJson) return; lastStateJson = raw; state = res.state; render(); } return; } if (!roomCode) return; // Wait for room // Pause updates if Performance Modal is open const perfModal = document.getElementById('performance-modal'); if (perfModal && perfModal.style.display !== 'none') return; const headers = { 'X-Room-Id': roomCode }; if (sessionToken) headers['X-Session-Token'] = sessionToken; const res = await fetch('api/state?viewer=' + perspectivePlayer, { headers: headers }); if (res.status === 404) { // Room invalid or expired console.warn("Room not found (404). Resetting."); roomCode = null; state = null; localStorage.removeItem('lovelive_room_code'); document.getElementById('room-modal').style.display = 'flex'; console.log("DEBUG: Room 404 Reset"); updateRoomDisplay(); return; } const raw = await res.text(); if (raw === lastStateJson) return; // Skip if no change lastStateJson = raw; const data = JSON.parse(raw); if (data.success) { state = data.state; console.log("[DEBUG] State Mode:", state.mode); } else { console.error("State fetch unsuccessful:", data.error); return; } // Auto-sync perspective if authenticated if (state.my_player_id !== undefined && state.my_player_id !== -1 && !hotseatMode) { perspectivePlayer = state.my_player_id; } // Keep track of last performance for review if (state.performance_results && Object.keys(state.performance_results).length > 0) { lastPerformanceData = state.performance_results; } if (state.last_performance_results && Object.keys(state.last_performance_results).length > 0) { lastPerformanceData = state.last_performance_results; } render(); } catch (e) { console.error("Fetch Error:", e); } } let renderRequested = false; function render() { if (renderRequested) return; renderRequested = true; requestAnimationFrame(() => { renderInternal(); renderRequested = false; }); } function renderInternal() { if (!state || !state.players) return; // --- Proactive Pre-loading --- const assetsToLoad = []; state.players.forEach(p => { if (p.hand) p.hand.forEach(c => { if (c.img) assetsToLoad.push(fixImg(c.img)); }); if (p.stage) p.stage.forEach(c => { if (c && c.img) assetsToLoad.push(fixImg(c.img)); }); }); if (state.legal_actions) { state.legal_actions.forEach(a => { if (a.img) assetsToLoad.push(fixImg(a.img)); }); } // Check if assets actually changed to avoid redundant work const assetsHash = assetsToLoad.join('|'); if (window.lastAssetsHash !== assetsHash) { preloadAssets(assetsToLoad); window.lastAssetsHash = assetsHash; } // ---------------------------- if (hotseatMode && state.active_player !== undefined) { // In hotseat, view whoever's turn it is perspectivePlayer = state.active_player; } const p0 = state.players[perspectivePlayer] || state.players[0]; const p1 = state.players[1 - (state.players.indexOf(p0) === -1 ? 0 : state.players.indexOf(p0))] || state.players[1]; // Alias looked_cards for helper functions (renderSelectionModal, resolves) if (p0) state.looked_cards = p0.looked_cards || []; if (!p0 || !p1) return; // Final safety // Update Mode Buttons // Use Actual mode from server state if available, otherwise fallback to local hotseat toggle const isPvP = state.mode === "pvp" || hotseatMode; const pvpBtn = document.getElementById('pvp-btn'); if (pvpBtn) { pvpBtn.textContent = isPvP ? "PvP (vs Human)" : "Solo (vs AI)"; pvpBtn.style.background = isPvP ? "linear-gradient(135deg, #ec4899, #be185d)" : "linear-gradient(135deg, #2ecc71, #27ae60)"; } // Update Debug Info Header const debugInfo = document.getElementById('header-debug-info'); if (debugInfo) { let debugText = isPvP ? "PvP" : "PvE"; if (state.is_pvp) debugText += " (Network)"; else if (hotseatMode) debugText += " (Hotseat)"; // Show Perspective debugText += ` | View: P${perspectivePlayer + 1}`; debugInfo.textContent = debugText; // Color code if (isPvP) debugInfo.style.borderColor = "#ec4899"; else debugInfo.style.borderColor = "#2ecc71"; } const switchBtn = document.getElementById('switch-btn'); if (switchBtn) { switchBtn.style.display = hotseatMode ? 'inline-block' : 'none'; switchBtn.textContent = `View: P${perspectivePlayer + 1}`; } // Header info const t = translations[currentLang]; let phaseKey = 'wait'; if (state.phase === Phase.SETUP) phaseKey = 'setup'; if (state.phase === Phase.MULLIGAN_P1) phaseKey = (perspectivePlayer === 0) ? 'mulligan_you' : 'mulligan_opp'; if (state.phase === Phase.MULLIGAN_P2) phaseKey = (perspectivePlayer === 1) ? 'mulligan_you' : 'mulligan_opp'; if (state.phase === Phase.ACTIVE) phaseKey = 'active'; if (state.phase === Phase.ENERGY) phaseKey = 'energy'; if (state.phase === Phase.DRAW) phaseKey = 'draw'; if (state.phase === Phase.MAIN) phaseKey = 'main'; if (state.phase === Phase.LIVE_SET) phaseKey = 'live_set'; if (state.phase === Phase.PERFORMANCE_P1) phaseKey = (perspectivePlayer === 0) ? 'perf_p1' : 'perf_p2'; // perf_p1 text is "(P1)", perf_p2 is "(P2)" if (state.phase === Phase.PERFORMANCE_P2) phaseKey = (perspectivePlayer === 1) ? 'perf_p1' : 'perf_p2'; // Wait, language keys are 'perf_p1'='Performance (P1)'. // Improved Logic: // If language key is literal "Performance (P1)", then we want to start consistently using that or "Your Performance"? // The previous keys were 'perf_p1': 'Performance (P1)', 'perf_p2': 'Performance (P2)'. // If user wants generic "Your Performance", we might need new keys. // For now, let's stick to literal P1/P2 for performance to avoid ambiguity, // BUT fix Mulligan which says "(You)". if (state.phase === Phase.PERFORMANCE_P1) phaseKey = 'perf_p1'; if (state.phase === Phase.PERFORMANCE_P2) phaseKey = 'perf_p2'; if (state.phase === Phase.LIVE_RESULT) phaseKey = 'live_result'; // Update header stats const turnEl = document.getElementById('turn'); if (turnEl) { turnEl.textContent = state.turn_number || state.turn || 1; const turnLabel = turnEl.parentElement.querySelector('.stat-label'); if (turnLabel) { turnLabel.style.display = 'none'; // Remove the colon text node Array.from(turnEl.parentElement.childNodes).forEach(node => { if (node.nodeType === 3 && node.textContent.includes(':')) node.textContent = ''; }); } } const phaseEl = document.getElementById('phase'); if (phaseEl) { phaseEl.textContent = t[phaseKey] || state.phase; const phaseLabel = phaseEl.parentElement.querySelector('.stat-label'); if (phaseLabel) { phaseLabel.style.display = 'none'; Array.from(phaseEl.parentElement.childNodes).forEach(node => { if (node.nodeType === 3 && node.textContent.includes(':')) node.textContent = ''; }); } } const scoreEl = document.getElementById('score'); if (scoreEl) { scoreEl.textContent = `${state.players[0].score} - ${state.players[1].score}`; const scoreLabel = scoreEl.parentElement.querySelector('.stat-label'); if (scoreLabel) { scoreLabel.style.display = 'none'; Array.from(scoreEl.parentElement.childNodes).forEach(node => { if (node.nodeType === 3 && node.textContent.includes(':')) node.textContent = ''; }); } } // Header Energy & Hearts (Space Efficient) if (p0) { const infoBar = document.querySelector('.header .info-bar'); if (infoBar) { // Update Energy directly const current = p0.energy_untapped || 0; const total = p0.energy_count || 0; const energyEl = document.getElementById('header-energy'); if (energyEl) { energyEl.textContent = `${current}/${total}`; energyEl.parentElement.style.color = current > 0 ? 'var(--accent-gold)' : '#888'; // Hide label to save space const label = energyEl.parentElement.querySelector('.stat-label'); if (label) { label.style.display = 'none'; Array.from(energyEl.parentElement.childNodes).forEach(node => { if (node.nodeType === 3 && node.textContent.includes(':')) node.textContent = ''; }); } } // Update Hearts (Compact Bar) let totalHearts = document.getElementById('total-hearts-summary'); if (!totalHearts) { totalHearts = document.createElement('span'); totalHearts.id = 'total-hearts-summary'; totalHearts.className = 'stat hearts-stat'; totalHearts.title = "Total Hearts"; // Insert after energy const energyStat = document.querySelector('.stat[title="Energy"]'); if (energyStat) { energyStat.after(totalHearts); } else { infoBar.appendChild(totalHearts); } } if (p0.total_hearts) { // Python Order: Pink, Red, Yellow, Green, Blue, Purple, Any // Asset Order: heart_01 (Pink) ... heart_06 (Purple), icon_all (Any) let html = ''; p0.total_hearts.forEach((count, idx) => { const iconName = idx < 6 ? `heart_0${idx + 1}.png` : 'icon_all.png'; html += ` heart ${count} `; }); totalHearts.innerHTML = html; } // Update Yells (Compact Total) let totalYells = document.getElementById('total-yells-header'); if (!totalYells) { totalYells = document.createElement('span'); totalYells.id = 'total-yells-header'; totalYells.className = 'stat yells-stat'; totalYells.title = "Total Yells (Score potential)"; // Insert after hearts totalHearts.after(totalYells); } const yellCount = p0.total_blades || 0; totalYells.innerHTML = ` ${yellCount}`; } } // Agent Names const myAgentKey = perspectivePlayer === 0 ? 'p0_agent' : 'p1_agent'; const oppAgentKey = perspectivePlayer === 0 ? 'p1_agent' : 'p0_agent'; const myDefault = perspectivePlayer === 0 ? 'Player 1' : 'Player 2'; const oppDefault = perspectivePlayer === 0 ? 'Player 2' : 'Player 1'; document.getElementById('my-agent-name').textContent = state[myAgentKey] ? `(${state[myAgentKey]})` : `(${myDefault})`; document.getElementById('opp-agent-name').textContent = state[oppAgentKey] ? `(${state[oppAgentKey]})` : `(${oppDefault})`; // Opponent document.getElementById('opp-score').textContent = `${p1.score} ライブ`; document.getElementById('opp-deck').textContent = p1.deck_count; document.getElementById('opp-energy-deck').textContent = 12 - p1.energy_count; document.getElementById('opp-energy-count').textContent = p1.energy_count; document.getElementById('opp-hand-count').textContent = p1.hand_count; document.getElementById('opp-discard-count').textContent = p1.discard_count; renderEnergy('opp-energy', p1.energy); renderStage('opp-stage', p1.stage, false); renderCards('opp-hand', p1.hand, false, true); renderCards('opp-discard', p1.discard, false, true); renderCards('opp-success', p1.success_lives, false, true); // Player document.getElementById('my-score').textContent = `${p0.score} ライブ`; document.getElementById('my-deck').textContent = p0.deck_count; document.getElementById('my-energy-deck').textContent = 12 - p0.energy_count; document.getElementById('my-energy-count').textContent = `${p0.energy_untapped}/${p0.energy_count}`; document.getElementById('my-hand-count').textContent = p0.hand_count; document.getElementById('my-discard-count').textContent = p0.discard_count; renderEnergy('my-energy', p0.energy); renderStage('my-stage', p0.stage, true); // Render Hand // Mulligan selection OR standard selection let selectedIndices = []; if (state.phase === Phase.MULLIGAN_P1 || state.phase === Phase.MULLIGAN_P2) { // In mulligan phase, P0 might have selection selectedIndices = Array.from(p0.mulligan_selection || []); } else { if (selectedHandIdx !== -1) selectedIndices = [selectedHandIdx]; } // For live set, we also allow multiple selection visualization technically, but currently only one active choice // Actually, live set is one by one. renderCards('my-hand', p0.hand, true, false, selectedIndices); renderCards('my-discard', p0.discard); renderCards('my-success', p0.success_lives, true); // Made clickable renderLiveZone('my-live', p0.live_zone, p0.live_zone_revealed); renderPerformanceGuide(); renderLookedCards(); renderSelectionModal(); renderLogs(); renderActiveAbilities(p0); // Check for performance popup logic... const modal = document.getElementById('performance-modal'); const perfResults = state.performance_results || {}; // DEBUG: Log phase and results for persistent "not popping up" issue // console.log(`[Render] Phase: ${state.phase}, PerfResults: ${Object.keys(perfResults).length}`); // Prevent duplicate auto-shows using a simple content hash const currentPerfHash = JSON.stringify(perfResults); // Auto-show Performance Result (at start of next turn) // Check if we have results and if we haven't shown them for this turn yet if (state.turn > lastPerformanceTurn) { const resultsToShow = lastPerformanceData || perfResults; if (resultsToShow && Object.keys(resultsToShow).length > 0) { const hash = JSON.stringify(resultsToShow); if (window.lastShownPerformanceHash !== hash) { renderPerformanceResult(resultsToShow); modal.style.display = 'flex'; window.lastShownPerformanceHash = hash; lastPerformanceTurn = state.turn; } } } // No auto-hide needed really, user dismisses it. // Check for game over if (state.game_over) { const actionsDiv = document.getElementById('actions'); if (actionsDiv) { let winnerText = ''; if (state.winner === 0) winnerText = '🎉 P0 (あなた) Wins!'; else if (state.winner === 1) winnerText = '🎉 P1 (相手) Wins!'; else if (state.winner === -2) winnerText = '❌ Illegal Move (Error)'; else winnerText = '⚖️ Draw (引き分け)'; actionsDiv.innerHTML = `

${winnerText}

Game Over - Final Score: ${state.players[0].score} - ${state.players[1].score}

`; } } else { renderActions(); } // Highlight source card of pending choice highlightPendingSource(); } function renderLogs() { const ruleLogEl = document.getElementById('rule-log'); if (ruleLogEl) { let logData = state.rule_log || []; // Apply filtering if (selectedTurn !== -1) { const turnStr = `[Turn ${selectedTurn}]`; logData = logData.filter(entry => entry.includes(turnStr)); } ruleLogEl.innerHTML = ''; logData.forEach(entry => { const div = document.createElement('div'); div.className = 'log-entry'; let displayText = entry; let isAbility = false; // Detect and translate absolute ability logs: [TRIGGER:ID]CardName: Pseudocode // Regex to handle optional turn prefix: (?:\[Turn \d+\] )?\[TRIGGER:(\d+)\](.*?): (.*) const abilityMatch = entry.match(/(?:\[Turn \d+\] )?\[TRIGGER:(\d+)\](.*?): (.*)/); if (abilityMatch) { isAbility = true; const triggerId = parseInt(abilityMatch[1]); const cardName = abilityMatch[2].trim(); const pseudocode = abilityMatch[3].trim(); // Translate Trigger let triggerLabel = `[${triggerId}]`; if (window.Translations && currentLang && window.Translations[currentLang]) { const t = window.Translations[currentLang]; if (t.triggers && t.triggers[triggerId]) { triggerLabel = t.triggers[triggerId]; } } // Translate Effect Pseudocode let translatedEffect = pseudocode; const shouldTranslate = (currentLang === 'en' || showFriendlyAbilities); if (shouldTranslate && window.translateAbility) { // Prepend "EFFECT:" so translateAbility treats it as a pseudo-line translatedEffect = window.translateAbility("EFFECT: " + pseudocode, currentLang); // Remove the "→ " or "Effect: " prefix added by translator if we want a clean line translatedEffect = translatedEffect.replace(/^.*?: /, '').replace(/^→ /, ''); } else if (currentLang === 'jp' && !showFriendlyAbilities) { // If Japanese and Friendly OFF, try to find original text from state const srcCard = resolveCardDataByName(cardName); if (srcCard && srcCard.original_text) { translatedEffect = srcCard.original_text; } } // Translate Card Name if it's in our map (optional, usually card names are kept JP or from DB) let displayCardName = cardName; if (currentLang === 'en' && window.NAME_MAP && window.NAME_MAP[cardName]) { displayCardName = window.NAME_MAP[cardName]; } // Reconstruct (preserve Turn number if present) const turnMatch = entry.match(/^\[Turn \d+\]/); const turnPrefix = turnMatch ? turnMatch[0] + " " : ""; displayText = `${turnPrefix}${triggerLabel} ${displayCardName}: ${translatedEffect}`; } // Apply semantic classes for color-coding if (entry.includes('PLAYS')) div.classList.add('action'); else if (entry.includes('EFFECT') && !isAbility) div.classList.add('effect'); else if (entry.includes('SCORE')) div.classList.add('score'); else if (entry.includes('[Turn') && !isAbility) div.classList.add('turn'); else if (isAbility) div.classList.add('ability'); div.textContent = displayText; ruleLogEl.appendChild(div); }); if (!showingFullLog) ruleLogEl.scrollTop = ruleLogEl.scrollHeight; } // Legacy simple log for sidebar if exists const simpleLogEl = document.getElementById('log'); if (simpleLogEl) { simpleLogEl.innerHTML = logs.map(l => `
[${l.timestamp}] ${l.msg}
`).join(''); } } function copyToClipboard(text) { navigator.clipboard.writeText(text).then(() => { log(`Copied ID: ${text}`); }).catch(err => { console.error('Failed to copy: ', err); }); if (event) event.stopPropagation(); } function renderActiveAbilities(player) { const listEl = document.getElementById('active-abilities-list'); const panelEl = document.getElementById('active-abilities-panel'); if (!listEl || !panelEl) return; const effects = player.active_effects || []; if (effects.length === 0) { panelEl.style.display = 'none'; return; } panelEl.style.display = 'block'; listEl.innerHTML = effects.map(eff => { const enrichedDesc = enrichAbilityText(eff.description); return `
[${eff.source}] Expires: ${eff.expiry}
${enrichedDesc}
`; }).join(''); } function enrichAbilityText(text) { if (!text) return ""; const iconMap = { '登場時': 'texticon/toujyou.png', '自動': 'texticon/jidou.png', '永続': 'texticon/jyouji.png', '起動': 'texticon/kidou.png', 'ターン1': 'texticon/turn1.png', 'ライブスタート': 'texticon/live_start.png', 'LIVE START': 'texticon/live_start.png', 'オール': 'texticon/icon_all.png', 'ブレード': 'texticon/icon_blade.png', 'エナジー': 'texticon/icon_energy.png' }; for (const [key, path] of Object.entries(iconMap)) { const regex = new RegExp(`\\[${key}\\]`, 'g'); let style = ""; if (path.includes('live_start') || path.includes('live_success')) { style = ' style="min-width: 3.5em;"'; } else if (path.includes('turn1')) { style = ' style="min-width: 2em;"'; } let src = `img/${path}`; if (typeof ICON_DATA_URIs !== 'undefined' && ICON_DATA_URIs[key]) { src = ICON_DATA_URIs[key]; } text = text.replace(regex, `${key}`); } text = text.replace(/{{(.*?)\|(.*?)}}/g, (match, img, alt) => { let src = "img/" + img; if (img.endsWith('.png') && !img.includes('/')) { src = "img/texticon/" + img; } let style = ""; if (img.includes('live_start') || img.includes('live_success')) { style = ' min-width: 3.5em;'; } const iconKey = img.replace('.png', ''); if (typeof ICON_DATA_URIs !== 'undefined' && ICON_DATA_URIs[iconKey]) { src = ICON_DATA_URIs[iconKey]; } return `${alt}${alt}`; }); text = text.replace(/\\n/g, '
'); return text; } function renderSelectionModal() { const modal = document.getElementById('selection-modal'); const choice = state.pending_choice; // Should we show selection modal? // Types that involve picking from a list of cards const selectionTypes = ["SELECT_FROM_LIST", "SELECT_FROM_DISCARD", "SELECT_SUCCESS_LIVE", "SELECT_SWAP_SOURCE", "SELECT_FORMATION_SLOT", "ORDER_DECK"]; const isSelectionChoice = choice && selectionTypes.includes(choice.type); const hasLookedCards = state.looked_cards && state.looked_cards.length > 0; if (!isSelectionChoice && !hasLookedCards) { modal.style.display = 'none'; return; } // If it's a selection choice, but for the AI, don't show it? // Actually, we should show it if we are WATCHING, but normally solo means user is P0. if (choice && choice.params && choice.params.player_id !== 0 && !hasLookedCards) { modal.style.display = 'none'; return; } modal.style.display = 'flex'; const titleEl = document.getElementById('selection-title'); const descEl = document.getElementById('selection-description'); const contentEl = document.getElementById('selection-content'); const footerEl = document.getElementById('selection-footer'); if (isSelectionChoice) { titleEl.textContent = choice.source_member ? `${choice.source_member} の効果` : "カード選択"; descEl.textContent = choice.description || "対象のカードを選択してください。"; if (choice.type === "ORDER_DECK") { // ORDER_DECK is now iterative in the backend. // We use the general selection logic below, but we'll add a 'Done' button if legal. // (The general logic handles IDs 600-659, which is where ORDER_DECK selection actions live) } // Find legal actions in that range let actionRange = [600, 659]; if (choice.type === "SELECT_FROM_DISCARD") actionRange = [660, 719]; if (choice.type === "SELECT_FORMATION_SLOT") actionRange = [720, 759]; if (choice.type === "SELECT_SUCCESS_LIVE") actionRange = [760, 819]; if (choice.type === "TARGET_LIVE") actionRange = [820, 829]; if (choice.type === "TARGET_ENERGY") actionRange = [830, 849]; if (choice.type === "TARGET_REMOVED") actionRange = [850, 909]; if (choice.type === "ORDER_DECK") actionRange = [600, 659]; const legalOptions = state.legal_actions.filter(a => a.id >= actionRange[0] && a.id <= actionRange[1]); const canPass = state.legal_actions.some(a => a.id === 0); // Auto-select if only one option and forced (cannot pass) // This makes forced discards/selections "automatic" if (legalOptions.length === 1 && !canPass) { console.log("Auto-selecting single forced option:", legalOptions[0].id); // Use setTimeout to allow render to finish or at least not block setTimeout(() => doAction(legalOptions[0].id), 10); modal.style.display = 'none'; // Hide modal immediately return; } let html = ''; const params = choice.params || {}; const cards = params.cards || params.available_members || []; cards.forEach((cId, idx) => { const absId = actionRange[0] + idx; const action = state.legal_actions.find(a => a.id === absId); const isLegal = !!action; // Use enriched data from action if available, otherwise resolveCardData let card = resolveCardData(cId); if (action && action.name) { card = { ...card, name: action.name, img: action.img || card.img }; } const isLive = card.type === 'live'; const tooltip = getEffectiveAbilityText(card); html += `
${card.name}
`; }); contentEl.innerHTML = html; // Footer (Pass button OR Done button) const doneAction = state.legal_actions.find(a => a.id === actionRange[0] + cards.length); if (choice.type === "ORDER_DECK") { // Relabel Skip to "Discard Rest" and hide redundant Done button footerEl.innerHTML = ``; } else if (doneAction) { footerEl.innerHTML = ``; } else if (canPass) { footerEl.innerHTML = ``; } else { footerEl.innerHTML = ''; } } else if (hasLookedCards) { const currentLookedHash = JSON.stringify(state.looked_cards.map(c => c.id)); if (window.dismissedLookedCardsHash === currentLookedHash) { if (modal.style.display === 'flex') modal.style.display = 'none'; return; } titleEl.textContent = "確認したカード"; descEl.textContent = "現在公開されているカードです。"; let html = ''; state.looked_cards.forEach(card => { const isLive = card.type === 'live'; const tooltip = getEffectiveAbilityText(card); html += `
${card.name}
`; }); contentEl.innerHTML = html; // Store hash in a data attribute or global for the onclick handler window.currentLookedCardsHashToDismiss = currentLookedHash; footerEl.innerHTML = ``; } // Highlight source card in the board if applicable highlightPendingSource(); } // Helper to resolve card info from cardName function resolveCardDataByName(name) { if (!state) return null; for (const p of state.players) { if (!p) continue; const allZones = [(p.hand || []), (p.stage || []), (p.live_zone || []), (p.energy || []), (p.discard || []), (p.success_zone || [])]; for (const zone of allZones) { for (const c of zone) { const card = (typeof c === 'object' && c !== null) ? (c.card || c) : null; if (card && card.name === name) return card; } } } if (state.looked_cards) { const found = state.looked_cards.find(c => c.name === name); if (found) return found; } return null; } // Helper to resolve card info from state caches if possible function resolveCardData(cid) { if (cid === null || cid === undefined || cid < 0) return null; // First try to look in the looked_cards if it's there if (state && state.looked_cards) { const found = state.looked_cards.find(c => c.id === cid); if (found) return found; } // Fallback to basic (we might not have full info if it's just an ID) return { id: cid, name: `Card ${cid}`, img: 'icon_blade.png', text: "", original_text: "" }; } function renderActions() { const actionsDiv = document.getElementById('actions'); if (!actionsDiv || !state) return; // Check if relevant action state changed to avoid redundant wipes const actionsHash = JSON.stringify({ actions: state.legal_actions, active: state.active_player, phase: state.phase, choice: state.pending_choice, perspective: perspectivePlayer, lang: currentLang, friendly: showFriendlyAbilities }); if (actionsHash === lastActionsHash) { return; // No change, skip wipe to preserve hover/tooltips } lastActionsHash = actionsHash; actionsDiv.innerHTML = ''; const mobileActionBar = document.getElementById('mobile-action-bar'); if (mobileActionBar) mobileActionBar.innerHTML = ''; // Check if it's our turn const isOurTurn = (state.active_player === perspectivePlayer); if (!isOurTurn && !hotseatMode) { const turnBanner = document.createElement('div'); turnBanner.className = 'turn-banner waiting'; turnBanner.innerHTML = ` OPPONENT'S TURN (P${state.active_player + 1})`; turnBanner.style.background = 'rgba(239, 83, 80, 0.15)'; turnBanner.style.border = '1px solid #ef5350'; turnBanner.style.color = '#ef5350'; turnBanner.style.padding = '10px'; turnBanner.style.marginBottom = '10px'; turnBanner.style.textAlign = 'center'; turnBanner.style.borderRadius = '6px'; turnBanner.style.fontWeight = 'bold'; actionsDiv.appendChild(turnBanner); // Selection safety: clear selection if it's not our turn selectedHandIdx = -1; return; // Hiding Opponent's Actions } // 0. Mulligan Phase Indicator if (state.phase === Phase.MULLIGAN_P1 || state.phase === Phase.MULLIGAN_P2) { const isP1 = state.phase === Phase.MULLIGAN_P1; const phaseName = currentLang === 'en' ? `Mulligan Phase (${isP1 ? 'P1' : 'P2'})` : `マリガンフェーズ (${isP1 ? 'P1' : 'P2'})`; const helpText = currentLang === 'en' ? "Tap cards in your hand to select for swapping, then press 'DONE'." : "交換したいカードをタップして選択してください。
「完了」を押すと確定します。"; const mulliganDiv = document.createElement('div'); mulliganDiv.className = 'mulligan-indicator'; mulliganDiv.innerHTML = `

${phaseName}

${helpText}

`; actionsDiv.appendChild(mulliganDiv); } // 0.1 Pending Choice Indicator // Suppress if in Main phase and the choice is likely a board-driven selection (1000-1999 range) const isSpecialSelect = state.pending_choice && (state.pending_choice.type === 'SELECT_FROM_LIST' || state.pending_choice.type === 'SELECT_MODE'); if (state.pending_choice && !(state.phase === Phase.MAIN && isSpecialSelect)) { const choiceDiv = document.createElement('div'); choiceDiv.className = 'pending-choice-indicator'; choiceDiv.style.padding = '10px'; choiceDiv.style.marginBottom = '15px'; choiceDiv.style.background = 'rgba(255, 215, 0, 0.08)'; choiceDiv.style.border = '1px solid var(--accent-gold)'; choiceDiv.style.borderRadius = '8px'; const desc = state.pending_choice.description || '選択してください'; const member = state.pending_choice.source_member || 'Unknown Source'; const ability = state.pending_choice.source_ability ? `[${state.pending_choice.source_ability}]` : ''; let stepInfo = ''; if (state.pending_choice.params && state.pending_choice.params.step_progress) { const steps = state.pending_choice.params.step_progress; if (steps !== '?') { let label = (steps === 'Cost') ? 'Paying Cost' : `Step ${steps}`; stepInfo = `(${label})`; } } const imgPath = state.pending_choice.source_img ? fixImg(state.pending_choice.source_img) : null; const imgHtml = imgPath ? `` : ''; choiceDiv.onmouseenter = () => highlightPendingSource(); choiceDiv.onmouseleave = () => clearHighlights(); choiceDiv.innerHTML = `
${imgHtml}
${member} ${ability} ${stepInfo}
${desc}
`; actionsDiv.appendChild(choiceDiv); } if (!state.legal_actions || state.legal_actions.length === 0) { // Diagnostic data for debugging missing actions const diag = { phase: state.phase, phase_name: Phase[state.phase] || 'Unknown', turn: state.turn || state.turn_number, active: state.active_player }; console.warn('[UI] Empty actions:', diag); if (state.phase === Phase.LIVE_SET) { console.error('[DIAG] Stuck in LIVE_SET with no actions. Perspective:', perspectivePlayer, 'Active:', state.active_player); } actionsDiv.innerHTML += `
⚠️ No actions available
`; return; } if (state.phase === Phase.LIVE_SET) { console.log('[DIAG] LIVE_SET Phase. Legal actions:', state.legal_actions); } // 1. Render Action List Container const listDiv = document.createElement('div'); listDiv.className = 'action-list'; actionsDiv.appendChild(listDiv); const myStage = (state.players && state.players[perspectivePlayer]) ? state.players[perspectivePlayer].stage : null; // 0. Pin "Pass" / "Confirm" Action (ID 0) to TOP const passAction = state.legal_actions.find(a => a.id === 0); if (passAction) { const btn = document.createElement('button'); btn.className = 'action-btn pass'; // Always pass style for top button if (state.phase === Phase.MULLIGAN_P1 || state.phase === Phase.MULLIGAN_P2) { btn.classList.add('confirm'); btn.innerHTML = `${currentLang === 'en' ? 'DONE (Mulligan)' : 'マリガン完了'}`; } else if (state.phase === Phase.LIVE_RESULT) { btn.classList.add('confirm'); btn.innerHTML = `${currentLang === 'en' ? 'CONFIRM' : '確認して次へ'}`; } else if (state.phase === Phase.LIVE_SET) { btn.classList.add('confirm'); btn.innerHTML = `${currentLang === 'en' ? 'CONFIRM LIVE SET' : 'ライブセット完了'}`; } else if (state.pending_choice) { btn.innerHTML = `${currentLang === 'en' ? 'SKIP OPTIONAL' : '能力を使用しない (Skip)'}`; } else { btn.innerHTML = `${currentLang === 'en' ? 'END PHASE' : 'フェイズ終了'}`; } btn.style.marginBottom = '10px'; btn.style.justifyContent = 'center'; btn.onclick = () => doAction(0); btn.onmouseenter = () => { clearHighlights(); hideTooltip(); }; btn.onmouseleave = () => clearHighlights(); listDiv.appendChild(btn); // Mirror to mobile action bar if (mobileActionBar && window.innerWidth <= 768) { const mBtn = btn.cloneNode(true); mBtn.onclick = btn.onclick; // cloneNode doesn't copy listeners mobileActionBar.appendChild(mBtn); } } // Group actions: Key = "type-handIdx-name" or similar, Value = [actions] const groups = {}; const abilityActions = []; const otherActions = []; state.legal_actions.forEach(action => { if (action.id === 0) return; // Already handled if (action.type === 'ABILITY') { abilityActions.push(action); } else if (action.type === 'PLAY') { // Group PLAY actions by the card they are playing (hand_idx) const key = `PLAY-${action.hand_idx}`; if (!groups[key]) groups[key] = { type: 'PLAY', actions: [], main: action }; groups[key].actions.push(action); } else if (action.type === 'SELECT_STAGE' && state.phase === Phase.RESPONSE) { // Group SELECT_STAGE actions by the card they are placing (source_card_id) const key = `SELECT_STAGE-${action.source_card_id || 'unknown'}`; if (!groups[key]) groups[key] = { type: 'SELECT_STAGE', actions: [], main: action }; groups[key].actions.push(action); } else { otherActions.push(action); } }); // 1. Render Ability Actions (Below Pass, Above Play) abilityActions.forEach(action => { const btn = document.createElement('button'); btn.className = 'action-btn'; // Premium "Golden Action" styling for activated abilities if (action.location === 'discard' || (action.id >= 200 && action.id < 300) || (action.id >= 2000 && action.id < 3000)) { btn.classList.add('golden'); } btn.style.marginBottom = '4px'; btn.style.padding = '6px'; btn.style.textAlign = 'left'; let imgHtml = ''; if (action.img) { imgHtml = ``; } const abilityTagsHtml = getActionTags(action); const translatedText = getEffectiveActionText(action); const lines = translatedText.split('\n'); // Extract the effect part from translated text const translatedEffect = (lines.length > 1 ? lines.slice(lines[0].includes('【') || lines[0].includes('[') ? 1 : 0).join('
') : lines[0] || ''); // Prioritize translated text if in friendly/EN mode, otherwise fallback to backend 'description' const isFriendly = currentLang === 'en' || showFriendlyAbilities; let displayBody = (isFriendly && translatedEffect) ? translatedEffect : (action.description || translatedEffect || 'Ability'); btn.innerHTML = ` ${imgHtml}
${action.name}
${displayBody}
${abilityTagsHtml}
`; btn.setAttribute('data-text', translatedText || action.text || action.description || ""); btn.setAttribute('data-slot-idx', action.area_idx); btn.onmouseenter = () => highlightAction(action); btn.onmouseleave = () => clearHighlights(); btn.onclick = () => doAction(action.id); listDiv.appendChild(btn); // Mirror to mobile action bar if (mobileActionBar && window.innerWidth <= 768) { const mBtn = btn.cloneNode(true); mBtn.onclick = btn.onclick; mBtn.onmouseenter = btn.onmouseenter; mBtn.onmouseleave = btn.onmouseleave; mobileActionBar.appendChild(mBtn); } }); // 2. Render Grouped Actions (PLAY / SELECT_STAGE) Object.values(groups).forEach(group => { const btn = document.createElement('div'); btn.className = 'action-group'; btn.style.background = 'rgba(255, 255, 255, 0.05)'; btn.style.border = '1px solid var(--border)'; btn.style.borderRadius = '6px'; btn.style.padding = '5px'; btn.style.marginBottom = '6px'; const mainAction = group.main; const tooltipText = getEffectiveActionText(mainAction); btn.setAttribute('data-text', tooltipText); btn.onmouseenter = () => highlightAction(mainAction); btn.onmouseleave = () => clearHighlights(); let displayImg = mainAction.img || "icon_blade.png"; let titleText = mainAction.name || "Action Group"; let subTitle = "登場させる先を選択"; if (group.type === 'SELECT_STAGE') { titleText = mainAction.source_name || mainAction.name || "メンバーの登場先選択"; displayImg = mainAction.source_img || mainAction.img || "icon_blade.png"; subTitle = "に置く場所を選択"; } // Clean up title text (strip technical prefixes) titleText = titleText.replace(/TRIGGER:\s*/g, ''); const abilityTagsHtml = getActionTags(mainAction, true); const displayBase = mainAction.base_cost !== undefined ? mainAction.base_cost : mainAction.cost; const baseCostHtml = displayBase !== undefined ? `${displayBase}` : ''; let content = `
${titleText}${abilityTagsHtml}${baseCostHtml}
${subTitle}
`; content += `
`; for (let i = 0; i < 3; i++) { const subAct = group.actions.find(a => a.area_idx === i); if (subAct) { let displayCost = ""; let extraInfo = ""; if (group.type === 'PLAY') { const finalCost = typeof subAct.cost === 'number' ? subAct.cost : 0; if (myStage && myStage[i] && myStage[i].id !== -1) { const stageCard = myStage[i]; displayCost = finalCost === 0 ? `0` : `-${finalCost}`; if (finalCost > 0) { displayCost = `${displayCost}`; extraInfo = `
BATON ▲
`; } else { const handBase = mainAction.base_cost !== undefined ? mainAction.base_cost : (mainAction.cost || 0); const targetCost = stageCard.cost || 0; if (handBase < targetCost) { displayCost = `${displayCost}`; extraInfo = `
BATON ▼
`; } else { displayCost = `${displayCost}`; extraInfo = `
BATON =
`; } } } else { displayCost = finalCost === 0 ? `0` : `-${finalCost}`; extraInfo = `
PLACE
`; if (finalCost > 0) { displayCost = `${displayCost}`; } else { displayCost = `${displayCost}`; } } } else if (group.type === 'SELECT_STAGE') { displayCost = `[${["左", "中", "右"][i]}]`; } const subTooltip = getEffectiveActionText(subAct); content += ` `; } else { content += `
`; } } content += `
`; btn.innerHTML = content; listDiv.appendChild(btn); // Mirror to mobile action bar if (mobileActionBar && window.innerWidth <= 768) { const mBtn = document.createElement('div'); mBtn.className = 'mobile-action-group-container'; mBtn.style.display = 'flex'; mBtn.style.gap = '4px'; // For grouped actions, we want to mirror the sub-actions as individual buttons on mobile // or at least make them accessible. Mirroring the whole group might be too bulky. // Let's mirror the individual sub-buttons if they exist. const subButtons = btn.querySelectorAll('.sub-action'); subButtons.forEach(sb => { const clonedSb = sb.cloneNode(true); clonedSb.onclick = (e) => { e.stopPropagation(); const actionIdAttr = sb.getAttribute('onclick').match(/doAction\((\d+)\)/); if (actionIdAttr) doAction(parseInt(actionIdAttr[1])); }; // Copy labels/costs for mobile context if needed, but clonedSb should have them mBtn.appendChild(clonedSb); }); if (mBtn.children.length > 0) { mobileActionBar.appendChild(mBtn); } } }); // 2. Render Other Actions (Standard) - BUT filtering out Action 0 (Pass) to pin it const nonPassActions = otherActions.filter(a => a.id !== 0); // MULLIGAN SPECIAL: If in mulligan, we want to show actions for cards even if they are already selected // (if they aren't in legal_actions anymore) if (state.phase === Phase.MULLIGAN_P1 || state.phase === Phase.MULLIGAN_P2) { const p0 = state.players[perspectivePlayer]; const selectedMulliganIndices = p0.mulligan_selection || []; // Show all hand cards as potential mulligan actions if (p0.hand) { p0.hand.forEach((card, idx) => { const actionId = 300 + idx; const isSelected = selectedMulliganIndices.includes(idx); const actionInLegal = state.legal_actions.find(a => a.id === actionId); // If it's already selected OR in legal actions, we show it if (isSelected || actionInLegal) { const btn = document.createElement('button'); btn.className = 'action-btn'; if (isSelected) { btn.style.opacity = '0.8'; // btn.style.border = '2px solid var(--accent-pink)'; } const cardName = card.name || `Card ${idx}`; // Toggle Text const actionText = isSelected ? `Return: ${cardName}` : `Mulligan: ${cardName}`; const icon = isSelected ? '↩️ ' : ''; btn.innerHTML = `${icon}${actionText}`; // FIX: Ensure Mulligan buttons use standardized tooltip logic if (card) { btn.setAttribute('data-text', getEffectiveAbilityText(card)); } btn.onclick = () => doAction(actionId); btn.onmouseenter = () => highlightAction(actionInLegal || { id: actionId, type: 'MULLIGAN', hand_idx: idx }); btn.onmouseleave = () => clearHighlights(); listDiv.appendChild(btn); } }); } } else { nonPassActions.forEach(action => { const btn = document.createElement('button'); btn.className = 'action-btn'; // Content construction let content = ''; // Add source image if available (Ability/Play) if (action.img) { content += ``; } content += `
`; if (action.type === 'COLOR_SELECT') { const colors = { 'Pink': '#ff80ab', 'Blue': '#42a5f5', 'Green': '#66bb6a', 'Yellow': '#fdd835', 'Red': '#ef5350', 'Purple': '#ab47bc', 'All': 'linear-gradient(45deg, #ff80ab, #42a5f5, #66bb6a, #fdd835, #ef5350, #ab47bc)', 'None': '#757575' }; const bg = colors[action.color] || '#555'; btn.style.background = bg; // Text color contrast btn.style.color = (['Yellow', 'Pink', 'All'].includes(action.color)) ? '#333' : '#fff'; btn.style.fontWeight = 'bold'; btn.style.textAlign = 'center'; btn.innerHTML = `${action.name}`; } else if (action.type === 'SELECT_MODE') { const displayTitle = action.description || action.name; btn.style.borderLeft = '4px solid var(--accent-gold)'; content += `${displayTitle}`; const tooltipText = getEffectiveActionText(action); btn.setAttribute('data-text', tooltipText); } else if (action.type === 'ABILITY' && action.name) { let actionTitle = action.name; if (actionTitle.startsWith('【能力】')) { content += `${getEffectiveActionText(action)}`; } else { content += `Ability: ${getEffectiveActionText(action)}`; } const tooltipText = getEffectiveActionText(action); btn.setAttribute('data-text', tooltipText); } else if (action.type === 'PLAY' && action.name) { content += `Play: ${getEffectiveActionText(action)}`; const tooltipText = getEffectiveActionText(action); btn.setAttribute('data-text', tooltipText); } else if ((action.type === 'SELECT' || action.type === 'SELECT_HAND' || action.type === 'SELECT_STAGE' || action.type === 'SELECT_DISCARD' || action.type === 'TARGET_OPPONENT' || action.type === 'SELECT_LIVE') && action.name) { // Actions enriched with card/slot metadata let actionPrefix = ""; if (action.type === 'SELECT') actionPrefix = "Top: "; if (action.type === 'SELECT_HAND') actionPrefix = "Hand: "; if (action.type === 'SELECT_DISCARD') actionPrefix = "Discard: "; if (action.type === 'SELECT_STAGE') actionPrefix = "Stage: "; if (action.type === 'TARGET_OPPONENT') actionPrefix = "Target: "; if (action.type === 'SELECT_LIVE') actionPrefix = "Performance: "; content += `${actionPrefix}${getEffectiveActionText(action)}`; const tooltipText = getEffectiveActionText(action); btn.setAttribute('data-text', tooltipText); if (action.img) { // Update content to include image if not already added by general logic if (!content.includes('` + content; } } } else { const displayTitle = action.description || action.desc || action.name || 'Action'; content += `${displayTitle}`; btn.setAttribute('data-text', displayTitle); } content += `
`; btn.innerHTML = content; btn.onclick = () => doAction(action.id); // Source Highlighting logic btn.onmouseenter = () => highlightAction(action); btn.onmouseleave = () => clearHighlights(); listDiv.appendChild(btn); }); } } function highlightStageCard(areaIdx) { const myStage = document.getElementById('my-stage'); if (!myStage) return; // The stage has 3 slots, standard naming logic matches areaIdx usually 0-2 const slot = myStage.children[areaIdx]; if (slot) { // Find inner member-slot or apply to wrapper const memberSlot = slot.querySelector('.member-slot') || slot; memberSlot.classList.add('highlight-source'); } } function clearHighlights() { document.querySelectorAll('.highlight-source').forEach(el => el.classList.remove('highlight-source')); document.querySelectorAll('.highlight-target').forEach(el => el.classList.remove('highlight-target')); // hideTooltip() removed from here to prevent flickering during highlightAction transitions. // Tooltips are managed via global mouseover/mouseout listeners. } async function toggleFullLog() { showingFullLog = !showingFullLog; if (showingFullLog) { try { const btn = document.getElementById('full-log-btn'); if (btn) btn.textContent = 'Loading...'; const res = await fetch('api/full_log'); const data = await res.json(); fullLogData = data.log; render(); } catch (e) { console.error("Failed to fetch full log", e); showingFullLog = false; render(); } } else { render(); } } function playCard(idx) { // Check if this card can be toggled for Mulligan by looking at legal_actions // IMPORTANT: Backend uses 301-360 for Mulligan (300 + index + 1) const mulliganActionId = 301 + idx; const isMulliganAction = state.legal_actions && state.legal_actions.find(a => a.id === mulliganActionId); if (isMulliganAction) { // We're in Mulligan phase and this is a valid toggle action doAction(mulliganActionId); return; } // Standard Phase: Select card for later placement if (state.phase === Phase.LIVE_SET) { // If we have a valid action for setting this card as a live card const liveSetActionId = 400 + idx; const isLiveSetAction = state.legal_actions && state.legal_actions.find(a => a.id === liveSetActionId); if (isLiveSetAction) { doAction(liveSetActionId); return; } } if (selectedHandIdx === idx) { selectedHandIdx = -1; // Deselect } else { selectedHandIdx = idx; // Select } render(); // Re-render to show selection } function onStageSlotClick(areaIdx) { console.log('[DEBUG] Stage clicked:', areaIdx, 'selectedHandIdx:', selectedHandIdx); // Check for TARGET_MEMBER / MEMBER_SELECT (Action 560-562) const selectActionId = 560 + areaIdx; if (state.legal_actions && state.legal_actions.find(a => a.id === selectActionId)) { doAction(selectActionId); return; } if (selectedHandIdx === -1) { // If clicking an existing member, maybe use ability? (Actions 200-202) const actionId = 200 + areaIdx; const action = state.legal_actions.find(a => a.id === actionId); if (action) { doAction(actionId); } else { console.log(`No ability available for slot ${areaIdx}`); } return; } // Try to play selected card to this area const actionId = 1 + (selectedHandIdx * 3) + areaIdx; const action = state.legal_actions.find(a => a.id === actionId); if (action) { doAction(actionId); selectedHandIdx = -1; } } function onLiveSlotClick(idx) { console.log('[DEBUG] Live slot clicked:', idx); // Check for Performance (900-902) const perfActionId = 900 + idx; if (state.legal_actions && state.legal_actions.find(a => a.id === perfActionId)) { doAction(perfActionId); return; } // Check for TARGET_LIVE (820-822) const liveSelectId = 820 + idx; if (state.legal_actions && state.legal_actions.find(a => a.id === liveSelectId)) { doAction(liveSelectId); return; } } function onEnergySlotClick(idx) { console.log('[DEBUG] Energy clicked:', idx); const actionId = 830 + idx; // TARGET_ENERGY_ZONE if (state.legal_actions && state.legal_actions.find(a => a.id === actionId)) { doAction(actionId); } } function onSuccessSlotClick(idx) { console.log('[DEBUG] Success slot clicked:', idx); const actionId = 760 + idx; // TARGET_SUCCESS_LIVES if (state.legal_actions && state.legal_actions.find(a => a.id === actionId)) { doAction(actionId); } } function renderCards(containerId, cards, clickable = false, mini = false, selectedIndices = []) { const el = document.getElementById(containerId); if (!el) return; el.innerHTML = ''; if (!cards) return; // Null safety cards.forEach((card, idx) => { const div = document.createElement('div'); const isSelected = selectedIndices.includes(idx); // Logic for highlighting: Pink for mulligan, Gold for standard selection let highlightClass = ''; if (isSelected) { highlightClass = (state.phase === Phase.MULLIGAN_P1 || state.phase === Phase.MULLIGAN_P2) ? ' mulligan-selected' : ' selected'; } if (card.is_new) { highlightClass += ' new-card'; } div.className = 'card' + (card.hidden ? ' hidden' : '') + (card.type === 'live' ? ' type-live' : '') + (mini ? ' card-mini' : '') + highlightClass; div.id = `${containerId}-card-${idx}`; // Hand card lock logic const isMulligan = (state.phase === Phase.MULLIGAN_P1 || state.phase === Phase.MULLIGAN_P2); if (containerId === 'my-hand' && card.type === 'member' && !isMulligan) { if (state.phase !== 5 && card.valid_actions) { const playActions = card.valid_actions.filter(aid => aid >= 1 && aid <= 180); if (playActions.length === 0) { div.classList.add('card-locked'); } else { const emptySlots = []; const myStage = (state.players && state.players[perspectivePlayer]) ? state.players[perspectivePlayer].stage : null; if (myStage) { for (let s = 0; s < 3; s++) { if (myStage[s] === -1) emptySlots.push(s); } } const canPlayToEmpty = playActions.some(aid => { const areaIdx = (aid - 1) % 3; return emptySlots.includes(areaIdx); }); if (canPlayToEmpty) div.classList.add('card-playable'); else div.classList.add('card-baton-playable'); } } } if (containerId === 'my-hand') div.setAttribute('draggable', 'true'); const isLiveSet = state.phase === Phase.LIVE_SET; const tooltip = isLiveSet ? getEffectiveAbilityText(card) : getEffectiveAbilityText(card); // During Live Set, we want the tooltip to be very prominent if it's a Live card if (tooltip) div.setAttribute('data-text', tooltip); if (!card.hidden) { let imgPath = card.img || card.img_path || ''; const imgHtml = imgPath ? `` : ''; let heartPipsHtml = ''; if (card.type === 'member' && card.hearts) { const colorNames = ['Red', 'Blue', 'Green', 'Yellow', 'Purple', 'Pink']; let pips = ''; for (let c = 0; c < 6; c++) { for (let h = 0; h < (card.hearts[c] || 0); h++) { pips += `
`; } } if (pips) heartPipsHtml = `
${pips}
`; } if (card.type === 'live' && card.required_hearts) { const colorNames = ['Red', 'Blue', 'Green', 'Yellow', 'Purple', 'Pink', 'Any']; let pips = ''; const filled = card.filled_hearts || [0, 0, 0, 0, 0, 0, 0]; const req = card.required_hearts || [0, 0, 0, 0, 0, 0, 0]; let totalFilled = 0; let totalReq = 0; for (let c = 0; c < 7; c++) { const r = req[c] || 0; const f = filled[c] || 0; totalReq += r; for (let i = 0; i < r; i++) { const isFilled = i < f; const colorClass = c === 6 ? 'color-any' : `color-${c}`; const fillClass = isFilled ? 'filled' : 'empty'; pips += `
`; } totalFilled += Math.min(f, r); } if (pips) { const progressText = `
${totalFilled}/${totalReq}
`; heartPipsHtml = `
${progressText}
${pips}
`; } } let bladeHeartsHtml = ''; if (card.type === 'member' && card.blade_hearts) { let pips = ''; for (let c = 0; c < 6; c++) { for (let h = 0; h < (card.blade_hearts[c] || 0); h++) { pips += `
`; } } if (pips) bladeHeartsHtml = `
${pips}
`; } let modifiersHtml = ''; if (card.modifiers && card.modifiers.length > 0) { let modLines = ''; card.modifiers.forEach(mod => { let expiryIcon = mod.expiry === 'TURN_END' ? '⏳' : (mod.expiry === 'CONSTANT' ? '♾️' : (mod.expiry === 'LIVE_END' ? '🏁' : '⏲️')); modLines += `
${expiryIcon} ${mod.description}
`; }); modifiersHtml = `
${modLines}
`; } const displayId = card.card_no || card.id; const idHtml = `
ID:${displayId}
`; div.innerHTML = `${imgHtml}${card.cost !== undefined ? card.cost : '★'} ${bladeHeartsHtml}${heartPipsHtml}
${card.name || ''}
${modifiersHtml}${idHtml}`; } else { div.innerHTML = ``; } if (clickable) { div.style.cursor = 'pointer'; div.onclick = () => { if (containerId === 'my-success') onSuccessSlotClick(idx); else playCard(idx); }; } el.appendChild(div); }); } function renderStage(containerId, stage, clickable) { const el = document.getElementById(containerId); if (!el) return; el.innerHTML = ''; const labels = ['左', 'C', '右']; for (let i = 0; i < 3; i++) { const slot = stage[i]; const area = document.createElement('div'); area.className = 'member-area board-slot-container'; // Check for Opponent Target Actions (600-602) let isOppTarget = false; let oppActionId = -1; if (containerId === 'opp-stage') { oppActionId = 600 + i; if (state && state.legal_actions) { isOppTarget = state.legal_actions.some(a => a.id === oppActionId); } } // Check for Activated Ability (200-202) let abilityAction = null; if (containerId === 'my-stage' && state.legal_actions) { abilityAction = state.legal_actions.find(a => a.id === (200 + i)); } if (clickable) { area.style.cursor = 'pointer'; area.onclick = () => onStageSlotClick(i); } else if (isOppTarget) { area.style.cursor = 'pointer'; area.onclick = () => doAction(oppActionId); area.classList.add('highlight-target'); } const slotDiv = document.createElement('div'); slotDiv.className = 'member-slot' + (slot ? ' filled' : '') + (slot?.tapped ? ' tapped' : ''); slotDiv.id = `${containerId}-slot-${i}`; if (clickable && selectedHandIdx !== -1) { const actionId = 1 + (selectedHandIdx * 3) + i; const isValid = state.legal_actions.some(a => a.id === actionId); if (isValid) { slotDiv.style.borderColor = 'var(--accent-green)'; slotDiv.style.boxShadow = '0 0 10px var(--accent-green)'; } } else if (isOppTarget) { slotDiv.style.borderColor = 'var(--accent-pink)'; slotDiv.style.boxShadow = '0 0 10px var(--accent-pink)'; } if (slot) { if (slot.locked) { slotDiv.classList.add('locked'); const lockIcon = document.createElement('div'); lockIcon.className = 'lock-overlay'; lockIcon.innerHTML = '🔒'; slotDiv.appendChild(lockIcon); } let imgPath = slot.img || slot.img_path || ''; const imgHtml = imgPath ? `${slot.name}` : ''; let heartPipsHtml = ''; if (slot.hearts) { const colorNames = ['Red', 'Blue', 'Green', 'Yellow', 'Purple', 'Pink']; let pips = ''; for (let c = 0; c < 6; c++) { for (let h = 0; h < (slot.hearts[c] || 0); h++) { pips += `
`; } } if (pips) heartPipsHtml = `
${pips}
`; } let bladeHeartsHtml = ''; if (slot.blade_hearts) { let pips = ''; for (let c = 0; c < 6; c++) { for (let h = 0; h < (slot.blade_hearts[c] || 0); h++) { pips += `
`; } } if (pips) bladeHeartsHtml = `
${pips}
`; } const displayId = slot.card_no || slot.id; const idHtml = `
ID:${displayId}
`; let modifiersHtml = ''; if (slot.modifiers && slot.modifiers.length > 0) { let modLines = ''; slot.modifiers.forEach(mod => { let expiryIcon = mod.expiry === 'TURN_END' ? '⏳' : (mod.expiry === 'CONSTANT' ? '♾️' : (mod.expiry === 'LIVE_END' ? '🏁' : '⏲️')); modLines += `
${expiryIcon} ${mod.description}
`; }); modifiersHtml = `
${modLines}
`; } slotDiv.innerHTML = `${imgHtml} ${slot.cost}${bladeHeartsHtml}${heartPipsHtml}
${slot.name}
${modifiersHtml}${idHtml}`; const tooltipText = getEffectiveAbilityText(slot); if (tooltipText) slotDiv.setAttribute('data-text', tooltipText); } const label = document.createElement('div'); label.className = 'area-label'; label.textContent = labels[i]; const energyDiv = document.createElement('div'); energyDiv.className = 'attached-energy'; if (slot && slot.energy > 0) { for (let e = 0; e < slot.energy; e++) { const pip = document.createElement('div'); pip.className = 'energy-mini'; pip.style.background = 'var(--accent-gold)'; pip.style.boxShadow = '0 0 5px var(--accent-gold)'; energyDiv.appendChild(pip); } } area.appendChild(slotDiv); area.appendChild(label); area.appendChild(energyDiv); if (abilityAction) { const actBtn = document.createElement('button'); actBtn.className = 'slot-action-btn'; actBtn.textContent = translations[currentLang]['act_ability'] || '能力発動'; actBtn.onclick = (e) => { e.stopPropagation(); doAction(abilityAction.id); }; // Tooltip for the button as well actBtn.setAttribute('data-text', getEffectiveActionText(abilityAction)); area.appendChild(actBtn); } el.appendChild(area); } } function renderEnergy(containerId, energy) { const el = document.getElementById(containerId); if (!el) return; el.innerHTML = ''; if (!energy || !Array.isArray(energy)) return; energy.forEach((e, i) => { const div = document.createElement('div'); // Check if we have card data if (e.card && e.card.img) { div.className = 'card card-mini' + (e.tapped ? ' tapped' : ''); // Customize size for energy zone to fit more div.style.width = '45px'; div.style.height = '63px'; div.style.marginRight = '4px'; div.id = `${containerId}-slot-${i}`; const imgPath = fixImg(e.card.img || e.card.img_path); const imgHtml = ``; // Overlay "Tapped" visual const tapOverlay = e.tapped ? '
' : ''; div.innerHTML = imgHtml + tapOverlay; let tooltipText = `Energy #${i + 1}: ${e.card.name}`; const abilityText = getEffectiveAbilityText(e.card); if (abilityText) tooltipText += `\n\n${abilityText}`; div.setAttribute('data-text', tooltipText); // Interaction div.style.cursor = 'pointer'; div.onclick = () => onEnergySlotClick(i); } else { // Fallback to pip if no card data div.className = 'energy-pip' + (e.tapped ? ' tapped' : ''); div.textContent = i + 1; } el.appendChild(div); }); } function renderRuleLog(log, containerId = 'log-content') { const el = document.getElementById(containerId); if (!el) return; // Optimization: Only render the last 200 entries to prevent DOM slowdown // The full log is still available in memory if we need it, but for display 200 is plenty. const renderLimit = 200; const logToRender = log.length > renderLimit ? log.slice(-renderLimit) : log; // Use a DocumentFragment for batch appending const fragment = document.createDocumentFragment(); if (log.length > renderLimit) { const warning = document.createElement('div'); warning.className = 'log-entry'; warning.style.color = '#777'; warning.style.fontStyle = 'italic'; warning.textContent = `... (Oldest ${log.length - renderLimit} entries hidden) ...`; fragment.appendChild(warning); } logToRender.forEach(entry => { const div = document.createElement('div'); div.className = 'log-entry'; // Highlight error/warning if (entry.includes('Error')) div.style.color = '#ff6b6b'; if (entry.includes('Warning')) div.style.color = '#ffd93d'; div.textContent = entry; fragment.appendChild(div); }); el.innerHTML = ''; el.appendChild(fragment); // Auto-scroll to bottom el.scrollTop = el.scrollHeight; } function renderLiveZone(containerId, liveCards, clickable) { const el = document.getElementById(containerId); if (!el) return; el.innerHTML = ''; // Always show 3 slots for (let i = 0; i < 3; i++) { const card = liveCards[i]; const prefix = containerId.split('-')[0]; // 'my' or 'opp' const area = document.createElement('div'); area.className = 'board-slot-container'; const slot = document.createElement('div'); const isHidden = card && card.hidden && !card.face_down; slot.className = 'card card-mini' + (card ? ' type-live' : '') + (isHidden ? ' hidden' : ''); slot.id = `${prefix}-live-slot-${i}`; // Check for Performance Action (900-902) let perfAction = null; if (prefix === 'my' && state.legal_actions) { perfAction = state.legal_actions.find(a => a.id === (900 + i)); } if (card) { let imgPath = card.img || card.img_path || ''; const imgHtml = imgPath ? `` : ''; let reqHtml = '
'; if (card.required_hearts) { for (let c = 0; c < 6; c++) { const count = card.required_hearts[c]; for (let k = 0; k < count; k++) { reqHtml += `
`; } } if (card.required_hearts.length > 6) { const anyCount = card.required_hearts[6]; for (let k = 0; k < anyCount; k++) { reqHtml += `
`; } } } reqHtml += '
'; const opacityStyle = card.face_down ? 'opacity: 0.6; border: 2px dashed var(--accent-gold);' : ''; const peekLabel = card.face_down ? '
PEEK
' : ''; slot.innerHTML = `${peekLabel}${imgHtml}
${card.score || 0}
${reqHtml}
${card.name || ''}
`; if (card.face_down) slot.style.cssText += opacityStyle; const tooltipText = getEffectiveAbilityText(card); if (tooltipText) slot.setAttribute('data-text', tooltipText); } else { slot.style.border = '2px dashed #555'; slot.style.opacity = '0.5'; slot.innerHTML = `
空き ${i + 1}
`; } slot.style.cursor = 'pointer'; slot.onclick = () => onLiveSlotClick(i); area.appendChild(slot); if (perfAction) { const perfBtn = document.createElement('button'); perfBtn.className = 'slot-action-btn btn-perform'; perfBtn.textContent = translations[currentLang]['perform_live'] || 'ライブ開催'; perfBtn.onclick = (e) => { e.stopPropagation(); doAction(perfAction.id); }; perfBtn.setAttribute('data-text', getEffectiveActionText(perfAction)); area.appendChild(perfBtn); } el.appendChild(area); } } // Tooltip Logic let tooltipTimeout = null; let tooltipHideTimeout = null; let currentTooltipTarget = null; let currentTooltipSidebarTarget = null; function showTooltip(target, e, forceTarget = null, useSidebar = false, explicitText = null) { const effectiveTarget = forceTarget || target; const descPanel = document.getElementById('card-desc-panel'); const descContent = document.getElementById('card-desc-content'); const textRaw = explicitText || (target && target.dataset ? target.dataset.text : null); if (!textRaw || !descPanel || !descContent) return; // Reliability check: if we are hovering a new instance of the same button (due to re-render), // or if the content is exactly the same, skip to avoid flicker. if (descContent.dataset.rawText === textRaw && descPanel.style.display === 'block') { currentTooltipTarget = effectiveTarget; return; } if (currentTooltipTarget === effectiveTarget && !explicitText) return; currentTooltipTarget = effectiveTarget; let text = enrichAbilityText(textRaw); descContent.innerHTML = text; descContent.dataset.rawText = textRaw; // Store to prevent redundant renders descPanel.style.display = 'flex'; } function hideTooltip(immediate = false) { clearTimeout(tooltipTimeout); tooltipTimeout = null; if (immediate) { clearTimeout(tooltipHideTimeout); tooltipHideTimeout = null; currentTooltipTarget = null; currentTooltipSidebarTarget = null; const descPanel = document.getElementById('card-desc-panel'); if (descPanel) descPanel.style.display = 'none'; return; } if (tooltipHideTimeout) return; tooltipHideTimeout = setTimeout(() => { tooltipHideTimeout = null; currentTooltipTarget = null; currentTooltipSidebarTarget = null; const descPanel = document.getElementById('card-desc-panel'); if (descPanel) descPanel.style.display = 'none'; }, 100); // 100ms grace period for movement } document.body.addEventListener('mouseover', (e) => { const selector = '.card, .member-slot, .modifier-line, .action-btn, .action-group, .btn, .modal-content'; const target = e.target.closest(selector); if (target) { // Only trigger tooltip update if the target has text attribute. // If it doesn't (like a pass button without desc), we skip hideTooltip() // to prevent closing tooltips manually triggered by onmouseenter (highlightAction). if (target.dataset.text) { // Cancel any pending hide if (tooltipHideTimeout) { clearTimeout(tooltipHideTimeout); tooltipHideTimeout = null; } showTooltip(target, e, null, false); } } else { // If mouse is definitely NOT on any valid target, then hide it. hideTooltip(); } }); document.body.addEventListener('mouseout', (e) => { const selector = '.card, .member-slot, .modifier-line, .action-btn, .action-group, .btn, .modal-content'; const target = e.target.closest(selector); if (target) { // Only hide if we are actually leaving the descriptive area entirely. // If we move from a button to its container (action-group), keep it. const nextTarget = e.relatedTarget ? e.relatedTarget.closest(selector) : null; if (!nextTarget) { hideTooltip(); } } }); // Hide tooltip globally on click or scroll to prevent "stuck" tooltips window.addEventListener('scroll', () => hideTooltip(true), { passive: true }); window.addEventListener('click', () => hideTooltip(true)); // Shared helper for highlighting elements function addHighlight(id, className) { const el = document.getElementById(id); if (el) { el.classList.add(className); // Auto-scroll logic for hand cards to bring them into view if (el.closest('.card-area.hand')) { el.scrollIntoView({ behavior: 'smooth', block: 'nearest', inline: 'center' }); } console.log(`[DEBUG] Added ${className} to ${id} `); } } function highlightAction(a) { console.log('[DEBUG] Highlighting action:', a); clearHighlights(); // Don't hideTooltip() here, as mouseover listener already showed the action tooltip. // If we have a sub-target card, showTooltip will just update the content. const selfPrefix = (state.active_player === perspectivePlayer ? 'my' : 'opp'); const oppPrefix = (state.active_player === perspectivePlayer ? 'opp' : 'my'); // PLAY (1-180) or LIVE_SET (400-459) or MULLIGAN (300-359) if (a.type === 'PLAY') { if (a.hand_idx !== undefined) { addHighlight(`${selfPrefix}-hand-card-${a.hand_idx}`, 'highlight-source'); // NO showTooltip here if we are already hovering an action button } if (a.area_idx !== undefined) addHighlight(`${selfPrefix}-stage-slot-${a.area_idx}`, 'highlight-target'); } else if (a.type === 'LIVE_SET') { if (a.hand_idx !== undefined) { addHighlight(`${selfPrefix}-hand-card-${a.hand_idx}`, 'highlight-source'); } addHighlight(`${selfPrefix}-live-zone`, 'highlight-target'); } else if (a.type === 'ABILITY') { if (a.area_idx !== undefined) { addHighlight(`${selfPrefix}-stage-slot-${a.area_idx}`, 'highlight-source'); } } else if (a.type === 'MULLIGAN') { if (a.hand_idx !== undefined) { addHighlight(`${selfPrefix}-hand-card-${a.hand_idx}`, 'highlight-target'); } } else if (a.type === 'SELECT_HAND') { const prefix = (a.player_id !== undefined ? (a.player_id === perspectivePlayer ? 'my' : 'opp') : selfPrefix); const id = `${prefix}-hand-card-${a.hand_idx}`; addHighlight(id, 'highlight-source'); const card = document.getElementById(id); if (card && card.dataset.text) { showTooltip(card, { clientX: card.getBoundingClientRect().right + 10, clientY: card.getBoundingClientRect().top }); } } else if (a.type === 'SELECT_STAGE') { const prefix = (a.player_id !== undefined ? (a.player_id === perspectivePlayer ? 'my' : 'opp') : selfPrefix); const id = `${prefix}-stage-slot-${a.area_idx}`; addHighlight(id, 'highlight-target'); const slot = document.getElementById(id); if (slot && slot.dataset.text) { showTooltip(slot, { clientX: slot.getBoundingClientRect().right + 10, clientY: slot.getBoundingClientRect().top }); } } else if (a.type === 'SELECT') { if (state.pending_choice && state.pending_choice.type === 'TARGET_OPPONENT_MEMBER') { const prefix = (a.player_id !== undefined ? (a.player_id === perspectivePlayer ? 'my' : 'opp') : oppPrefix); const id = `${prefix}-stage-slot-${a.index}`; addHighlight(id, 'highlight-target'); const slot = document.getElementById(id); if (slot && slot.dataset.text) { showTooltip(slot, { clientX: slot.getBoundingClientRect().right + 10, clientY: slot.getBoundingClientRect().top }); } } else { addHighlight(`select-list-item-${a.index}`, 'highlight-target'); } } else if (a.type === 'SELECT_DISCARD') { const prefix = (a.player_id !== undefined ? (a.player_id === perspectivePlayer ? 'my' : 'opp') : selfPrefix); addHighlight(`${prefix}-discard`, 'highlight-target'); } else if (a.type === 'SELECT_LIVE') { const prefix = (a.player_id !== undefined ? (a.player_id === perspectivePlayer ? 'my' : 'opp') : selfPrefix); const id = `${prefix}-live-slot-${a.area_idx}`; addHighlight(id, 'highlight-target'); const slot = document.getElementById(id); if (slot && slot.dataset.text) { showTooltip(slot, { clientX: slot.getBoundingClientRect().right + 10, clientY: slot.getBoundingClientRect().top }); } } // Fallback: Highlight source card by ID if provided and not highlighted by index if (a.source_card_id !== undefined && a.source_card_id !== -1) { highlightCardById(a.source_card_id); } } function highlightPendingSource() { if (!state || !state.pending_choice) return; const srcId = state.pending_choice.source_card_id; if (srcId === undefined || srcId === -1) return; highlightCardById(srcId); } function highlightCardById(srcId) { if (!state) return; // Search both "my" and "opp" zones based on perspective const playersMap = [ { id: perspectivePlayer, prefix: 'my' }, { id: 1 - perspectivePlayer, prefix: 'opp' } ]; playersMap.forEach(pMap => { const p = state.players[pMap.id]; if (!p) return; // Stage if (p.stage) { p.stage.forEach((card, idx) => { const cid = card ? card.id : -1; if (cid === srcId) { const id = `${pMap.prefix}-stage-slot-${idx}`; addHighlight(id, 'highlight-source'); const slot = document.getElementById(id); if (slot) { // showTooltip removed to prevent conflict with action button tooltip } } }); } // Hand if (p.hand) { p.hand.forEach((card, idx) => { const cid = card ? card.id : -1; if (cid === srcId) { const id = `${pMap.prefix}-hand-card-${idx}`; addHighlight(id, 'highlight-source'); const cardEl = document.getElementById(id); if (cardEl) { // showTooltip removed } } }); } // Live if (p.live_zone) { p.live_zone.forEach((cardObj, idx) => { const cid = cardObj ? cardObj.id : -1; if (cid === srcId) { addHighlight(`${pMap.prefix}-live-slot-${idx}`, 'highlight-source'); } }); } // Discard if (p.discard && p.discard.some(c => (typeof c === 'object' ? c.id === srcId : c === srcId))) { addHighlight(`${pMap.prefix}-discard`, 'highlight-source'); } // Energy if (p.energy) { p.energy.forEach((e, idx) => { const cid = (e && e.card) ? e.card.id : -1; if (cid === srcId) { addHighlight(`${pMap.prefix}-energy-slot-${idx}`, 'highlight-source'); } }); } }); } function clearHighlight() { // Alias to plural for compatibility clearHighlights(); } async function doAction(id) { if (state.active_player !== perspectivePlayer && !hotseatMode) { console.warn("[UI] Action blocked: Not your turn."); return; } if (window.pendingAction) return; window.pendingAction = true; // Visual feedback document.body.classList.add('action-pending'); log(`Action: ${id} `); try { if (offlineMode) { const res = await wasmAdapter.doAction(id); if (res.success) { state = res.state; lastStateJson = JSON.stringify(state); render(); log('Action completed'); } else { alert(res.error); } return; } const headers = { 'Content-Type': 'application/json', 'X-Room-Id': roomCode }; if (sessionToken) headers['X-Session-Token'] = sessionToken; // Fast Poll: set polling to fast mode immediately on user action setPollingInterval(500); const res = await fetch('api/action', { method: 'POST', headers: headers, body: JSON.stringify({ action_id: id }) }); const text = await res.text(); lastStateJson = text; const data = JSON.parse(text); if (data.success) { state = data.state; render(); log('Action completed'); setTimeout(() => updateAdaptivePolling(), 2000); } else { alert(data.error || 'Unknown error'); updateAdaptivePolling(); } } finally { window.pendingAction = false; document.body.classList.remove('action-pending'); } } async function changeAI() { const aiMode = document.getElementById('ai-selector').value; console.log(`Switching AI to ${aiMode} `); const headers = { 'Content-Type': 'application/json', 'X-Room-Id': roomCode }; if (sessionToken) headers['X-Session-Token'] = sessionToken; try { const res = await fetch('api/set_ai', { method: 'POST', headers: headers, body: JSON.stringify({ ai_mode: aiMode }) }); const data = await res.json(); if (data.success) { console.log(`AI switched to ${aiMode} `); } else { alert('Failed to switch AI: ' + data.error); } } catch (e) { console.error("AI Switch failed", e); } } async function forceAction() { const id = parseInt(document.getElementById('force-id').value); console.log(`Force action: ${id} `); const headers = { 'Content-Type': 'application/json', 'X-Room-Id': roomCode }; if (sessionToken) headers['X-Session-Token'] = sessionToken; const res = await fetch('api/action', { method: 'POST', headers: headers, body: JSON.stringify({ action_id: id, force: true }) }); const text = await res.text(); lastStateJson = text; const data = JSON.parse(text); if (data.success) { state = data.state; render(); console.log('Forced'); } else { console.log('Error: ' + data.error); } } function showLastPerformance() { const modal = document.getElementById('performance-modal'); if (!modal) return; selectedPerfTurn = -1; // Reset to latest // Try to render using current performance results first let results = null; if (state && state.performance_results && Object.keys(state.performance_results).length > 0) { results = state.performance_results; } else if (state && state.last_performance_results && Object.keys(state.last_performance_results).length > 0) { results = state.last_performance_results; } else if (lastPerformanceData) { results = lastPerformanceData; } const titleEl = document.getElementById('perf-title'); if (titleEl) titleEl.textContent = "Performance Result"; if (results || (state && state.performance_history && state.performance_history.length > 0)) { renderPerformanceResult(results); modal.style.display = 'flex'; } else { console.log("No performance data available yet."); } } function selectHistoryTurn(turn) { selectedPerfTurn = turn; renderPerformanceResult(null); // Will use selectedPerfTurn internally } function showMatchHistory() { const modal = document.getElementById('performance-modal'); if (!modal || !state || !state.performance_history) return; const content = document.getElementById('perf-content'); const titleEl = document.getElementById('perf-title'); if (titleEl) titleEl.textContent = "Match Performance History"; if (state.performance_history.length === 0) { content.innerHTML = '
履歴がありません
'; } else { let html = '
'; // Sort by turn descending const sorted = [...state.performance_history].reverse(); sorted.forEach((res, idx) => { html += `< div style = "border-bottom: 2px dashed rgba(255,255,255,0.1); padding-bottom:30px;" >

Turn ${res.turn} - Player ${res.player_id === 0 ? 'P0 (You)' : 'P1 (Opp)'}

`; // We temporarily wrap one result to reuse renderPerformanceResult logic partly // or just manually render a mini version. // Let's do a simplified manual render for history list. html += `< div style = "display:flex; flex-wrap:wrap; gap:10px; margin-bottom:15px;" > `; res.lives.forEach(live => { const opacity = live.passed ? 1 : 0.4; const borderColor = live.passed ? 'var(--accent-green)' : '#555'; html += ` < div style = "width:120px; text-align:center; opacity:${opacity}; border:1px solid ${borderColor}; padding:5px; border-radius:6px; background:rgba(255,255,255,0.05);" >
${live.name}
${live.passed ? 'SUCCESS' : 'FAILED'}
`; }); html += ` `; html += ` `; }); html += ''; content.innerHTML = html; } modal.style.display = 'flex'; } function dismissPerformanceModal() { document.getElementById('performance-modal').style.display = 'none'; } async function forcedTurnEnd() { const headers = { 'X-Room-Id': roomCode }; // Simple endpoint, might need session? if (sessionToken) headers['X-Session-Token'] = sessionToken; const res = await fetch('api/force_turn_end', { method: 'POST', headers: headers }); fetchState(); } async function execCode() { const code = document.getElementById('god-code').value; console.log(`Exec: ${code.substring(0, 30)}...`); const headers = { 'Content-Type': 'application/json', 'X-Room-Id': roomCode }; if (sessionToken) headers['X-Session-Token'] = sessionToken; const res = await fetch('api/exec', { method: 'POST', headers: headers, body: JSON.stringify({ code: code }) }); const text = await res.text(); lastStateJson = text; const data = JSON.parse(text); if (data.success) { state = data.state; render(); log('Code executed'); } else { log('Error: ' + data.error); } } async function resetGame() { log('Resetting game...'); if (offlineMode) { const res = await wasmAdapter.resetGame(); if (res.success) { state = res.state; window.lastShownPerformanceHash = ""; render(); log('New game started'); } return; } const modeToUse = (state && state.mode) ? state.mode : (hotseatMode ? "pvp" : "pve"); const headers = { 'Content-Type': 'application/json', 'X-Room-Id': roomCode }; if (sessionToken) headers['X-Session-Token'] = sessionToken; try { const res = await fetch('api/reset', { method: 'POST', headers: headers, body: JSON.stringify({ mode: modeToUse }) }); if (!res.ok) { const errorText = await res.text(); log(`Reset failed: ${res.status} - ${errorText} `); console.error('Reset error:', errorText); return; } const text = await res.text(); lastStateJson = text; const data = JSON.parse(text); if (data.success) { state = data.state; window.lastShownPerformanceHash = ""; // Reset hash on new game render(); log('New game started'); fetchState(); } else { log(`Reset failed: ${data.error || 'Unknown error'} `); } } catch (error) { log(`Reset error: ${error.message} `); console.error('Reset exception:', error); } } // (Removed duplicate showLastPerformance) function selectLogTurn(t) { selectedTurn = t; render(); } // ===== REPLAY SYSTEM ===== let replayMode = false; let replayData = null; let currentFrame = 0; let playInterval = null; window.toggleReplayMode = function () { replayMode = !replayMode; const controls = document.getElementById('replay-controls'); if (controls) controls.style.display = replayMode ? 'flex' : 'none'; if (!replayMode && playInterval) stopPlay(); }; window.loadReplay = async function () { const filename = (document.getElementById('replay-file')?.value || 'ai_match.json').trim(); try { const res = await fetch(`/ api / replay / ${filename}?t = ${Date.now()} `); replayData = await res.json(); currentFrame = 0; const totalEl = document.getElementById('total-frames'); if (totalEl) totalEl.textContent = replayData.states.length; displayReplayFrame(); log(`Loaded: Game ${replayData.game_id + 1}, Winner: ${replayData.winner}, Phase: ${replayData.states[0]?.phase}, Frames: ${replayData.states.length} `); } catch (e) { log('Failed to load replay: ' + e.message); } }; window.loadReplayFromFile = function (input) { if (!input.files || !input.files[0]) return; const file = input.files[0]; const reader = new FileReader(); reader.onload = function (e) { try { const json = JSON.parse(e.target.result); // Basic validation if (!json.states || !Array.isArray(json.states)) { throw new Error("Invalid replay format: missing 'states' array"); } replayData = json; currentFrame = 0; const totalEl = document.getElementById('total-frames'); if (totalEl) totalEl.textContent = replayData.states.length; displayReplayFrame(); log(`Loaded File: ${file.name} (Phase: ${replayData.states[0]?.phase}, Frames: ${replayData.states.length})`); // Stop auto-play if running if (playInterval) stopPlay(); // Ensure replay mode UI is active if (!replayMode) toggleReplayMode(); } catch (err) { log('Error reading replay file: ' + err.message); console.error(err); alert('Error loading replay: ' + err.message); } }; reader.readAsText(file); input.value = ''; // Reset }; window.openPasteReplayModal = function () { document.getElementById('paste-replay-modal').style.display = 'flex'; const input = document.getElementById('paste-replay-input'); input.value = ''; input.focus(); }; window.closePasteReplayModal = function () { document.getElementById('paste-replay-modal').style.display = 'none'; }; window.submitPasteReplay = function () { const text = document.getElementById('paste-replay-input').value; if (!text.trim()) return; try { const json = JSON.parse(text); if (!json.states || !Array.isArray(json.states)) throw new Error("Invalid replay format: missing 'states'"); replayData = json; currentFrame = 0; const totalEl = document.getElementById('total-frames'); if (totalEl) totalEl.textContent = replayData.states.length; displayReplayFrame(); log(`Loaded from Clipboard(Phase: ${replayData.states[0]?.phase}, Frames: ${replayData.states.length})`); if (playInterval) stopPlay(); if (!replayMode) toggleReplayMode(); closePasteReplayModal(); } catch (e) { alert("Invalid JSON: " + e.message); } }; window.jumpToFrame = function (val) { const n = parseInt(val); if (replayData && !isNaN(n) && n >= 0 && n < replayData.states.length) { currentFrame = n; displayReplayFrame(); } }; // Keyboard Navigation for Replay window.addEventListener('keydown', (e) => { if (!replayMode) return; if (document.activeElement.tagName === 'INPUT') return; // Don't trigger when typing in boxes if (e.key === 'ArrowLeft') { replayPrev(); } else if (e.key === 'ArrowRight') { replayNext(); } }); function displayReplayFrame() { if (!replayData || currentFrame >= replayData.states.length) return; const frame = replayData.states[currentFrame]; const frameEl = document.getElementById('frame-num'); if (frameEl) frameEl.textContent = currentFrame; const jumpInput = document.getElementById('jump-frame'); if (jumpInput) jumpInput.value = currentFrame; // If we have full state data, render it completely if (frame.players && frame.players.length >= 2) { state = frame; // Set global state to the replay frame render(); // Use the existing render function log(`Frame ${currentFrame}: T${frame.turn} ${frame.phase} P${frame.current_player} (Act: ${frame.action_taken || 0})`); } else { // Fallback to minimal display document.getElementById('turn').textContent = frame.turn; document.getElementById('phase').textContent = frame.phase_name || frame.phase; document.getElementById('score').textContent = `${frame.p0_score} - ${frame.p1_score} `; log(`Frame ${currentFrame}: T${frame.turn} ${frame.phase_name} ${frame.p0_score} - ${frame.p1_score} `); } } window.replayPrev = function () { if (currentFrame > 0) { currentFrame--; displayReplayFrame(); } }; window.replayNext = function () { if (replayData && currentFrame < replayData.states.length - 1) { currentFrame++; displayReplayFrame(); } else if (playInterval) stopPlay(); }; window.replayPrevTurn = function () { if (!replayData || currentFrame <= 0) return; const currentTurn = replayData.states[currentFrame].turn; let i = currentFrame - 1; // Go back to the start of the current turn or to the previous turn while (i > 0 && replayData.states[i].turn === currentTurn) i--; currentFrame = i; displayReplayFrame(); }; window.replayNextTurn = function () { if (!replayData || currentFrame >= replayData.states.length - 1) return; const currentTurn = replayData.states[currentFrame].turn; let i = currentFrame + 1; while (i < replayData.states.length && replayData.states[i].turn === currentTurn) i++; if (i < replayData.states.length) currentFrame = i; displayReplayFrame(); }; window.replayPrevPhase = function () { if (!replayData || currentFrame <= 0) return; const currentPhase = replayData.states[currentFrame].phase; const currentTurn = replayData.states[currentFrame].turn; let i = currentFrame - 1; while (i > 0 && replayData.states[i].phase === currentPhase && replayData.states[i].turn === currentTurn) i--; currentFrame = i; displayReplayFrame(); }; window.replayNextPhase = function () { if (!replayData || currentFrame >= replayData.states.length - 1) return; const currentPhase = replayData.states[currentFrame].phase; const currentTurn = replayData.states[currentFrame].turn; let i = currentFrame + 1; while (i < replayData.states.length && replayData.states[i].phase === currentPhase && replayData.states[i].turn === currentTurn) i++; if (i < replayData.states.length) currentFrame = i; displayReplayFrame(); }; window.togglePlay = function () { playInterval ? stopPlay() : startPlay(); }; function startPlay() { const btn = document.getElementById('play-btn'); if (btn) btn.textContent = '⏸ Pause'; playInterval = setInterval(window.replayNext, 500); } function stopPlay() { if (playInterval) { clearInterval(playInterval); playInterval = null; const btn = document.getElementById('play-btn'); if (btn) btn.textContent = '▶ Play'; } } async function forceAdvance() { log('Forcing advance...'); const headers = { 'Content-Type': 'application/json', 'X-Room-Id': roomCode }; if (sessionToken) headers['X-Session-Token'] = sessionToken; try { const res = await fetch('api/advance', { method: 'POST', headers: headers, body: JSON.stringify({}) }); if (res.ok) { const data = await res.json(); if (data.success) { console.log("Fetched state:", data.state); state = data.state; render(); log('Advance complete'); } else { log('Advance error: ' + data.error); } } else { console.warn("Advance failed with status:", res.status); log('Advance failed: Server responded with status ' + res.status); } } catch (e) { log('Advance failed: ' + e.message); } } async function dismissPerformanceModal() { try { console.log("Dismissing performance modal..."); document.getElementById('performance-modal').style.display = 'none'; // No auto-action. Just close. } catch (e) { console.error("Error clearing performance:", e); } } function renderPerformanceResult(results) { const content = document.getElementById('perf-content'); if (!content) return; let html = ''; // Add History Selector at the top (grouped by turn) if (state && state.performance_history && state.performance_history.length > 0) { // Extract unique turns const availableTurns = [...new Set(state.performance_history.map(h => h.turn))].sort((a, b) => b - a); html += '
'; html += ``; availableTurns.forEach(t => { html += ``; }); html += '
'; } // Determine what to display let displayResults = results; if (selectedPerfTurn >= 0) { // Group by player for the selected turn displayResults = {}; if (state && state.performance_history) { state.performance_history.forEach(h => { if (h.turn === selectedPerfTurn) { displayResults[h.player_id] = h; } }); } } else if (!displayResults) { // Fallback for current if none passed (e.g. from selectHistoryTurn) if (state && state.performance_results && Object.keys(state.performance_results).length > 0) { displayResults = state.performance_results; } else if (state && state.last_performance_results && Object.keys(state.last_performance_results).length > 0) { displayResults = state.last_performance_results; } else { displayResults = lastPerformanceData; } } // Safety fallback to prevent crashes on [0, 1].forEach if (!displayResults) displayResults = {}; html += '
'; const fixImg = (path) => { if (!path) return 'icon_blade.png'; if (path.startsWith('/') || path.startsWith('http')) return path; return 'img/' + path; }; // Iterate over all players who might have performed [0, 1].forEach(pid => { const res = displayResults[pid]; const playerName = pid === 0 ? "Player (You)" : "Opponent (AI)"; // Use a darker, more premium background const bgColor = pid === 0 ? "rgba(255,105,180,0.15)" : "rgba(100,200,255,0.15)"; const borderColor = pid === 0 ? "#ff69b4" : "#64c8ff"; if (!res) { html += `

${playerName}

No Performance / Skip
`; return; } html += `
`; // Header (Status) const statusColor = res.success ? '#4f4' : '#f44'; const statusIcon = res.success ? '🎉 SUCCESS' : '💔 FAILURE'; const statusBg = res.success ? 'rgba(76, 175, 80, 0.2)' : 'rgba(244, 67, 54, 0.2)'; html += `

${playerName}

${statusIcon} ${Array.isArray(res.lives) ? `(+${res.lives.reduce((acc, l) => acc + (l.passed ? (l.score || 0) : 0), 0)} PTS)` : ''}
`; // Main Layout: 3 Columns (Sources -> Total -> Goal) html += `
`; // 1. SOURCES (Detailed Breakdown) html += `
Source Logs
BLADE LOG:
${res.breakdown && res.breakdown.blades ? res.breakdown.blades.map(b => { const isInactive = b.type === 'inactive' || b.type === 'empty'; const opacity = isInactive ? '0.5' : '1'; const srcCard = resolveCardData(b.source_id); const tooltipText = srcCard ? getEffectiveAbilityText(srcCard) : b.source; return `
${b.source_id && b.source_id >= 0 ? `` : ''} ${b.source}
${b.value > 0 ? '+' : ''}${b.value}
`; }).join('') : '
No voltage sources
'}
HEART LOG:
${res.breakdown && res.breakdown.hearts ? res.breakdown.hearts.map(h => { if (h.type === 'transform') { return `
${h.source}: ${h.desc || h.text}
`; } const isInactive = h.type === 'inactive' || h.type === 'empty'; const opacity = isInactive ? '0.5' : '1'; const srcCard = resolveCardData(h.source_id); const tooltipText = srcCard ? getEffectiveAbilityText(srcCard) : h.source; return `
${h.source_id && h.source_id >= 0 ? `` : ''} ${h.source}
${renderHeartsCompact(h.value)}
`; }).join('') : '
No heart sources
'}
${res.breakdown && (res.breakdown.requirements || res.breakdown.transforms) ? `
GLOBAL / REQ LOG:
${res.breakdown.transforms ? res.breakdown.transforms.map(t => `
${t.desc}
` ).join('') : ''} ${res.breakdown.requirements ? res.breakdown.requirements.map(r => (r.type === 'req_mod') ? `
${r.source} (Req) ${renderHeartsCompact(r.value)}
` : '' ).join('') : ''}
` : ''}
Judgment Score:
Base (Live Cards) ${res.lives.reduce((acc, l) => acc + (l.passed ? (l.score || 0) : 0), 0)}
Score Icons (Notes) +${res.yell_score_bonus || 0}
${res.breakdown && res.breakdown.score_modifiers && res.breakdown.score_modifiers.length > 0 ? res.breakdown.score_modifiers.map(m => `
${m.source} +${m.value}
`).join('') : ''}
TOTAL ${res.lives.reduce((acc, l) => acc + (l.passed ? (l.score || 0) : 0), 0) + (res.yell_score_bonus || 0) + (res.breakdown && res.breakdown.score_modifiers ? res.breakdown.score_modifiers.reduce((acc, m) => acc + m.value, 0) : 0)}
${res.member_contributions && res.member_contributions.length > 0 ? `
MEMBER CONTRIBUTIONS:
${res.member_contributions.map(m => { const hasValue = m.hearts.some(v => v > 0) || m.blades > 0; if (!hasValue) return ''; return `
${m.source}
${m.volume_icons > 0 ? `${m.volume_icons}` : ''} ${m.draw_icons > 0 ? `${m.draw_icons}` : ''} ${m.blades > 0 ? `${m.blades}` : ''} ${renderHeartsCompact(m.hearts)}
`; }).join('')}
` : ''}
${res.yell_cards.length > 0 ? `
Yell Cards:
${res.yell_cards.map(y => { const hasHearts = y.blade_hearts && y.blade_hearts.some(h => h > 0); const hasVol = y.volume_icons > 0; const hasDraw = y.draw_icons > 0; return `
${hasHearts ? `
${renderBladeHeartsCompact(y.blade_hearts)}
` : ''} ${hasVol ? `
` : ''} ${hasDraw ? `
` : ''}
`; }).join('')}
` : ''}
`; // Arrow html += `
`; // 2. TOTAL (The Bank) html += `
Total Hearts
${renderTotalHeartsBreakdown(res.total_hearts)}
`; // Arrow html += `
`; // 3. GOAL (Live Cards) html += `
Live Requirements
`; res.lives.forEach(l => { const lColor = l.passed ? '#4f4' : '#f44'; const lBg = l.passed ? 'rgba(76, 175, 80, 0.1)' : 'rgba(244, 67, 54, 0.1)'; html += `
${l.name}
Progress:
${renderHeartProgress(l.filled, l.required)}
${l.reason ? `
${l.reason}
` : ''}
${l.passed ? '✓' : '✗'}
${l.passed ? 'PASSED' : 'FAILED'}
${l.passed ? `
+${l.score} PTS
` : ''}
`; }); html += `
`; // End Goals html += `
`; // End Flex Row html += `
`; // End Player Box }); html += ''; content.innerHTML = html; } function renderHeartProgress(filled, required) { if (!required) return ''; const safeFilled = [...(filled || new Array(7).fill(0))]; const safeReq = [...required]; let html = '
'; // 1. Process specific requirements (0-5) for (let idx = 0; idx < 6; idx++) { const reqCount = safeReq[idx]; if (reqCount > 0) { const fillCount = Math.min(reqCount, safeFilled[idx]); safeFilled[idx] -= fillCount; // Keep track of remaining for 'Any' const colorClass = `color-${idx}`; html += `
`; for (let i = 0; i < fillCount; i++) html += `
`; for (let i = 0; i < (reqCount - fillCount); i++) html += `
`; html += `
`; } } // 2. Process 'Any' requirements (6) const anyReqCount = safeReq[6] || 0; if (anyReqCount > 0) { html += `
`; let remainingAny = anyReqCount; // Show specific colors that satisfied 'Any' for (let idx = 0; idx < 7; idx++) { const colorClass = idx === 6 ? 'color-any' : `color-${idx}`; const fillCount = Math.min(remainingAny, safeFilled[idx]); for (let i = 0; i < fillCount; i++) { html += `
`; remainingAny--; } } // Remaining empty slots for (let i = 0; i < remainingAny; i++) { html += `
`; } html += `
`; } html += '
'; return html; } function renderHeartsCompact(hearts) { // Renders hearts as grouped numbers (e.g. [Pink Icon]x2) instead of individual pips if (!hearts) return ''; let html = '
'; let hasAny = false; hearts.forEach((count, idx) => { if (count > 0) { hasAny = true; const isAny = idx === 6; const colorClass = isAny ? 'color-any' : `color-${idx}`; const iconHtml = isAny ? `` : `
`; html += `
${iconHtml} ${count}
`; } }); if (!hasAny) return '-'; html += '
'; return html; } function renderBladeHeartsCompact(hearts) { return renderHeartsCompact(hearts).replace(/border-radius:50%/g, 'clip-path: polygon(50% 0%, 100% 85%, 50% 100%, 0% 85%)'); // Crude blade shape } function renderTotalHeartsBreakdown(hearts) { if (!hearts) return ''; let html = ''; // Python Order: Pink, Red, Yellow, Green, Blue, Purple const colors = ['Pink', 'Red', 'Yellow', 'Green', 'Blue', 'Purple', 'Any']; const colorCodes = ['#ff8da1', '#ef5350', '#ffd32a', '#76d672', '#4aaef7', '#ba68c8', 'white']; hearts.forEach((count, idx) => { if (count > 0) { html += `
${colors[idx]}
${count}
`; } }); if (html === '') return '
None
'; return html; } // Render Report Modal function renderReportModal() { // ... existing report modal code if present or just placeholder } function renderPerformanceGuide() { const p0 = state.players[perspectivePlayer] || state.players[0]; // Respect perspective const guide = p0.performance_guide; const panel = document.getElementById('perf-guide-panel'); if (!guide || guide.lives.length === 0) { panel.style.display = 'none'; return; } panel.style.display = 'block'; let html = `
Blades: ${guide.total_blades} Hearts: ${renderHeartsCompact(guide.total_hearts)}
`; guide.lives.forEach(l => { const color = l.passed ? '#4f4' : '#f44'; html += `
${l.name} (${l.score}pts)
${renderHeartProgress(l.filled, l.required)}
${!l.passed ? `
${l.reason}
` : ''}
${l.passed ? '✓' : '✗'}
`; }); // Add Detailed Breakdown Log if (guide.breakdown) { html += `
`; // Detailed log content (Voltage) html += `
BLADE LOG
${guide.breakdown.blades ? guide.breakdown.blades.map(b => { const isInactive = b.type === 'inactive' || b.type === 'empty'; const srcCard = resolveCardData(b.source_id); const tooltipText = srcCard ? getEffectiveAbilityText(srcCard) : b.source; return `
${b.source} ${b.value > 0 ? '+' : ''}${b.value}
`; }).join('') : '' }
`; // Detailed log content (Hearts) html += `
HEART LOG
${guide.breakdown.hearts ? guide.breakdown.hearts.map(h => { if (h.type === 'transform') { return `
${h.desc || h.text}
`; } const isInactive = h.type === 'inactive' || h.type === 'empty'; const srcCard = resolveCardData(h.source_id); const tooltipText = srcCard ? getEffectiveAbilityText(srcCard) : h.source; return `
${h.source} ${renderHeartsCompact(h.value)}
`; }).join('') : '' }
`; // Requirements / Global Log if (guide.breakdown.requirements || guide.breakdown.transforms) { html += `
GLOBAL / REQUIREMENTS
${guide.breakdown.transforms ? guide.breakdown.transforms.map(t => `
${t.desc}
` ).join('') : '' } ${guide.breakdown.requirements ? guide.breakdown.requirements.map(r => (r.type === 'req_mod') ? `
${r.source} (Req Reduced) ${renderHeartsCompact(r.value)}
` : '' ).join('') : '' }
`; } html += `
`; } document.getElementById('perf-guide-content').innerHTML = html; } function renderLookedCards() { const panel = document.getElementById('looked-cards-panel'); const content = document.getElementById('looked-cards-content'); const cards = state.looked_cards || []; if (cards.length === 0) { panel.style.display = 'none'; return; } panel.style.display = 'block'; let html = ''; cards.forEach((c, idx) => { const actionId = 600 + idx; // SELECT action IDs start at 600 const isLive = c.type === 'live'; const tooltip = getEffectiveAbilityText(c); html += `
${c.name}
`; }); content.innerHTML = html; } async function fetchState() { try { if (replayMode) return; // Stop polling while watching replay if (offlineMode) { if (!wasmAdapter) return; const res = await wasmAdapter.fetchState(); if (res.success) { // Check change const raw = JSON.stringify(res.state); if (raw === lastStateJson) return; lastStateJson = raw; state = res.state; render(); } return; } if (!roomCode) return; // Wait for room // Pause updates if Performance Modal is open const perfModal = document.getElementById('performance-modal'); if (perfModal && perfModal.style.display !== 'none') return; const headers = { 'X-Room-Id': roomCode }; if (sessionToken) headers['X-Session-Token'] = sessionToken; const res = await fetch('api/state?viewer=' + perspectivePlayer, { headers: headers }); if (res.status === 404) { // Room invalid or expired console.warn("Room not found (404). Resetting."); roomCode = null; state = null; localStorage.removeItem('lovelive_room_code'); document.getElementById('room-modal').style.display = 'flex'; console.log("DEBUG: Room 404 Reset"); updateRoomDisplay(); return; } const raw = await res.text(); if (raw === lastStateJson) return; // Skip if no change lastStateJson = raw; // Check if raw is HTML (typical for 404 or errors not handled by status check) if (raw.trim().startsWith('<')) { console.error("Received HTML instead of JSON state. API might be down or path incorrect."); return; } const data = JSON.parse(raw); if (data.success) { state = data.state; console.log("[DEBUG] State Mode:", state.mode); } else { console.error("State fetch unsuccessful:", data.error); return; } // Auto-sync perspective if authenticated if (state.my_player_id !== undefined && state.my_player_id !== -1 && !hotseatMode) { perspectivePlayer = state.my_player_id; } // Keep track of last performance for review if (state.performance_results && Object.keys(state.performance_results).length > 0) { lastPerformanceData = state.performance_results; } if (state.last_performance_results && Object.keys(state.last_performance_results).length > 0) { lastPerformanceData = state.last_performance_results; } render(); } catch (e) { console.error("Fetch Error:", e); } } // Initial setup fetchState(); setInterval(fetchState, 200); // Polling every 200ms (near-instant phase detection) // Extended logic for reporting let actionHistory = []; window.openReportModal = function () { document.getElementById('report-modal').style.display = 'flex'; document.getElementById('report-explanation').focus(); }; window.closeReportModal = function () { document.getElementById('report-modal').style.display = 'none'; }; window.downloadReport = function () { const explanation = document.getElementById('report-explanation').value; // Use current global state const reportData = { timestamp: new Date().toISOString(), explanation: explanation, state: state, history: actionHistory, errors: window.capturedErrors || [], navigator: navigator.userAgent }; const blob = new Blob([JSON.stringify(reportData, null, 2)], { type: 'application/json' }); const url = URL.createObjectURL(blob); const a = document.createElement('a'); a.href = url; // Format: report_YYYYMMDD_HHMMSS.json const dateStr = new Date().toISOString().replace(/[:.]/g, '-').slice(0, 19); a.download = `loveca_report_${dateStr}.json`; document.body.appendChild(a); a.click(); document.body.removeChild(a); URL.revokeObjectURL(url); }; window.submitReport = async function () { const explanation = document.getElementById('report-explanation').value; if (!explanation.trim()) { alert("Please provide an explanation."); return; } console.log("Submitting report..."); const headers = { 'Content-Type': 'application/json', 'X-Room-Id': roomCode }; if (sessionToken) headers['X-Session-Token'] = sessionToken; try { const res = await fetch('api/report_issue', { method: 'POST', headers: headers, body: JSON.stringify({ explanation: explanation, state: state, history: actionHistory }) }); const data = await res.json(); if (data.success) { console.log(`Report saved: ${data.filename} `); alert(`Bug report saved to ${data.filename}.\n\nPlease include this path when asking for a fix!`); closeReportModal(); } else { console.log(`Report failed: ${data.error} `); alert(`Failed to save report: ${data.error} `); } } catch (e) { console.log(`Report error: ${e.message} `); alert(`Error submitting report: ${e.message} `); } }; // Update doAction to track history const originalDoAction = window.doAction; window.doAction = async function (id) { actionHistory.push({ action_id: id, timestamp: new Date().toISOString(), phase: state ? state.phase : 'unknown', legal_actions: state ? state.legal_actions : [] }); // Keep only last 50 actions if (actionHistory.length > 50) actionHistory.shift(); return originalDoAction(id); }; // Deck Modal Functions window.openDeckModal = function () { document.getElementById('deck-modal').style.display = 'flex'; document.getElementById('deck-html-input').value = ''; document.getElementById('deck-preview').textContent = ''; }; // --- Perspective Switching --- window.setPerspective = function (pid) { perspectivePlayer = pid; document.getElementById('perspective-modal').style.display = 'none'; fetchState(); // Update labels const myName = document.getElementById('my-agent-name'); const oppName = document.getElementById('opp-agent-name'); if (pid === 0) { if (myName) myName.innerText = "(Player 1)"; if (oppName) oppName.innerText = "(Player 2)"; } else { if (myName) myName.innerText = "(Player 2)"; if (oppName) oppName.innerText = "(Player 1)"; } }; window.closeDeckModal = function () { document.getElementById('deck-modal').style.display = 'none'; }; window.submitDeck = async function () { const playerVal = document.getElementById('deck-player-select').value; let content = ''; const fileInput = document.getElementById('deck-file-input'); const file = fileInput.files[0]; const textInput = document.getElementById('deck-html-input').value; if (file) { try { content = await file.text(); } catch (e) { alert("Failed to read file: " + e.message); return; } } else if (textInput.trim()) { content = textInput; } else { alert('Please select a file or paste deck HTML.'); return; } const playerIds = (playerVal === 'both') ? [0, 1] : [parseInt(playerVal)]; if (offlineMode) { for (const pid of playerIds) { const res = await wasmAdapter.uploadDeck(pid, content); if (res.success) { console.log(`[Offline] Deck loaded for P${pid}: ${res.message}`); } else { alert(`Failed to load deck for P${pid}: ${res.error}`); } } closeDeckModal(); fetchState(); return; } console.log("Uploading deck..."); const headers = { 'Content-Type': 'application/json', 'X-Room-Id': roomCode }; if (sessionToken) headers['X-Session-Token'] = sessionToken; for (const pid of playerIds) { try { const resp = await fetch('api/upload_deck', { method: 'POST', headers: headers, body: JSON.stringify({ player: pid, content: content }) }); const result = await resp.json(); if (result.success) { alert(`Deck set for Player ${pid + 1}: ${result.message} `); closeDeckModal(); // Refresh state to see new deck fetchState(); } else { alert(`Failed to set deck for P${pid + 1}: ` + (result.error || 'Unknown error')); } } catch (e) { alert(`Error submitting deck for P${pid + 1}: ` + e.message); } } }; // Load Test Deck Preset window.loadTestDeck = async function () { const playerVal = document.getElementById('deck-player-select').value; const playerIds = (playerVal === 'both') ? [0, 1] : [parseInt(playerVal)]; if (!confirm(`Load 'Test Deck'(doubled quantities) for Player ${playerVal === 'both' ? 'Both' : parseInt(playerVal) + 1}?`)) return; try { // 1. Get the compiled deck list from server const res = await fetch('api/get_test_deck'); const data = await res.json(); if (!data.success) { alert("Failed to load test deck: " + data.error); return; } const cards = data.content; // Array of strings e.g. ["PL!...", "PL!..."] // 2. Upload this list as a "custom deck" // We reuse upload_deck but pass JSON content console.log(`Uploading test deck(${cards.length} cards)...`); for (const pid of playerIds) { const resp = await fetch('api/upload_deck', { method: 'POST', headers: { 'Content-Type': 'application/json', 'X-Room-Id': roomCode }, body: JSON.stringify({ player: pid, content: JSON.stringify(cards) // Send as JSON string of list }) }); const result = await resp.json(); if (result.success) { alert(`Test Deck loaded for P${pid + 1}: ${result.message} `); closeDeckModal(); fetchState(); } else { alert(`Failed to load P${pid + 1}: ` + result.error); } } } catch (e) { console.error(e); alert("Error loading test deck: " + e.message); } }; // Help Modal window.openHelpModal = function () { document.getElementById('help-modal').style.display = 'flex'; }; window.closeHelpModal = function () { document.getElementById('help-modal').style.display = 'none'; }; // Load Random Deck window.loadRandomDeck = async function () { const playerVal = document.getElementById('deck-player-select').value; const playerIds = (playerVal === 'both') ? [0, 1] : [parseInt(playerVal)]; if (!confirm(`Generate Random Deck for Player ${playerVal === 'both' ? 'Both' : parseInt(playerVal) + 1}?`)) return; try { const res = await fetch('api/get_random_deck'); const data = await res.json(); if (!data.success) { alert("Failed to generate deck: " + data.error); return; } const cards = data.content; console.log(`Uploading random deck(${cards.length} cards)...`); for (const pid of playerIds) { const resp = await fetch('api/upload_deck', { method: 'POST', headers: { 'Content-Type': 'application/json', 'X-Room-Id': roomCode }, body: JSON.stringify({ player: pid, content: JSON.stringify(cards) }) }); const result = await resp.json(); if (result.success) { alert(`Random Deck Loaded for P${pid + 1}`); closeDeckModal(); fetchState(); } else { alert(`Failed to load P${pid + 1}: ` + result.error); } } } catch (e) { console.error(e); alert("Error loading random deck: " + e.message); } }; // Preview deck as user types - validates card IDs against database let previewTimeout = null; const deckHtmlInput = document.getElementById('deck-html-input'); if (deckHtmlInput) { deckHtmlInput.addEventListener('input', function () { const html = this.value; const regex = /title="([^"]+?) :[^"]*"[^>]*>.*?class="num">(\d+)<\/span>/gs; const cards = {}; let count = 0; let match; while ((match = regex.exec(html)) !== null) { const cardId = match[1].trim(); const qty = parseInt(match[2], 10); cards[cardId] = (cards[cardId] || 0) + qty; count += qty; } if (count === 0) { document.getElementById('deck-preview').textContent = ''; return; } // Debounce validation API call clearTimeout(previewTimeout); previewTimeout = setTimeout(async () => { const cardIds = Object.keys(cards); try { const resp = await fetch('api/validate_cards', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ card_ids: cardIds, card_counts: cards }) }); const result = await resp.json(); const unknownSet = new Set(result.unknown); const breakdown = result.breakdown || { member: 0, live: 0, energy: 0 }; // Build card list with known/unknown highlighting and names const cardList = Object.entries(cards) .sort((a, b) => a[0].localeCompare(b[0])) .map(([id, n]) => { const isUnknown = unknownSet.has(id); const info = result.card_info?.[id]; const name = info?.name || ''; const type = info?.type || ''; const typeColor = type === 'Member' ? '#4CAF50' : type === 'Live' ? '#2196F3' : type === 'Energy' ? '#FF9800' : '#888'; const style = isUnknown ? 'color: #ff4444; font-weight: bold;' : ''; const badge = isUnknown ? ' ⚠️' : ''; const typeTag = type ? `${type}` : ''; return `
${id} x${n}${typeTag} ${name}${badge}
`; }) .join(''); document.getElementById('deck-preview').innerHTML = `Detected ${count} cards(${cardIds.length} unique):
👤 ${breakdown.member} Member 🎵 ${breakdown.live} Live ● ${breakdown.energy} Energy
${result.unknown_count > 0 ? `
⚠️ ${result.unknown_count} NOT in database
` : ''}
${cardList}
`; } catch (e) { console.error('Validation error:', e); } }, 300); }); } // Live Watch Logic let liveWatchInterval = null; let isLiveWatchOn = false; function toggleLiveWatch() { const btn = document.getElementById('live-watch-btn'); if (isLiveWatchOn) { // Turn OFF isLiveWatchOn = false; if (liveWatchInterval) { clearInterval(liveWatchInterval); liveWatchInterval = null; } btn.style.background = 'linear-gradient(135deg, #cc4444, #882222)'; console.log('Live Watch Disabled'); } else { // Turn ON isLiveWatchOn = true; btn.style.background = 'linear-gradient(135deg, #44cc44, #228822)'; console.log('Live Watch Enabled - Polling /api/state'); liveWatchInterval = setInterval(() => { fetchState(); }, 1000); } if (typeof updateLanguage === 'function') updateLanguage(); } // Init Language if (typeof updateLanguage === 'function') updateLanguage(); // --- Mobile Sidebar Logic --- function toggleSidebar() { const sidebar = document.querySelector('.sidebar'); const btn = document.getElementById('mobile-sidebar-toggle'); sidebar.classList.toggle('active'); document.body.classList.toggle('sidebar-open'); if (sidebar.classList.contains('active')) { btn.textContent = '✕'; btn.style.background = '#444'; } else { btn.textContent = '☰'; btn.style.background = 'var(--accent-pink)'; } } // --- Drag and Drop Logic --- let draggedCardIdx = -1; function initDragAndDrop() { // This function will be called after render to attach events // But since we render dynamically, we can attach inline or delegate. // Let's attach via delegation or re-attach in render functions. // Actually, simplest is to add attributes in renderCards and global listeners. } document.addEventListener('dragstart', (e) => { if (e.target.classList.contains('card') && e.target.closest('#my-hand')) { const idParts = e.target.id.split('-'); // id is 'my-hand-card-X' draggedCardIdx = parseInt(idParts[3]); e.dataTransfer.setData('text/plain', draggedCardIdx); e.dataTransfer.effectAllowed = 'move'; e.target.style.opacity = '0.4'; } }); document.addEventListener('dragend', (e) => { if (e.target.classList.contains('card')) { e.target.style.opacity = '1'; draggedCardIdx = -1; document.querySelectorAll('.member-slot').forEach(el => el.classList.remove('drag-over')); } }); document.addEventListener('dragover', (e) => { const slot = e.target.closest('.member-slot'); if (slot && draggedCardIdx !== -1) { e.preventDefault(); // Allow dropping e.dataTransfer.dropEffect = 'move'; slot.classList.add('drag-over'); } }); document.addEventListener('dragleave', (e) => { const slot = e.target.closest('.member-slot'); if (slot) { slot.classList.remove('drag-over'); } }); document.addEventListener('drop', (e) => { const slot = e.target.closest('.member-slot'); if (slot && draggedCardIdx !== -1) { e.preventDefault(); slot.classList.remove('drag-over'); // Identify Area Index const slotIdParts = slot.id.split('-'); // id is 'my-stage-slot-X' if (slotIdParts[1] === 'stage') { const areaIdx = parseInt(slotIdParts[3]); handleCardDrop(draggedCardIdx, areaIdx); } } }); function handleCardDrop(handIdx, areaIdx) { console.log(`[DnD] Dropped Hand[${handIdx}] onto Area[${areaIdx}]`); // Logic similar to onStageSlotClick but triggered by drop // Calculate Action ID: 1 + (hand_idx * 3) + area_idx const actionId = 1 + (handIdx * 3) + areaIdx; // Check legality const action = state.legal_actions.find(a => a.id === actionId); if (action) { log(`Playing Hand[${handIdx}] to Area ${areaIdx} (DnD)`); doAction(actionId); } else { log(`Cannot play Hand[${handIdx}] to Area ${areaIdx} (Invalid Move)`); // Maybe visual shake effect? } } // Game Loop (Adaptive Polling) let pollingTimer = null; let currentPollingInterval = 2000; function setPollingInterval(ms) { if (currentPollingInterval === ms && pollingTimer) return; console.log(`[POLLING] Changing interval to ${ms}ms`); currentPollingInterval = ms; if (pollingTimer) clearInterval(pollingTimer); pollingTimer = setInterval(() => { if (roomCode && document.getElementById('room-modal').style.display === 'none') { fetchState(); updateAdaptivePolling(); // Periodically re-evaluate } }, ms); } function updateAdaptivePolling() { if (!state) return; const isOurTurn = (state.active_player === perspectivePlayer); const isAutoPhase = [Phase.ACTIVE, Phase.ENERGY, Phase.DRAW, Phase.PERFORMANCE_P1, Phase.PERFORMANCE_P2, Phase.LIVE_RESULT].includes(state.phase); if (!isOurTurn || isAutoPhase) { // Fast poll when waiting for AI or auto-advance setPollingInterval(500); } else { // Slow poll when user is thinking to reduce server load setPollingInterval(2000); } } // Initial Start setPollingInterval(2000); // --- Image Pre-loading for Instant Tooltips --- const preloadedUrls = new Set(); function preloadAssets(additionalUrls = []) { // Icons are now inlined as Base64 in ICON_DATA_URIs additionalUrls.forEach(url => { if (url && !preloadedUrls.has(url)) { const img = new Image(); img.src = fixImg(url); preloadedUrls.add(url); } }); } // Initial run preloadAssets(); // Global Image Preloading (PRO) async function preloadAllCards() { console.log("[PRELOAD] Starting background card image preloading..."); try { const res = await fetch('data/cards_compiled.json'); if (!res.ok) return; const data = await res.json(); const cards = data.member_db || {}; const urls = []; for (const key in cards) { const card = cards[key]; const imgPath = card.img_path || card.img || card._img; if (imgPath) { urls.push(imgPath); } } console.log(`[PRELOAD] Found ${urls.length} card images to preload.`); // Chunk preloading to avoid flooding the network const chunkSize = 20; for (let i = 0; i < urls.length; i += chunkSize) { const chunk = urls.slice(i, i + chunkSize); await Promise.all(chunk.map(url => { return new Promise((resolve) => { const img = new Image(); img.onload = () => resolve(); img.onerror = () => resolve(); // Skip failures img.src = fixImg(url); }); })); if (i % 100 === 0 && i > 0) console.log(`[PRELOAD] Progress: ${i}/${urls.length}...`); } console.log("[PRELOAD] All card images cached."); } catch (e) { console.warn("[PRELOAD] Failed to preload cards:", e); } } preloadAllCards(); // Initial debug console.log("Main.js loaded v4 (Settings Fixed)"); updateLanguage(); preloadAssets(); fetchAndPopulateDecks(); // --- End of main.js ---