チェスアプリ開発(3) ポーンの最初の動きを追加

Python プログラムで動かすフェアリーチェスアプリ開発、連載第3回です。

は、先手の駒色と駒の初期配置を修正しました。

今回は、ポーンの最初の動きを付け加えてみましょう。

本来ポーンは、最初の動きだけは2歩まで進むことができます。


駒に関するクラス

まず、Pawn を見ると、class Pawn(Piece):とあります。

これはPieceクラスを継承していることがわかります。

クラスを定義するとき、()内に書かれたものが基底クラス(親クラス)になります。

継承をすると、基底クラスで定義された

を子クラスでも使うことができます。

あと、子クラスから生成された

から基底クラスの属性やメソッドにアクセスできるようになります。

それでは、Pawnの中身を見ていきましょう。(下は原文ママ)

pieces.py
1
class Pawn(Piece):
2
def __init__(self,color,name,direction):
3
self.name = name
4
self.Color = color
5
#of course, the smallest piece is the hardest to code. direction should be either 1 or -1, should be -1 if the pawn is traveling "backwards"
6
self.direction = direction
7
def availableMoves(self,x,y,gameboard, Color = None):
8
if Color is None : Color = self.Color
9
answers = []
10
if (x+1,y+self.direction) in gameboard and self.noConflict(gameboard, Color, x+1, y+self.direction) : answers.append((x+1,y+self.direction))
11
if (x-1,y+self.direction) in gameboard and self.noConflict(gameboard, Color, x-1, y+self.direction) : answers.append((x-1,y+self.direction))
12
if (x,y+self.direction) not in gameboard and Color == self.Color : answers.append((x,y+self.direction))# the condition after the and is to make sure the non-capturing movement (the only fucking one in the game) is not used in the calculation of checkmate
13
return answers

__init__() がある

他の駒にはないのにPawnにだけ

`__init__()`があります。

これはPawnにだけdirectionというプロパティをつける必要があったからでしょう。

他の駒は前後対称な動きをするけど、ポーンだけは前後が決まっています。

directionには1を指定すれば白から見て前方向、-1を指定すれば後方向ですね。


ポーンの特殊な動きを再現

ポーンはちょっと特殊な動きをします。

動き始めだけは2歩まで進めるということの他にも、

進むときは前に1歩しか動けないが、敵駒を取るときにのみ斜め前に1歩進める(斜め前の駒を取れる)のです。

次のavailableMoves()は盤面上の(x, y)の位置にいる駒の動けるマスを出力します。

❓なぜわかる?

これは例えば、

その駒が本来動けない場所への移動を指定したときにinvalid moveとプリントされる
Gameクラスのmain()の中のif target.isValid(startpos,endpos,target.Color,self.gameboard):にかからなかった
target.isValid(startpos,endpos,target.Color,self.gameboard)False
PieceクラスのisValid(self,startpos,endpos,Color,gameboard)endpos in self.availableMoves(startpos[0],startpos[1],gameboard, Color = Color)ならばTrueを返す
availableMoves()は移動可能なマスの位置を表している

というふうにしてわかります。


以下のコードは原文を少し整形しています。

pieces.py
1
def availableMoves(self, x, y, gameboard, Color=None):
2
if Color is None:
3
Color = self.Color
4
answers = []
5
if ((x + 1, y + self.direction) in gameboard
6
and self.noConflict(gameboard, Color, x + 1, y + self.direction)):
7
answers.append((x + 1, y + self.direction))
8
if ((x - 1, y + self.direction) in gameboard
9
and self.noConflict(gameboard, Color, x - 1, y + self.direction)):
10
answers.append((x - 1, y + self.direction))
11
if (x, y + self.direction) not in gameboard and Color == self.Color:
12
answers.append((x, y + self.direction))
13
return answers

if Color is None:

最初のこのif文は、Colorが特に指定されなかったときのための受け皿みたいなものなので、ここではあまり重要ではありません。

answers

answersに移動可能なマスを二要素

として格納し、リストを作っています。

最終的には[(int, int), ...]という形になります。(intは整数という意味)

if ... in gameboard and self.noConflict ...

ここの3つのif文がこの部分の勘所です。

最初の2文は、ポーンの攻撃を表しています。

ポーンは敵駒を取るときにのみ斜め前に1歩進めます。

