2024-01-15 16:03:06 -08:00
|
|
|
// Eryn Wells <eryn@erynwells.me>
|
|
|
|
|
|
2024-03-08 08:04:21 -08:00
|
|
|
//! Defines routines for computing sight of a piece. Sight is the set of squares
|
|
|
|
|
//! that a piece can see. In other words, it's the set of squares attacked or
|
|
|
|
|
//! controled by a piece.
|
|
|
|
|
|
2024-01-24 09:18:12 -08:00
|
|
|
use chessfriend_bitboard::BitBoard;
|
2024-07-13 12:08:20 -07:00
|
|
|
use chessfriend_board::Board;
|
2024-01-28 09:46:38 -08:00
|
|
|
use chessfriend_core::{Color, Direction, PlacedPiece, Shape, Square};
|
2024-01-15 16:03:06 -08:00
|
|
|
|
2024-02-01 08:43:03 -08:00
|
|
|
macro_rules! ray_in_direction {
|
2025-05-08 17:37:51 -07:00
|
|
|
($square:expr, $blockers:expr, $direction:ident, $first_occupied_square:tt) => {{
|
2024-02-01 08:43:03 -08:00
|
|
|
let ray = BitBoard::ray($square, Direction::$direction);
|
2025-05-08 17:37:51 -07:00
|
|
|
let ray_blockers = ray & $blockers;
|
|
|
|
|
if let Some(first_occupied_square) = ray_blockers.$first_occupied_square() {
|
2024-02-01 08:43:03 -08:00
|
|
|
let remainder = BitBoard::ray(first_occupied_square, Direction::$direction);
|
|
|
|
|
let attack_ray = ray & !remainder;
|
|
|
|
|
attack_ray
|
|
|
|
|
} else {
|
2025-05-08 17:37:51 -07:00
|
|
|
ray
|
2024-02-01 08:43:03 -08:00
|
|
|
}
|
|
|
|
|
}};
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-08 08:04:21 -08:00
|
|
|
/// Compute sight of a white pawn.
|
|
|
|
|
fn _white_pawn_sight(
|
2024-07-13 12:08:20 -07:00
|
|
|
pawn: BitBoard,
|
|
|
|
|
occupancy: BitBoard,
|
|
|
|
|
blockers: BitBoard,
|
|
|
|
|
en_passant_square: BitBoard,
|
2024-03-08 08:04:21 -08:00
|
|
|
) -> BitBoard {
|
|
|
|
|
let possible_squares = !occupancy | blockers | en_passant_square;
|
|
|
|
|
let pawn = pawn.shift_north_west_one() | pawn.shift_north_east_one();
|
|
|
|
|
pawn & possible_squares
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn _black_pawn_sight(
|
2024-07-13 12:08:20 -07:00
|
|
|
pawn: BitBoard,
|
|
|
|
|
occupancy: BitBoard,
|
|
|
|
|
blockers: BitBoard,
|
|
|
|
|
en_passant_square: BitBoard,
|
2024-03-08 08:04:21 -08:00
|
|
|
) -> BitBoard {
|
|
|
|
|
let possible_squares = !occupancy | blockers | en_passant_square;
|
|
|
|
|
let pawn = pawn.shift_south_west_one() | pawn.shift_south_east_one();
|
|
|
|
|
pawn & possible_squares
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-13 12:08:20 -07:00
|
|
|
fn _knight_sight(knight_square: Square, blockers: BitBoard) -> BitBoard {
|
2024-03-08 08:04:21 -08:00
|
|
|
BitBoard::knight_moves(knight_square) & !blockers
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-13 12:08:20 -07:00
|
|
|
fn _bishop_sight(bishop_square: Square, occupancy: BitBoard) -> BitBoard {
|
2024-03-08 08:04:21 -08:00
|
|
|
#[rustfmt::skip]
|
2025-05-08 17:37:51 -07:00
|
|
|
let sight = ray_in_direction!(bishop_square, occupancy, NorthEast, first_occupied_square_trailing)
|
|
|
|
|
| ray_in_direction!(bishop_square, occupancy, NorthWest, first_occupied_square_trailing)
|
|
|
|
|
| ray_in_direction!(bishop_square, occupancy, SouthEast, first_occupied_square_leading)
|
|
|
|
|
| ray_in_direction!(bishop_square, occupancy, SouthWest, first_occupied_square_leading);
|
2024-03-08 08:04:21 -08:00
|
|
|
|
|
|
|
|
sight
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-13 12:08:20 -07:00
|
|
|
fn _rook_sight(rook_square: Square, occupancy: BitBoard) -> BitBoard {
|
2024-03-08 08:04:21 -08:00
|
|
|
#[rustfmt::skip]
|
2025-05-08 17:37:51 -07:00
|
|
|
let sight = ray_in_direction!(rook_square, occupancy, North, first_occupied_square_trailing)
|
|
|
|
|
| ray_in_direction!(rook_square, occupancy, East, first_occupied_square_trailing)
|
|
|
|
|
| ray_in_direction!(rook_square, occupancy, South, first_occupied_square_leading)
|
|
|
|
|
| ray_in_direction!(rook_square, occupancy, West, first_occupied_square_leading);
|
2024-03-08 08:04:21 -08:00
|
|
|
|
|
|
|
|
sight
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-13 12:08:20 -07:00
|
|
|
fn _queen_sight(queen_square: Square, occupancy: BitBoard) -> BitBoard {
|
2024-03-08 08:04:21 -08:00
|
|
|
#[rustfmt::skip]
|
2025-05-08 17:37:51 -07:00
|
|
|
let sight = ray_in_direction!(queen_square, occupancy, NorthWest, first_occupied_square_trailing)
|
|
|
|
|
| ray_in_direction!(queen_square, occupancy, North, first_occupied_square_trailing)
|
|
|
|
|
| ray_in_direction!(queen_square, occupancy, NorthEast, first_occupied_square_trailing)
|
|
|
|
|
| ray_in_direction!(queen_square, occupancy, East, first_occupied_square_trailing)
|
|
|
|
|
| ray_in_direction!(queen_square, occupancy, SouthEast, first_occupied_square_leading)
|
|
|
|
|
| ray_in_direction!(queen_square, occupancy, South, first_occupied_square_leading)
|
|
|
|
|
| ray_in_direction!(queen_square, occupancy, SouthWest, first_occupied_square_leading)
|
|
|
|
|
| ray_in_direction!(queen_square, occupancy, West, first_occupied_square_leading);
|
2024-03-08 08:04:21 -08:00
|
|
|
|
|
|
|
|
sight
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-13 12:08:20 -07:00
|
|
|
fn _king_sight(king_square: Square, blockers: BitBoard) -> BitBoard {
|
2024-03-08 08:04:21 -08:00
|
|
|
BitBoard::king_moves(king_square) & !blockers
|
|
|
|
|
}
|
|
|
|
|
pub(crate) trait BishopSightExt {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn bishop_sight(&self, occupancy: BitBoard) -> BitBoard;
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub(crate) trait KingSightExt {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn king_sight(&self, board: &Board) -> BitBoard;
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub(crate) trait KnightSightExt {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn knight_sight(&self, board: &Board) -> BitBoard;
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub(crate) trait PawnSightExt {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn pawn_sight(&self, board: &Board, en_passant_square: Option<Square>) -> BitBoard;
|
|
|
|
|
fn white_pawn_sight(&self, board: &Board, en_passant_square: Option<Square>) -> BitBoard;
|
|
|
|
|
fn black_pawn_sight(&self, board: &Board, en_passant_square: Option<Square>) -> BitBoard;
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
|
|
|
|
pub(crate) trait QueenSightExt {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn queen_sight(&self, occupancy: BitBoard) -> BitBoard;
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
2024-01-28 09:46:38 -08:00
|
|
|
|
2024-03-08 08:04:21 -08:00
|
|
|
pub(crate) trait RookSightExt {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn rook_sight(&self, occupancy: BitBoard) -> BitBoard;
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub(crate) trait SliderSightExt: BishopSightExt + QueenSightExt + RookSightExt {}
|
|
|
|
|
|
|
|
|
|
pub(crate) trait SightExt {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn sight(&self, board: &Board, en_passant_square: Option<Square>) -> BitBoard;
|
2024-01-15 16:03:06 -08:00
|
|
|
}
|
|
|
|
|
|
2024-02-01 08:42:19 -08:00
|
|
|
pub(crate) trait SliderRayToSquareExt {
|
2024-02-02 07:29:43 -08:00
|
|
|
fn ray_to_square(&self, origin: Square, target: Square) -> Option<BitBoard>;
|
2024-02-01 08:42:19 -08:00
|
|
|
}
|
|
|
|
|
|
2024-01-24 17:08:27 -08:00
|
|
|
impl SightExt for PlacedPiece {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn sight(&self, board: &Board, en_passant_square: Option<Square>) -> BitBoard {
|
2024-01-15 16:03:06 -08:00
|
|
|
match self.shape() {
|
|
|
|
|
Shape::Pawn => match self.color() {
|
2024-07-13 12:08:20 -07:00
|
|
|
Color::White => self.white_pawn_sight(board, en_passant_square),
|
|
|
|
|
Color::Black => self.black_pawn_sight(board, en_passant_square),
|
2024-01-15 16:03:06 -08:00
|
|
|
},
|
2024-07-13 12:08:20 -07:00
|
|
|
Shape::Knight => self.knight_sight(board),
|
2025-05-08 17:37:51 -07:00
|
|
|
Shape::Bishop => self.bishop_sight(board.pieces.all_pieces()),
|
|
|
|
|
Shape::Rook => self.rook_sight(board.pieces.all_pieces()),
|
|
|
|
|
Shape::Queen => self.queen_sight(board.pieces.all_pieces()),
|
2024-07-13 12:08:20 -07:00
|
|
|
Shape::King => self.king_sight(board),
|
2024-01-15 16:03:06 -08:00
|
|
|
}
|
|
|
|
|
}
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl KingSightExt for PlacedPiece {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn king_sight(&self, board: &Board) -> BitBoard {
|
2025-05-08 17:37:51 -07:00
|
|
|
_king_sight(self.square, board.pieces.all_pieces_of_color(self.color))
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl KnightSightExt for PlacedPiece {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn knight_sight(&self, board: &Board) -> BitBoard {
|
2025-05-08 17:37:51 -07:00
|
|
|
_knight_sight(self.square, board.pieces.all_pieces_of_color(self.color))
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl PawnSightExt for PlacedPiece {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn pawn_sight(&self, board: &Board, en_passant_square: Option<Square>) -> BitBoard {
|
2025-05-08 17:37:51 -07:00
|
|
|
match self.color {
|
2024-07-13 12:08:20 -07:00
|
|
|
Color::White => self.white_pawn_sight(board, en_passant_square),
|
|
|
|
|
Color::Black => self.black_pawn_sight(board, en_passant_square),
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
|
|
|
|
}
|
2024-01-15 16:03:06 -08:00
|
|
|
|
2024-07-13 12:08:20 -07:00
|
|
|
fn white_pawn_sight(&self, board: &Board, en_passant_square: Option<Square>) -> BitBoard {
|
2025-05-08 17:37:51 -07:00
|
|
|
let opponent = self.color.other();
|
2024-03-08 08:04:21 -08:00
|
|
|
_white_pawn_sight(
|
2025-05-08 17:37:51 -07:00
|
|
|
self.square.into(),
|
|
|
|
|
board.pieces.all_pieces(),
|
|
|
|
|
board.pieces.all_pieces_of_color(opponent),
|
2024-07-13 12:08:20 -07:00
|
|
|
en_passant_square.into(),
|
2024-03-08 08:04:21 -08:00
|
|
|
)
|
2024-01-15 16:03:06 -08:00
|
|
|
}
|
|
|
|
|
|
2024-07-13 12:08:20 -07:00
|
|
|
fn black_pawn_sight(&self, board: &Board, en_passant_square: Option<Square>) -> BitBoard {
|
2025-05-08 17:37:51 -07:00
|
|
|
let opponent = self.piece.color.other();
|
2024-03-08 08:04:21 -08:00
|
|
|
_black_pawn_sight(
|
2025-05-08 17:37:51 -07:00
|
|
|
self.square.into(),
|
|
|
|
|
board.pieces.all_pieces(),
|
|
|
|
|
board.pieces.all_pieces_of_color(opponent),
|
2024-07-13 12:08:20 -07:00
|
|
|
en_passant_square.into(),
|
2024-03-08 08:04:21 -08:00
|
|
|
)
|
2024-01-15 16:03:06 -08:00
|
|
|
}
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
2024-01-15 16:03:06 -08:00
|
|
|
|
2024-03-08 08:04:21 -08:00
|
|
|
impl BishopSightExt for PlacedPiece {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn bishop_sight(&self, occupancy: BitBoard) -> BitBoard {
|
2025-05-08 17:37:51 -07:00
|
|
|
_bishop_sight(self.square, occupancy)
|
2024-01-15 16:03:06 -08:00
|
|
|
}
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
2024-01-15 16:03:06 -08:00
|
|
|
|
2024-03-08 08:04:21 -08:00
|
|
|
impl RookSightExt for PlacedPiece {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn rook_sight(&self, occupancy: BitBoard) -> BitBoard {
|
2025-05-08 17:37:51 -07:00
|
|
|
_rook_sight(self.square, occupancy)
|
2024-01-15 16:03:06 -08:00
|
|
|
}
|
2024-03-08 08:04:21 -08:00
|
|
|
}
|
2024-01-15 16:03:06 -08:00
|
|
|
|
2024-03-08 08:04:21 -08:00
|
|
|
impl QueenSightExt for PlacedPiece {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn queen_sight(&self, occupancy: BitBoard) -> BitBoard {
|
2025-05-08 17:37:51 -07:00
|
|
|
_queen_sight(self.square, occupancy)
|
2024-01-15 16:03:06 -08:00
|
|
|
}
|
|
|
|
|
}
|
2024-01-17 08:44:27 -08:00
|
|
|
|
2024-03-08 08:04:21 -08:00
|
|
|
impl SliderSightExt for PlacedPiece {}
|
|
|
|
|
|
2024-02-02 07:29:43 -08:00
|
|
|
impl SliderRayToSquareExt for Shape {
|
|
|
|
|
fn ray_to_square(&self, origin: Square, target: Square) -> Option<BitBoard> {
|
2024-02-01 08:42:19 -08:00
|
|
|
macro_rules! ray {
|
|
|
|
|
($square:expr, $direction:ident) => {
|
|
|
|
|
(
|
|
|
|
|
BitBoard::ray($square, Direction::$direction),
|
|
|
|
|
Direction::$direction,
|
|
|
|
|
)
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let target_bitboard: BitBoard = target.into();
|
|
|
|
|
|
2024-02-02 07:29:43 -08:00
|
|
|
let ray_and_direction = match self {
|
2024-02-01 08:42:19 -08:00
|
|
|
Shape::Bishop => [
|
2024-02-02 07:29:43 -08:00
|
|
|
ray!(origin, NorthEast),
|
|
|
|
|
ray!(origin, SouthEast),
|
|
|
|
|
ray!(origin, SouthWest),
|
|
|
|
|
ray!(origin, NorthWest),
|
2024-02-01 08:42:19 -08:00
|
|
|
]
|
|
|
|
|
.into_iter()
|
2025-05-08 17:37:51 -07:00
|
|
|
.find(|(ray, _)| !(target_bitboard & ray).is_empty()),
|
2024-02-01 08:42:19 -08:00
|
|
|
Shape::Rook => [
|
2024-02-02 07:29:43 -08:00
|
|
|
ray!(origin, North),
|
|
|
|
|
ray!(origin, East),
|
|
|
|
|
ray!(origin, South),
|
|
|
|
|
ray!(origin, West),
|
2024-02-01 08:42:19 -08:00
|
|
|
]
|
|
|
|
|
.into_iter()
|
2025-05-08 17:37:51 -07:00
|
|
|
.find(|(ray, _)| !(target_bitboard & ray).is_empty()),
|
2024-02-01 08:42:19 -08:00
|
|
|
Shape::Queen => [
|
2024-02-02 07:29:43 -08:00
|
|
|
ray!(origin, North),
|
|
|
|
|
ray!(origin, NorthEast),
|
|
|
|
|
ray!(origin, East),
|
|
|
|
|
ray!(origin, SouthEast),
|
|
|
|
|
ray!(origin, South),
|
|
|
|
|
ray!(origin, SouthWest),
|
|
|
|
|
ray!(origin, West),
|
|
|
|
|
ray!(origin, NorthWest),
|
2024-02-01 08:42:19 -08:00
|
|
|
]
|
|
|
|
|
.into_iter()
|
2025-05-08 17:37:51 -07:00
|
|
|
.find(|(ray, _)| !(target_bitboard & ray).is_empty()),
|
2024-02-01 08:42:19 -08:00
|
|
|
_ => None,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if let Some((ray, direction)) = ray_and_direction {
|
2024-02-02 07:29:43 -08:00
|
|
|
let remainder = BitBoard::ray(target, direction);
|
|
|
|
|
return Some(ray & !remainder);
|
2024-02-01 08:42:19 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
None
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-08 08:04:21 -08:00
|
|
|
impl BishopSightExt for Square {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn bishop_sight(&self, occupancy: BitBoard) -> BitBoard {
|
2024-03-08 08:04:21 -08:00
|
|
|
_bishop_sight(*self, occupancy)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl QueenSightExt for Square {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn queen_sight(&self, occupancy: BitBoard) -> BitBoard {
|
2024-03-08 08:04:21 -08:00
|
|
|
_queen_sight(*self, occupancy)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl RookSightExt for Square {
|
2024-07-13 12:08:20 -07:00
|
|
|
fn rook_sight(&self, occupancy: BitBoard) -> BitBoard {
|
2024-03-08 08:04:21 -08:00
|
|
|
_rook_sight(*self, occupancy)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-17 08:44:27 -08:00
|
|
|
#[cfg(test)]
|
|
|
|
|
mod tests {
|
2024-02-01 08:42:19 -08:00
|
|
|
use super::*;
|
|
|
|
|
use chessfriend_bitboard::bitboard;
|
|
|
|
|
use chessfriend_core::{piece, Square};
|
|
|
|
|
|
2024-01-17 08:44:27 -08:00
|
|
|
macro_rules! sight_test {
|
|
|
|
|
($test_name:ident, $position:expr, $piece:expr, $bitboard:expr) => {
|
|
|
|
|
#[test]
|
|
|
|
|
fn $test_name() {
|
|
|
|
|
let pos = $position;
|
2024-01-28 00:25:53 -08:00
|
|
|
let piece = $piece;
|
|
|
|
|
let sight = pos.sight_of_piece(&piece);
|
2024-01-17 08:44:27 -08:00
|
|
|
|
|
|
|
|
assert_eq!(sight, $bitboard);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
($test_name:ident, $piece:expr, $bitboard:expr) => {
|
|
|
|
|
sight_test! {$test_name, $crate::Position::empty(), $piece, $bitboard}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-01 08:42:19 -08:00
|
|
|
#[test]
|
|
|
|
|
fn pawns_and_knights_cannot_make_rays() {
|
2024-02-02 07:29:43 -08:00
|
|
|
assert_eq!(Shape::Pawn.ray_to_square(Square::F7, Square::E8), None);
|
|
|
|
|
assert_eq!(Shape::Knight.ray_to_square(Square::F6, Square::E8), None);
|
2024-02-01 08:42:19 -08:00
|
|
|
}
|
|
|
|
|
|
2024-01-17 08:44:27 -08:00
|
|
|
mod pawn {
|
2024-01-28 00:25:53 -08:00
|
|
|
use crate::test_position;
|
2024-01-24 09:18:12 -08:00
|
|
|
use chessfriend_bitboard::{bitboard, BitBoard};
|
2025-05-08 17:37:51 -07:00
|
|
|
use chessfriend_core::piece;
|
2024-01-17 08:44:27 -08:00
|
|
|
|
2025-05-08 17:37:51 -07:00
|
|
|
sight_test!(e4_pawn, piece!(White Pawn on E4), bitboard![D5 F5]);
|
2024-01-17 08:44:27 -08:00
|
|
|
|
|
|
|
|
sight_test!(
|
|
|
|
|
e4_pawn_one_blocker,
|
2024-01-21 13:06:44 -08:00
|
|
|
test_position![
|
2024-01-17 08:44:27 -08:00
|
|
|
White Bishop on D5,
|
|
|
|
|
White Pawn on E4,
|
|
|
|
|
],
|
|
|
|
|
piece!(White Pawn on E4),
|
|
|
|
|
bitboard!(F5)
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn e4_pawn_two_blocker() {
|
2024-01-21 13:06:44 -08:00
|
|
|
let pos = test_position!(
|
2024-01-17 08:44:27 -08:00
|
|
|
White Bishop on D5,
|
|
|
|
|
White Queen on F5,
|
|
|
|
|
White Pawn on E4,
|
|
|
|
|
);
|
2024-01-21 13:06:44 -08:00
|
|
|
|
2024-01-28 00:25:53 -08:00
|
|
|
let piece = piece!(White Pawn on E4);
|
|
|
|
|
let sight = pos.sight_of_piece(&piece);
|
2024-01-17 08:44:27 -08:00
|
|
|
|
|
|
|
|
assert_eq!(sight, BitBoard::empty());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn e4_pawn_capturable() {
|
2024-01-21 13:06:44 -08:00
|
|
|
let pos = test_position!(
|
2024-01-17 08:44:27 -08:00
|
|
|
Black Bishop on D5,
|
|
|
|
|
White Queen on F5,
|
|
|
|
|
White Pawn on E4,
|
|
|
|
|
);
|
2024-01-21 13:06:44 -08:00
|
|
|
|
2024-01-28 00:25:53 -08:00
|
|
|
let piece = piece!(White Pawn on E4);
|
|
|
|
|
let sight = pos.sight_of_piece(&piece);
|
2024-01-17 08:44:27 -08:00
|
|
|
|
|
|
|
|
assert_eq!(sight, bitboard!(D5));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn e5_en_passant() {
|
2025-05-08 17:37:51 -07:00
|
|
|
let pos = test_position!(White, [
|
2024-01-17 08:44:27 -08:00
|
|
|
White Pawn on E5,
|
|
|
|
|
Black Pawn on D5,
|
2025-05-08 17:37:51 -07:00
|
|
|
], D6);
|
2024-01-28 00:25:53 -08:00
|
|
|
let piece = piece!(White Pawn on E5);
|
|
|
|
|
let sight = pos.sight_of_piece(&piece);
|
2024-01-17 08:44:27 -08:00
|
|
|
|
2025-05-08 17:37:51 -07:00
|
|
|
assert_eq!(sight, bitboard!(D6 F6));
|
2024-01-17 08:44:27 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[macro_use]
|
|
|
|
|
mod knight {
|
2024-01-24 09:18:12 -08:00
|
|
|
use chessfriend_bitboard::bitboard;
|
2024-01-24 17:08:27 -08:00
|
|
|
use chessfriend_core::piece;
|
2024-01-17 08:44:27 -08:00
|
|
|
|
|
|
|
|
sight_test!(
|
|
|
|
|
f6_knight,
|
|
|
|
|
piece!(Black Knight on F6),
|
2025-05-08 17:37:51 -07:00
|
|
|
bitboard![H7 G8 E8 D7 D5 E4 G4 H5]
|
2024-01-17 08:44:27 -08:00
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mod bishop {
|
2024-02-01 08:42:19 -08:00
|
|
|
use super::*;
|
2024-01-17 08:44:27 -08:00
|
|
|
|
|
|
|
|
sight_test!(
|
|
|
|
|
c2_bishop,
|
|
|
|
|
piece!(Black Bishop on C2),
|
2025-05-08 17:37:51 -07:00
|
|
|
bitboard![D1 B3 A4 B1 D3 E4 F5 G6 H7]
|
2024-01-17 08:44:27 -08:00
|
|
|
);
|
2024-02-01 08:42:19 -08:00
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn ray_to_square() {
|
2024-02-02 07:29:43 -08:00
|
|
|
let generated_ray = Shape::Bishop.ray_to_square(Square::C5, Square::E7);
|
2025-05-08 17:37:51 -07:00
|
|
|
let expected_ray = bitboard![D6 E7];
|
2024-02-01 08:42:19 -08:00
|
|
|
assert_eq!(generated_ray, Some(expected_ray));
|
|
|
|
|
}
|
2024-01-17 08:44:27 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mod rook {
|
2024-02-01 08:42:19 -08:00
|
|
|
use super::*;
|
2024-01-28 00:25:53 -08:00
|
|
|
use crate::test_position;
|
2024-01-17 08:44:27 -08:00
|
|
|
|
|
|
|
|
sight_test!(
|
|
|
|
|
g3_rook,
|
|
|
|
|
piece!(White Rook on G3),
|
2025-05-08 17:37:51 -07:00
|
|
|
bitboard![G1 G2 G4 G5 G6 G7 G8 A3 B3 C3 D3 E3 F3 H3]
|
2024-01-17 08:44:27 -08:00
|
|
|
);
|
2024-01-28 00:25:53 -08:00
|
|
|
|
|
|
|
|
sight_test!(
|
|
|
|
|
e4_rook_with_e1_white_king_e7_black_king,
|
|
|
|
|
test_position![
|
|
|
|
|
White Rook on E4,
|
|
|
|
|
White King on E1,
|
|
|
|
|
Black King on E7,
|
|
|
|
|
],
|
|
|
|
|
piece!(White Rook on E4),
|
2025-05-08 17:37:51 -07:00
|
|
|
bitboard![A4 B4 C4 D4 F4 G4 H4 E2 E3 E5 E6 E7 E1]
|
2024-01-28 00:25:53 -08:00
|
|
|
);
|
2024-02-01 08:42:19 -08:00
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn ray_to_square() {
|
2024-02-02 07:29:43 -08:00
|
|
|
let generated_ray = Shape::Rook.ray_to_square(Square::C2, Square::C6);
|
2025-05-08 17:37:51 -07:00
|
|
|
let expected_ray = bitboard![C3 C4 C5 C6];
|
2024-02-01 08:42:19 -08:00
|
|
|
assert_eq!(generated_ray, Some(expected_ray));
|
|
|
|
|
|
2024-02-02 07:29:43 -08:00
|
|
|
let generated_ray = Shape::Rook.ray_to_square(Square::D2, Square::H2);
|
2025-05-08 17:37:51 -07:00
|
|
|
let expected_ray = bitboard![E2 F2 G2 H2];
|
2024-02-01 08:42:19 -08:00
|
|
|
assert_eq!(generated_ray, Some(expected_ray));
|
|
|
|
|
|
2024-02-02 07:29:43 -08:00
|
|
|
let generated_ray = Shape::Rook.ray_to_square(Square::G6, Square::B6);
|
2025-05-08 17:37:51 -07:00
|
|
|
let expected_ray = bitboard![B6 C6 D6 E6 F6];
|
2024-02-01 08:42:19 -08:00
|
|
|
assert_eq!(generated_ray, Some(expected_ray));
|
|
|
|
|
|
2024-02-02 07:29:43 -08:00
|
|
|
let generated_ray = Shape::Rook.ray_to_square(Square::A6, Square::A3);
|
2025-05-08 17:37:51 -07:00
|
|
|
let expected_ray = bitboard![A5 A4 A3];
|
2024-02-01 08:42:19 -08:00
|
|
|
assert_eq!(generated_ray, Some(expected_ray));
|
|
|
|
|
}
|
2024-01-28 00:25:53 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mod king {
|
|
|
|
|
use chessfriend_bitboard::bitboard;
|
|
|
|
|
use chessfriend_core::piece;
|
|
|
|
|
|
2025-05-08 17:37:51 -07:00
|
|
|
sight_test!(e1_king, piece!(White King on E1), bitboard![D1 D2 E2 F2 F1]);
|
2024-01-17 08:44:27 -08:00
|
|
|
}
|
|
|
|
|
}
|