chessbot/chess/board.go

1030 lines
20 KiB
Go

package chess
import (
"fmt"
"image"
"image/draw"
"image/png"
"log"
"os"
"strings"
"golang.org/x/image/font"
"golang.org/x/image/font/basicfont"
"golang.org/x/image/math/fixed"
)
type Board struct {
tiles [8][8]*Piece
turn Color
moves []string
}
func NewBoard() *Board {
board := &Board{turn: Light}
board.mustSetPiece(Rook, Light, "a1")
board.mustSetPiece(Knight, Light, "b1")
board.mustSetPiece(Bishop, Light, "c1")
board.mustSetPiece(Queen, Light, "d1")
board.mustSetPiece(King, Light, "e1")
board.mustSetPiece(Bishop, Light, "f1")
board.mustSetPiece(Knight, Light, "g1")
board.mustSetPiece(Rook, Light, "h1")
board.mustSetPiece(Pawn, Light, "a2")
board.mustSetPiece(Pawn, Light, "b2")
board.mustSetPiece(Pawn, Light, "c2")
board.mustSetPiece(Pawn, Light, "d2")
board.mustSetPiece(Pawn, Light, "e2")
board.mustSetPiece(Pawn, Light, "f2")
board.mustSetPiece(Pawn, Light, "g2")
board.mustSetPiece(Pawn, Light, "h2")
board.mustSetPiece(Rook, Dark, "a8")
board.mustSetPiece(Knight, Dark, "b8")
board.mustSetPiece(Bishop, Dark, "c8")
board.mustSetPiece(Queen, Dark, "d8")
board.mustSetPiece(King, Dark, "e8")
board.mustSetPiece(Bishop, Dark, "f8")
board.mustSetPiece(Knight, Dark, "g8")
board.mustSetPiece(Rook, Dark, "h8")
board.mustSetPiece(Pawn, Dark, "a7")
board.mustSetPiece(Pawn, Dark, "b7")
board.mustSetPiece(Pawn, Dark, "c7")
board.mustSetPiece(Pawn, Dark, "d7")
board.mustSetPiece(Pawn, Dark, "e7")
board.mustSetPiece(Pawn, Dark, "f7")
board.mustSetPiece(Pawn, Dark, "g7")
board.mustSetPiece(Pawn, Dark, "h7")
return board
}
func NewGame(moves string) (*Board, error) {
board := NewBoard()
if err := board.Parse(moves); err != nil {
return nil, err
}
return board, nil
}
func (b *Board) Save(filename string) error {
var (
file *os.File
err error
)
if file, err = os.Create(filename); err != nil {
return err
}
defer file.Close()
return png.Encode(file, b.Image())
}
func (b *Board) Image() *image.RGBA {
var (
img *image.RGBA
piece *Piece
bg *image.Uniform
rect image.Rectangle
p = image.Point{0, 0}
)
img = image.NewRGBA(image.Rect(0, 0, 1024, 1024))
for yi := 0; yi < 8; yi++ {
for xi := 0; xi < 8; xi++ {
x := xi * 128
y := yi * 128
rect = image.Rect(x, y, x+128, y+128)
bg = image.NewUniform(getTileColor(xi, yi))
draw.Draw(img, rect, bg, p, draw.Src)
piece = b.tiles[xi][yi]
if piece != nil {
pieceImg := piece.Image
if b.turn == Dark {
pieceImg = flipImage(pieceImg)
}
draw.Draw(img, rect, pieceImg, p, draw.Over)
}
}
}
if b.turn == Dark {
img = flipImage(img)
}
for yi := 0; yi < 8; yi++ {
for xi := 0; xi < 8; xi++ {
if b.turn == Light {
drawCoordinate(img, xi, yi, false)
}
if b.turn == Dark {
drawCoordinate(img, xi, yi, true)
}
}
}
return img
}
func drawCoordinate(img *image.RGBA, x, y int, flipped bool) {
if x != 7 && y != 7 {
return
}
var column, row string
if y == 7 {
switch x {
case 0:
column = "a"
case 1:
column = "b"
case 2:
column = "c"
case 3:
column = "d"
case 4:
column = "e"
case 5:
column = "f"
case 6:
column = "g"
case 7:
column = "h"
}
}
if x == 7 {
yRow := y
if flipped {
yRow = 7 - y
}
switch yRow {
case 0:
row = "8"
case 1:
row = "7"
case 2:
row = "6"
case 3:
row = "5"
case 4:
row = "4"
case 5:
row = "3"
case 6:
row = "2"
case 7:
row = "1"
}
}
drawString := func(s string, origin fixed.Point26_6) {
color := getTileColor(x, y)
if !flipped && color == Light {
color = Dark
} else if !flipped {
color = Light
}
// TODO: use SN font and make it bold
d := &font.Drawer{
Dst: img,
Src: image.NewUniform(color),
Face: basicfont.Face7x13,
Dot: origin,
}
d.DrawString(s)
}
var origin fixed.Point26_6
if column != "" {
origin = fixed.P(x*128+5, (y+1)*128-5)
drawString(column, origin)
}
if row != "" {
origin = fixed.P((x+1)*128-12, y*128+15)
drawString(row, origin)
}
}
func (b *Board) SetPiece(name PieceName, color Color, position string) error {
var (
piece *Piece
x int
y int
err error
)
if len(position) != 2 {
return fmt.Errorf("invalid position: %s", position)
}
if piece, err = NewPiece(name, color); err != nil {
return err
}
if x, y, err = getXY(position); err != nil {
return err
}
b.tiles[x][y] = piece
return nil
}
func (b *Board) AlgebraicNotation() string {
var text string
for i, m := range b.moves {
if i%2 == 0 {
text += fmt.Sprintf("%d. %s", i/2+1, m)
} else {
text += fmt.Sprintf(" %s ", m)
}
}
return fmt.Sprintf("`%s`", text)
}
func (b *Board) Parse(pgn string) error {
var (
moves = strings.Split(strings.Trim(pgn, " "), " ")
err error
)
for _, move := range moves {
move = strings.Trim(move, " ")
if move == "" {
continue
}
if err = b.Move(move); err != nil {
return err
}
}
return nil
}
func (b *Board) Move(move string) error {
var (
to string
piece string
// if the move is ambiguous, the originating square rank must be given
// see https://en.wikipedia.org/wiki/Algebraic_notation_(chess)#Disambiguating_moves
fromX int
fromY int
collisionPiece *Piece
err error
)
if piece, fromX, fromY, to, err = parseMove(move); err != nil {
return err
}
// TODO: parse ambiguous captures for all pieces
// TODO: parse promotions
// TODO: parse checks e.g. e5+
// TODO: parse checkmates e.g. e5#
// TODO: parse O-O as kingside castle and O-O-O as queenside castle
// TODO: make sure pinned pieces cannot move
// TODO: make sure king is not in check after move
// ( this avoids moving into check and moving a piece that exposes the king to check e.g. pinned pieces )
move_ := func() error {
// collision detection
if collisionPiece, err = b.getCollision(to); err != nil {
return fmt.Errorf("invalid move %s: %v", move, err)
} else if collisionPiece != nil {
return fmt.Errorf("invalid move %s: position %s blocked by %s", move, to, collisionPiece)
}
switch strings.ToLower(piece) {
case "p":
return b.movePawn(to, fromX, fromY)
case "r":
return b.moveRook(to, false, fromX, fromY)
case "b":
return b.moveBishop(to, false)
case "n":
return b.moveKnight(to, fromX, fromY)
case "q":
return b.moveQueen(to)
case "k":
return b.moveKing(to)
default:
return fmt.Errorf("invalid move %s: %v", move, err)
}
}
if err = move_(); err != nil {
return err
}
if b.turn == Light {
b.turn = Dark
} else {
b.turn = Light
}
b.moves = append(b.moves, move)
return nil
}
func parseMove(move string) (string, int, int, string, error) {
var (
piece string
fromX = -1
fromY = -1
to string
from string
)
if strings.Contains(move, "x") {
return parseCaptureMove(move)
}
if len(move) < 2 {
return piece, fromX, fromY, to, fmt.Errorf("invalid move: %s", move)
}
if len(move) == 2 {
// pawn move
piece = "p"
to = move
return piece, fromX, fromY, to, nil
}
if len(move) == 3 {
// unambiguous piece move
piece = move[0:1]
to = move[1:]
return piece, fromX, fromY, to, nil
}
// last two characters are the target position
to = move[len(move)-2:]
// everything before is about origin
from = move[:len(move)-2]
// piece is first character
piece = from[0:1]
// we can ignore piece from here on
from = from[1:]
if len(from) == 1 {
// rank or file given
if from[0] >= 'a' && from[0] <= 'h' {
// file given
return piece, toFile(rune(from[0])), -1, to, nil
}
if from[0] >= '1' && from[0] <= '8' {
// rank given
return piece, -1, toRank(rune(from[0])), to, nil
}
return "", -1, -1, "", fmt.Errorf("invalid move: %s", move)
}
if len(from) == 2 {
// file and rank given
return piece, toFile(rune(from[0])), toRank(rune(from[1])), to, nil
}
return "", -1, -1, "", fmt.Errorf("invalid move: %s", move)
}
func parseCaptureMove(move string) (string, int, int, string, error) {
var (
parts = strings.Split(move, "x")
piece string
fromX = -1
fromY = -1
from = parts[0]
to = parts[1]
)
if len(from) == 1 {
// pawn move with rank given (exd4) or piece move (Nxe5)
if strings.ToLower(from) == from {
piece = "p"
fromX = toFile(rune(from[0]))
return piece, fromX, fromY, to, nil
}
piece = from
return piece, fromX, fromY, to, nil
}
// piece is always first character
piece = from[0:1]
if len(from) == 2 {
// rank or file given
fromX = toFile(rune(from[1]))
fromY = toRank(rune(from[1]))
if fromX == -1 && fromY == -1 {
return "", -1, -1, "", fmt.Errorf("invalid move: %s", move)
}
return piece, fromX, fromY, to, nil
}
if len(from) == 3 {
// both file and rank given
fromX = toFile(rune(from[1]))
fromY = toRank(rune(from[1]))
if fromX == -1 || fromY == -1 {
return "", -1, -1, "", fmt.Errorf("invalid move: %s", move)
}
return piece, fromX, fromY, to, nil
}
return "", -1, -1, "", fmt.Errorf("invalid move: %s", move)
}
func toFile(r rune) int {
if r >= 'a' && r <= 'h' {
return int(r - 'a')
}
return -1
}
func toRank(r rune) int {
if r >= '1' && r <= '8' {
return int('8' - r)
}
return -1
}
func (b *Board) validateMove(search PieceName, fromX int, fromY int) func(*Piece, int, int) bool {
return func(p *Piece, xPrev int, yPrev int) bool {
// does piece originate from given origin?
from := (fromX == -1 || xPrev == fromX) && (fromY == -1 || yPrev == fromY)
// is this the piece we're looking for?
found := p != nil && p.Name == search && p.Color == b.turn
return from && found
}
}
func (b *Board) movePawn(position string, fromX int, fromY int) error {
var (
toX int
toY int
yPrev int
piece *Piece
err error
)
if toX, toY, err = getXY(position); err != nil {
return err
}
if fromX != -1 {
if b.turn == Light {
fromY = toY + 1
} else {
fromY = toY - 1
}
piece = b.getPiece(fromX, fromY)
if piece == nil || piece.Name != Pawn || piece.Color != b.turn {
// not your pawn
return fmt.Errorf("invalid capture move for pawn: %s", position)
}
if fromX != toX-1 && fromX != toX+1 || (b.turn == Light && fromY != toY+1) || (b.turn == Dark && fromY != toY-1) {
// invalid capture move
return fmt.Errorf("invalid capture move for pawn: %s", position)
}
b.tiles[fromX][fromY] = nil
b.tiles[toX][toY] = piece
return nil
}
// TODO: assert move is valid:
// * 2 moves from start position
// * 1 move otherwise
// * diagonal if attacking
// * no collision with other pieces
if b.turn == Light {
yPrev = toY + 1
} else {
yPrev = toY - 1
}
piece = b.tiles[toX][yPrev]
if piece != nil && piece.Name == Pawn && piece.Color == b.turn {
b.tiles[toX][yPrev] = nil
b.tiles[toX][toY] = piece
return nil
}
if b.turn == Light {
yPrev = toY + 2
} else {
yPrev = toY - 2
}
piece = b.tiles[toX][yPrev]
if piece != nil && piece.Name == Pawn && piece.Color == b.turn {
b.tiles[toX][yPrev] = nil
b.tiles[toX][toY] = piece
return nil
}
return fmt.Errorf("no pawn found that can move to %s", position)
}
func (b *Board) moveRook(position string, queen bool, fromX int, fromY int) error {
var (
x int
y int
xPrev int
yPrev int
p *Piece
validPrev []*Square
err error
)
if x, y, err = getXY(position); err != nil {
return err
}
checkRookMove := b.validateMove(Rook, fromX, fromY)
checkQueenMove := b.validateMove(Queen, fromX, fromY)
checkMove := func(p *Piece, xPrev int, yPrev int) bool {
return (!queen && checkRookMove(p, xPrev, yPrev)) || (queen && checkQueenMove(p, xPrev, yPrev))
}
xPrev = x
yPrev = y
for xPrev >= 0 && xPrev < 8 && yPrev >= 0 && yPrev < 8 {
xPrev++
p = b.getPiece(xPrev, yPrev)
if p != nil {
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
}
break
}
}
xPrev = x
yPrev = y
for xPrev >= 0 && xPrev < 8 && yPrev >= 0 && yPrev < 8 {
yPrev--
p = b.getPiece(xPrev, yPrev)
if p != nil {
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
} else {
break
}
}
}
xPrev = x
yPrev = y
for xPrev >= 0 && xPrev < 8 && yPrev >= 0 && yPrev < 8 {
xPrev--
p = b.getPiece(xPrev, yPrev)
if p != nil {
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
} else {
break
}
}
}
xPrev = x
yPrev = y
for xPrev >= 0 && xPrev < 8 && yPrev >= 0 && yPrev < 8 {
yPrev++
p = b.getPiece(xPrev, yPrev)
if p != nil {
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
} else {
break
}
}
}
if len(validPrev) > 1 {
return fmt.Errorf("move ambiguous: %d rooks can move to %s", len(validPrev), position)
}
if len(validPrev) == 1 {
xPrev = validPrev[0].X
yPrev = validPrev[0].Y
p = b.getPiece(xPrev, yPrev)
b.tiles[x][y] = p
b.tiles[xPrev][yPrev] = nil
return nil
}
return fmt.Errorf("no rook found that can move to %s", position)
}
func (b *Board) moveBishop(position string, queen bool) error {
var (
x int
y int
xPrev int
yPrev int
piece *Piece
err error
)
if x, y, err = getXY(position); err != nil {
return err
}
xPrev = x
yPrev = y
for xPrev >= 0 && xPrev < 8 && yPrev >= 0 && yPrev < 8 {
xPrev++
yPrev--
piece = b.getPiece(xPrev, yPrev)
if piece != nil {
if ((!queen && piece.Name == Bishop) || (queen && piece.Name == Queen)) && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
} else {
// direction blocked by other piece
break
}
}
}
xPrev = x
yPrev = y
for xPrev >= 0 && xPrev < 8 && yPrev >= 0 && yPrev < 8 {
xPrev++
yPrev++
piece = b.getPiece(xPrev, yPrev)
if piece != nil {
if ((!queen && piece.Name == Bishop) || (queen && piece.Name == Queen)) && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
} else {
break
}
}
}
xPrev = x
yPrev = y
for xPrev >= 0 && xPrev < 8 && yPrev >= 0 && yPrev < 8 {
xPrev--
yPrev++
piece = b.getPiece(xPrev, yPrev)
if piece != nil {
if ((!queen && piece.Name == Bishop) || (queen && piece.Name == Queen)) && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
} else {
break
}
}
}
xPrev = x
yPrev = y
for xPrev >= 0 && xPrev < 8 && yPrev >= 0 && yPrev < 8 {
xPrev--
yPrev--
piece = b.getPiece(xPrev, yPrev)
if piece != nil {
if ((!queen && piece.Name == Bishop) || (queen && piece.Name == Queen)) && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
} else {
break
}
}
}
return fmt.Errorf("no bishop found that can move to %s", position)
}
func (b *Board) moveKnight(position string, fromX int, fromY int) error {
var (
x int
y int
xPrev int
yPrev int
p *Piece
validPrev []*Square
err error
)
if x, y, err = getXY(position); err != nil {
return err
}
checkMove := b.validateMove(Knight, fromX, fromY)
xPrev = x + 1
yPrev = y - 2
p = b.getPiece(xPrev, yPrev)
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
}
xPrev = x + 2
yPrev = y - 1
p = b.getPiece(xPrev, yPrev)
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
}
xPrev = x + 2
yPrev = y + 1
p = b.getPiece(xPrev, yPrev)
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
}
xPrev = x + 1
yPrev = y + 2
p = b.getPiece(xPrev, yPrev)
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
}
xPrev = x - 1
yPrev = y + 2
p = b.getPiece(xPrev, yPrev)
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
}
xPrev = x - 2
yPrev = y + 1
p = b.getPiece(xPrev, yPrev)
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
}
xPrev = x - 2
yPrev = y - 1
p = b.getPiece(xPrev, yPrev)
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
}
xPrev = x - 1
yPrev = y - 2
p = b.getPiece(xPrev, yPrev)
if checkMove(p, xPrev, yPrev) {
validPrev = append(validPrev, &Square{X: xPrev, Y: yPrev})
}
if len(validPrev) > 1 {
return fmt.Errorf("move ambiguous: %d knights can move to %s", len(validPrev), position)
}
if len(validPrev) == 1 {
xPrev = validPrev[0].X
yPrev = validPrev[0].Y
p = b.getPiece(xPrev, yPrev)
b.tiles[x][y] = p
b.tiles[xPrev][yPrev] = nil
return nil
}
return fmt.Errorf("no knight found that can move to %s", position)
}
func (b *Board) moveQueen(position string) error {
var (
err error
)
if err = b.moveBishop(position, true); err == nil {
return nil
}
if err = b.moveRook(position, true, -1, -1); err == nil {
return nil
}
return fmt.Errorf("no queen found that can move to %s", position)
}
func (b *Board) moveKing(position string) error {
var (
x int
y int
xPrev int
yPrev int
piece *Piece
err error
)
if x, y, err = getXY(position); err != nil {
return err
}
// ^
xPrev = x + 0
yPrev = y - 1
piece = b.getPiece(xPrev, yPrev)
if piece != nil && piece.Name == King && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
}
// ^>
xPrev = x + 1
yPrev = y - 1
piece = b.getPiece(xPrev, yPrev)
if piece != nil && piece.Name == King && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
}
// >
xPrev = x + 1
yPrev = y + 0
piece = b.getPiece(xPrev, yPrev)
if piece != nil && piece.Name == King && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
}
// v>
xPrev = x + 1
yPrev = y + 1
piece = b.getPiece(xPrev, yPrev)
if piece != nil && piece.Name == King && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
}
// v
xPrev = x + 0
yPrev = y + 1
piece = b.getPiece(xPrev, yPrev)
if piece != nil && piece.Name == King && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
}
// <v
xPrev = x - 1
yPrev = y + 1
piece = b.getPiece(xPrev, yPrev)
if piece != nil && piece.Name == King && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
}
// <
xPrev = x - 1
yPrev = y + 0
piece = b.getPiece(xPrev, yPrev)
if piece != nil && piece.Name == King && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
}
// <^
xPrev = x - 1
yPrev = y - 1
piece = b.getPiece(xPrev, yPrev)
if piece != nil && piece.Name == King && piece.Color == b.turn {
b.tiles[xPrev][yPrev] = nil
b.tiles[x][y] = piece
return nil
}
return fmt.Errorf("no king found that can move to %s", position)
}
func (b *Board) mustSetPiece(name PieceName, color Color, position string) {
if err := b.SetPiece(name, color, position); err != nil {
log.Fatalf("cannot set piece %s: %v", name, err)
}
}
func (b *Board) At(position string) *Piece {
var (
x int
y int
err error
)
if x, y, err = getXY(position); err != nil {
return nil
}
return b.tiles[x][y]
}
func getXY(position string) (int, int, error) {
var (
posX rune
posY rune
x int
y int
)
runes := []rune(position)
posX = runes[0]
posY = runes[1]
if posX < 'a' || posX > 'h' {
return -1, -1, fmt.Errorf("invalid move: %s", position)
}
if posY < '1' || posY > '8' {
return -1, -1, fmt.Errorf("invalid move: %s", position)
}
// image origin (0,0) is at top-left corner (a8)
x = int(posX - 'a')
y = int('8' - posY)
return x, y, nil
}
func (b *Board) getPiece(x int, y int) *Piece {
if x < 0 || x >= 8 || y < 0 || y >= 8 {
return nil
}
return b.tiles[x][y]
}
func (b *Board) getCollision(position string) (*Piece, error) {
var (
x, y int
p *Piece
collision bool
err error
)
if x, y, err = getXY(position); err != nil {
return nil, err
}
p = b.getPiece(x, y)
// check if position is occupied by own piece
collision = p != nil && p.Color == b.turn
if collision {
return p, nil
}
// TODO: check if path is blocked by some piece
return nil, nil
}
func getTileColor(x, y int) Color {
if x%2 == y%2 {
return Light
} else {
return Dark
}
}
func flipImage(img image.Image) *image.RGBA {
bounds := img.Bounds()
flipped := image.NewRGBA(bounds)
// Flip the image vertically by reversing the Y coordinate
for y := bounds.Min.Y; y < bounds.Max.Y; y++ {
for x := bounds.Min.X; x < bounds.Max.X; x++ {
flipped.Set(x, bounds.Max.Y-y-1, img.At(x, y))
}
}
return flipped
}