もし自分の斜め前に駒があり(... in gameboard)、その駒が自分と異なる色である(self.noConflict(...))とき、そのマスに移動可能になりますよ~ということです。

noConflict()Pieceクラスで定義されていて、中身はこんな感じです。

pieces.py
1
def noConflict(self, gameboard, initialColor, x, y):
2
if (self.isInBounds(x, y)
3
and ((x, y) not in gameboard
4
or gameboard[(x, y)].Color != initialColor)):
5
return True
6
return False

isInBounds()(x, y)が盤面の範囲内にあるときTrueを返すものなので、

noConflict()(x, y)が盤面の範囲内にあって味方の駒がない(駒がないもしくは敵の駒がある)マスであるときにTrueとなります。

そこには駒が進出可能ということですね。

if (x, y + self.direction) not in gameboard and Color == self.Color:

目の前に駒がなければ(if (x, y + self.direction) not in gameboard)、1歩前に移動できますよ~ということです。

Color == self.Colorの部分ですが、

コメントでは、駒を取らない動きがチェックメイトの計算に使われないことを保証するため、と書かれています。

たぶん、キングが相手のポーンの目の前に移動すると、そのマスにポーンが移動できるためチェックと判定されるのを防ごうとしているのかなと思いますが、

すでに(x, y + self.direction) not in gameboardを通っているので、この部分はなくていいんじゃないかなと思います。

そもそも、コード内でColor != self.Colorになることがないので…


最初の動きを追加

さて、駒の動きの仕組みを理解したところで、最初の動きだけは2歩まで動けるという動きを追加していきます。

最初だけはというのがちょっと難しいかもしれないです。

まずは、駒がないマスならば前2歩まで進めることを表現してみましょう。

pieces.py
1
def availableMoves(self, x, y, gameboard, Color=None):
2
if Color is None:
3
Color = self.Color
4
answers = []
5
if ((x + 1, y + self.direction) in gameboard
6
and self.noConflict(gameboard, Color, x + 1, y + self.direction)):
7
answers.append((x + 1, y + self.direction))
8
if ((x - 1, y + self.direction) in gameboard
9
and self.noConflict(gameboard, Color, x - 1, y + self.direction)):
10
answers.append((x - 1, y + self.direction))
11
if (x, y + self.direction) not in gameboard:
12
answers.append((x, y + self.direction))
13
if ((x, y + self.direction) not in gameboard
14
and (x, y + 2*self.direction) not in gameboard):
15
answers.append((x, y + 2*self.direction))
16
return answers

ポイントは2マス先に駒がなくても1マス先に駒があれば通れないということです。

さて、最初の動きだけ、という条件をつけていくのですが、

それぞれのポーンにとって最初の動きに特有の環境というものがあります。

初期位置です。

初期配置

通常のチェスの場合、白ポーンは初めて動くとき必ず2ランクにいて

2ランクにいるのはいつでもまだ動いていないときです。

この位置情報を条件として付与すればよさそうです。

pieces.py
1
def availableMoves(self, x, y, gameboard, Color=None):
2
if Color is None:
3
Color = self.Color
4
answers = []
5
if ((x + 1, y + self.direction) in gameboard
6
and self.noConflict(gameboard, Color, x + 1, y + self.direction)):
7
answers.append((x + 1, y + self.direction))
8
if ((x - 1, y + self.direction) in gameboard
9
and self.noConflict(gameboard, Color, x - 1, y + self.direction)):
10
answers.append((x - 1, y + self.direction))
11
if (x, y + self.direction) not in gameboard:
12
answers.append((x, y + self.direction))
13
if (((self.Color == WHITE and y == 1)
14
or (self.Color == BLACK and y == 6))
15
and (x, y + self.direction) not in gameboard
16
and (x, y + 2*self.direction) not in gameboard):
17
answers.append((x, y + 2*self.direction))
18
return answers

(x, y)の形では0からのスタートになるので、

2ランクはy == 1となります。

コマンドライン

ちゃんと2歩動きました!

コマンドライン

最初だけ!

ただ、このやり方は通常のチェスであったから通用しただけであって、

今後フェアリーチェスへの拡張を考えると、もっと柔軟な対応が必要になるのですが、

そのへんはまた今度にしましょう。


今回は駒に動きを追加しました。

はチェックまわりについて見ていきたいと思います。

長くなってしまいましたが、お読みいただきありがとうございました!

それではまた👋