import pygame
def main():
# Setup the properties of the game / screen
pygame.init()
SCREEN_WIDTH ,SCREEN_HEIGHT = 1280, 720
FPS = 60
screen = pygame.display.set_mode( (SCREEN_WIDTH, SCREEN_HEIGHT) )
pygame.display.set_caption('Tic Tac Toe')
clock = pygame.time.Clock()
running = True
clicked = False
# 0 = no winner, 1 = player1 won, 2 = player2 won, 3 = tie
winner = 0
restart = False
BG_COLOR = (0, 0, 0)
b = board(SCREEN_WIDTH, SCREEN_HEIGHT)
# Counts whose turn it is
# Even turn = Player1
# Odd turn = Player2
turn = 0
# Fill an array with 9 area objects, for each
# area of the board
#
# area 0.0 | area 0.1 | area 0.2
# _______________________________
# area 1.0 | area 1.1 | area 1.2
# _______________________________
# area 2.0 | area 2.1 | area 2.2
areas = []
for i in range(3):
row = []
for j in range(3):
row.append(area(
(b.get_board_posx() + (j * ( (b.get_board_size() / 3) +
(b.get_thickness() /2 ) ) ) ),
(b.get_board_posy() + (i * ( (b.get_board_size() / 3) +
(b.get_thickness() / 2 ) ) ) ),
b.get_field_size()
))
areas.append(row)
###########
# GAME LOOP
###########
while running:
# Abort condition
# If (x) is clicked, terminate program
# If clicked is True, the user clicked in this
# iteration of the game loop
for event in pygame.event.get():
if event.type == pygame.QUIT:
running = False
elif event.type == pygame.MOUSEBUTTONDOWN:
clicked = True
if (not (winner == 0)):
restart = True
# If any key or mousebutton is pressed
# while game is over, game restarts
elif event.type == pygame.KEYDOWN:
if (not winner == 0):
restart = True
if event.key == pygame.K_ESCAPE:
running = False
# Setup screen
screen.fill(BG_COLOR)
b.draw_board(screen)
mouse_posx, mouse_posy = pygame.mouse.get_pos()
# Looping through every area on the board
# Most of the game logic happens here
for i in range(len(areas)):
for j in range(len(areas[i])):
# Draw already set X tokens
if (areas[i][j].get_state() == 2):
draw_x_token(
areas[i][j].get_posx(),
areas[i][j].get_posy(),
b.get_field_size(),
screen,
b.get_color()
)
# Draw already set O tokens
if (areas[i][j].get_state() == 3):
draw_o_token(
areas[i][j].get_posx(),
areas[i][j].get_posy(),
b.get_field_size(),
screen,
b.get_color()
)
# Checking if mouse is hovering over any
# of the areas
# If so, preview token that is about to be drawn
if(areas[i][j].is_in(mouse_posx, mouse_posy)):
# print('Mouse is in Area({}, {})'.format(i, j))
# Handling clicks (Draw tokens)
if clicked:
if (not(areas[i][j].get_state() == 2) and
not(areas[i][j].get_state() ==3) and winner == 0 ):
if (turn % 2 == 0):
draw_x_token(
areas[i][j].get_posx(),
areas[i][j].get_posy(),
b.get_field_size(),
screen,
b.get_color()
)
areas[i][j].set_state(2)
turn += 1
else:
draw_o_token(
areas[i][j].get_posx(),
areas[i][j].get_posy(),
b.get_field_size(),
screen,
b.get_color()
)
areas[i][j].set_state(3)
turn += 1
if ( (areas[i][j].get_state() == 0
or areas[i][j].get_state() == 1) and winner == 0 ):
# even turn: Player1
if (turn % 2 == 0):
draw_x_token(
areas[i][j].get_posx(),
areas[i][j].get_posy(),
b.get_field_size(),
screen,
b.get_color()
)
areas[i][j].set_state(1)
else:
draw_o_token(
areas[i][j].get_posx(),
areas[i][j].get_posy(),
b.get_field_size(),
screen,
b.get_color()
)
areas[i][j].set_state(1)
# If mouse is not hovering over an area
# and area is not already used
# Set state to inactive
else:
if (
not(areas[i][j].get_state() == 2) and
not(areas[i][j].get_state() ==3)
):
areas[i][j].set_state(0)
# Check if there is a winner
# after this round
win_cons = [
# Horizontal
[ areas[0][0], areas[0][1], areas[0][2] ],
[ areas[1][0], areas[1][1], areas[1][2] ],
[ areas[2][0], areas[2][1], areas[2][2] ],
# Vertical
[ areas[0][0], areas[1][0], areas[2][0] ],
[ areas[0][1], areas[1][1], areas[2][1] ],
[ areas[0][2], areas[1][2], areas[2][2] ],
# Diagonal
[ areas[0][2], areas[1][1], areas[2][0] ],
[ areas[0][0], areas[1][1], areas[2][2] ]
]
for i in range( len(win_cons) ):
if (
win_cons[i][0].get_state() ==
win_cons[i][1].get_state() ==
win_cons[i][2].get_state() ==
2
):
winner = 1
if (
win_cons[i][0].get_state() ==
win_cons[i][1].get_state() ==
win_cons[i][2].get_state() ==
3
):
winner = 2
# Check for tie
fields_filled = 0
for i in range(len(areas)):
for j in range(len(areas[i])):
if (
areas[i][j].get_state() == 2 or
areas[i][j].get_state() == 3
):
fields_filled += 1
if (fields_filled == 9 and winner == 0):
winner = 3
if ( not (winner == 0) ):
if winner == 1:
draw_text(screen, 'Player X has Won. Press any key to restart',
b.get_board_posx(),
b.get_board_posy() - b.get_board_size() * 0.08,
int( round(b.get_board_size() * 0.055) ) )
elif winner == 2:
draw_text(screen, 'Player O has Won. Press any key to restart',
b.get_board_posx(),
b.get_board_posy() - b.get_board_size() * 0.08,
int( round(b.get_board_size() * 0.055) ))
elif winner == 3:
draw_text(screen, 'You both win! Press any key to restart',
b.get_board_posx(),
b.get_board_posy() - b.get_board_size() * 0.08,
int( round(b.get_board_size() * 0.055) ) )
# Reset all changing variables
# and set area states to inactive
if restart:
restart = False
winner = 0
turn = 0
for i in range(len(areas)):
for j in range(len(areas[i])):
areas[i][j].set_state(0)
# Pygame stuff
pygame.display.update()
clock.tick(FPS)
clicked = False
##################
# END OF GAME LOOP
##################
# closing the pygame application
pygame.quit()
def draw_text(screen, text, posx, posy, size):
my_font = pygame.font.SysFont('Arial', size)
textsurface = my_font.render(text, False, (255, 255, 255))
screen.blit(textsurface, (posx, posy))
def draw_x_token(posx, posy, size, screen, color):
offset = size * 0.1
thickness = size * 0.1
points0 = [
(posx + offset, posy + offset),
(posx + offset + thickness, posy + offset),
(posx + size - offset, posy + size - offset),
(posx + size - (offset + thickness), posy + size - offset)]
pygame.draw.polygon(screen, color, points0, 0 )
points1 = [
( posx + size - (offset + thickness), posy + offset),
( posx + size - offset, posy + offset),
( posx + offset + thickness, posy + size - offset),
( posx + offset, posy + size - offset)]
pygame.draw.polygon(screen, color, points1, 0 )
def draw_o_token(posx, posy, size, screen, color):
offset = size * 0.1
thickness = size * 0.1
radius = size / 2 - offset
pygame.draw.circle(
screen,
color,
( int( round(posx + (size / 2), 0) ),
int( round(posy + (size / 2), 0) ) ),
int( round(radius) ),
int( round (thickness) ) )
class board:
def __init__(self, SCREEN_WIDTH, SCREEN_HEIGHT):
self.SCREEN_WIDTH = SCREEN_WIDTH
self.SCREEN_HEIGHT = SCREEN_HEIGHT
self.BOARD_WIDTH = self.BOARD_HEIGHT = 550
self.COLOR = (255, 255, 255)
self.THICKNESS = self.BOARD_WIDTH * 0.025
# The point (posx, posy) is at the top left position
# of the board
# Setting up posx and posy in a way
# that the board is always drawn in the centre
# of the pygame application/screen
self.posx = (self.SCREEN_WIDTH / 2) - (self.BOARD_WIDTH / 2)
self.posy = (self.SCREEN_HEIGHT / 2) - (self.BOARD_HEIGHT / 2)
# Printing positions of the 4 board lines
# for debug purposes
#print( 'Left vertical line {}, {}'.format(
#self.posx + (self.BOARD_WIDTH / 3) - (self.THICKNESS / 2),
#self.posy
#) )
#print( 'Right vertical line {}, {}'.format(
#self.posx + ( ( (self.BOARD_WIDTH / 3) - (self.THICKNESS / 2) ) * 2),
#self.posy
#))
#print( 'Upper horizontal line {}, {}'.format(
#self.posx,
#self.posy + (self.BOARD_HEIGHT / 3) - (self.THICKNESS / 2)
#) )
#print( 'Lower horizontal line {}, {}'.format(
#self.posx,
#self.posy + ( ( (self.BOARD_HEIGHT / 3) - (self.THICKNESS / 2) ) * 2)
#))
# Draws the game board in the middle of the screen
def draw_board(self, screen):
# Draw left vertical line
pygame.draw.rect(screen, self.COLOR, (
self.posx + (self.BOARD_WIDTH / 3) - (self.THICKNESS / 2),
self.posy, self.THICKNESS, self.BOARD_HEIGHT), 0)
# Draw right verical line
pygame.draw.rect(screen, self.COLOR, (
self.posx + ( (self.BOARD_WIDTH / 3) * 2 ),
self.posy, self.THICKNESS, self.BOARD_HEIGHT), 0)
# Draw upper horizontal line
pygame.draw.rect(screen, self.COLOR, (
self.posx,
self.posy + (self.BOARD_HEIGHT / 3) - (self.THICKNESS / 2),
self.BOARD_WIDTH, self.THICKNESS), 0)
# Draw lower horizontal line
pygame.draw.rect(screen, self.COLOR, (
self.posx,
self.posy + ( (self.BOARD_HEIGHT / 3) * 2),
self.BOARD_WIDTH, self.THICKNESS), 0)
# Getters to access board vars
def get_board_posx(self):
return self.posx
def get_board_posy(self):
return self.posy
def get_board_size(self):
return self.BOARD_WIDTH
def get_thickness(self):
return self.THICKNESS
def get_field_size(self):
return (self.BOARD_WIDTH / 3) - (self.THICKNESS / 2)
def get_color(self):
return self.COLOR
# The board is divided into 9 areas:
#
# area 0.0 | area 0.1 | area 0.2
# _______________________________
# area 1.0 | area 1.1 | area 1.2
# _______________________________
# area 2.0 | area 2.1 | area 2.2
class area:
active = False
def __init__(self, posx, posy, size):
self.posx = posx
self.posy = posy
# width and height of an area
self.size = size
# 0 = inactive
# 1 = mouse is hovering over area
# 2 = area is filled with an X
# 3 = area is filled with an O
self.state = 0
# Debug
#print(posx, posy, size)
def __repr__(self):
return 'Area({}, {})'.format(self.posx, self.posy)
# Function tests if Point(x, y) lies in the Area
# Typically used with mouseposition
def is_in(self, x, y):
if (
(x >= self.posx) and # Top left limit
(x <= (self.posx + self.size)) and # Top right limit
(y >= self.posy) and # Bottom left limit
(y <= (self.posy + self.size)) # Bottom right limit
):
return True
else:
return False
# only for debug
def print_pos(self):
print(self.posx, self.posy, self.size)
def get_posx(self):
return self.posx
def get_posy(self):
return self.posy
def get_active(self):
return self.active
def change_active(self):
self.active = not self.active
def get_state(self):
return self.state
def set_state(self, state):
if (state == 0 or
state == 1 or
state == 2 or
state == 3):
self.state = state
main()
import pygame



