Files
basic-computer-games/15_Boxing/python/boxing.py
2024-08-19 03:46:27 +03:00

181 lines
5.3 KiB
Python
Executable File

#!/usr/bin/env python3
import json
import random
from dataclasses import dataclass
from pathlib import Path
from typing import Dict, Literal, NamedTuple, Tuple
class PunchProfile(NamedTuple):
choices: int
threshold: int
hit_damage: int
block_damage: int
pre_msg: str
hit_msg: str
blocked_msg: str
knockout_possible: bool = False
def is_hit(self) -> bool:
return random.randint(1, self.choices) <= self.threshold
@dataclass
class Player:
name: str
best: int # this hit guarantees 2 damage on opponent
weakness: int # you're always hit when your opponent uses this punch
is_computer: bool
# for each of the 4 punch types, we have a probability of hitting
punch_profiles: Dict[Literal[1, 2, 3, 4], PunchProfile]
damage: int = 0
score: int = 0
knockedout: bool = False
def get_punch_choice(self) -> Literal[1, 2, 3, 4]:
if self.is_computer:
return random.randint(1, 4) # type: ignore
punch = -1
while punch not in [1, 2, 3, 4]:
print(f"{self.name}'S PUNCH", end="? ")
punch = int(input())
return punch # type: ignore
KNOCKOUT_THRESHOLD = 35
QUESTION_PROMPT = "? "
KNOCKED_COLD = "{loser} IS KNOCKED COLD AND {winner} IS THE WINNER AND CHAMP"
def get_vulnerability() -> int:
print("WHAT IS HIS VULNERABILITY", end=QUESTION_PROMPT)
return int(input())
def get_opponent_stats() -> Tuple[int, int]:
opponent_best = 0
opponent_weakness = 0
while opponent_best == opponent_weakness:
opponent_best = random.randint(1, 4)
opponent_weakness = random.randint(1, 4)
return opponent_best, opponent_weakness
def read_punch_profiles(filepath: Path) -> Dict[Literal[1, 2, 3, 4], PunchProfile]:
with open(filepath) as f:
punch_profile_dict = json.load(f)
return {
int(key): PunchProfile(**value)
for key, value in punch_profile_dict.items()
}
def main() -> None:
print("BOXING")
print("CREATIVE COMPUTING MORRISTOWN, NEW JERSEY")
print("\n\n")
print("BOXING OLYMPIC STYLE (3 ROUNDS -- 2 OUT OF 3 WINS)")
print("WHAT IS YOUR OPPONENT'S NAME", end=QUESTION_PROMPT)
opponent_name = input()
print("WHAT IS YOUR MAN'S NAME", end=QUESTION_PROMPT)
player_name = input()
print("DIFFERENT PUNCHES ARE 1 FULL SWING 2 HOOK 3 UPPERCUT 4 JAB")
print("WHAT IS YOUR MAN'S BEST", end=QUESTION_PROMPT)
player_best = int(input()) # noqa: TODO - this likely is a bug!
player_weakness = get_vulnerability()
player = Player(
name=player_name,
best=player_best,
weakness=player_weakness,
is_computer=False,
punch_profiles=read_punch_profiles(
Path(__file__).parent / "player-profile.json"
),
)
opponent_best, opponent_weakness = get_opponent_stats()
opponent = Player(
name=opponent_name,
best=opponent_best,
weakness=opponent_weakness,
is_computer=True,
punch_profiles=read_punch_profiles(
Path(__file__).parent / "opponent-profile.json"
),
)
print(
f"{opponent.name}'S ADVANTAGE is {opponent.weakness} AND VULNERABILITY IS SECRET."
)
for round_number in (1, 2, 3):
play_round(round_number, player, opponent)
if player.knockedout:
print(KNOCKED_COLD.format(loser=player.name, winner=opponent.name))
elif opponent.knockedout:
print(KNOCKED_COLD.format(loser=opponent.name, winner=player.name))
elif opponent.score > player.score:
print(f"{opponent.name} WINS (NICE GOING), {player.name}")
else:
print(f"{player.name} AMAZINGLY WINS")
print("\n\nAND NOW GOODBYE FROM THE OLYMPIC ARENA.")
def is_opponents_turn() -> bool:
return random.randint(1, 10) > 5
def play_round(round_number: int, player: Player, opponent: Player) -> None:
print(f"ROUND {round_number} BEGINS...\n")
if opponent.score >= 2 or player.score >= 2:
return
for _action in range(7):
if is_opponents_turn():
punch = opponent.get_punch_choice()
active = opponent
passive = player
else:
punch = player.get_punch_choice()
active = player
passive = opponent
# Load the hit characteristics of the current player's punch
punch_profile = active.punch_profiles[punch]
if punch == active.best:
passive.damage += 2
print(punch_profile.pre_msg.format(active=active, passive=passive), end=" ")
if passive.weakness == punch or punch_profile.is_hit():
print(punch_profile.hit_msg.format(active=active, passive=passive))
if punch_profile.knockout_possible and passive.damage > KNOCKOUT_THRESHOLD:
passive.knockedout = True
break
passive.damage += punch_profile.hit_damage
else:
print(punch_profile.blocked_msg.format(active=active, passive=passive))
active.damage += punch_profile.block_damage
if player.knockedout or opponent.knockedout:
return
elif player.damage > opponent.damage:
print(f"{opponent.name} WINS ROUND {round_number}")
opponent.score += 1
else:
print(f"{player.name} WINS ROUND {round_number}")
player.score += 1
if __name__ == "__main__":
main()