#!/usr/bin/env python # John Conway's "Game of Life" using a GUI. # Copyright (C) 2018 PySimpleGUI.org # GUI provided by PySimpleGUI. # Core game engine provied by Christian Jacobs # An implementation of Conway's Game of Life in Python. # Copyright (C) 2013 Christian Jacobs. # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # You should have received a copy of the GNU General Public License # along with this program. If not, see . import numpy import PySimpleGUI as sg BOX_SIZE = 15 class GameOfLife: def __init__(self, N=20, T=200): """ Set up Conway's Game of Life. """ # Here we create two grids to hold the old and new configurations. # This assumes an N*N grid of points. # Each point is either alive or dead, represented by integer values of 1 and 0, respectively. self.N = N self.old_grid = numpy.zeros(N * N, dtype='i').reshape(N, N) self.new_grid = numpy.zeros(N * N, dtype='i').reshape(N, N) self.T = T # The maximum number of generations # Set up a random initial configuration for the grid. for i in range(0, self.N): for j in range(0, self.N): self.old_grid[i][j] = 0 self.init_graphics() self.manual_board_setup() def live_neighbours(self, i, j): """ Count the number of live neighbours around point (i, j). """ s = 0 # The total number of live neighbours. # Loop over all the neighbours. for x in [i - 1, i, i + 1]: for y in [j - 1, j, j + 1]: if (x == i and y == j): continue # Skip the current point itself - we only want to count the neighbours! if (x != self.N and y != self.N): s += self.old_grid[x][y] # The remaining branches handle the case where the neighbour is off the end of the grid. # In this case, we loop back round such that the grid becomes a "toroidal array". elif (x == self.N and y != self.N): s += self.old_grid[0][y] elif (x != self.N and y == self.N): s += self.old_grid[x][0] else: s += self.old_grid[0][0] return s def play(self): """ Play Conway's Game of Life. """ # Write the initial configuration to file. self.t = 1 # Current time level while self.t <= self.T: # Evolve! # print( "At time level %d" % t) # Loop over each cell of the grid and apply Conway's rules. for i in range(self.N): for j in range(self.N): live = self.live_neighbours(i, j) if (self.old_grid[i][j] == 1 and live < 2): self.new_grid[i][j] = 0 # Dead from starvation. elif (self.old_grid[i][j] == 1 and (live == 2 or live == 3)): self.new_grid[i][j] = 1 # Continue living. elif (self.old_grid[i][j] == 1 and live > 3): self.new_grid[i][j] = 0 # Dead from overcrowding. elif (self.old_grid[i][j] == 0 and live == 3): self.new_grid[i][j] = 1 # Alive from reproduction. # Output the new configuration. # The new configuration becomes the old configuration for the next generation. self.old_grid = self.new_grid.copy() self.draw_board() # Move on to the next time level self.t += 1 def init_graphics(self): self.graph = sg.Graph((600, 600), (0, 0), (450, 450), key='-GRAPH-', change_submits=True, drag_submits=False, background_color='lightblue') layout = [ [sg.Text('Game of Life', font='ANY 15'), sg.Text('Click below to place cells', key='-OUTPUT-', size=(30, 1), font='ANY 15')], [self.graph], [sg.Button('Go!', key='-DONE-'), sg.Text(' Delay (ms)'), sg.Slider((0, 800), 100, orientation='h', key='-SLIDER-', enable_events=True, size=(15, 15)), sg.Text('', size=(3, 1), key='-S1-OUT-'), sg.Text(' Num Generations'), sg.Slider([0, 20000], default_value=4000, orientation='h', size=(15, 15), enable_events=True, key='-SLIDER2-'), sg.Text('', size=(3, 1), key='-S2-OUT-')] ] self.window = sg.Window('John Conway\'s Game of Life', layout, finalize=True) event, values = self.window.read(timeout=0) self.delay = values['-SLIDER-'] self.window['-S1-OUT-'].update(values['-SLIDER-']) self.window['-S2-OUT-'].update(values['-SLIDER2-']) def draw_board(self): BOX_SIZE = 15 self.graph.erase() for i in range(self.N): for j in range(self.N): if self.old_grid[i][j]: self.graph.draw_rectangle((i * BOX_SIZE, j * BOX_SIZE), (i * BOX_SIZE + BOX_SIZE, j * (BOX_SIZE) + BOX_SIZE), line_color='black', fill_color='yellow') event, values = self.window.read(timeout=self.delay) if event in (None, '-DONE-'): sg.popup('Click OK to exit the program...') self.window.close() exit() self.delay = values['-SLIDER-'] self.T = int(values['-SLIDER2-']) self.window['-S1-OUT-'].update(values['-SLIDER-']) self.window['-S2-OUT-'].update(values['-SLIDER2-']) self.window['-OUTPUT-'].update('Generation {}'.format(self.t)) def manual_board_setup(self): ids = [] for i in range(self.N): ids.append([]) for j in range(self.N): ids[i].append(0) while True: # Event Loop event, values = self.window.read() if event is None or event == '-DONE-': break self.window['-S1-OUT-'].update(values['-SLIDER-']) self.window['-S2-OUT-'].update(values['-SLIDER2-']) mouse = values['-GRAPH-'] if event == '-GRAPH-': if mouse == (None, None): continue box_x = mouse[0] // BOX_SIZE box_y = mouse[1] // BOX_SIZE if self.old_grid[box_x][box_y] == 1: id_val = ids[box_x][box_y] self.graph.delete_figure(id_val) self.old_grid[box_x][box_y] = 0 else: id_val = self.graph.draw_rectangle((box_x * BOX_SIZE, box_y * BOX_SIZE), (box_x * BOX_SIZE + BOX_SIZE, box_y * (BOX_SIZE) + BOX_SIZE), line_color='black', fill_color='yellow') ids[box_x][box_y] = id_val self.old_grid[box_x][box_y] = 1 if event is None: self.window.close() else: self.window['-DONE-'].update(text='Exit') if (__name__ == "__main__"): game = GameOfLife(N=35, T=200) game.play() sg.popup('Completed running.', 'Click OK to exit the program') game.window.close()