def main():

    # Setup the properties of the game / screen
    pygame.init()
    SCREEN_WIDTH ,SCREEN_HEIGHT = 1280, 720
    FPS = 60
    screen = pygame.display.set_mode( (SCREEN_WIDTH, SCREEN_HEIGHT) )
    pygame.display.set_caption('Tic Tac Toe')
    clock = pygame.time.Clock()
    running = True
    clicked = False
    # 0 = no winner, 1 = player1 won, 2 = player2 won, 3 = tie
    winner = 0
    restart = False
    BG_COLOR = (0, 0, 0)
    b = board(SCREEN_WIDTH, SCREEN_HEIGHT)

    # Counts whose turn it is
    # Even turn = Player1
    # Odd turn = Player2
    turn = 0

    # Fill an array with 9 area objects, for each
    # area of the board
    #
    # area 0.0 | area 0.1 | area 0.2
    # _______________________________
    # area 1.0 | area 1.1 | area 1.2
    # _______________________________
    # area 2.0 | area 2.1 | area 2.2
    areas = []
    for i in range(3):
        row = []
        for j in range(3):
            row.append(area(
            (b.get_board_posx() + (j * ( (b.get_board_size() / 3) +
            (b.get_thickness() /2 ) ) ) ),
            (b.get_board_posy() + (i * ( (b.get_board_size() / 3) +
            (b.get_thickness() / 2 ) ) ) ),
            b.get_field_size()
            ))

        areas.append(row)


    ###########
    # GAME LOOP
    ###########
    while running:

        # Abort condition
        # If (x) is clicked, terminate program
        # If clicked is True, the user clicked in this
        # iteration of the game loop
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                running = False
            elif event.type == pygame.MOUSEBUTTONDOWN:
                clicked = True
                if (not (winner == 0)):
                    restart = True

            # If any key or mousebutton is pressed
            # while game is over, game restarts
            elif event.type == pygame.KEYDOWN:
                if (not winner == 0):
                    restart = True

                if event.key == pygame.K_ESCAPE:
                    running = False




        # Setup screen
        screen.fill(BG_COLOR)
        b.draw_board(screen)
        mouse_posx, mouse_posy = pygame.mouse.get_pos()


        # Looping through every area on the board
        # Most of the game logic happens here
        for i in range(len(areas)):
            for j in range(len(areas[i])):

                # Draw already set X tokens
                if (areas[i][j].get_state() == 2):
                    draw_x_token(
                    areas[i][j].get_posx(),
                    areas[i][j].get_posy(),
                    b.get_field_size(),
                    screen,
                    b.get_color()
                    )

                # Draw already set O tokens
                if (areas[i][j].get_state() == 3):
                    draw_o_token(
                    areas[i][j].get_posx(),
                    areas[i][j].get_posy(),
                    b.get_field_size(),
                    screen,
                    b.get_color()
                    )


                # Checking if mouse is hovering over any
                # of the areas
                # If so, preview token that is about to be drawn
                if(areas[i][j].is_in(mouse_posx, mouse_posy)):
                    # print('Mouse is in Area({}, {})'.format(i, j))


                    # Handling clicks (Draw tokens)
                    if clicked:
                        if (not(areas[i][j].get_state() == 2) and
                        not(areas[i][j].get_state() ==3) and winner == 0 ):

                            if (turn % 2 == 0):
                                draw_x_token(
                                areas[i][j].get_posx(),
                                areas[i][j].get_posy(),
                                b.get_field_size(),
                                screen,
                                b.get_color()
                                )

                                areas[i][j].set_state(2)
                                turn += 1

                            else:
                                draw_o_token(
                                areas[i][j].get_posx(),
                                areas[i][j].get_posy(),
                                b.get_field_size(),
                                screen,
                                b.get_color()
                                )

                                areas[i][j].set_state(3)
                                turn += 1



                    if ( (areas[i][j].get_state() == 0
                    or areas[i][j].get_state() == 1) and winner == 0 ):

                        # even turn: Player1
                        if (turn % 2 == 0):
                            draw_x_token(
                            areas[i][j].get_posx(),
                            areas[i][j].get_posy(),
                            b.get_field_size(),
                            screen,
                            b.get_color()
                            )

                            areas[i][j].set_state(1)

                        else:
                            draw_o_token(
                            areas[i][j].get_posx(),
                            areas[i][j].get_posy(),
                            b.get_field_size(),
                            screen,
                            b.get_color()
                            )

                            areas[i][j].set_state(1)



                # If mouse is not hovering over an area
                # and area is not already used
                # Set state to inactive
                else:
                    if (
                    not(areas[i][j].get_state() == 2) and
                    not(areas[i][j].get_state() ==3)
                    ):
                        areas[i][j].set_state(0)


        # Check if there is a winner
        # after this round
        win_cons = [
        # Horizontal
        [ areas[0][0], areas[0][1], areas[0][2] ],
        [ areas[1][0], areas[1][1], areas[1][2] ],
        [ areas[2][0], areas[2][1], areas[2][2] ],
        # Vertical
        [ areas[0][0], areas[1][0], areas[2][0] ],
        [ areas[0][1], areas[1][1], areas[2][1] ],
        [ areas[0][2], areas[1][2], areas[2][2] ],
        # Diagonal
        [ areas[0][2], areas[1][1], areas[2][0] ],
        [ areas[0][0], areas[1][1], areas[2][2] ]
        ]
        for i in range( len(win_cons) ):
            if (
            win_cons[i][0].get_state() ==
            win_cons[i][1].get_state() ==
            win_cons[i][2].get_state() ==
            2
            ):
                winner = 1

            if (
            win_cons[i][0].get_state() ==
            win_cons[i][1].get_state() ==
            win_cons[i][2].get_state() ==
            3
            ):
                winner = 2

            # Check for tie
            fields_filled = 0
            for i in range(len(areas)):
                for j in range(len(areas[i])):
                    if (
                    areas[i][j].get_state() == 2 or
                    areas[i][j].get_state() == 3
                    ):
                        fields_filled += 1

            if (fields_filled == 9 and winner == 0):
                winner = 3

        if ( not (winner == 0) ):

            if winner == 1:
                draw_text(screen, 'Player X has Won. Press any key to restart',
                b.get_board_posx(),
                b.get_board_posy() - b.get_board_size() * 0.08,
                int( round(b.get_board_size() * 0.055) ) )
            elif winner == 2:
                draw_text(screen, 'Player O has Won. Press any key to restart',
                b.get_board_posx(),
                b.get_board_posy() - b.get_board_size() * 0.08,
                int( round(b.get_board_size() * 0.055) ))
            elif winner == 3:
                draw_text(screen, 'You both win! Press any key to restart',
                b.get_board_posx(),
                b.get_board_posy() - b.get_board_size() * 0.08,
                int( round(b.get_board_size() * 0.055) ) )



        # Reset all changing variables
        # and set area states to inactive
        if restart:
            restart = False
            winner = 0
            turn = 0

            for i in range(len(areas)):
                for j in range(len(areas[i])):
                    areas[i][j].set_state(0)



        # Pygame stuff
        pygame.display.update()
        clock.tick(FPS)
        clicked = False

        ##################
        # END OF GAME LOOP
        ##################

    # closing the pygame application
    pygame.quit()


