234 lines
5.5 KiB
Python
234 lines
5.5 KiB
Python
from __future__ import annotations
|
|
|
|
import random
|
|
from typing import ClassVar, Any
|
|
|
|
from .config import Config
|
|
from .utils import Utils
|
|
from .storage import Storage
|
|
from .window import Window
|
|
|
|
|
|
class Ant:
|
|
def __init__(self) -> None:
|
|
now = Utils.now()
|
|
self.created = now
|
|
self.updated = now
|
|
self.name = ""
|
|
self.status = ""
|
|
self.method = ""
|
|
self.triumph = 0
|
|
self.hits = 0
|
|
|
|
def to_dict(self) -> dict[str, Any]:
|
|
return {
|
|
"created": self.created,
|
|
"updated": self.updated,
|
|
"name": self.name,
|
|
"status": self.status,
|
|
"method": self.method,
|
|
"hits": self.hits,
|
|
"triumph": self.triumph,
|
|
}
|
|
|
|
def from_dict(self, data: dict[str, Any]) -> None:
|
|
self.created = data["created"]
|
|
self.updated = data["updated"]
|
|
self.name = data["name"]
|
|
self.status = data["status"]
|
|
self.method = data["method"]
|
|
self.hits = data["hits"]
|
|
self.triumph = data["triumph"]
|
|
|
|
def get_name(self) -> str:
|
|
return self.name or "Nameless"
|
|
|
|
def get_age(self) -> str:
|
|
now = Utils.now()
|
|
return Utils.time_ago(self.created, now)
|
|
|
|
def describe(self) -> None:
|
|
Utils.print(f"Name is {self.get_name()}")
|
|
Utils.print(f"It hatched {self.get_age()}")
|
|
|
|
def get_score(self) -> int:
|
|
return self.triumph - self.hits
|
|
|
|
|
|
class Ants:
|
|
ants: ClassVar[list[Ant]] = []
|
|
|
|
@staticmethod
|
|
def prepare() -> None:
|
|
Ants.get_ants()
|
|
|
|
@staticmethod
|
|
def hatch(num: int = 1) -> None:
|
|
from .game import Game
|
|
|
|
if len(Ants.ants) >= Config.max_ants:
|
|
Window.alert("Max ants reached\nTerminate some to hatch new ones")
|
|
return
|
|
|
|
now = Utils.now()
|
|
|
|
for _ in range(num):
|
|
ant = Ant()
|
|
ant.created = now
|
|
ant.updated = now
|
|
ant.name = Ants.random_name()
|
|
|
|
Ants.ants.append(ant)
|
|
image_path = Config.hatched_image_path
|
|
Game.add_message("Hatched", f"{ant.name} is born", image_path)
|
|
|
|
if len(Ants.ants) >= Config.max_ants:
|
|
break
|
|
|
|
Ants.save()
|
|
Game.update_info()
|
|
|
|
@staticmethod
|
|
def hatch_burst() -> None:
|
|
Ants.hatch(Config.hatch_burst)
|
|
|
|
@staticmethod
|
|
def terminate() -> None:
|
|
from .game import Game
|
|
|
|
if Ants.empty():
|
|
return
|
|
|
|
ant = Ants.random_ant()
|
|
|
|
if not ant:
|
|
return
|
|
|
|
Ants.ants.remove(ant)
|
|
Ants.save()
|
|
|
|
image_path = Config.terminated_image_path
|
|
Game.add_message("Terminated", f"{ant.name} is gone", image_path)
|
|
Game.update_info()
|
|
|
|
@staticmethod
|
|
def terminate_all() -> None:
|
|
from .game import Game
|
|
|
|
def action() -> None:
|
|
Ants.ants = []
|
|
Ants.save()
|
|
Window.clear_view()
|
|
Game.update_info()
|
|
|
|
Window.confirm("Terminate all ants?", action)
|
|
|
|
@staticmethod
|
|
def random_ant() -> Ant | None:
|
|
if Ants.empty():
|
|
return None
|
|
|
|
return random.choice(Ants.ants)
|
|
|
|
@staticmethod
|
|
def get_names() -> list[str]:
|
|
return [ant.name for ant in Ants.ants]
|
|
|
|
@staticmethod
|
|
def save() -> None:
|
|
Storage.save_ants(Ants.ants)
|
|
|
|
@staticmethod
|
|
def empty() -> bool:
|
|
return len(Ants.ants) == 0
|
|
|
|
@staticmethod
|
|
def get_next() -> Ant | None:
|
|
if Ants.empty():
|
|
return None
|
|
|
|
now = Utils.now()
|
|
ages = [(now - ant.updated) for ant in Ants.ants]
|
|
|
|
# Normalize ages to create weights
|
|
total_age = sum(ages)
|
|
|
|
if total_age == 0:
|
|
weights = [1] * len(Ants.ants) # If all ages are zero, use equal weights
|
|
else:
|
|
weights = [
|
|
int(age / total_age * 1000) for age in ages
|
|
] # Scale and cast to int
|
|
|
|
# Perform weighted random selection
|
|
return random.choices(Ants.ants, weights=weights, k=1)[0]
|
|
|
|
@staticmethod
|
|
def get_current() -> Ant | None:
|
|
if Ants.empty():
|
|
return None
|
|
|
|
return max(Ants.ants, key=lambda ant: ant.updated)
|
|
|
|
@staticmethod
|
|
def set_status(ant: Ant, status: str, method: str) -> None:
|
|
from .game import Game
|
|
|
|
status = status.strip()
|
|
ant.status = status
|
|
ant.method = method
|
|
ant.updated = Utils.now()
|
|
|
|
Game.add_status(ant)
|
|
Ants.save()
|
|
|
|
@staticmethod
|
|
def get_other(ant: Ant) -> Ant:
|
|
ants = [a for a in Ants.ants if a.name != ant.name]
|
|
return random.choice(ants)
|
|
|
|
@staticmethod
|
|
def get_ants() -> None:
|
|
objs = Storage.get_ants()
|
|
changed = False
|
|
|
|
if len(objs) > Config.max_ants:
|
|
objs = objs[: Config.max_ants]
|
|
changed = True
|
|
|
|
for obj in objs:
|
|
ant = Ant()
|
|
ant.from_dict(obj)
|
|
Ants.ants.append(ant)
|
|
|
|
if changed:
|
|
Ants.save()
|
|
|
|
@staticmethod
|
|
def random_name() -> str:
|
|
return Utils.random_name(Ants.get_names())
|
|
|
|
@staticmethod
|
|
def get_top_ant() -> tuple[Ant, int] | None:
|
|
if Ants.empty():
|
|
return None
|
|
|
|
top = None
|
|
top_score = 0
|
|
|
|
# This could be a one-liner but I might expand the algorithm later
|
|
for ant in Ants.ants:
|
|
score = ant.get_score()
|
|
|
|
if score <= 0:
|
|
continue
|
|
|
|
if (not top) or (score > top_score):
|
|
top = ant
|
|
top_score = score
|
|
|
|
if not top:
|
|
return None
|
|
|
|
return top, top_score
|