#!/usr/bin/python
#coding: utf-8
import Tkinter as Tk
import Dialog as D
import ScrolledText as ST
import random, time
import sys
BLACK, WHITE, NONE = range(3)
def get_pos(i):
return i % 8, i / 8
class Reversi:
def __init__(self):
self.xlist = (-1, 0, 1, -1, 1, -1, 0, 1)
self.ylist = (-1, -1, -1, 0, 0, 1, 1, 1)
self.init_game()
def init_game(self):
global board
board = [NONE for i in range(64)]
board[8*3+3] = board[8*4+4] = WHITE
board[8*4+3] = board[8*3+4] = BLACK
global COUNT
COUNT = 1
def put_stone(self, x, y, turn, count=True):
global COUNT
if board[8*y+x] != NONE:
return False
elif self.check_put(x, y, turn):
if count: COUNT += 1
self.put_stones(x, y, turn)
return True
else:
return False
def check_put(self, x, y, turn):
""" (x,y)の位置に石を置けるか判定する """
revnum = 0
for i in range(8):
ax = self.xlist[i] + x
ay = self.ylist[i] + y
prevnum = 0
while True:
if not(0 <= ax <= 7 and 0 <= ay <= 7):
break
if board[8*ay+ax] == (not turn):
prevnum += 1
elif board[8*ay+ax] == turn and prevnum:
revnum += prevnum
break
else:
break
ax += self.xlist[i]
ay += self.ylist[i]
return revnum
def put_stones(self, x, y, turn):
""" (x, y)に石を置く """
board[8*y+x] = turn
for i in range(8):
ax = self.xlist[i] + x
ay = self.ylist[i] + y
check_flag = put_flag = False
px = ax; py = ay
while (0 <= px <= 7 and 0 <= py <= 7):
if board[8*py+px] == (not turn):
check_flag = True
elif board[8*py+px] == turn and check_flag:
put_flag = True
else:
break
px += self.xlist[i]
py += self.ylist[i]
while put_flag:
if board[8*ay+ax] == (not turn):
board[8*ay+ax] = turn
else:
break
ax += self.xlist[i]
ay += self.ylist[i]
def get_nummoves(self, turn):
""" そのターンの手数をリストにする """
nummoves = []
for i in range(64):
x, y = get_pos(i)
if board[8*y+x] == NONE and self.check_put(x, y, turn):
nummoves.append((x, y))
return nummoves
def judge_remain(self, state):
""" 白、黒が全滅、置く所がなくなったか判定する """
return state not in board
def judge_end(self):
""" どちらも置けなくなって終局になるか判定する """
for stone in (BLACK, WHITE):
if self.get_nummoves(stone):
return False
return True
class AI(Reversi):
def __init__(self, stone):
Reversi.__init__(self)
self.stone = stone
"""
weights = (120, -20, 20, 5, 5, 20, -20, 120, -20, -40, -5, -5,
-5, -5, -40, -20, 20, -5, 15, 3, 3, 15, -5, 20, 5,
-5, 3, 3, 3, 3, -5, 20)
"""
weights = (30,-12,0,-1,-1,0,-12,30,-12,-15,-3,-3,-3,-3,-15,-12,
0,-3,0,-1,-1,0,-3,0,-1,-3,-1,-1,-1,-1,-3,-1)
self.wboard = [0 for i in range(64)]
for i in range(len(weights)):
self.wboard[i] = weights[i]
self.wboard[63-i] = weights[i]
def put(self):
self.nummoves = self.get_nummoves(self.stone)
x, y = self.thinking()
return x, y
def calc_weights(self):
score = 0
for i in range(64):
if board[i] == self.stone:
score += self.wboard[i]
elif board[i] == (not self.stone):
score -= self.wboard[i]
return score
def judge_score(self):
global board
score_list = []
for x, y in self.nummoves:
backup = board[:]
if self.put_stone(x, y, self.stone, count=False):
score_list.append((self.calc_weights(), x, y))
board = backup[:]
x, y = max(score_list)[1:]
return x, y
def thinking(self):
# 鼠定石にならないようにする
if self.stone == WHITE and COUNT == 2:
nummoves = []
for x, y in self.nummoves:
count = 0
for ax, ay in [(-1, 0), (1, 0), (0, -1), (0, 1)]:
if board[(y+ay)*8+(x+ax)] == (not self.stone):
count += 1
if count != 2:
nummoves.append((x, y))
else:
random.choice(nummoves)
return self.judge_score()
if COUNT <= 15: #序盤
pass
elif COUNT <= 40: #中盤
pass
else: #終盤
pass
class Board(Tk.Canvas):
cs = 50
def __init__(self, master, msg):
Tk.Canvas.__init__(self, master, bg="white",
width=self.cs*8, height=self.cs*8)
self.msg = msg
self.init()
self.bind_all("<Button-1>", self.clicked)
self.after(0, self.loop)
def init(self):
self.cpp_id = None
self.cpp = (-1, -1)
self.ply_turn = random.randrange(2)
if len(sys.argv) >= 2:
self.ply_turn = int(sys.argv[1])
self.com_turn = not self.ply_turn
self.reversi = Reversi()
self.ai = AI(self.com_turn)
self.turn = BLACK
self.draw_board()
self.draw_stone()
self.msg.update("プレイヤーの番です" if self.ply_turn == BLACK else "")
self.player_time = time.time()
def game_set(self):
self.end_game()
for i in self.find_all(): self.delete(i)
self.init()
def draw_board(self):
self.create_rectangle(0, 0, self.cs*8, self.cs*8, fill="#008B00")
for pos in range(1,8):
pos *= self.cs
self.create_line(pos, 0, pos, self.cs*8)
self.create_line(0, pos, self.cs*8, pos)
wid = 3
for x, y in (2, 2), (2, 6), (6, 2), (6, 6):
self.create_oval(self.cs*x-wid, self.cs*y-wid,
self.cs*x+wid, self.cs*y+wid, fill="black")
def draw_stone(self):
for y in range(8):
for x in range(8):
x0 = self.cs * x
y0 = self.cs * y
if board[8*y+x] == BLACK:
self.create_oval(x0+3, y0+3, x0+self.cs-3,
y0+self.cs-3, fill="#000000", width=1)
elif board[8*y+x] == WHITE:
self.create_oval(x0+3, y0+3, x0+self.cs-3,
y0+self.cs-3, fill="#FFFFFF", width=1)
if self.cpp_id: self.delete(self.cpp_id)
x, y = self.cpp
x = self.cs * x + self.cs / 2
y = self.cs * y + self.cs / 2
self.cpp_id = self.create_oval(x-5, y-5, x+5, y+5, fill="red")
def loop(self):
if self.reversi.judge_remain(NONE) or self.reversi.judge_end():
self.game_set()
if (self.turn == self.com_turn and
(time.time() - self.player_time) >= 0.3):
self.turn = self.ply_turn
if not self.reversi.get_nummoves(self.com_turn):
self.msg.update(text="COMはパスしました")
else:
x, y = self.ai.put()
self.reversi.put_stone(x, y, self.com_turn)
stone = "黒" if self.ply_turn==WHITE else "白"
self.msg.update(text="プレイヤーの番です",
pos="COM (%d,%d) %s"%(x, y, stone))
self.cpp = (x, y)
self.draw_stone()
elif self.turn == self.ply_turn:
if not self.reversi.get_nummoves(self.ply_turn):
self.turn = self.com_turn
self.after(50, self.loop)
def clicked(self, event):
self.msg.update(text="")
if self.turn != self.ply_turn: return
x = event.x / self.cs
y = event.y / self.cs
result = self.reversi.put_stone(x, y, self.ply_turn)
if result:
self.draw_stone()
self.turn = self.com_turn
if self.reversi.judge_remain(self.turn): self.game_set()
self.player_time = time.time()
stone = "黒" if self.ply_turn == BLACK else "白"
self.msg.update(pos="PLAYER (%d,%d) %s" %(x, y, stone))
else:
self.msg.update(text="そこには置けません")
def end_game(self):
bn = wn = 0
ply = com = 0
for i in range(64):
x, y = get_pos(i)
if board[8*y+x] == BLACK:
bn += 1
elif board[8*y+x] == WHITE:
wn += 1
if board[8*y+x] == self.ply_turn:
ply += 1
elif board[8*y+x] == self.com_turn:
com += 1
if ply > com:
text = "プレイヤーの勝ち"
elif ply < com:
text = "COMの勝ち"
else:
text = "引き分け"
text = "黒:%d 白:%d " % (bn,wn) + "\n" + text
dialog = D.Dialog(title="勝敗", text=text, bitmap="question",
default=0, strings=["OK"])
self.msg.clear()
self.turn = BLACK
class Message:
def __init__(self, master):
self.msg = Tk.Label(master, relief=Tk.RIDGE, bd=2)
self.msg.pack(fill=Tk.X)
self.log = ST.ScrolledText(master, state=Tk.DISABLED,
font=("Monospace", "12"),
height=5, width=1)
self.log.pack(side=Tk.BOTTOM, fill=Tk.X)
def update(self, text=None, pos=None):
if text is not None:
self.msg.config(text=text)
if pos is not None:
self.log.config(state=Tk.NORMAL)
self.log.insert(Tk.END, pos+"\n")
self.log.config(state=Tk.DISABLED)
self.log.see(Tk.END)
def clear(self):
self.msg.config(text="")
self.log.config(state=Tk.NORMAL)
self.log.delete("1.0", Tk.END)
self.log.config(state=Tk.DISABLED)
class Application(Tk.Frame):
def __init__(self, master=None):
Tk.Frame.__init__(self, master)
frame = Tk.Frame()
message = Message(frame)
self.board = Board(frame, message)
self.board.pack(side=Tk.TOP)
frame.pack()
self.set_menubar()
def set_menubar(self):
menubar = Tk.Menu(self, tearoff=0)
filemenu = Tk.Menu(menubar, tearoff=0)
menubar.add_cascade(label=u"ゲーム(G)", menu=filemenu, underline=5)
filemenu.add_command(label=u"最初から", command=self.board.init)
self.master.config(menu=menubar)
if __name__ == "__main__":
frame = Application()
frame.pack()
frame.mainloop()
#!/usr/bin/python
#coding: utf-8

