Skip to content

Instantly share code, notes, and snippets.

@Mon-Ouie
Last active December 22, 2021 14:38
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save Mon-Ouie/e69a0cb2e2c8ed369b176483696e240e to your computer and use it in GitHub Desktop.
Save Mon-Ouie/e69a0cb2e2c8ed369b176483696e240e to your computer and use it in GitHub Desktop.
/*
* Copyright (c) 2021, Jeff Hlywa (jhlywa@gmail.com)
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
*----------------------------------------------------------------------------*/
var Chess = function(fen) {
var BLACK = 'b'
var WHITE = 'w'
var EMPTY = -1
var PAWN = 'p'
var KNIGHT = 'n'
var BISHOP = 'b'
var ROOK = 'r'
var QUEEN = 'q'
var KING = 'k'
var SYMBOLS = 'pnbrqkPNBRQK'
var DEFAULT_POSITION =
'rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR w KQkq - 0 1'
var POSSIBLE_RESULTS = ['1-0', '0-1', '1/2-1/2', '*']
var PAWN_OFFSETS = {
b: [16, 32, 17, 15],
w: [-16, -32, -17, -15]
}
var PIECE_OFFSETS = {
n: [-18, -33, -31, -14, 18, 33, 31, 14],
b: [-17, -15, 17, 15],
r: [-16, 1, 16, -1],
q: [-17, -16, -15, 1, 17, 16, 15, -1],
k: [-17, -16, -15, 1, 17, 16, 15, -1]
}
// prettier-ignore
var ATTACKS = [
20, 0, 0, 0, 0, 0, 0, 24, 0, 0, 0, 0, 0, 0,20, 0,
0,20, 0, 0, 0, 0, 0, 24, 0, 0, 0, 0, 0,20, 0, 0,
0, 0,20, 0, 0, 0, 0, 24, 0, 0, 0, 0,20, 0, 0, 0,
0, 0, 0,20, 0, 0, 0, 24, 0, 0, 0,20, 0, 0, 0, 0,
0, 0, 0, 0,20, 0, 0, 24, 0, 0,20, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0,20, 2, 24, 2,20, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 2,53, 56, 53, 2, 0, 0, 0, 0, 0, 0,
24,24,24,24,24,24,56, 0, 56,24,24,24,24,24,24, 0,
0, 0, 0, 0, 0, 2,53, 56, 53, 2, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0,20, 2, 24, 2,20, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0,20, 0, 0, 24, 0, 0,20, 0, 0, 0, 0, 0,
0, 0, 0,20, 0, 0, 0, 24, 0, 0, 0,20, 0, 0, 0, 0,
0, 0,20, 0, 0, 0, 0, 24, 0, 0, 0, 0,20, 0, 0, 0,
0,20, 0, 0, 0, 0, 0, 24, 0, 0, 0, 0, 0,20, 0, 0,
20, 0, 0, 0, 0, 0, 0, 24, 0, 0, 0, 0, 0, 0,20
];
// prettier-ignore
var RAYS = [
17, 0, 0, 0, 0, 0, 0, 16, 0, 0, 0, 0, 0, 0, 15, 0,
0, 17, 0, 0, 0, 0, 0, 16, 0, 0, 0, 0, 0, 15, 0, 0,
0, 0, 17, 0, 0, 0, 0, 16, 0, 0, 0, 0, 15, 0, 0, 0,
0, 0, 0, 17, 0, 0, 0, 16, 0, 0, 0, 15, 0, 0, 0, 0,
0, 0, 0, 0, 17, 0, 0, 16, 0, 0, 15, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 17, 0, 16, 0, 15, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 17, 16, 15, 0, 0, 0, 0, 0, 0, 0,
1, 1, 1, 1, 1, 1, 1, 0, -1, -1, -1,-1, -1, -1, -1, 0,
0, 0, 0, 0, 0, 0,-15,-16,-17, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0,-15, 0,-16, 0,-17, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0,-15, 0, 0,-16, 0, 0,-17, 0, 0, 0, 0, 0,
0, 0, 0,-15, 0, 0, 0,-16, 0, 0, 0,-17, 0, 0, 0, 0,
0, 0,-15, 0, 0, 0, 0,-16, 0, 0, 0, 0,-17, 0, 0, 0,
0,-15, 0, 0, 0, 0, 0,-16, 0, 0, 0, 0, 0,-17, 0, 0,
-15, 0, 0, 0, 0, 0, 0,-16, 0, 0, 0, 0, 0, 0,-17
];
var SHIFTS = { p: 0, n: 1, b: 2, r: 3, q: 4, k: 5 }
var FLAGS = {
NORMAL: 'n',
CAPTURE: 'c',
BIG_PAWN: 'b',
EP_CAPTURE: 'e',
PROMOTION: 'p',
KSIDE_CASTLE: 'k',
QSIDE_CASTLE: 'q'
}
var BITS = {
NORMAL: 1,
CAPTURE: 2,
BIG_PAWN: 4,
EP_CAPTURE: 8,
PROMOTION: 16,
KSIDE_CASTLE: 32,
QSIDE_CASTLE: 64
}
var RANK_1 = 7
var RANK_2 = 6
var RANK_3 = 5
var RANK_4 = 4
var RANK_5 = 3
var RANK_6 = 2
var RANK_7 = 1
var RANK_8 = 0
// prettier-ignore
var SQUARES = {
a8: 0, b8: 1, c8: 2, d8: 3, e8: 4, f8: 5, g8: 6, h8: 7,
a7: 16, b7: 17, c7: 18, d7: 19, e7: 20, f7: 21, g7: 22, h7: 23,
a6: 32, b6: 33, c6: 34, d6: 35, e6: 36, f6: 37, g6: 38, h6: 39,
a5: 48, b5: 49, c5: 50, d5: 51, e5: 52, f5: 53, g5: 54, h5: 55,
a4: 64, b4: 65, c4: 66, d4: 67, e4: 68, f4: 69, g4: 70, h4: 71,
a3: 80, b3: 81, c3: 82, d3: 83, e3: 84, f3: 85, g3: 86, h3: 87,
a2: 96, b2: 97, c2: 98, d2: 99, e2: 100, f2: 101, g2: 102, h2: 103,
a1: 112, b1: 113, c1: 114, d1: 115, e1: 116, f1: 117, g1: 118, h1: 119
};
var ROOKS = {
w: [
{ square: SQUARES.a1, flag: BITS.QSIDE_CASTLE },
{ square: SQUARES.h1, flag: BITS.KSIDE_CASTLE }
],
b: [
{ square: SQUARES.a8, flag: BITS.QSIDE_CASTLE },
{ square: SQUARES.h8, flag: BITS.KSIDE_CASTLE }
]
}
var board = new Array(128)
var kings = { w: EMPTY, b: EMPTY }
var turn = WHITE
var castling = { w: 0, b: 0 }
var ep_square = EMPTY
var half_moves = 0
var move_number = 1
var history = []
var header = {}
var comments = {}
/* if the user passes in a fen string, load it, else default to
* starting position
*/
if (typeof fen === 'undefined') {
load(DEFAULT_POSITION)
} else {
load(fen)
}
function clear(keep_headers) {
if (typeof keep_headers === 'undefined') {
keep_headers = false
}
board = new Array(128)
kings = { w: EMPTY, b: EMPTY }
turn = WHITE
castling = { w: 0, b: 0 }
ep_square = EMPTY
half_moves = 0
move_number = 1
history = []
if (!keep_headers) header = {}
comments = {}
update_setup(generate_fen())
}
function prune_comments() {
var reversed_history = [];
var current_comments = {};
var copy_comment = function(fen) {
if (fen in comments) {
current_comments[fen] = comments[fen];
}
};
while (history.length > 0) {
reversed_history.push(undo_move());
}
copy_comment(generate_fen());
while (reversed_history.length > 0) {
make_move(reversed_history.pop());
copy_comment(generate_fen());
}
comments = current_comments;
}
function reset() {
load(DEFAULT_POSITION)
}
function load(fen, keep_headers) {
if (typeof keep_headers === 'undefined') {
keep_headers = false
}
var tokens = fen.split(/\s+/)
var position = tokens[0]
var square = 0
if (!validate_fen(fen).valid) {
return false
}
clear(keep_headers)
for (var i = 0; i < position.length; i++) {
var piece = position.charAt(i)
if (piece === '/') {
square += 8
} else if (is_digit(piece)) {
square += parseInt(piece, 10)
} else {
var color = piece < 'a' ? WHITE : BLACK
put({ type: piece.toLowerCase(), color: color }, algebraic(square))
square++
}
}
turn = tokens[1]
if (tokens[2].indexOf('K') > -1) {
castling.w |= BITS.KSIDE_CASTLE
}
if (tokens[2].indexOf('Q') > -1) {
castling.w |= BITS.QSIDE_CASTLE
}
if (tokens[2].indexOf('k') > -1) {
castling.b |= BITS.KSIDE_CASTLE
}
if (tokens[2].indexOf('q') > -1) {
castling.b |= BITS.QSIDE_CASTLE
}
ep_square = tokens[3] === '-' ? EMPTY : SQUARES[tokens[3]]
half_moves = parseInt(tokens[4], 10)
move_number = parseInt(tokens[5], 10)
update_setup(generate_fen())
return true
}
/* TODO: this function is pretty much crap - it validates structure but
* completely ignores content (e.g. doesn't verify that each side has a king)
* ... we should rewrite this, and ditch the silly error_number field while
* we're at it
*/
function validate_fen(fen) {
var errors = {
0: 'No errors.',
1: 'FEN string must contain six space-delimited fields.',
2: '6th field (move number) must be a positive integer.',
3: '5th field (half move counter) must be a non-negative integer.',
4: '4th field (en-passant square) is invalid.',
5: '3rd field (castling availability) is invalid.',
6: '2nd field (side to move) is invalid.',
7: "1st field (piece positions) does not contain 8 '/'-delimited rows.",
8: '1st field (piece positions) is invalid [consecutive numbers].',
9: '1st field (piece positions) is invalid [invalid piece].',
10: '1st field (piece positions) is invalid [row too large].',
11: 'Illegal en-passant square'
}
/* 1st criterion: 6 space-seperated fields? */
var tokens = fen.split(/\s+/)
if (tokens.length !== 6) {
return { valid: false, error_number: 1, error: errors[1] }
}
/* 2nd criterion: move number field is a integer value > 0? */
if (isNaN(tokens[5]) || parseInt(tokens[5], 10) <= 0) {
return { valid: false, error_number: 2, error: errors[2] }
}
/* 3rd criterion: half move counter is an integer >= 0? */
if (isNaN(tokens[4]) || parseInt(tokens[4], 10) < 0) {
return { valid: false, error_number: 3, error: errors[3] }
}
/* 4th criterion: 4th field is a valid e.p.-string? */
if (!/^(-|[abcdefgh][36])$/.test(tokens[3])) {
return { valid: false, error_number: 4, error: errors[4] }
}
/* 5th criterion: 3th field is a valid castle-string? */
if (!/^(KQ?k?q?|Qk?q?|kq?|q|-)$/.test(tokens[2])) {
return { valid: false, error_number: 5, error: errors[5] }
}
/* 6th criterion: 2nd field is "w" (white) or "b" (black)? */
if (!/^(w|b)$/.test(tokens[1])) {
return { valid: false, error_number: 6, error: errors[6] }
}
/* 7th criterion: 1st field contains 8 rows? */
var rows = tokens[0].split('/')
if (rows.length !== 8) {
return { valid: false, error_number: 7, error: errors[7] }
}
/* 8th criterion: every row is valid? */
for (var i = 0; i < rows.length; i++) {
/* check for right sum of fields AND not two numbers in succession */
var sum_fields = 0
var previous_was_number = false
for (var k = 0; k < rows[i].length; k++) {
if (!isNaN(rows[i][k])) {
if (previous_was_number) {
return { valid: false, error_number: 8, error: errors[8] }
}
sum_fields += parseInt(rows[i][k], 10)
previous_was_number = true
} else {
if (!/^[prnbqkPRNBQK]$/.test(rows[i][k])) {
return { valid: false, error_number: 9, error: errors[9] }
}
sum_fields += 1
previous_was_number = false
}
}
if (sum_fields !== 8) {
return { valid: false, error_number: 10, error: errors[10] }
}
}
if (
(tokens[3][1] == '3' && tokens[1] == 'w') ||
(tokens[3][1] == '6' && tokens[1] == 'b')
) {
return { valid: false, error_number: 11, error: errors[11] }
}
/* everything's okay! */
return { valid: true, error_number: 0, error: errors[0] }
}
function generate_fen() {
var empty = 0
var fen = ''
for (var i = SQUARES.a8; i <= SQUARES.h1; i++) {
if (board[i] == null) {
empty++
} else {
if (empty > 0) {
fen += empty
empty = 0
}
var color = board[i].color
var piece = board[i].type
fen += color === WHITE ? piece.toUpperCase() : piece.toLowerCase()
}
if ((i + 1) & 0x88) {
if (empty > 0) {
fen += empty
}
if (i !== SQUARES.h1) {
fen += '/'
}
empty = 0
i += 8
}
}
var cflags = ''
if (castling[WHITE] & BITS.KSIDE_CASTLE) {
cflags += 'K'
}
if (castling[WHITE] & BITS.QSIDE_CASTLE) {
cflags += 'Q'
}
if (castling[BLACK] & BITS.KSIDE_CASTLE) {
cflags += 'k'
}
if (castling[BLACK] & BITS.QSIDE_CASTLE) {
cflags += 'q'
}
/* do we have an empty castling flag? */
cflags = cflags || '-'
var epflags = ep_square === EMPTY ? '-' : algebraic(ep_square)
return [fen, turn, cflags, epflags, half_moves, move_number].join(' ')
}
function set_header(args) {
for (var i = 0; i < args.length; i += 2) {
if (typeof args[i] === 'string' && typeof args[i + 1] === 'string') {
header[args[i]] = args[i + 1]
}
}
return header
}
/* called when the initial board setup is changed with put() or remove().
* modifies the SetUp and FEN properties of the header object. if the FEN is
* equal to the default position, the SetUp and FEN are deleted
* the setup is only updated if history.length is zero, ie moves haven't been
* made.
*/
function update_setup(fen) {
if (history.length > 0) return
if (fen !== DEFAULT_POSITION) {
header['SetUp'] = '1'
header['FEN'] = fen
} else {
delete header['SetUp']
delete header['FEN']
}
}
function get(square) {
var piece = board[SQUARES[square]]
return piece ? { type: piece.type, color: piece.color } : null
}
function put(piece, square) {
/* check for valid piece object */
if (!('type' in piece && 'color' in piece)) {
return false
}
/* check for piece */
if (SYMBOLS.indexOf(piece.type.toLowerCase()) === -1) {
return false
}
/* check for valid square */
if (!(square in SQUARES)) {
return false
}
var sq = SQUARES[square]
/* don't let the user place more than one king */
if (
piece.type == KING &&
!(kings[piece.color] == EMPTY || kings[piece.color] == sq)
) {
return false
}
board[sq] = { type: piece.type, color: piece.color }
if (piece.type === KING) {
kings[piece.color] = sq
}
update_setup(generate_fen())
return true
}
function remove(square) {
var piece = get(square)
board[SQUARES[square]] = null
if (piece && piece.type === KING) {
kings[piece.color] = EMPTY
}
update_setup(generate_fen())
return piece
}
function build_move(board, from, to, flags, promotion) {
var move = {
color: turn,
from: from,
to: to,
flags: flags,
piece: board[from].type
}
if (promotion) {
move.flags |= BITS.PROMOTION
move.promotion = promotion
}
if (board[to]) {
move.captured = board[to].type
} else if (flags & BITS.EP_CAPTURE) {
move.captured = PAWN
}
return move
}
function generate_moves(options) {
function add_move(board, moves, from, to, flags) {
/* if pawn promotion */
if (
board[from].type === PAWN &&
(rank(to) === RANK_8 || rank(to) === RANK_1)
) {
var pieces = [QUEEN, ROOK, BISHOP, KNIGHT]
for (var i = 0, len = pieces.length; i < len; i++) {
moves.push(build_move(board, from, to, flags, pieces[i]))
}
} else {
moves.push(build_move(board, from, to, flags))
}
}
var moves = []
var us = turn
var them = swap_color(us)
var second_rank = { b: RANK_7, w: RANK_2 }
var first_sq = SQUARES.a8
var last_sq = SQUARES.h1
var single_square = false
/* do we want legal moves? */
var legal =
typeof options !== 'undefined' && 'legal' in options
? options.legal
: true
/* are we generating moves for a single square? */
if (typeof options !== 'undefined' && 'square' in options) {
if (options.square in SQUARES) {
first_sq = last_sq = SQUARES[options.square]
single_square = true
} else {
/* invalid square */
return []
}
}
for (var i = first_sq; i <= last_sq; i++) {
/* did we run off the end of the board */
if (i & 0x88) {
i += 7
continue
}
var piece = board[i]
if (piece == null || piece.color !== us) {
continue
}
if (piece.type === PAWN) {
/* single square, non-capturing */
var square = i + PAWN_OFFSETS[us][0]
if (board[square] == null) {
add_move(board, moves, i, square, BITS.NORMAL)
/* double square */
var square = i + PAWN_OFFSETS[us][1]
if (second_rank[us] === rank(i) && board[square] == null) {
add_move(board, moves, i, square, BITS.BIG_PAWN)
}
}
/* pawn captures */
for (j = 2; j < 4; j++) {
var square = i + PAWN_OFFSETS[us][j]
if (square & 0x88) continue
if (board[square] != null && board[square].color === them) {
add_move(board, moves, i, square, BITS.CAPTURE)
} else if (square === ep_square) {
add_move(board, moves, i, ep_square, BITS.EP_CAPTURE)
}
}
} else {
for (var j = 0, len = PIECE_OFFSETS[piece.type].length; j < len; j++) {
var offset = PIECE_OFFSETS[piece.type][j]
var square = i
while (true) {
square += offset
if (square & 0x88) break
if (board[square] == null) {
add_move(board, moves, i, square, BITS.NORMAL)
} else {
if (board[square].color === us) break
add_move(board, moves, i, square, BITS.CAPTURE)
break
}
/* break, if knight or king */
if (piece.type === 'n' || piece.type === 'k') break
}
}
}
}
/* check for castling if: a) we're generating all moves, or b) we're doing
* single square move generation on the king's square
*/
if (!single_square || last_sq === kings[us]) {
/* king-side castling */
if (castling[us] & BITS.KSIDE_CASTLE) {
var castling_from = kings[us]
var castling_to = castling_from + 2
if (
board[castling_from + 1] == null &&
board[castling_to] == null &&
!attacked(them, kings[us]) &&
!attacked(them, castling_from + 1) &&
!attacked(them, castling_to)
) {
add_move(board, moves, kings[us], castling_to, BITS.KSIDE_CASTLE)
}
}
/* queen-side castling */
if (castling[us] & BITS.QSIDE_CASTLE) {
var castling_from = kings[us]
var castling_to = castling_from - 2
if (
board[castling_from - 1] == null &&
board[castling_from - 2] == null &&
board[castling_from - 3] == null &&
!attacked(them, kings[us]) &&
!attacked(them, castling_from - 1) &&
!attacked(them, castling_to)
) {
add_move(board, moves, kings[us], castling_to, BITS.QSIDE_CASTLE)
}
}
}
/* return all pseudo-legal moves (this includes moves that allow the king
* to be captured)
*/
if (!legal) {
return moves
}
/* filter out illegal moves */
var legal_moves = []
for (var i = 0, len = moves.length; i < len; i++) {
make_move(moves[i])
if (!king_attacked(us)) {
legal_moves.push(moves[i])
}
undo_move()
}
return legal_moves
}
/* convert a move from 0x88 coordinates to Standard Algebraic Notation
* (SAN)
*
* @param {boolean} sloppy Use the sloppy SAN generator to work around over
* disambiguation bugs in Fritz and Chessbase. See below:
*
* r1bqkbnr/ppp2ppp/2n5/1B1pP3/4P3/8/PPPP2PP/RNBQK1NR b KQkq - 2 4
* 4. ... Nge7 is overly disambiguated because the knight on c6 is pinned
* 4. ... Ne7 is technically the valid SAN
*/
function move_to_san(move, sloppy) {
var output = ''
if (move.flags & BITS.KSIDE_CASTLE) {
output = 'O-O'
} else if (move.flags & BITS.QSIDE_CASTLE) {
output = 'O-O-O'
} else {
var disambiguator = get_disambiguator(move, sloppy)
if (move.piece !== PAWN) {
output += move.piece.toUpperCase() + disambiguator
}
if (move.flags & (BITS.CAPTURE | BITS.EP_CAPTURE)) {
if (move.piece === PAWN) {
output += algebraic(move.from)[0]
}
output += 'x'
}
output += algebraic(move.to)
if (move.flags & BITS.PROMOTION) {
output += '=' + move.promotion.toUpperCase()
}
}
make_move(move)
if (in_check()) {
if (in_checkmate()) {
output += '#'
} else {
output += '+'
}
}
undo_move()
return output
}
// parses all of the decorators out of a SAN string
function stripped_san(move) {
return move.replace(/=/, '').replace(/[+#]?[?!]*$/, '')
}
function attacked(color, square) {
for (var i = SQUARES.a8; i <= SQUARES.h1; i++) {
/* did we run off the end of the board */
if (i & 0x88) {
i += 7
continue
}
/* if empty square or wrong color */
if (board[i] == null || board[i].color !== color) continue
var piece = board[i]
var difference = i - square
var index = difference + 119
if (ATTACKS[index] & (1 << SHIFTS[piece.type])) {
if (piece.type === PAWN) {
if (difference > 0) {
if (piece.color === WHITE) return true
} else {
if (piece.color === BLACK) return true
}
continue
}
/* if the piece is a knight or a king */
if (piece.type === 'n' || piece.type === 'k') return true
var offset = RAYS[index]
var j = i + offset
var blocked = false
while (j !== square) {
if (board[j] != null) {
blocked = true
break
}
j += offset
}
if (!blocked) return true
}
}
return false
}
function king_attacked(color) {
return attacked(swap_color(color), kings[color])
}
function in_check() {
return king_attacked(turn)
}
function in_checkmate() {
return in_check() && generate_moves().length === 0
}
function in_stalemate() {
return !in_check() && generate_moves().length === 0
}
function insufficient_material() {
var pieces = {}
var bishops = []
var num_pieces = 0
var sq_color = 0
for (var i = SQUARES.a8; i <= SQUARES.h1; i++) {
sq_color = (sq_color + 1) % 2
if (i & 0x88) {
i += 7
continue
}
var piece = board[i]
if (piece) {
pieces[piece.type] = piece.type in pieces ? pieces[piece.type] + 1 : 1
if (piece.type === BISHOP) {
bishops.push(sq_color)
}
num_pieces++
}
}
/* k vs. k */
if (num_pieces === 2) {
return true
} else if (
/* k vs. kn .... or .... k vs. kb */
num_pieces === 3 &&
(pieces[BISHOP] === 1 || pieces[KNIGHT] === 1)
) {
return true
} else if (num_pieces === pieces[BISHOP] + 2) {
/* kb vs. kb where any number of bishops are all on the same color */
var sum = 0
var len = bishops.length
for (var i = 0; i < len; i++) {
sum += bishops[i]
}
if (sum === 0 || sum === len) {
return true
}
}
return false
}
function in_threefold_repetition() {
/* TODO: while this function is fine for casual use, a better
* implementation would use a Zobrist key (instead of FEN). the
* Zobrist key would be maintained in the make_move/undo_move functions,
* avoiding the costly that we do below.
*/
var moves = []
var positions = {}
var repetition = false
while (true) {
var move = undo_move()
if (!move) break
moves.push(move)
}
while (true) {
/* remove the last two fields in the FEN string, they're not needed
* when checking for draw by rep */
var fen = generate_fen()
.split(' ')
.slice(0, 4)
.join(' ')
/* has the position occurred three or move times */
positions[fen] = fen in positions ? positions[fen] + 1 : 1
if (positions[fen] >= 3) {
repetition = true
}
if (!moves.length) {
break
}
make_move(moves.pop())
}
return repetition
}
function push(move) {
history.push({
move: move,
kings: { b: kings.b, w: kings.w },
turn: turn,
castling: { b: castling.b, w: castling.w },
ep_square: ep_square,
half_moves: half_moves,
move_number: move_number
})
}
function make_move(move) {
var us = turn
var them = swap_color(us)
push(move)
board[move.to] = board[move.from]
board[move.from] = null
/* if ep capture, remove the captured pawn */
if (move.flags & BITS.EP_CAPTURE) {
if (turn === BLACK) {
board[move.to - 16] = null
} else {
board[move.to + 16] = null
}
}
/* if pawn promotion, replace with new piece */
if (move.flags & BITS.PROMOTION) {
board[move.to] = { type: move.promotion, color: us }
}
/* if we moved the king */
if (board[move.to].type === KING) {
kings[board[move.to].color] = move.to
/* if we castled, move the rook next to the king */
if (move.flags & BITS.KSIDE_CASTLE) {
var castling_to = move.to - 1
var castling_from = move.to + 1
board[castling_to] = board[castling_from]
board[castling_from] = null
} else if (move.flags & BITS.QSIDE_CASTLE) {
var castling_to = move.to + 1
var castling_from = move.to - 2
board[castling_to] = board[castling_from]
board[castling_from] = null
}
/* turn off castling */
castling[us] = ''
}
/* turn off castling if we move a rook */
if (castling[us]) {
for (var i = 0, len = ROOKS[us].length; i < len; i++) {
if (
move.from === ROOKS[us][i].square &&
castling[us] & ROOKS[us][i].flag
) {
castling[us] ^= ROOKS[us][i].flag
break
}
}
}
/* turn off castling if we capture a rook */
if (castling[them]) {
for (var i = 0, len = ROOKS[them].length; i < len; i++) {
if (
move.to === ROOKS[them][i].square &&
castling[them] & ROOKS[them][i].flag
) {
castling[them] ^= ROOKS[them][i].flag
break
}
}
}
/* if big pawn move, update the en passant square */
if (move.flags & BITS.BIG_PAWN) {
if (turn === 'b') {
ep_square = move.to - 16
} else {
ep_square = move.to + 16
}
} else {
ep_square = EMPTY
}
/* reset the 50 move counter if a pawn is moved or a piece is captured */
if (move.piece === PAWN) {
half_moves = 0
} else if (move.flags & (BITS.CAPTURE | BITS.EP_CAPTURE)) {
half_moves = 0
} else {
half_moves++
}
if (turn === BLACK) {
move_number++
}
turn = swap_color(turn)
}
function undo_move() {
var old = history.pop()
if (old == null) {
return null
}
var move = old.move
kings = old.kings
turn = old.turn
castling = old.castling
ep_square = old.ep_square
half_moves = old.half_moves
move_number = old.move_number
var us = turn
var them = swap_color(turn)
board[move.from] = board[move.to]
board[move.from].type = move.piece // to undo any promotions
board[move.to] = null
if (move.flags & BITS.CAPTURE) {
board[move.to] = { type: move.captured, color: them }
} else if (move.flags & BITS.EP_CAPTURE) {
var index
if (us === BLACK) {
index = move.to - 16
} else {
index = move.to + 16
}
board[index] = { type: PAWN, color: them }
}
if (move.flags & (BITS.KSIDE_CASTLE | BITS.QSIDE_CASTLE)) {
var castling_to, castling_from
if (move.flags & BITS.KSIDE_CASTLE) {
castling_to = move.to + 1
castling_from = move.to - 1
} else if (move.flags & BITS.QSIDE_CASTLE) {
castling_to = move.to - 2
castling_from = move.to + 1
}
board[castling_to] = board[castling_from]
board[castling_from] = null
}
return move
}
/* this function is used to uniquely identify ambiguous moves */
function get_disambiguator(move, sloppy) {
var moves = generate_moves({ legal: !sloppy })
var from = move.from
var to = move.to
var piece = move.piece
var ambiguities = 0
var same_rank = 0
var same_file = 0
for (var i = 0, len = moves.length; i < len; i++) {
var ambig_from = moves[i].from
var ambig_to = moves[i].to
var ambig_piece = moves[i].piece
/* if a move of the same piece type ends on the same to square, we'll
* need to add a disambiguator to the algebraic notation
*/
if (piece === ambig_piece && from !== ambig_from && to === ambig_to) {
ambiguities++
if (rank(from) === rank(ambig_from)) {
same_rank++
}
if (file(from) === file(ambig_from)) {
same_file++
}
}
}
if (ambiguities > 0) {
/* if there exists a similar moving piece on the same rank and file as
* the move in question, use the square as the disambiguator
*/
if (same_rank > 0 && same_file > 0) {
return algebraic(from)
} else if (same_file > 0) {
/* if the moving piece rests on the same file, use the rank symbol as the
* disambiguator
*/
return algebraic(from).charAt(1)
} else {
/* else use the file symbol */
return algebraic(from).charAt(0)
}
}
return ''
}
function ascii() {
var s = ' +------------------------+\n'
for (var i = SQUARES.a8; i <= SQUARES.h1; i++) {
/* display the rank */
if (file(i) === 0) {
s += ' ' + '87654321'[rank(i)] + ' |'
}
/* empty piece */
if (board[i] == null) {
s += ' . '
} else {
var piece = board[i].type
var color = board[i].color
var symbol = color === WHITE ? piece.toUpperCase() : piece.toLowerCase()
s += ' ' + symbol + ' '
}
if ((i + 1) & 0x88) {
s += '|\n'
i += 8
}
}
s += ' +------------------------+\n'
s += ' a b c d e f g h\n'
return s
}
// convert a move from Standard Algebraic Notation (SAN) to 0x88 coordinates
function move_from_san(move, sloppy) {
// strip off any move decorations: e.g Nf3+?!
var clean_move = stripped_san(move)
// if we're using the sloppy parser run a regex to grab piece, to, and from
// this should parse invalid SAN like: Pe2-e4, Rc1c4, Qf3xf7
if (sloppy) {
var matches = clean_move.match(
/([pnbrqkPNBRQK])?([a-h][1-8])x?-?([a-h][1-8])([qrbnQRBN])?/
)
if (matches) {
var piece = matches[1]
var from = matches[2]
var to = matches[3]
var promotion = matches[4]
}
}
var moves = generate_moves()
for (var i = 0, len = moves.length; i < len; i++) {
// try the strict parser first, then the sloppy parser if requested
// by the user
if (
clean_move === stripped_san(move_to_san(moves[i])) ||
(sloppy && clean_move === stripped_san(move_to_san(moves[i], true)))
) {
return moves[i]
} else {
if (
matches &&
(!piece || piece.toLowerCase() == moves[i].piece) &&
SQUARES[from] == moves[i].from &&
SQUARES[to] == moves[i].to &&
(!promotion || promotion.toLowerCase() == moves[i].promotion)
) {
return moves[i]
}
}
}
return null
}
/*****************************************************************************
* UTILITY FUNCTIONS
****************************************************************************/
function rank(i) {
return i >> 4
}
function file(i) {
return i & 15
}
function algebraic(i) {
var f = file(i),
r = rank(i)
return 'abcdefgh'.substring(f, f + 1) + '87654321'.substring(r, r + 1)
}
function swap_color(c) {
return c === WHITE ? BLACK : WHITE
}
function is_digit(c) {
return '0123456789'.indexOf(c) !== -1
}
/* pretty = external move object */
function make_pretty(ugly_move) {
var move = clone(ugly_move)
move.san = move_to_san(move, false)
move.to = algebraic(move.to)
move.from = algebraic(move.from)
var flags = ''
for (var flag in BITS) {
if (BITS[flag] & move.flags) {
flags += FLAGS[flag]
}
}
move.flags = flags
return move
}
function clone(obj) {
var dupe = obj instanceof Array ? [] : {}
for (var property in obj) {
if (typeof property === 'object') {
dupe[property] = clone(obj[property])
} else {
dupe[property] = obj[property]
}
}
return dupe
}
function trim(str) {
return str.replace(/^\s+|\s+$/g, '')
}
/*****************************************************************************
* DEBUGGING UTILITIES
****************************************************************************/
function perft(depth) {
var moves = generate_moves({ legal: false })
var nodes = 0
var color = turn
for (var i = 0, len = moves.length; i < len; i++) {
make_move(moves[i])
if (!king_attacked(color)) {
if (depth - 1 > 0) {
var child_nodes = perft(depth - 1)
nodes += child_nodes
} else {
nodes++
}
}
undo_move()
}
return nodes
}
return {
/***************************************************************************
* PUBLIC CONSTANTS (is there a better way to do this?)
**************************************************************************/
WHITE: WHITE,
BLACK: BLACK,
PAWN: PAWN,
KNIGHT: KNIGHT,
BISHOP: BISHOP,
ROOK: ROOK,
QUEEN: QUEEN,
KING: KING,
SQUARES: (function() {
/* from the ECMA-262 spec (section 12.6.4):
* "The mechanics of enumerating the properties ... is
* implementation dependent"
* so: for (var sq in SQUARES) { keys.push(sq); } might not be
* ordered correctly
*/
var keys = []
for (var i = SQUARES.a8; i <= SQUARES.h1; i++) {
if (i & 0x88) {
i += 7
continue
}
keys.push(algebraic(i))
}
return keys
})(),
FLAGS: FLAGS,
/***************************************************************************
* PUBLIC API
**************************************************************************/
load: function(fen) {
return load(fen)
},
reset: function() {
return reset()
},
moves: function(options) {
/* The internal representation of a chess move is in 0x88 format, and
* not meant to be human-readable. The code below converts the 0x88
* square coordinates to algebraic coordinates. It also prunes an
* unnecessary move keys resulting from a verbose call.
*/
var ugly_moves = generate_moves(options)
var moves = []
for (var i = 0, len = ugly_moves.length; i < len; i++) {
/* does the user want a full move object (most likely not), or just
* SAN
*/
if (
typeof options !== 'undefined' &&
'verbose' in options &&
options.verbose
) {
moves.push(make_pretty(ugly_moves[i]))
} else {
moves.push(move_to_san(ugly_moves[i], false))
}
}
return moves
},
in_check: function() {
return in_check()
},
in_checkmate: function() {
return in_checkmate()
},
in_stalemate: function() {
return in_stalemate()
},
in_draw: function() {
return (
half_moves >= 100 ||
in_stalemate() ||
insufficient_material() ||
in_threefold_repetition()
)
},
insufficient_material: function() {
return insufficient_material()
},
in_threefold_repetition: function() {
return in_threefold_repetition()
},
game_over: function() {
return (
half_moves >= 100 ||
in_checkmate() ||
in_stalemate() ||
insufficient_material() ||
in_threefold_repetition()
)
},
validate_fen: function(fen) {
return validate_fen(fen)
},
fen: function() {
return generate_fen()
},
board: function() {
var output = [],
row = []
for (var i = SQUARES.a8; i <= SQUARES.h1; i++) {
if (board[i] == null) {
row.push(null)
} else {
row.push({ type: board[i].type, color: board[i].color })
}
if ((i + 1) & 0x88) {
output.push(row)
row = []
i += 8
}
}
return output
},
pgn: function(options) {
/* using the specification from http://www.chessclub.com/help/PGN-spec
* example for html usage: .pgn({ max_width: 72, newline_char: "<br />" })
*/
var newline =
typeof options === 'object' && typeof options.newline_char === 'string'
? options.newline_char
: '\n'
var max_width =
typeof options === 'object' && typeof options.max_width === 'number'
? options.max_width
: 0
var result = []
var header_exists = false
/* add the PGN header headerrmation */
for (var i in header) {
/* TODO: order of enumerated properties in header object is not
* guaranteed, see ECMA-262 spec (section 12.6.4)
*/
result.push('[' + i + ' "' + header[i] + '"]' + newline)
header_exists = true
}
if (header_exists && history.length) {
result.push(newline)
}
var append_comment = function(move_string) {
var comment = comments[generate_fen()]
if (typeof comment !== 'undefined') {
var delimiter = move_string.length > 0 ? ' ' : '';
move_string = `${move_string}${delimiter}{${comment}}`
}
return move_string
}
/* pop all of history onto reversed_history */
var reversed_history = []
while (history.length > 0) {
reversed_history.push(undo_move())
}
var moves = []
var move_string = ''
/* special case of a commented starting position with no moves */
if (reversed_history.length === 0) {
moves.push(append_comment(''))
}
/* build the list of moves. a move_string looks like: "3. e3 e6" */
while (reversed_history.length > 0) {
move_string = append_comment(move_string)
var move = reversed_history.pop()
/* if the position started with black to move, start PGN with 1. ... */
if (!history.length && move.color === 'b') {
move_string = move_number + '. ...'
} else if (move.color === 'w') {
/* store the previous generated move_string if we have one */
if (move_string.length) {
moves.push(move_string)
}
move_string = move_number + '.'
}
move_string = move_string + ' ' + move_to_san(move, false)
make_move(move)
}
/* are there any other leftover moves? */
if (move_string.length) {
moves.push(append_comment(move_string))
}
/* is there a result? */
if (typeof header.Result !== 'undefined') {
moves.push(header.Result)
}
/* history should be back to what it was before we started generating PGN,
* so join together moves
*/
if (max_width === 0) {
return result.join('') + moves.join(' ')
}
var strip = function() {
if (result.length > 0 && result[result.length - 1] === ' ') {
result.pop();
return true;
}
return false;
};
/* NB: this does not preserve comment whitespace. */
var wrap_comment = function(width, move) {
for (var token of move.split(' ')) {
if (!token) {
continue;
}
if (width + token.length > max_width) {
while (strip()) {
width--;
}
result.push(newline);
width = 0;
}
result.push(token);
width += token.length;
result.push(' ');
width++;
}
if (strip()) {
width--;
}
return width;
};
/* wrap the PGN output at max_width */
var current_width = 0
for (var i = 0; i < moves.length; i++) {
if (current_width + moves[i].length > max_width) {
if (moves[i].includes('{')) {
current_width = wrap_comment(current_width, moves[i]);
continue;
}
}
/* if the current move will push past max_width */
if (current_width + moves[i].length > max_width && i !== 0) {
/* don't end the line with whitespace */
if (result[result.length - 1] === ' ') {
result.pop()
}
result.push(newline)
current_width = 0
} else if (i !== 0) {
result.push(' ')
current_width++
}
result.push(moves[i])
current_width += moves[i].length
}
return result.join('')
},
load_pgn: function(pgn, options) {
// allow the user to specify the sloppy move parser to work around over
// disambiguation bugs in Fritz and Chessbase
var sloppy =
typeof options !== 'undefined' && 'sloppy' in options
? options.sloppy
: false
function mask(str) {
return str.replace(/\\/g, '\\')
}
function has_keys(object) {
for (var key in object) {
return true
}
return false
}
function parse_pgn_header(header, options) {
var newline_char =
typeof options === 'object' &&
typeof options.newline_char === 'string'
? options.newline_char
: '\r?\n'
var header_obj = {}
var headers = header.split(new RegExp(mask(newline_char)))
var key = ''
var value = ''
for (var i = 0; i < headers.length; i++) {
key = headers[i].replace(/^\[([A-Z][A-Za-z]*)\s.*\]$/, '$1')
value = headers[i].replace(/^\[[A-Za-z]+\s"(.*)"\ *\]$/, '$1')
if (trim(key).length > 0) {
header_obj[key] = value
}
}
return header_obj
}
var newline_char =
typeof options === 'object' && typeof options.newline_char === 'string'
? options.newline_char
: '\r?\n'
// RegExp to split header. Takes advantage of the fact that header and movetext
// will always have a blank line between them (ie, two newline_char's).
// With default newline_char, will equal: /^(\[((?:\r?\n)|.)*\])(?:\r?\n){2}/
var header_regex = new RegExp(
'^(\\[((?:' +
mask(newline_char) +
')|.)*\\])' +
'(?:' +
mask(newline_char) +
'){2}'
)
// If no header given, begin with moves.
var header_string = header_regex.test(pgn)
? header_regex.exec(pgn)[1]
: ''
// Put the board in the starting position
reset()
/* parse PGN header */
var headers = parse_pgn_header(header_string, options)
for (var key in headers) {
set_header([key, headers[key]])
}
/* load the starting position indicated by [Setup '1'] and
* [FEN position] */
if (headers['SetUp'] === '1') {
if (!('FEN' in headers && load(headers['FEN'], true))) {
// second argument to load: don't clear the headers
return false
}
}
/* NB: the regexes below that delete move numbers, recursive
* annotations, and numeric annotation glyphs may also match
* text in comments. To prevent this, we transform comments
* by hex-encoding them in place and decoding them again after
* the other tokens have been deleted.
*
* While the spec states that PGN files should be ASCII encoded,
* we use {en,de}codeURIComponent here to support arbitrary UTF8
* as a convenience for modern users */
var to_hex = function(string) {
return Array
.from(string)
.map(function(c) {
/* encodeURI doesn't transform most ASCII characters,
* so we handle these ourselves */
return c.charCodeAt(0) < 128
? c.charCodeAt(0).toString(16)
: encodeURIComponent(c).replace(/\%/g, '').toLowerCase()
})
.join('')
}
var from_hex = function(string) {
return string.length == 0
? ''
: decodeURIComponent('%' + string.match(/.{1,2}/g).join('%'))
}
var encode_comment = function(string) {
string = string.replace(new RegExp(mask(newline_char), 'g'), ' ')
return `{${to_hex(string.slice(1, string.length - 1))}}`
}
var decode_comment = function(string) {
if (string.startsWith('{') && string.endsWith('}')) {
return from_hex(string.slice(1, string.length - 1))
}
}
/* delete header to get the moves */
var ms = pgn
.replace(header_string, '')
.replace(
/* encode comments so they don't get deleted below */
new RegExp(`(\{[^}]*\})+?|;([^${mask(newline_char)}]*)`, 'g'),
function(match, bracket, semicolon) {
return bracket !== undefined
? encode_comment(bracket)
: ' ' + encode_comment(`{${semicolon.slice(1)}}`)
}
)
.replace(new RegExp(mask(newline_char), 'g'), ' ')
/* delete recursive annotation variations */
var rav_regex = /(\([^\(\)]+\))+?/g
while (rav_regex.test(ms)) {
ms = ms.replace(rav_regex, '')
}
/* delete move numbers */
ms = ms.replace(/\d+\.(\.\.)?/g, '')
/* delete ... indicating black to move */
ms = ms.replace(/\.\.\./g, '')
/* delete numeric annotation glyphs */
ms = ms.replace(/\$\d+/g, '')
/* trim and get array of moves */
var moves = trim(ms).split(new RegExp(/\s+/))
/* delete empty entries */
moves = moves
.join(',')
.replace(/,,+/g, ',')
.split(',')
var move = ''
for (var half_move = 0; half_move < moves.length - 1; half_move++) {
var comment = decode_comment(moves[half_move])
if (comment !== undefined) {
comments[generate_fen()] = comment
continue
}
move = move_from_san(moves[half_move], sloppy)
/* move not possible! (don't clear the board to examine to show the
* latest valid position)
*/
if (move == null) {
return false
} else {
make_move(move)
}
}
comment = decode_comment(moves[moves.length - 1])
if (comment !== undefined) {
comments[generate_fen()] = comment
moves.pop()
}
/* examine last move */
move = moves[moves.length - 1]
if (POSSIBLE_RESULTS.indexOf(move) > -1) {
if (has_keys(header) && typeof header.Result === 'undefined') {
set_header(['Result', move])
}
} else {
move = move_from_san(move, sloppy)
if (move == null) {
return false
} else {
make_move(move)
}
}
return true
},
header: function() {
return set_header(arguments)
},
ascii: function() {
return ascii()
},
turn: function() {
return turn
},
move: function(move, options) {
/* The move function can be called with in the following parameters:
*
* .move('Nxb7') <- where 'move' is a case-sensitive SAN string
*
* .move({ from: 'h7', <- where the 'move' is a move object (additional
* to :'h8', fields are ignored)
* promotion: 'q',
* })
*/
// allow the user to specify the sloppy move parser to work around over
// disambiguation bugs in Fritz and Chessbase
var sloppy =
typeof options !== 'undefined' && 'sloppy' in options
? options.sloppy
: false
var move_obj = null
if (typeof move === 'string') {
move_obj = move_from_san(move, sloppy)
} else if (typeof move === 'object') {
var moves = generate_moves()
/* convert the pretty move object to an ugly move object */
for (var i = 0, len = moves.length; i < len; i++) {
if (
move.from === algebraic(moves[i].from) &&
move.to === algebraic(moves[i].to) &&
(!('promotion' in moves[i]) ||
move.promotion === moves[i].promotion)
) {
move_obj = moves[i]
break
}
}
}
/* failed to find move */
if (!move_obj) {
return null
}
/* need to make a copy of move because we can't generate SAN after the
* move is made
*/
var pretty_move = make_pretty(move_obj)
make_move(move_obj)
return pretty_move
},
undo: function() {
var move = undo_move()
return move ? make_pretty(move) : null
},
clear: function() {
return clear()
},
put: function(piece, square) {
return put(piece, square)
},
get: function(square) {
return get(square)
},
remove: function(square) {
return remove(square)
},
perft: function(depth) {
return perft(depth)
},
square_color: function(square) {
if (square in SQUARES) {
var sq_0x88 = SQUARES[square]
return (rank(sq_0x88) + file(sq_0x88)) % 2 === 0 ? 'light' : 'dark'
}
return null
},
history: function(options) {
var reversed_history = []
var move_history = []
var verbose =
typeof options !== 'undefined' &&
'verbose' in options &&
options.verbose
while (history.length > 0) {
reversed_history.push(undo_move())
}
while (reversed_history.length > 0) {
var move = reversed_history.pop()
if (verbose) {
move_history.push(make_pretty(move))
} else {
move_history.push(move_to_san(move))
}
make_move(move)
}
return move_history
},
get_comment: function() {
return comments[generate_fen()];
},
set_comment: function(comment) {
comments[generate_fen()] = comment.replace('{', '[').replace('}', ']');
},
delete_comment: function() {
var comment = comments[generate_fen()];
delete comments[generate_fen()];
return comment;
},
get_comments: function() {
prune_comments();
return Object.keys(comments).map(function(fen) {
return {fen: fen, comment: comments[fen]};
});
},
delete_comments: function() {
prune_comments();
return Object.keys(comments)
.map(function(fen) {
var comment = comments[fen];
delete comments[fen];
return {fen: fen, comment: comment};
});
}
}
}
/* export Chess object if using node or any other CommonJS compatible
* environment */
if (typeof exports !== 'undefined') exports.Chess = Chess
/* export Chess object for any RequireJS compatible environment */
if (typeof define !== 'undefined')
define(function() {
return Chess
})
.cg-wrap {
box-sizing: content-box;
position: relative;
display: block;
}
cg-container {
position: absolute;
width: 100%;
height: 100%;
display: block;
top: 0;
}
cg-board {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
line-height: 0;
background-size: cover;
cursor: pointer;
}
cg-board square {
position: absolute;
top: 0;
left: 0;
width: 12.5%;
height: 12.5%;
pointer-events: none;
}
cg-board square.move-dest {
pointer-events: auto;
}
cg-board square.last-move {
will-change: transform;
}
.cg-wrap piece {
position: absolute;
top: 0;
left: 0;
width: 12.5%;
height: 12.5%;
background-size: cover;
z-index: 2;
will-change: transform;
pointer-events: none;
}
cg-board piece.dragging {
cursor: move;
/* !important to override z-index from 3D piece inline style */
z-index: 11 !important;
}
piece.anim {
z-index: 8;
}
piece.fading {
z-index: 1;
opacity: 0.5;
}
.cg-wrap piece.ghost {
opacity: 0.3;
}
.cg-wrap piece svg {
overflow: hidden;
position: relative;
top: 0px;
left: 0px;
width: 100%;
height: 100%;
pointer-events: none;
z-index: 2;
opacity: 0.6;
}
.cg-wrap piece svg image {
opacity: 0.5;
}
.cg-wrap .cg-shapes,
.cg-wrap .cg-custom-svgs {
overflow: hidden;
position: absolute;
top: 0px;
left: 0px;
width: 100%;
height: 100%;
pointer-events: none;
}
.cg-wrap .cg-shapes {
opacity: 0.6;
z-index: 2;
}
.cg-wrap .cg-custom-svgs {
/* over piece.anim = 8, but under piece.dragging = 11 */
z-index: 9;
overflow: visible;
}
.cg-wrap .cg-custom-svgs svg {
overflow: visible;
}
.cg-wrap coords {
position: absolute;
display: flex;
pointer-events: none;
opacity: 0.8;
font-family: sans-serif;
font-size: 9px;
}
.cg-wrap coords.ranks {
left: 4px;
top: -20px;
flex-flow: column-reverse;
height: 100%;
width: 12px;
}
.cg-wrap coords.ranks.black {
flex-flow: column;
}
.cg-wrap coords.files {
bottom: -4px;
left: 24px;
flex-flow: row;
width: 100%;
height: 16px;
text-transform: uppercase;
text-align: center;
}
.cg-wrap coords.files.black {
flex-flow: row-reverse;
}
.cg-wrap coords coord {
flex: 1 1 auto;
}
.cg-wrap coords.ranks coord {
transform: translateY(39%);
}
/** Colored board squares as an embedded SVG */
cg-board {
background-image: url('');
}
/** Interactive board square colors */
cg-board square.move-dest {
background: radial-gradient(rgba(20, 85, 30, 0.5) 22%, #208530 0, rgba(0, 0, 0, 0.3) 0, rgba(0, 0, 0, 0) 0);
}
cg-board square.premove-dest {
background: radial-gradient(rgba(20, 30, 85, 0.5) 22%, #203085 0, rgba(0, 0, 0, 0.3) 0, rgba(0, 0, 0, 0) 0);
}
cg-board square.oc.move-dest {
background: radial-gradient(transparent 0%, transparent 80%, rgba(20, 85, 0, 0.3) 80%);
}
cg-board square.oc.premove-dest {
background: radial-gradient(transparent 0%, transparent 80%, rgba(20, 30, 85, 0.2) 80%);
}
cg-board square.move-dest:hover {
background: rgba(20, 85, 30, 0.3);
}
cg-board square.premove-dest:hover {
background: rgba(20, 30, 85, 0.2);
}
cg-board square.last-move {
background-color: rgba(155, 199, 0, 0.41);
}
cg-board square.selected {
background-color: rgba(20, 85, 30, 0.5);
}
cg-board square.check {
background: radial-gradient(
ellipse at center,
rgba(255, 0, 0, 1) 0%,
rgba(231, 0, 0, 1) 25%,
rgba(169, 0, 0, 0) 89%,
rgba(158, 0, 0, 0) 100%
);
}
cg-board square.current-premove {
background-color: rgba(20, 30, 85, 0.5);
}
/** Alternating colors in rank/file labels */
.cg-wrap.orientation-white coords.ranks coord:nth-child(2n),
.cg-wrap.orientation-white coords.files coord:nth-child(2n),
.cg-wrap.orientation-black coords.ranks coord:nth-child(2n + 1),
.cg-wrap.orientation-black coords.files coord:nth-child(2n + 1) {
color: #b58863;
}
.cg-wrap.orientation-black coords.ranks coord:nth-child(2n),
.cg-wrap.orientation-black coords.files coord:nth-child(2n),
.cg-wrap.orientation-white coords.ranks coord:nth-child(2n + 1),
.cg-wrap.orientation-white coords.files coord:nth-child(2n + 1) {
color: #f0d9b5;
}
/** Embedded SVGs for all chess pieces */
.cg-wrap piece.pawn.white {
background-image: url('');
}
.cg-wrap piece.bishop.white {
background-image: url('');
}
.cg-wrap piece.knight.white {
background-image: url('');
}
.cg-wrap piece.rook.white {
background-image: url('');
}
.cg-wrap piece.queen.white {
background-image: url('');
}
.cg-wrap piece.king.white {
background-image: url('');
}
.cg-wrap piece.pawn.black {
background-image: url('');
}
.cg-wrap piece.bishop.black {
background-image: url('');
}
.cg-wrap piece.knight.black {
background-image: url('');
}
.cg-wrap piece.rook.black {
background-image: url('');
}
.cg-wrap piece.queen.black {
background-image: url('');
}
.cg-wrap piece.king.black {
background-image: url('');
}
var Chessground;(()=>{"use strict";var e={d:(t,o)=>{for(var n in o)e.o(o,n)&&!e.o(t,n)&&Object.defineProperty(t,n,{enumerable:!0,get:o[n]})},o:(e,t)=>Object.prototype.hasOwnProperty.call(e,t),r:e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})}},t={};(()=>{e.r(t),e.d(t,{Chessground:()=>Ve});const o=["white","black"],n=["a","b","c","d","e","f","g","h"],r=["1","2","3","4","5","6","7","8"],s=[...r].reverse(),i=Array.prototype.concat(...n.map((e=>r.map((t=>e+t))))),c=e=>i[8*e[0]+e[1]],a=e=>[e.charCodeAt(0)-97,e.charCodeAt(1)-49],l=i.map(a),d=()=>{let e;return{start(){e=performance.now()},cancel(){e=void 0},stop(){if(!e)return 0;const t=performance.now()-e;return e=void 0,t}}},u=e=>"white"===e?"black":"white",p=(e,t)=>{const o=e[0]-t[0],n=e[1]-t[1];return o*o+n*n},f=(e,t)=>e.role===t.role&&e.color===t.color,g=e=>(t,o)=>[(o?t[0]:7-t[0])*e.width/8,(o?7-t[1]:t[1])*e.height/8],h=(e,t)=>{e.style.transform=`translate(${t[0]}px,${t[1]}px)`},m=(e,t)=>{e.style.visibility=t?"visible":"hidden"},v=e=>{var t;return e.clientX||0===e.clientX?[e.clientX,e.clientY]:(null===(t=e.targetTouches)||void 0===t?void 0:t[0])?[e.targetTouches[0].clientX,e.targetTouches[0].clientY]:void 0},b=e=>2===e.buttons||2===e.button,w=(e,t)=>{const o=document.createElement(e);return t&&(o.className=t),o};function y(e,t,o){const n=a(e);return t||(n[0]=7-n[0],n[1]=7-n[1]),[o.left+o.width*n[0]/8+o.width/16,o.top+o.height*(7-n[1])/8+o.height/16]}function k(e,t){return Math.abs(e-t)}const C=(e,t,o,n)=>{const r=k(e,o),s=k(t,n);return 1===r&&2===s||2===r&&1===s},M=(e,t,o,n)=>k(e,o)===k(t,n),S=(e,t,o,n)=>e===o||t===n,P=(e,t,o,n)=>M(e,t,o,n)||S(e,t,o,n);function x(e,t,o){const n=e.get(t);if(!n)return[];const r=a(t),s=n.role,i="pawn"===s?(d=n.color,(e,t,o,n)=>k(e,o)<2&&("white"===d?n===t+1||t<=1&&n===t+2&&e===o:n===t-1||t>=6&&n===t-2&&e===o)):"knight"===s?C:"bishop"===s?M:"rook"===s?S:"queen"===s?P:function(e,t,o){return(n,r,s,i)=>k(n,s)<2&&k(r,i)<2||o&&r===i&&r===("white"===e?0:7)&&(4===n&&(2===s&&t.includes(0)||6===s&&t.includes(7))||t.includes(s))}(n.color,function(e,t){const o="white"===t?"1":"8",n=[];for(const[r,s]of e)r[1]===o&&s.color===t&&"rook"===s.role&&n.push(a(r)[0]);return n}(e,n.color),o);var d;return l.filter((e=>(r[0]!==e[0]||r[1]!==e[1])&&i(r[0],r[1],e[0],e[1]))).map(c)}function A(e,...t){e&&setTimeout((()=>e(...t)),1)}function K(e){e.premovable.current&&(e.premovable.current=void 0,A(e.premovable.events.unset))}function T(e){const t=e.predroppable;t.current&&(t.current=void 0,A(t.events.unset))}function q(e,t,o){const n=e.pieces.get(t),r=e.pieces.get(o);if(t===o||!n)return!1;const s=r&&r.color!==n.color?r:void 0;return o===e.selected&&W(e),A(e.events.move,t,o,s),function(e,t,o){if(!e.autoCastle)return!1;const n=e.pieces.get(t);if(!n||"king"!==n.role)return!1;const r=a(t),s=a(o);if(0!==r[1]&&7!==r[1]||r[1]!==s[1])return!1;4!==r[0]||e.pieces.has(o)||(6===s[0]?o=c([7,s[1]]):2===s[0]&&(o=c([0,s[1]])));const i=e.pieces.get(o);return!(!i||i.color!==n.color||"rook"!==i.role||(e.pieces.delete(t),e.pieces.delete(o),r[0]<s[0]?(e.pieces.set(c([6,s[1]]),n),e.pieces.set(c([5,s[1]]),i)):(e.pieces.set(c([2,s[1]]),n),e.pieces.set(c([3,s[1]]),i)),0))}(e,t,o)||(e.pieces.set(o,n),e.pieces.delete(t)),e.lastMove=[t,o],e.check=void 0,A(e.events.change),s||!0}function L(e,t,o,n){if(e.pieces.has(o)){if(!n)return!1;e.pieces.delete(o)}return A(e.events.dropNewPiece,t,o),e.pieces.set(o,t),e.lastMove=[o],e.check=void 0,A(e.events.change),e.movable.dests=void 0,e.turnColor=u(e.turnColor),!0}function O(e,t,o){const n=q(e,t,o);return n&&(e.movable.dests=void 0,e.turnColor=u(e.turnColor),e.animation.current=void 0),n}function N(e,t,o){if(R(e,t,o)){const n=O(e,t,o);if(n){const r=e.hold.stop();W(e);const s={premove:!1,ctrlKey:e.stats.ctrlKey,holdTime:r};return!0!==n&&(s.captured=n),A(e.movable.events.after,t,o,s),!0}}else if(function(e,t,o){return t!==o&&j(e,t)&&x(e.pieces,t,e.premovable.castle).includes(o)}(e,t,o))return function(e,t,o,n){T(e),e.premovable.current=[t,o],A(e.premovable.events.set,t,o,n)}(e,t,o,{ctrlKey:e.stats.ctrlKey}),W(e),!0;return W(e),!1}function D(e,t,o,n){const r=e.pieces.get(t);r&&(function(e,t,o){const n=e.pieces.get(t);return!(!n||t!==o&&e.pieces.has(o)||"both"!==e.movable.color&&(e.movable.color!==n.color||e.turnColor!==n.color))}(e,t,o)||n)?(e.pieces.delete(t),L(e,r,o,n),A(e.movable.events.afterNewPiece,r.role,o,{premove:!1,predrop:!1})):r&&function(e,t,o){const n=e.pieces.get(t),r=e.pieces.get(o);return!!n&&(!r||r.color!==e.movable.color)&&e.predroppable.enabled&&("pawn"!==n.role||"1"!==o[1]&&"8"!==o[1])&&e.movable.color===n.color&&e.turnColor!==n.color}(e,t,o)?function(e,t,o){K(e),e.predroppable.current={role:t,key:o},A(e.predroppable.events.set,t,o)}(e,r.role,o):(K(e),T(e)),e.pieces.delete(t),W(e)}function $(e,t,o){if(A(e.events.select,t),e.selected){if(e.selected===t&&!e.draggable.enabled)return W(e),void e.hold.cancel();if((e.selectable.enabled||o)&&e.selected!==t&&N(e,e.selected,t))return void(e.stats.dragged=!1)}(H(e,t)||j(e,t))&&(E(e,t),e.hold.start())}function E(e,t){e.selected=t,j(e,t)?e.premovable.dests=x(e.pieces,t,e.premovable.castle):e.premovable.dests=void 0}function W(e){e.selected=void 0,e.premovable.dests=void 0,e.hold.cancel()}function H(e,t){const o=e.pieces.get(t);return!!o&&("both"===e.movable.color||e.movable.color===o.color&&e.turnColor===o.color)}function R(e,t,o){var n,r;return t!==o&&H(e,t)&&(e.movable.free||!!(null===(r=null===(n=e.movable.dests)||void 0===n?void 0:n.get(t))||void 0===r?void 0:r.includes(o)))}function j(e,t){const o=e.pieces.get(t);return!!o&&e.premovable.enabled&&e.movable.color===o.color&&e.turnColor!==o.color}function B(e){const t=e.premovable.current;if(!t)return!1;const o=t[0],n=t[1];let r=!1;if(R(e,o,n)){const t=O(e,o,n);if(t){const s={premove:!0};!0!==t&&(s.captured=t),A(e.movable.events.after,o,n,s),r=!0}}return K(e),r}function F(e){K(e),T(e),W(e)}function z(e){e.movable.color=e.movable.dests=e.animation.current=void 0,F(e)}function I(e,t,o){let n=Math.floor(8*(e[0]-o.left)/o.width);t||(n=7-n);let r=7-Math.floor(8*(e[1]-o.top)/o.height);return t||(r=7-r),n>=0&&n<8&&r>=0&&r<8?c([n,r]):void 0}function V(e){return"white"===e.orientation}const G="rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR",U={p:"pawn",r:"rook",n:"knight",b:"bishop",q:"queen",k:"king"},X={pawn:"p",rook:"r",knight:"n",bishop:"b",queen:"q",king:"k"};function Y(e){"start"===e&&(e=G);const t=new Map;let o=7,n=0;for(const r of e)switch(r){case" ":return t;case"/":if(--o,o<0)return t;n=0;break;case"~":{const e=t.get(c([n,o]));e&&(e.promoted=!0);break}default:{const e=r.charCodeAt(0);if(e<57)n+=e-48;else{const e=r.toLowerCase();t.set(c([n,o]),{role:U[e],color:r===e?"black":"white"}),++n}}}return t}function Z(e,t){t.animation&&(_(e.animation,t.animation),(e.animation.duration||0)<70&&(e.animation.enabled=!1))}function Q(e,t){var o,n;if((null===(o=t.movable)||void 0===o?void 0:o.dests)&&(e.movable.dests=void 0),(null===(n=t.drawable)||void 0===n?void 0:n.autoShapes)&&(e.drawable.autoShapes=[]),_(e,t),t.fen&&(e.pieces=Y(t.fen),e.drawable.shapes=[]),"check"in t&&function(e,t){if(e.check=void 0,!0===t&&(t=e.turnColor),t)for(const[o,n]of e.pieces)"king"===n.role&&n.color===t&&(e.check=o)}(e,t.check||!1),"lastMove"in t&&!t.lastMove?e.lastMove=void 0:t.lastMove&&(e.lastMove=t.lastMove),e.selected&&E(e,e.selected),Z(e,t),!e.movable.rookCastle&&e.movable.dests){const t="white"===e.movable.color?"1":"8",o="e"+t,n=e.movable.dests.get(o),r=e.pieces.get(o);if(!n||!r||"king"!==r.role)return;e.movable.dests.set(o,n.filter((e=>!(e==="a"+t&&n.includes("c"+t)||e==="h"+t&&n.includes("g"+t)))))}}function _(e,t){for(const o in t)J(e[o])&&J(t[o])?_(e[o],t[o]):e[o]=t[o]}function J(e){return"object"==typeof e}function ee(e,t){return t.animation.enabled?function(e,t){const o=new Map(t.pieces),n=e(t),r=function(e,t){const o=new Map,n=[],r=new Map,s=[],c=[],a=new Map;let l,d,u;for(const[t,o]of e)a.set(t,oe(t,o));for(const e of i)l=t.pieces.get(e),d=a.get(e),l?d?f(l,d.piece)||(s.push(d),c.push(oe(e,l))):c.push(oe(e,l)):d&&s.push(d);for(const e of c)d=ne(e,s.filter((t=>f(e.piece,t.piece)))),d&&(u=[d.pos[0]-e.pos[0],d.pos[1]-e.pos[1]],o.set(e.key,u.concat(u)),n.push(d.key));for(const e of s)n.includes(e.key)||r.set(e.key,e.piece);return{anims:o,fadings:r}}(o,t);if(r.anims.size||r.fadings.size){const e=t.animation.current&&t.animation.current.start;t.animation.current={start:performance.now(),frequency:1/t.animation.duration,plan:r},e||re(t,performance.now())}else t.dom.redraw();return n}(e,t):te(e,t)}function te(e,t){const o=e(t);return t.dom.redraw(),o}function oe(e,t){return{key:e,pos:a(e),piece:t}}function ne(e,t){return t.sort(((t,o)=>p(e.pos,t.pos)-p(e.pos,o.pos)))[0]}function re(e,t){const o=e.animation.current;if(void 0===o)return void(e.dom.destroyed||e.dom.redrawNow());const n=1-(t-o.start)*o.frequency;if(n<=0)e.animation.current=void 0,e.dom.redrawNow();else{const t=(r=n)<.5?4*r*r*r:(r-1)*(2*r-2)*(2*r-2)+1;for(const e of o.plan.anims.values())e[2]=e[0]*t,e[3]=e[1]*t;e.dom.redrawNow(!0),requestAnimationFrame(((t=performance.now())=>re(e,t)))}var r}const se=["green","red","blue","yellow"];function ie(e,t){if(t.touches&&t.touches.length>1)return;t.stopPropagation(),t.preventDefault(),t.ctrlKey?W(e):F(e);const o=v(t),n=I(o,V(e),e.dom.bounds());n&&(e.drawable.current={orig:n,pos:o,brush:ue(t),snapToValidMove:e.drawable.defaultSnapToValidMove},ce(e))}function ce(e){requestAnimationFrame((()=>{const t=e.drawable.current;if(t){const o=I(t.pos,V(e),e.dom.bounds());o||(t.snapToValidMove=!1);const n=t.snapToValidMove?function(e,t,o,n){const r=a(e),s=l.filter((e=>P(r[0],r[1],e[0],e[1])||C(r[0],r[1],e[0],e[1]))),i=s.map((e=>y(c(e),o,n))).map((e=>p(t,e))),[,d]=i.reduce(((e,t,o)=>e[0]<t?e:[t,o]),[i[0],0]);return c(s[d])}(t.orig,t.pos,V(e),e.dom.bounds()):o;n!==t.mouseSq&&(t.mouseSq=n,t.dest=n!==t.orig?n:void 0,e.dom.redrawNow()),ce(e)}}))}function ae(e,t){e.drawable.current&&(e.drawable.current.pos=v(t))}function le(e){const t=e.drawable.current;t&&(t.mouseSq&&function(e,t){const o=e=>e.orig===t.orig&&e.dest===t.dest,n=e.shapes.find(o);n&&(e.shapes=e.shapes.filter((e=>!o(e)))),n&&n.brush===t.brush||e.shapes.push(t),pe(e)}(e.drawable,t),de(e))}function de(e){e.drawable.current&&(e.drawable.current=void 0,e.dom.redraw())}function ue(e){var t;const o=(e.shiftKey||e.ctrlKey)&&b(e),n=e.altKey||e.metaKey||(null===(t=e.getModifierState)||void 0===t?void 0:t.call(e,"AltGraph"));return se[(o?1:0)+(n?2:0)]}function pe(e){e.onChange&&e.onChange(e.shapes)}function fe(e){requestAnimationFrame((()=>{var t;const o=e.draggable.current;if(!o)return;(null===(t=e.animation.current)||void 0===t?void 0:t.plan.anims.has(o.orig))&&(e.animation.current=void 0);const n=e.pieces.get(o.orig);if(n&&f(n,o.piece)){if(!o.started&&p(o.pos,o.origPos)>=Math.pow(e.draggable.distance,2)&&(o.started=!0),o.started){if("function"==typeof o.element){const e=o.element();if(!e)return;e.cgDragging=!0,e.classList.add("dragging"),o.element=e}const t=e.dom.bounds();h(o.element,[o.pos[0]-t.left-t.width/16,o.pos[1]-t.top-t.height/16]),o.keyHasChanged||(o.keyHasChanged=o.orig!==I(o.pos,V(e),t))}}else me(e);fe(e)}))}function ge(e,t){e.draggable.current&&(!t.touches||t.touches.length<2)&&(e.draggable.current.pos=v(t))}function he(e,t){const o=e.draggable.current;if(!o)return;if("touchend"===t.type&&!1!==t.cancelable&&t.preventDefault(),"touchend"===t.type&&o.originTarget!==t.target&&!o.newPiece)return void(e.draggable.current=void 0);K(e),T(e);const n=I(v(t)||o.pos,V(e),e.dom.bounds());n&&o.started&&o.orig!==n?o.newPiece?D(e,o.orig,n,o.force):(e.stats.ctrlKey=t.ctrlKey,N(e,o.orig,n)&&(e.stats.dragged=!0)):o.newPiece?e.pieces.delete(o.orig):e.draggable.deleteOnDropOff&&!n&&(e.pieces.delete(o.orig),A(e.events.change)),(o.orig!==o.previouslySelected&&!o.keyHasChanged||o.orig!==n&&n)&&e.selectable.enabled||W(e),ve(e),e.draggable.current=void 0,e.dom.redraw()}function me(e){const t=e.draggable.current;t&&(t.newPiece&&e.pieces.delete(t.orig),e.draggable.current=void 0,W(e),ve(e),e.dom.redraw())}function ve(e){const t=e.dom.elements;t.ghost&&m(t.ghost,!1)}function be(e,t){let o=e.dom.elements.board.firstChild;for(;o;){if(o.cgKey===t&&"PIECE"===o.tagName)return o;o=o.nextSibling}}function we(e,t){e.exploding&&(t?e.exploding.stage=t:e.exploding=void 0,e.dom.redraw())}function ye(e){return document.createElementNS("http://www.w3.org/2000/svg",e)}function ke(e,t,o,n,r){const s=e.dom.bounds(),i=new Map,c=[];for(const e of t)i.set(e.hash,!1);let a,l=r.firstChild;for(;l;)a=l.getAttribute("cgHash"),i.has(a)?i.set(a,!0):c.push(l),l=l.nextSibling;for(const e of c)r.removeChild(e);for(const c of t)i.get(c.hash)||r.appendChild(Pe(e,c,o,n,s))}function Ce({orig:e,dest:t,brush:o,piece:n,modifiers:r,customSvg:s},i,c,a){return[a.width,a.height,c,e,t,o,t&&(i.get(t)||0)>1,n&&Me(n),r&&(l=r,""+(l.lineWidth||"")),s&&Se(s)].filter((e=>e)).join(",");var l}function Me(e){return[e.color,e.role,e.scale].filter((e=>e)).join(",")}function Se(e){let t=0;for(let o=0;o<e.length;o++)t=(t<<5)-t+e.charCodeAt(o)>>>0;return"custom-"+t.toString()}function Pe(e,{shape:t,current:o,hash:n},r,s,i){let c;if(t.customSvg){const o=Ke(a(t.orig),e.orientation);c=function(e,t,o){const[n,r]=Oe(t,o),s=Ae(ye("g"),{transform:`translate(${n},${r})`}),i=Ae(ye("svg"),{width:1,height:1,viewBox:"0 0 100 100"});return s.appendChild(i),i.innerHTML=e,s}(t.customSvg,o,i)}else if(t.piece)c=function(e,t,o,n){const r=Oe(t,n),s=o.color[0]+("knight"===o.role?"n":o.role[0]).toUpperCase();return Ae(ye("image"),{className:`${o.role} ${o.color}`,x:r[0]-.5,y:r[1]-.5,width:1,height:1,href:e+s+".svg",transform:`scale(${o.scale||1})`,"transform-origin":`${r[0]} ${r[1]}`})}(e.drawable.pieces.baseUrl,Ke(a(t.orig),e.orientation),t.piece,i);else{const n=Ke(a(t.orig),e.orientation);if(t.dest){let l=r[t.brush];t.modifiers&&(l=Te(l,t.modifiers)),c=function(e,t,o,n,r,s){const i=function(e){return(e?20:10)/64}(r&&!n),c=Oe(t,s),a=Oe(o,s),l=a[0]-c[0],d=a[1]-c[1],u=Math.atan2(d,l),p=Math.cos(u)*i,f=Math.sin(u)*i;return Ae(ye("line"),{stroke:e.color,"stroke-width":qe(e,n),"stroke-linecap":"round","marker-end":"url(#arrowhead-"+e.key+")",opacity:Le(e,n),x1:c[0],y1:c[1],x2:a[0]-p,y2:a[1]-f})}(l,n,Ke(a(t.dest),e.orientation),o,(s.get(t.dest)||0)>1,i)}else c=function(e,t,o,n){const r=Oe(t,n),s=[3/64,4/64],i=(n.width+n.height)/(4*Math.max(n.width,n.height));return Ae(ye("circle"),{stroke:e.color,"stroke-width":s[o?0:1],fill:"none",opacity:Le(e,o),cx:r[0],cy:r[1],r:i-s[1]/2})}(r[t.brush],n,o,i)}return c.setAttribute("cgHash",n),c}function xe(e){const t=Ae(ye("marker"),{id:"arrowhead-"+e.key,orient:"auto",markerWidth:4,markerHeight:8,refX:2.05,refY:2.01});return t.appendChild(Ae(ye("path"),{d:"M0,0 V4 L3,2 Z",fill:e.color})),t.setAttribute("cgKey",e.key),t}function Ae(e,t){for(const o in t)e.setAttribute(o,t[o]);return e}function Ke(e,t){return"white"===t?e:[7-e[0],7-e[1]]}function Te(e,t){return{color:e.color,opacity:Math.round(10*e.opacity)/10,lineWidth:Math.round(t.lineWidth||e.lineWidth),key:[e.key,t.lineWidth].filter((e=>e)).join("")}}function qe(e,t){return(e.lineWidth||10)*(t?.85:1)/64}function Le(e,t){return(e.opacity||1)*(t?.9:1)}function Oe(e,t){const o=Math.min(1,t.width/t.height),n=Math.min(1,t.height/t.width);return[(e[0]-3.5)*o,(3.5-e[1])*n]}function Ne(e,t){const o=w("coords",t);let n;for(const t of e)n=w("coord"),n.textContent=t,o.appendChild(n);return o}function De(e,t){const o=e.dom.elements.board;if("ResizeObserver"in window&&new ResizeObserver(t).observe(e.dom.elements.wrap),e.viewOnly)return;const n=function(e){return t=>{e.draggable.current?me(e):e.drawable.current?de(e):t.shiftKey||b(t)?e.drawable.enabled&&ie(e,t):e.viewOnly||(e.dropmode.active?function(e,t){if(!e.dropmode.active)return;K(e),T(e);const o=e.dropmode.piece;if(o){e.pieces.set("a0",o);const n=v(t),r=n&&I(n,V(e),e.dom.bounds());r&&D(e,"a0",r)}e.dom.redraw()}(e,t):function(e,t){if(!t.isTrusted||void 0!==t.button&&0!==t.button)return;if(t.touches&&t.touches.length>1)return;const o=e.dom.bounds(),n=v(t),r=I(n,V(e),o);if(!r)return;const s=e.pieces.get(r),i=e.selected;var c;i||!e.drawable.enabled||!e.drawable.eraseOnClick&&s&&s.color===e.turnColor||(c=e).drawable.shapes.length&&(c.drawable.shapes=[],c.dom.redraw(),pe(c.drawable)),!1!==t.cancelable&&(!t.touches||e.blockTouchScroll||s||i||function(e,t){const o=V(e),n=e.dom.bounds(),r=Math.pow(n.width/8,2);for(const s of e.pieces.keys()){const e=y(s,o,n);if(p(e,t)<=r)return!0}return!1}(e,n))&&t.preventDefault();const l=!!e.premovable.current,d=!!e.predroppable.current;e.stats.ctrlKey=t.ctrlKey,e.selected&&R(e,e.selected,r)?ee((e=>$(e,r)),e):$(e,r);const u=e.selected===r,f=be(e,r);if(s&&f&&u&&function(e,t){const o=e.pieces.get(t);return!!o&&e.draggable.enabled&&("both"===e.movable.color||e.movable.color===o.color&&(e.turnColor===o.color||e.premovable.enabled))}(e,r)){e.draggable.current={orig:r,piece:s,origPos:n,pos:n,started:e.draggable.autoDistance&&e.stats.dragged,element:f,previouslySelected:i,originTarget:t.target,keyHasChanged:!1},f.cgDragging=!0,f.classList.add("dragging");const c=e.dom.elements.ghost;c&&(c.className=`ghost ${s.color} ${s.role}`,h(c,g(o)(a(r),V(e))),m(c,!0)),fe(e)}else l&&K(e),d&&T(e);e.dom.redraw()}(e,t))}}(e);o.addEventListener("touchstart",n,{passive:!1}),o.addEventListener("mousedown",n,{passive:!1}),(e.disableContextMenu||e.drawable.enabled)&&o.addEventListener("contextmenu",(e=>e.preventDefault()))}function $e(e,t,o,n){return e.addEventListener(t,o,n),()=>e.removeEventListener(t,o,n)}function Ee(e,t,o){return n=>{e.drawable.current?e.drawable.enabled&&o(e,n):e.viewOnly||t(e,n)}}function We(e){const t=e.dom.elements.wrap.getBoundingClientRect(),o=e.dom.elements.container,n=t.height/t.width,r=8*Math.floor(t.width*window.devicePixelRatio/8)/window.devicePixelRatio,s=r*n;o.style.width=r+"px",o.style.height=s+"px",e.dom.bounds.clear(),e.addDimensionsCssVars&&(document.documentElement.style.setProperty("--cg-width",r+"px"),document.documentElement.style.setProperty("--cg-height",s+"px"))}function He(e){return"PIECE"===e.tagName}function Re(e){return"SQUARE"===e.tagName}function je(e,t){for(const o of t)e.dom.elements.board.removeChild(o)}function Be(e,t){const o=e[1];return`${t?10-o:3+o}`}function Fe(e){return`${e.color} ${e.role}`}function ze(e,t,o){const n=e.get(t);n?e.set(t,`${n} ${o}`):e.set(t,o)}function Ie(e,t,o){const n=e.get(t);n?n.push(o):e.set(t,[o])}function Ve(e,t){const i={pieces:Y(G),orientation:"white",turnColor:"white",coordinates:!0,autoCastle:!0,viewOnly:!1,disableContextMenu:!1,addPieceZIndex:!1,addDimensionsCssVars:!1,blockTouchScroll:!1,pieceKey:!1,highlight:{lastMove:!0,check:!0},animation:{enabled:!0,duration:200},movable:{free:!0,color:"both",showDests:!0,events:{},rookCastle:!0},premovable:{enabled:!0,showDests:!0,castle:!0,events:{}},predroppable:{enabled:!1,events:{}},draggable:{enabled:!0,distance:3,autoDistance:!0,showGhost:!0,deleteOnDropOff:!1},dropmode:{active:!1},selectable:{enabled:!0},stats:{dragged:!("ontouchstart"in window)},events:{},drawable:{enabled:!0,visible:!0,defaultSnapToValidMove:!0,eraseOnClick:!0,shapes:[],autoShapes:[],brushes:{green:{key:"g",color:"#15781B",opacity:1,lineWidth:10},red:{key:"r",color:"#882020",opacity:1,lineWidth:10},blue:{key:"b",color:"#003088",opacity:1,lineWidth:10},yellow:{key:"y",color:"#e68f00",opacity:1,lineWidth:10},paleBlue:{key:"pb",color:"#003088",opacity:.4,lineWidth:15},paleGreen:{key:"pg",color:"#15781B",opacity:.4,lineWidth:15},paleRed:{key:"pr",color:"#882020",opacity:.4,lineWidth:15},paleGrey:{key:"pgr",color:"#4a4a4a",opacity:.35,lineWidth:15}},pieces:{baseUrl:"https://lichess1.org/assets/piece/cburnett/"},prevSvgHash:""},hold:d()};function c(){const t="dom"in i?i.dom.unbind:void 0,s=function(e,t){e.innerHTML="",e.classList.add("cg-wrap");for(const n of o)e.classList.toggle("orientation-"+n,t.orientation===n);e.classList.toggle("manipulable",!t.viewOnly);const s=w("cg-container");e.appendChild(s);const i=w("cg-board");let c,a,l;if(s.appendChild(i),t.drawable.visible&&(c=Ae(ye("svg"),{class:"cg-shapes",viewBox:"-4 -4 8 8",preserveAspectRatio:"xMidYMid slice"}),c.appendChild(ye("defs")),c.appendChild(ye("g")),a=Ae(ye("svg"),{class:"cg-custom-svgs",viewBox:"-3.5 -3.5 8 8",preserveAspectRatio:"xMidYMid slice"}),a.appendChild(ye("g")),s.appendChild(c),s.appendChild(a)),t.coordinates){const e="black"===t.orientation?" black":"";s.appendChild(Ne(r,"ranks"+e)),s.appendChild(Ne(n,"files"+e))}return t.draggable.showGhost&&(l=w("piece","ghost"),m(l,!1),s.appendChild(l)),{board:i,container:s,wrap:e,ghost:l,svg:c,customSvg:a}}(e,i),c=function(e){let t;const o=()=>(void 0===t&&(t=s.board.getBoundingClientRect()),t);return o.clear=()=>{t=void 0},o}(),l=e=>{(function(e){const t=V(e),o=g(e.dom.bounds()),n=e.dom.elements.board,r=e.pieces,s=e.animation.current,i=s?s.plan.anims:new Map,c=s?s.plan.fadings:new Map,l=e.draggable.current,d=function(e){var t;const o=new Map;if(e.lastMove&&e.highlight.lastMove)for(const t of e.lastMove)ze(o,t,"last-move");if(e.check&&e.highlight.check&&ze(o,e.check,"check"),e.selected&&(ze(o,e.selected,"selected"),e.movable.showDests)){const n=null===(t=e.movable.dests)||void 0===t?void 0:t.get(e.selected);if(n)for(const t of n)ze(o,t,"move-dest"+(e.pieces.has(t)?" oc":""));const r=e.premovable.dests;if(r)for(const t of r)ze(o,t,"premove-dest"+(e.pieces.has(t)?" oc":""))}const n=e.premovable.current;if(n)for(const e of n)ze(o,e,"current-premove");else e.predroppable.current&&ze(o,e.predroppable.current.key,"current-premove");const r=e.exploding;if(r)for(const e of r.keys)ze(o,e,"exploding"+r.stage);return o}(e),u=new Set,p=new Set,f=new Map,m=new Map;let v,b,y,k,C,M,S,P,x,A;for(b=n.firstChild;b;){if(v=b.cgKey,He(b))if(y=r.get(v),C=i.get(v),M=c.get(v),k=b.cgPiece,!b.cgDragging||l&&l.orig===v||(b.classList.remove("dragging"),h(b,o(a(v),t)),b.cgDragging=!1),!M&&b.cgFading&&(b.cgFading=!1,b.classList.remove("fading")),y){if(C&&b.cgAnimating&&k===Fe(y)){const e=a(v);e[0]+=C[2],e[1]+=C[3],b.classList.add("anim"),h(b,o(e,t))}else b.cgAnimating&&(b.cgAnimating=!1,b.classList.remove("anim"),h(b,o(a(v),t)),e.addPieceZIndex&&(b.style.zIndex=Be(a(v),t)));k!==Fe(y)||M&&b.cgFading?M&&k===Fe(M)?(b.classList.add("fading"),b.cgFading=!0):Ie(f,k,b):u.add(v)}else Ie(f,k,b);else if(Re(b)){const e=b.className;d.get(v)===e?p.add(v):Ie(m,e,b)}b=b.nextSibling}for(const[e,r]of d)if(!p.has(e)){x=m.get(r),A=x&&x.pop();const s=o(a(e),t);if(A)A.cgKey=e,h(A,s);else{const t=w("square",r);t.cgKey=e,h(t,s),n.insertBefore(t,n.firstChild)}}for(const[s,c]of r)if(C=i.get(s),!u.has(s))if(S=f.get(Fe(c)),P=S&&S.pop(),P){P.cgKey=s,P.cgFading&&(P.classList.remove("fading"),P.cgFading=!1);const n=a(s);e.addPieceZIndex&&(P.style.zIndex=Be(n,t)),C&&(P.cgAnimating=!0,P.classList.add("anim"),n[0]+=C[2],n[1]+=C[3]),h(P,o(n,t))}else{const r=Fe(c),i=w("piece",r),l=a(s);i.cgPiece=r,i.cgKey=s,C&&(i.cgAnimating=!0,l[0]+=C[2],l[1]+=C[3]),h(i,o(l,t)),e.addPieceZIndex&&(i.style.zIndex=Be(l,t)),n.appendChild(i)}for(const t of f.values())je(e,t);for(const t of m.values())je(e,t)})(u),!e&&s.svg&&function(e,t,o){const n=e.drawable,r=n.current,s=r&&r.mouseSq?r:void 0,i=new Map,c=e.dom.bounds();for(const e of n.shapes.concat(n.autoShapes).concat(s?[s]:[]))e.dest&&i.set(e.dest,(i.get(e.dest)||0)+1);const a=n.shapes.concat(n.autoShapes).map((e=>({shape:e,current:!1,hash:Ce(e,i,!1,c)})));s&&a.push({shape:s,current:!0,hash:Ce(s,i,!0,c)});const l=a.map((e=>e.hash)).join(";");if(l===e.drawable.prevSvgHash)return;e.drawable.prevSvgHash=l;const d=t.querySelector("defs"),u=t.querySelector("g"),p=o.querySelector("g");!function(e,t,o){const n=new Map;let r;for(const o of t)o.shape.dest&&(r=e.brushes[o.shape.brush],o.shape.modifiers&&(r=Te(r,o.shape.modifiers)),n.set(r.key,r));const s=new Set;let i=o.firstChild;for(;i;)s.add(i.getAttribute("cgKey")),i=i.nextSibling;for(const[e,t]of n.entries())s.has(e)||o.appendChild(xe(t))}(n,a,d),ke(e,a.filter((e=>!e.shape.customSvg)),n.brushes,i,u),ke(e,a.filter((e=>e.shape.customSvg)),n.brushes,i,p)}(u,s.svg,s.customSvg)},d=()=>{We(u),function(e){const t=V(e),o=g(e.dom.bounds());let n=e.dom.elements.board.firstChild;for(;n;)(He(n)&&!n.cgAnimating||Re(n))&&h(n,o(a(n.cgKey),t)),n=n.nextSibling}(u)},u=i;return u.dom={elements:s,bounds:c,redraw:Ge(l),redrawNow:l,unbind:t},u.drawable.prevSvgHash="",We(u),l(!1),De(u,d),t||(u.dom.unbind=function(e,t){const o=[];if("ResizeObserver"in window||o.push($e(document.body,"chessground.resize",t)),!e.viewOnly){const t=Ee(e,ge,ae),n=Ee(e,he,le);for(const e of["touchmove","mousemove"])o.push($e(document,e,t));for(const e of["touchend","mouseup"])o.push($e(document,e,n));const r=()=>e.dom.bounds.clear();o.push($e(document,"scroll",r,{capture:!0,passive:!0})),o.push($e(window,"resize",r,{passive:!0}))}return()=>o.forEach((e=>e()))}(u,d)),u.events.insert&&u.events.insert(s),u}return Q(i,t||{}),function(e,t){function o(){!function(e){e.orientation=u(e.orientation),e.animation.current=e.draggable.current=e.selected=void 0}(e),t()}return{set(t){t.orientation&&t.orientation!==e.orientation&&o(),Z(e,t),(t.fen?ee:te)((e=>Q(e,t)),e)},state:e,getFen:()=>{return t=e.pieces,s.map((e=>n.map((o=>{const n=t.get(o+e);if(n){const e=X[n.role];return"white"===n.color?e.toUpperCase():e}return"1"})).join(""))).join("/").replace(/1{2,}/g,(e=>e.length.toString()));var t},toggleOrientation:o,setPieces(t){ee((e=>function(e,t){for(const[o,n]of t)n?e.pieces.set(o,n):e.pieces.delete(o)}(e,t)),e)},selectSquare(t,o){t?ee((e=>$(e,t,o)),e):e.selected&&(W(e),e.dom.redraw())},move(t,o){ee((e=>q(e,t,o)),e)},newPiece(t,o){ee((e=>L(e,t,o)),e)},playPremove(){if(e.premovable.current){if(ee(B,e))return!0;e.dom.redraw()}return!1},playPredrop(t){if(e.predroppable.current){const o=function(e,t){const o=e.predroppable.current;let n=!1;return!!o&&(t(o)&&L(e,{role:o.role,color:e.movable.color},o.key)&&(A(e.movable.events.afterNewPiece,o.role,o.key,{premove:!1,predrop:!0}),n=!0),T(e),n)}(e,t);return e.dom.redraw(),o}return!1},cancelPremove(){te(K,e)},cancelPredrop(){te(T,e)},cancelMove(){te((e=>{F(e),me(e)}),e)},stop(){te((e=>{z(e),me(e)}),e)},explode(t){!function(e,t){e.exploding={stage:1,keys:t},e.dom.redraw(),setTimeout((()=>{we(e,2),setTimeout((()=>we(e,void 0)),120)}),120)}(e,t)},setAutoShapes(t){te((e=>e.drawable.autoShapes=t),e)},setShapes(t){te((e=>e.drawable.shapes=t),e)},getKeyAtDomPos:t=>I(t,V(e),e.dom.bounds()),redrawAll:t,dragNewPiece(t,o,n){!function(e,t,o,n){const r="a0";e.pieces.set(r,t),e.dom.redraw();const s=v(o);e.draggable.current={orig:r,piece:t,origPos:s,pos:s,started:!0,element:()=>be(e,r),originTarget:o.target,newPiece:!0,force:!!n,keyHasChanged:!1},fe(e)}(e,t,o,n)},destroy(){z(e),e.dom.unbind&&e.dom.unbind(),e.dom.destroyed=!0}}}(c(),c)}function Ge(e){let t=!1;return()=>{t||(t=!0,requestAnimationFrame((()=>{e(),t=!1})))}}})(),Chessground=t})();
<!doctype html>
<html>
<head>
<title>The Knight's Voyage</title>
<meta charset="UTF-8">
<link rel="stylesheet" href="chessground.css">
<style type="text/css">
.timer {
text-align: center;
font-size: 80px;
}
.board-container-side-by-side {
display: flex;
}
#board-column { margin-pottom: 20px; }
body {
background-color: black;
color: white;
}
table {
background-color: #262421;
}
tr {
border: none;
}
p { text-align: center; font-size: 20px; }
td, th {
border: none;
text-align: center;
padding-right: 20px;
padding-left: 20px;
}
.splits-pb { color: rgb(22, 166, 255); }
.splits-ahead.splits-gain { color: rgb(41, 204, 84); }
.splits-ahead.splits-loss { color: rgb(112, 204, 137); }
.splits-behind.splits-gain { color: rgb(204, 120, 112); }
.splits-behind.splits-loss { color: rgb(204, 55, 41); }
.splits-glod { color: rgb(216, 175, 31); }
table { margin: auto; font-size: 30px; }
</style>
</head>
<script src="chessground.js"></script>
<script src="chess.js"></script>
<body>
<div id="board-container">
<div id="board-column">
<div id="board" style="width: 900px; height: 900px; margin: auto"></div>
</div>
<div id="splits">
<table>
<thead>
<tr>
<th>Square</th>
<th>Time</th>
<th></th>
<th>Best</th>
<th># Moves</th>
<th>Min. #Moves</th>
</tr>
</thead>
<tbody>
<tr id="splits-total">
<td class="splits-square">Total</td>
<td class="splits-time"></td>
<td class="splits-delta"></td>
<td class="splits-best-time"></td>
<td class="splits-num-moves"></td>
<td class="splits-min-moves"></td>
</tr>
<tr id="splits-best">
<td class="splits-square">Sum of Best</td>
<td class="splits-time"></td>
<td class="splits-delta"></td>
<td class="splits-best-time"></td>
<td class="splits-num-moves"></td>
<td class="splits-min-moves"></td>
</tr>
</tbody>
</table>
</div>
</div>
<p class="timer">Time: <span id="timer-count">0.00</span></p>
<p>
<label for="fen">FEN:</label>
<input type="text" id="fen" name="fen" value="">
<button id="load-fen">Load</button>
<br/>
<label for="fen-selector">Preset: </label>
<select name="fen-selector" id="fen-selector">
<option value="7n/8/8/3Q4/8/8/8/8 b - - 0 1">One Queen</option>
<option value="7n/8/8/8/8/8/8/8 b - - 0 1">Clear Board</option>
<option value="7n/8/8/3B4/8/8/5B2/8 b - - 0 1">Two Bishops</option>
<option value="7n/8/8/1B6/5R2/8/8/8 b - - 0 1">Bishop & Rook</option>
<option value="7n/8/2N5/3N4/8/8/8/3B4 b - - 0 1">Two Knights & Bishop</option>
<option value="5n2/6R1/8/8/8/2R5/8/8 b - - 0 1">Two Rooks</option>
</select>
<br/>
<input type="checkbox" id="show-inaccessible" name="show-inaccessible"
value="on">
<label for="show-inaccessible">Show Inaccessible squares</label>
<br/>
</p>
</body>
<script>
function formatTime(delta) {
return (delta / 1000).toFixed(2).toString();
}
function updateTimer(timestamp) {
if (!startTime) {
startTime = timestamp;
lastSplitTime = timestamp;
}
document.querySelector("#timer-count").innerHTML = formatTime(timestamp - startTime);
if (targetId < targets.length) {
updateSplits(targets[targetId]);
timerUpdateRequest = requestAnimationFrame(updateTimer);
}
}
function isAttacked(game, target) {
let moves = game.moves({legal: false, verbose: true});
return moves.some(move => move.to == target);
}
function passTurn(game) {
let parts = game.fen().split(" ");
parts[1] = parts[1] == "w" ? "b" : "w";
let newFen = parts.join(" ");
game.load(newFen);
}
function successors(game) {
let successors = [];
let moves = game.moves({legal: false, verbose: true});
moves.forEach((move) => {
if (game.get(move.to)) return;
let old_source = game.get(move.from);
let old_target = game.get(move.to);
game.put(old_source, move.to);
game.remove(move.from);
passTurn(game);
if (!isAttacked(game, move.to)) {
passTurn(game);
successors.push([move.from, move.to, game.fen()])
passTurn(game);
}
if (old_target)
game.put(old_target, move.to);
else
game.remove(move.to);
if (old_source)
game.put(old_source, move.from);
else
game.remove(move.from);
passTurn(game);
});
return successors;
}
function legalMoves(game) {
let moves = new Map();
successors(game).forEach((entry) => {
let [previous, square, fen] = entry;
if (!moves.has(previous))
moves.set(previous, [square]);
else
moves.get(previous).push(square);
});
return moves;
}
function updateLegalMoves() {
board.set({
turnColor: game.turn() == game.WHITE ? "white" : "black",
movable: {dests: legalMoves(game)},
});
}
function currentSquare(game) {
for (let y = 0; y < 8; y++) {
for (let x = 0; x < 8; x++) {
let file = ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h'][y];
let rank = (x + 1).toString();
let square = file + rank;
if (game.get(square) && game.get(square).color == game.turn())
return square;
}
}
return null;
}
function reachableSquares() {
let square = currentSquare(game);
let reachable = {};
let stack = [[square, game.fen()]];
while (stack.length != 0) {
let [square, fen] = stack.pop();
if (reachable[square]) continue;
reachable[square] = [];
successors(Chess(fen)).forEach((entry) => {
let [previous, nextSquare, fen] = entry;
reachable[square].push(nextSquare);
if (!reachable[nextSquare])
stack.push([nextSquare, fen]);
});
}
return reachable;
}
function graphDistance(graph, source, dest) {
let queue = [[source, 0]];
let i = 0;
let seen = {};
seen[source] = true;
while (i < queue.length) {
let [node, dist] = queue[i++];
if (node == dest)
return dist;
graph[node].forEach(succ => {
if (!seen[succ]) {
queue.push([succ, dist + 1]);
seen[succ] = true;
}
});
}
}
function targetSquares() {
let reachable = reachableSquares();
let targets = [];
for (let x = 0; x < 8; x++) {
for (let y = 0; y < 8; y++) {
let file = ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h'][8 - y - 1];
let rank = ((8 - x - 1) + 1).toString();
let square = file + rank;
if (reachable[square])
targets.push(square);
}
}
return targets;
}
function distances() {
let reachable = reachableSquares();
let targets = targetSquares();
let current = currentSquare(game);
let out = [];
targets.forEach(target => {
out.push(graphDistance(reachable, current, target));
current = target;
});
return out;
}
function createSplits() {
let numMoves = distances();
let table = document.querySelector("#splits tbody")
let summaryRow = document.querySelector("#splits-total")
let bestRow = document.querySelector("#splits-best")
let moveSum = 0;
let sumOfBest = 0;
let bestRun = bestSplits["bestRun"] || {};
let bestTime = 0;
targetSquares().forEach((square, i) => {
let row = document.createElement("tr");
row.id = "splits-" + square;
let squareNode = document.createElement("td");
squareNode.classList.add("splits-square");
squareNode.appendChild(document.createTextNode(square));
row.appendChild(squareNode);
let timeNode = document.createElement("td");
timeNode.classList.add("splits-time");
row.appendChild(timeNode);
let deltaNode = document.createElement("td");
deltaNode.classList.add("splits-delta");
row.appendChild(deltaNode);
let bestTimeNode = document.createElement("td");
bestTimeNode.classList.add("splits-best-time");
row.appendChild(bestTimeNode);
if (square in bestRun) {
bestTime += bestRun[square];
bestTimeNode.appendChild(document.createTextNode(formatTime(bestTime)));
}
if (square in bestSplits) {
if (sumOfBest != null) sumOfBest += bestSplits[square];
}
else
sumOfBest = null;
let numMovesNode = document.createElement("td");
numMovesNode.classList.add("splits-num-moves");
row.appendChild(numMovesNode);
let minMovesNode = document.createElement("td");
minMovesNode.appendChild(document.createTextNode(numMoves[i]));
minMovesNode.classList.add("splits-min-moves");
row.appendChild(minMovesNode);
moveSum += numMoves[i];
table.insertBefore(row, summaryRow);
});
document.querySelector("#splits-total .splits-min-moves").innerHTML = moveSum;
if ("total" in bestSplits) {
summaryRow.querySelector(".splits-best-time").innerHTML =
formatTime(bestSplits["total"]);
}
if (sumOfBest != null) {
bestRow.querySelector(".splits-best-time").innerHTML =
formatTime(sumOfBest);
}
}
function clearSplits() {
const table = document.querySelector("#splits tbody")
targets.forEach(square => {
table.removeChild(document.querySelector("#splits-" + square));
});
let summaryRow = document.querySelector("#splits-total");
let bestRow = document.querySelector("#splits-best");
summaryRow.querySelector(".splits-time").innerHTML = "";
summaryRow.querySelector(".splits-time").classList.remove("splits-pb");
summaryRow.querySelector(".splits-delta").innerHTML = "";
summaryRow.querySelector(".splits-best-time").innerHTML = "";
summaryRow.querySelector(".splits-num-moves").innerHTML = "";
summaryRow.querySelector(".splits-min-moves").innerHTML = "";
bestRow.querySelector(".splits-time").innerHTML = "";
bestRow.querySelector(".splits-delta").innerHTML = "";
bestRow.querySelector(".splits-best-time").innerHTML = "";
bestRow.querySelector(".splits-num-moves").innerHTML = "";
bestRow.querySelector(".splits-min-moves").innerHTML = "";
}
function bestRunTime(target) {
let sum = 0;
for (let i = 0; i < targets.length; i++) {
if (!(targets[i] in bestSplits["bestRun"]))
return null;
sum += bestSplits["bestRun"][targets[i]];
if (targets[i] == target)
return sum;
}
}
function updateSplits(target) {
const targetRow = document.querySelector("#splits-" + target);
const summaryRow = document.querySelector("#splits-total");
if (startTime) {
const time = performance.now() - startTime;
const segmentTime = performance.now() - (lastSplitTime || startTime);
summaryRow.querySelector(".splits-time").innerHTML = formatTime(time);
targetRow.querySelector(".splits-time").innerHTML = formatTime(time);
if (bestSplits["bestRun"] && target in bestSplits["bestRun"]) {
let bestRunSegment = bestSplits["bestRun"][target];
let previousTime = bestRunTime(target);
const deltaNode = targetRow.querySelector(".splits-delta");
deltaNode.classList.remove("splits-ahead");
deltaNode.classList.remove("splits-behind");
deltaNode.classList.remove("splits-gain");
deltaNode.classList.remove("splits-loss");
deltaNode.classList.remove("splits-glod");
if (time <= previousTime)
deltaNode.classList.add("splits-ahead");
if (time > previousTime)
deltaNode.classList.add("splits-behind");
if (segmentTime > bestRunSegment)
deltaNode.classList.add("splits-loss");
if (segmentTime <= bestRunSegment)
deltaNode.classList.add("splits-gain");
const timeStr = formatTime(time - previousTime);
deltaNode.innerHTML = (time > previousTime ? "+" : "") + timeStr;
}
}
else {
summaryRow.querySelector(".splits-time").innerHTML = formatTime(0);
targetRow.querySelector(".splits-time").innerHTML = formatTime(0);
}
targetRow.querySelector(".splits-num-moves").innerHTML = numMovesMadeSinceSplit;
summaryRow.querySelector(".splits-num-moves").innerHTML = totalMovesMade;
}
function storeSplit(target, time) {
splits[target] = time;
if (!bestSplits[target] || time <= bestSplits[target]) {
bestSplits[target] = time;
if (target == "total") {
bestSplits["bestRun"] = splits;
document.querySelector("#splits-total .splits-time").classList.add("splits-pb");
}
else {
document.querySelector("#splits-" + target + " .splits-time").classList.add("splits-glod");
}
}
localStorage.setItem(startPosition, JSON.stringify(bestSplits));
let sumOfBest = 0;
targets.forEach(square => {
if (square in bestSplits && sumOfBest != null) sumOfBest += bestSplits[square];
else sumOfBest = null;
});
if (sumOfBest != null) {
document.querySelector("#splits-best .splits-best-time").innerHTML =
formatTime(sumOfBest);
}
}
function updateTarget() {
while (targetId < targets.length && game.get(targets[targetId])) {
updateSplits(targets[targetId]);
numMovesMadeSinceSplit = 0;
const endTime = performance.now();
const splitStart = lastSplitTime || startTime || endTime;
storeSplit(targets[targetId], endTime - splitStart);
lastSplitTime = endTime;
targetId += 1;
if (targetId == targets.length)
storeSplit("total", endTime - startTime);
}
let shapes = [];
if (showInaccessible)
shapes = coveredSquaresHighlights();
if (targetId < targets.length) {
shapes.push({orig: targets[targetId], brush: "green"});
}
board.set({drawable: {shapes: shapes}});
}
function coveredSquares() {
let reachable = reachableSquares();
let squares = [];
for (let x = 0; x < 8; x++) {
for (let y = 0; y < 8; y++) {
let file = ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h'][8 - y - 1];
let rank = ((8 - x - 1) + 1).toString();
let square = file + rank;
if (!reachable[square])
squares.push(square);
}
}
return squares;
}
function coveredSquaresHighlights() {
return coveredSquares().map(s => { return {orig: s, brush: "red"} });
}
function onMove(source, target, meta) {
totalMovesMade++;
numMovesMadeSinceSplit++;
game.put(game.get(source), target);
game.remove(source);
updateLegalMoves();
if (targetId < targets.length) {
updateSplits(targets[targetId]);
if (!startTime) updateTimer(performance.now());
}
updateTarget();
board.selectSquare(target, true);
}
function resizeBoard() {
const viewportWidth =
Math.max(document.documentElement.clientWidth || 0, window.innerWidth || 0);
const sideBySideWidth = viewportWidth -
(document.querySelector("#splits thead").clientWidth || 0);
const viewportHeight = Math.max(document.documentElement.clientHeight|| 0,
window.innerHeight || 0) - 100;
const usedWidth = sideBySideWidth > 320 ? sideBySideWidth : viewportWidth;
const size = Math.min(usedWidth, viewportHeight).toString() + "px";
document.querySelector("#board").style.width = size;
document.querySelector("#board").style.height = size;
if (sideBySideWidth > 300) {
document.querySelector("#board-container").classList.add("board-container-side-by-side");
document.querySelector("#board-column").style.flex = size;
}
else {
document.querySelector("#board-container").classList.remove("board-container-side-by-side");
document.querySelector("#board-column").style.flex = null;
}
}
function soundOnMove(source, target, capture) {
new Audio('chess-move.ogg').play();
}
function onSelect(square) {
if (!square || !game.get(square)) {
square = currentSquare(game);
board.selectSquare(square);
}
}
let board = null;
let startPosition = "7n/8/8/3Q4/8/8/8/8 b - - 0 1";
let game = Chess(startPosition);
let startTime = null;
let showInaccessible = false;
let timerUpdateRequest = null;
let totalMovesMade = 0;
let numMovesMadeSinceSplit = 0;
let lastSplitTime = null;
let targets = targetSquares();
let targetId = 0;
let splits = {};
let bestSplits = {};
if (localStorage.getItem(startPosition))
bestSplits = JSON.parse(localStorage.getItem(startPosition));
function setupPosition(fen) {
if (timerUpdateRequest)
cancelAnimationFrame(timerUpdateRequest);
clearSplits();
startPosition = fen;
game = Chess(startPosition);
startTime = null;
totalMovesMade = 0;
numMovesMadeSinceSplit = 0;
lastSplitTime = null;
targets = targetSquares()
targetId = 0;
splits = {};
bestSplits = {};
if (localStorage.getItem(startPosition))
bestSplits = JSON.parse(localStorage.getItem(startPosition));
board.set(defaultConfig());
createSplits();
updateLegalMoves(board);
updateTarget();
resizeBoard();
}
function defaultConfig() {
const isWhite = game.turn() == game.WHITE;
return {
fen: game.fen(),
orientation: "white",
movable: {
free: false,
color: isWhite ? "white" : "black",
showDests: false,
events: {after: onMove},
},
selected: currentSquare(game),
drawable: {enabled: false},
animation: {enabled: false},
events: {move: soundOnMove, select: onSelect},
coordinates: false
};
}
document.addEventListener("DOMContentLoaded", function() {
board = Chessground.Chessground(document.querySelector("#board"), defaultConfig());
createSplits();
updateLegalMoves(board);
updateTarget();
resizeBoard();
window.addEventListener("resize", resizeBoard);
const showInaccessibleBox = document.querySelector("#show-inaccessible");
showInaccessibleBox.onchange = () => {
showInaccessible = showInaccessibleBox.checked;
updateTarget();
};
const fenSelector = document.querySelector("#fen-selector");
const fenInput = document.querySelector("#fen");
fenSelector.onchange = () => {
fenInput.value = fenSelector.value;
};
document.querySelector("#load-fen").onclick = () => {
setupPosition(fenInput.value);
};
});
</script>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment