<!DOCTYPE html>
<html>
<head>
    <title>Info Chess</title>
    <style>
        * { box-sizing: border-box; margin: 0; padding: 0; }
        body {
            background: #1c1c1e;
            color: #f0f0f0;
            font-family: 'Courier New', monospace;
            display: flex;
            flex-direction: column;
            align-items: center;
            padding: 20px;
            min-height: 100vh;
        }

        h1 { font-size: 20px; letter-spacing: 2px; margin-bottom: 6px; color: #aaa; }

        #team-banner {
            font-size: 26px;
            font-weight: bold;
            margin-bottom: 10px;
            padding: 6px 24px;
            border-radius: 6px;
        }
        .banner-white { background: #e8e0c8; color: #111; }
        .banner-black { background: #2a2a2a; color: #eee; border: 1px solid #555; }
        .banner-spectator { background: #333; color: #999; }

        #phase-bar {
            font-size: 15px;
            margin-bottom: 14px;
            padding: 6px 16px;
            background: #2a2a2a;
            border-radius: 6px;
            border-left: 4px solid #888;
            min-width: 340px;
            text-align: center;
        }
        .phase-move   { border-left-color: #4caf50; }
        .phase-king   { border-left-color: #2196f3; }
        .phase-guess  { border-left-color: #ff9800; }
        .phase-wait   { border-left-color: #555; color: #777; }
        .phase-over   { border-left-color: #f44336; }

        .board-area {
            display: flex;
            align-items: flex-start;
            gap: 6px;
        }
        .coord-col {
            display: flex;
            flex-direction: column;
            justify-content: space-around;
            height: 480px;
            width: 18px;
            text-align: right;
            font-size: 11px;
            color: #666;
            padding-right: 3px;
        }
        .coord-row {
            display: flex;
            width: 480px;
            font-size: 11px;
            color: #666;
            margin-top: 3px;
            margin-left: 18px;
        }
        .coord-row span { flex: 1; text-align: center; }

        #board {
            display: grid;
            grid-template-columns: repeat(8, 60px);
            grid-template-rows: repeat(8, 60px);
            border: 2px solid #444;
        }

        .cell {
            width: 60px; height: 60px;
            display: flex; align-items: center; justify-content: center;
            position: relative;
            cursor: pointer;
            font-size: 34px;
            user-select: none;
            transition: filter 0.1s;
        }
        .cell:hover { filter: brightness(1.2); }

        .light { background: #d4b896; }
        .dark  { background: #8b6343; }

        .fog { background: #111 !important; cursor: default; }
        .fog:hover { filter: none; }

        .selected    { outline: 3px solid #00e676 inset; }
        .legal-empty::after {
            content: '';
            width: 20px; height: 20px;
            background: rgba(0, 230, 118, 0.5);
            border-radius: 50%;
            position: absolute;
            pointer-events: none;
        }
        .legal-occ   { outline: 3px solid rgba(0,230,118,0.7) inset; }
        .guess-target { outline: 3px solid #ff9800 inset; }
        .king-sq      { outline: 3px solid #2196f3 inset; }

        /* Piece symbols */
        .piece-white { color: #fff; text-shadow: 0 0 3px #000, 0 0 6px #000; }
        .piece-black { color: #111; text-shadow: 0 0 3px #fff3, 0 0 6px #999; }

        #sidebar {
            margin-left: 20px;
            width: 220px;
            display: flex;
            flex-direction: column;
            gap: 12px;
        }

        .score-box {
            background: #2a2a2a;
            border-radius: 6px;
            padding: 10px 14px;
            font-size: 14px;
        }
        .score-box .label { color: #888; font-size: 11px; margin-bottom: 4px; }
        .score-row { display: flex; justify-content: space-between; margin: 2px 0; }
        .score-num { font-size: 22px; font-weight: bold; }

        #log-box {
            background: #111;
            border-radius: 6px;
            padding: 8px;
            height: 260px;
            overflow-y: auto;
            font-size: 12px;
        }
        .log-entry { padding: 2px 0; border-bottom: 1px solid #1e1e1e; color: #aaa; }
        .log-entry.good { color: #4caf50; }
        .log-entry.bad  { color: #f44336; }
        .log-entry.info { color: #888; }

        button {
            padding: 8px;
            background: #333;
            color: #ddd;
            border: 1px solid #555;
            border-radius: 5px;
            cursor: pointer;
            font-family: inherit;
            font-size: 13px;
        }
        button:hover { background: #444; }

        #skip-king-btn {
            background: #1a3a5c;
            border-color: #2196f3;
            color: #90caf9;
            display: none;
        }

        .turn-indicator {
            font-size: 12px;
            color: #666;
            text-align: center;
        }
    </style>
</head>
<body>
<h1>INFO CHESS</h1>
<div id="team-banner" class="banner-spectator">Connecting…</div>
<div id="phase-bar">—</div>

<div style="display:flex">
    <div class="board-area">
        <div class="coord-col" id="coord-col"></div>
        <div>
            <div id="board"></div>
            <div class="coord-row" id="coord-row"></div>
        </div>
    </div>
    <div id="sidebar">
        <div class="score-box">
            <div class="label">SCORE (correct guesses / 10)</div>
            <div class="score-row">
                <span>⬜ White</span><span class="score-num" id="score-white">0</span>
            </div>
            <div class="score-row">
                <span>⬛ Black</span><span class="score-num" id="score-black">0</span>
            </div>
            <div class="turn-indicator" id="turn-counter">Turn 0 / 20</div>
        </div>
        <button id="skip-king-btn" onclick="skipKing()">Skip king move (S)</button>
        <button onclick="requestNewGame()">New Game</button>
        <div class="label" style="color:#555;font-size:11px">Event log</div>
        <div id="log-box"></div>
    </div>
</div>

<script>
const PIECES = {
    white: { king:'♔', rook:'♖', bishop:'♗', pawn:'♙' },
    black: { king:'♚', rook:'♜', bishop:'♝', pawn:'♟' },
};
const FILES = ['a','b','c','d','e','f','g','h'];

let socket = new WebSocket('ws://localhost:8080');
let myTeam = null;
let state   = null;
let selected = null;
let legalMovesMap = {};

const boardEl = document.getElementById('board');
const cells = [];
for (let r = 0; r < 8; r++) {
    cells[r] = [];
    for (let c = 0; c < 8; c++) {
        const cell = document.createElement('div');
        cell.className = 'cell ' + ((r+c)%2===0 ? 'light' : 'dark');
        cell.dataset.r = r; cell.dataset.c = c;
        cell.addEventListener('click', () => onCellClick(r, c));
        boardEl.appendChild(cell);
        cells[r][c] = cell;
    }
}

const coordCol = document.getElementById('coord-col');
for (let r = 0; r < 8; r++) {
    const d = document.createElement('div'); d.textContent = 8-r;
    coordCol.appendChild(d);
}
const coordRow = document.getElementById('coord-row');
for (let c = 0; c < 8; c++) {
    const s = document.createElement('span'); s.textContent = FILES[c];
    coordRow.appendChild(s);
}

socket.onopen = () => log('Connected to server', 'info');
socket.onclose = () => log('Disconnected', 'bad');

socket.onmessage = (event) => {
    const msg = JSON.parse(event.data);

    if (msg.type === 'assigned') {
        myTeam = msg.team;
        const banner = document.getElementById('team-banner');
        banner.textContent = myTeam === 'spectator' ? 'Spectator' : `You are ${myTeam.toUpperCase()}`;
        banner.className = 'banner-' + myTeam;
        log(`Assigned as ${myTeam}`, 'info');
    }

    if (msg.type === 'gameState') {
        const prev = state;
        state = msg.state;
        selected = null;
        legalMovesMap = {};

        if (state.legalMoves) {
            for (const entry of state.legalMoves) {
                const key = entry.from[0] + ',' + entry.from[1];
                legalMovesMap[key] = entry.moves;
            }
        }

        render();
        updatePhaseBar();
        updateScores();
        logStateChange(prev, state);
    }

    if (msg.type === 'error') {
        log('⚠ ' + msg.message, 'bad');
    }
};

document.addEventListener('keydown', (e) => {
    if (e.key === 's' || e.key === 'S') {
        if (state && state.turn === myTeam && state.phase === 'move_king') skipKing();
    }
    if (e.key === 'r' || e.key === 'R') {
        selected = null; renderHighlights();
    }
});

function onCellClick(r, c) {
    if (!state || state.status === 'over') return;
    if (state.turn !== myTeam) return;

    const phase = state.phase;

    if (phase === 'guess') {
        log(`Guessing opponent king at ${sq(r,c)}`, 'info');
        socket.send(JSON.stringify({ type: 'guess', pos: [r, c] }));
        return;
    }

    if (phase === 'move_piece' || phase === 'move_king') {
        const key = `${r},${c}`;

        if (selected) {
            const selKey = `${selected[0]},${selected[1]}`;
            const legal = legalMovesMap[selKey] || [];
            if (legal.some(([lr,lc]) => lr===r && lc===c)) {
                if (phase === 'move_piece') {
                    socket.send(JSON.stringify({ type: 'movePiece', from: selected, to: [r,c] }));
                } else {
                    socket.send(JSON.stringify({ type: 'moveKing', to: [r,c] }));
                }
                selected = null;
                return;
            }
        }

        if (legalMovesMap[key]) {
            selected = [r, c];
            renderHighlights();
        } else {
            selected = null;
            renderHighlights();
        }
    }
}

function skipKing() {
    if (!state || state.turn !== myTeam || state.phase !== 'move_king') return;
    socket.send(JSON.stringify({ type: 'moveKing', to: null }));
    selected = null;
}

function requestNewGame() {
    socket.send(JSON.stringify({ type: 'newGame' }));
    document.getElementById('log-box').innerHTML = '';
    log('New game started', 'info');
}

function render() {
    if (!state) return;

    for (let r = 0; r < state.rows; r++) {
        for (let c = 0; c < state.cols; c++) {
            const cell = cells[r][c];
            const base = (r+c)%2===0 ? 'light' : 'dark';
            const sq = state.board[r][c];

            if (sq === null) {
                cell.className = 'cell fog';
                cell.textContent = '';
            } else if (sq === 'empty') {
                cell.className = 'cell ' + base;
                cell.textContent = '';
            } else {
                cell.className = 'cell ' + base + ' piece-' + sq.team;
                cell.textContent = PIECES[sq.team][sq.type] || '?';
            }
        }
    }

    renderHighlights();
}

function renderHighlights() {
    if (!state) return;

    for (let r = 0; r < state.rows; r++) {
        for (let c = 0; c < state.cols; c++) {
            const cell = cells[r][c];
            cell.classList.remove('selected','legal-empty','legal-occ','guess-target','king-sq');
        }
    }

    if (state.turn !== myTeam || state.status !== 'active') return;

    const phase = state.phase;

    if (phase === 'move_piece' || phase === 'move_king') {
        for (const key of Object.keys(legalMovesMap)) {
            const [r,c] = key.split(',').map(Number);
            if (!selected || (selected[0]!==r || selected[1]!==c)) {
                cells[r][c].classList.add('king-sq');
            }
        }

        if (selected) {
            const [sr, sc] = selected;
            cells[sr][sc].classList.add('selected');
            const legal = legalMovesMap[`${sr},${sc}`] || [];
            for (const [lr, lc] of legal) {
                const sq = state.board[lr][lc];
                cells[lr][lc].classList.add(sq && sq !== 'empty' ? 'legal-occ' : 'legal-empty');
            }
        }
    }

    if (phase === 'guess') {
        for (let r = 0; r < state.rows; r++) {
            for (let c = 0; c < state.cols; c++) {
                if (!cells[r][c].classList.contains('fog')) {
                    cells[r][c].classList.add('guess-target');
                }
            }
        }
    }

    document.getElementById('skip-king-btn').style.display =
        (phase === 'move_king') ? 'block' : 'none';
}

function updatePhaseBar() {
    if (!state) return;
    const bar = document.getElementById('phase-bar');

    if (state.status === 'over') {
        const { white, black } = state.scores;
        const winner = white > black ? 'White' : black > white ? 'Black' : 'Draw';
        bar.textContent = `Game over! ${winner === 'Draw' ? 'Draw' : winner + ' wins'} — W:${white} B:${black}`;
        bar.className = 'phase-over';
        return;
    }

    const isMyTurn = state.turn === myTeam;
    const turnLabel = isMyTurn ? 'Your turn' : `${cap(state.turn)}'s turn`;

    const phaseDesc = {
        move_piece: 'Move a piece',
        move_king:  'Move your king  (S to skip)',
        guess:      "Guess opponent's king square",
    };

    if (!isMyTurn) {
        bar.textContent = `Waiting for ${cap(state.turn)}…`;
        bar.className = 'phase-wait';
    } else {
        bar.textContent = `${turnLabel} — ${phaseDesc[state.phase] || state.phase}`;
        bar.className = {
            move_piece: 'phase-move',
            move_king:  'phase-king',
            guess:      'phase-guess',
        }[state.phase] || '';
    }
}

function updateScores() {
    if (!state) return;
    document.getElementById('score-white').textContent = state.scores.white;
    document.getElementById('score-black').textContent = state.scores.black;
    document.getElementById('turn-counter').textContent =
        `Turn ${Math.floor(state.turnCount/2)} / ${state.maxTurns/2}`;
}

function logStateChange(prev, curr) {
    if (!curr) return;
    if (prev && curr.lastGuess && (!prev.lastGuess || prev.lastGuess.turnCount !== curr.lastGuess.turnCount)) {
        const g = curr.lastGuess;
        const msg = `${cap(g.team)} guessed ${sq(...g.guess)} — ` +
                    (g.correct ? `✓ correct! King was at ${sq(...g.kingWas)}` : `✗ wrong. King was at ${sq(...g.kingWas)}`);
        log(msg, g.correct ? 'good' : 'bad');
    }
}

function log(msg, cls='') {
    const box = document.getElementById('log-box');
    const d = document.createElement('div');
    d.className = 'log-entry ' + cls;
    d.textContent = msg;
    box.appendChild(d);
    box.scrollTop = box.scrollHeight;
}

function sq(r, c) { return FILES[c] + (8-r); }
function cap(s) { return s.charAt(0).toUpperCase() + s.slice(1); }
</script>
</body>
</html>