def draw_text(screen, text, posx, posy, size):
    my_font = pygame.font.SysFont('Arial', size)
    textsurface = my_font.render(text, False, (255, 255, 255))
    screen.blit(textsurface, (posx, posy))

def draw_x_token(posx, posy, size, screen, color):
    offset = size * 0.1
    thickness = size * 0.1

    points0 = [
    (posx + offset, posy + offset),
    (posx + offset + thickness, posy + offset),
    (posx + size - offset, posy + size - offset),
    (posx + size - (offset + thickness), posy + size - offset)]

    pygame.draw.polygon(screen, color, points0, 0 )

    points1 = [
    ( posx +  size - (offset + thickness), posy + offset),
    ( posx + size - offset, posy + offset),
    ( posx + offset + thickness, posy + size - offset),
    ( posx + offset, posy + size - offset)]

    pygame.draw.polygon(screen, color, points1, 0 )

def draw_o_token(posx, posy, size, screen, color):
    offset = size * 0.1
    thickness = size * 0.1
    radius = size / 2 - offset

    pygame.draw.circle(
    screen,
    color,
    ( int( round(posx + (size / 2), 0) ),
    int( round(posy + (size / 2), 0) ) ),
    int( round(radius) ),
    int( round (thickness) ) )


class board:

    def __init__(self, SCREEN_WIDTH, SCREEN_HEIGHT):
        self.SCREEN_WIDTH = SCREEN_WIDTH
        self.SCREEN_HEIGHT = SCREEN_HEIGHT

        self.BOARD_WIDTH = self.BOARD_HEIGHT = 550
        self.COLOR = (255, 255, 255)
        self.THICKNESS = self.BOARD_WIDTH * 0.025

        # The point (posx, posy) is at the top left position
        # of the board
        # Setting up posx and posy in a way
        # that the board is always drawn in the centre
        # of the pygame application/screen
        self.posx = (self.SCREEN_WIDTH / 2) - (self.BOARD_WIDTH / 2)
        self.posy = (self.SCREEN_HEIGHT / 2) - (self.BOARD_HEIGHT / 2)

        # Printing positions of the 4 board lines
        # for debug purposes
        #print( 'Left vertical line {}, {}'.format(
        #self.posx + (self.BOARD_WIDTH / 3) - (self.THICKNESS / 2),
        #self.posy
        #) )

        #print( 'Right vertical line {}, {}'.format(
        #self.posx + ( ( (self.BOARD_WIDTH / 3) - (self.THICKNESS / 2) ) * 2),
        #self.posy
        #))

        #print( 'Upper horizontal line {}, {}'.format(
        #self.posx,
        #self.posy + (self.BOARD_HEIGHT / 3) - (self.THICKNESS / 2)
        #) )

        #print( 'Lower horizontal line {}, {}'.format(
        #self.posx,
        #self.posy + ( ( (self.BOARD_HEIGHT / 3) - (self.THICKNESS / 2) ) * 2)
        #))


    # Draws the game board in the middle of the screen
    def draw_board(self, screen):

        # Draw left vertical  line
        pygame.draw.rect(screen, self.COLOR, (
        self.posx + (self.BOARD_WIDTH / 3) - (self.THICKNESS / 2),
        self.posy, self.THICKNESS, self.BOARD_HEIGHT), 0)

        # Draw right verical line
        pygame.draw.rect(screen, self.COLOR, (
        self.posx + ( (self.BOARD_WIDTH / 3) * 2 ),
        self.posy, self.THICKNESS, self.BOARD_HEIGHT), 0)

        # Draw upper horizontal line
        pygame.draw.rect(screen, self.COLOR, (
        self.posx,
        self.posy + (self.BOARD_HEIGHT / 3) - (self.THICKNESS / 2),
        self.BOARD_WIDTH, self.THICKNESS), 0)

        # Draw lower horizontal line
        pygame.draw.rect(screen, self.COLOR, (
        self.posx,
        self.posy + ( (self.BOARD_HEIGHT / 3) * 2),
        self.BOARD_WIDTH, self.THICKNESS), 0)

    # Getters to access board vars
    def get_board_posx(self):
        return self.posx
    def get_board_posy(self):
        return self.posy
    def get_board_size(self):
        return self.BOARD_WIDTH
    def get_thickness(self):
        return self.THICKNESS
    def get_field_size(self):
        return (self.BOARD_WIDTH / 3) - (self.THICKNESS / 2)
    def get_color(self):
        return self.COLOR

