mirror of
https://github.com/csd4ni3l/game-of-life.git
synced 2026-01-01 04:23:42 +01:00
Optimize game of life implementation to bitwise operations to achieve 450-500 fps for a glider gun, use int instead of math.ceil for positions and fix generation fps reset after file manager
This commit is contained in:
39
game/game_of_life.py
Normal file
39
game/game_of_life.py
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
from utils.constants import ROWS, COLS, NEIGHBORS
|
||||||
|
TOTAL_CELLS = ROWS * COLS
|
||||||
|
|
||||||
|
def get_index(row, col):
|
||||||
|
return row * COLS + col
|
||||||
|
|
||||||
|
def get_neighbors(cell_grid, neighbor_mask):
|
||||||
|
return (cell_grid & neighbor_mask).bit_count()
|
||||||
|
|
||||||
|
def unset_bit(number, i):
|
||||||
|
return number & ~(1 << i)
|
||||||
|
|
||||||
|
def set_bit(number, i):
|
||||||
|
return number | (1 << i)
|
||||||
|
|
||||||
|
def get_bit(number, i):
|
||||||
|
return (number >> i) & 1
|
||||||
|
|
||||||
|
def print_bits(n: int, width: int = 8):
|
||||||
|
print(f"{n:0{width}b}")
|
||||||
|
|
||||||
|
def create_zeroed_int(n):
|
||||||
|
zero_val = 0
|
||||||
|
bitmask = (1 << n) -1
|
||||||
|
return zero_val & bitmask
|
||||||
|
|
||||||
|
def precompute_neighbor_masks():
|
||||||
|
masks = [0] * TOTAL_CELLS
|
||||||
|
for row in range(ROWS):
|
||||||
|
for col in range(COLS):
|
||||||
|
index = get_index(row, col)
|
||||||
|
mask = 0
|
||||||
|
for dy, dx in NEIGHBORS:
|
||||||
|
ny, nx = row + dy, col + dx
|
||||||
|
if 0 <= ny < ROWS and 0 <= nx < COLS:
|
||||||
|
neighbor_index = get_index(ny, nx)
|
||||||
|
mask |= 1 << neighbor_index
|
||||||
|
masks[index] = mask
|
||||||
|
return masks
|
||||||
107
game/play.py
107
game/play.py
@@ -1,20 +1,25 @@
|
|||||||
import arcade, arcade.gui, random, math, copy, time, json, os
|
import arcade, arcade.gui, random, time, json, os
|
||||||
from game.file_support import load_file
|
from game.file_support import load_file
|
||||||
from utils.constants import COLS, ROWS, CELL_SIZE, SPACING, NEIGHBORS, button_style
|
from utils.constants import COLS, ROWS, CELL_SIZE, SPACING, button_style
|
||||||
from utils.preload import create_sound, destroy_sound, button_texture, button_hovered_texture
|
from utils.preload import create_sound, destroy_sound, button_texture, button_hovered_texture, NEIGHBOUR_MASKS
|
||||||
|
from game.game_of_life import get_index, get_bit, get_neighbors, set_bit, unset_bit, create_zeroed_int
|
||||||
|
|
||||||
class Game(arcade.gui.UIView):
|
class Game(arcade.gui.UIView):
|
||||||
def __init__(self, pypresence_client=None, generation=None, running=False, cell_grid=None, load_from=None):
|
def __init__(self, pypresence_client=None, generation=None, running=False, cell_grid=None, generation_fps=10, load_from=None):
|
||||||
super().__init__()
|
super().__init__()
|
||||||
|
|
||||||
self.generation = generation or 0
|
self.generation = generation or 0
|
||||||
self.population = 0
|
self.population = 0
|
||||||
self.running = running or False
|
self.running = running or False
|
||||||
self.generation_fps = 10
|
self.cell_grid = cell_grid or 0
|
||||||
self.cell_grid = cell_grid or {}
|
|
||||||
self.sprite_grid = {}
|
self.sprite_grid = {}
|
||||||
self.load_from = load_from
|
self.load_from = load_from
|
||||||
|
|
||||||
self.pypresence_generation_count = 0
|
self.pypresence_generation_count = 0
|
||||||
|
self.generation_fps = generation_fps
|
||||||
|
self.generation_time = 1 / self.generation_fps
|
||||||
|
self.generation_delta_time = 1 / self.generation_fps
|
||||||
|
self.last_generation_update = time.perf_counter()
|
||||||
|
|
||||||
self.pypresence_client = pypresence_client
|
self.pypresence_client = pypresence_client
|
||||||
self.spritelist = arcade.SpriteList()
|
self.spritelist = arcade.SpriteList()
|
||||||
@@ -26,6 +31,8 @@ class Game(arcade.gui.UIView):
|
|||||||
with open("settings.json", "r") as file:
|
with open("settings.json", "r") as file:
|
||||||
self.settings_dict = json.load(file)
|
self.settings_dict = json.load(file)
|
||||||
|
|
||||||
|
arcade.schedule(self.update_generation, 1 / self.generation_fps)
|
||||||
|
|
||||||
def on_show_view(self):
|
def on_show_view(self):
|
||||||
super().on_show_view()
|
super().on_show_view()
|
||||||
self.setup_grid(load_existing=bool(self.cell_grid))
|
self.setup_grid(load_existing=bool(self.cell_grid))
|
||||||
@@ -36,12 +43,15 @@ class Game(arcade.gui.UIView):
|
|||||||
self.population_label = arcade.gui.UILabel(text="Population: 0", font_name="Roboto", font_size=16)
|
self.population_label = arcade.gui.UILabel(text="Population: 0", font_name="Roboto", font_size=16)
|
||||||
self.info_box.add(self.population_label)
|
self.info_box.add(self.population_label)
|
||||||
|
|
||||||
self.generation_label = arcade.gui.UILabel(text="Generation: 0", font_name="Roboto", font_size=16)
|
self.generation_label = arcade.gui.UILabel(text=f"Generation: {self.generation}", font_name="Roboto", font_size=16)
|
||||||
self.info_box.add(self.generation_label)
|
self.info_box.add(self.generation_label)
|
||||||
|
|
||||||
self.fps_label = arcade.gui.UILabel(text="FPS: 10", font_name="Roboto", font_size=16)
|
self.fps_label = arcade.gui.UILabel(text=f"FPS: {self.generation_fps}", font_name="Roboto", font_size=16)
|
||||||
self.info_box.add(self.fps_label)
|
self.info_box.add(self.fps_label)
|
||||||
|
|
||||||
|
self.actual_fps_label = arcade.gui.UILabel(text=f"Actual FPS: 0", font_name="Roboto", font_size=16)
|
||||||
|
self.info_box.add(self.actual_fps_label)
|
||||||
|
|
||||||
self.back_button = arcade.gui.UITextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text='<--', style=button_style, width=100, height=50)
|
self.back_button = arcade.gui.UITextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text='<--', style=button_style, width=100, height=50)
|
||||||
self.back_button.on_click = lambda event: self.main_exit()
|
self.back_button.on_click = lambda event: self.main_exit()
|
||||||
self.anchor.add(self.back_button, anchor_x="left", anchor_y="top", align_x=5, align_y=-5)
|
self.anchor.add(self.back_button, anchor_x="left", anchor_y="top", align_x=5, align_y=-5)
|
||||||
@@ -54,8 +64,6 @@ class Game(arcade.gui.UIView):
|
|||||||
self.save_button.on_click = lambda event: self.save()
|
self.save_button.on_click = lambda event: self.save()
|
||||||
self.anchor.add(self.save_button, anchor_x="right", anchor_y="bottom", align_x=-5, align_y=5)
|
self.anchor.add(self.save_button, anchor_x="right", anchor_y="bottom", align_x=-5, align_y=5)
|
||||||
|
|
||||||
arcade.schedule(self.update_generation, 1 / self.generation_fps)
|
|
||||||
|
|
||||||
def main_exit(self):
|
def main_exit(self):
|
||||||
from menus.main import Main
|
from menus.main import Main
|
||||||
self.window.show_view(Main(self.pypresence_client))
|
self.window.show_view(Main(self.pypresence_client))
|
||||||
@@ -66,30 +74,28 @@ class Game(arcade.gui.UIView):
|
|||||||
if self.load_from:
|
if self.load_from:
|
||||||
loaded_data = load_file(COLS / 2, ROWS / 2, self.load_from)
|
loaded_data = load_file(COLS / 2, ROWS / 2, self.load_from)
|
||||||
|
|
||||||
|
self.cell_grid = create_zeroed_int(ROWS * COLS)
|
||||||
|
|
||||||
for row in range(ROWS):
|
for row in range(ROWS):
|
||||||
if not row in self.cell_grid:
|
self.sprite_grid[row] = {}
|
||||||
self.cell_grid[row] = {}
|
|
||||||
|
|
||||||
if not row in self.sprite_grid:
|
|
||||||
self.sprite_grid[row] = {}
|
|
||||||
|
|
||||||
for col in range(COLS):
|
for col in range(COLS):
|
||||||
if self.load_from:
|
if self.load_from:
|
||||||
self.cell_grid[row][col] = 1 if (row, col) in loaded_data else 0
|
if (row, col) in loaded_data:
|
||||||
|
self.cell_grid = set_bit(self.cell_grid, get_index(row, col))
|
||||||
elif not load_existing:
|
elif not load_existing:
|
||||||
if randomized and random.randint(0, 1):
|
if randomized and random.randint(0, 1):
|
||||||
self.cell_grid[row][col] = 1
|
self.cell_grid = set_bit(self.cell_grid, get_index(row, col))
|
||||||
self.population += 1
|
self.population += 1
|
||||||
continue
|
continue
|
||||||
|
|
||||||
self.cell_grid[row][col] = 0
|
|
||||||
|
|
||||||
cell = arcade.SpriteSolidColor(CELL_SIZE, CELL_SIZE, center_x=self.start_x + col * (CELL_SIZE + SPACING), center_y=self.start_y + row * (CELL_SIZE + SPACING), color=arcade.color.WHITE)
|
cell = arcade.SpriteSolidColor(CELL_SIZE, CELL_SIZE, center_x=self.start_x + col * (CELL_SIZE + SPACING), center_y=self.start_y + row * (CELL_SIZE + SPACING), color=arcade.color.WHITE)
|
||||||
cell.visible = self.cell_grid[row][col]
|
cell.visible = get_bit(self.cell_grid, get_index(row, col))
|
||||||
self.sprite_grid[row][col] = cell
|
self.sprite_grid[row][col] = cell
|
||||||
self.spritelist.append(cell)
|
self.spritelist.append(cell)
|
||||||
|
|
||||||
def update_generation(self, _):
|
def update_generation(self, delta_time):
|
||||||
|
self.generation_delta_time = delta_time
|
||||||
|
|
||||||
if self.running:
|
if self.running:
|
||||||
self.generation += 1
|
self.generation += 1
|
||||||
|
|
||||||
@@ -99,38 +105,28 @@ class Game(arcade.gui.UIView):
|
|||||||
self.pypresence_generation_count = 0
|
self.pypresence_generation_count = 0
|
||||||
self.pypresence_client.update(state='In Game', details=f'Generation: {self.generation} Population: {self.population}', start=self.pypresence_client.start_time)
|
self.pypresence_client.update(state='In Game', details=f'Generation: {self.generation} Population: {self.population}', start=self.pypresence_client.start_time)
|
||||||
|
|
||||||
next_grid = copy.deepcopy(self.cell_grid) # create a copy of the old grid so we dont modify it in-place
|
next_grid = self.cell_grid | 0
|
||||||
|
|
||||||
grid = self.cell_grid
|
|
||||||
|
|
||||||
for x in range(0, COLS):
|
for x in range(0, COLS):
|
||||||
for y in range(0, ROWS):
|
for y in range(0, ROWS):
|
||||||
cell_neighbors = 0
|
index = get_index(y, x)
|
||||||
for neighbor_y, neighbor_x in NEIGHBORS:
|
cell_neighbors = get_neighbors(self.cell_grid, NEIGHBOUR_MASKS[index])
|
||||||
if neighbor_x == 0 and neighbor_y == 0:
|
|
||||||
continue
|
|
||||||
|
|
||||||
if grid.get(y + neighbor_y, {}).get(x + neighbor_x):
|
if get_bit(self.cell_grid, index):
|
||||||
cell_neighbors += 1
|
|
||||||
|
|
||||||
if grid[y][x]:
|
|
||||||
if (cell_neighbors == 2 or cell_neighbors == 3):
|
if (cell_neighbors == 2 or cell_neighbors == 3):
|
||||||
pass # survives
|
pass # survives
|
||||||
else: # dies
|
else: # dies
|
||||||
self.population -= 1
|
self.population -= 1
|
||||||
self.sprite_grid[y][x].visible = False
|
self.sprite_grid[y][x].visible = False
|
||||||
next_grid[y][x] = 0
|
next_grid = unset_bit(next_grid, index)
|
||||||
|
|
||||||
elif cell_neighbors == 3: # newborn
|
elif cell_neighbors == 3: # newborn
|
||||||
self.population += 1
|
self.population += 1
|
||||||
self.sprite_grid[y][x].visible = True
|
self.sprite_grid[y][x].visible = True
|
||||||
next_grid[y][x] = 1
|
next_grid = set_bit(next_grid, index)
|
||||||
|
|
||||||
self.cell_grid = next_grid
|
self.cell_grid = next_grid
|
||||||
|
|
||||||
self.population_label.text = f"Population: {self.population}"
|
|
||||||
self.generation_label.text = f"Generation: {self.generation}"
|
|
||||||
|
|
||||||
def on_key_press(self, symbol: int, modifiers: int) -> bool | None:
|
def on_key_press(self, symbol: int, modifiers: int) -> bool | None:
|
||||||
super().on_key_press(symbol, modifiers)
|
super().on_key_press(symbol, modifiers)
|
||||||
|
|
||||||
@@ -143,7 +139,7 @@ class Game(arcade.gui.UIView):
|
|||||||
self.population_label.text = f"Population: {self.population}"
|
self.population_label.text = f"Population: {self.population}"
|
||||||
self.generation_label.text = f"Generation: {self.generation}"
|
self.generation_label.text = f"Generation: {self.generation}"
|
||||||
|
|
||||||
self.cell_grid.clear()
|
self.cell_grid = 0
|
||||||
self.spritelist.clear()
|
self.spritelist.clear()
|
||||||
self.sprite_grid.clear()
|
self.sprite_grid.clear()
|
||||||
|
|
||||||
@@ -154,23 +150,32 @@ class Game(arcade.gui.UIView):
|
|||||||
def on_update(self, delta_time):
|
def on_update(self, delta_time):
|
||||||
super().on_update(delta_time)
|
super().on_update(delta_time)
|
||||||
|
|
||||||
|
self.actual_fps_label.text = f"Actual FPS: {round(1 / self.generation_delta_time, 2)}"
|
||||||
|
self.population_label.text = f"Population: {self.population}"
|
||||||
|
self.generation_label.text = f"Generation: {self.generation}"
|
||||||
|
|
||||||
if self.window.keyboard[arcade.key.UP] or self.window.keyboard[arcade.key.DOWN]: # type: ignore
|
if self.window.keyboard[arcade.key.UP] or self.window.keyboard[arcade.key.DOWN]: # type: ignore
|
||||||
self.generation_fps += 1 if self.window.keyboard[arcade.key.UP] else -1 # type: ignore
|
self.generation_fps += 1 if self.window.keyboard[arcade.key.UP] else -1 # type: ignore
|
||||||
|
|
||||||
if self.generation_fps < 1:
|
if self.generation_fps < 1:
|
||||||
self.generation_fps = 1
|
self.generation_fps = 1
|
||||||
|
|
||||||
|
self.generation_time = 1 / self.generation_fps
|
||||||
self.fps_label.text = f"FPS: {self.generation_fps}"
|
self.fps_label.text = f"FPS: {self.generation_fps}"
|
||||||
|
|
||||||
arcade.unschedule(self.update_generation)
|
arcade.unschedule(self.update_generation)
|
||||||
arcade.schedule(self.update_generation, 1 / self.generation_fps)
|
arcade.schedule(self.update_generation, self.generation_time)
|
||||||
|
|
||||||
if self.window.mouse[arcade.MOUSE_BUTTON_LEFT]: # type: ignore
|
if self.window.mouse[arcade.MOUSE_BUTTON_LEFT]: # type: ignore
|
||||||
grid_col = math.ceil((self.window.mouse.data["x"] - self.start_x + (CELL_SIZE / 2)) // (CELL_SIZE + SPACING)) # type: ignore
|
grid_col = int((self.window.mouse.data["x"] - self.start_x + (CELL_SIZE / 2)) // (CELL_SIZE + SPACING)) # type: ignore
|
||||||
grid_row = math.ceil((self.window.mouse.data["y"] - self.start_y + (CELL_SIZE / 2)) // (CELL_SIZE + SPACING)) # type: ignore
|
grid_row = int((self.window.mouse.data["y"] - self.start_y + (CELL_SIZE / 2)) // (CELL_SIZE + SPACING)) # type: ignore
|
||||||
|
|
||||||
if grid_col < 0 or grid_row < 0 or grid_row >= ROWS or grid_col >= COLS:
|
if grid_col < 0 or grid_row < 0 or grid_row >= ROWS or grid_col >= COLS:
|
||||||
return
|
return
|
||||||
|
|
||||||
if self.cell_grid[grid_row][grid_col] == 0:
|
index = get_index(grid_row, grid_col)
|
||||||
|
|
||||||
|
if not get_bit(self.cell_grid, index):
|
||||||
self.population += 1
|
self.population += 1
|
||||||
|
|
||||||
if time.perf_counter() - self.last_create_sound >= 0.05:
|
if time.perf_counter() - self.last_create_sound >= 0.05:
|
||||||
@@ -179,31 +184,33 @@ class Game(arcade.gui.UIView):
|
|||||||
create_sound.play(volume=self.settings_dict.get("sfx_volume", 50) / 100)
|
create_sound.play(volume=self.settings_dict.get("sfx_volume", 50) / 100)
|
||||||
|
|
||||||
self.sprite_grid[grid_row][grid_col].visible = True
|
self.sprite_grid[grid_row][grid_col].visible = True
|
||||||
self.cell_grid[grid_row][grid_col] = 1
|
self.cell_grid = set_bit(self.cell_grid, index)
|
||||||
|
|
||||||
elif self.window.mouse[arcade.MOUSE_BUTTON_RIGHT]: # type: ignore
|
elif self.window.mouse[arcade.MOUSE_BUTTON_RIGHT]: # type: ignore
|
||||||
grid_col = math.ceil((self.window.mouse.data["x"] - self.start_x + (CELL_SIZE / 2)) // (CELL_SIZE + SPACING)) # type: ignore
|
grid_col = int((self.window.mouse.data["x"] - self.start_x + (CELL_SIZE / 2)) // (CELL_SIZE + SPACING)) # type: ignore
|
||||||
grid_row = math.ceil((self.window.mouse.data["y"] - self.start_y + (CELL_SIZE / 2)) // (CELL_SIZE + SPACING)) # type: ignore
|
grid_row = int((self.window.mouse.data["y"] - self.start_y + (CELL_SIZE / 2)) // (CELL_SIZE + SPACING)) # type: ignore
|
||||||
|
|
||||||
if grid_col < 0 or grid_row < 0 or grid_row >= ROWS or grid_col >= COLS:
|
if grid_col < 0 or grid_row < 0 or grid_row >= ROWS or grid_col >= COLS:
|
||||||
return
|
return
|
||||||
|
|
||||||
if self.cell_grid[grid_row][grid_col]:
|
index = get_index(grid_row, grid_col)
|
||||||
|
|
||||||
|
if get_bit(self.cell_grid, index):
|
||||||
self.population -= 1
|
self.population -= 1
|
||||||
if self.settings_dict.get("sfx", True):
|
if self.settings_dict.get("sfx", True):
|
||||||
destroy_sound.play(volume=self.settings_dict.get("sfx_volume", 50) / 100)
|
destroy_sound.play(volume=self.settings_dict.get("sfx_volume", 50) / 100)
|
||||||
self.sprite_grid[grid_row][grid_col].visible = False
|
self.sprite_grid[grid_row][grid_col].visible = False
|
||||||
self.cell_grid[grid_row][grid_col] = 0
|
self.cell_grid = unset_bit(self.cell_grid, index)
|
||||||
|
|
||||||
def load(self):
|
def load(self):
|
||||||
arcade.unschedule(self.update_generation)
|
arcade.unschedule(self.update_generation)
|
||||||
from game.file_manager import FileManager
|
from game.file_manager import FileManager
|
||||||
self.window.show_view(FileManager(os.path.expanduser("~"), [".txt", ".rle"], False, self.pypresence_client, self.generation, self.running, self.cell_grid))
|
self.window.show_view(FileManager(os.path.expanduser("~"), [".txt", ".rle"], False, self.pypresence_client, self.generation, self.running, self.cell_grid, self.generation_fps))
|
||||||
|
|
||||||
def save(self):
|
def save(self):
|
||||||
arcade.unschedule(self.update_generation)
|
arcade.unschedule(self.update_generation)
|
||||||
from game.file_manager import FileManager
|
from game.file_manager import FileManager
|
||||||
self.window.show_view(FileManager(os.path.expanduser("~"), [".txt", ".rle"], True, self.pypresence_client, self.generation, self.running, self.cell_grid))
|
self.window.show_view(FileManager(os.path.expanduser("~"), [".txt", ".rle"], True, self.pypresence_client, self.generation, self.running, self.cell_grid, self.generation_fps))
|
||||||
|
|
||||||
def on_draw(self):
|
def on_draw(self):
|
||||||
super().on_draw()
|
super().on_draw()
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ COLS = 80
|
|||||||
ROWS = 60
|
ROWS = 60
|
||||||
CELL_SIZE = 10
|
CELL_SIZE = 10
|
||||||
SPACING = 2
|
SPACING = 2
|
||||||
NEIGHBORS = [(-1, 0), (-1, 1), (-1, -1),(0, 0), (0, 1), (0, -1), (1, 0), (1, 1), (1, -1)]
|
NEIGHBORS = [(-1, 0), (-1, 1), (-1, -1), (0, 1), (0, -1), (1, 0), (1, 1), (1, -1)]
|
||||||
|
|
||||||
discord_presence_id = 1363780625928028200
|
discord_presence_id = 1363780625928028200
|
||||||
log_dir = 'logs'
|
log_dir = 'logs'
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import arcade.gui, arcade
|
import arcade.gui, arcade
|
||||||
|
from game.game_of_life import precompute_neighbor_masks
|
||||||
|
|
||||||
button_texture = arcade.gui.NinePatchTexture(64 // 4, 64 // 4, 64 // 4, 64 // 4, arcade.load_texture("assets/graphics/button.png"))
|
button_texture = arcade.gui.NinePatchTexture(64 // 4, 64 // 4, 64 // 4, 64 // 4, arcade.load_texture("assets/graphics/button.png"))
|
||||||
button_hovered_texture = arcade.gui.NinePatchTexture(64 // 4, 64 // 4, 64 // 4, 64 // 4, arcade.load_texture("assets/graphics/button_hovered.png"))
|
button_hovered_texture = arcade.gui.NinePatchTexture(64 // 4, 64 // 4, 64 // 4, 64 // 4, arcade.load_texture("assets/graphics/button_hovered.png"))
|
||||||
@@ -6,3 +7,5 @@ button_hovered_texture = arcade.gui.NinePatchTexture(64 // 4, 64 // 4, 64 // 4,
|
|||||||
create_sound = arcade.Sound("assets/sound/create.mp3")
|
create_sound = arcade.Sound("assets/sound/create.mp3")
|
||||||
destroy_sound = arcade.Sound("assets/sound/destroy.mp3")
|
destroy_sound = arcade.Sound("assets/sound/destroy.mp3")
|
||||||
theme_sound = arcade.Sound("assets/sound/music.mp3")
|
theme_sound = arcade.Sound("assets/sound/music.mp3")
|
||||||
|
|
||||||
|
NEIGHBOUR_MASKS = precompute_neighbor_masks()
|
||||||
|
|||||||
Reference in New Issue
Block a user