346 lines
8.7 KiB
Python
346 lines
8.7 KiB
Python
from __future__ import annotations
|
|
|
|
import random
|
|
|
|
from PySide6.QtCore import Qt # type: ignore
|
|
from PySide6.QtWidgets import QHBoxLayout # type: ignore
|
|
from PySide6.QtWidgets import QVBoxLayout
|
|
from PySide6.QtWidgets import QLabel
|
|
from PySide6.QtWidgets import QWidget
|
|
from PySide6.QtGui import QMouseEvent # type: ignore
|
|
from PySide6.QtGui import QPixmap
|
|
from PySide6.QtCore import QTimer
|
|
|
|
from .config import Config
|
|
from .utils import Utils
|
|
from .ants import Ant
|
|
from .ants import Ants
|
|
from .window import Window
|
|
from .settings import Settings
|
|
|
|
|
|
class Method:
|
|
merge = 0
|
|
triumph = 1
|
|
hit = 2
|
|
travel = 3
|
|
thinking_1 = 4
|
|
thinking_2 = 5
|
|
sentence_1 = 6
|
|
sentence_2 = 7
|
|
sentence_3 = 8
|
|
|
|
|
|
class Game:
|
|
timer: QTimer | None = None
|
|
playing_song: bool = False
|
|
last_update: int = 0
|
|
|
|
@staticmethod
|
|
def prepare() -> None:
|
|
Game.initial_fill()
|
|
Game.update_info()
|
|
|
|
@staticmethod
|
|
def add_update(
|
|
ant: Ant,
|
|
) -> None:
|
|
container = QHBoxLayout()
|
|
image_label = Game.get_image(ant)
|
|
right_container = Game.make_right_container(ant)
|
|
|
|
container.addWidget(image_label)
|
|
container.addSpacing(Config.space_1)
|
|
container.addWidget(right_container)
|
|
Game.add_container(container)
|
|
|
|
@staticmethod
|
|
def add_container(container: QHBoxLayout) -> None:
|
|
from .filter import Filter
|
|
|
|
root = QWidget()
|
|
root.setContentsMargins(0, 0, 0, 0)
|
|
container.setContentsMargins(0, 0, 0, 0)
|
|
root.setLayout(container)
|
|
Window.view.insertWidget(0, root)
|
|
|
|
while Window.view.count() > Config.max_updates:
|
|
item = Window.view.takeAt(Window.view.count() - 1)
|
|
|
|
if item.widget():
|
|
item.widget().deleteLater()
|
|
elif item.layout():
|
|
Window.delete_layout(item.layout())
|
|
|
|
Filter.check()
|
|
|
|
@staticmethod
|
|
def make_right_container(ant: Ant) -> QWidget:
|
|
if ant.method == "hatched":
|
|
title = "Hatched"
|
|
message = f"{ant.name} is born"
|
|
elif ant.method == "terminated":
|
|
title = "Terminated"
|
|
message = f"{ant.name} is gone"
|
|
else:
|
|
title = ant.name
|
|
message = ant.get_status()
|
|
|
|
root = QWidget()
|
|
root.setObjectName("view_right")
|
|
container = QVBoxLayout()
|
|
container.setAlignment(Qt.AlignTop)
|
|
|
|
title_label = QLabel(title)
|
|
title_label.setTextInteractionFlags(Qt.TextSelectableByMouse)
|
|
title_label.setStyleSheet("font-weight: bold;")
|
|
title_label.setWordWrap(True)
|
|
title_label.setObjectName("view_title")
|
|
Window.expand(title_label)
|
|
|
|
message_label = QLabel(message)
|
|
message_label.setTextInteractionFlags(Qt.TextSelectableByMouse)
|
|
message_label.setWordWrap(True)
|
|
message_label.setObjectName("view_message")
|
|
Window.expand(message_label)
|
|
|
|
container.addWidget(title_label)
|
|
container.addWidget(message_label)
|
|
root.setLayout(container)
|
|
|
|
return root
|
|
|
|
@staticmethod
|
|
def get_image(
|
|
ant: Ant,
|
|
) -> QLabel:
|
|
if ant.method == "hatched":
|
|
path = Config.hatched_image_path
|
|
elif ant.method == "terminated":
|
|
path = Config.terminated_image_path
|
|
else:
|
|
path = Config.status_image_path
|
|
|
|
if ant.method == "triumph":
|
|
color = Config.triumph_color
|
|
elif ant.method == "hit":
|
|
color = Config.hit_color
|
|
else:
|
|
color = None
|
|
|
|
tooltip = ant.tooltip()
|
|
image_label = QLabel()
|
|
image_label.setObjectName("view_image")
|
|
pixmap = QPixmap(str(path))
|
|
|
|
scaled_pixmap = pixmap.scaled(
|
|
Config.image_size,
|
|
pixmap.height(),
|
|
Qt.KeepAspectRatio,
|
|
Qt.SmoothTransformation,
|
|
)
|
|
|
|
image_label.setPixmap(scaled_pixmap)
|
|
image_label.setFixedSize(scaled_pixmap.size())
|
|
|
|
if color:
|
|
rgb = Utils.get_rgb(color)
|
|
|
|
style = f"""
|
|
QLabel#view_image {{
|
|
border: 2px solid {rgb};
|
|
}}
|
|
"""
|
|
|
|
image_label.setStyleSheet(style)
|
|
|
|
if tooltip:
|
|
image_label.setToolTip(tooltip)
|
|
|
|
image_label.mousePressEvent = lambda event: Game.image_action(event, ant)
|
|
return image_label
|
|
|
|
@staticmethod
|
|
def get_status() -> None:
|
|
ant = Ants.get_next()
|
|
|
|
if not ant:
|
|
return
|
|
|
|
min_num = 0
|
|
max_num = 12
|
|
no_repeat = [Method.merge]
|
|
|
|
num = random.randint(min_num, max_num)
|
|
|
|
if num in no_repeat:
|
|
if Game.last_update == num:
|
|
num = max_num
|
|
|
|
Game.last_update = num
|
|
|
|
if num == Method.merge:
|
|
if Ants.merge():
|
|
return
|
|
|
|
# If merge failed
|
|
num = max_num
|
|
|
|
status = ""
|
|
method = "normal"
|
|
|
|
if num == Method.triumph:
|
|
ant.triumph += 1
|
|
method = "triumph"
|
|
|
|
elif num == Method.hit:
|
|
ant.hits += 1
|
|
method = "hit"
|
|
|
|
elif num == Method.travel:
|
|
status = Utils.random_country([])
|
|
method = "travel"
|
|
|
|
elif num == Method.thinking_1:
|
|
status = Utils.random_name([], Ants.get_names())
|
|
method = "thinking"
|
|
|
|
elif num == Method.thinking_2:
|
|
status = Utils.random_emoji(3)
|
|
method = "thinking"
|
|
|
|
elif num == Method.sentence_1:
|
|
status = Utils.rand_sentence.simple_sentence()
|
|
|
|
elif num == Method.sentence_2:
|
|
status = Utils.rand_sentence.bare_bone_sentence()
|
|
|
|
elif num == Method.sentence_3:
|
|
status = Utils.rand_sentence.bare_bone_with_adjective()
|
|
|
|
else:
|
|
status = Utils.rand_sentence.sentence()
|
|
|
|
Ants.set_status(ant, status, method)
|
|
|
|
@staticmethod
|
|
def initial_fill() -> None:
|
|
if not len(Ants.ants):
|
|
return
|
|
|
|
ants = sorted(Ants.ants, key=lambda ant: ant.updated)
|
|
|
|
for ant in ants:
|
|
Game.add_update(ant)
|
|
|
|
@staticmethod
|
|
def start_loop() -> None:
|
|
if Game.timer:
|
|
Game.timer.stop()
|
|
|
|
speed = Settings.speed
|
|
|
|
if speed == "fast":
|
|
delay = Config.loop_delay_fast
|
|
elif speed == "normal":
|
|
delay = Config.loop_delay_normal
|
|
elif speed == "slow":
|
|
delay = Config.loop_delay_slow
|
|
else:
|
|
return
|
|
|
|
Game.timer = QTimer()
|
|
Game.timer.timeout.connect(Game.get_status)
|
|
Game.timer.start(delay)
|
|
|
|
@staticmethod
|
|
def update_speed() -> None:
|
|
speed = Window.speed.currentText().lower()
|
|
|
|
if speed == Settings.speed:
|
|
return
|
|
|
|
Settings.set_speed(speed)
|
|
Game.start_loop()
|
|
|
|
@staticmethod
|
|
def update_info() -> None:
|
|
text = []
|
|
|
|
# Non-breaking space
|
|
nb = "\u00a0"
|
|
|
|
if not len(Ants.ants):
|
|
text.append("Hatch some ants")
|
|
else:
|
|
text.append(f"Ants:{nb}{len(Ants.ants)}")
|
|
top = Ants.get_top_ant()
|
|
|
|
if top:
|
|
ant = top[0]
|
|
score = top[1]
|
|
text.append(f"Top:{nb}{ant.name} ({score})")
|
|
|
|
Window.info.setText(Config.info_separator.join(text))
|
|
|
|
@staticmethod
|
|
def toggle_song() -> None:
|
|
if Game.playing_song:
|
|
Window.stop_audio()
|
|
Game.playing_song = False
|
|
else:
|
|
path = str(Config.song_path)
|
|
|
|
def on_stop() -> None:
|
|
Game.playing_song = False
|
|
|
|
Window.play_audio(path, on_stop)
|
|
Game.playing_song = True
|
|
|
|
@staticmethod
|
|
def restart() -> None:
|
|
opts = ["25", "50", "100", "250"]
|
|
defindex = 0
|
|
|
|
for i, opt in enumerate(opts):
|
|
if int(opt) == Config.default_population:
|
|
defindex = i
|
|
break
|
|
|
|
opts = [f"{opt} ants" for opt in opts]
|
|
size = Window.prompt_combobox("Size of the population", opts, defindex)
|
|
|
|
if not size:
|
|
return
|
|
|
|
num = int(size.split(" ")[0])
|
|
|
|
Window.clear_view()
|
|
Ants.populate(num)
|
|
Window.to_top()
|
|
Game.start_loop()
|
|
|
|
@staticmethod
|
|
def update_size() -> None:
|
|
pass
|
|
|
|
@staticmethod
|
|
def image_action(event: QMouseEvent, ant: Ant) -> None:
|
|
def is_terminated() -> bool:
|
|
return ant.method == "terminated"
|
|
|
|
if event.button() == Qt.LeftButton:
|
|
if is_terminated():
|
|
return
|
|
|
|
Ants.terminate(ant)
|
|
Game.start_loop()
|
|
elif event.button() == Qt.MiddleButton:
|
|
if is_terminated():
|
|
return
|
|
|
|
Ants.merge(ant)
|
|
Game.start_loop()
|
|
else:
|
|
Game.toggle_song()
|