# The board is divided into 9 areas:
#
# area 0.0 | area 0.1 | area 0.2
# _______________________________
# area 1.0 | area 1.1 | area 1.2
# _______________________________
# area 2.0 | area 2.1 | area 2.2
class area:
    active = False


    def __init__(self, posx, posy, size):
        self.posx = posx
        self.posy = posy

        # width and height of an area
        self.size = size

        # 0 = inactive
        # 1 = mouse is hovering over area
        # 2 = area is filled with an X
        # 3 = area is filled with an O
        self.state = 0

        # Debug
        #print(posx, posy, size)

    def __repr__(self):
        return 'Area({}, {})'.format(self.posx, self.posy)

    # Function tests if Point(x, y) lies in the Area
    # Typically used with mouseposition
    def is_in(self, x, y):
        if (
        (x >= self.posx) and                # Top left limit
        (x <= (self.posx + self.size)) and  # Top right limit
        (y >= self.posy) and                # Bottom left limit
        (y <= (self.posy + self.size))      # Bottom right limit
        ):
            return True

        else:
            return False

    # only for debug
    def print_pos(self):
        print(self.posx, self.posy, self.size)



    def get_posx(self):
        return self.posx

    def get_posy(self):
        return self.posy


    def get_active(self):
        return self.active

    def change_active(self):
        self.active = not self.active


    def get_state(self):
        return self.state

    def set_state(self, state):
        if (state == 0 or
        state == 1 or
        state == 2 or
        state == 3):
            self.state = state







main()