import Tkinter as Tk
import Dialog as D
import ScrolledText as ST
import random, time
import sys

BLACK, WHITE, NONE = range(3)

def get_pos(i):
    return i % 8, i / 8

class Reversi:
    def __init__(self):
        self.xlist = (-1, 0, 1, -1, 1, -1, 0, 1)
        self.ylist = (-1, -1, -1, 0, 0, 1, 1, 1)
        self.init_game()

    def init_game(self):
        global board
        board = [NONE for i in range(64)]
        board[8*3+3] = board[8*4+4] = WHITE
        board[8*4+3] = board[8*3+4] = BLACK
        global COUNT
        COUNT = 1

    def put_stone(self, x, y, turn, count=True):
        global COUNT
        if board[8*y+x] != NONE:
            return False
        elif self.check_put(x, y, turn):
            if count: COUNT += 1
            self.put_stones(x, y, turn)
            return True
        else:
            return False

    def check_put(self, x, y, turn):
        """ (x,y)の位置に石を置けるか判定する """
        revnum = 0
        for i in range(8):
            ax = self.xlist[i] + x
            ay = self.ylist[i] + y
            prevnum = 0
            while True:
                if not(0 <= ax <= 7 and 0 <= ay <= 7):
                    break
                if board[8*ay+ax] == (not turn):
                    prevnum += 1
                elif board[8*ay+ax] == turn and prevnum:
                    revnum += prevnum
                    break
                else:
                    break
                ax += self.xlist[i]
                ay += self.ylist[i]
        return revnum

    def put_stones(self, x, y, turn):
        """ (x, y)に石を置く """
        board[8*y+x] = turn
        for i in range(8):
            ax = self.xlist[i] + x
            ay = self.ylist[i] + y
            check_flag = put_flag = False

            px = ax; py = ay
            while (0 <= px <= 7 and 0 <= py <= 7):
                if board[8*py+px] == (not turn):
                    check_flag = True
                elif board[8*py+px] == turn and check_flag:
                    put_flag = True
                else:
                    break
                px += self.xlist[i]
                py += self.ylist[i]

            while put_flag:
                if board[8*ay+ax] == (not turn):
                    board[8*ay+ax] = turn
                else:
                    break
                ax += self.xlist[i]
                ay += self.ylist[i]

    def get_nummoves(self, turn):
        """ そのターンの手数をリストにする """
        nummoves = []
        for i in range(64):
            x, y = get_pos(i)
            if board[8*y+x] == NONE and self.check_put(x, y, turn):
                nummoves.append((x, y))
        return nummoves

    def judge_remain(self, state):
        """ 白、黒が全滅、置く所がなくなったか判定する """
        return state not in board

    def judge_end(self):
        """ どちらも置けなくなって終局になるか判定する """
        for stone in (BLACK, WHITE):
            if self.get_nummoves(stone):
                return False
        return True

