squirrel-battle/squirrelbattle/entities/monsters.py

125 lines
4.2 KiB
Python
Raw Permalink Normal View History

2021-01-10 09:46:17 +00:00
# Copyright (C) 2020-2021 by ÿnérant, eichhornchen, nicomarg, charlse
2020-11-27 15:33:17 +00:00
# SPDX-License-Identifier: GPL-3.0-or-later
from random import shuffle
2020-11-10 21:59:02 +00:00
from .player import Player
2020-11-06 14:33:26 +00:00
from ..interfaces import FightingEntity, Map
2020-10-23 14:51:48 +00:00
class Monster(FightingEntity):
"""
The class for all monsters in the dungeon.
All specific monster classes overwrite this class,
and the parameters are given in the __init__ function.
An example of the specification of a monster that has a strength of 4
and 20 max HP:
class MyMonster(Monster):
def __init__(self, strength: int = 4, maxhealth: int = 20,
*args, **kwargs) -> None:
super().__init__(name="my_monster", strength=strength,
maxhealth=maxhealth, *args, **kwargs)
With that way, attributes can be overwritten when the entity is created.
"""
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
2020-11-06 15:13:28 +00:00
def act(self, m: Map) -> None:
"""
By default, a monster will move randomly where it is possible
If the player is closeby, the monster runs to the player.
"""
super().act(m)
2020-11-10 21:59:02 +00:00
target = None
for entity in m.entities:
if self.distance_squared(entity) <= 25 and \
isinstance(entity, Player):
target = entity
break
2020-10-23 14:51:48 +00:00
# Monsters move according to a Dijkstra algorithm
# that targets the player.
# If they can not move and are already close to the player,
# they hit.
if target and (self.y, self.x) in target.paths and \
self.map.is_visible_from(self.y, self.x,
target.y, target.x, 5):
# Moves to target player by choosing the best available path
for next_y, next_x in target.paths[(self.y, self.x)]:
moved = self.check_move(next_y, next_x, True)
if moved:
break
if self.distance_squared(target) <= 1:
self.map.logs.add_message(self.hit(target))
break
2020-11-10 21:59:02 +00:00
else:
# Moves in a random direction
# If the direction is not available, tries another one
moves = [self.move_up, self.move_down,
self.move_left, self.move_right]
shuffle(moves)
for move in moves:
if move():
2020-11-10 21:59:02 +00:00
break
2020-12-18 16:46:38 +00:00
def move(self, y: int, x: int) -> None:
"""
Overwrites the move function to recalculate paths.
"""
super().move(y, x)
self.recalculate_paths()
2020-11-06 14:33:26 +00:00
2020-11-20 17:02:08 +00:00
class Tiger(Monster):
"""
A tiger monster.
"""
def __init__(self, name: str = "tiger", strength: int = 5,
maxhealth: int = 30, *args, **kwargs) -> None:
super().__init__(name=name, strength=strength,
maxhealth=maxhealth, *args, **kwargs)
2020-11-11 16:39:48 +00:00
2020-11-10 20:47:36 +00:00
class Hedgehog(Monster):
"""
A really mean hedgehog monster.
"""
def __init__(self, name: str = "hedgehog", strength: int = 3,
maxhealth: int = 10, *args, **kwargs) -> None:
super().__init__(name=name, strength=strength,
maxhealth=maxhealth, *args, **kwargs)
2020-11-11 16:39:48 +00:00
class Rabbit(Monster):
"""
A rabbit monster.
"""
def __init__(self, name: str = "rabbit", strength: int = 1,
maxhealth: int = 20, critical: int = 30,
2021-01-06 17:02:58 +00:00
*args, **kwargs) -> None:
super().__init__(name=name, strength=strength,
2021-01-06 17:02:58 +00:00
maxhealth=maxhealth, critical=critical,
*args, **kwargs)
2020-11-11 16:39:48 +00:00
class TeddyBear(Monster):
"""
A cute teddybear monster.
"""
def __init__(self, name: str = "teddy_bear", strength: int = 0,
maxhealth: int = 50, *args, **kwargs) -> None:
super().__init__(name=name, strength=strength,
maxhealth=maxhealth, *args, **kwargs)
2021-01-06 17:02:58 +00:00
class GiantSeaEagle(Monster):
"""
An eagle boss
"""
def __init__(self, name: str = "eagle", strength: int = 1000,
maxhealth: int = 5000, *args, **kwargs) -> None:
super().__init__(name=name, strength=strength,
maxhealth=maxhealth, *args, **kwargs)