| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | from __future__ import annotations | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |  | import logging | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | import typing | 
					
						
							|  |  |  |  | from collections import Counter | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | logger = logging.getLogger("Hollow Knight") | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-04 00:15:16 +02:00
										 |  |  |  | from .Items import item_table, lookup_type_to_names, item_name_groups | 
					
						
							| 
									
										
										
										
											2021-02-24 06:02:51 +01:00
										 |  |  |  | from .Regions import create_regions | 
					
						
							|  |  |  |  | from .Rules import set_rules | 
					
						
							| 
									
										
										
										
											2022-04-02 21:25:03 +02:00
										 |  |  |  | from .Options import hollow_knight_options, hollow_knight_randomize_options, disabled | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | from .ExtractedData import locations, starts, multi_locations, location_to_region_lookup, \ | 
					
						
							|  |  |  |  |     event_names, item_effects, connectors, one_ways | 
					
						
							| 
									
										
										
										
											2022-04-08 19:22:50 +02:00
										 |  |  |  | from .Charms import names as charm_names | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-18 18:19:21 +01:00
										 |  |  |  | from BaseClasses import Region, Entrance, Location, MultiWorld, Item, RegionType | 
					
						
							| 
									
										
										
										
											2021-07-15 13:31:33 +02:00
										 |  |  |  | from ..AutoWorld import World, LogicMixin | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | white_palace_locations = { | 
					
						
							|  |  |  |  |     "Soul_Totem-Path_of_Pain_Below_Thornskip", | 
					
						
							|  |  |  |  |     "Soul_Totem-White_Palace_Final", | 
					
						
							|  |  |  |  |     "Lore_Tablet-Path_of_Pain_Entrance", | 
					
						
							|  |  |  |  |     "Soul_Totem-Path_of_Pain_Left_of_Lever", | 
					
						
							|  |  |  |  |     "Soul_Totem-Path_of_Pain_Hidden", | 
					
						
							|  |  |  |  |     "Soul_Totem-Path_of_Pain_Entrance", | 
					
						
							|  |  |  |  |     "Soul_Totem-Path_of_Pain_Final", | 
					
						
							|  |  |  |  |     "Soul_Totem-White_Palace_Entrance", | 
					
						
							|  |  |  |  |     "Soul_Totem-Path_of_Pain_Below_Lever", | 
					
						
							|  |  |  |  |     "Lore_Tablet-Palace_Throne", | 
					
						
							|  |  |  |  |     "Soul_Totem-Path_of_Pain_Second", | 
					
						
							|  |  |  |  |     "Soul_Totem-White_Palace_Left", | 
					
						
							|  |  |  |  |     "Lore_Tablet-Palace_Workshop", | 
					
						
							|  |  |  |  |     "Soul_Totem-White_Palace_Hub", | 
					
						
							|  |  |  |  |     "Journal_Entry-Seal_of_Binding", | 
					
						
							|  |  |  |  |     "Soul_Totem-White_Palace_Right", | 
					
						
							|  |  |  |  |     "King_Fragment", | 
					
						
							|  |  |  |  |     # Events: | 
					
						
							|  |  |  |  |     "Palace_Entrance_Lantern_Lit", | 
					
						
							|  |  |  |  |     "Palace_Left_Lantern_Lit", | 
					
						
							|  |  |  |  |     "Palace_Right_Lantern_Lit", | 
					
						
							|  |  |  |  |     "Warp-Path_of_Pain_Complete", | 
					
						
							|  |  |  |  |     "Defeated_Path_of_Pain_Arena", | 
					
						
							|  |  |  |  |     "Palace_Atrium_Gates_Opened", | 
					
						
							|  |  |  |  |     "Completed_Path_of_Pain", | 
					
						
							|  |  |  |  |     "Warp-White_Palace_Atrium_to_Palace_Grounds", | 
					
						
							|  |  |  |  |     "Warp-White_Palace_Entrance_to_Palace_Grounds", | 
					
						
							|  |  |  |  |     # Event-Regions: | 
					
						
							|  |  |  |  |     "White_Palace_03_hub", | 
					
						
							|  |  |  |  |     "White_Palace_13", | 
					
						
							|  |  |  |  |     "White_Palace_01", | 
					
						
							|  |  |  |  |     # Event-Transitions: | 
					
						
							|  |  |  |  |     "White_Palace_12[bot1]", "White_Palace_12[bot1]", "White_Palace_03_hub[bot1]", "White_Palace_16[left2]", | 
					
						
							|  |  |  |  |     "White_Palace_16[left2]", "White_Palace_11[door2]", "White_Palace_11[door2]", "White_Palace_18[top1]", | 
					
						
							|  |  |  |  |     "White_Palace_18[top1]", "White_Palace_15[left1]", "White_Palace_15[left1]", "White_Palace_05[left2]", | 
					
						
							|  |  |  |  |     "White_Palace_05[left2]", "White_Palace_14[bot1]", "White_Palace_14[bot1]", "White_Palace_13[left2]", | 
					
						
							|  |  |  |  |     "White_Palace_13[left2]", "White_Palace_03_hub[left1]", "White_Palace_03_hub[left1]", "White_Palace_15[right2]", | 
					
						
							|  |  |  |  |     "White_Palace_15[right2]", "White_Palace_06[top1]", "White_Palace_06[top1]", "White_Palace_03_hub[bot1]", | 
					
						
							|  |  |  |  |     "White_Palace_08[right1]", "White_Palace_08[right1]", "White_Palace_03_hub[right1]", "White_Palace_03_hub[right1]", | 
					
						
							|  |  |  |  |     "White_Palace_01[right1]", "White_Palace_01[right1]", "White_Palace_08[left1]", "White_Palace_08[left1]", | 
					
						
							|  |  |  |  |     "White_Palace_19[left1]", "White_Palace_19[left1]", "White_Palace_04[right2]", "White_Palace_04[right2]", | 
					
						
							|  |  |  |  |     "White_Palace_01[left1]", "White_Palace_01[left1]", "White_Palace_17[right1]", "White_Palace_17[right1]", | 
					
						
							|  |  |  |  |     "White_Palace_07[bot1]", "White_Palace_07[bot1]", "White_Palace_20[bot1]", "White_Palace_20[bot1]", | 
					
						
							|  |  |  |  |     "White_Palace_03_hub[left2]", "White_Palace_03_hub[left2]", "White_Palace_18[right1]", "White_Palace_18[right1]", | 
					
						
							|  |  |  |  |     "White_Palace_05[right1]", "White_Palace_05[right1]", "White_Palace_17[bot1]", "White_Palace_17[bot1]", | 
					
						
							|  |  |  |  |     "White_Palace_09[right1]", "White_Palace_09[right1]", "White_Palace_16[left1]", "White_Palace_16[left1]", | 
					
						
							|  |  |  |  |     "White_Palace_13[left1]", "White_Palace_13[left1]", "White_Palace_06[bot1]", "White_Palace_06[bot1]", | 
					
						
							|  |  |  |  |     "White_Palace_15[right1]", "White_Palace_15[right1]", "White_Palace_06[left1]", "White_Palace_06[left1]", | 
					
						
							|  |  |  |  |     "White_Palace_05[right2]", "White_Palace_05[right2]", "White_Palace_04[top1]", "White_Palace_04[top1]", | 
					
						
							|  |  |  |  |     "White_Palace_19[top1]", "White_Palace_19[top1]", "White_Palace_14[right1]", "White_Palace_14[right1]", | 
					
						
							|  |  |  |  |     "White_Palace_03_hub[top1]", "White_Palace_03_hub[top1]", "Grubfather_2", "White_Palace_13[left3]", | 
					
						
							|  |  |  |  |     "White_Palace_13[left3]", "White_Palace_02[left1]", "White_Palace_02[left1]", "White_Palace_12[right1]", | 
					
						
							|  |  |  |  |     "White_Palace_12[right1]", "White_Palace_07[top1]", "White_Palace_07[top1]", "White_Palace_05[left1]", | 
					
						
							|  |  |  |  |     "White_Palace_05[left1]", "White_Palace_13[right1]", "White_Palace_13[right1]", "White_Palace_01[top1]", | 
					
						
							|  |  |  |  |     "White_Palace_01[top1]", | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-06 00:41:15 +02:00
										 |  |  |  | progression_charms = { | 
					
						
							|  |  |  |  |     # Baulder Killers | 
					
						
							|  |  |  |  |     "Grubberfly's_Elegy", "Weaversong", "Glowing_Womb", | 
					
						
							|  |  |  |  |     # Spore Shroom spots in fungle wastes | 
					
						
							|  |  |  |  |     "Spore_Shroom", | 
					
						
							|  |  |  |  |     # Tuk gives egg, | 
					
						
							|  |  |  |  |     "Defender's_Crest", | 
					
						
							|  |  |  |  |     # Unlocks Grimm Troupe | 
					
						
							|  |  |  |  |     "Grimmchild1", "Grimmchild2" | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-18 18:19:21 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-11 14:22:44 +02:00
										 |  |  |  | class HKWorld(World): | 
					
						
							| 
									
										
										
										
											2022-04-02 20:49:27 +02:00
										 |  |  |  |     """Beneath the fading town of Dirtmouth sleeps a vast, ancient kingdom. Many are drawn beneath the surface, 
 | 
					
						
							|  |  |  |  |     searching for riches, or glory, or answers to old secrets. | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-02 20:56:26 +02:00
										 |  |  |  |     As the enigmatic Knight, you’ll traverse the depths, unravel its mysteries and conquer its evils. | 
					
						
							| 
									
										
										
										
											2022-04-02 20:49:27 +02:00
										 |  |  |  |     """  # from https://www.hollowknight.com
 | 
					
						
							| 
									
										
										
										
											2021-06-11 14:22:44 +02:00
										 |  |  |  |     game: str = "Hollow Knight" | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  |     options = hollow_knight_options | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     item_name_to_id = {name: data.id for name, data in item_table.items()} | 
					
						
							|  |  |  |  |     location_name_to_id = {location_name: location_id for location_id, location_name in | 
					
						
							|  |  |  |  |                            enumerate(locations, start=0x1000000)} | 
					
						
							| 
									
										
										
										
											2022-04-04 00:15:16 +02:00
										 |  |  |  |     item_name_groups = item_name_groups | 
					
						
							| 
									
										
										
										
											2021-07-12 18:05:46 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     ranges: typing.Dict[str, typing.Tuple[int, int]] | 
					
						
							| 
									
										
										
										
											2022-04-04 00:40:19 +02:00
										 |  |  |  |     shops: typing.Dict[str, str] = { | 
					
						
							|  |  |  |  |         "Egg_Shop": "Egg", | 
					
						
							|  |  |  |  |         "Grubfather": "Grub", | 
					
						
							|  |  |  |  |         "Seer": "Essence", | 
					
						
							|  |  |  |  |         "Salubra_(Requires_Charms)": "Charm" | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     charm_costs: typing.List[int] | 
					
						
							|  |  |  |  |     data_version = 2 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     allow_white_palace = False | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __init__(self, world, player): | 
					
						
							|  |  |  |  |         super(HKWorld, self).__init__(world, player) | 
					
						
							|  |  |  |  |         self.created_multi_locations: typing.Dict[str, int] = Counter() | 
					
						
							|  |  |  |  |         self.ranges = {} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def generate_early(self): | 
					
						
							|  |  |  |  |         world = self.world | 
					
						
							| 
									
										
										
										
											2022-04-12 17:13:52 +02:00
										 |  |  |  |         charm_costs = world.RandomCharmCosts[self.player].get_costs(world.random) | 
					
						
							|  |  |  |  |         self.charm_costs = world.PlandoCharmCosts[self.player].get_costs(charm_costs) | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         world.exclude_locations[self.player].value.update(white_palace_locations) | 
					
						
							|  |  |  |  |         world.local_items[self.player].value.add("Mimic_Grub") | 
					
						
							|  |  |  |  |         for vendor, unit in self.shops.items(): | 
					
						
							| 
									
										
										
										
											2022-04-03 18:23:40 -04:00
										 |  |  |  |             mini = getattr(world, f"Minimum{unit}Price")[self.player] | 
					
						
							|  |  |  |  |             maxi = getattr(world, f"Maximum{unit}Price")[self.player] | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |             # if minimum > maximum, set minimum to maximum | 
					
						
							|  |  |  |  |             mini.value = min(mini.value, maxi.value) | 
					
						
							|  |  |  |  |             self.ranges[unit] = mini.value, maxi.value | 
					
						
							| 
									
										
										
										
											2022-04-03 18:23:40 -04:00
										 |  |  |  |         world.push_precollected(HKItem(starts[world.StartLocation[self.player].current_key], | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |                                        True, None, "Event", self.player)) | 
					
						
							| 
									
										
										
										
											2022-04-02 21:25:03 +02:00
										 |  |  |  |         for option_name in disabled: | 
					
						
							|  |  |  |  |             getattr(world, option_name)[self.player].value = 0 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     def create_regions(self): | 
					
						
							|  |  |  |  |         menu_region: Region = create_region(self.world, self.player, 'Menu') | 
					
						
							|  |  |  |  |         self.world.regions.append(menu_region) | 
					
						
							| 
									
										
										
										
											2021-08-27 14:52:33 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  |         # Link regions | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         for event_name in event_names: | 
					
						
							|  |  |  |  |             loc = HKLocation(self.player, event_name, None, menu_region) | 
					
						
							|  |  |  |  |             loc.place_locked_item(HKItem(event_name, | 
					
						
							|  |  |  |  |                                          self.allow_white_palace or event_name not in white_palace_locations, | 
					
						
							|  |  |  |  |                                          None, "Event", self.player)) | 
					
						
							|  |  |  |  |             menu_region.locations.append(loc) | 
					
						
							|  |  |  |  |         for entry_transition, exit_transition in connectors.items(): | 
					
						
							|  |  |  |  |             if exit_transition: | 
					
						
							|  |  |  |  |                 # if door logic fulfilled -> award vanilla target as event | 
					
						
							|  |  |  |  |                 loc = HKLocation(self.player, entry_transition, None, menu_region) | 
					
						
							|  |  |  |  |                 loc.place_locked_item(HKItem(exit_transition, | 
					
						
							|  |  |  |  |                                              self.allow_white_palace or exit_transition not in white_palace_locations, | 
					
						
							|  |  |  |  |                                              None, "Event", self.player)) | 
					
						
							|  |  |  |  |                 menu_region.locations.append(loc) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def create_items(self): | 
					
						
							|  |  |  |  |         # Generate item pool and associated locations (paired in HK) | 
					
						
							|  |  |  |  |         pool: typing.List[HKItem] = [] | 
					
						
							|  |  |  |  |         geo_replace: typing.Set[str] = set() | 
					
						
							|  |  |  |  |         if self.world.RemoveSpellUpgrades[self.player]: | 
					
						
							|  |  |  |  |             geo_replace.add("Abyss_Shriek") | 
					
						
							|  |  |  |  |             geo_replace.add("Shade_Soul") | 
					
						
							|  |  |  |  |             geo_replace.add("Descending_Dark") | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         for option_key, option in hollow_knight_randomize_options.items(): | 
					
						
							|  |  |  |  |             if getattr(self.world, option_key)[self.player]: | 
					
						
							|  |  |  |  |                 for item_name, location_name in zip(option.items, option.locations): | 
					
						
							|  |  |  |  |                     if item_name in geo_replace: | 
					
						
							|  |  |  |  |                         item_name = "Geo_Rock-Default" | 
					
						
							| 
									
										
										
										
											2022-04-02 21:16:23 +02:00
										 |  |  |  |                     item = self.create_item(item_name) | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |                     if location_name in white_palace_locations: | 
					
						
							| 
									
										
										
										
											2022-04-02 21:16:23 +02:00
										 |  |  |  |                         self.create_location(location_name).place_locked_item(item) | 
					
						
							|  |  |  |  |                     elif location_name == "Start": | 
					
						
							|  |  |  |  |                         self.world.push_precollected(item) | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |                     else: | 
					
						
							|  |  |  |  |                         self.create_location(location_name) | 
					
						
							| 
									
										
										
										
											2022-04-02 21:16:23 +02:00
										 |  |  |  |                         pool.append(item) | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  |             else: | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |                 for item_name, location_name in zip(option.items, option.locations): | 
					
						
							|  |  |  |  |                     item = self.create_item(item_name) | 
					
						
							|  |  |  |  |                     if location_name == "Start": | 
					
						
							|  |  |  |  |                         self.world.push_precollected(item) | 
					
						
							|  |  |  |  |                     else: | 
					
						
							|  |  |  |  |                         self.create_location(location_name).place_locked_item(item) | 
					
						
							| 
									
										
										
										
											2022-04-03 18:23:40 -04:00
										 |  |  |  |         for i in range(self.world.EggShopSlots[self.player].value): | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |             self.create_location("Egg_Shop") | 
					
						
							|  |  |  |  |             pool.append(self.create_item("Geo_Rock-Default")) | 
					
						
							|  |  |  |  |         if not self.allow_white_palace: | 
					
						
							|  |  |  |  |             loc = self.world.get_location("King_Fragment", self.player) | 
					
						
							|  |  |  |  |             if loc.item and loc.item.name == loc.name: | 
					
						
							|  |  |  |  |                 loc.item.advancement = False | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  |         self.world.itempool += pool | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-03 22:05:20 +02:00
										 |  |  |  |         for shopname in self.shops: | 
					
						
							|  |  |  |  |             prices: typing.List[int] = [] | 
					
						
							|  |  |  |  |             locations: typing.List[HKLocation] = [] | 
					
						
							|  |  |  |  |             for x in range(1, self.created_multi_locations[shopname]+1): | 
					
						
							|  |  |  |  |                 loc = self.world.get_location(self.get_multi_location_name(shopname, x), self.player) | 
					
						
							|  |  |  |  |                 locations.append(loc) | 
					
						
							|  |  |  |  |                 prices.append(loc.cost) | 
					
						
							|  |  |  |  |             prices.sort() | 
					
						
							|  |  |  |  |             for loc, price in zip(locations, prices): | 
					
						
							|  |  |  |  |                 loc.cost = price | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  |     def set_rules(self): | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         world = self.world | 
					
						
							|  |  |  |  |         player = self.player | 
					
						
							|  |  |  |  |         if world.logic[player] != 'nologic': | 
					
						
							|  |  |  |  |             world.completion_condition[player] = lambda state: state.has('DREAMER', player, 3) | 
					
						
							|  |  |  |  |         set_rules(self) | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     def fill_slot_data(self): | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  |         slot_data = {} | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         options = slot_data["options"] = {} | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  |         for option_name in self.options: | 
					
						
							|  |  |  |  |             option = getattr(self.world, option_name)[self.player] | 
					
						
							| 
									
										
										
										
											2022-04-12 17:13:52 +02:00
										 |  |  |  |             try: | 
					
						
							|  |  |  |  |                 optionvalue = int(option.value) | 
					
						
							|  |  |  |  |             except TypeError: | 
					
						
							|  |  |  |  |                 pass  # C# side is currently typed as dict[str, int], drop what doesn't fit | 
					
						
							|  |  |  |  |             else: | 
					
						
							|  |  |  |  |                 options[option_name] = optionvalue | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         # 32 bit int | 
					
						
							|  |  |  |  |         slot_data["seed"] = self.world.slot_seeds[self.player].randint(-2147483647, 2147483646) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         for shop, unit in self.shops.items(): | 
					
						
							|  |  |  |  |             slot_data[f"{unit}_costs"] = { | 
					
						
							|  |  |  |  |                 f"{shop}_{i}": | 
					
						
							|  |  |  |  |                     self.world.get_location(f"{shop}_{i}", self.player).cost | 
					
						
							|  |  |  |  |                 for i in range(1, 1 + self.created_multi_locations[shop]) | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         slot_data["notch_costs"] = self.charm_costs | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  |         return slot_data | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     def create_item(self, name: str) -> HKItem: | 
					
						
							| 
									
										
										
										
											2021-07-12 13:54:47 +02:00
										 |  |  |  |         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
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     def create_location(self, name: str) -> HKLocation: | 
					
						
							|  |  |  |  |         unit = self.shops.get(name, None) | 
					
						
							|  |  |  |  |         if unit: | 
					
						
							|  |  |  |  |             cost = self.world.random.randint(*self.ranges[unit]) | 
					
						
							|  |  |  |  |         else: | 
					
						
							|  |  |  |  |             cost = 0 | 
					
						
							|  |  |  |  |         if name in multi_locations: | 
					
						
							|  |  |  |  |             self.created_multi_locations[name] += 1 | 
					
						
							| 
									
										
										
										
											2022-04-03 22:05:20 +02:00
										 |  |  |  |             name = self.get_multi_location_name(name, self.created_multi_locations[name]) | 
					
						
							| 
									
										
										
										
											2021-07-21 18:08:15 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         region = self.world.get_region("Menu", self.player) | 
					
						
							|  |  |  |  |         loc = HKLocation(self.player, name, self.location_name_to_id[name], region) | 
					
						
							|  |  |  |  |         if unit: | 
					
						
							|  |  |  |  |             loc.unit = unit | 
					
						
							|  |  |  |  |             loc.cost = cost | 
					
						
							|  |  |  |  |         region.locations.append(loc) | 
					
						
							|  |  |  |  |         return loc | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def collect(self, state, item: HKItem) -> bool: | 
					
						
							|  |  |  |  |         change = super(HKWorld, self).collect(state, item) | 
					
						
							| 
									
										
										
										
											2022-04-08 21:30:38 +02:00
										 |  |  |  |         if change: | 
					
						
							|  |  |  |  |             for effect_name, effect_value in item_effects.get(item.name, {}).items(): | 
					
						
							|  |  |  |  |                 state.prog_items[effect_name, item.player] += effect_value | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         return change | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def remove(self, state, item: HKItem) -> bool: | 
					
						
							|  |  |  |  |         change = super(HKWorld, self).remove(state, item) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-08 21:30:38 +02:00
										 |  |  |  |         if change: | 
					
						
							|  |  |  |  |             for effect_name, effect_value in item_effects.get(item.name, {}).items(): | 
					
						
							|  |  |  |  |                 if state.prog_items[effect_name, item.player] == effect_value: | 
					
						
							|  |  |  |  |                     del state.prog_items[effect_name, item.player] | 
					
						
							|  |  |  |  |                 state.prog_items[effect_name, item.player] -= effect_value | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         return change | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-03 00:12:37 +02:00
										 |  |  |  |     @classmethod | 
					
						
							| 
									
										
										
										
											2022-04-04 00:40:19 +02:00
										 |  |  |  |     def stage_write_spoiler(cls, world: MultiWorld, spoiler_handle): | 
					
						
							| 
									
										
										
										
											2022-04-03 00:12:37 +02:00
										 |  |  |  |         hk_players = world.get_game_players(cls.game) | 
					
						
							|  |  |  |  |         spoiler_handle.write('\n\nCharm Notches:') | 
					
						
							|  |  |  |  |         for player in hk_players: | 
					
						
							|  |  |  |  |             name = world.get_player_name(player) | 
					
						
							|  |  |  |  |             spoiler_handle.write(f'\n{name}\n') | 
					
						
							|  |  |  |  |             hk_world: HKWorld = world.worlds[player] | 
					
						
							| 
									
										
										
										
											2022-04-08 19:22:50 +02:00
										 |  |  |  |             for charm_number, cost in enumerate(hk_world.charm_costs): | 
					
						
							|  |  |  |  |                 spoiler_handle.write(f"\n{charm_names[charm_number]}: {cost}") | 
					
						
							| 
									
										
										
										
											2022-04-03 00:12:37 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-04 00:40:19 +02:00
										 |  |  |  |         spoiler_handle.write('\n\nShop Prices:') | 
					
						
							|  |  |  |  |         for player in hk_players: | 
					
						
							|  |  |  |  |             name = world.get_player_name(player) | 
					
						
							|  |  |  |  |             spoiler_handle.write(f'\n{name}\n') | 
					
						
							|  |  |  |  |             hk_world: HKWorld = world.worlds[player] | 
					
						
							|  |  |  |  |             for shop_name, unit_name in cls.shops.items(): | 
					
						
							|  |  |  |  |                 for x in range(1, hk_world.created_multi_locations[shop_name]+1): | 
					
						
							|  |  |  |  |                     loc = world.get_location(hk_world.get_multi_location_name(shop_name, x), player) | 
					
						
							|  |  |  |  |                     spoiler_handle.write(f"\n{loc}: {loc.item} costing {loc.cost} {unit_name}") | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-03 22:05:20 +02:00
										 |  |  |  |     def get_multi_location_name(self, base: str, i: typing.Optional[int]) -> str: | 
					
						
							|  |  |  |  |         if i is None: | 
					
						
							|  |  |  |  |             i = self.created_multi_locations[base] | 
					
						
							|  |  |  |  |         assert 0 < i < 18, "limited number of multi location IDs reserved." | 
					
						
							|  |  |  |  |         return f"{base}_{i}" | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | def create_region(world: MultiWorld, player: int, name: str, location_names=None, exits=None) -> Region: | 
					
						
							| 
									
										
										
										
											2022-03-18 18:19:21 +01:00
										 |  |  |  |     ret = Region(name, RegionType.Generic, name, player) | 
					
						
							| 
									
										
										
										
											2021-02-24 06:02:51 +01:00
										 |  |  |  |     ret.world = world | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     if location_names: | 
					
						
							|  |  |  |  |         for location in location_names: | 
					
						
							|  |  |  |  |             loc_id = HKWorld.location_name_to_id.get(location, None) | 
					
						
							| 
									
										
										
										
											2021-02-24 06:02:51 +01:00
										 |  |  |  |             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" | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     cost: int = 0 | 
					
						
							|  |  |  |  |     unit: typing.Optional[str] = None | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     def __init__(self, player: int, name: str, code=None, parent=None): | 
					
						
							|  |  |  |  |         super(HKLocation, self).__init__(player, name, code if code else None, parent) | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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): | 
					
						
							| 
									
										
										
										
											2022-03-18 18:19:21 +01:00
										 |  |  |  |         super(HKItem, self).__init__(name, advancement, code if code else None, player) | 
					
						
							| 
									
										
										
										
											2021-03-21 00:47:17 +01:00
										 |  |  |  |         self.type = type | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         if name == "Mimic_Grub": | 
					
						
							|  |  |  |  |             self.trap = True | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-06 00:41:15 +02:00
										 |  |  |  |         if type in ("Grub", "DreamWarrior", "Root", "Egg"): | 
					
						
							|  |  |  |  |             self.skip_in_prog_balancing = True | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if type == "Charm" and name not in progression_charms: | 
					
						
							|  |  |  |  |             self.skip_in_prog_balancing = True | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-24 06:02:51 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | class HKLogicMixin(LogicMixin): | 
					
						
							|  |  |  |  |     world: MultiWorld | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     def _hk_notches(self, player: int, *notches: int) -> int: | 
					
						
							|  |  |  |  |         return sum(self.world.worlds[player].charm_costs[notch] for notch in notches) | 
					
						
							| 
									
										
										
										
											2021-07-15 13:31:33 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-05 15:01:33 +02:00
										 |  |  |  |     def _hk_option(self, player: int, option_name: str) -> int: | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         return getattr(self.world, option_name)[player].value | 
					
						
							| 
									
										
										
										
											2021-07-15 13:31:33 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-05 15:01:33 +02:00
										 |  |  |  |     def _hk_start(self, player, start_location: str) -> bool: | 
					
						
							| 
									
										
										
										
											2022-04-03 18:23:40 -04:00
										 |  |  |  |         return self.world.StartLocation[player] == start_location |