class AI(Reversi):
    def __init__(self, stone):
        Reversi.__init__(self)
        self.stone = stone
        """
        weights = (120, -20, 20, 5, 5, 20, -20, 120, -20, -40, -5, -5,
                   -5, -5, -40, -20, 20, -5, 15, 3, 3, 15, -5, 20, 5,
                   -5, 3, 3, 3, 3, -5, 20)
                   """
        weights = (30,-12,0,-1,-1,0,-12,30,-12,-15,-3,-3,-3,-3,-15,-12,
                   0,-3,0,-1,-1,0,-3,0,-1,-3,-1,-1,-1,-1,-3,-1)

        self.wboard = [0 for i in range(64)]
        for i in range(len(weights)):
            self.wboard[i] = weights[i]
            self.wboard[63-i] = weights[i]

    def put(self):
        self.nummoves = self.get_nummoves(self.stone)
        x, y = self.thinking()
        return x, y

    def calc_weights(self):
        score = 0
        for i in range(64):
            if board[i] == self.stone:
                score += self.wboard[i]
            elif board[i] == (not self.stone):
                score -= self.wboard[i]
        return score

    def judge_score(self):
        global board
        score_list = []
        for x, y in self.nummoves:
            backup = board[:]
            if self.put_stone(x, y, self.stone, count=False):
                score_list.append((self.calc_weights(), x, y))
                board = backup[:]
        x, y = max(score_list)[1:]
        return x, y

    def thinking(self):
        # 鼠定石にならないようにする
        if self.stone == WHITE and COUNT == 2:
            nummoves = []
            for x, y in self.nummoves:
                count = 0
                for ax, ay in [(-1, 0), (1, 0), (0, -1), (0, 1)]:
                    if board[(y+ay)*8+(x+ax)] == (not self.stone):
                        count += 1
                if count != 2:
                    nummoves.append((x, y))
            else:
                random.choice(nummoves)

        return self.judge_score()

        if COUNT <= 15: #序盤
            pass
        elif COUNT <= 40: #中盤
            pass
        else: #終盤
            pass

