| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | from copy import deepcopy | 
					
						
							|  |  |  |  | import itertools | 
					
						
							|  |  |  |  | import operator | 
					
						
							| 
									
										
										
										
											2024-08-08 13:33:13 -05:00
										 |  |  |  | from collections import defaultdict, 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 | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  | from .Rules import set_rules, cost_terms, _hk_can_beat_thk, _hk_siblings_ending, _hk_can_beat_radiance | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | from .Options import hollow_knight_options, hollow_knight_randomize_options, Goal, WhitePalace, CostSanity, \ | 
					
						
							| 
									
										
										
										
											2024-08-08 13:33:13 -05:00
										 |  |  |  |     shop_to_option, HKOptions, GrubHuntGoal | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | from .ExtractedData import locations, starts, multi_locations, location_to_region_lookup, \ | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |     event_names, item_effects, connectors, one_ways, vanilla_shop_costs, vanilla_location_costs | 
					
						
							| 
									
										
										
										
											2022-04-08 19:22:50 +02:00
										 |  |  |  | from .Charms import names as charm_names | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-08 13:33:13 -05:00
										 |  |  |  | from BaseClasses import Region, Location, MultiWorld, Item, LocationProgressType, Tutorial, ItemClassification, CollectionState | 
					
						
							| 
									
										
										
										
											2023-06-25 03:47:38 +02:00
										 |  |  |  | from worlds.AutoWorld import World, LogicMixin, WebWorld | 
					
						
							| 
									
										
										
										
											2021-02-21 20:17:24 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Hollow Knight updates (goals, WP/POP, etc.) (#438)
* Hollow Knight updates:
- Add configurable goals (Any, THK, Siblings, Radiance)
  - Change base logic to require Opened_Black_Egg_Temple instead of
    requiring 3 dreamers.  This is future-proof for transition rando,
    where Black Egg might not have been located yet.
  - Add combat logic for THK and Radiance on par with Rando4's boss logic,
    so itemless HK shouldn't be required.
- Existing completion logic now uses Black_Egg_te
- Add White Palace options
  (Exclude, King Fragment Only, No Path of Pain, Include)
  - Excluded WP may still be required for King Fragment if Charms are
    not randomized
  - Simply don't place WP locations that are excluded
  - Distinguish between POP locations (required for POP), WP checks (
    actual item locations), WP transitions (relevant for future transition
    rando), and WP events (logically required to reach King Fragment)
  - Many transitions were listed twice.  Remove duplicates.
  - Sort transitions by scene
- For randomizable locations that have no logical significance when not
    randomized, simply skip adding them to the pool entirely for
    theoretically faster generation.
* Hollow Knight updates
  - Support random starting geo up to 1000 geo.
  - Always include locations rather than dropping unrandomized "logicless"
    ones, as it is required to best support same-slot coop.
											
										 
											2022-06-12 23:23:03 -07:00
										 |  |  |  | path_of_pain_locations = { | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     "Soul_Totem-Path_of_Pain_Below_Thornskip", | 
					
						
							|  |  |  |  |     "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-Path_of_Pain_Below_Lever", | 
					
						
							|  |  |  |  |     "Soul_Totem-Path_of_Pain_Second", | 
					
						
							| 
									
										
											  
											
												Hollow Knight updates (goals, WP/POP, etc.) (#438)
* Hollow Knight updates:
- Add configurable goals (Any, THK, Siblings, Radiance)
  - Change base logic to require Opened_Black_Egg_Temple instead of
    requiring 3 dreamers.  This is future-proof for transition rando,
    where Black Egg might not have been located yet.
  - Add combat logic for THK and Radiance on par with Rando4's boss logic,
    so itemless HK shouldn't be required.
- Existing completion logic now uses Black_Egg_te
- Add White Palace options
  (Exclude, King Fragment Only, No Path of Pain, Include)
  - Excluded WP may still be required for King Fragment if Charms are
    not randomized
  - Simply don't place WP locations that are excluded
  - Distinguish between POP locations (required for POP), WP checks (
    actual item locations), WP transitions (relevant for future transition
    rando), and WP events (logically required to reach King Fragment)
  - Many transitions were listed twice.  Remove duplicates.
  - Sort transitions by scene
- For randomizable locations that have no logical significance when not
    randomized, simply skip adding them to the pool entirely for
    theoretically faster generation.
* Hollow Knight updates
  - Support random starting geo up to 1000 geo.
  - Always include locations rather than dropping unrandomized "logicless"
    ones, as it is required to best support same-slot coop.
											
										 
											2022-06-12 23:23:03 -07:00
										 |  |  |  |     "Journal_Entry-Seal_of_Binding", | 
					
						
							|  |  |  |  |     "Warp-Path_of_Pain_Complete", | 
					
						
							|  |  |  |  |     "Defeated_Path_of_Pain_Arena", | 
					
						
							|  |  |  |  |     "Completed_Path_of_Pain", | 
					
						
							|  |  |  |  |     # Path of Pain transitions | 
					
						
							|  |  |  |  |     "White_Palace_17[right1]", "White_Palace_17[bot1]", | 
					
						
							|  |  |  |  |     "White_Palace_18[top1]", "White_Palace_18[right1]", | 
					
						
							|  |  |  |  |     "White_Palace_19[left1]", "White_Palace_19[top1]", | 
					
						
							|  |  |  |  |     "White_Palace_20[bot1]", | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | white_palace_transitions = { | 
					
						
							|  |  |  |  |     # Event-Transitions: | 
					
						
							|  |  |  |  |     # "Grubfather_2", | 
					
						
							|  |  |  |  |     "White_Palace_01[left1]", "White_Palace_01[right1]", "White_Palace_01[top1]", | 
					
						
							|  |  |  |  |     "White_Palace_02[left1]", | 
					
						
							|  |  |  |  |     "White_Palace_03_hub[bot1]", "White_Palace_03_hub[left1]", "White_Palace_03_hub[left2]", | 
					
						
							|  |  |  |  |     "White_Palace_03_hub[right1]", "White_Palace_03_hub[top1]", | 
					
						
							|  |  |  |  |     "White_Palace_04[right2]", "White_Palace_04[top1]", | 
					
						
							|  |  |  |  |     "White_Palace_05[left1]", "White_Palace_05[left2]", "White_Palace_05[right1]", "White_Palace_05[right2]", | 
					
						
							|  |  |  |  |     "White_Palace_06[bot1]", "White_Palace_06[left1]", "White_Palace_06[top1]", "White_Palace_07[bot1]", | 
					
						
							|  |  |  |  |     "White_Palace_07[top1]", "White_Palace_08[left1]", "White_Palace_08[right1]", | 
					
						
							|  |  |  |  |     "White_Palace_09[right1]", | 
					
						
							|  |  |  |  |     "White_Palace_11[door2]", | 
					
						
							|  |  |  |  |     "White_Palace_12[bot1]", "White_Palace_12[right1]", | 
					
						
							|  |  |  |  |     "White_Palace_13[left1]", "White_Palace_13[left2]", "White_Palace_13[left3]", "White_Palace_13[right1]", | 
					
						
							|  |  |  |  |     "White_Palace_14[bot1]", "White_Palace_14[right1]", | 
					
						
							|  |  |  |  |     "White_Palace_15[left1]", "White_Palace_15[right1]", "White_Palace_15[right2]", | 
					
						
							|  |  |  |  |     "White_Palace_16[left1]", "White_Palace_16[left2]", | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | white_palace_checks = { | 
					
						
							|  |  |  |  |     "Soul_Totem-White_Palace_Final", | 
					
						
							|  |  |  |  |     "Soul_Totem-White_Palace_Entrance", | 
					
						
							|  |  |  |  |     "Lore_Tablet-Palace_Throne", | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     "Soul_Totem-White_Palace_Left", | 
					
						
							|  |  |  |  |     "Lore_Tablet-Palace_Workshop", | 
					
						
							|  |  |  |  |     "Soul_Totem-White_Palace_Hub", | 
					
						
							| 
									
										
											  
											
												Hollow Knight updates (goals, WP/POP, etc.) (#438)
* Hollow Knight updates:
- Add configurable goals (Any, THK, Siblings, Radiance)
  - Change base logic to require Opened_Black_Egg_Temple instead of
    requiring 3 dreamers.  This is future-proof for transition rando,
    where Black Egg might not have been located yet.
  - Add combat logic for THK and Radiance on par with Rando4's boss logic,
    so itemless HK shouldn't be required.
- Existing completion logic now uses Black_Egg_te
- Add White Palace options
  (Exclude, King Fragment Only, No Path of Pain, Include)
  - Excluded WP may still be required for King Fragment if Charms are
    not randomized
  - Simply don't place WP locations that are excluded
  - Distinguish between POP locations (required for POP), WP checks (
    actual item locations), WP transitions (relevant for future transition
    rando), and WP events (logically required to reach King Fragment)
  - Many transitions were listed twice.  Remove duplicates.
  - Sort transitions by scene
- For randomizable locations that have no logical significance when not
    randomized, simply skip adding them to the pool entirely for
    theoretically faster generation.
* Hollow Knight updates
  - Support random starting geo up to 1000 geo.
  - Always include locations rather than dropping unrandomized "logicless"
    ones, as it is required to best support same-slot coop.
											
										 
											2022-06-12 23:23:03 -07:00
										 |  |  |  |     "Soul_Totem-White_Palace_Right" | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | white_palace_events = { | 
					
						
							|  |  |  |  |     "White_Palace_03_hub", | 
					
						
							|  |  |  |  |     "White_Palace_13", | 
					
						
							|  |  |  |  |     "White_Palace_01", | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     "Palace_Entrance_Lantern_Lit", | 
					
						
							|  |  |  |  |     "Palace_Left_Lantern_Lit", | 
					
						
							|  |  |  |  |     "Palace_Right_Lantern_Lit", | 
					
						
							|  |  |  |  |     "Palace_Atrium_Gates_Opened", | 
					
						
							|  |  |  |  |     "Warp-White_Palace_Atrium_to_Palace_Grounds", | 
					
						
							|  |  |  |  |     "Warp-White_Palace_Entrance_to_Palace_Grounds", | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-06 00:41:15 +02:00
										 |  |  |  | progression_charms = { | 
					
						
							| 
									
										
											  
											
												Hollow Knight updates (goals, WP/POP, etc.) (#438)
* Hollow Knight updates:
- Add configurable goals (Any, THK, Siblings, Radiance)
  - Change base logic to require Opened_Black_Egg_Temple instead of
    requiring 3 dreamers.  This is future-proof for transition rando,
    where Black Egg might not have been located yet.
  - Add combat logic for THK and Radiance on par with Rando4's boss logic,
    so itemless HK shouldn't be required.
- Existing completion logic now uses Black_Egg_te
- Add White Palace options
  (Exclude, King Fragment Only, No Path of Pain, Include)
  - Excluded WP may still be required for King Fragment if Charms are
    not randomized
  - Simply don't place WP locations that are excluded
  - Distinguish between POP locations (required for POP), WP checks (
    actual item locations), WP transitions (relevant for future transition
    rando), and WP events (logically required to reach King Fragment)
  - Many transitions were listed twice.  Remove duplicates.
  - Sort transitions by scene
- For randomizable locations that have no logical significance when not
    randomized, simply skip adding them to the pool entirely for
    theoretically faster generation.
* Hollow Knight updates
  - Support random starting geo up to 1000 geo.
  - Always include locations rather than dropping unrandomized "logicless"
    ones, as it is required to best support same-slot coop.
											
										 
											2022-06-12 23:23:03 -07:00
										 |  |  |  |     # Baldur Killers | 
					
						
							| 
									
										
										
										
											2022-04-06 00:41:15 +02:00
										 |  |  |  |     "Grubberfly's_Elegy", "Weaversong", "Glowing_Womb", | 
					
						
							| 
									
										
											  
											
												Hollow Knight updates (goals, WP/POP, etc.) (#438)
* Hollow Knight updates:
- Add configurable goals (Any, THK, Siblings, Radiance)
  - Change base logic to require Opened_Black_Egg_Temple instead of
    requiring 3 dreamers.  This is future-proof for transition rando,
    where Black Egg might not have been located yet.
  - Add combat logic for THK and Radiance on par with Rando4's boss logic,
    so itemless HK shouldn't be required.
- Existing completion logic now uses Black_Egg_te
- Add White Palace options
  (Exclude, King Fragment Only, No Path of Pain, Include)
  - Excluded WP may still be required for King Fragment if Charms are
    not randomized
  - Simply don't place WP locations that are excluded
  - Distinguish between POP locations (required for POP), WP checks (
    actual item locations), WP transitions (relevant for future transition
    rando), and WP events (logically required to reach King Fragment)
  - Many transitions were listed twice.  Remove duplicates.
  - Sort transitions by scene
- For randomizable locations that have no logical significance when not
    randomized, simply skip adding them to the pool entirely for
    theoretically faster generation.
* Hollow Knight updates
  - Support random starting geo up to 1000 geo.
  - Always include locations rather than dropping unrandomized "logicless"
    ones, as it is required to best support same-slot coop.
											
										 
											2022-06-12 23:23:03 -07:00
										 |  |  |  |     # Spore Shroom spots in fungal wastes and elsewhere | 
					
						
							| 
									
										
										
										
											2022-04-06 00:41:15 +02:00
										 |  |  |  |     "Spore_Shroom", | 
					
						
							|  |  |  |  |     # Tuk gives egg, | 
					
						
							|  |  |  |  |     "Defender's_Crest", | 
					
						
							|  |  |  |  |     # Unlocks Grimm Troupe | 
					
						
							|  |  |  |  |     "Grimmchild1", "Grimmchild2" | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Hollow Knight updates (goals, WP/POP, etc.) (#438)
* Hollow Knight updates:
- Add configurable goals (Any, THK, Siblings, Radiance)
  - Change base logic to require Opened_Black_Egg_Temple instead of
    requiring 3 dreamers.  This is future-proof for transition rando,
    where Black Egg might not have been located yet.
  - Add combat logic for THK and Radiance on par with Rando4's boss logic,
    so itemless HK shouldn't be required.
- Existing completion logic now uses Black_Egg_te
- Add White Palace options
  (Exclude, King Fragment Only, No Path of Pain, Include)
  - Excluded WP may still be required for King Fragment if Charms are
    not randomized
  - Simply don't place WP locations that are excluded
  - Distinguish between POP locations (required for POP), WP checks (
    actual item locations), WP transitions (relevant for future transition
    rando), and WP events (logically required to reach King Fragment)
  - Many transitions were listed twice.  Remove duplicates.
  - Sort transitions by scene
- For randomizable locations that have no logical significance when not
    randomized, simply skip adding them to the pool entirely for
    theoretically faster generation.
* Hollow Knight updates
  - Support random starting geo up to 1000 geo.
  - Always include locations rather than dropping unrandomized "logicless"
    ones, as it is required to best support same-slot coop.
											
										 
											2022-06-12 23:23:03 -07:00
										 |  |  |  | # Vanilla placements of the following items have no impact on logic, thus we can avoid creating these items and | 
					
						
							|  |  |  |  | # locations entirely when the option to randomize them is disabled. | 
					
						
							|  |  |  |  | logicless_options = { | 
					
						
							|  |  |  |  |     "RandomizeVesselFragments", "RandomizeGeoChests", "RandomizeJunkPitChests", "RandomizeRelics", | 
					
						
							|  |  |  |  |     "RandomizeMaps", "RandomizeJournalEntries", "RandomizeGeoRocks", "RandomizeBossGeo", | 
					
						
							|  |  |  |  |     "RandomizeLoreTablets", "RandomizeSoulTotems", | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | # Options that affect vanilla starting items | 
					
						
							|  |  |  |  | randomizable_starting_items: typing.Dict[str, typing.Tuple[str, ...]] = { | 
					
						
							|  |  |  |  |     "RandomizeFocus": ("Focus",), | 
					
						
							|  |  |  |  |     "RandomizeSwim": ("Swim",), | 
					
						
							|  |  |  |  |     "RandomizeNail": ('Upslash', 'Leftslash', 'Rightslash') | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | # Shop cost types. | 
					
						
							|  |  |  |  | shop_cost_types: typing.Dict[str, typing.Tuple[str, ...]] = { | 
					
						
							|  |  |  |  |     "Egg_Shop": ("RANCIDEGGS",), | 
					
						
							|  |  |  |  |     "Grubfather": ("GRUBS",), | 
					
						
							|  |  |  |  |     "Seer": ("ESSENCE",), | 
					
						
							|  |  |  |  |     "Salubra_(Requires_Charms)": ("CHARMS", "GEO"), | 
					
						
							|  |  |  |  |     "Sly": ("GEO",), | 
					
						
							|  |  |  |  |     "Sly_(Key)": ("GEO",), | 
					
						
							|  |  |  |  |     "Iselda": ("GEO",), | 
					
						
							|  |  |  |  |     "Salubra": ("GEO",), | 
					
						
							|  |  |  |  |     "Leg_Eater": ("GEO",), | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-03-18 18:19:21 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-03 07:41:27 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-11 13:05:53 -05:00
										 |  |  |  | class HKWeb(WebWorld): | 
					
						
							|  |  |  |  |     tutorials = [Tutorial( | 
					
						
							|  |  |  |  |         "Mod Setup and Use Guide", | 
					
						
							|  |  |  |  |         "A guide to playing Hollow Knight with Archipelago.", | 
					
						
							|  |  |  |  |         "English", | 
					
						
							|  |  |  |  |         "setup_en.md", | 
					
						
							|  |  |  |  |         "setup/en", | 
					
						
							|  |  |  |  |         ["Ijwu"] | 
					
						
							|  |  |  |  |     )] | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-03 07:41:27 -05:00
										 |  |  |  |     bug_report_page = "https://github.com/Ijwu/Archipelago.HollowKnight/issues/new?assignees=&labels=bug%2C+needs+investigation&template=bug_report.md&title=" | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-11 13:05:53 -05: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" | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |     options_dataclass = HKOptions | 
					
						
							|  |  |  |  |     options: HKOptions | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-11 13:05:53 -05:00
										 |  |  |  |     web = HKWeb() | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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]] | 
					
						
							|  |  |  |  |     charm_costs: typing.List[int] | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |     cached_filler_items = {} | 
					
						
							| 
									
										
										
										
											2024-08-08 13:33:13 -05:00
										 |  |  |  |     grub_count: int | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |     def __init__(self, multiworld, player): | 
					
						
							|  |  |  |  |         super(HKWorld, self).__init__(multiworld, player) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |         self.created_multi_locations: typing.Dict[str, typing.List[HKLocation]] = { | 
					
						
							|  |  |  |  |             location: list() for location in multi_locations | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         self.ranges = {} | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |         self.created_shop_items = 0 | 
					
						
							|  |  |  |  |         self.vanilla_shop_costs = deepcopy(vanilla_shop_costs) | 
					
						
							| 
									
										
										
										
											2024-08-08 13:33:13 -05:00
										 |  |  |  |         self.grub_count = 0 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     def generate_early(self): | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         options = self.options | 
					
						
							|  |  |  |  |         charm_costs = options.RandomCharmCosts.get_costs(self.random) | 
					
						
							|  |  |  |  |         self.charm_costs = options.PlandoCharmCosts.get_costs(charm_costs) | 
					
						
							|  |  |  |  |         # options.exclude_locations.value.update(white_palace_locations) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |         for term, data in cost_terms.items(): | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             mini = getattr(options, f"Minimum{data.option}Price") | 
					
						
							|  |  |  |  |             maxi = getattr(options, f"Maximum{data.option}Price") | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |             # if minimum > maximum, set minimum to maximum | 
					
						
							|  |  |  |  |             mini.value = min(mini.value, maxi.value) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |             self.ranges[term] = mini.value, maxi.value | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         self.multiworld.push_precollected(HKItem(starts[options.StartLocation.current_key], | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |                                        True, None, "Event", self.player)) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Hollow Knight updates (goals, WP/POP, etc.) (#438)
* Hollow Knight updates:
- Add configurable goals (Any, THK, Siblings, Radiance)
  - Change base logic to require Opened_Black_Egg_Temple instead of
    requiring 3 dreamers.  This is future-proof for transition rando,
    where Black Egg might not have been located yet.
  - Add combat logic for THK and Radiance on par with Rando4's boss logic,
    so itemless HK shouldn't be required.
- Existing completion logic now uses Black_Egg_te
- Add White Palace options
  (Exclude, King Fragment Only, No Path of Pain, Include)
  - Excluded WP may still be required for King Fragment if Charms are
    not randomized
  - Simply don't place WP locations that are excluded
  - Distinguish between POP locations (required for POP), WP checks (
    actual item locations), WP transitions (relevant for future transition
    rando), and WP events (logically required to reach King Fragment)
  - Many transitions were listed twice.  Remove duplicates.
  - Sort transitions by scene
- For randomizable locations that have no logical significance when not
    randomized, simply skip adding them to the pool entirely for
    theoretically faster generation.
* Hollow Knight updates
  - Support random starting geo up to 1000 geo.
  - Always include locations rather than dropping unrandomized "logicless"
    ones, as it is required to best support same-slot coop.
											
										 
											2022-06-12 23:23:03 -07:00
										 |  |  |  |     def white_palace_exclusions(self): | 
					
						
							|  |  |  |  |         exclusions = set() | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         wp = self.options.WhitePalace | 
					
						
							| 
									
										
											  
											
												Hollow Knight updates (goals, WP/POP, etc.) (#438)
* Hollow Knight updates:
- Add configurable goals (Any, THK, Siblings, Radiance)
  - Change base logic to require Opened_Black_Egg_Temple instead of
    requiring 3 dreamers.  This is future-proof for transition rando,
    where Black Egg might not have been located yet.
  - Add combat logic for THK and Radiance on par with Rando4's boss logic,
    so itemless HK shouldn't be required.
- Existing completion logic now uses Black_Egg_te
- Add White Palace options
  (Exclude, King Fragment Only, No Path of Pain, Include)
  - Excluded WP may still be required for King Fragment if Charms are
    not randomized
  - Simply don't place WP locations that are excluded
  - Distinguish between POP locations (required for POP), WP checks (
    actual item locations), WP transitions (relevant for future transition
    rando), and WP events (logically required to reach King Fragment)
  - Many transitions were listed twice.  Remove duplicates.
  - Sort transitions by scene
- For randomizable locations that have no logical significance when not
    randomized, simply skip adding them to the pool entirely for
    theoretically faster generation.
* Hollow Knight updates
  - Support random starting geo up to 1000 geo.
  - Always include locations rather than dropping unrandomized "logicless"
    ones, as it is required to best support same-slot coop.
											
										 
											2022-06-12 23:23:03 -07:00
										 |  |  |  |         if wp <= WhitePalace.option_nopathofpain: | 
					
						
							|  |  |  |  |             exclusions.update(path_of_pain_locations) | 
					
						
							|  |  |  |  |         if wp <= WhitePalace.option_kingfragment: | 
					
						
							|  |  |  |  |             exclusions.update(white_palace_checks) | 
					
						
							| 
									
										
										
										
											2022-06-25 11:15:03 -07:00
										 |  |  |  |         if wp == WhitePalace.option_exclude: | 
					
						
							| 
									
										
											  
											
												Hollow Knight updates (goals, WP/POP, etc.) (#438)
* Hollow Knight updates:
- Add configurable goals (Any, THK, Siblings, Radiance)
  - Change base logic to require Opened_Black_Egg_Temple instead of
    requiring 3 dreamers.  This is future-proof for transition rando,
    where Black Egg might not have been located yet.
  - Add combat logic for THK and Radiance on par with Rando4's boss logic,
    so itemless HK shouldn't be required.
- Existing completion logic now uses Black_Egg_te
- Add White Palace options
  (Exclude, King Fragment Only, No Path of Pain, Include)
  - Excluded WP may still be required for King Fragment if Charms are
    not randomized
  - Simply don't place WP locations that are excluded
  - Distinguish between POP locations (required for POP), WP checks (
    actual item locations), WP transitions (relevant for future transition
    rando), and WP events (logically required to reach King Fragment)
  - Many transitions were listed twice.  Remove duplicates.
  - Sort transitions by scene
- For randomizable locations that have no logical significance when not
    randomized, simply skip adding them to the pool entirely for
    theoretically faster generation.
* Hollow Knight updates
  - Support random starting geo up to 1000 geo.
  - Always include locations rather than dropping unrandomized "logicless"
    ones, as it is required to best support same-slot coop.
											
										 
											2022-06-12 23:23:03 -07:00
										 |  |  |  |             exclusions.add("King_Fragment") | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             if self.options.RandomizeCharms: | 
					
						
							| 
									
										
										
										
											2022-06-25 11:15:03 -07:00
										 |  |  |  |                 # If charms are randomized, this will be junk-filled -- so transitions and events are not progression | 
					
						
							|  |  |  |  |                 exclusions.update(white_palace_transitions) | 
					
						
							|  |  |  |  |                 exclusions.update(white_palace_events) | 
					
						
							| 
									
										
											  
											
												Hollow Knight updates (goals, WP/POP, etc.) (#438)
* Hollow Knight updates:
- Add configurable goals (Any, THK, Siblings, Radiance)
  - Change base logic to require Opened_Black_Egg_Temple instead of
    requiring 3 dreamers.  This is future-proof for transition rando,
    where Black Egg might not have been located yet.
  - Add combat logic for THK and Radiance on par with Rando4's boss logic,
    so itemless HK shouldn't be required.
- Existing completion logic now uses Black_Egg_te
- Add White Palace options
  (Exclude, King Fragment Only, No Path of Pain, Include)
  - Excluded WP may still be required for King Fragment if Charms are
    not randomized
  - Simply don't place WP locations that are excluded
  - Distinguish between POP locations (required for POP), WP checks (
    actual item locations), WP transitions (relevant for future transition
    rando), and WP events (logically required to reach King Fragment)
  - Many transitions were listed twice.  Remove duplicates.
  - Sort transitions by scene
- For randomizable locations that have no logical significance when not
    randomized, simply skip adding them to the pool entirely for
    theoretically faster generation.
* Hollow Knight updates
  - Support random starting geo up to 1000 geo.
  - Always include locations rather than dropping unrandomized "logicless"
    ones, as it is required to best support same-slot coop.
											
										 
											2022-06-12 23:23:03 -07:00
										 |  |  |  |         return exclusions | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     def create_regions(self): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |  |         menu_region: Region = create_region(self.multiworld, self.player, 'Menu') | 
					
						
							|  |  |  |  |         self.multiworld.regions.append(menu_region) | 
					
						
							| 
									
										
										
										
											2022-06-25 11:15:03 -07:00
										 |  |  |  |         # wp_exclusions = self.white_palace_exclusions() | 
					
						
							| 
									
										
										
										
											2021-08-27 14:52:33 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-02 08:26:17 -05:00
										 |  |  |  |         # check for any goal that godhome events are relevant to | 
					
						
							|  |  |  |  |         all_event_names = event_names.copy() | 
					
						
							| 
									
										
										
										
											2024-08-08 13:33:13 -05:00
										 |  |  |  |         if self.options.Goal in [Goal.option_godhome, Goal.option_godhome_flower, Goal.option_any]: | 
					
						
							| 
									
										
										
										
											2024-05-02 08:26:17 -05:00
										 |  |  |  |             from .GodhomeData import godhome_event_names | 
					
						
							|  |  |  |  |             all_event_names.update(set(godhome_event_names)) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  |         # Link regions | 
					
						
							| 
									
										
										
										
											2024-05-02 08:26:17 -05:00
										 |  |  |  |         for event_name in all_event_names: | 
					
						
							| 
									
										
										
										
											2022-06-25 11:15:03 -07:00
										 |  |  |  |             #if event_name in wp_exclusions: | 
					
						
							|  |  |  |  |             #    continue | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |             loc = HKLocation(self.player, event_name, None, menu_region) | 
					
						
							|  |  |  |  |             loc.place_locked_item(HKItem(event_name, | 
					
						
							| 
									
										
										
										
											2022-06-25 11:15:03 -07:00
										 |  |  |  |                                          True, #event_name not in wp_exclusions, | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |                                          None, "Event", self.player)) | 
					
						
							|  |  |  |  |             menu_region.locations.append(loc) | 
					
						
							|  |  |  |  |         for entry_transition, exit_transition in connectors.items(): | 
					
						
							| 
									
										
										
										
											2022-06-25 11:15:03 -07:00
										 |  |  |  |             #if entry_transition in wp_exclusions: | 
					
						
							|  |  |  |  |             #    continue | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |             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, | 
					
						
							| 
									
										
										
										
											2022-06-25 11:15:03 -07:00
										 |  |  |  |                                              True, #exit_transition not in wp_exclusions, | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |                                              None, "Event", self.player)) | 
					
						
							|  |  |  |  |                 menu_region.locations.append(loc) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def create_items(self): | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |         unfilled_locations = 0 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         # Generate item pool and associated locations (paired in HK) | 
					
						
							|  |  |  |  |         pool: typing.List[HKItem] = [] | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |         wp_exclusions = self.white_palace_exclusions() | 
					
						
							|  |  |  |  |         junk_replace: typing.Set[str] = set() | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         if self.options.RemoveSpellUpgrades: | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |             junk_replace.update(("Abyss_Shriek", "Shade_Soul", "Descending_Dark")) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         randomized_starting_items = set() | 
					
						
							|  |  |  |  |         for attr, items in randomizable_starting_items.items(): | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             if getattr(self.options, attr): | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                 randomized_starting_items.update(items) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         # noinspection PyShadowingNames | 
					
						
							| 
									
										
										
										
											2024-03-13 06:45:43 -05:00
										 |  |  |  |         def _add(item_name: str, location_name: str, randomized: bool): | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |             """
 | 
					
						
							|  |  |  |  |             Adds a pairing of an item and location, doing appropriate checks to see if it should be vanilla or not. | 
					
						
							|  |  |  |  |             """
 | 
					
						
							|  |  |  |  |             nonlocal unfilled_locations | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             vanilla = not randomized | 
					
						
							|  |  |  |  |             excluded = False | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if not vanilla and location_name in wp_exclusions: | 
					
						
							|  |  |  |  |                 if location_name == 'King_Fragment': | 
					
						
							|  |  |  |  |                     excluded = True | 
					
						
							|  |  |  |  |                 else: | 
					
						
							|  |  |  |  |                     vanilla = True | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if item_name in junk_replace: | 
					
						
							|  |  |  |  |                 item_name = self.get_filler_item_name() | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             item = self.create_item(item_name) if not vanilla or location_name == "Start" or self.options.AddUnshuffledLocations else self.create_event(item_name) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |             if location_name == "Start": | 
					
						
							|  |  |  |  |                 if item_name in randomized_starting_items: | 
					
						
							| 
									
										
										
										
											2022-07-25 16:19:07 -04:00
										 |  |  |  |                     if item_name == "Focus": | 
					
						
							|  |  |  |  |                         self.create_location("Focus") | 
					
						
							|  |  |  |  |                         unfilled_locations += 1 | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                     pool.append(item) | 
					
						
							|  |  |  |  |                 else: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |  |                     self.multiworld.push_precollected(item) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                 return | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if vanilla: | 
					
						
							|  |  |  |  |                 location = self.create_vanilla_location(location_name, item) | 
					
						
							|  |  |  |  |             else: | 
					
						
							|  |  |  |  |                 pool.append(item) | 
					
						
							|  |  |  |  |                 if location_name in multi_locations:  # Create shop locations later. | 
					
						
							|  |  |  |  |                     return | 
					
						
							|  |  |  |  |                 location = self.create_location(location_name) | 
					
						
							|  |  |  |  |                 unfilled_locations += 1 | 
					
						
							|  |  |  |  |             if excluded: | 
					
						
							|  |  |  |  |                 location.progress_type = LocationProgressType.EXCLUDED | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         for option_key, option in hollow_knight_randomize_options.items(): | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             randomized = getattr(self.options, option_key) | 
					
						
							|  |  |  |  |             if all([not randomized, option_key in logicless_options, not self.options.AddUnshuffledLocations]): | 
					
						
							| 
									
										
										
										
											2024-03-13 06:45:43 -05:00
										 |  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2022-06-25 11:15:03 -07:00
										 |  |  |  |             for item_name, location_name in zip(option.items, option.locations): | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                 if item_name in junk_replace: | 
					
						
							|  |  |  |  |                     item_name = self.get_filler_item_name() | 
					
						
							| 
									
										
										
										
											2022-06-25 11:15:03 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |                 if (item_name == "Crystal_Heart" and self.options.SplitCrystalHeart) or \ | 
					
						
							|  |  |  |  |                         (item_name == "Mothwing_Cloak" and self.options.SplitMothwingCloak): | 
					
						
							| 
									
										
										
										
											2024-03-13 06:45:43 -05:00
										 |  |  |  |                     _add("Left_" + item_name, location_name, randomized) | 
					
						
							|  |  |  |  |                     _add("Right_" + item_name, "Split_" + location_name, randomized) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |                 if item_name == "Mantis_Claw" and self.options.SplitMantisClaw: | 
					
						
							| 
									
										
										
										
											2024-03-13 06:45:43 -05:00
										 |  |  |  |                     _add("Left_" + item_name, "Left_" + location_name, randomized) | 
					
						
							|  |  |  |  |                     _add("Right_" + item_name, "Right_" + location_name, randomized) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |                 if item_name == "Shade_Cloak" and self.options.SplitMothwingCloak: | 
					
						
							|  |  |  |  |                     if self.random.randint(0, 1): | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                         item_name = "Left_Mothwing_Cloak" | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |                     else: | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                         item_name = "Right_Mothwing_Cloak" | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |                 if item_name == "Grimmchild2" and self.options.RandomizeGrimmkinFlames and self.options.RandomizeCharms: | 
					
						
							| 
									
										
										
										
											2024-03-13 06:45:43 -05:00
										 |  |  |  |                     _add("Grimmchild1", location_name, randomized) | 
					
						
							|  |  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-13 06:45:43 -05:00
										 |  |  |  |                 _add(item_name, location_name, randomized) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         if self.options.RandomizeElevatorPass: | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |             randomized = True | 
					
						
							| 
									
										
										
										
											2024-03-13 06:45:43 -05:00
										 |  |  |  |             _add("Elevator_Pass", "Elevator_Pass", randomized) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         for shop, locations in self.created_multi_locations.items(): | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             for _ in range(len(locations), getattr(self.options, shop_to_option[shop]).value): | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                 loc = self.create_location(shop) | 
					
						
							|  |  |  |  |                 unfilled_locations += 1 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         # Balance the pool | 
					
						
							|  |  |  |  |         item_count = len(pool) | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         additional_shop_items = max(item_count - unfilled_locations, self.options.ExtraShopSlots.value) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         # Add additional shop items, as needed. | 
					
						
							|  |  |  |  |         if additional_shop_items > 0: | 
					
						
							|  |  |  |  |             shops = list(shop for shop, locations in self.created_multi_locations.items() if len(locations) < 16) | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             if not self.options.EggShopSlots:  # No eggshop, so don't place items there | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                 shops.remove('Egg_Shop') | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-14 18:57:47 -06:00
										 |  |  |  |             if shops: | 
					
						
							|  |  |  |  |                 for _ in range(additional_shop_items): | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |                     shop = self.random.choice(shops) | 
					
						
							| 
									
										
										
										
											2022-11-14 18:57:47 -06:00
										 |  |  |  |                     loc = self.create_location(shop) | 
					
						
							|  |  |  |  |                     unfilled_locations += 1 | 
					
						
							|  |  |  |  |                     if len(self.created_multi_locations[shop]) >= 16: | 
					
						
							|  |  |  |  |                         shops.remove(shop) | 
					
						
							|  |  |  |  |                         if not shops: | 
					
						
							|  |  |  |  |                             break | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         # Create filler items, if needed | 
					
						
							|  |  |  |  |         if item_count < unfilled_locations: | 
					
						
							|  |  |  |  |             pool.extend(self.create_item(self.get_filler_item_name()) for _ in range(unfilled_locations - item_count)) | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |  |         self.multiworld.itempool += pool | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |         self.apply_costsanity() | 
					
						
							|  |  |  |  |         self.sort_shops_by_cost() | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def sort_shops_by_cost(self): | 
					
						
							|  |  |  |  |         for shop, locations in self.created_multi_locations.items(): | 
					
						
							|  |  |  |  |             randomized_locations = list(loc for loc in locations if not loc.vanilla) | 
					
						
							|  |  |  |  |             prices = sorted( | 
					
						
							|  |  |  |  |                 (loc.costs for loc in randomized_locations), | 
					
						
							|  |  |  |  |                 key=lambda costs: (len(costs),) + tuple(costs.values()) | 
					
						
							|  |  |  |  |             ) | 
					
						
							|  |  |  |  |             for loc, costs in zip(randomized_locations, prices): | 
					
						
							|  |  |  |  |                 loc.costs = costs | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def apply_costsanity(self): | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         setting = self.options.CostSanity.value | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |         if not setting: | 
					
						
							|  |  |  |  |             return  # noop | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         def _compute_weights(weights: dict, desc: str) -> typing.Dict[str, int]: | 
					
						
							|  |  |  |  |             if all(x == 0 for x in weights.values()): | 
					
						
							|  |  |  |  |                 logger.warning( | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |  |                     f"All {desc} weights were zero for {self.multiworld.player_name[self.player]}." | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                     f" Setting them to one instead." | 
					
						
							|  |  |  |  |                 ) | 
					
						
							|  |  |  |  |                 weights = {k: 1 for k in weights} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             return {k: v for k, v in weights.items() if v} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         random = self.random | 
					
						
							|  |  |  |  |         hybrid_chance = getattr(self.options, f"CostSanityHybridChance").value | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |         weights = { | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             data.term: getattr(self.options, f"CostSanity{data.option}Weight").value | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |             for data in cost_terms.values() | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         weights_geoless = dict(weights) | 
					
						
							|  |  |  |  |         del weights_geoless["GEO"] | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         weights = _compute_weights(weights, "CostSanity") | 
					
						
							|  |  |  |  |         weights_geoless = _compute_weights(weights_geoless, "Geoless CostSanity") | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if hybrid_chance > 0: | 
					
						
							|  |  |  |  |             if len(weights) == 1: | 
					
						
							|  |  |  |  |                 logger.warning( | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |  |                     f"Only one cost type is available for CostSanity in {self.multiworld.player_name[self.player]}'s world." | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                     f" CostSanityHybridChance will not trigger." | 
					
						
							|  |  |  |  |                 ) | 
					
						
							|  |  |  |  |             if len(weights_geoless) == 1: | 
					
						
							|  |  |  |  |                 logger.warning( | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |  |                     f"Only one cost type is available for CostSanity in {self.multiworld.player_name[self.player]}'s world." | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                     f" CostSanityHybridChance will not trigger in geoless locations." | 
					
						
							|  |  |  |  |                 ) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |  |         for region in self.multiworld.get_regions(self.player): | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |             for location in region.locations: | 
					
						
							|  |  |  |  |                 if location.vanilla: | 
					
						
							|  |  |  |  |                     continue | 
					
						
							|  |  |  |  |                 if not location.costs: | 
					
						
							|  |  |  |  |                     continue | 
					
						
							|  |  |  |  |                 if location.name == "Vessel_Fragment-Basin": | 
					
						
							|  |  |  |  |                     continue | 
					
						
							|  |  |  |  |                 if setting == CostSanity.option_notshops and location.basename in multi_locations: | 
					
						
							|  |  |  |  |                     continue | 
					
						
							|  |  |  |  |                 if setting == CostSanity.option_shopsonly and location.basename not in multi_locations: | 
					
						
							|  |  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2024-06-02 21:39:34 -05:00
										 |  |  |  |                 if location.basename in {'Grubfather', 'Seer', 'Egg_Shop'}: | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                     our_weights = dict(weights_geoless) | 
					
						
							| 
									
										
										
										
											2022-06-25 11:15:03 -07:00
										 |  |  |  |                 else: | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                     our_weights = dict(weights) | 
					
						
							| 
									
										
										
										
											2022-06-25 11:15:03 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                 rolls = 1 | 
					
						
							|  |  |  |  |                 if random.randrange(100) < hybrid_chance: | 
					
						
							|  |  |  |  |                     rolls = 2 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 if rolls > len(our_weights): | 
					
						
							|  |  |  |  |                     terms = list(our_weights.keys())  # Can't randomly choose cost types, using all of them. | 
					
						
							|  |  |  |  |                 else: | 
					
						
							|  |  |  |  |                     terms = [] | 
					
						
							|  |  |  |  |                     for _ in range(rolls): | 
					
						
							|  |  |  |  |                         term = random.choices(list(our_weights.keys()), list(our_weights.values()))[0] | 
					
						
							|  |  |  |  |                         del our_weights[term] | 
					
						
							|  |  |  |  |                         terms.append(term) | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                 location.costs = {term: random.randint(*self.ranges[term]) for term in terms} | 
					
						
							|  |  |  |  |                 location.sort_costs() | 
					
						
							| 
									
										
										
										
											2022-04-03 22:05:20 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  |     def set_rules(self): | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         multiworld = self.multiworld | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         player = self.player | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         goal = self.options.Goal | 
					
						
							| 
									
										
										
										
											2023-12-12 20:11:10 -06:00
										 |  |  |  |         if goal == Goal.option_hollowknight: | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             multiworld.completion_condition[player] = lambda state: _hk_can_beat_thk(state, player) | 
					
						
							| 
									
										
										
										
											2023-12-12 20:11:10 -06:00
										 |  |  |  |         elif goal == Goal.option_siblings: | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             multiworld.completion_condition[player] = lambda state: _hk_siblings_ending(state, player) | 
					
						
							| 
									
										
										
										
											2023-12-12 20:11:10 -06:00
										 |  |  |  |         elif goal == Goal.option_radiance: | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             multiworld.completion_condition[player] = lambda state: _hk_can_beat_radiance(state, player) | 
					
						
							| 
									
										
										
										
											2024-04-09 14:12:50 -05:00
										 |  |  |  |         elif goal == Goal.option_godhome: | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             multiworld.completion_condition[player] = lambda state: state.count("Defeated_Pantheon_5", player) | 
					
						
							| 
									
										
										
										
											2024-04-09 14:12:50 -05:00
										 |  |  |  |         elif goal == Goal.option_godhome_flower: | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             multiworld.completion_condition[player] = lambda state: state.count("Godhome_Flower_Quest", player) | 
					
						
							| 
									
										
										
										
											2024-08-08 13:33:13 -05:00
										 |  |  |  |         elif goal == Goal.option_grub_hunt: | 
					
						
							|  |  |  |  |             pass  # will set in stage_pre_fill() | 
					
						
							| 
									
										
										
										
											2023-12-12 20:11:10 -06:00
										 |  |  |  |         else: | 
					
						
							|  |  |  |  |             # Any goal | 
					
						
							| 
									
										
										
										
											2024-08-08 13:33:13 -05:00
										 |  |  |  |             multiworld.completion_condition[player] = lambda state: _hk_siblings_ending(state, player) and \ | 
					
						
							|  |  |  |  |                 _hk_can_beat_radiance(state, player) and state.count("Godhome_Flower_Quest", player) | 
					
						
							| 
									
										
											  
											
												Hollow Knight updates (goals, WP/POP, etc.) (#438)
* Hollow Knight updates:
- Add configurable goals (Any, THK, Siblings, Radiance)
  - Change base logic to require Opened_Black_Egg_Temple instead of
    requiring 3 dreamers.  This is future-proof for transition rando,
    where Black Egg might not have been located yet.
  - Add combat logic for THK and Radiance on par with Rando4's boss logic,
    so itemless HK shouldn't be required.
- Existing completion logic now uses Black_Egg_te
- Add White Palace options
  (Exclude, King Fragment Only, No Path of Pain, Include)
  - Excluded WP may still be required for King Fragment if Charms are
    not randomized
  - Simply don't place WP locations that are excluded
  - Distinguish between POP locations (required for POP), WP checks (
    actual item locations), WP transitions (relevant for future transition
    rando), and WP events (logically required to reach King Fragment)
  - Many transitions were listed twice.  Remove duplicates.
  - Sort transitions by scene
- For randomizable locations that have no logical significance when not
    randomized, simply skip adding them to the pool entirely for
    theoretically faster generation.
* Hollow Knight updates
  - Support random starting geo up to 1000 geo.
  - Always include locations rather than dropping unrandomized "logicless"
    ones, as it is required to best support same-slot coop.
											
										 
											2022-06-12 23:23:03 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         set_rules(self) | 
					
						
							| 
									
										
										
										
											2021-06-26 11:18:12 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-08 13:33:13 -05:00
										 |  |  |  |     @classmethod | 
					
						
							|  |  |  |  |     def stage_pre_fill(cls, multiworld: "MultiWorld"): | 
					
						
							|  |  |  |  |         def set_goal(player, grub_rule: typing.Callable[[CollectionState], bool]): | 
					
						
							|  |  |  |  |             world = multiworld.worlds[player] | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if world.options.Goal == "grub_hunt": | 
					
						
							|  |  |  |  |                 multiworld.completion_condition[player] = grub_rule | 
					
						
							|  |  |  |  |             else: | 
					
						
							|  |  |  |  |                 old_rule = multiworld.completion_condition[player] | 
					
						
							|  |  |  |  |                 multiworld.completion_condition[player] = lambda state: old_rule(state) and grub_rule(state) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         worlds = [world for world in multiworld.get_game_worlds(cls.game) if world.options.Goal in ["any", "grub_hunt"]] | 
					
						
							|  |  |  |  |         if worlds: | 
					
						
							|  |  |  |  |             grubs = [item for item in multiworld.get_items() if item.name == "Grub"] | 
					
						
							| 
									
										
										
										
											2024-08-09 16:02:41 +01:00
										 |  |  |  |         all_grub_players = [world.player for world in worlds if world.options.GrubHuntGoal == GrubHuntGoal.special_range_names["all"]] | 
					
						
							| 
									
										
										
										
											2024-08-08 13:33:13 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         if all_grub_players: | 
					
						
							|  |  |  |  |             group_lookup = defaultdict(set) | 
					
						
							|  |  |  |  |             for group_id, group in multiworld.groups.items(): | 
					
						
							|  |  |  |  |                 for player in group["players"]: | 
					
						
							|  |  |  |  |                     group_lookup[group_id].add(player) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             grub_count_per_player = Counter() | 
					
						
							|  |  |  |  |             per_player_grubs_per_player = defaultdict(Counter) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             for grub in grubs: | 
					
						
							|  |  |  |  |                 player = grub.player | 
					
						
							|  |  |  |  |                 if player in group_lookup: | 
					
						
							|  |  |  |  |                     for real_player in group_lookup[player]: | 
					
						
							|  |  |  |  |                         per_player_grubs_per_player[real_player][player] += 1 | 
					
						
							|  |  |  |  |                 else: | 
					
						
							|  |  |  |  |                     per_player_grubs_per_player[player][player] += 1 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 if grub.location and grub.location.player in group_lookup.keys(): | 
					
						
							|  |  |  |  |                     for real_player in group_lookup[grub.location.player]: | 
					
						
							|  |  |  |  |                         grub_count_per_player[real_player] += 1 | 
					
						
							|  |  |  |  |                 else: | 
					
						
							|  |  |  |  |                     grub_count_per_player[player] += 1 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             for player, count in grub_count_per_player.items(): | 
					
						
							|  |  |  |  |                 multiworld.worlds[player].grub_count = count | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             for player, grub_player_count in per_player_grubs_per_player.items(): | 
					
						
							|  |  |  |  |                 if player in all_grub_players: | 
					
						
							|  |  |  |  |                     set_goal(player, lambda state, g=grub_player_count: all(state.has("Grub", owner, count) for owner, count in g.items())) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         for world in worlds: | 
					
						
							|  |  |  |  |             if world.player not in all_grub_players: | 
					
						
							|  |  |  |  |                 world.grub_count = world.options.GrubHuntGoal.value | 
					
						
							|  |  |  |  |                 player = world.player | 
					
						
							|  |  |  |  |                 set_goal(player, lambda state, p=player, c=world.grub_count: state.has("Grub", p, c)) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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"] = {} | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         for option_name in hollow_knight_options: | 
					
						
							|  |  |  |  |             option = getattr(self.options, option_name) | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         slot_data["seed"] = self.random.randint(-2147483647, 2147483646) | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |         # Backwards compatibility for shop cost data (HKAP < 0.1.0) | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         if not self.options.CostSanity: | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |             for shop, terms in shop_cost_types.items(): | 
					
						
							|  |  |  |  |                 unit = cost_terms[next(iter(terms))].option | 
					
						
							|  |  |  |  |                 if unit == "Geo": | 
					
						
							|  |  |  |  |                     continue | 
					
						
							|  |  |  |  |                 slot_data[f"{unit}_costs"] = { | 
					
						
							|  |  |  |  |                     loc.name: next(iter(loc.costs.values())) | 
					
						
							|  |  |  |  |                     for loc in self.created_multi_locations[shop] | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         # HKAP 0.1.0 and later cost data. | 
					
						
							|  |  |  |  |         location_costs = {} | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |  |         for region in self.multiworld.get_regions(self.player): | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |             for location in region.locations: | 
					
						
							|  |  |  |  |                 if location.costs: | 
					
						
							|  |  |  |  |                     location_costs[location.name] = location.costs | 
					
						
							|  |  |  |  |         slot_data["location_costs"] = location_costs | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         slot_data["notch_costs"] = self.charm_costs | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-08 13:33:13 -05:00
										 |  |  |  |         slot_data["grub_count"] = self.grub_count | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-13 06:45:43 -05:00
										 |  |  |  |     def create_event(self, name: str) -> HKItem: | 
					
						
							|  |  |  |  |         item_data = item_table[name] | 
					
						
							|  |  |  |  |         return HKItem(name, item_data.advancement, None, item_data.type, self.player) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |     def create_location(self, name: str, vanilla=False) -> HKLocation: | 
					
						
							|  |  |  |  |         costs = None | 
					
						
							|  |  |  |  |         basename = name | 
					
						
							|  |  |  |  |         if name in shop_cost_types: | 
					
						
							|  |  |  |  |             costs = { | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |                 term: self.random.randint(*self.ranges[term]) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                 for term in shop_cost_types[name] | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         elif name in vanilla_location_costs: | 
					
						
							|  |  |  |  |             costs = vanilla_location_costs[name] | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         multi = self.created_multi_locations.get(name) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if multi is not None: | 
					
						
							|  |  |  |  |             i = len(multi) + 1 | 
					
						
							|  |  |  |  |             name = f"{name}_{i}" | 
					
						
							| 
									
										
										
										
											2021-07-21 18:08:15 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |  |         region = self.multiworld.get_region("Menu", self.player) | 
					
						
							| 
									
										
										
										
											2024-03-13 06:45:43 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         if vanilla and not self.options.AddUnshuffledLocations: | 
					
						
							| 
									
										
										
										
											2024-03-13 06:45:43 -05:00
										 |  |  |  |             loc = HKLocation(self.player, name, | 
					
						
							|  |  |  |  |                              None, region, costs=costs, vanilla=vanilla, | 
					
						
							|  |  |  |  |                              basename=basename) | 
					
						
							|  |  |  |  |         else: | 
					
						
							|  |  |  |  |             loc = HKLocation(self.player, name, | 
					
						
							|  |  |  |  |                              self.location_name_to_id[name], region, costs=costs, vanilla=vanilla, | 
					
						
							|  |  |  |  |                              basename=basename) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         if multi is not None: | 
					
						
							|  |  |  |  |             multi.append(loc) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         region.locations.append(loc) | 
					
						
							|  |  |  |  |         return loc | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |     def create_vanilla_location(self, location: str, item: Item): | 
					
						
							|  |  |  |  |         costs = self.vanilla_shop_costs.get((location, item.name)) | 
					
						
							|  |  |  |  |         location = self.create_location(location, vanilla=True) | 
					
						
							|  |  |  |  |         location.place_locked_item(item) | 
					
						
							|  |  |  |  |         if costs: | 
					
						
							|  |  |  |  |             location.costs = costs.pop() | 
					
						
							| 
									
										
										
										
											2022-07-25 16:19:07 -04:00
										 |  |  |  |         return location | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     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(): | 
					
						
							| 
									
										
										
										
											2023-11-02 00:41:20 -05:00
										 |  |  |  |                 state.prog_items[item.player][effect_name] += effect_value | 
					
						
							| 
									
										
										
										
											2024-09-05 14:19:37 -05:00
										 |  |  |  |             if item.name in {"Left_Mothwing_Cloak", "Right_Mothwing_Cloak"}: | 
					
						
							|  |  |  |  |                 if state.prog_items[item.player].get('RIGHTDASH', 0) and \ | 
					
						
							|  |  |  |  |                         state.prog_items[item.player].get('LEFTDASH', 0): | 
					
						
							|  |  |  |  |                     (state.prog_items[item.player]["RIGHTDASH"], state.prog_items[item.player]["LEFTDASH"]) = \ | 
					
						
							|  |  |  |  |                         ([max(state.prog_items[item.player]["RIGHTDASH"], state.prog_items[item.player]["LEFTDASH"])] * 2) | 
					
						
							| 
									
										
										
										
											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(): | 
					
						
							| 
									
										
										
										
											2023-11-02 00:41:20 -05:00
										 |  |  |  |                 if state.prog_items[item.player][effect_name] == effect_value: | 
					
						
							|  |  |  |  |                     del state.prog_items[item.player][effect_name] | 
					
						
							| 
									
										
										
										
											2024-07-24 20:08:58 -05:00
										 |  |  |  |                 else: | 
					
						
							|  |  |  |  |                     state.prog_items[item.player][effect_name] -= effect_value | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         return change | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-03 00:12:37 +02:00
										 |  |  |  |     @classmethod | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |     def stage_write_spoiler(cls, multiworld: MultiWorld, spoiler_handle): | 
					
						
							|  |  |  |  |         hk_players = multiworld.get_game_players(cls.game) | 
					
						
							| 
									
										
										
										
											2022-04-03 00:12:37 +02:00
										 |  |  |  |         spoiler_handle.write('\n\nCharm Notches:') | 
					
						
							|  |  |  |  |         for player in hk_players: | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             name = multiworld.get_player_name(player) | 
					
						
							| 
									
										
										
										
											2022-04-03 00:12:37 +02:00
										 |  |  |  |             spoiler_handle.write(f'\n{name}\n') | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             hk_world: HKWorld = multiworld.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: | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             name = multiworld.get_player_name(player) | 
					
						
							| 
									
										
										
										
											2022-04-04 00:40:19 +02:00
										 |  |  |  |             spoiler_handle.write(f'\n{name}\n') | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             hk_world: HKWorld = multiworld.worlds[player] | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |             if hk_world.options.CostSanity: | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                 for loc in sorted( | 
					
						
							|  |  |  |  |                     ( | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |                         loc for loc in itertools.chain(*(region.locations for region in multiworld.get_regions(player))) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                         if loc.costs | 
					
						
							|  |  |  |  |                     ), key=operator.attrgetter('name') | 
					
						
							|  |  |  |  |                 ): | 
					
						
							|  |  |  |  |                     spoiler_handle.write(f"\n{loc}: {loc.item} costing {loc.cost_text()}") | 
					
						
							|  |  |  |  |             else: | 
					
						
							|  |  |  |  |                 for shop_name, locations in hk_world.created_multi_locations.items(): | 
					
						
							|  |  |  |  |                     for loc in locations: | 
					
						
							|  |  |  |  |                         spoiler_handle.write(f"\n{loc}: {loc.item} costing {loc.cost_text()}") | 
					
						
							| 
									
										
										
										
											2022-04-04 00:40:19 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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: | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |             i = len(self.created_multi_locations[base]) + 1 | 
					
						
							|  |  |  |  |         assert 1 <= 16, "limited number of multi location IDs reserved." | 
					
						
							| 
									
										
										
										
											2022-04-03 22:05:20 +02:00
										 |  |  |  |         return f"{base}_{i}" | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |     def get_filler_item_name(self) -> str: | 
					
						
							|  |  |  |  |         if self.player not in self.cached_filler_items: | 
					
						
							|  |  |  |  |             fillers = ["One_Geo", "Soul_Refill"] | 
					
						
							|  |  |  |  |             exclusions = self.white_palace_exclusions() | 
					
						
							|  |  |  |  |             for group in ( | 
					
						
							|  |  |  |  |                     'RandomizeGeoRocks', 'RandomizeSoulTotems', 'RandomizeLoreTablets', 'RandomizeJunkPitChests', | 
					
						
							|  |  |  |  |                     'RandomizeRancidEggs' | 
					
						
							|  |  |  |  |             ): | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |                 if getattr(self.options, group): | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |                     fillers.extend(item for item in hollow_knight_randomize_options[group].items if item not in | 
					
						
							|  |  |  |  |                                    exclusions) | 
					
						
							|  |  |  |  |             self.cached_filler_items[self.player] = fillers | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         return self.random.choice(self.cached_filler_items[self.player]) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  | def create_region(multiworld: MultiWorld, player: int, name: str, location_names=None) -> Region: | 
					
						
							|  |  |  |  |     ret = Region(name, player, multiworld) | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  |  |     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-07-03 08:10:10 -07:00
										 |  |  |  |     costs: typing.Dict[str, int] = None | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |     unit: typing.Optional[str] = None | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |     vanilla = False | 
					
						
							|  |  |  |  |     basename: str | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def sort_costs(self): | 
					
						
							|  |  |  |  |         if self.costs is None: | 
					
						
							|  |  |  |  |             return | 
					
						
							|  |  |  |  |         self.costs = {k: self.costs[k] for k in sorted(self.costs.keys(), key=lambda x: cost_terms[x].sort)} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def __init__( | 
					
						
							|  |  |  |  |             self, player: int, name: str, code=None, parent=None, | 
					
						
							|  |  |  |  |             costs: typing.Dict[str, int] = None, vanilla: bool = False, basename: str = None | 
					
						
							|  |  |  |  |     ): | 
					
						
							|  |  |  |  |         self.basename = basename or name | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         super(HKLocation, self).__init__(player, name, code if code else None, parent) | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |         self.vanilla = vanilla | 
					
						
							|  |  |  |  |         if costs: | 
					
						
							|  |  |  |  |             self.costs = dict(costs) | 
					
						
							|  |  |  |  |             self.sort_costs() | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     def cost_text(self, separator=" and "): | 
					
						
							|  |  |  |  |         if self.costs is None: | 
					
						
							|  |  |  |  |             return None | 
					
						
							|  |  |  |  |         return separator.join( | 
					
						
							|  |  |  |  |             f"{value} {cost_terms[term].singular if value == 1 else cost_terms[term].plural}" | 
					
						
							|  |  |  |  |             for term, value in self.costs.items() | 
					
						
							|  |  |  |  |         ) | 
					
						
							| 
									
										
										
										
											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" | 
					
						
							| 
									
										
										
										
											2022-08-06 00:49:54 +02:00
										 |  |  |  |     type: str | 
					
						
							| 
									
										
										
										
											2021-02-22 11:18:53 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-06 00:49:54 +02:00
										 |  |  |  |     def __init__(self, name, advancement, code, type: str, player: int = None): | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  |         if name == "Mimic_Grub": | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  |  |             classification = ItemClassification.trap | 
					
						
							| 
									
										
										
										
											2024-05-28 20:37:44 -05:00
										 |  |  |  |         elif name == "Godtuner": | 
					
						
							|  |  |  |  |             classification = ItemClassification.progression | 
					
						
							| 
									
										
										
										
											2022-10-09 19:26:33 -05:00
										 |  |  |  |         elif type in ("Grub", "DreamWarrior", "Root", "Egg", "Dreamer"): | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  |  |             classification = ItemClassification.progression_skip_balancing | 
					
						
							|  |  |  |  |         elif type == "Charm" and name not in progression_charms: | 
					
						
							|  |  |  |  |             classification = ItemClassification.progression_skip_balancing | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |         elif type in ("Map", "Journal"): | 
					
						
							|  |  |  |  |             classification = ItemClassification.filler | 
					
						
							| 
									
										
										
										
											2023-07-11 04:49:40 -05:00
										 |  |  |  |         elif type in ("Ore", "Vessel"): | 
					
						
							| 
									
										
										
										
											2022-07-03 08:10:10 -07:00
										 |  |  |  |             classification = ItemClassification.useful | 
					
						
							| 
									
										
										
										
											2022-06-17 03:23:27 +02:00
										 |  |  |  |         elif advancement: | 
					
						
							|  |  |  |  |             classification = ItemClassification.progression | 
					
						
							|  |  |  |  |         else: | 
					
						
							|  |  |  |  |             classification = ItemClassification.filler | 
					
						
							|  |  |  |  |         super(HKItem, self).__init__(name, classification, code if code else None, player) | 
					
						
							|  |  |  |  |         self.type = type | 
					
						
							| 
									
										
										
										
											2022-04-06 00:41:15 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-24 06:02:51 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 03:23:52 +02:00
										 |  |  |  | class HKLogicMixin(LogicMixin): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |  |     multiworld: 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: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |  |         return sum(self.multiworld.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: | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         return getattr(self.multiworld.worlds[player].options, option_name).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: | 
					
						
							| 
									
										
										
										
											2024-07-28 16:27:39 -05:00
										 |  |  |  |         return self.multiworld.worlds[player].options.StartLocation == start_location |