| 
									
										
										
										
											2021-07-22 15:51:50 +02:00
										 |  |  | import random | 
					
						
							| 
									
										
										
										
											2021-08-09 06:50:11 +02:00
										 |  |  | import logging | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  | import os | 
					
						
							|  |  |  | import threading | 
					
						
							| 
									
										
										
										
											2021-07-22 15:51:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 15:11:48 +02:00
										 |  |  | from BaseClasses import Item, CollectionState | 
					
						
							|  |  |  | from .SubClasses import ALttPItem | 
					
						
							| 
									
										
										
										
											2021-06-11 14:22:44 +02:00
										 |  |  | from ..AutoWorld import World | 
					
						
							| 
									
										
										
										
											2021-07-04 16:18:21 +02:00
										 |  |  | from .Options import alttp_options | 
					
						
							| 
									
										
										
										
											2021-07-12 13:54:47 +02:00
										 |  |  | from .Items import as_dict_item_table, item_name_groups, item_table | 
					
						
							| 
									
										
										
										
											2021-07-22 15:51:50 +02:00
										 |  |  | from .Regions import lookup_name_to_id, create_regions, mark_light_world_regions | 
					
						
							|  |  |  | from .Rules import set_rules | 
					
						
							|  |  |  | from .ItemPool import generate_itempool | 
					
						
							|  |  |  | from .Shops import create_shops | 
					
						
							|  |  |  | from .Dungeons import create_dungeons | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  | from .Rom import LocalRom, patch_rom, patch_race_rom, patch_enemizer, apply_rom_settings, get_hash_string | 
					
						
							|  |  |  | import Patch | 
					
						
							| 
									
										
										
										
											2021-07-12 13:54:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-22 15:51:50 +02:00
										 |  |  | from .InvertedRegions import create_inverted_regions, mark_dark_world_regions | 
					
						
							|  |  |  | from .EntranceShuffle import link_entrances, link_inverted_entrances, plando_connect | 
					
						
							| 
									
										
										
										
											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") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-11 14:22:44 +02:00
										 |  |  | class ALTTPWorld(World): | 
					
						
							|  |  |  |     game: str = "A Link to the Past" | 
					
						
							| 
									
										
										
										
											2021-07-04 16:18:21 +02:00
										 |  |  |     options = 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 | 
					
						
							|  |  |  |     item_names = frozenset(item_table) | 
					
						
							| 
									
										
										
										
											2021-07-12 15:11:48 +02:00
										 |  |  |     location_names = frozenset(lookup_name_to_id) | 
					
						
							| 
									
										
										
										
											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-07-14 10:02:39 +02:00
										 |  |  |     data_version = 7 | 
					
						
							| 
									
										
										
										
											2021-07-13 19:14:57 +02:00
										 |  |  |     remote_items: bool = False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-22 15:51:50 +02:00
										 |  |  |     set_rules = set_rules | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     create_items = generate_itempool | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def create_regions(self): | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  |         self.rom_name_available_event = threading.Event() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-23 12:03:19 +02:00
										 |  |  |         player = self.player | 
					
						
							| 
									
										
										
										
											2021-07-22 15:51:50 +02:00
										 |  |  |         world = self.world | 
					
						
							| 
									
										
										
										
											2021-07-23 12:03:19 +02:00
										 |  |  |         if world.open_pyramid[player] == 'goal': | 
					
						
							|  |  |  |             world.open_pyramid[player] = world.goal[player] in {'crystals', 'ganontriforcehunt', | 
					
						
							|  |  |  |                                                                 'localganontriforcehunt', 'ganonpedestal'} | 
					
						
							|  |  |  |         elif world.open_pyramid[player] == 'auto': | 
					
						
							|  |  |  |             world.open_pyramid[player] = world.goal[player] in {'crystals', 'ganontriforcehunt', | 
					
						
							|  |  |  |                                                                 'localganontriforcehunt', 'ganonpedestal'} and \ | 
					
						
							|  |  |  |                                          (world.shuffle[player] in {'vanilla', 'dungeonssimple', 'dungeonsfull', | 
					
						
							|  |  |  |                                                                     'dungeonscrossed'} or not world.shuffle_ganon) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             world.open_pyramid[player] = {'on': True, 'off': False, 'yes': True, 'no': False}.get( | 
					
						
							|  |  |  |                 world.open_pyramid[player], 'auto') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         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 | 
					
						
							|  |  |  |         world.random = random.Random(world.er_seeds[player]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if world.mode[player] != 'inverted': | 
					
						
							|  |  |  |             link_entrances(world, player) | 
					
						
							|  |  |  |             mark_light_world_regions(world, player) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             link_inverted_entrances(world, player) | 
					
						
							|  |  |  |             mark_dark_world_regions(world, player) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         world.random = old_random | 
					
						
							|  |  |  |         plando_connect(world, player) | 
					
						
							| 
									
										
										
										
											2021-07-22 15:51:50 +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 | 
					
						
							|  |  |  |                     elif state.has('Tempered Sword', item.player) and self.world.difficulty_requirements[ | 
					
						
							|  |  |  |                         item.player].progressive_sword_limit >= 4: | 
					
						
							|  |  |  |                         return 'Golden Sword' | 
					
						
							|  |  |  |                     elif state.has('Master Sword', item.player) and self.world.difficulty_requirements[ | 
					
						
							|  |  |  |                         item.player].progressive_sword_limit >= 3: | 
					
						
							|  |  |  |                         return 'Tempered Sword' | 
					
						
							|  |  |  |                     elif state.has('Fighter Sword', item.player) and self.world.difficulty_requirements[item.player].progressive_sword_limit >= 2: | 
					
						
							|  |  |  |                         return 'Master Sword' | 
					
						
							|  |  |  |                     elif self.world.difficulty_requirements[item.player].progressive_sword_limit >= 1: | 
					
						
							|  |  |  |                         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 | 
					
						
							|  |  |  |                     elif state.has('Red Shield', item.player) and self.world.difficulty_requirements[item.player].progressive_shield_limit >= 3: | 
					
						
							|  |  |  |                         return 'Mirror Shield' | 
					
						
							|  |  |  |                     elif state.has('Blue Shield', item.player)  and self.world.difficulty_requirements[item.player].progressive_shield_limit >= 2: | 
					
						
							|  |  |  |                         return 'Red Shield' | 
					
						
							|  |  |  |                     elif self.world.difficulty_requirements[item.player].progressive_shield_limit >= 1: | 
					
						
							|  |  |  |                         return 'Blue Shield' | 
					
						
							|  |  |  |                 elif 'Bow' in item_name: | 
					
						
							|  |  |  |                     if state.has('Silver', item.player): | 
					
						
							|  |  |  |                         return | 
					
						
							|  |  |  |                     elif state.has('Bow', item.player) and self.world.difficulty_requirements[item.player].progressive_bow_limit >= 2: | 
					
						
							|  |  |  |                         return 'Silver Bow' | 
					
						
							|  |  |  |                     elif self.world.difficulty_requirements[item.player].progressive_bow_limit >= 1: | 
					
						
							|  |  |  |                         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 | 
					
						
							|  |  |  |         world = self.world | 
					
						
							|  |  |  |         player = self.player | 
					
						
							|  |  |  |         all_state = world.get_all_state(keys=True) | 
					
						
							|  |  |  |         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 | 
					
						
							|  |  |  |         fill_dungeons_restrictive(world) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  |     def generate_output(self, output_directory: str): | 
					
						
							|  |  |  |         world = self.world | 
					
						
							|  |  |  |         player = self.player | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  |         try: | 
					
						
							|  |  |  |             use_enemizer = (world.boss_shuffle[player] != 'none' or world.enemy_shuffle[player] | 
					
						
							|  |  |  |                             or world.enemy_health[player] != 'default' or world.enemy_damage[player] != 'default' | 
					
						
							|  |  |  |                             or world.shufflepots[player] or world.bush_shuffle[player] | 
					
						
							|  |  |  |                             or world.killable_thieves[player]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             rom = LocalRom(world.alttp_rom) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             patch_rom(world, rom, player, use_enemizer) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if use_enemizer: | 
					
						
							|  |  |  |                 patch_enemizer(world, player, rom, world.enemizer, output_directory) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             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], | 
					
						
							|  |  |  |                 'link': world.link_palettes[player] | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             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, | 
					
						
							|  |  |  |                                triforcehud=world.triforcehud[player].current_key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             outfilepname = f'_P{player}' | 
					
						
							|  |  |  |             outfilepname += f"_{world.player_name[player].replace(' ', '_')}" \ | 
					
						
							|  |  |  |                 if world.player_name[player] != 'Player%d' % player else '' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             rompath = os.path.join(output_directory, f'AP_{world.seed_name}{outfilepname}.sfc') | 
					
						
							|  |  |  |             rom.write_to_file(rompath, hide_enemizer=True) | 
					
						
							|  |  |  |             Patch.create_patch_file(rompath, player=player, player_name=world.player_name[player]) | 
					
						
							|  |  |  |             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
										 |  |  | 
 | 
					
						
							|  |  |  |     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() | 
					
						
							|  |  |  |             payload = multidata["connect_names"][self.world.player_name[self.player]] | 
					
						
							|  |  |  |             multidata["connect_names"][new_name] = payload | 
					
						
							|  |  |  |             del (multidata["connect_names"][self.world.player_name[self.player]]) | 
					
						
							| 
									
										
										
										
											2021-08-09 09:15:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-10 07:37:56 +02:00
										 |  |  |     def get_required_client_version(self) -> tuple: | 
					
						
							|  |  |  |         return max((0, 1, 4), super(ALTTPWorld, self).get_required_client_version()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 13:54:47 +02:00
										 |  |  |     def create_item(self, name: str) -> Item: | 
					
						
							|  |  |  |         return ALttPItem(name, self.player, **as_dict_item_table[name]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-10 09:03:44 +02:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def stage_fill_hook(cls, world, progitempool, nonexcludeditempool, localrestitempool, restitempool, fill_locations): | 
					
						
							|  |  |  |         trash_counts = {} | 
					
						
							|  |  |  |         standard_keyshuffle_players = set() | 
					
						
							|  |  |  |         for player in world.get_game_players("A Link to the Past"): | 
					
						
							|  |  |  |             if world.mode[player] == 'standard' and world.keyshuffle[player] is True: | 
					
						
							|  |  |  |                 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: | 
					
						
							| 
									
										
										
										
											2021-08-14 00:37:58 +02:00
										 |  |  |             viable = [] | 
					
						
							|  |  |  |             for location in world.get_locations(): | 
					
						
							|  |  |  |                 if location.player in standard_keyshuffle_players \ | 
					
						
							|  |  |  |                         and location.item is None \ | 
					
						
							|  |  |  |                         and location.can_reach(world.state): | 
					
						
							|  |  |  |                     viable.append(location) | 
					
						
							|  |  |  |             world.random.shuffle(viable) | 
					
						
							|  |  |  |             for player in standard_keyshuffle_players: | 
					
						
							|  |  |  |                 key = world.create_item("Small Key (Hyrule Castle)", player) | 
					
						
							|  |  |  |                 loc = viable.pop() | 
					
						
							|  |  |  |                 loc.place_locked_item(key) | 
					
						
							|  |  |  |                 fill_locations.remove(loc) | 
					
						
							|  |  |  |             world.random.shuffle(fill_locations) | 
					
						
							|  |  |  |             # TODO: investigate not creating the key in the first place | 
					
						
							|  |  |  |             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) | 
					
						
							|  |  |  |                 localrest = localrestitempool[player] | 
					
						
							|  |  |  |                 if localrest: | 
					
						
							|  |  |  |                     gt_item_pool = restitempool + localrest | 
					
						
							|  |  |  |                     world.random.shuffle(gt_item_pool) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     gt_item_pool = restitempool.copy() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 while gtower_locations and gt_item_pool and trash_count > 0: | 
					
						
							|  |  |  |                     spot_to_fill = gtower_locations.pop() | 
					
						
							|  |  |  |                     item_to_place = gt_item_pool.pop() | 
					
						
							|  |  |  |                     if item_to_place in localrest: | 
					
						
							|  |  |  |                         localrest.remove(item_to_place) | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         restitempool.remove(item_to_place) | 
					
						
							|  |  |  |                     world.push_item(spot_to_fill, item_to_place, False) | 
					
						
							|  |  |  |                     fill_locations.remove(spot_to_fill)  # very slow, unfortunately | 
					
						
							|  |  |  |                     trash_count -= 1 |