const WebSocket = require('ws');
const wss = new WebSocket.Server({ port: 8080 });

const DEFAULT_CONFIG = {
    rows: 8,
    cols: 8,
    pieces: [
        { type: 'king',   team: 'white', position: [4, 0] },
        { type: 'rook',   team: 'white', position: [7, 0] },
        { type: 'bishop', team: 'white', position: [0, 0] },
        { type: 'pawn',   team: 'white', position: [3, 1] },
        { type: 'pawn',   team: 'white', position: [4, 1] },

        { type: 'king',   team: 'black', position: [3, 7] },
        { type: 'rook',   team: 'black', position: [0, 7] },
        { type: 'bishop', team: 'black', position: [7, 7] },
        { type: 'pawn',   team: 'black', position: [3, 6] },
        { type: 'pawn',   team: 'black', position: [4, 6] },
    ]
};

const MAX_TURNS = 20;

const ALL_DIRS    = [[0,1],[0,-1],[1,0],[-1,0],[1,1],[1,-1],[-1,1],[-1,-1]];
const ROOK_DIRS   = [[0,1],[0,-1],[1,0],[-1,0]];
const BISHOP_DIRS = [[1,1],[1,-1],[-1,1],[-1,-1]];

function inBounds(r, c, rows, cols) {
    return r >= 0 && r < rows && c >= 0 && c < cols;
}

function getPossibleMoves(piece, grid, rows, cols) {
    const [r0, c0] = piece.position;
    const moves = [];
    for (const [dr, dc] of ALL_DIRS) {
        const r = r0 + dr, c = c0 + dc;
        if (inBounds(r, c, rows, cols) && grid[r][c] === null) {
            moves.push([r, c]);
        }
    }
    return moves;
}

function getVision(piece, grid, rows, cols) {
    const [r0, c0] = piece.position;
    const visible = new Set([`${r0},${c0}`]);
    for (const [dr, dc] of ALL_DIRS) {
        const r = r0 + dr, c = c0 + dc;
        if (inBounds(r, c, rows, cols)) visible.add(`${r},${c}`);
    }
    const longDirs = piece.type === 'rook' ? ROOK_DIRS
                   : piece.type === 'bishop' ? BISHOP_DIRS
                   : [];

    for (const [dr, dc] of longDirs) {
        let r = r0 + dr, c = c0 + dc;
        while (inBounds(r, c, rows, cols)) {
            visible.add(`${r},${c}`);
            const occ = grid[r][c];
            if (occ !== null && occ.type === 'pawn') break;
            r += dr; c += dc;
        }
    }
    return visible;
}

function computeFog(team, pieces, grid, rows, cols) {
    const visible = new Set();
    for (const p of pieces) {
        if (p.team === team) {
            for (const sq of getVision(p, grid, rows, cols)) visible.add(sq);
        }
    }
    return visible;
}

let game = null;

function newGame() {
    const { rows, cols } = DEFAULT_CONFIG;
    const grid = Array.from({ length: rows }, () => Array(cols).fill(null));
    const pieces = DEFAULT_CONFIG.pieces.map(pc => ({
        type: pc.type,
        team: pc.team,
        position: [...pc.position],
    }));
    for (const p of pieces) grid[p.position[0]][p.position[1]] = p;

    game = {
        rows, cols, grid, pieces,
        turn: 'white',
        turnCount: 0,
        maxTurns: MAX_TURNS,
        phase: 'move_piece',  
        scores: { white: 0, black: 0 },
        guessHistory: [],
        status: 'active', 
    };
}

function movePiece(team, from, to) {
    if (game.status === 'over')      return { error: 'Game is over.' };
    if (game.turn !== team)          return { error: `It is ${game.turn}'s turn.` };
    if (game.phase !== 'move_piece') return { error: 'Not in move_piece phase.' };

    const piece = game.grid[from[0]][from[1]];
    if (!piece)                  return { error: 'No piece there.' };
    if (piece.team !== team)     return { error: 'Not your piece.' };
    if (piece.type === 'king')   return { error: 'King is moved in the king phase.' };

    const legal = getPossibleMoves(piece, game.grid, game.rows, game.cols);
    if (!legal.some(([r,c]) => r === to[0] && c === to[1])) return { error: 'Illegal move.' };

    game.grid[from[0]][from[1]] = null;
    piece.position = [...to];
    game.grid[to[0]][to[1]] = piece;

    game.phase = 'move_king';
    return { success: true };
}

