-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtictactoe.py
146 lines (110 loc) · 3.74 KB
/
tictactoe.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
# create a 3x3 board using list comprehension
board = [[" " for _ in range(3)] for _ in range(3)]
# constants
MIN_ROW_COL = 0
MAX_ROW_COL = 2
ERROR_MESSAGE = "Invalid input. Try again."
def print_board(board):
for i in range(len(board)):
print(' | '.join(board[i]))
if i < len(board) - 1:
print("---------")
print("\n")
def make_move(board, row, col, player):
if board[row][col] == " ":
board[row][col] = player
return True
else:
return False
def is_game_over(board, round_number):
"""
Check if game is over.
This function checks rows, columns, and diagonals for a winning combination.
If the board is full and there is no winning combination, it is a tie.
Return 1 if player won, 2 if tie, 0 if game is not over.
"""
# check rows
for row in board:
if row.count(row[0]) == len(row) and row[0] != " ":
return 1
# check columns
for col in range(len(board[0])):
check_col = [row[col] for row in board]
if check_col.count(check_col[0]) == len(check_col) and check_col[0] != " ":
return 1
# check diagonals
if board[0][0] == board[1][1] == board[2][2] and board[0][0] != " ":
return 1
if board[0][2] == board[1][1] == board[2][0] and board[0][2] != " ":
return 1
# check if board is full
if round_number == 9:
return 2
return 0
def switch_players(player):
return "O" if player == "X" else "X"
def is_valid_input(row_col):
if len(row_col) != 2:
print("Error: Enter two values (each 0 - 2) separated by a space.")
return False
return True
def within_range(row_col):
try:
row_col = [int(i) for i in row_col]
except ValueError:
print(f"{ERROR_MESSAGE} Enter integers only.")
return None
if row_col[0] < MIN_ROW_COL or MAX_ROW_COL < row_col[0] or row_col[1] < MIN_ROW_COL or MAX_ROW_COL < row_col[1]:
print(ERROR_MESSAGE + " Out of range.")
return None
return row_col
def get_row_col():
# get move
while True:
try:
# get row and col in one line
row_col = input("Enter row and col: ").split()
if row_col[0] == "-1":
print("You quit the game.")
quit()
# check if two values are given
if not is_valid_input(row_col):
continue
# convert to int
row_col = [int(i) for i in row_col]
# check if row and col are within range
row_col = within_range(row_col)
if row_col is not None:
return row_col
except ValueError:
print(f"{ERROR_MESSAGE} Enter integers only.")
continue
def main():
player = "X"
round_number = 0
successful_move = True
print("\nWelcome to TicTacToe! \nEvery turn, enter the row and column (i.e. '0 2') you want to place your move in or enter -1 to quit.")
while True:
if successful_move:
print(f"\nRound {round_number}:\n")
print_board(board)
print(f"Player {player}'s turn.")
else:
print("!! That spot is taken, try again !!")
# get row and col from user
row, col = get_row_col()
if make_move(board, row, col, player):
round_number += 1
game_over = is_game_over(board, round_number)
if game_over == 1:
print(f"Player {player} won!")
break
elif game_over == 2:
print("It's a tie!")
break
player = switch_players(player)
successful_move = True
else:
successful_move = False
if __name__ == "__main__":
main()