2020-11-27 15:33:17 +00:00
|
|
|
# Copyright (C) 2020 by ÿnérant, eichhornchen, nicomarg, charlse
|
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
2020-11-27 19:42:19 +00:00
|
|
|
|
2020-11-27 17:09:08 +00:00
|
|
|
from json import JSONDecodeError
|
2020-11-10 20:41:54 +00:00
|
|
|
from random import randint
|
2020-11-18 13:56:59 +00:00
|
|
|
from typing import Any, Optional
|
2020-12-11 15:56:22 +00:00
|
|
|
import curses
|
2020-11-16 00:01:18 +00:00
|
|
|
import json
|
|
|
|
import os
|
2020-11-18 13:56:59 +00:00
|
|
|
import sys
|
2020-11-06 14:33:26 +00:00
|
|
|
|
2020-11-06 16:48:47 +00:00
|
|
|
from .entities.player import Player
|
2020-11-11 22:48:46 +00:00
|
|
|
from .enums import GameMode, KeyValues, DisplayActions
|
2020-11-19 11:03:05 +00:00
|
|
|
from .interfaces import Map, Logs
|
2020-11-19 01:49:59 +00:00
|
|
|
from .resources import ResourceManager
|
2020-11-06 13:59:27 +00:00
|
|
|
from .settings import Settings
|
2020-11-06 17:06:28 +00:00
|
|
|
from . import menus
|
2020-11-28 00:59:52 +00:00
|
|
|
from .translations import gettext as _, Translator
|
2020-11-06 16:24:20 +00:00
|
|
|
|
2020-11-06 17:11:59 +00:00
|
|
|
|
2020-10-23 12:53:08 +00:00
|
|
|
class Game:
|
2020-11-18 11:19:27 +00:00
|
|
|
"""
|
|
|
|
The game object controls all actions in the game.
|
|
|
|
"""
|
2020-11-08 22:26:54 +00:00
|
|
|
map: Map
|
|
|
|
player: Player
|
2020-12-03 23:27:25 +00:00
|
|
|
screen: Any
|
2020-11-13 17:08:48 +00:00
|
|
|
# display_actions is a display interface set by the bootstrapper
|
2020-12-11 15:56:22 +00:00
|
|
|
display_actions: callable
|
2020-11-08 22:26:54 +00:00
|
|
|
|
2020-11-06 17:39:55 +00:00
|
|
|
def __init__(self) -> None:
|
2020-11-08 22:31:17 +00:00
|
|
|
"""
|
2020-12-13 20:29:25 +00:00
|
|
|
Initiates the game.
|
2020-11-08 22:31:17 +00:00
|
|
|
"""
|
2020-11-10 18:40:59 +00:00
|
|
|
self.state = GameMode.MAINMENU
|
2020-12-07 20:29:57 +00:00
|
|
|
self.waiting_for_friendly_key = False
|
2020-12-18 00:05:50 +00:00
|
|
|
self.is_in_store_menu = True
|
2020-11-06 13:59:27 +00:00
|
|
|
self.settings = Settings()
|
|
|
|
self.settings.load_settings()
|
|
|
|
self.settings.write_settings()
|
2020-11-28 00:59:52 +00:00
|
|
|
Translator.setlocale(self.settings.LOCALE)
|
2020-11-27 21:19:41 +00:00
|
|
|
self.main_menu = menus.MainMenu()
|
|
|
|
self.settings_menu = menus.SettingsMenu()
|
2020-11-27 20:51:54 +00:00
|
|
|
self.settings_menu.update_values(self.settings)
|
2020-12-04 13:41:59 +00:00
|
|
|
self.inventory_menu = menus.InventoryMenu()
|
2020-12-07 19:54:53 +00:00
|
|
|
self.store_menu = menus.StoreMenu()
|
2020-11-19 11:03:05 +00:00
|
|
|
self.logs = Logs()
|
2020-11-27 17:09:08 +00:00
|
|
|
self.message = None
|
2020-10-23 12:53:08 +00:00
|
|
|
|
2020-11-10 17:08:06 +00:00
|
|
|
def new_game(self) -> None:
|
2020-11-08 22:31:17 +00:00
|
|
|
"""
|
2020-12-13 20:29:25 +00:00
|
|
|
Creates a new game on the screen.
|
2020-11-08 22:31:17 +00:00
|
|
|
"""
|
2020-10-23 16:01:39 +00:00
|
|
|
# TODO generate a new map procedurally
|
2020-12-18 00:50:11 +00:00
|
|
|
self.map = Map.load(ResourceManager.get_asset_path("example_map_2.txt"))
|
2020-11-19 11:03:05 +00:00
|
|
|
self.map.logs = self.logs
|
2020-11-19 11:55:06 +00:00
|
|
|
self.logs.clear()
|
2020-10-23 16:01:39 +00:00
|
|
|
self.player = Player()
|
2020-11-08 22:26:54 +00:00
|
|
|
self.map.add_entity(self.player)
|
2020-11-11 15:58:20 +00:00
|
|
|
self.player.move(self.map.start_y, self.map.start_x)
|
2020-11-11 15:23:27 +00:00
|
|
|
self.map.spawn_random_entities(randint(3, 10))
|
2020-12-04 15:28:37 +00:00
|
|
|
self.inventory_menu.update_player(self.player)
|
2020-10-23 16:01:39 +00:00
|
|
|
|
2020-12-18 15:40:52 +00:00
|
|
|
def run(self, screen: Any) -> None: # pragma no cover
|
2020-11-08 22:31:17 +00:00
|
|
|
"""
|
|
|
|
Main infinite loop.
|
2020-12-13 20:29:25 +00:00
|
|
|
We wait for the player's action, then we do what should be done
|
|
|
|
when a key gets pressed.
|
2020-11-08 22:31:17 +00:00
|
|
|
"""
|
2020-12-18 15:40:52 +00:00
|
|
|
screen.refresh()
|
|
|
|
while True:
|
2020-11-26 21:20:14 +00:00
|
|
|
screen.erase()
|
2020-12-18 15:40:52 +00:00
|
|
|
screen.noutrefresh()
|
2020-11-11 22:48:46 +00:00
|
|
|
self.display_actions(DisplayActions.REFRESH)
|
2020-12-18 15:40:52 +00:00
|
|
|
curses.doupdate()
|
2020-10-23 12:53:08 +00:00
|
|
|
key = screen.getkey()
|
2020-12-11 17:17:59 +00:00
|
|
|
if key == "KEY_MOUSE":
|
|
|
|
_ignored1, x, y, _ignored2, _ignored3 = curses.getmouse()
|
|
|
|
self.display_actions(DisplayActions.MOUSE, y, x)
|
|
|
|
else:
|
|
|
|
self.handle_key_pressed(
|
|
|
|
KeyValues.translate_key(key, self.settings), key)
|
2020-11-06 16:24:20 +00:00
|
|
|
|
2020-11-11 21:45:15 +00:00
|
|
|
def handle_key_pressed(self, key: Optional[KeyValues], raw_key: str = '')\
|
|
|
|
-> None:
|
2020-11-08 22:31:17 +00:00
|
|
|
"""
|
2020-12-13 20:29:25 +00:00
|
|
|
Indicates what should be done when a given key is pressed,
|
2020-11-08 22:31:17 +00:00
|
|
|
according to the current game state.
|
|
|
|
"""
|
2020-11-27 16:35:51 +00:00
|
|
|
if self.message:
|
|
|
|
self.message = None
|
2020-11-27 17:12:27 +00:00
|
|
|
self.display_actions(DisplayActions.REFRESH)
|
2020-11-27 16:35:51 +00:00
|
|
|
return
|
|
|
|
|
2020-11-06 17:06:28 +00:00
|
|
|
if self.state == GameMode.PLAY:
|
2020-12-07 20:29:57 +00:00
|
|
|
if self.waiting_for_friendly_key:
|
|
|
|
# The player requested to talk with a friendly entity
|
|
|
|
self.handle_friendly_entity_chat(key)
|
|
|
|
else:
|
|
|
|
self.handle_key_pressed_play(key)
|
2020-12-04 13:57:53 +00:00
|
|
|
elif self.state == GameMode.INVENTORY:
|
|
|
|
self.handle_key_pressed_inventory(key)
|
2020-11-08 22:26:54 +00:00
|
|
|
elif self.state == GameMode.MAINMENU:
|
2020-11-16 00:01:18 +00:00
|
|
|
self.handle_key_pressed_main_menu(key)
|
2020-11-08 22:26:54 +00:00
|
|
|
elif self.state == GameMode.SETTINGS:
|
2020-11-11 21:36:42 +00:00
|
|
|
self.settings_menu.handle_key_pressed(key, raw_key, self)
|
2020-12-07 19:54:53 +00:00
|
|
|
elif self.state == GameMode.STORE:
|
|
|
|
self.handle_key_pressed_store(key)
|
2020-12-18 21:24:41 +00:00
|
|
|
elif self.state == GameMode.CREDITS:
|
|
|
|
self.state = GameMode.MAINMENU
|
2020-11-11 22:48:46 +00:00
|
|
|
self.display_actions(DisplayActions.REFRESH)
|
2020-11-08 22:26:54 +00:00
|
|
|
|
|
|
|
def handle_key_pressed_play(self, key: KeyValues) -> None:
|
2020-11-08 22:31:17 +00:00
|
|
|
"""
|
2020-11-18 11:19:27 +00:00
|
|
|
In play mode, arrows or zqsd move the main character.
|
2020-11-08 22:31:17 +00:00
|
|
|
"""
|
2020-11-08 22:26:54 +00:00
|
|
|
if key == KeyValues.UP:
|
2020-11-10 23:38:02 +00:00
|
|
|
if self.player.move_up():
|
2020-12-18 16:29:59 +00:00
|
|
|
self.map.tick(self.player)
|
2020-11-08 22:26:54 +00:00
|
|
|
elif key == KeyValues.DOWN:
|
2020-11-10 23:38:02 +00:00
|
|
|
if self.player.move_down():
|
2020-12-18 16:29:59 +00:00
|
|
|
self.map.tick(self.player)
|
2020-11-08 22:26:54 +00:00
|
|
|
elif key == KeyValues.LEFT:
|
2020-11-10 23:38:02 +00:00
|
|
|
if self.player.move_left():
|
2020-12-18 16:29:59 +00:00
|
|
|
self.map.tick(self.player)
|
2020-11-08 22:26:54 +00:00
|
|
|
elif key == KeyValues.RIGHT:
|
2020-11-10 23:38:02 +00:00
|
|
|
if self.player.move_right():
|
2020-12-18 16:29:59 +00:00
|
|
|
self.map.tick(self.player)
|
2020-12-04 13:41:59 +00:00
|
|
|
elif key == KeyValues.INVENTORY:
|
|
|
|
self.state = GameMode.INVENTORY
|
2020-11-08 22:26:54 +00:00
|
|
|
elif key == KeyValues.SPACE:
|
|
|
|
self.state = GameMode.MAINMENU
|
2020-12-09 14:32:37 +00:00
|
|
|
elif key == KeyValues.CHAT:
|
2020-12-07 20:29:57 +00:00
|
|
|
# Wait for the direction of the friendly entity
|
|
|
|
self.waiting_for_friendly_key = True
|
2020-12-12 17:12:37 +00:00
|
|
|
elif key == KeyValues.WAIT:
|
2020-12-18 16:29:59 +00:00
|
|
|
self.map.tick(self.player)
|
2020-12-07 20:22:06 +00:00
|
|
|
|
2020-12-07 20:29:57 +00:00
|
|
|
def handle_friendly_entity_chat(self, key: KeyValues) -> None:
|
|
|
|
"""
|
2020-12-13 20:29:25 +00:00
|
|
|
If the player tries to talk to a friendly entity, the game waits for
|
|
|
|
a directional key to be pressed, verifies there is a friendly entity
|
|
|
|
in that direction and then lets the player interact with it.
|
2020-12-07 20:29:57 +00:00
|
|
|
"""
|
|
|
|
if not self.waiting_for_friendly_key:
|
|
|
|
return
|
|
|
|
self.waiting_for_friendly_key = False
|
|
|
|
|
|
|
|
if key == KeyValues.UP:
|
2020-12-07 20:22:06 +00:00
|
|
|
xp = self.player.x
|
|
|
|
yp = self.player.y - 1
|
2020-12-07 20:29:57 +00:00
|
|
|
elif key == KeyValues.DOWN:
|
2020-12-07 20:22:06 +00:00
|
|
|
xp = self.player.x
|
|
|
|
yp = self.player.y + 1
|
2020-12-07 20:29:57 +00:00
|
|
|
elif key == KeyValues.LEFT:
|
2020-12-07 20:22:06 +00:00
|
|
|
xp = self.player.x - 1
|
|
|
|
yp = self.player.y
|
2020-12-07 20:29:57 +00:00
|
|
|
elif key == KeyValues.RIGHT:
|
2020-12-07 20:22:06 +00:00
|
|
|
xp = self.player.x + 1
|
|
|
|
yp = self.player.y
|
|
|
|
else:
|
|
|
|
return
|
|
|
|
if self.map.entity_is_present(yp, xp):
|
|
|
|
for entity in self.map.entities:
|
|
|
|
if entity.is_friendly() and entity.x == xp and \
|
|
|
|
entity.y == yp:
|
|
|
|
msg = entity.talk_to(self.player)
|
|
|
|
self.logs.add_message(msg)
|
|
|
|
if entity.is_merchant():
|
|
|
|
self.state = GameMode.STORE
|
2020-12-18 00:50:11 +00:00
|
|
|
self.is_in_store_menu = True
|
2020-12-07 20:22:06 +00:00
|
|
|
self.store_menu.update_merchant(entity)
|
2020-12-18 14:15:47 +00:00
|
|
|
self.display_actions(DisplayActions.UPDATE)
|
2020-11-16 00:01:18 +00:00
|
|
|
|
2020-12-04 13:57:53 +00:00
|
|
|
def handle_key_pressed_inventory(self, key: KeyValues) -> None:
|
|
|
|
"""
|
|
|
|
In the inventory menu, we can interact with items or close the menu.
|
|
|
|
"""
|
|
|
|
if key == KeyValues.SPACE or key == KeyValues.INVENTORY:
|
|
|
|
self.state = GameMode.PLAY
|
2020-12-04 15:31:15 +00:00
|
|
|
elif key == KeyValues.UP:
|
|
|
|
self.inventory_menu.go_up()
|
|
|
|
elif key == KeyValues.DOWN:
|
|
|
|
self.inventory_menu.go_down()
|
2020-12-05 12:20:52 +00:00
|
|
|
if self.inventory_menu.values and not self.player.dead:
|
2020-12-04 16:01:00 +00:00
|
|
|
if key == KeyValues.USE:
|
|
|
|
self.inventory_menu.validate().use()
|
|
|
|
elif key == KeyValues.EQUIP:
|
|
|
|
self.inventory_menu.validate().equip()
|
|
|
|
elif key == KeyValues.DROP:
|
2020-12-04 16:19:06 +00:00
|
|
|
self.inventory_menu.validate().drop()
|
2020-12-04 16:01:00 +00:00
|
|
|
|
|
|
|
# Ensure that the cursor has a good position
|
|
|
|
self.inventory_menu.position = min(self.inventory_menu.position,
|
|
|
|
len(self.inventory_menu.values)
|
|
|
|
- 1)
|
2020-12-04 13:57:53 +00:00
|
|
|
|
2020-12-07 19:54:53 +00:00
|
|
|
def handle_key_pressed_store(self, key: KeyValues) -> None:
|
2020-12-06 10:43:48 +00:00
|
|
|
"""
|
2020-12-07 19:54:53 +00:00
|
|
|
In a store menu, we can buy items or close the menu.
|
2020-12-06 10:43:48 +00:00
|
|
|
"""
|
2020-12-18 00:05:50 +00:00
|
|
|
menu = self.store_menu if self.is_in_store_menu else self.inventory_menu
|
|
|
|
|
|
|
|
if key == KeyValues.SPACE or key == KeyValues.INVENTORY:
|
2020-12-06 10:43:48 +00:00
|
|
|
self.state = GameMode.PLAY
|
|
|
|
elif key == KeyValues.UP:
|
2020-12-18 00:05:50 +00:00
|
|
|
menu.go_up()
|
2020-12-06 10:43:48 +00:00
|
|
|
elif key == KeyValues.DOWN:
|
2020-12-18 00:05:50 +00:00
|
|
|
menu.go_down()
|
|
|
|
elif key == KeyValues.LEFT:
|
|
|
|
self.is_in_store_menu = False
|
2020-12-18 14:15:47 +00:00
|
|
|
self.display_actions(DisplayActions.UPDATE)
|
2020-12-18 00:05:50 +00:00
|
|
|
elif key == KeyValues.RIGHT:
|
|
|
|
self.is_in_store_menu = True
|
2020-12-18 14:15:47 +00:00
|
|
|
self.display_actions(DisplayActions.UPDATE)
|
2020-12-18 00:05:50 +00:00
|
|
|
if menu.values and not self.player.dead:
|
2020-12-07 19:54:53 +00:00
|
|
|
if key == KeyValues.ENTER:
|
2020-12-18 00:05:50 +00:00
|
|
|
item = menu.validate()
|
|
|
|
owner = self.store_menu.merchant if self.is_in_store_menu \
|
|
|
|
else self.player
|
|
|
|
buyer = self.player if self.is_in_store_menu \
|
|
|
|
else self.store_menu.merchant
|
|
|
|
flag = item.be_sold(buyer, owner)
|
2020-12-11 16:06:30 +00:00
|
|
|
if not flag:
|
2020-12-18 00:05:50 +00:00
|
|
|
self.message = _("The buyer does not have enough money")
|
2020-12-18 14:15:47 +00:00
|
|
|
self.display_actions(DisplayActions.UPDATE)
|
2020-12-06 10:43:48 +00:00
|
|
|
# Ensure that the cursor has a good position
|
2020-12-18 00:05:50 +00:00
|
|
|
menu.position = min(menu.position, len(menu.values) - 1)
|
2020-12-06 10:43:48 +00:00
|
|
|
|
2020-11-16 00:01:18 +00:00
|
|
|
def handle_key_pressed_main_menu(self, key: KeyValues) -> None:
|
|
|
|
"""
|
2020-12-13 20:29:25 +00:00
|
|
|
In the main menu, we can navigate through different options.
|
2020-11-16 00:01:18 +00:00
|
|
|
"""
|
|
|
|
if key == KeyValues.DOWN:
|
|
|
|
self.main_menu.go_down()
|
|
|
|
if key == KeyValues.UP:
|
|
|
|
self.main_menu.go_up()
|
|
|
|
if key == KeyValues.ENTER:
|
|
|
|
option = self.main_menu.validate()
|
|
|
|
if option == menus.MainMenuValues.START:
|
2020-11-19 00:32:52 +00:00
|
|
|
self.new_game()
|
|
|
|
self.display_actions(DisplayActions.UPDATE)
|
|
|
|
self.state = GameMode.PLAY
|
|
|
|
if option == menus.MainMenuValues.RESUME:
|
2020-11-16 00:01:18 +00:00
|
|
|
self.state = GameMode.PLAY
|
|
|
|
elif option == menus.MainMenuValues.SAVE:
|
|
|
|
self.save_game()
|
|
|
|
elif option == menus.MainMenuValues.LOAD:
|
|
|
|
self.load_game()
|
|
|
|
elif option == menus.MainMenuValues.SETTINGS:
|
|
|
|
self.state = GameMode.SETTINGS
|
|
|
|
elif option == menus.MainMenuValues.EXIT:
|
|
|
|
sys.exit(0)
|
|
|
|
|
2020-11-18 13:56:59 +00:00
|
|
|
def save_state(self) -> dict:
|
2020-11-18 11:19:27 +00:00
|
|
|
"""
|
2020-12-13 20:29:25 +00:00
|
|
|
Saves the game to a dictionary.
|
2020-11-18 11:19:27 +00:00
|
|
|
"""
|
2020-11-16 00:01:18 +00:00
|
|
|
return self.map.save_state()
|
|
|
|
|
|
|
|
def load_state(self, d: dict) -> None:
|
2020-11-18 11:19:27 +00:00
|
|
|
"""
|
2020-12-13 20:29:25 +00:00
|
|
|
Loads the game from a dictionary.
|
2020-11-18 11:19:27 +00:00
|
|
|
"""
|
2020-11-27 17:16:54 +00:00
|
|
|
try:
|
|
|
|
self.map.load_state(d)
|
|
|
|
except KeyError:
|
2020-11-27 19:42:19 +00:00
|
|
|
self.message = _("Some keys are missing in your save file.\n"
|
|
|
|
"Your save seems to be corrupt. It got deleted.")
|
2020-11-27 17:16:54 +00:00
|
|
|
os.unlink(ResourceManager.get_config_path("save.json"))
|
|
|
|
self.display_actions(DisplayActions.UPDATE)
|
|
|
|
return
|
|
|
|
|
2020-11-27 17:09:08 +00:00
|
|
|
players = self.map.find_entities(Player)
|
|
|
|
if not players:
|
2020-11-27 19:42:19 +00:00
|
|
|
self.message = _("No player was found on this map!\n"
|
|
|
|
"Maybe you died?")
|
2020-11-27 17:09:08 +00:00
|
|
|
self.player.health = 0
|
|
|
|
self.display_actions(DisplayActions.UPDATE)
|
|
|
|
return
|
|
|
|
|
|
|
|
self.player = players[0]
|
2020-11-18 14:04:15 +00:00
|
|
|
self.display_actions(DisplayActions.UPDATE)
|
2020-11-18 13:56:59 +00:00
|
|
|
|
2020-11-16 00:01:18 +00:00
|
|
|
def load_game(self) -> None:
|
|
|
|
"""
|
2020-12-13 20:29:25 +00:00
|
|
|
Loads the game from a file.
|
2020-11-16 00:01:18 +00:00
|
|
|
"""
|
2020-11-19 02:13:01 +00:00
|
|
|
file_path = ResourceManager.get_config_path("save.json")
|
|
|
|
if os.path.isfile(file_path):
|
|
|
|
with open(file_path, "r") as f:
|
2020-11-27 17:09:08 +00:00
|
|
|
try:
|
|
|
|
state = json.loads(f.read())
|
|
|
|
self.load_state(state)
|
|
|
|
except JSONDecodeError:
|
2020-11-27 19:42:19 +00:00
|
|
|
self.message = _("The JSON file is not correct.\n"
|
2020-11-27 21:21:16 +00:00
|
|
|
"Your save seems corrupted. "
|
2020-11-27 19:42:19 +00:00
|
|
|
"It got deleted.")
|
2020-11-27 17:09:08 +00:00
|
|
|
os.unlink(file_path)
|
|
|
|
self.display_actions(DisplayActions.UPDATE)
|
2020-11-16 00:01:18 +00:00
|
|
|
|
|
|
|
def save_game(self) -> None:
|
|
|
|
"""
|
2020-12-13 20:29:25 +00:00
|
|
|
Saves the game to a file.
|
2020-11-16 00:01:18 +00:00
|
|
|
"""
|
2020-11-19 02:13:01 +00:00
|
|
|
with open(ResourceManager.get_config_path("save.json"), "w") as f:
|
2020-11-16 00:01:18 +00:00
|
|
|
f.write(json.dumps(self.save_state()))
|