| 
									
										
										
										
											2021-08-09 06:50:11 +02:00
										 |  |  | import logging | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2022-08-16 02:40:05 +02:00
										 |  |  | import random | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  | import threading | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  | import typing | 
					
						
							| 
									
										
										
										
											2021-07-22 15:51:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-30 17:12:33 +02:00
										 |  |  | import Utils | 
					
						
							| 
									
										
										
										
											2022-05-11 13:05:53 -05:00
										 |  |  | from BaseClasses import Item, CollectionState, Tutorial | 
					
						
							| 
									
										
										
										
											2022-08-16 02:40:05 +02:00
										 |  |  | from .Dungeons import create_dungeons | 
					
						
							| 
									
										
										
										
											2022-10-20 03:35:28 +02:00
										 |  |  | from .EntranceShuffle import link_entrances, link_inverted_entrances, plando_connect, \ | 
					
						
							|  |  |  |     indirect_connections, indirect_connections_inverted, indirect_connections_not_inverted | 
					
						
							| 
									
										
										
										
											2022-08-16 02:40:05 +02:00
										 |  |  | from .InvertedRegions import create_inverted_regions, mark_dark_world_regions | 
					
						
							|  |  |  | from .ItemPool import generate_itempool, difficulties | 
					
						
							| 
									
										
										
										
											2022-08-06 00:49:54 +02:00
										 |  |  | from .Items import item_init_table, item_name_groups, item_table, GetBeemizerItem | 
					
						
							| 
									
										
										
										
											2022-08-16 02:40:05 +02:00
										 |  |  | from .Options import alttp_options, smallkey_shuffle | 
					
						
							| 
									
										
										
										
											2022-09-18 14:30:43 +02:00
										 |  |  | from .Regions import lookup_name_to_id, create_regions, mark_light_world_regions, lookup_vanilla_location_to_entrance, \ | 
					
						
							|  |  |  |     is_main_entrance | 
					
						
							| 
									
										
										
										
											2022-10-25 13:54:43 -04:00
										 |  |  | from .Client import ALTTPSNIClient | 
					
						
							| 
									
										
										
										
											2022-08-10 13:21:52 -07:00
										 |  |  | from .Rom import LocalRom, patch_rom, patch_race_rom, check_enemizer, patch_enemizer, apply_rom_settings, \ | 
					
						
							|  |  |  |     get_hash_string, get_base_rom_path, LttPDeltaPatch | 
					
						
							| 
									
										
										
										
											2022-08-16 02:40:05 +02:00
										 |  |  | from .Rules import set_rules | 
					
						
							|  |  |  | from .Shops import create_shops, ShopSlotFill | 
					
						
							|  |  |  | from .SubClasses import ALttPItem | 
					
						
							| 
									
										
										
										
											2022-09-30 04:58:19 +02:00
										 |  |  | from worlds.AutoWorld import World, WebWorld, LogicMixin | 
					
						
							| 
									
										
										
										
											2020-10-24 05:38:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-09 06:50:11 +02:00
										 |  |  | lttp_logger = logging.getLogger("A Link to the Past") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-10 07:23:03 -04:00
										 |  |  | extras_list = sum(difficulties['normal'].extras[0:5], []) | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-18 14:30:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-11 13:05:53 -05:00
										 |  |  | class ALTTPWeb(WebWorld): | 
					
						
							|  |  |  |     setup_en = Tutorial( | 
					
						
							|  |  |  |         "Multiworld Setup Tutorial", | 
					
						
							|  |  |  |         "A guide to setting up the Archipelago ALttP Software on your computer. This guide covers single-player, multiworld, and related software.", | 
					
						
							|  |  |  |         "English", | 
					
						
							|  |  |  |         "multiworld_en.md", | 
					
						
							|  |  |  |         "multiworld/en", | 
					
						
							|  |  |  |         ["Farrak Kilhn"] | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     setup_de = Tutorial( | 
					
						
							|  |  |  |         setup_en.tutorial_name, | 
					
						
							|  |  |  |         setup_en.description, | 
					
						
							|  |  |  |         "Deutsch", | 
					
						
							|  |  |  |         "multiworld_de.md", | 
					
						
							|  |  |  |         "multiworld/de", | 
					
						
							|  |  |  |         ["Fischfilet"] | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     setup_es = Tutorial( | 
					
						
							|  |  |  |         setup_en.tutorial_name, | 
					
						
							|  |  |  |         setup_en.description, | 
					
						
							|  |  |  |         "Español", | 
					
						
							|  |  |  |         "multiworld_es.md", | 
					
						
							|  |  |  |         "multiworld/es", | 
					
						
							|  |  |  |         ["Edos"] | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     setup_fr = Tutorial( | 
					
						
							|  |  |  |         setup_en.tutorial_name, | 
					
						
							|  |  |  |         setup_en.description, | 
					
						
							|  |  |  |         "Français", | 
					
						
							|  |  |  |         "multiworld_fr.md", | 
					
						
							|  |  |  |         "multiworld/fr", | 
					
						
							|  |  |  |         ["Coxla"] | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     msu = Tutorial( | 
					
						
							|  |  |  |         "MSU-1 Setup Tutorial", | 
					
						
							|  |  |  |         "A guide to setting up MSU-1, which allows for custom in-game music.", | 
					
						
							|  |  |  |         "English", | 
					
						
							|  |  |  |         "msu1_en.md", | 
					
						
							|  |  |  |         "msu1/en", | 
					
						
							|  |  |  |         ["Farrak Kilhn"] | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     msu_es = Tutorial( | 
					
						
							|  |  |  |         msu.tutorial_name, | 
					
						
							|  |  |  |         msu.description, | 
					
						
							|  |  |  |         "Español", | 
					
						
							|  |  |  |         "msu1_es.md", | 
					
						
							| 
									
										
										
										
											2022-06-25 06:15:57 -05:00
										 |  |  |         "msu1/es", | 
					
						
							| 
									
										
										
										
											2022-05-11 13:05:53 -05:00
										 |  |  |         ["Edos"] | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     msu_fr = Tutorial( | 
					
						
							|  |  |  |         msu.tutorial_name, | 
					
						
							|  |  |  |         msu.description, | 
					
						
							|  |  |  |         "Français", | 
					
						
							|  |  |  |         "msu1_fr.md", | 
					
						
							|  |  |  |         "msu1/fr", | 
					
						
							|  |  |  |         ["Coxla"] | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     plando = Tutorial( | 
					
						
							|  |  |  |         "Plando Tutorial", | 
					
						
							|  |  |  |         "A guide to creating Multiworld Plandos with LTTP", | 
					
						
							|  |  |  |         "English", | 
					
						
							|  |  |  |         "plando_en.md", | 
					
						
							|  |  |  |         "plando/en", | 
					
						
							|  |  |  |         ["Berserker"] | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     tutorials = [setup_en, setup_de, setup_es, setup_fr, msu, msu_es, msu_fr, plando] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-11 14:22:44 +02:00
										 |  |  | class ALTTPWorld(World): | 
					
						
							| 
									
										
										
										
											2021-08-31 17:28:46 -04:00
										 |  |  |     """
 | 
					
						
							|  |  |  |     The Legend of Zelda: A Link to the Past is an action/adventure game. Take on the role of | 
					
						
							|  |  |  |     Link, a boy who is destined to save the land of Hyrule. Delve through three palaces and nine | 
					
						
							|  |  |  |     dungeons on your quest to rescue the descendents of the seven wise men and defeat the evil | 
					
						
							|  |  |  |     Ganon! | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2021-06-11 14:22:44 +02:00
										 |  |  |     game: str = "A Link to the Past" | 
					
						
							| 
									
										
										
										
											2022-08-15 16:46:59 -05:00
										 |  |  |     option_definitions = alttp_options | 
					
						
							| 
									
										
										
										
											2021-07-08 11:07:41 +02:00
										 |  |  |     topology_present = True | 
					
						
							| 
									
										
										
										
											2021-07-12 13:54:47 +02:00
										 |  |  |     item_name_groups = item_name_groups | 
					
						
							| 
									
										
										
										
											2021-07-12 15:33:20 +02:00
										 |  |  |     hint_blacklist = {"Triforce"} | 
					
						
							| 
									
										
										
										
											2021-07-08 11:07:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 18:05:46 +02:00
										 |  |  |     item_name_to_id = {name: data.item_code for name, data in item_table.items() if type(data.item_code) == int} | 
					
						
							|  |  |  |     location_name_to_id = lookup_name_to_id | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-27 14:52:33 +02:00
										 |  |  |     data_version = 8 | 
					
						
							| 
									
										
										
										
											2021-07-13 19:14:57 +02:00
										 |  |  |     remote_items: bool = False | 
					
						
							| 
									
										
										
										
											2021-09-23 03:48:37 +02:00
										 |  |  |     remote_start_inventory: bool = False | 
					
						
							| 
									
										
										
										
											2022-04-08 11:16:36 +02:00
										 |  |  |     required_client_version = (0, 3, 2) | 
					
						
							| 
									
										
										
										
											2022-05-11 13:05:53 -05:00
										 |  |  |     web = ALTTPWeb() | 
					
						
							| 
									
										
										
										
											2021-07-13 19:14:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-06 00:49:54 +02:00
										 |  |  |     pedestal_credit_texts: typing.Dict[int, str] = \ | 
					
						
							|  |  |  |         {data.item_code: data.pedestal_credit for data in item_table.values() if data.pedestal_credit} | 
					
						
							|  |  |  |     sickkid_credit_texts: typing.Dict[int, str] = \ | 
					
						
							|  |  |  |         {data.item_code: data.sick_kid_credit for data in item_table.values() if data.sick_kid_credit} | 
					
						
							|  |  |  |     zora_credit_texts: typing.Dict[int, str] = \ | 
					
						
							|  |  |  |         {data.item_code: data.zora_credit for data in item_table.values() if data.zora_credit} | 
					
						
							|  |  |  |     magicshop_credit_texts: typing.Dict[int, str] = \ | 
					
						
							|  |  |  |         {data.item_code: data.witch_credit for data in item_table.values() if data.witch_credit} | 
					
						
							|  |  |  |     fluteboy_credit_texts: typing.Dict[int, str] = \ | 
					
						
							|  |  |  |         {data.item_code: data.flute_boy_credit for data in item_table.values() if data.flute_boy_credit} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-22 15:51:50 +02:00
										 |  |  |     set_rules = set_rules | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     create_items = generate_itempool | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-30 17:12:33 +02:00
										 |  |  |     enemizer_path: str = Utils.get_options()["generator"]["enemizer_path"] \ | 
					
						
							|  |  |  |         if os.path.isabs(Utils.get_options()["generator"]["enemizer_path"]) \ | 
					
						
							|  |  |  |         else Utils.local_path(Utils.get_options()["generator"]["enemizer_path"]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-30 16:31:56 +02:00
										 |  |  |     def __init__(self, *args, **kwargs): | 
					
						
							|  |  |  |         self.dungeon_local_item_names = set() | 
					
						
							|  |  |  |         self.dungeon_specific_item_names = set() | 
					
						
							|  |  |  |         self.rom_name_available_event = threading.Event() | 
					
						
							| 
									
										
										
										
											2021-09-20 01:00:09 +02:00
										 |  |  |         self.has_progressive_bows = False | 
					
						
							| 
									
										
										
										
											2021-08-30 16:31:56 +02:00
										 |  |  |         super(ALTTPWorld, self).__init__(*args, **kwargs) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-29 20:37:28 -05:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def stage_assert_generate(cls, world): | 
					
						
							|  |  |  |         rom_file = get_base_rom_path() | 
					
						
							|  |  |  |         if not os.path.exists(rom_file): | 
					
						
							|  |  |  |             raise FileNotFoundError(rom_file) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  |     def generate_early(self): | 
					
						
							| 
									
										
										
										
											2022-08-30 17:12:33 +02:00
										 |  |  |         if self.use_enemizer(): | 
					
						
							|  |  |  |             check_enemizer(self.enemizer_path) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  |         player = self.player | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         world = self.multiworld | 
					
						
							| 
									
										
										
										
											2021-08-30 16:31:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  |         # system for sharing ER layouts | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |         self.er_seed = str(world.random.randint(0, 2 ** 64)) | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if "-" in world.shuffle[player]: | 
					
						
							|  |  |  |             shuffle, seed = world.shuffle[player].split("-", 1) | 
					
						
							|  |  |  |             world.shuffle[player] = shuffle | 
					
						
							|  |  |  |             if shuffle == "vanilla": | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                 self.er_seed = "vanilla" | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  |             elif seed.startswith("group-") or world.is_race: | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                 self.er_seed = get_same_seed(world, ( | 
					
						
							| 
									
										
										
										
											2022-06-01 17:29:21 +02:00
										 |  |  |                     shuffle, seed, world.retro_caves[player], world.mode[player], world.logic[player])) | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  |             else:  # not a race or group seed, use set seed as is. | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |                 self.er_seed = seed | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  |         elif world.shuffle[player] == "vanilla": | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |             self.er_seed = "vanilla" | 
					
						
							| 
									
										
										
										
											2021-08-30 18:00:39 +02:00
										 |  |  |         for dungeon_item in ["smallkey_shuffle", "bigkey_shuffle", "compass_shuffle", "map_shuffle"]: | 
					
						
							| 
									
										
										
										
											2021-08-30 16:31:56 +02:00
										 |  |  |             option = getattr(world, dungeon_item)[player] | 
					
						
							|  |  |  |             if option == "own_world": | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  |                 world.local_items[player].value |= self.item_name_groups[option.item_name_group] | 
					
						
							| 
									
										
										
										
											2021-08-30 16:31:56 +02:00
										 |  |  |             elif option == "different_world": | 
					
						
							| 
									
										
										
										
											2021-09-17 00:17:54 +02:00
										 |  |  |                 world.non_local_items[player].value |= self.item_name_groups[option.item_name_group] | 
					
						
							| 
									
										
										
										
											2021-08-30 16:31:56 +02:00
										 |  |  |             elif option.in_dungeon: | 
					
						
							|  |  |  |                 self.dungeon_local_item_names |= self.item_name_groups[option.item_name_group] | 
					
						
							|  |  |  |                 if option == "original_dungeon": | 
					
						
							|  |  |  |                     self.dungeon_specific_item_names |= self.item_name_groups[option.item_name_group] | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         world.difficulty_requirements[player] = difficulties[world.difficulty[player]] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-22 15:51:50 +02:00
										 |  |  |     def create_regions(self): | 
					
						
							| 
									
										
										
										
											2021-07-23 12:03:19 +02:00
										 |  |  |         player = self.player | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         world = self.multiworld | 
					
						
							| 
									
										
										
										
											2021-07-23 12:03:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         world.triforce_pieces_available[player] = max(world.triforce_pieces_available[player], | 
					
						
							|  |  |  |                                                       world.triforce_pieces_required[player]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if world.mode[player] != 'inverted': | 
					
						
							|  |  |  |             create_regions(world, player) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             create_inverted_regions(world, player) | 
					
						
							|  |  |  |         create_shops(world, player) | 
					
						
							|  |  |  |         create_dungeons(world, player) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if world.logic[player] not in ["noglitches", "minorglitches"] and world.shuffle[player] in \ | 
					
						
							|  |  |  |                 {"vanilla", "dungeonssimple", "dungeonsfull", "simple", "restricted", "full"}: | 
					
						
							|  |  |  |             world.fix_fake_world[player] = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # seeded entrance shuffle | 
					
						
							|  |  |  |         old_random = world.random | 
					
						
							| 
									
										
										
										
											2021-10-06 11:32:49 +02:00
										 |  |  |         world.random = random.Random(self.er_seed) | 
					
						
							| 
									
										
										
										
											2021-07-23 12:03:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if world.mode[player] != 'inverted': | 
					
						
							|  |  |  |             link_entrances(world, player) | 
					
						
							|  |  |  |             mark_light_world_regions(world, player) | 
					
						
							| 
									
										
										
										
											2022-10-20 03:35:28 +02:00
										 |  |  |             for region_name, entrance_name in indirect_connections_not_inverted.items(): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 world.register_indirect_condition(world.get_region(region_name, player), | 
					
						
							|  |  |  |                                                   world.get_entrance(entrance_name, player)) | 
					
						
							| 
									
										
										
										
											2021-07-23 12:03:19 +02:00
										 |  |  |         else: | 
					
						
							|  |  |  |             link_inverted_entrances(world, player) | 
					
						
							|  |  |  |             mark_dark_world_regions(world, player) | 
					
						
							| 
									
										
										
										
											2022-10-20 03:35:28 +02:00
										 |  |  |             for region_name, entrance_name in indirect_connections_inverted.items(): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                 world.register_indirect_condition(world.get_region(region_name, player), | 
					
						
							|  |  |  |                                                   world.get_entrance(entrance_name, player)) | 
					
						
							| 
									
										
										
										
											2021-07-23 12:03:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         world.random = old_random | 
					
						
							|  |  |  |         plando_connect(world, player) | 
					
						
							| 
									
										
										
										
											2021-07-22 15:51:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-30 04:58:19 +02:00
										 |  |  |         for region_name, entrance_name in indirect_connections.items(): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             world.register_indirect_condition(world.get_region(region_name, player), | 
					
						
							|  |  |  |                                               world.get_entrance(entrance_name, player)) | 
					
						
							| 
									
										
										
										
											2022-09-30 04:58:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-20 03:35:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |     def collect_item(self, state: CollectionState, item: Item, remove=False): | 
					
						
							|  |  |  |         item_name = item.name | 
					
						
							|  |  |  |         if item_name.startswith('Progressive '): | 
					
						
							|  |  |  |             if remove: | 
					
						
							|  |  |  |                 if 'Sword' in item_name: | 
					
						
							|  |  |  |                     if state.has('Golden Sword', item.player): | 
					
						
							|  |  |  |                         return 'Golden Sword' | 
					
						
							|  |  |  |                     elif state.has('Tempered Sword', item.player): | 
					
						
							|  |  |  |                         return 'Tempered Sword' | 
					
						
							|  |  |  |                     elif state.has('Master Sword', item.player): | 
					
						
							|  |  |  |                         return 'Master Sword' | 
					
						
							|  |  |  |                     elif state.has('Fighter Sword', item.player): | 
					
						
							|  |  |  |                         return 'Fighter Sword' | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         return None | 
					
						
							|  |  |  |                 elif 'Glove' in item.name: | 
					
						
							|  |  |  |                     if state.has('Titans Mitts', item.player): | 
					
						
							|  |  |  |                         return 'Titans Mitts' | 
					
						
							|  |  |  |                     elif state.has('Power Glove', item.player): | 
					
						
							|  |  |  |                         return 'Power Glove' | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         return None | 
					
						
							|  |  |  |                 elif 'Shield' in item_name: | 
					
						
							|  |  |  |                     if state.has('Mirror Shield', item.player): | 
					
						
							|  |  |  |                         return 'Mirror Shield' | 
					
						
							|  |  |  |                     elif state.has('Red Shield', item.player): | 
					
						
							|  |  |  |                         return 'Red Shield' | 
					
						
							|  |  |  |                     elif state.has('Blue Shield', item.player): | 
					
						
							|  |  |  |                         return 'Blue Shield' | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         return None | 
					
						
							|  |  |  |                 elif 'Bow' in item_name: | 
					
						
							|  |  |  |                     if state.has('Silver Bow', item.player): | 
					
						
							|  |  |  |                         return 'Silver Bow' | 
					
						
							|  |  |  |                     elif state.has('Bow', item.player): | 
					
						
							|  |  |  |                         return 'Bow' | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         return None | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 if 'Sword' in item_name: | 
					
						
							|  |  |  |                     if state.has('Golden Sword', item.player): | 
					
						
							|  |  |  |                         pass | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     elif state.has('Tempered Sword', item.player) and self.multiworld.difficulty_requirements[ | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |                         item.player].progressive_sword_limit >= 4: | 
					
						
							|  |  |  |                         return 'Golden Sword' | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     elif state.has('Master Sword', item.player) and self.multiworld.difficulty_requirements[ | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |                         item.player].progressive_sword_limit >= 3: | 
					
						
							|  |  |  |                         return 'Tempered Sword' | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     elif state.has('Fighter Sword', item.player) and self.multiworld.difficulty_requirements[item.player].progressive_sword_limit >= 2: | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |                         return 'Master Sword' | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     elif self.multiworld.difficulty_requirements[item.player].progressive_sword_limit >= 1: | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |                         return 'Fighter Sword' | 
					
						
							|  |  |  |                 elif 'Glove' in item_name: | 
					
						
							|  |  |  |                     if state.has('Titans Mitts', item.player): | 
					
						
							|  |  |  |                         return | 
					
						
							|  |  |  |                     elif state.has('Power Glove', item.player): | 
					
						
							|  |  |  |                         return 'Titans Mitts' | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         return 'Power Glove' | 
					
						
							|  |  |  |                 elif 'Shield' in item_name: | 
					
						
							|  |  |  |                     if state.has('Mirror Shield', item.player): | 
					
						
							|  |  |  |                         return | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     elif state.has('Red Shield', item.player) and self.multiworld.difficulty_requirements[item.player].progressive_shield_limit >= 3: | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |                         return 'Mirror Shield' | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     elif state.has('Blue Shield', item.player) and self.multiworld.difficulty_requirements[item.player].progressive_shield_limit >= 2: | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |                         return 'Red Shield' | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     elif self.multiworld.difficulty_requirements[item.player].progressive_shield_limit >= 1: | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |                         return 'Blue Shield' | 
					
						
							|  |  |  |                 elif 'Bow' in item_name: | 
					
						
							| 
									
										
										
										
											2021-08-26 16:03:22 -05:00
										 |  |  |                     if state.has('Silver Bow', item.player): | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |                         return | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     elif state.has('Bow', item.player) and (self.multiworld.difficulty_requirements[item.player].progressive_bow_limit >= 2 | 
					
						
							|  |  |  |                                                             or self.multiworld.logic[item.player] == 'noglitches' | 
					
						
							|  |  |  |                                                             or self.multiworld.swordless[item.player]): # modes where silver bow is always required for ganon | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |                         return 'Silver Bow' | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                     elif self.multiworld.difficulty_requirements[item.player].progressive_bow_limit >= 1: | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |                         return 'Bow' | 
					
						
							| 
									
										
										
										
											2021-08-10 09:47:28 +02:00
										 |  |  |         elif item.advancement: | 
					
						
							| 
									
										
										
										
											2021-08-21 06:55:08 +02:00
										 |  |  |             return item_name | 
					
						
							| 
									
										
										
										
											2020-10-24 05:38:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-09 06:50:11 +02:00
										 |  |  |     def pre_fill(self): | 
					
						
							|  |  |  |         from Fill import fill_restrictive, FillError | 
					
						
							|  |  |  |         attempts = 5 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         world = self.multiworld | 
					
						
							| 
									
										
										
										
											2021-08-09 06:50:11 +02:00
										 |  |  |         player = self.player | 
					
						
							| 
									
										
										
										
											2021-09-01 14:01:54 -05:00
										 |  |  |         all_state = world.get_all_state(use_cache=True) | 
					
						
							| 
									
										
										
										
											2021-08-09 06:50:11 +02:00
										 |  |  |         crystals = [self.create_item(name) for name in ['Red Pendant', 'Blue Pendant', 'Green Pendant', 'Crystal 1', 'Crystal 2', 'Crystal 3', 'Crystal 4', 'Crystal 7', 'Crystal 5', 'Crystal 6']] | 
					
						
							|  |  |  |         crystal_locations = [world.get_location('Turtle Rock - Prize', player), | 
					
						
							|  |  |  |                              world.get_location('Eastern Palace - Prize', player), | 
					
						
							|  |  |  |                              world.get_location('Desert Palace - Prize', player), | 
					
						
							|  |  |  |                              world.get_location('Tower of Hera - Prize', player), | 
					
						
							|  |  |  |                              world.get_location('Palace of Darkness - Prize', player), | 
					
						
							|  |  |  |                              world.get_location('Thieves\' Town - Prize', player), | 
					
						
							|  |  |  |                              world.get_location('Skull Woods - Prize', player), | 
					
						
							|  |  |  |                              world.get_location('Swamp Palace - Prize', player), | 
					
						
							|  |  |  |                              world.get_location('Ice Palace - Prize', player), | 
					
						
							|  |  |  |                              world.get_location('Misery Mire - Prize', player)] | 
					
						
							|  |  |  |         placed_prizes = {loc.item.name for loc in crystal_locations if loc.item} | 
					
						
							|  |  |  |         unplaced_prizes = [crystal for crystal in crystals if crystal.name not in placed_prizes] | 
					
						
							|  |  |  |         empty_crystal_locations = [loc for loc in crystal_locations if not loc.item] | 
					
						
							|  |  |  |         for attempt in range(attempts): | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 prizepool = unplaced_prizes.copy() | 
					
						
							|  |  |  |                 prize_locs = empty_crystal_locations.copy() | 
					
						
							|  |  |  |                 world.random.shuffle(prize_locs) | 
					
						
							|  |  |  |                 fill_restrictive(world, all_state, prize_locs, prizepool, True, lock=True) | 
					
						
							|  |  |  |             except FillError as e: | 
					
						
							|  |  |  |                 lttp_logger.exception("Failed to place dungeon prizes (%s). Will retry %s more times", e, | 
					
						
							|  |  |  |                                                 attempts - attempt) | 
					
						
							|  |  |  |                 for location in empty_crystal_locations: | 
					
						
							|  |  |  |                     location.item = None | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             break | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             raise FillError('Unable to place dungeon prizes') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def stage_pre_fill(cls, world): | 
					
						
							|  |  |  |         from .Dungeons import fill_dungeons_restrictive | 
					
						
							| 
									
										
										
										
											2022-02-13 23:02:18 +01:00
										 |  |  |         fill_dungeons_restrictive(world) | 
					
						
							| 
									
										
										
										
											2021-08-09 06:50:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-30 01:16:04 +02:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def stage_post_fill(cls, world): | 
					
						
							| 
									
										
										
										
											2021-08-30 01:18:30 +02:00
										 |  |  |         ShopSlotFill(world) | 
					
						
							| 
									
										
										
										
											2021-08-30 01:16:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-10 13:21:52 -07:00
										 |  |  |     def use_enemizer(self): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         world = self.multiworld | 
					
						
							| 
									
										
										
										
											2022-08-10 13:21:52 -07:00
										 |  |  |         player = self.player | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |         return (world.boss_shuffle[player] or world.enemy_shuffle[player] | 
					
						
							| 
									
										
										
										
											2022-08-10 13:21:52 -07:00
										 |  |  |                 or world.enemy_health[player] != 'default' or world.enemy_damage[player] != 'default' | 
					
						
							|  |  |  |                 or world.pot_shuffle[player] or world.bush_shuffle[player] | 
					
						
							|  |  |  |                 or world.killable_thieves[player]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  |     def generate_output(self, output_directory: str): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         world = self.multiworld | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  |         player = self.player | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2022-08-10 13:21:52 -07:00
										 |  |  |             use_enemizer = self.use_enemizer() | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-18 22:13:19 +02:00
										 |  |  |             rom = LocalRom(get_base_rom_path()) | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             patch_rom(world, rom, player, use_enemizer) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if use_enemizer: | 
					
						
							| 
									
										
										
										
											2022-08-30 17:12:33 +02:00
										 |  |  |                 patch_enemizer(world, player, rom, self.enemizer_path, output_directory) | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if world.is_race: | 
					
						
							|  |  |  |                 patch_race_rom(rom, world, player) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             world.spoiler.hashes[player] = get_hash_string(rom.hash) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             palettes_options = { | 
					
						
							|  |  |  |                 'dungeon': world.uw_palettes[player], | 
					
						
							|  |  |  |                 'overworld': world.ow_palettes[player], | 
					
						
							|  |  |  |                 'hud': world.hud_palettes[player], | 
					
						
							|  |  |  |                 'sword': world.sword_palettes[player], | 
					
						
							|  |  |  |                 'shield': world.shield_palettes[player], | 
					
						
							| 
									
										
										
										
											2022-09-07 11:16:32 -07:00
										 |  |  |                 # 'link': world.link_palettes[player] | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |             palettes_options = {key: option.current_key for key, option in palettes_options.items()} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             apply_rom_settings(rom, world.heartbeep[player].current_key, | 
					
						
							|  |  |  |                                world.heartcolor[player].current_key, | 
					
						
							|  |  |  |                                world.quickswap[player], | 
					
						
							|  |  |  |                                world.menuspeed[player].current_key, | 
					
						
							|  |  |  |                                world.music[player], | 
					
						
							|  |  |  |                                world.sprite[player], | 
					
						
							|  |  |  |                                palettes_options, world, player, True, | 
					
						
							|  |  |  |                                reduceflashing=world.reduceflashing[player] or world.is_race, | 
					
						
							| 
									
										
										
										
											2021-11-08 16:34:54 +01:00
										 |  |  |                                triforcehud=world.triforcehud[player].current_key, | 
					
						
							| 
									
										
										
										
											2022-04-04 18:54:49 -07:00
										 |  |  |                                deathlink=world.death_link[player], | 
					
						
							|  |  |  |                                allowcollect=world.allow_collect[player]) | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             rompath = os.path.join(output_directory, f"{self.multiworld.get_out_file_name_base(self.player)}.sfc") | 
					
						
							| 
									
										
										
										
											2021-09-15 01:02:06 +02:00
										 |  |  |             rom.write_to_file(rompath) | 
					
						
							| 
									
										
										
										
											2022-03-18 04:53:09 +01:00
										 |  |  |             patch = LttPDeltaPatch(os.path.splitext(rompath)[0]+LttPDeltaPatch.patch_file_ending, player=player, | 
					
						
							|  |  |  |                                    player_name=world.player_name[player], patched_path=rompath) | 
					
						
							|  |  |  |             patch.write() | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  |             os.unlink(rompath) | 
					
						
							|  |  |  |             self.rom_name = rom.name | 
					
						
							|  |  |  |         except: | 
					
						
							|  |  |  |             raise | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             self.rom_name_available_event.set() # make sure threading continues and errors are collected | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-18 14:30:43 +02:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def stage_extend_hint_information(cls, world, hint_data: typing.Dict[int, typing.Dict[int, str]]): | 
					
						
							|  |  |  |         er_hint_data = {player: {} for player in world.get_game_players("A Link to the Past") if | 
					
						
							|  |  |  |                         world.shuffle[player] != "vanilla" or world.retro_caves[player]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for region in world.regions: | 
					
						
							|  |  |  |             if region.player in er_hint_data and region.locations: | 
					
						
							|  |  |  |                 main_entrance = region.get_connecting_entrance(is_main_entrance) | 
					
						
							|  |  |  |                 for location in region.locations: | 
					
						
							|  |  |  |                     if type(location.address) == int:  # skips events and crystals | 
					
						
							|  |  |  |                         if lookup_vanilla_location_to_entrance[location.address] != main_entrance.name: | 
					
						
							|  |  |  |                             er_hint_data[region.player][location.address] = main_entrance.name | 
					
						
							|  |  |  |         hint_data.update(er_hint_data) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  |     def modify_multidata(self, multidata: dict): | 
					
						
							|  |  |  |         import base64 | 
					
						
							|  |  |  |         # wait for self.rom_name to be available. | 
					
						
							|  |  |  |         self.rom_name_available_event.wait() | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  |         rom_name = getattr(self, "rom_name", None) | 
					
						
							|  |  |  |         # we skip in case of error, so that the original error in the output thread is the one that gets raised | 
					
						
							|  |  |  |         if rom_name: | 
					
						
							|  |  |  |             new_name = base64.b64encode(bytes(self.rom_name)).decode() | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             multidata["connect_names"][new_name] = multidata["connect_names"][self.multiworld.player_name[self.player]] | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 13:54:47 +02:00
										 |  |  |     def create_item(self, name: str) -> Item: | 
					
						
							| 
									
										
										
										
											2022-08-06 00:49:54 +02:00
										 |  |  |         return ALttPItem(name, self.player, **item_init_table[name]) | 
					
						
							| 
									
										
										
										
											2021-07-12 13:54:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  |     @classmethod | 
					
						
							| 
									
										
										
										
											2022-09-16 20:06:25 -04:00
										 |  |  |     def stage_fill_hook(cls, world, progitempool, usefulitempool, filleritempool, fill_locations): | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  |         trash_counts = {} | 
					
						
							|  |  |  |         standard_keyshuffle_players = set() | 
					
						
							|  |  |  |         for player in world.get_game_players("A Link to the Past"): | 
					
						
							| 
									
										
										
										
											2021-08-30 18:00:39 +02:00
										 |  |  |             if world.mode[player] == 'standard' and world.smallkey_shuffle[player] \ | 
					
						
							| 
									
										
										
										
											2022-01-09 04:48:31 +01:00
										 |  |  |                     and world.smallkey_shuffle[player] != smallkey_shuffle.option_universal and \ | 
					
						
							|  |  |  |                     world.smallkey_shuffle[player] != smallkey_shuffle.option_own_dungeons: | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  |                 standard_keyshuffle_players.add(player) | 
					
						
							|  |  |  |             if not world.ganonstower_vanilla[player] or \ | 
					
						
							|  |  |  |                     world.logic[player] in {'owglitches', 'hybridglitches', "nologic"}: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  |             elif 'triforcehunt' in world.goal[player] and ('local' in world.goal[player] or world.players == 1): | 
					
						
							|  |  |  |                 trash_counts[player] = world.random.randint(world.crystals_needed_for_gt[player] * 2, | 
					
						
							|  |  |  |                                                             world.crystals_needed_for_gt[player] * 4) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 trash_counts[player] = world.random.randint(0, world.crystals_needed_for_gt[player] * 2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Make sure the escape small key is placed first in standard with key shuffle to prevent running out of spots | 
					
						
							| 
									
										
										
										
											2021-08-14 00:37:58 +02:00
										 |  |  |         # TODO: this might be worthwhile to introduce as generic option for various games and then optimize it | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  |         if standard_keyshuffle_players: | 
					
						
							| 
									
										
										
										
											2022-01-09 04:32:17 +01:00
										 |  |  |             viable = {} | 
					
						
							| 
									
										
										
										
											2021-08-14 00:37:58 +02:00
										 |  |  |             for location in world.get_locations(): | 
					
						
							|  |  |  |                 if location.player in standard_keyshuffle_players \ | 
					
						
							|  |  |  |                         and location.item is None \ | 
					
						
							|  |  |  |                         and location.can_reach(world.state): | 
					
						
							| 
									
										
										
										
											2022-01-09 04:32:17 +01:00
										 |  |  |                     viable.setdefault(location.player, []).append(location) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-14 00:37:58 +02:00
										 |  |  |             for player in standard_keyshuffle_players: | 
					
						
							| 
									
										
										
										
											2022-01-09 04:32:17 +01:00
										 |  |  |                 loc = world.random.choice(viable[player]) | 
					
						
							| 
									
										
										
										
											2021-08-14 00:37:58 +02:00
										 |  |  |                 key = world.create_item("Small Key (Hyrule Castle)", player) | 
					
						
							|  |  |  |                 loc.place_locked_item(key) | 
					
						
							|  |  |  |                 fill_locations.remove(loc) | 
					
						
							|  |  |  |             world.random.shuffle(fill_locations) | 
					
						
							|  |  |  |             # TODO: investigate not creating the key in the first place | 
					
						
							| 
									
										
										
										
											2022-02-13 23:02:18 +01:00
										 |  |  |             progitempool[:] = [item for item in progitempool if | 
					
						
							|  |  |  |                                item.player not in standard_keyshuffle_players or | 
					
						
							|  |  |  |                                item.name != "Small Key (Hyrule Castle)"] | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if trash_counts: | 
					
						
							|  |  |  |             locations_mapping = {player: [] for player in trash_counts} | 
					
						
							|  |  |  |             for location in fill_locations: | 
					
						
							|  |  |  |                 if 'Ganons Tower' in location.name and location.player in locations_mapping: | 
					
						
							|  |  |  |                     locations_mapping[location.player].append(location) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for player, trash_count in trash_counts.items(): | 
					
						
							|  |  |  |                 gtower_locations = locations_mapping[player] | 
					
						
							|  |  |  |                 world.random.shuffle(gtower_locations) | 
					
						
							| 
									
										
										
										
											2022-09-16 20:06:25 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 while gtower_locations and filleritempool and trash_count > 0: | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  |                     spot_to_fill = gtower_locations.pop() | 
					
						
							| 
									
										
										
										
											2022-10-17 09:52:34 +02:00
										 |  |  |                     for index, item in enumerate(filleritempool): | 
					
						
							|  |  |  |                         if spot_to_fill.item_rule(item): | 
					
						
							|  |  |  |                             filleritempool.pop(index)  # remove from outer fill | 
					
						
							|  |  |  |                             world.push_item(spot_to_fill, item, False) | 
					
						
							|  |  |  |                             fill_locations.remove(spot_to_fill)  # very slow, unfortunately | 
					
						
							|  |  |  |                             trash_count -= 1 | 
					
						
							|  |  |  |                             break | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         logging.warning(f"Could not trash fill Ganon's Tower for player {player}.") | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  |     def get_filler_item_name(self) -> str: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         if self.multiworld.goal[self.player] == "icerodhunt": | 
					
						
							| 
									
										
										
										
											2022-05-19 09:37:26 -04:00
										 |  |  |             item = "Nothing" | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             item = self.multiworld.random.choice(extras_list) | 
					
						
							|  |  |  |         return GetBeemizerItem(self.multiworld, self.player, item) | 
					
						
							| 
									
										
										
										
											2022-02-05 15:49:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-13 23:02:18 +01:00
										 |  |  |     def get_pre_fill_items(self): | 
					
						
							|  |  |  |         res = [] | 
					
						
							|  |  |  |         if self.dungeon_local_item_names: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |             for (name, player), dungeon in self.multiworld.dungeons.items(): | 
					
						
							| 
									
										
										
										
											2022-02-13 23:02:18 +01:00
										 |  |  |                 if player == self.player: | 
					
						
							|  |  |  |                     for item in dungeon.all_items: | 
					
						
							|  |  |  |                         if item.name in self.dungeon_local_item_names: | 
					
						
							|  |  |  |                             res.append(item) | 
					
						
							|  |  |  |         return res | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-28 00:26:02 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | def get_same_seed(world, seed_def: tuple) -> str: | 
					
						
							|  |  |  |     seeds: typing.Dict[tuple, str] = getattr(world, "__named_seeds", {}) | 
					
						
							|  |  |  |     if seed_def in seeds: | 
					
						
							|  |  |  |         return seeds[seed_def] | 
					
						
							|  |  |  |     seeds[seed_def] = str(world.random.randint(0, 2 ** 64)) | 
					
						
							|  |  |  |     world.__named_seeds = seeds | 
					
						
							| 
									
										
										
										
											2021-08-30 19:11:12 +02:00
										 |  |  |     return seeds[seed_def] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ALttPLogic(LogicMixin): | 
					
						
							|  |  |  |     def _lttp_has_key(self, item, player, count: int = 1): | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         if self.multiworld.logic[player] == 'nologic': | 
					
						
							| 
									
										
										
										
											2021-08-30 19:11:12 +02:00
										 |  |  |             return True | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |         if self.multiworld.smallkey_shuffle[player] == smallkey_shuffle.option_universal: | 
					
						
							| 
									
										
										
										
											2021-08-30 19:11:12 +02:00
										 |  |  |             return self.can_buy_unlimited('Small Key (Universal)', player) | 
					
						
							| 
									
										
										
										
											2021-08-31 17:28:46 -04:00
										 |  |  |         return self.prog_items[item, player] >= count |