class Board(Tk.Canvas):
    cs = 50
    def __init__(self, master, msg):
        Tk.Canvas.__init__(self, master, bg="white",
                           width=self.cs*8, height=self.cs*8)
        self.msg = msg
        self.init()
        self.bind_all("<Button-1>", self.clicked)
        self.after(0, self.loop)

    def init(self):
        self.cpp_id = None
        self.cpp = (-1, -1)
        self.ply_turn = random.randrange(2)
        if len(sys.argv) >= 2:
            self.ply_turn = int(sys.argv[1])
        self.com_turn = not self.ply_turn
        self.reversi = Reversi()
        self.ai = AI(self.com_turn)
        self.turn = BLACK
        self.draw_board()
        self.draw_stone()
        self.msg.update("プレイヤーの番です" if self.ply_turn == BLACK else "")
        self.player_time = time.time()

    def game_set(self):
        self.end_game()
        for i in self.find_all(): self.delete(i)
        self.init()

    def draw_board(self):
        self.create_rectangle(0, 0, self.cs*8, self.cs*8, fill="#008B00")
        for pos in range(1,8):
            pos *= self.cs
            self.create_line(pos, 0, pos, self.cs*8)
            self.create_line(0, pos, self.cs*8, pos)
        wid = 3
        for x, y in (2, 2), (2, 6), (6, 2), (6, 6):
            self.create_oval(self.cs*x-wid, self.cs*y-wid,
                             self.cs*x+wid, self.cs*y+wid, fill="black")

    def draw_stone(self):
        for y in range(8):
            for x in range(8):
                x0 = self.cs * x
                y0 = self.cs * y
                if board[8*y+x] == BLACK:
                    self.create_oval(x0+3, y0+3, x0+self.cs-3,
                                     y0+self.cs-3, fill="#000000", width=1)
                elif board[8*y+x] == WHITE:
                    self.create_oval(x0+3, y0+3, x0+self.cs-3,
                                     y0+self.cs-3, fill="#FFFFFF", width=1)

        if self.cpp_id: self.delete(self.cpp_id)
        x, y = self.cpp
        x = self.cs * x + self.cs / 2
        y = self.cs * y + self.cs / 2
        self.cpp_id = self.create_oval(x-5, y-5, x+5, y+5, fill="red")

    def loop(self):
        if self.reversi.judge_remain(NONE) or self.reversi.judge_end():
            self.game_set()

        if (self.turn == self.com_turn and
            (time.time() - self.player_time) >= 0.3):
            self.turn = self.ply_turn

            if not self.reversi.get_nummoves(self.com_turn):
                self.msg.update(text="COMはパスしました")
            else:
                x, y = self.ai.put()
                self.reversi.put_stone(x, y, self.com_turn)
                stone = "黒" if self.ply_turn==WHITE else "白"
                self.msg.update(text="プレイヤーの番です",
                                pos="COM    (%d,%d) %s"%(x, y, stone))
                self.cpp = (x, y)
            self.draw_stone()
        elif self.turn == self.ply_turn:
            if not self.reversi.get_nummoves(self.ply_turn):
                self.turn = self.com_turn
        self.after(50, self.loop)

    def clicked(self, event):
        self.msg.update(text="")
        if self.turn != self.ply_turn: return

        x = event.x / self.cs
        y = event.y / self.cs

        result = self.reversi.put_stone(x, y, self.ply_turn)
        if result:
            self.draw_stone()
            self.turn = self.com_turn
            if self.reversi.judge_remain(self.turn): self.game_set()
            self.player_time = time.time()
            stone = "黒" if self.ply_turn == BLACK else "白"
            self.msg.update(pos="PLAYER (%d,%d) %s" %(x, y, stone))
        else:
            self.msg.update(text="そこには置けません")

    def end_game(self):
        bn = wn = 0
        ply = com = 0
        for i in range(64):
            x, y = get_pos(i)
            if board[8*y+x] == BLACK:
                bn += 1
            elif board[8*y+x] == WHITE:
                wn += 1
            if board[8*y+x] == self.ply_turn:
                ply += 1
            elif board[8*y+x] == self.com_turn:
                com += 1

        if ply > com:
            text = "プレイヤーの勝ち"
        elif ply < com:
            text = "COMの勝ち"
        else:
            text = "引き分け"
        text = "黒:%d 白:%d " % (bn,wn) + "\n" + text
        dialog = D.Dialog(title="勝敗", text=text, bitmap="question",
                          default=0, strings=["OK"])
        self.msg.clear()
        self.turn = BLACK

