+ return this.getColor(x1,y1) !== this.getColor(x2,y2);
+ }
+
+ // Is (x,y) on the chessboard?
+ static OnBoard(x,y)
+ {
+ return (x>=0 && x<V.size.x && y>=0 && y<V.size.y);
+ }
+
+ // Used in interface: 'side' arg == player color
+ canIplay(side, [x,y])
+ {
+ return (this.turn == side && this.getColor(x,y) == side);
+ }
+
+ // On which squares is color under check ? (for interface)
+ getCheckSquares(color)
+ {
+ return this.isAttacked(this.kingPos[color], [this.getOppCol(color)])
+ ? [JSON.parse(JSON.stringify(this.kingPos[color]))] //need to duplicate!
+ : [];
+ }
+
+ /////////////
+ // FEN UTILS
+
+ // Setup the initial random (assymetric) position
+ static GenRandInitFen()
+ {
+ let pieces = { "w": new Array(8), "b": new Array(8) };
+ // Shuffle pieces on first and last rank
+ for (let c of ["w","b"])
+ {
+ let positions = _.range(8);
+
+ // Get random squares for bishops
+ let randIndex = 2 * _.random(3);
+ const bishop1Pos = positions[randIndex];
+ // The second bishop must be on a square of different color
+ let randIndex_tmp = 2 * _.random(3) + 1;
+ const bishop2Pos = positions[randIndex_tmp];
+ // Remove chosen squares
+ positions.splice(Math.max(randIndex,randIndex_tmp), 1);
+ positions.splice(Math.min(randIndex,randIndex_tmp), 1);
+
+ // Get random squares for knights
+ randIndex = _.random(5);
+ const knight1Pos = positions[randIndex];
+ positions.splice(randIndex, 1);
+ randIndex = _.random(4);
+ const knight2Pos = positions[randIndex];
+ positions.splice(randIndex, 1);
+
+ // Get random square for queen
+ randIndex = _.random(3);
+ const queenPos = positions[randIndex];
+ positions.splice(randIndex, 1);
+
+ // Rooks and king positions are now fixed,
+ // because of the ordering rook-king-rook
+ const rook1Pos = positions[0];
+ const kingPos = positions[1];
+ const rook2Pos = positions[2];
+
+ // Finally put the shuffled pieces in the board array
+ pieces[c][rook1Pos] = 'r';
+ pieces[c][knight1Pos] = 'n';
+ pieces[c][bishop1Pos] = 'b';
+ pieces[c][queenPos] = 'q';
+ pieces[c][kingPos] = 'k';
+ pieces[c][bishop2Pos] = 'b';
+ pieces[c][knight2Pos] = 'n';
+ pieces[c][rook2Pos] = 'r';
+ }
+ return pieces["b"].join("") +
+ "/pppppppp/8/8/8/8/PPPPPPPP/" +
+ pieces["w"].join("").toUpperCase() +
+ " w 1111 -"; //add turn + flags + enpassant
+ }
+
+ // "Parse" FEN: just return untransformed string data
+ static ParseFen(fen)
+ {
+ const fenParts = fen.split(" ");
+ let res =
+ {
+ position: fenParts[0],
+ turn: fenParts[1],
+ };
+ let nextIdx = 2;
+ if (V.HasFlags)
+ Object.assign(res, {flags: fenParts[nextIdx++]});
+ if (V.HasEnpassant)
+ Object.assign(res, {enpassant: fenParts[nextIdx]});
+ return res;