-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathConnectFour.py
170 lines (138 loc) · 5.91 KB
/
ConnectFour.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
from config import *
from Board import Board
import random
import copy
import time
class ConnectFour:
def __init__(self):
self.board = Board()
self.gameOver = False
self.turn = PLAYER1
self.winner = None
self.opponent_type = None
self.depth = None
def getBoard(self):
return self.board
def makeMove(self):
print(f"\nPlayer {self.turn}'s turn")
column = input("Enter column number: ")
# check if column is not integer or out of range
if not column.isdigit() or int(column) < 1 or int(column) > 7:
print("Invalid input. Please enter a number between 1 and 7...")
return self.makeMove()
column = int(column)
# check if column is full
if column - 1 not in self.board.getValidMoves():
print("Column is full. Please choose another column...")
return self.makeMove()
self.board.dropPiece(column - 1, self.turn)
print ("\n--------------------------------------\n")
def makeAIMove(self):
print(f"\nPlayer {self.turn}'s turn. AI is thinking...")
column, minimax_score = self.minimax(self.board, self.depth, True)
time.sleep(1)
print(f"Player {self.turn} chooses column {column + 1}")
self.board.dropPiece(column, self.turn)
print ("\n--------------------------------------\n")
def switchTurn(self):
self.turn = PLAYER1 if self.turn == PLAYER2 else PLAYER2
def minimax(self, board, depth, maximizingPlayer, alpha=-float('inf'), beta=float('inf')):
valid_moves = board.getValidMoves()
is_terminal = board.isBoardFull() or board.checkWin()
if depth == 0 or is_terminal:
if is_terminal:
if board.checkWin() == PLAYER2:
return (None, 1000000 + depth)
elif board.checkWin() == PLAYER1:
return (None, -1000000 - depth)
else:
return (None, 0)
else:
return (None, board.evalScore(PLAYER2))
if maximizingPlayer:
value = -float('inf')
column = random.choice(valid_moves)
for col in valid_moves:
board_copy = copy.deepcopy(board)
board_copy.dropPiece(col, PLAYER2)
new_score = self.minimax(board_copy, depth - 1, False, alpha, beta)[1]
if new_score > value:
value = new_score
column = col
alpha = max(alpha, value)
if alpha >= beta:
break
return (column, value)
else:
value = float('inf')
column = random.choice(valid_moves)
for col in valid_moves:
board_copy = copy.deepcopy(board)
board_copy.dropPiece(col, PLAYER1)
new_score = self.minimax(board_copy, depth - 1, True, alpha, beta)[1]
if new_score < value:
value = new_score
column = col
beta = min(beta, value)
if alpha >= beta:
break
return (column, value)
def playGame(self):
print ("Welcome to Connect Four!")
# choose between playing against AI or another player
while self.opponent_type is None:
print ("Do you want to play against AI or another player?")
print ("1. AI")
print ("2. Player")
opponent = input("\nEnter your choice (1 or 2): ")
print ()
if opponent.isdigit() and int(opponent) == 1:
self.opponent_type = "ai"
print("Choose the difficulty level:")
print("1. Very Easy")
print("2. Easy")
print("3. Medium")
print("4. Hard")
print("5. Expert")
print("6. Master")
print("7. Impossible")
print("8. Inhuman")
print("9. Godlike")
print("10. Legendary")
while self.depth is None:
level = input("\nEnter the difficulty level (1-10): ")
if level.isdigit() and 1 <= int(level) <= 10:
# verify it if user chooses number > 8
if int(level) > 8:
print("It may take up to 10 minutes to make a move for ai.\nAre you sure you want to continue? (y/N)")
answer = input()
if answer.lower() != "y":
continue
self.depth = int(level)
else:
print("Invalid input. Please enter a number between 1 and 10...")
elif opponent.isdigit() and int(opponent) == 2:
self.opponent_type = "player"
else:
print("Invalid input. Please enter 1 or 2...")
# random turn
self.turn = PLAYER1 if random.randint(0, 1) == 0 else PLAYER2
print ("\n---- Let's start the game! ----\n")
self.board.printBoard()
while not self.gameOver:
# if ai turn
if self.opponent_type == "ai" and self.turn == PLAYER2:
self.makeAIMove()
else:
self.makeMove()
self.board.printBoard()
if self.board.checkWin():
self.gameOver = True
self.winner = self.turn
print(f"\n\n**** Player {self.winner} wins! ****\n\n")
break
if self.board.isBoardFull():
self.gameOver = True
print("\n\n**** It's a tie! ****\n\n")
break
self.switchTurn()