class Message:
    def __init__(self, master):
        self.msg = Tk.Label(master, relief=Tk.RIDGE, bd=2)
        self.msg.pack(fill=Tk.X)
        self.log = ST.ScrolledText(master, state=Tk.DISABLED,
                                   font=("Monospace", "12"),
                                   height=5, width=1)
        self.log.pack(side=Tk.BOTTOM, fill=Tk.X)

    def update(self, text=None, pos=None):
        if text is not None:
            self.msg.config(text=text)
        if pos is not None:
            self.log.config(state=Tk.NORMAL)
            self.log.insert(Tk.END, pos+"\n")
            self.log.config(state=Tk.DISABLED)
            self.log.see(Tk.END)

    def clear(self):
        self.msg.config(text="")
        self.log.config(state=Tk.NORMAL)
        self.log.delete("1.0", Tk.END)
        self.log.config(state=Tk.DISABLED)

class Application(Tk.Frame):
    def __init__(self, master=None):
        Tk.Frame.__init__(self, master)
        frame = Tk.Frame()
        message = Message(frame)
        self.board = Board(frame, message)
        self.board.pack(side=Tk.TOP)
        frame.pack()
        self.set_menubar()

    def set_menubar(self):
        menubar = Tk.Menu(self, tearoff=0)
        filemenu = Tk.Menu(menubar, tearoff=0)
        menubar.add_cascade(label=u"ゲーム(G)", menu=filemenu,  underline=5)
        filemenu.add_command(label=u"最初から", command=self.board.init)
        self.master.config(menu=menubar)

if __name__ == "__main__":
    frame = Application()
    frame.pack()
    frame.mainloop()