function moveKing(team, to) {
    if (game.status === 'over')      return { error: 'Game is over.' };
    if (game.turn !== team)          return { error: `It is ${game.turn}'s turn.` };
    if (game.phase !== 'move_king')  return { error: 'Not in move_king phase.' };

    if (to !== null) {
        const king = game.pieces.find(p => p.team === team && p.type === 'king');
        const legal = getPossibleMoves(king, game.grid, game.rows, game.cols);
        if (!legal.some(([r,c]) => r === to[0] && c === to[1])) return { error: 'Illegal king move.' };
        game.grid[king.position[0]][king.position[1]] = null;
        king.position = [...to];
        game.grid[to[0]][to[1]] = king;
    }

    game.phase = 'guess';
    return { success: true };
}

function makeGuess(team, pos) {
    if (game.status === 'over')   return { error: 'Game is over.' };
    if (game.turn !== team)       return { error: `It is ${game.turn}'s turn.` };
    if (game.phase !== 'guess')   return { error: 'Not in guess phase.' };

    const oppKing = game.pieces.find(p => p.team !== team && p.type === 'king');
    const correct = oppKing.position[0] === pos[0] && oppKing.position[1] === pos[1];
    if (correct) game.scores[team] += 1;

    game.guessHistory.push({ turnCount: game.turnCount, team, guess: pos, correct, kingWas: [...oppKing.position] });

    game.turn = team === 'white' ? 'black' : 'white';
    game.turnCount += 1;
    game.phase = 'move_piece';
    if (game.turnCount >= game.maxTurns) game.status = 'over';

    return { success: true };
}

function getClientState(team) {
    const fog = computeFog(team, game.pieces, game.grid, game.rows, game.cols);
    const board = [];
    for (let r = 0; r < game.rows; r++) {
        board[r] = [];
        for (let c = 0; c < game.cols; c++) {
            if (!fog.has(`${r},${c}`)) {
                board[r][c] = null; // fog
            } else {
                const p = game.grid[r][c];
                board[r][c] = p ? { type: p.type, team: p.team } : 'empty';
            }
        }
    }

    let legalMoves = [];
    if (game.turn === team && game.status === 'active') {
        if (game.phase === 'move_piece') {
            for (const p of game.pieces) {
                if (p.team === team && p.type !== 'king') {
                    legalMoves.push({ from: p.position, moves: getPossibleMoves(p, game.grid, game.rows, game.cols) });
                }
            }
        } else if (game.phase === 'move_king') {
            const king = game.pieces.find(p => p.team === team && p.type === 'king');
            legalMoves.push({ from: king.position, moves: getPossibleMoves(king, game.grid, game.rows, game.cols) });
        }
    }

    const lastGuess = game.guessHistory.length ? game.guessHistory[game.guessHistory.length - 1] : null;

    return {
        board,
        rows: game.rows,
        cols: game.cols,
        turn: game.turn,
        phase: game.phase,
        turnCount: game.turnCount,
        maxTurns: game.maxTurns,
        scores: game.scores,
        status: game.status,
        legalMoves,
        myPieces: game.pieces.filter(p => p.team === team).map(p => ({ type: p.type, team: p.team, position: p.position })),
        lastGuess,
    };
}

const clients = new Map(); // ws -> team
const slots = { white: null, black: null };

newGame();

wss.on('connection', (ws) => {
    let team = 'spectator';
    if (!slots.white) { team = 'white'; slots.white = ws; }
    else if (!slots.black) { team = 'black'; slots.black = ws; }
    clients.set(ws, team);

    console.log(`[+] ${team} connected`);
    ws.send(JSON.stringify({ type: 'assigned', team }));
    ws.send(JSON.stringify({ type: 'gameState', state: getClientState(team) }));

    ws.on('message', (raw) => {
        let msg;
        try { msg = JSON.parse(raw); } catch { return; }
        const team = clients.get(ws);

        let result;
        if      (msg.type === 'movePiece') result = movePiece(team, msg.from, msg.to);
        else if (msg.type === 'moveKing')  result = moveKing(team, msg.to);
        else if (msg.type === 'guess')     result = makeGuess(team, msg.pos);
        else if (msg.type === 'newGame')   { newGame(); broadcast(); return; }
        else return;

        if (result.error) {
            ws.send(JSON.stringify({ type: 'error', message: result.error }));
        } else {
            broadcast();
        }
    });

    ws.on('close', () => {
        const team = clients.get(ws);
        console.log(`[-] ${team} disconnected`);
        if (slots.white === ws)  slots.white = null;
        if (slots.black === ws) slots.black = null;
        clients.delete(ws);
    });
});

function broadcast() {
    for (const [ws, team] of clients) {
        if (ws.readyState === WebSocket.OPEN) {
            ws.send(JSON.stringify({ type: 'gameState', state: getClientState(team) }));
        }
    }
}

console.log('Info Chess server running on ws://localhost:8080');
console.log('Open TWO browser tabs: first = White, second = Black');
