2021-02-21 20:17:24 +01:00
|
|
|
import logging
|
2021-07-12 13:54:47 +02:00
|
|
|
from typing import Set
|
2021-02-21 20:17:24 +01:00
|
|
|
|
|
|
|
logger = logging.getLogger("Hollow Knight")
|
|
|
|
|
2021-02-24 06:02:51 +01:00
|
|
|
from .Locations import lookup_name_to_id
|
2021-07-15 13:31:33 +02:00
|
|
|
from .Items import item_table, lookup_type_to_names
|
2021-02-24 06:02:51 +01:00
|
|
|
from .Regions import create_regions
|
|
|
|
from .Rules import set_rules
|
2021-06-26 11:18:12 -05:00
|
|
|
from .Options import hollow_knight_options
|
2021-02-21 20:17:24 +01:00
|
|
|
|
|
|
|
from BaseClasses import Region, Entrance, Location, MultiWorld, Item
|
2021-07-15 13:31:33 +02:00
|
|
|
from ..AutoWorld import World, LogicMixin
|
2021-02-21 20:17:24 +01:00
|
|
|
|
2021-06-11 14:22:44 +02:00
|
|
|
class HKWorld(World):
|
|
|
|
game: str = "Hollow Knight"
|
2021-06-26 11:18:12 -05:00
|
|
|
options = hollow_knight_options
|
2021-07-12 13:54:47 +02:00
|
|
|
item_names: Set[str] = frozenset(item_table)
|
2021-07-12 15:11:48 +02:00
|
|
|
location_names: Set[str] = frozenset(lookup_name_to_id)
|
2021-06-26 11:18:12 -05:00
|
|
|
|
2021-07-12 18:47:58 +02:00
|
|
|
item_name_to_id = {name: data.id for name, data in item_table.items() if data.type != "Event"}
|
2021-07-12 18:05:46 +02:00
|
|
|
location_name_to_id = lookup_name_to_id
|
|
|
|
|
2021-06-26 11:18:12 -05:00
|
|
|
def generate_basic(self):
|
|
|
|
# Link regions
|
|
|
|
self.world.get_entrance('Hollow Nest S&Q', self.player).connect(self.world.get_region('Hollow Nest', self.player))
|
|
|
|
|
|
|
|
# Generate item pool
|
|
|
|
pool = []
|
|
|
|
for item_name, item_data in item_table.items():
|
2021-07-12 13:54:47 +02:00
|
|
|
item = self.create_item(item_name)
|
2021-06-26 11:18:12 -05:00
|
|
|
|
|
|
|
if item_data.type == "Event":
|
|
|
|
event_location = self.world.get_location(item_name, self.player)
|
|
|
|
self.world.push_item(event_location, item, collect=False)
|
|
|
|
event_location.event = True
|
|
|
|
event_location.locked = True
|
|
|
|
if item.name == "King's_Pass":
|
|
|
|
self.world.push_precollected(item)
|
|
|
|
elif item_data.type == "Cursed":
|
|
|
|
if self.world.CURSED[self.player]:
|
|
|
|
pool.append(item)
|
|
|
|
else:
|
|
|
|
# fill Focus Location with Focus and add it to start inventory as well.
|
|
|
|
event_location = self.world.get_location(item_name, self.player)
|
|
|
|
self.world.push_item(event_location, item)
|
|
|
|
event_location.event = True
|
|
|
|
event_location.locked = True
|
|
|
|
|
|
|
|
elif item_data.type == "Fake":
|
|
|
|
pass
|
|
|
|
elif item_data.type in not_shufflable_types:
|
|
|
|
location = self.world.get_location(item_name, self.player)
|
|
|
|
self.world.push_item(location, item, collect=False)
|
|
|
|
location.event = item.advancement
|
|
|
|
location.locked = True
|
|
|
|
else:
|
|
|
|
target = option_to_type_lookup[item.type]
|
|
|
|
shuffle_it = getattr(self.world, target)
|
|
|
|
if shuffle_it[self.player]:
|
|
|
|
pool.append(item)
|
|
|
|
else:
|
|
|
|
location = self.world.get_location(item_name, self.player)
|
|
|
|
self.world.push_item(location, item, collect=False)
|
|
|
|
location.event = item.advancement
|
|
|
|
location.locked = True
|
|
|
|
logger.debug(f"Placed {item_name} to vanilla for player {self.player}")
|
|
|
|
|
|
|
|
self.world.itempool += pool
|
|
|
|
|
|
|
|
|
|
|
|
def set_rules(self):
|
|
|
|
set_rules(self.world, self.player)
|
|
|
|
|
|
|
|
|
|
|
|
def create_regions(self):
|
|
|
|
create_regions(self.world, self.player)
|
|
|
|
|
|
|
|
|
|
|
|
def generate_output(self):
|
|
|
|
pass # Hollow Knight needs no output files
|
|
|
|
|
|
|
|
|
|
|
|
def fill_slot_data(self):
|
|
|
|
slot_data = {}
|
|
|
|
for option_name in self.options:
|
|
|
|
option = getattr(self.world, option_name)[self.player]
|
|
|
|
slot_data[option_name] = int(option.value)
|
|
|
|
return slot_data
|
|
|
|
|
2021-07-12 13:54:47 +02:00
|
|
|
def create_item(self, name: str) -> Item:
|
|
|
|
item_data = item_table[name]
|
|
|
|
return HKItem(name, item_data.advancement, item_data.id, item_data.type, self.player)
|
2021-02-21 20:17:24 +01:00
|
|
|
|
2021-02-24 06:02:51 +01:00
|
|
|
def create_region(world: MultiWorld, player: int, name: str, locations=None, exits=None):
|
|
|
|
ret = Region(name, None, name, player)
|
|
|
|
ret.world = world
|
|
|
|
if locations:
|
|
|
|
for location in locations:
|
|
|
|
loc_id = lookup_name_to_id.get(location, 0)
|
|
|
|
location = HKLocation(player, location, loc_id, ret)
|
|
|
|
ret.locations.append(location)
|
|
|
|
if exits:
|
|
|
|
for exit in exits:
|
|
|
|
ret.exits.append(Entrance(player, exit, ret))
|
|
|
|
|
|
|
|
return ret
|
|
|
|
|
2021-04-01 11:40:58 +02:00
|
|
|
|
2021-02-21 20:17:24 +01:00
|
|
|
class HKLocation(Location):
|
|
|
|
game: str = "Hollow Knight"
|
|
|
|
|
|
|
|
def __init__(self, player: int, name: str, address=None, parent=None):
|
|
|
|
super(HKLocation, self).__init__(player, name, address, parent)
|
|
|
|
|
2021-04-01 11:40:58 +02:00
|
|
|
|
2021-02-21 20:17:24 +01:00
|
|
|
class HKItem(Item):
|
2021-02-22 11:18:53 +01:00
|
|
|
game = "Hollow Knight"
|
|
|
|
|
2021-03-21 00:47:17 +01:00
|
|
|
def __init__(self, name, advancement, code, type, player: int = None):
|
2021-02-21 20:17:24 +01:00
|
|
|
super(HKItem, self).__init__(name, advancement, code, player)
|
2021-03-21 00:47:17 +01:00
|
|
|
self.type = type
|
2021-02-21 20:17:24 +01:00
|
|
|
|
2021-02-24 06:02:51 +01:00
|
|
|
|
2021-03-21 00:47:17 +01:00
|
|
|
not_shufflable_types = {"Essence_Boss"}
|
|
|
|
|
|
|
|
option_to_type_lookup = {
|
|
|
|
"Root": "RandomizeWhisperingRoots",
|
|
|
|
"Dreamer": "RandomizeDreamers",
|
|
|
|
"Geo": "RandomizeGeoChests",
|
|
|
|
"Skill": "RandomizeSkills",
|
|
|
|
"Map": "RandomizeMaps",
|
|
|
|
"Relic": "RandomizeRelics",
|
|
|
|
"Charm": "RandomizeCharms",
|
|
|
|
"Notch": "RandomizeCharmNotches",
|
|
|
|
"Key": "RandomizeKeys",
|
|
|
|
"Stag": "RandomizeStags",
|
|
|
|
"Flame": "RandomizeFlames",
|
|
|
|
"Grub": "RandomizeGrubs",
|
|
|
|
"Cocoon": "RandomizeLifebloodCocoons",
|
|
|
|
"Mask": "RandomizeMaskShards",
|
|
|
|
"Ore": "RandomizePaleOre",
|
|
|
|
"Egg": "RandomizeRancidEggs",
|
|
|
|
"Vessel": "RandomizeVesselFragments",
|
|
|
|
}
|
2021-02-21 20:17:24 +01:00
|
|
|
|
|
|
|
|
2021-07-15 13:31:33 +02:00
|
|
|
class HKLogic(LogicMixin):
|
|
|
|
# these are all wip
|
|
|
|
def _hk_has_essence(self, player: int, count: int):
|
|
|
|
return self.prog_items["Dream_Nail", player]
|
|
|
|
# return self.prog_items["Essence", player] >= count
|
2021-02-21 20:17:24 +01:00
|
|
|
|
2021-07-15 13:31:33 +02:00
|
|
|
def _hk_has_grubs(self, player: int, count: int):
|
|
|
|
found = 0
|
|
|
|
for item_name in lookup_type_to_names["Grub"]:
|
|
|
|
found += self.prog_items[item_name, player]
|
|
|
|
if found >= count:
|
|
|
|
return True
|
2021-02-21 20:17:24 +01:00
|
|
|
|
2021-07-15 13:31:33 +02:00
|
|
|
return False
|
|
|
|
|
|
|
|
def _hk_has_flames(self, player: int, count: int):
|
|
|
|
found = 0
|
|
|
|
for item_name in lookup_type_to_names["Flame"]:
|
|
|
|
found += self.prog_items[item_name, player]
|
|
|
|
if found >= count:
|
|
|
|
return True
|
|
|
|
|
|
|
|
return False
|